<!-- -*- mode: Text; fill-column: 100 -*- vim: set textwidth=100 :
 !   EDITOR NOTES
 !
 !   Adding a new element involves editing the following sections:
 !    - section for the element itself
 !    - descriptions of the element's categories
 !    - images/content-venn.svg
 !    - syntax, if it's void or otherwise special
 !    - parser, if it's not phrasing-level
 !    - rendering
 !    - obsolete section
 !    - element, attribute, content model, and interface indices
 !
 !   Adding a new attribute involves editing the following sections:
 !    - The IDL and content attributes for the relevant elements
 !    - element and attribute indices
 !-->

<!--
 ! https://lists.w3.org/Archives/Public/www-archive/2014Apr/0034.html
 !-->

<!DOCTYPE html> <!-- Note: This file is NOT HTML, it's a proprietary language that is then post-processed into HTML. -->
<html lang="en-US-x-hixie">
 <head>
  <meta charset="utf-8">
  <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
  <title w-nodev>HTML Standard</title>
  <title w-dev>HTML Standard, Edition for Web Developers</title>
  <meta name="theme-color" content="#3c790a">
  <meta name="color-scheme" content="light dark">
  <link w-noreview rel="stylesheet" crossorigin href="https://resources.whatwg.org/standard-shared-with-dev.css">
  <link w-nodev w-noreview crossorigin rel="stylesheet" href="https://resources.whatwg.org/standard.css">
  <link w-nodev crossorigin rel="stylesheet" href="https://resources.whatwg.org/spec.css">
  <link w-nohtml w-nodev w-nosnap rel="stylesheet" crossorigin href="https://resources.whatwg.org/review-draft.css">
  <link w-nosnap w-noreview rel="icon" crossorigin href="https://resources.whatwg.org/logo.svg">
  <link w-nohtml w-nodev rel="icon" crossorigin href="https://resources.whatwg.org/logo-snapshot.svg">
  <link w-dev rel="stylesheet" crossorigin href="/dev/styles.css">
  <link w-nodev rel="stylesheet" crossorigin href="/styles.css">
  <script>
   function toggleStatus(div) {
     div.parentNode.classList.toggle('wrapped');
   }
   function setLinkFragment(link) {
     link.hash = location.hash;
   }
  </script>
 </head>
 <body>
  <script w-dev src="/dev/search.js" async></script>
  <script w-nodev w-noreview crossorigin src="/html-dfn.js" defer></script>
  <script w-nodev w-noreview w-nosplit crossorigin src="https://resources.whatwg.org/commit-snapshot-shortcut-key.js" defer></script>
  <header class="head with-buttons" id="head">
   <a href="https://whatwg.org/" class="logo"><img crossorigin width="100" height="100" alt="WHATWG" src="https://resources.whatwg.org/logo.svg" class="darkmode-aware"></a>
   <hgroup w-nodev>
    <h1 class="allcaps">HTML</h1>
    <p w-nosnap w-noreview id="living-standard">Living Standard &mdash; Last Updated <span class="pubdate">[DATE: 01 Jan 1901]</span></p>
    <p w-nohtml w-noreview id="living-standard">Commit Snapshot &mdash; Last Updated <span class="pubdate">[DATE: 01 Jan 1901]</span></p>
    <p w-nohtml w-nosnap id="living-standard">Review Draft &mdash; Published <span class="pubdate">[DATE: 01 Jan 1901]</span></p>
   </hgroup>
   <hgroup w-dev>
    <h1><a href="/dev/" rel=home>HTML: The Living Standard</a></h1>
    <p id="dev-edition-h2">Edition for Web Developers &mdash; Last Updated <span class="pubdate">[DATE: 01 Jan 1901]</span></p>
   </hgroup>
   <nav w-nosplit w-nodev w-noreview>
    <!-- Note: be sure to keep an even number of <a>s, for small screens where they appear in two columns. -->
    <div>
     <a w-nohtml href="/" id="commit-snapshot-link"><span data-x=""><strong>One-Page Version</strong> <code data-x="">html.spec.whatwg.org</code></span></a>
     <a w-nosnap href="/"><span data-x=""><strong>One-Page Version</strong> <code data-x="">html.spec.whatwg.org</code></span></a>
     <a href="/multipage/" id="multipage-link" onclick="setLinkFragment(this);"><span data-x=""><strong>Multipage Version</strong> <code data-x="">/multipage</code></span></a>
     <a href="/dev/" onclick="setLinkFragment(this);"><span data-x=""><strong>Version for Web Devs</strong> <code data-x="">/dev</code></span></a>
     <a href="/print.pdf"><span data-x=""><strong>PDF Version</strong> <code data-x="">/print.pdf</code></span></a>
     <a href="https://github.com/whatwg/html/wiki/Translations"><span data-x=""><strong>Translations</strong> <code data-x="">日本語 • 简体中文</code></span></a>
    </div>
    <div>
     <a class="misc" href="https://github.com/whatwg/html/blob/main/FAQ.md"><span data-x=""><strong>FAQ</strong> <code data-x="">on GitHub</code></span></a>
     <a class="comms" href="https://whatwg.org/chat"><span data-x=""><strong>Chat</strong> <code data-x="">on Matrix</code></span></a>
    </div>
    <div>
     <a class="changes" href="https://github.com/whatwg/html"><span data-x=""><strong>Contribute on GitHub</strong> <code data-x="">whatwg/html repository</code></span></a>
     <a class="changes" href="https://github.com/whatwg/html/commits"><span data-x=""><strong>Commits</strong> <code data-x="">on GitHub</code></span></a>
     <a w-nosnap class="changes" href="/commit-snapshots/[SHA]" id="commit-snapshot-link"><span data-x=""><strong>Snapshot</strong> <code data-x="">as of this commit</code></span></a>
     <a class="changes" href="https://twitter.com/htmlstandard"><span data-x=""><strong>Twitter Updates</strong> <code data-x="">@htmlstandard</code></span></a>
    </div>
    <div>
     <a class="feedback" href="https://github.com/whatwg/html/issues"><span data-x=""><strong>Open Issues</strong> <code data-x="">filed on GitHub</code></span></a>
     <a class="feedback" href="https://whatwg.org/newbug"><span data-x=""><strong>Open an Issue</strong> <code data-x="">whatwg.org/newbug</code></span></a>
     <a class="feedback" href="https://github.com/web-platform-tests/wpt/tree/master/html"><span data-x=""><strong>Tests</strong> <code data-x="">web-platform-tests html/</code></span></a>
     <a class="feedback" href="https://github.com/web-platform-tests/wpt/labels/html"><span data-x=""><strong>Issues for Tests</strong> <code data-x="">ongoing work</code></span></a>
    </div>
   </nav>

   <div id="search" w-dev>
    <input name="query" type="search" placeholder="Search. Press '/'" autocomplete="off" id="query">
    <ol id="results"></ol>
   </div>
  </header>

  <details w-nohtml w-nodev w-noreview class="annoying-warning" open="">
   <summary>This is a Commit Snapshot of the standard</summary>

   <p>This document contains the contents of the standard as of the <a class="sha-link"></a>, and
   should only be used as a historical reference. This commit may not even have been merged into the
   main branch.</p>

   <p>Do not attempt to implement this version of the standard. Do not reference this version as
   authoritative in any way. Instead, see <a
   href="https://html.spec.whatwg.org/">https://html.spec.whatwg.org/</a> for the living
   standard.</p>
  </details>

  <details w-nohtml w-nodev w-nosnap class="annoying-warning" open>
   <summary>This is a Review Draft of the standard</summary>

   <p>This is a Review Draft. It is published primarily for purposes of patent review by Workstream
   Participants; it mirrors the <a href="https://html.spec.whatwg.org/">Living Standard</a> closely,
   redacting only text that is identified as "Objection Pending" or "Confirmation Pending".
   Developers should refer to the <a href="https://html.spec.whatwg.org/">Living Standard</a> for
   the most current error corrections and other developments.</p>

   <p>For information regarding patent commitments, please see the <a
   href="https://whatwg.org/ipr-policy">IPR Policy</a> and <a
   href="https://github.com/whatwg/html/labels/exclusion%20notices">exclusion notices</a>.</p>

   <p>Do not attempt to implement this version of the standard. Do not reference this version as
   authoritative in any way. Instead, see <a
   href="https://html.spec.whatwg.org/">https://html.spec.whatwg.org/</a> for the Living
   Standard.</p>
  </details>

  <hr w-nodev w-nosplit>

  <h2 class="no-num no-toc" w-nosplit>Table of contents</h2>
  <!--smalltoc-->

  <h2 class="no-num no-toc" id="contents" w-nosplit w-nodev>Full table of contents</h2>
  <!--toc-->

  <div w-dev w-nosplit>

  <h2 class="no-num no-toc" id="about-dev-edition">About this specification</h2>

  <p>This specification is like no other — it has been processed with <em>you</em>, the humble web
  developer, in mind.</p>

  <p>The focus of this specification is readability and ease of access. Unlike the <a
  href="/multipage/">full HTML Standard</a>, this "developer's edition" removes information that
  only browser vendors need know. It is automatically produced from the full specification by our
  build tooling, and thus always in sync with the latest developments in HTML.</p>

  <p>To read about its conception, construction, and future, read the <a
  href="https://web.archive.org/web/20150220020906/http://archive.germanforblack.com:80/articles/html5-for-web-developers">original
  press release</a>, and the <a href="https://blog.whatwg.org/developers-edition-comeback">blog post
  about its relaunch</a>.</p>

  <p>Finally, feel free to <a href="https://github.com/whatwg/html/labels/dev%20edition">contribute
  on GitHub</a> to make this edition better for everyone!</p>

  </div>

  <h2 split-filename="introduction" id="introduction">Introduction</h2>

  <div w-nodev>

  <h3 id="abstract">Where does this specification fit?</h3>

  <p>This specification defines a big part of the web platform, in lots of detail. Its place in the
  web platform specification stack relative to other specifications can be best summed up as
  follows:</p>

  <svg viewBox="0 0 398 359" width=398 height=359 role=img aria-label="It consists of everything else, above such core technologies as HTTP, TLS, DOM, Unicode, Web IDL, MIME, URL, XML, JavaScript, and Encoding; below higher-level technologies like CSS, SVG, MathML, and Service Workers; and to the side of technologies like IndexedDB, Fetch, CSP, AV1, Opus, and PNG." id="abstractimg">
   <rect width=398 height=80 />
   <text x=199 y=45 class=horizontal>CSS SVG MathML Service&thinsp;Workers</text>
   <rect width=67 height=177 y=85 />
   <text transform="translate(25 173.5) rotate(-90)" class="left">IDB Fetch CSP</text>
   <text transform="translate(50 173.5) rotate(-90)" class="left">AV1 Opus PNG</text>
   <!-- https://www.flickr.com/photos/wonderlane/2986252088/ -->
   <image x=72 y=85 width=326 height=177 xlink:href="/images/abstract.jpeg"/>
   <text x=130 y=250 class=right>THIS SPECIFICATION</text>
   <rect width=398 height=92 y=267 />
   <text x=199 y=300 class=horizontal>HTTP TLS DOM Unicode Web&thinsp;IDL</text>
   <text x=199 y=330 class=horizontal>MIME URL XML JavaScript Encoding</text>
  </svg>

  </div>


  <h3 id="is-this-html5?">Is this HTML5?</h3>

  <!-- NON-NORMATIVE SECTION -->

  <p>In short: Yes.</p>

  <p>In more length: the term "HTML5" is widely used as a buzzword to refer to modern web
  technologies, many of which (though by no means all) are developed at the WHATWG. This document is
  one such; others are available from <a href="https://spec.whatwg.org/">the WHATWG Standards
  overview</a>.</p>


  <h3>Background</h3>

  <!-- NON-NORMATIVE SECTION -->

  <p>HTML is the World Wide Web's core markup language. Originally, HTML was primarily designed as a
  language for semantically describing scientific documents. Its general design, however, has
  enabled it to be adapted, over the subsequent years, to describe a number of other types of
  documents and even applications.</p>


  <h3>Audience</h3>

  <!-- NON-NORMATIVE SECTION -->

  <p>This specification is intended for authors of documents and scripts that use the features
  defined in this specification<span w-nodev>, implementers of tools that operate on pages that
  use the features defined in this specification, and individuals wishing to establish the
  correctness of documents or implementations with respect to the requirements of this
  specification</span>.</p>

  <p>This document is probably not suited to readers who do not already have at least a passing
  familiarity with web technologies, as in places it sacrifices clarity for precision, and brevity
  for completeness. More approachable tutorials and authoring guides can provide a gentler
  introduction to the topic.</p>

  <p>In particular, familiarity with the basics of DOM is necessary for a complete understanding of
  some of the more technical parts of this specification. An understanding of Web IDL, HTTP, XML,
  Unicode, character encodings, JavaScript, and CSS will also be helpful in places but is not
  essential.</p>


  <h3>Scope</h3>

  <!-- NON-NORMATIVE SECTION -->

  <p>This specification is limited to providing a semantic-level markup language and associated
  semantic-level scripting APIs for authoring accessible pages on the web ranging from static
  documents to dynamic applications.</p>

  <p>The scope of this specification does not include providing mechanisms for media-specific
  customization of presentation (although default rendering rules for web browsers are included at
  the end of this specification, and several mechanisms for hooking into CSS are provided as part of
  the language).</p>

  <p>The scope of this specification is not to describe an entire operating system. In particular,
  hardware configuration software, image manipulation tools, and applications that users would be
  expected to use with high-end workstations on a daily basis are out of scope. In terms of
  applications, this specification is targeted specifically at applications that would be expected
  to be used by users on an occasional basis, or regularly but from disparate locations, with low
  CPU requirements. Examples of such applications include online purchasing systems, searching
  systems, games (especially multiplayer online games), public telephone books or address books,
  communications software (email clients, instant messaging clients, discussion software), document
  editing software, etc.</p>


  <h3>History</h3>

  <!-- NON-NORMATIVE SECTION -->

  <p>For its first five years (1990-1995), HTML went through a number of revisions and experienced a
  number of extensions, primarily hosted first at CERN, and then at the IETF.</p>

  <p>With the creation of the W3C, HTML's development changed venue again. A first abortive attempt
  at extending HTML in 1995 known as HTML 3.0 then made way to a more pragmatic approach known as
  HTML 3.2, which was completed in 1997. HTML4 quickly followed later that same year.</p>

  <p>The following year, the W3C membership decided to stop evolving HTML and instead begin work on
  an XML-based equivalent, called XHTML. <!-- https://www.w3.org/MarkUp/future/#summary --> This
  effort started with a reformulation of HTML4 in XML, known as XHTML 1.0, which added no new
  features except the new serialization, and which was completed in 2000. After XHTML 1.0, the W3C's
  focus turned to making it easier for other working groups to extend XHTML, under the banner of
  XHTML Modularization. In parallel with this, the W3C also worked on a new language that was not
  compatible with the earlier HTML and XHTML languages, calling it XHTML2.</p>

  <p>Around the time that HTML's evolution was stopped in 1998, parts of the API for HTML developed
  by browser vendors were specified and published under the name DOM Level 1 (in 1998) and DOM Level
  2 Core and DOM Level 2 HTML (starting in 2000 and culminating in 2003). These efforts then petered
  out, with some DOM Level 3 specifications published in 2004 but the working group being closed
  before all the Level 3 drafts were completed.</p>

  <p>In 2003, the publication of XForms, a technology which was positioned as the next generation of
  web forms, sparked a renewed interest in evolving HTML itself, rather than finding replacements
  for it. This interest was borne from the realization that XML's deployment as a web technology was
  limited to entirely new technologies (like RSS and later Atom), rather than as a replacement for
  existing deployed technologies (like HTML).</p>

  <p>A proof of concept to show that it was possible to extend HTML4's forms to provide many of the
  features that XForms 1.0 introduced, without requiring browsers to implement rendering engines
  that were incompatible with existing HTML web pages, was the first result of this renewed
  interest. At this early stage, while the draft was already publicly available, and input was
  already being solicited from all sources, the specification was only under Opera Software's
  copyright.</p>

  <p>The idea that HTML's evolution should be reopened was tested at a W3C workshop in 2004, where
  some of the principles that underlie the HTML5 work (described below), as well as the
  aforementioned early draft proposal covering just forms-related features, were presented to the
  W3C jointly by Mozilla and Opera. The proposal was rejected on the grounds that the proposal
  conflicted with the previously chosen direction for the web's evolution; the W3C staff and
  membership voted to continue developing XML-based replacements instead.</p>

  <p>Shortly thereafter, Apple, Mozilla, and Opera jointly announced their intent to continue
  working on the effort under the umbrella of a new venue called the WHATWG. A public mailing list
  was created, and the draft was moved to the WHATWG site. The copyright was subsequently amended to
  be jointly owned by all three vendors, and to allow reuse of the specification.</p>

  <p>The WHATWG was based on several core principles, in particular that technologies need to be
  backwards compatible, that specifications and implementations need to match even if this means
  changing the specification rather than the implementations, and that specifications need to be
  detailed enough that implementations can achieve complete interoperability without
  reverse-engineering each other.</p>

  <p>The latter requirement in particular required that the scope of the HTML5 specification include
  what had previously been specified in three separate documents: HTML4, XHTML1, and DOM2 HTML. It
  also meant including significantly more detail than had previously been considered the norm.</p>

  <p>In 2006, the W3C indicated an interest to participate in the development of HTML5 after all,
  and in 2007 formed a working group chartered to work with the WHATWG on the development of the
  HTML5 specification. Apple, Mozilla, and Opera allowed the W3C to publish the specification under
  the W3C copyright, while keeping a version with the less restrictive license on the WHATWG
  site.</p>

  <p>For a number of years, both groups then worked together. In 2011, however, the groups came to
  the conclusion that they had different goals: the W3C wanted to publish a "finished" version of
  "HTML5", while the WHATWG wanted to continue working on a Living Standard for HTML, continuously
  maintaining the specification rather than freezing it in a state with known problems, and adding
  new features as needed to evolve the platform.</p>

  <p>In 2019, the WHATWG and W3C <a href="https://www.w3.org/blog/news/archives/7753">signed an
  agreement</a> to collaborate on a single version of HTML going forward: this document.</p>



  <h3>Design notes</h3>

  <!-- NON-NORMATIVE SECTION -->

  <p>It must be admitted that many aspects of HTML appear at first glance to be nonsensical and
  inconsistent.</p>

  <p>HTML, its supporting DOM APIs, as well as many of its supporting technologies, have been
  developed over a period of several decades by a wide array of people with different priorities
  who, in many cases, did not know of each other's existence.</p>

  <p>Features have thus arisen from many sources, and have not always been designed in especially
  consistent ways. Furthermore, because of the unique characteristics of the web, implementation
  bugs have often become de-facto, and now de-jure, standards, as content is often unintentionally
  written in ways that rely on them before they can be fixed.</p>

  <p>Despite all this, efforts have been made to adhere to certain design goals. These are described
  in the next few subsections.</p>


  <div w-nodev>

  <!--en-GB--><h4 id="serialisability-of-script-execution">Serializability of script execution</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>To avoid exposing web authors to the complexities of multithreading, the HTML and DOM APIs are
  designed such that no script can ever detect the simultaneous execution of other scripts. Even
  with <span data-x="Worker">workers</span>, the intent is that the behavior of implementations can
  be thought of as completely serializing the execution of all scripts in all globals.</p>

  <p>The exception to this general design principle is the JavaScript <code>SharedArrayBuffer</code>
  class. Using <code>SharedArrayBuffer</code> objects, it can in fact be observed that scripts in
  other <span data-x="agent">agents</span> are executing simultaneously. Furthermore, due to the
  JavaScript memory model, there are situations which not only are un-representable via serialized
  <em>script</em> execution, but also un-representable via serialized <em>statement</em> execution
  among those scripts.</p>

  </div>



  <h4>Extensibility</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>HTML has a wide array of extensibility mechanisms that can be used for adding semantics in a
  safe manner:</p>

  <ul>
   <li><p>Authors can use the <code data-x="attr-class">class</code> attribute to extend elements,
   effectively creating their own elements, while using the most applicable existing "real" HTML
   element, so that browsers and other tools that don't know of the extension can still support it
   somewhat well. This is the tack used by microformats, for example.</p></li>

   <li><p>Authors can include data for inline client-side scripts or server-side site-wide scripts
   to process using the <code data-x="attr-data-*">data-*=""</code> attributes. These are guaranteed
   to never be touched by browsers, and allow scripts to include data on HTML elements that scripts
   can then look for and process.</p></li>

   <li><p>Authors can use the <code data-x="meta">&lt;meta name="" content=""></code> mechanism to
   include page-wide metadata.</p></li>

   <li><p>Authors can use the <code data-x="attr-hyperlink-rel">rel=""</code> mechanism to annotate
   links with specific meanings by registering <span data-x="concept-rel-extensions">extensions to
   the predefined set of link types</span>. This is also used by microformats.</p></li>

   <li><p>Authors can embed raw data using the <code data-x="script">&lt;script type=""></code>
   mechanism with a custom type, for further handling by inline or server-side scripts.</p></li>

   <li><p>Authors can extend APIs using the JavaScript prototyping mechanism. This is widely used by
   script libraries, for instance.</p></li>

   <li><p>Authors can use the microdata feature (the <code
   data-x="attr-itemscope">itemscope=""</code> and <code data-x="attr-itemprop">itemprop=""</code>
   attributes) to embed nested name-value pairs of data to be shared with other applications and
   sites.</p></li>

   <li><p>Authors can define, share, and use <span data-x="custom element">custom elements</span>
   to extend the vocabulary of HTML. The requirements of <span data-x="valid custom element
   name">valid custom element names</span> ensure forward compatibility (since no elements will be
   added to HTML, SVG, or MathML with hyphen-containing local names in the future).</p></li>
  </ul>




  <h3 id="html-vs-xhtml">HTML vs XML syntax</h3>

  <!-- NON-NORMATIVE SECTION -->

  <p>This specification defines an abstract language for describing documents and applications, and
  some APIs for interacting with in-memory representations of resources that use this language.</p>

  <p>The in-memory representation is known as "DOM HTML", or "the DOM" for short.</p>

  <p>There are various concrete syntaxes that can be used to transmit resources that use this
  abstract language, two of which are defined in this specification.</p>

  <p>The first such concrete syntax is the HTML syntax. This is the format suggested for most
  authors. It is compatible with most legacy web browsers. If a document is transmitted with the
  <code>text/html</code> <span>MIME type</span>, then it will be processed as an HTML document by
  web browsers. This specification defines the latest HTML syntax, known simply as "HTML".</p>

  <p>The second concrete syntax is XML. When a document is transmitted with an <span>XML MIME
  type</span>, such as <code>application/xhtml+xml</code>, then it is treated as an XML document by
  web browsers, to be parsed by an XML processor. Authors are reminded that the processing for XML
  and HTML differs; in particular, even minor syntax errors will prevent a document labeled as XML
  from being rendered fully, whereas they would be ignored in the HTML syntax.</p>

  <p class="note">The XML syntax for HTML was formerly referred to as "XHTML", but this
  specification does not use that term (among other reasons, because no such term is used for the
  HTML syntaxes of MathML and SVG).</p>

  <p>The DOM, the HTML syntax, and the XML syntax cannot all represent the same content. For
  example, namespaces cannot be represented using the HTML syntax, but they are supported in the DOM
  and in the XML syntax. Similarly, documents that use the <code>noscript</code> feature can be
  represented using the HTML syntax, but cannot be represented with the DOM or in the XML syntax.
  Comments that contain the string "<code data-x="">--&gt;</code>" can only be represented in the
  DOM, not in the HTML and XML syntaxes.</p>


  <h3>Structure of this specification</h3>

  <!-- NON-NORMATIVE SECTION -->

  <p>This specification is divided into the following major sections:</p>

  <dl>
   <dt><a href="#introduction">Introduction</a></dt>
   <dd>Non-normative materials providing a context for the HTML standard.</dd>

   <dt><a href="#infrastructure">Common infrastructure</a></dt>
   <dd>The conformance classes, algorithms, definitions, and the common underpinnings of the rest of
   the specification.</dd>

   <dt><a href="#dom">Semantics, structure, and APIs of HTML documents</a></dt>
   <dd>Documents are built from elements. These elements form a tree using the DOM. This section
   defines the features of this DOM, as well as introducing the features common to all elements, and
   the concepts used in defining elements.</dd>

   <dt><a href="#semantics">The elements of HTML</a></dt>
   <dd>Each element has a predefined meaning, which is explained in this section. Rules for authors
   on how to use the element<span w-nodev>, along with user agent requirements for how to
   handle each element,</span> are also given. This includes large signature features of HTML such
   as video playback and subtitles, form controls and form submission, and a 2D graphics API known
   as the HTML canvas.</dd>

   <dt><a href="#microdata">Microdata</a></dt>
   <dd>This specification introduces a mechanism for adding machine-readable annotations to
   documents, so that tools can extract trees of name-value pairs from the document. This section
   describes this mechanism<span w-nodev> and some algorithms that can be used to convert HTML
   documents into other formats</span>. This section also defines some sample Microdata vocabularies
   for contact information, calendar events, and licensing works.</dd>

   <dt><a href="#editing">User interaction</a></dt>
   <dd>HTML documents can provide a number of mechanisms for users to interact with and modify
   content, which are described in this section, such as how focus works, and drag-and-drop.</dd>

   <dt><a href="#browsers">Loading web pages</a></dt>
   <dd>HTML documents do not exist in a vacuum &mdash; this section defines many of the features
   that affect environments that deal with multiple pages, such as web browsers.</dd>

   <dt><a href="#webappapis">Web application APIs</a></dt>
   <dd>This section introduces basic features for scripting of applications in HTML.</dd>

   <dt><a href="#workers">Web workers</a></dt>
   <dd>This section defines an API for background threads in JavaScript.</dd>

   <dt><a href="#worklets">Worklets</a></dt>
   <dd>This section defines infrastructure for APIs that need to run JavaScript separately from the
   main JavaScript execution environment.</dd>

   <dt><a href="#comms">The communication APIs</a></dt>
   <dd>This section describes some mechanisms that applications written in HTML can use to
   communicate with other applications from different domains running on the same client. It also
   introduces a server-push event stream mechanism known as Server Sent Events or
   <code>EventSource</code>, and a two-way full-duplex socket protocol for scripts known as Web
   Sockets.</dd>

   <dt><a href="#webstorage">Web storage</a></dt>
   <dd>This section defines a client-side storage mechanism based on name-value pairs.</dd>

   <dt><a href="#syntax">The HTML syntax</a></dt>
   <dt><a href="#xhtml">The XML syntax</a></dt>
   <dd>All of these features would be for naught if they couldn't be represented in a serialized
   form and sent to other people, and so these sections define the syntaxes of HTML and XML<span
   w-nodev>, along with rules for how to parse content using those syntaxes</span>.</dd>

   <dt w-nodev><a href="#rendering">Rendering</a></dt>
   <dd w-nodev>This section defines the default rendering rules for web browsers.</dd>
  </dl>

  <p>There are also some appendices, listing <a href="#obsolete">obsolete features</a> and <a
  href="#iana">IANA considerations</a>, and several indices.</p>


  <div w-nodev>

  <h4>How to read this specification</h4>

  <p>This specification should be read like all other specifications. First, it should be read
  cover-to-cover, multiple times. Then, it should be read backwards at least once. Then it should be
  read by picking random sections from the contents list and following all the cross-references.</p>

  <p>As described in the conformance requirements section below, this specification describes
  conformance criteria for a variety of conformance classes. In particular, there are conformance
  requirements that apply to <em>producers</em>, for example authors and the documents they create,
  and there are conformance requirements that apply to <em>consumers</em>, for example web browsers.
  They can be distinguished by what they are requiring: a requirement on a producer states what is
  allowed, while a requirement on a consumer states how software is to act.</p>

  <div class="example">

   <p>For example, "the <code data-x="">foo</code> attribute's value must be a <span>valid
   integer</span>" is a requirement on producers, as it lays out the allowed values; in contrast,
   the requirement "the <code data-x="">foo</code> attribute's value must be parsed using the
   <span>rules for parsing integers</span>" is a requirement on consumers, as it describes how to
   process the content.</p>

  </div>

  <p><strong>Requirements on producers have no bearing whatsoever on consumers.</strong></p>

  <div class="example">

   <p>Continuing the above example, a requirement stating that a particular attribute's value is
   constrained to being a <span>valid integer</span> emphatically does <em>not</em> imply anything
   about the requirements on consumers. It might be that the consumers are in fact required to treat
   the attribute as an opaque string, completely unaffected by whether the value conforms to the
   requirements or not. It might be (as in the previous example) that the consumers are required to
   parse the value using specific rules that define how invalid (non-numeric in this case) values
   are to be processed.</p>

  </div>

  </div>



  <h4>Typographic conventions</h4>

  <p>This is a definition, requirement, or explanation.</p>

  <p class="note">This is a note.</p>

  <p class="example">This is an example.</p>

  <p class="XXX">This is an open issue.</p>

  <p class="warning">This is a warning.</p>

  <pre class="extract" w-nodev><code class="idl">[Exposed=Window]
interface <dfn data-x="">Example</dfn> {
  // this is an IDL definition
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>variable</var> = <var>object</var>.<span data-x="x-that">method</span>([<var>optionalArgument</var>])</code></dt>

   <dd><p>This is a note to authors describing the usage of an interface.</p></dd>
  </dl>

  <pre><code class="css">/* this is a CSS fragment */</code></pre>

  <p>The defining instance of a term is marked up like <dfn data-x="x-this">this</dfn>. Uses of that
  term are marked up like <span data-x="x-this">this</span> or like <i data-x="x-this">this</i>.</p>

  <p>The defining instance of an element, attribute, or API is marked up like <dfn><code
  data-x="x-that">this</code></dfn>. References to that element, attribute, or API are marked up
  like <code data-x="x-that">this</code>.</p>

  <p>Other code fragments are marked up <code data-x="">like this</code>.</p>

  <p>Variables are marked up like <var>this</var>.</p>

  <p w-nodev>In an algorithm, steps in <span data-x="synchronous section">synchronous
  sections</span> are marked with &#x231B;.</p>

  <p>In some cases, requirements are given in the form of lists with conditions and corresponding
  requirements. In such cases, the requirements that apply to a condition are always the first set
  of requirements that follow the condition, even in the case of there being multiple sets of
  conditions for those requirements. Such cases are presented as follows:</p>

  <dl class="switch">

   <dt>This is a condition
   <dt>This is another condition
   <dd>This is the requirement that applies to the conditions above.

   <dt>This is a third condition
   <dd>This is the requirement that applies to the third condition.
  </dl>




  <h3>A quick introduction to HTML</h3>

  <!-- NON-NORMATIVE SECTION -->

  <p>A basic HTML document looks like this:</p>

  <pre id="intro-early-example"><code class="html">&lt;!DOCTYPE html>
&lt;html lang="en">
 &lt;head>
  &lt;title>Sample page&lt;/title>
 &lt;/head>
 &lt;body>
  &lt;h1>Sample page&lt;/h1>
  &lt;p>This is a &lt;a href="demo.html">simple&lt;/a> sample.&lt;/p>
  &lt;!-- this is a comment -->
 &lt;/body>
&lt;/html></code></pre>

  <p>HTML documents consist of a tree of elements and text. Each element is denoted in the source by
  a <span data-x="syntax-start-tag">start tag</span>, such as "<code data-x="">&lt;body></code>", and
  an <span data-x="syntax-end-tag">end tag</span>, such as "<code data-x="">&lt;/body></code>".
  (Certain start tags and end tags can in certain cases be <span
  data-x="syntax-tag-omission">omitted</span> and are implied by other tags.)</p>

  <p>Tags have to be nested such that elements are all completely within each other, without
  overlapping:</p>

  <pre class="bad"><code class="html">&lt;p>This is &lt;em>very &lt;strong>wrong&lt;/em>!&lt;/strong>&lt;/p></code></pre>
  <pre><code class="html">&lt;p>This &lt;em>is &lt;strong>correct&lt;/strong>.&lt;/em>&lt;/p></code></pre>

  <p>This specification defines a set of elements that can be used in HTML, along with rules about
  the ways in which the elements can be nested.</p>

  <p>Elements can have attributes, which control how the elements work. In the example below, there
  is a <span>hyperlink</span>, formed using the <code>a</code> element and its <code
  data-x="attr-hyperlink-href">href</code> attribute:</p>

  <pre><code class="html">&lt;a href="demo.html">simple&lt;/a></code></pre>

  <p><span data-x="syntax-attributes">Attributes</span> are placed inside the start tag, and consist
  of a <span data-x="syntax-attribute-name">name</span> and a <span
  data-x="syntax-attribute-value">value</span>, separated by an "<code data-x="">=</code>" character.
  The attribute value can remain <a href="#unquoted">unquoted</a> if it doesn't contain <span>ASCII
  whitespace</span> or any of <code data-x="">"</code> <code data-x="">'</code> <code
  data-x="">`</code> <code data-x="">=</code> <code data-x="">&lt;</code> or <code
  data-x="">&gt;</code>. Otherwise, it has to be quoted using either single or double quotes. The
  value, along with the "<code data-x="">=</code>" character, can be omitted altogether if the value
  is the empty string.</p>

  <pre><code class="html">&lt;!-- empty attributes -->
&lt;input name=address disabled>
&lt;input name=address disabled="">

&lt;!-- attributes with a value -->
&lt;input name=address maxlength=200>
&lt;input name=address maxlength='200'>
&lt;input name=address maxlength="200"></code></pre>

  <p>HTML user agents (e.g., web browsers) then <i>parse</i> this markup, turning it into a DOM
  (Document Object Model) tree. A DOM tree is an in-memory representation of a document.</p>

  <p>DOM trees contain several kinds of nodes, in particular a <code>DocumentType</code> node,
  <code>Element</code> nodes, <code>Text</code> nodes, <code>Comment</code> nodes, and in some cases
  <code>ProcessingInstruction</code> nodes.</p>

  <p>The <a href="#intro-early-example">markup snippet at the top of this section</a> would be
  turned into the following DOM tree:</p>

  <ul class="domTree"><li class="t10">DOCTYPE: <code data-x="">html</code></li><li class="t1"><code>html</code> <span data-x="" class="t2"><code class="attribute name" data-x="attr-lang">lang</code>="<code class="attribute value" data-x="">en</code>"</span><ul><li class="t1"><code>head</code><ul><li class="t3"><code>#text</code>: <span data-x="">&#x23CE;&#x2423;&#x2423;</span></li><li class="t1"><code>title</code><ul><li class="t3"><code>#text</code>: <span data-x="">Sample page</span></li></ul></li><li class="t3"><code>#text</code>: <span data-x="">&#x23CE;&#x2423;</span></li></ul></li><li class="t3"><code>#text</code>: <span data-x="">&#x23CE;&#x2423;</span></li><li class="t1"><code>body</code><ul><li class="t3"><code>#text</code>: <span data-x="">&#x23CE;&#x2423;&#x2423;</span></li><li class="t1"><code>h1</code><ul><li class="t3"><code>#text</code>: <span data-x="">Sample page</span></li></ul></li><li class="t3"><code>#text</code>: <span data-x="">&#x23CE;&#x2423;&#x2423;</span></li><li class="t1"><code>p</code><ul><li class="t3"><code>#text</code>: <span data-x="">This is a <!--grammar-check-override--></span></li><li class="t1"><code>a</code> <span data-x="" class="t2"><code class="attribute name" data-x="attr-hyperlink-href">href</code>="<code class="attribute value" data-x="">demo.html</code>"</span><ul><li class="t3"><code>#text</code>: <span data-x="">simple</span></li></ul></li><li class="t3"><code>#text</code>: <span data-x=""> sample.</span></li></ul></li><li class="t3"><code>#text</code>: <span data-x="">&#x23CE;&#x2423;&#x2423;</span></li><li class="t8"><code>#comment</code>: <span data-x=""> this is a comment </span></li><li class="t3"><code>#text</code>: <span data-x="">&#x23CE;&#x2423;&#x23CE;</span></li></ul></li></ul></li></ul>

  <p>The <span>document element</span> of this tree is the <code>html</code> element, which is the
  element always found in that position in HTML documents. It contains two elements,
  <code>head</code> and <code>body</code>, as well as a <code>Text</code> node between them.</p>

  <p>There are many more <code>Text</code> nodes in the DOM tree than one would initially expect,
  because the source contains a number of spaces (represented here by "&#x2423;") and line breaks
  ("&#x23CE;") that all end up as <code>Text</code> nodes in the DOM. However, for historical
  reasons not all of the spaces and line breaks in the original markup appear in the DOM. In
  particular, all the whitespace before <code>head</code> start tag ends up being dropped silently,
  and all the whitespace after the <code>body</code> end tag ends up placed at the end of the
  <code>body</code>.</p>

  <p>The <code>head</code> element contains a <code>title</code> element, which itself contains a
  <code>Text</code> node with the text "Sample page". Similarly, the <code>body</code> element
  contains an <code>h1</code> element, a <code>p</code> element, and a comment.</p>

  <hr>

  <p>This DOM tree can be manipulated from scripts in the page. Scripts (typically in JavaScript)
  are small programs that can be embedded using the <code>script</code> element or using <span>event
  handler content attributes</span>. For example, here is a form with a script that sets the value
  of the form's <code>output</code> element to say "Hello World":</p>

  <pre><code class="html">&lt;<span>form</span> <span data-x="attr-form-name">name</span>="main">
 Result: &lt;<span>output</span> <span data-x="attr-fe-name">name</span>="result">&lt;/output>
 &lt;<span>script</span>>
  <span data-x="Document">document</span>.<span data-x="dom-document-forms">forms</span>.main.<span data-x="dom-form-elements">elements</span>.result.<span data-x="dom-output-value">value</span> = 'Hello World';
 &lt;/script>
&lt;/form></code></pre>

  <p>Each element in the DOM tree is represented by an object, and these objects have APIs so that
  they can be manipulated. For instance, a link (e.g. the <code>a</code> element in the tree above)
  can have its "<code data-x="attr-hyperlink-href">href</code>" attribute changed in several
  ways:</p>

  <pre><code class="js">var a = <span data-x="Document">document</span>.<span data-x="dom-document-links">links</span>[0]; // obtain the first link in the document
a.<span data-x="dom-hyperlink-href">href</span> = 'sample.html'; // change the destination URL of the link
a.<span data-x="dom-hyperlink-protocol">protocol</span> = 'https'; // change just the scheme part of the URL
a.setAttribute('href', 'https://example.com/'); // change the content attribute directly</code></pre>

  <p>Since DOM trees are used as the way to represent HTML documents when they are processed and
  presented by implementations (especially interactive implementations like web browsers), this
  specification is mostly phrased in terms of DOM trees, instead of the markup described above.</p>

  <hr>

  <p>HTML documents represent a media-independent description of interactive content. HTML documents
  might be rendered to a screen, or through a speech synthesizer, or on a braille display. To
  influence exactly how such rendering takes place, authors can use a styling language such as
  CSS.</p>

  <p>In the following example, the page has been made yellow-on-blue using CSS.</p>

  <pre><code class="html">&lt;!DOCTYPE html>
&lt;html lang="en">
 &lt;head>
  &lt;title>Sample styled page&lt;/title>
  &lt;style>
   body { background: navy; color: yellow; }
  &lt;/style>
 &lt;/head>
 &lt;body>
  &lt;h1>Sample styled page&lt;/h1>
  &lt;p>This page is just a demo.&lt;/p>
 &lt;/body>
&lt;/html></code></pre>

  <p>For more details on how to use HTML, authors are encouraged to consult tutorials and guides.
  Some of the examples included in this specification might also be of use, but the novice author is
  cautioned that this specification, by necessity, defines the language with a level of detail that
  might be difficult to understand at first.</p>



  <h4>Writing secure applications with HTML</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>When HTML is used to create interactive sites, care needs to be taken to avoid introducing
  vulnerabilities through which attackers can compromise the integrity of the site itself or of the
  site's users.</p>

  <p>A comprehensive study of this matter is beyond the scope of this document, and authors are
  strongly encouraged to study the matter in more detail. However, this section attempts to provide
  a quick introduction to some common pitfalls in HTML application development.</p>

  <p>The security model of the web is based on the concept of "origins", and correspondingly many of
  the potential attacks on the web involve cross-origin actions. <ref>ORIGIN</ref></p>

  <dl>
   <dt>Not validating user input</dt>
   <dt>Cross-site scripting (XSS)</dt>
   <dt>SQL injection</dt>

   <dd>
    <p>When accepting untrusted input, e.g. user-generated content such as text comments, values in
    URL parameters, messages from third-party sites, etc, it is imperative that the data be
    validated before use, and properly escaped when displayed. Failing to do this can allow a
    hostile user to perform a variety of attacks, ranging from the potentially benign, such as
    providing bogus user information like a negative age, to the serious, such as running scripts
    every time a user looks at a page that includes the information, potentially propagating the
    attack in the process, to the catastrophic, such as deleting all data in the server.</p>

    <p>When writing filters to validate user input, it is imperative that filters always be
    safelist-based, allowing known-safe constructs and disallowing all other input. Blocklist-based
    filters that disallow known-bad inputs and allow everything else are not secure, as not
    everything that is bad is yet known (for example, because it might be invented in the
    future).</p>

    <div class="example">

     <p>For example, suppose a page looked at its URL's query string to determine what to display,
     and the site then redirected the user to that page to display a message, as in:</p>

     <pre><code class="html">&lt;ul>
 &lt;li>&lt;a href="message.cgi?say=Hello">Say Hello&lt;/a>
 &lt;li>&lt;a href="message.cgi?say=Welcome">Say Welcome&lt;/a>
 &lt;li>&lt;a href="message.cgi?say=Kittens">Say Kittens&lt;/a>
&lt;/ul></code></pre>

     <p>If the message was just displayed to the user without escaping, a hostile attacker could
     then craft a URL that contained a script element:</p>

     <pre>https://example.com/message.cgi?say=%3Cscript%3Ealert%28%27Oh%20no%21%27%29%3C/script%3E</pre>

     <p>If the attacker then convinced a victim user to visit this page, a script of the attacker's
     choosing would run on the page. Such a script could do any number of hostile actions, limited
     only by what the site offers: if the site is an e-commerce shop, for instance, such a script
     could cause the user to unknowingly make arbitrarily many unwanted purchases.</p>

     <p>This is called a cross-site scripting attack.</p>

    </div>

    <p>There are many constructs that can be used to try to trick a site into executing code. Here
    are some that authors are encouraged to consider when writing safelist filters:</p>

    <ul>
     <li>When allowing harmless-seeming elements like <code>img</code>, it is important to safelist
     any provided attributes as well. If one allowed all attributes then an attacker could, for
     instance, use the <code data-x="handler-onload">onload</code> attribute to run arbitrary
     script.</li>

     <li>When allowing URLs to be provided (e.g. for links), the scheme of each URL also needs to be
     explicitly safelisted, as there are many schemes that can be abused. The most prominent
     example is "<code data-x="javascript protocol">javascript:</code>", but user agents can
     implement (and indeed, have historically implemented) others.</li> <!-- IE had vbscript:,
     Netscape had livescript:, etc. -->

     <li>Allowing a <code>base</code> element to be inserted means any <code>script</code> elements
     in the page with relative links can be hijacked, and similarly that any form submissions can
     get redirected to a hostile site.</li>
    </ul>
   </dd>


   <dt>Cross-site request forgery (CSRF)</dt>

   <dd>
    <p>If a site allows a user to make form submissions with user-specific side-effects, for example
    posting messages on a forum under the user's name, making purchases, or applying for a passport,
    it is important to verify that the request was made by the user intentionally, rather than by
    another site tricking the user into making the request unknowingly.</p>

    <p>This problem exists because HTML forms can be submitted to other origins.</p>

    <p>Sites can prevent such attacks by populating forms with user-specific hidden tokens, or by
    checking `<code data-x="http-origin">Origin</code>` headers on all requests.</p>
   </dd>



   <dt>Clickjacking</dt>

   <dd>
    <p>A page that provides users with an interface to perform actions that the user might not wish
    to perform needs to be designed so as to avoid the possibility that users can be tricked into
    activating the interface.</p>

    <p>One way that a user could be so tricked is if a hostile site places the victim site in a
    small <code>iframe</code> and then convinces the user to click, for instance by having the user
    play a reaction game. Once the user is playing the game, the hostile site can quickly position
    the iframe under the mouse cursor just as the user is about to click, thus tricking the user
    into clicking the victim site's interface.</p>

    <p>To avoid this, sites that do not expect to be used in frames are encouraged to only enable
    their interface if they detect that they are not in a frame (e.g. by comparing the <code
    data-x="dom-window">window</code> object to the value of the <code data-x="dom-top">top</code>
    attribute).</p>
   </dd>
  </dl>


  <h4>Common pitfalls to avoid when using the scripting APIs</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>Scripts in HTML have "run-to-completion" semantics, meaning that the browser will generally run
  the script uninterrupted before doing anything else, such as firing further events or continuing
  to parse the document.</p>

  <p>On the other hand, parsing of HTML files happens incrementally, meaning that
  the parser can pause at any point to let scripts run. This is generally a good thing, but it does
  mean that authors need to be careful to avoid hooking event handlers after the events could have
  possibly fired.</p>

  <p>There are two techniques for doing this reliably: use <span>event handler content
  attributes</span>, or create the element and add the event handlers in the same script. The latter
  is safe because, as mentioned earlier, scripts are run to completion before further events can
  fire.</p>

  <div class="example">

   <p>One way this could manifest itself is with <code>img</code> elements and the <code
   data-x="event-load">load</code> event. The event could fire as soon as the element has been
   parsed, especially if the image has already been cached (which is common).</p>

   <p>Here, the author uses the <code data-x="handler-onload">onload</code> handler on an
   <code>img</code> element to catch the <code data-x="event-load">load</code> event:</p>

   <pre><code class="html">&lt;img src="games.png" alt="Games" onload="gamesLogoHasLoaded(event)"></code></pre>

   <p>If the element is being added by script, then so long as the event handlers are added in the
   same script, the event will still not be missed:</p>

   <pre><code class="html">&lt;script>
 var img = new Image();
 img.src = 'games.png';
 img.alt = 'Games';
 img.onload = gamesLogoHasLoaded;
 // img.addEventListener('load', gamesLogoHasLoaded, false); // would work also
&lt;/script></code></pre>

   <p>However, if the author first created the <code>img</code> element and then in a separate
   script added the event listeners, there's a chance that the <code data-x="event-load">load</code>
   event would be fired in between, leading it to be missed:</p>

   <pre class="bad"><code class="html">&lt;!-- Do not use this style, it has a race condition! -->
 &lt;img id="games" src="games.png" alt="Games">
 &lt;!-- the 'load' event might fire here while the parser is taking a
      break, in which case you will not see it! -->
 &lt;script>
  var img = document.getElementById('games');
  img.onload = gamesLogoHasLoaded; // might never fire!
 &lt;/script></code></pre>

  </div>



  <h4>How to catch mistakes when writing HTML: validators and conformance checkers</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>Authors are encouraged to make use of conformance checkers (also known as <i>validators</i>) to
  catch common mistakes. The WHATWG maintains a list of such tools at: <a
  href="https://whatwg.org/validator/">https://whatwg.org/validator/</a></p>



  <h3>Conformance requirements for authors</h3>

  <!-- NON-NORMATIVE SECTION -->

  <p>Unlike previous versions of the HTML specification, this specification defines in some detail
  the required processing for invalid documents as well as valid documents.</p> <!-- This has led to
  some questioning the purpose of conformance criteria: if there is no ambiguity in how something
  will be processed, why disallow it? -->

  <p>However, even though the processing of invalid content is in most cases well-defined,
  conformance requirements for documents are still important: in practice, interoperability (the
  situation in which all implementations process particular content in a reliable and identical or
  equivalent way) is not the only goal of document conformance requirements. This section details
  some of the more common reasons for still distinguishing between a conforming document and one
  with errors.</p>


  <h4>Presentational markup</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>The majority of presentational features from previous versions of HTML are no longer allowed.
  Presentational markup in general has been found to have a number of problems:</p>

  <dl>
   <dt>The use of presentational elements leads to poorer accessibility</dt>

   <dd>
    <p>While it is possible to use presentational markup in a way that provides users of assistive
    technologies (ATs) with an acceptable experience (e.g. using ARIA), doing so is significantly
    more difficult than doing so when using semantically-appropriate markup. Furthermore, even using
    such techniques doesn't help make pages accessible for non-AT non-graphical users, such as users
    of text-mode browsers.</p>

    <p>Using media-independent markup, on the other hand, provides an easy way for documents to be
    authored in such a way that they work for more users (e.g. users of text browsers).</p>
   </dd>


   <dt>Higher cost of maintenance</dt>

   <dd>
    <p>It is significantly easier to maintain a site written in such a way that the markup is
    style-independent. For example, changing the color of a site that uses <code
    data-x="">&lt;font&nbsp;color=""></code> throughout requires changes across the entire site,
    whereas a similar change to a site based on CSS can be done by changing a single file.</p>
   </dd>


   <dt>Larger document sizes</dt>

   <dd>
    <p>Presentational markup tends to be much more redundant, and thus results in larger document
    sizes.</p>
   </dd>
  </dl>

  <p>For those reasons, presentational markup has been removed from HTML in this version. This
  change should not come as a surprise; HTML4 deprecated presentational markup many years ago and
  provided a mode (HTML4 Transitional) to help authors move away from presentational markup; later,
  XHTML 1.1 went further and obsoleted those features altogether.</p>

  <p>The only remaining presentational markup features in HTML are the <code
  data-x="attr-style">style</code> attribute and the <code>style</code> element. Use of the <code
  data-x="attr-style">style</code> attribute is somewhat discouraged in production environments, but
  it can be useful for rapid prototyping (where its rules can be directly moved into a separate
  style sheet later) and for providing specific styles in unusual cases where a separate style sheet
  would be inconvenient. Similarly, the <code>style</code> element can be useful in syndication or
  for page-specific styles, but in general an external style sheet is likely to be more convenient
  when the styles apply to multiple pages.</p>

  <p>It is also worth noting that some elements that were previously presentational have been
  redefined in this specification to be media-independent: <code>b</code>, <code>i</code>,
  <code>hr</code>, <code>s</code>, <code>small</code>, and <code>u</code>.</p>


  <h4>Syntax errors</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>The syntax of HTML is constrained to avoid a wide variety of problems.</p>

  <dl>
   <dt>Unintuitive error-handling behavior</dt>

   <dd>
    <p>Certain invalid syntax constructs, when parsed, result in DOM trees that are highly
    unintuitive.</p>

    <div class="example">

     <p>For example, the following markup fragment results in a DOM with an <code>hr</code> element
     that is an <em>earlier</em> sibling of the corresponding <code>table</code> element:</p>

     <pre class="bad"><code class="html">&lt;table>&lt;hr>...</code></pre>

    </div>
   </dd>


   <dt>Errors with optional error recovery</dt>

   <dd>
    <p>To allow user agents to be used in controlled environments without having to implement the
    more bizarre and convoluted error handling rules, user agents are permitted to fail whenever
    encountering a <span>parse error</span>.</p>
   </dd>


   <dt>Errors where the error-handling behavior is not compatible with streaming user agents</dt>

   <dd>
    <p>Some error-handling behavior, such as the behavior for the <code
    data-x="">&lt;table>&lt;hr>...</code> example mentioned above, are incompatible with streaming
    user agents (user agents that process HTML files in one pass, without storing state). To avoid
    interoperability problems with such user agents, any syntax resulting in such behavior is
    considered invalid.</p>
   </dd>


   <dt>Errors that can result in infoset coercion</dt>

   <dd>
    <p>When a user agent based on XML is connected to an HTML parser, it is possible that certain
    invariants that XML enforces, such as element or attribute names never contain multiple colons,
    will be violated by an HTML file. Handling this can require that the parser coerce the HTML DOM
    into an XML-compatible infoset. Most syntax constructs that require such handling are considered
    invalid. (Comments containing two consecutive hyphens, or ending with a hyphen, are exceptions
    that are allowed in the HTML syntax.)</p>
   </dd>


   <dt>Errors that result in disproportionately poor performance</dt>

   <dd>
    <p>Certain syntax constructs can result in disproportionately poor performance. To discourage the
    use of such constructs, they are typically made non-conforming.</p>

    <div class="example">

     <p>For example, the following markup results in poor performance, since all the unclosed
     <code>i</code> elements have to be reconstructed in each paragraph, resulting in progressively
     more elements in each paragraph:</p>

     <pre class="bad"><code class="html">&lt;p>&lt;i>She dreamt.
&lt;p>&lt;i>She dreamt that she ate breakfast.
&lt;p>&lt;i>Then lunch.
&lt;p>&lt;i>And finally dinner.</code></pre>

     <p>The resulting DOM for this fragment would be:</p>

     <ul class="domTree"><li class="t1"><code>p</code><ul><li class="t1"><code>i</code><ul><li class="t3"><code>#text</code>: <span data-x="">She dreamt.</span></li></ul></li></ul></li><li class="t1"><code>p</code><ul><li class="t1"><code>i</code><ul><li class="t1"><code>i</code><ul><li class="t3"><code>#text</code>: <span data-x="">She dreamt that she ate breakfast.</span></li></ul></li></ul></li></ul></li><li class="t1"><code>p</code><ul><li class="t1"><code>i</code><ul><li class="t1"><code>i</code><ul><li class="t1"><code>i</code><ul><li class="t3"><code>#text</code>: <span data-x="">Then lunch.</span></li></ul></li></ul></li></ul></li></ul></li><li class="t1"><code>p</code><ul><li class="t1"><code>i</code><ul><li class="t1"><code>i</code><ul><li class="t1"><code>i</code><ul><li class="t1"><code>i</code><ul><li class="t3"><code>#text</code>: <span data-x="">And finally dinner.</span></li></ul></li></ul></li></ul></li></ul></li></ul></li></ul>

    </div>
   </dd>


   <dt>Errors involving fragile syntax constructs</dt>

   <dd>
    <p>There are syntax constructs that, for historical reasons, are relatively fragile. To help
    reduce the number of users who accidentally run into such problems, they are made
    non-conforming.</p>

    <div class="example">

     <p>For example, the parsing of certain named character references in attributes happens even
     with the closing semicolon being omitted. It is safe to include an ampersand followed by
     letters that do not form a named character reference, but if the letters are changed to a
     string that <em>does</em> form a named character reference, they will be interpreted as that
     character instead.</p>

     <p>In this fragment, the attribute's value is "<code data-x="">?bill&amp;ted</code>":</p>

     <pre class="bad"><code class="html">&lt;a href="?bill&amp;ted">Bill and Ted&lt;/a></code></pre>

     <p>In the following fragment, however, the attribute's value is actually "<code
     data-x="">?art&copy;</code>", <em>not</em> the intended "<code data-x="">?art&amp;copy</code>",
     because even without the final semicolon, "<code data-x="">&amp;copy</code>" is handled the same
     as "<code data-x="">&amp;copy;</code>" and thus gets interpreted as "<code
     data-x="">&copy;</code>":</p>

     <pre class="bad"><code class="html">&lt;a href="?art&amp;copy">Art and Copy&lt;/a></code></pre>

     <p>To avoid this problem, all named character references are required to end with a semicolon,
     and uses of named character references without a semicolon are flagged as errors.</p>

     <p>Thus, the correct way to express the above cases is as
     follows:</p>

     <pre><code class="html">&lt;a href="?bill&amp;ted">Bill and Ted&lt;/a> &lt;!-- &amp;ted is ok, since it's not a named character reference --></code></pre>
     <pre><code class="html">&lt;a href="?art&amp;amp;copy">Art and Copy&lt;/a> &lt;!-- the &amp; has to be escaped, since &amp;copy <em>is</em> a named character reference --></code></pre>

    </div>
   </dd>


   <dt>Errors involving known interoperability problems in legacy user agents</dt>

   <dd>
    <p>Certain syntax constructs are known to cause especially subtle or serious problems in legacy
    user agents, and are therefore marked as non-conforming to help authors avoid them.</p>

    <div class="example">

     <p>For example, this is why the U+0060 GRAVE ACCENT character (`) is not allowed in unquoted
     attributes. In certain legacy user agents, <!-- namely IE --> it is sometimes treated as a
     quote character.</p>

    </div>

    <div class="example">

     <p>Another example of this is the DOCTYPE, which is required to trigger <span>no-quirks
     mode</span>, because the behavior of legacy user agents in <span>quirks mode</span> is often
     largely undocumented.</p>

    </div>
   </dd>


   <dt>Errors that risk exposing authors to security attacks</dt>

   <dd>
    <p>Certain restrictions exist purely to avoid known security problems.</p>

    <div class="example">

     <p>For example, the restriction on using UTF-7 exists purely to avoid authors falling prey to a
     known cross-site-scripting attack using UTF-7. <ref>UTF7</ref></p>

    </div>
   </dd>


   <dt>Cases where the author's intent is unclear</dt>

   <dd>
    <p>Markup where the author's intent is very unclear is often made non-conforming. Correcting
    these errors early makes later maintenance easier.</p>

    <div class="example">

     <p>For example, it is unclear whether the author intended the following to be an
     <code>h1</code> heading or an <code>h2</code> heading:</p>

     <pre class="bad"><code class="html">&lt;h1>Contact details&lt;/h2></code></pre>

    </div>
   </dd>


   <dt>Cases that are likely to be typos</dt>

   <dd>
    <p>When a user makes a simple typo, it is helpful if the error can be caught early, as this can
    save the author a lot of debugging time. This specification therefore usually considers it an
    error to use element names, attribute names, and so forth, that do not match the names defined
    in this specification.</p>

    <div class="example">

     <p>For example, if the author typed <code data-x="">&lt;capton></code> instead of <code
     data-x="">&lt;caption></code>, this would be flagged as an error and the author could correct
     the typo immediately.</p>

    </div>
   </dd>


   <dt>Errors that could interfere with new syntax in the future</dt>

   <dd>
    <p>In order to allow the language syntax to be extended in the future, certain otherwise
    harmless features are disallowed.</p>

    <div class="example">

     <p>For example, "attributes" in end tags are ignored currently, but they are invalid, in case a
     future change to the language makes use of that syntax feature without conflicting with
     already-deployed (and valid!) content.</p>

    </div>
   </dd>
  </dl>

  <p>Some authors find it helpful to be in the practice of always quoting all attributes and always
  including all optional tags, preferring the consistency derived from such custom over the minor
  benefits of terseness afforded by making use of the flexibility of the HTML syntax. To aid such
  authors, conformance checkers can provide modes of operation wherein such conventions are
  enforced.</p>



  <h4>Restrictions on content models and on attribute values</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>Beyond the syntax of the language, this specification also places restrictions on how elements
  and attributes can be specified. These restrictions are present for similar reasons:</p>

  <dl>

   <dt>Errors involving content with dubious semantics</dt>

   <dd>
    <p>To avoid misuse of elements with defined meanings, content models are defined that restrict
    how elements can be nested when such nestings would be of dubious value.</p>

    <p class="example">For example, this specification disallows nesting a <code>section</code>
    element inside a <code>kbd</code> element, since it is highly unlikely for an author to indicate
    that an entire section <!--non-normative-->should be keyed in.</p>
   </dd>


   <dt>Errors that involve a conflict in expressed semantics</dt>

   <dd>
    <p>Similarly, to draw the author's attention to mistakes in the use of elements, clear
    contradictions in the semantics expressed are also considered conformance errors.</p>

    <div class="example">

     <p>In the fragments below, for example, the semantics are nonsensical: a separator cannot
     simultaneously be a cell, nor can a radio button be a progress bar.</p>

     <pre class="bad"><code class="html">&lt;hr role="cell"></code></pre>
     <pre class="bad"><code class="html">&lt;input type=radio role=progressbar></code></pre>

    </div>

    <p class="example">Another example is the restrictions on the content models of the
    <code>ul</code> element, which only allows <code>li</code> element children. Lists by definition
    consist just of zero or more list items, so if a <code>ul</code> element contains something
    other than an <code>li</code> element, it's not clear what was meant.</p>
   </dd>


   <dt>Cases where the default styles are likely to lead to confusion</dt>

   <dd>
    <p>Certain elements have default styles or behaviors that make certain combinations likely to
    lead to confusion. Where these have equivalent alternatives without this problem, the confusing
    combinations are disallowed.</p>

    <p class="example">For example, <code>div</code> elements are rendered as <span
    data-x="block box">block boxes</span>, and <code>span</code> elements as <span
    data-x="inline box">inline boxes</span>. Putting a <span>block box</span> in an
    <span>inline box</span> is unnecessarily confusing; since either nesting just <code>div</code>
    elements, or nesting just <code>span</code> elements, or nesting <code>span</code> elements
    inside <code>div</code> elements all serve the same purpose as nesting a <code>div</code>
    element in a <code>span</code> element, but only the latter involves a <span>block box</span> in
    an <span>inline box</span>, the latter combination is disallowed.</p>

    <p class="example">Another example would be the way <span>interactive content</span> cannot be
    nested. For example, a <code>button</code> element cannot contain a <code>textarea</code>
    element. This is because the default behavior of such nesting interactive elements would be
    highly confusing to users. Instead of nesting these elements, they can be placed side by
    side.</p>
   </dd>


   <dt>Errors that indicate a likely misunderstanding of the specification</dt>

   <dd>
    <p>Sometimes, something is disallowed because allowing it would likely cause author
    confusion.</p>

    <p class="example">For example, setting the <code data-x="attr-fe-disabled">disabled</code>
    attribute to the value "<code data-x="">false</code>" is disallowed, because despite the
    appearance of meaning that the element is enabled, it in fact means that the element is
    <em>disabled</em> (what matters for implementations is the presence of the attribute, not its
    value).</p>
   </dd>


   <dt>Errors involving limits that have been imposed merely to simplify the language</dt>

   <dd>
    <p>Some conformance errors simplify the language that authors need to learn.</p>

    <p class="example">For example, the <code>area</code> element's <code
    data-x="attr-area-shape">shape</code> attribute, despite accepting both <code
    data-x="attr-area-shape-keyword-circ">circ</code> and <code
    data-x="attr-area-shape-keyword-circle">circle</code> values in practice as synonyms, disallows
    the use of the <code data-x="attr-area-shape-keyword-circ">circ</code> value, so as to simplify
    tutorials and other learning aids. There would be no benefit to allowing both, but it would
    cause extra confusion when teaching the language.</p>
   </dd>


   <dt>Errors that involve peculiarities of the parser</dt>

   <dd>
    <p>Certain elements are parsed in somewhat eccentric ways (typically for historical reasons),
    and their content model restrictions are intended to avoid exposing the author to these
    issues.</p>

    <div class="example">

     <p>For example, a <code>form</code> element isn't allowed inside <span>phrasing content</span>,
     because when parsed as HTML, a <code>form</code> element's start tag will imply a
     <code>p</code> element's end tag. Thus, the following markup results in two <span
     data-x="paragraph">paragraphs</span>, not one:</p>

     <pre><code class="html">&lt;p>Welcome. &lt;form>&lt;label>Name:&lt;/label> &lt;input>&lt;/form></code></pre>

     <p>It is parsed exactly like the following:</p>

     <pre><code class="html">&lt;p>Welcome. &lt;/p>&lt;form>&lt;label>Name:&lt;/label> &lt;input>&lt;/form></code></pre>

    </div>
   </dd>


   <dt>Errors that would likely result in scripts failing in hard-to-debug ways</dt>

   <dd>
    <p>Some errors are intended to help prevent script problems that would be hard to debug.</p>

    <p class="example">This is why, for instance, it is non-conforming to have two <code
    data-x="attr-id">id</code> attributes with the same value. Duplicate IDs lead to the wrong
    element being selected, with sometimes disastrous effects whose cause is hard to determine.</p>
   </dd>


   <dt>Errors that waste authoring time</dt>

   <dd>
    <p>Some constructs are disallowed because historically they have been the cause of a lot of
    wasted authoring time, and by encouraging authors to avoid making them, authors can save time in
    future efforts.</p>

    <p class="example">For example, a <code>script</code> element's <code
    data-x="attr-script-src">src</code> attribute causes the element's contents to be ignored.
    However, this isn't obvious, especially if the element's contents appear to be executable script
    &mdash; which can lead to authors spending a lot of time trying to debug the inline script
    without realizing that it is not executing. To reduce this problem, this specification makes it
    non-conforming to have executable script in a <code>script</code> element when the <code
    data-x="attr-script-src">src</code> attribute is present. This means that authors who are
    validating their documents are less likely to waste time with this kind of mistake.</p>
   </dd>


   <dt>Errors that involve areas that affect authors migrating between the HTML and XML
   syntaxes</dt>

   <dd>
    <p>Some authors like to write files that can be interpreted as both XML and HTML with similar
    results. Though this practice is discouraged in general due to the myriad of subtle
    complications involved (especially when involving scripting, styling, or any kind of automated
    serialization), this specification has a few restrictions intended to at least somewhat mitigate
    the difficulties. This makes it easier for authors to use this as a transitionary step when
    migrating between the HTML and XML syntaxes.</p>

    <p class="example">For example, there are somewhat complicated rules surrounding the <code
    data-x="attr-lang">lang</code> and <code data-x="attr-xml-lang">xml:lang</code> attributes
    intended to keep the two synchronized.</p>

    <p class="example">Another example would be the restrictions on the values of <code
    data-x="">xmlns</code> attributes in the HTML serialization, which are intended to ensure that
    elements in conforming documents end up in the same namespaces whether processed as HTML or
    XML.</p>
   </dd>


   <dt>Errors that involve areas reserved for future expansion</dt>

   <dd>
    <p>As with the restrictions on the syntax intended to allow for new syntax in future revisions
    of the language, some restrictions on the content models of elements and values of attributes
    are intended to allow for future expansion of the HTML vocabulary.</p>

    <p class="example">For example, limiting the values of the <code
    data-x="attr-hyperlink-target">target</code> attribute that start with an U+005F LOW LINE
    character (_) to only specific predefined values allows new predefined values to be introduced
    at a future time without conflicting with author-defined values.</p>
   </dd>


   <dt>Errors that indicate a mis-use of other specifications</dt>

   <dd>
    <p>Certain restrictions are intended to support the restrictions made by other
    specifications.</p>

    <p class="example">For example, requiring that attributes that take media query lists use only
    <em>valid</em> media query lists reinforces the importance of following the conformance rules of
    that specification.</p>
   </dd>
  </dl>



  <h3>Suggested reading</h3>

  <!-- NON-NORMATIVE SECTION -->

  <p>The following documents might be of interest to readers of this specification.</p>

  <dl>
   <dt><cite>Character Model for the World Wide Web 1.0: Fundamentals</cite> <ref>CHARMOD</ref></dt>

   <dd><blockquote><p>This Architectural Specification provides authors of specifications, software
   developers, and content developers with a common reference for interoperable text manipulation on
   the World Wide Web, building on the Universal Character Set, defined jointly by the Unicode
   Standard and ISO/IEC 10646. Topics addressed include use of the terms 'character', 'encoding' and
   'string', a reference processing model, choice and identification of character encodings,
   character escaping, and string indexing.</p></blockquote></dd>

   <dt><cite>Unicode Security Considerations</cite> <ref>UTR36</ref></dt>

   <dd><blockquote><p>Because Unicode contains such a large number of characters and incorporates
   the varied writing systems of the world, incorrect usage can expose programs or systems to
   possible security attacks. This is especially important as more and more products are
   internationalized. This document describes some of the security considerations that programmers,
   system analysts, standards developers, and users should take into account, and provides specific
   recommendations to reduce the risk of problems.</p></blockquote></dd>

   <dt><cite>Web Content Accessibility Guidelines (WCAG)</cite> <ref>WCAG</ref></dt>

   <dd><blockquote><p>Web Content Accessibility Guidelines (WCAG) covers a wide range of
   recommendations for making web content more accessible. Following these guidelines will make
   content accessible to a wider range of people with disabilities, including blindness and low
   vision, deafness and hearing loss, learning disabilities, cognitive limitations, limited
   movement, speech disabilities, photosensitivity and combinations of these. Following these
   guidelines will also often make your web content more usable to users in
   general.</p></blockquote></dd>

   <dt w-nodev><cite>Authoring Tool Accessibility Guidelines (ATAG) 2.0</cite> <ref>ATAG</ref></dt>

   <dd w-nodev><blockquote><p>This specification provides guidelines for designing web content
   authoring tools that are more accessible for people with disabilities. An authoring tool that
   conforms to these guidelines will promote accessibility by providing an accessible user interface
   to authors with disabilities as well as by enabling, supporting, and promoting the production of
   accessible web content by all authors.</p></blockquote></dd>

   <dt w-nodev><cite>User Agent Accessibility Guidelines (UAAG) 2.0</cite> <ref>UAAG</ref></dt>

   <dd w-nodev><blockquote><p>This document provides guidelines for designing user agents that
   lower barriers to web accessibility for people with disabilities. User agents include browsers
   and other types of software that retrieve and render web content. A user agent that conforms to
   these guidelines will promote accessibility through its own user interface and through other
   internal facilities, including its ability to communicate with other technologies (especially
   assistive technologies). Furthermore, all users, not just users with disabilities, should find
   conforming user agents to be more usable.</p></blockquote></dd>
  </dl>



  <h2 split-filename="infrastructure" id="infrastructure">Common infrastructure</h2>

  <p>This specification depends on <cite>Infra</cite>. <ref>INFRA</ref></p>

  <h3>Terminology</h3>

  <p>This specification refers to both HTML and XML attributes and IDL attributes, often in the same
  context. When it is not clear which is being referred to, they are referred to as <dfn
  data-x="" data-lt="content attribute" export>content attributes</dfn> for HTML and XML attributes,
  and <dfn data-x="">IDL attributes</dfn> for those defined on IDL interfaces. Similarly, the term
  "properties" is used for both JavaScript object properties and CSS properties. When these are
  ambiguous they are qualified as <dfn data-x="">object properties</dfn> and <dfn data-x="">CSS properties</dfn> respectively.</p>

  <p>Generally, when the specification states that a feature applies to <span>the HTML syntax</span>
  or <span>the XML syntax</span>, it also includes the other. When a feature specifically only
  applies to one of the two languages, it is called out by explicitly stating that it does not apply
  to the other format, as in "for HTML, ... (this does not apply to XML)".</p>

  <p>This specification uses the term <dfn data-x="">document</dfn> to refer to any use of HTML,
  ranging from short static documents to long essays or reports with rich multimedia, as well as to
  fully-fledged interactive applications. The term is used to refer both to <code>Document</code>
  objects and their descendant DOM trees, and to serialized byte streams using the <span data-x="the
  HTML syntax">HTML syntax</span> or the <span data-x="the XML syntax">XML syntax</span>, depending
  on context.</p>

  <p>In the context of the DOM structures, the terms <span data-x="HTML documents">HTML
  document</span> and <span data-x="XML documents">XML document</span> are used as defined in
  <cite>DOM</cite>, and refer specifically to two different modes that <code>Document</code> objects
  can find themselves in. <ref>DOM</ref> (Such uses are always hyperlinked to their
  definition.)</p>

  <p>In the context of byte streams, the term HTML document refers to resources labeled as
  <code>text/html</code>, and the term XML document refers to resources labeled with an <span>XML
  MIME type</span>.</p>

  <hr>

  <p>For simplicity, terms such as <dfn data-x="">shown</dfn>, <dfn data-x="">displayed</dfn>, and
  <dfn data-x="">visible</dfn> might sometimes be used when referring to the way a document is
  rendered to the user. These terms are not meant to imply a visual medium; they must be considered
  to apply to other media in equivalent ways.</p>


  <div w-nodev>

  <h4>Parallelism</h4>

  <p>To run steps <dfn export>in parallel</dfn> means those steps are to be run, one after another,
  at the same time as other logic in the standard (e.g., at the same time as the <span>event
  loop</span>). This standard does not define the precise mechanism by which this is achieved, be it
  time-sharing cooperative multitasking, fibers, threads, processes, using different hyperthreads,
  cores, CPUs, machines, etc. By contrast, an operation that is to run <dfn>immediately</dfn> must
  interrupt the currently running task, run itself, and then resume the previously running task.</p>

  <p class="note">For guidance on writing specifications that leverage parallelism, see <a
  href="#event-loop-for-spec-authors">Dealing with the event loop from other specifications</a>.</p>

  <p>To avoid race conditions between different <span>in parallel</span> algorithms that operate on
  the same data, a <span>parallel queue</span> can be used.</p>

  <p>A <dfn export>parallel queue</dfn> represents a queue of algorithm steps that must be run in
  series.</p>

  <p>A <span>parallel queue</span> has an <dfn>algorithm queue</dfn> (a <span>queue</span>),
  initially empty.</p>

  <p>To <dfn data-x="enqueue the following steps" data-lt="enqueue steps|enqueue the following
  steps" export for="parallel queue">enqueue steps</dfn> to a <span>parallel queue</span>,
  <span>enqueue</span> the algorithm steps to the <span>parallel queue</span>'s <span>algorithm
  queue</span>.</p>

  <p>To <dfn data-x="starting a new parallel queue" data-lt="start a new parallel queue|starting a
  new parallel queue" export>start a new parallel queue</dfn>, run the following steps:</p>

  <ol>
   <li><p>Let <var>parallelQueue</var> be a new <span>parallel queue</span>.</p></li>

   <li>
    <p>Run the following steps <span>in parallel</span>:</p>

    <ol>
     <li>
      <p>While true:</p>

      <ol>
       <li><p>Let <var>steps</var> be the result of <span data-x="dequeue">dequeueing</span> from
       <var>parallelQueue</var>'s <span>algorithm queue</span>.</p></li>

       <li><p>If <var>steps</var> is not nothing, then run <var>steps</var>.</p></li>

       <li><p><span>Assert</span>: running <var>steps</var> did not throw an exception, as steps
       running <span>in parallel</span> are not allowed to throw.</p></li>
      </ol>

      <p class="note">Implementations are not expected to implement this as a continuously running
      loop. Algorithms in standards are to be easy to understand and are not necessarily great for
      battery life or performance.</p>
     </li>
    </ol>
   </li>

   <li><p>Return <var>parallelQueue</var>.</p></li>
  </ol>

  <p class="note">Steps running <span>in parallel</span> can themselves run other steps in <span>in
  parallel</span>. E.g., inside a <span>parallel queue</span> it can be useful to run a series of
  steps in parallel with the queue.</p>

  <div class="example">
   <p>Imagine a standard defined <var>nameList</var> (a <span>list</span>), along with a method to
   add a <var>name</var> to <var>nameList</var>, unless <var>nameList</var> already <span
   data-x="list contains">contains</span> <var>name</var>, in which case it rejects.</p>

   <p>The following solution suffers from race conditions:</p>

   <ol>
    <li><p>Let <var>p</var> be a new promise created in <span>this</span>'s <span
    data-x="concept-relevant-realm">relevant realm</span>.</p></li>

    <li><p>Let <var>global</var> be <span>this</span>'s <span>relevant global
    object</span>.</p></li>

    <li>
     <p>Run the following steps <span>in parallel</span>:</p>

     <ol>
      <li><p>If <var>nameList</var> <span data-x="list contains">contains</span> <var>name</var>,
      then <span>queue a global task</span> on the <span>DOM manipulation task source</span> given
      <var>global</var> to reject <var>p</var> with a <code>TypeError</code>, and abort these
      steps.</p></li>

      <li><p>Do some potentially lengthy work.</p></li>

      <li><p><span data-x="list append">Append</span> <var>name</var> to
      <var>nameList</var>.</p></li>

      <li><p><span>Queue a global task</span> on the <span>DOM manipulation task source</span> given
      <var>global</var> to resolve <var>p</var> with undefined.</p></li>
     </ol>
    </li>

    <li><p>Return <var>p</var>.</p></li>
   </ol>

   <p>Two invocations of the above could run simultaneously, meaning <var>name</var> isn't in
   <var>nameList</var> during step 2.1, but it <em>might be added</em> before step 2.3 runs,
   meaning <var>name</var> ends up in <var>nameList</var> twice.</p>

   <p>Parallel queues solve this. The standard would let <var>nameListQueue</var> be the result of
   <span>starting a new parallel queue</span>, then:</p>

   <ol>
    <li><p>Let <var>p</var> be a new promise created in <span>this</span>'s <span
    data-x="concept-relevant-realm">relevant realm</span>.</p></li>

    <li><p>Let <var>global</var> be <span>this</span>'s <span>relevant global
    object</span>.</p></li>

    <li>
     <p><mark><span>Enqueue the following steps</span> to <var>nameListQueue</var>:</mark></p>

     <ol>
      <li><p>If <var>nameList</var> <span data-x="list contains">contains</span> <var>name</var>,
      then <span>queue a global task</span> on the <span>DOM manipulation task source</span> given
      <var>global</var> to reject <var>p</var> with a <code>TypeError</code>, and abort these
      steps.</p></li>

      <li><p>Do some potentially lengthy work.</p></li>

      <li><p><span data-x="list append">Append</span> <var>name</var> to
      <var>nameList</var>.</p></li>

      <li><p><span>Queue a global task</span> on the <span>DOM manipulation task source</span> given
      <var>global</var> to resolve <var>p</var> with undefined.</p></li>
     </ol>
    </li>

    <li><p>Return <var>p</var>.</p></li>
   </ol>

   <p>The steps would now queue and the race is avoided.</p>
  </div>

  </div>


  <h4>Resources</h4>

  <p>The specification uses the term <dfn data-x="">supported</dfn> when referring to whether a user
  agent has an implementation capable of decoding the semantics of an external resource. A format or
  type is said to be <i>supported</i> if the implementation can process an external resource of that
  format or type without critical aspects of the resource being ignored. Whether a specific resource
  is <i>supported</i> can depend on what features of the resource's format are in use.</p>

  <p class="example">For example, a PNG image would be considered to be in a supported format if its
  pixel data could be decoded and rendered, even if, unbeknownst to the implementation, the image
  also contained animation data.</p>

  <p class="example">An MPEG-4 video file would not be considered to be in a supported format if the
  compression format used was not supported, even if the implementation could determine the
  dimensions of the movie from the file's metadata.</p>

  <p>What some specifications, in particular the HTTP specifications, refer to as a
  <i>representation</i> is referred to in this specification as a <dfn data-x="">resource</dfn>.
  <ref>HTTP</ref></p>

  <p>A resource's <dfn>critical subresources</dfn> are those that the resource needs to have
  available to be correctly processed. Which resources are considered critical or not is defined by
  the specification that defines the resource's format.</p>

  <p>For <span data-x="CSS style sheet">CSS style sheets</span>, we tentatively define here that
  their critical subresources are other style sheets imported via <code data-x="">@import</code>
  rules, including those indirectly imported by other imported style sheets.</p>

  <p class="XXX">This definition is not fully interoperable; furthermore, some user agents seem to
  count resources like background images or web fonts as critical subresources. Ideally, the CSS
  Working Group would define this; see <a
  href="https://github.com/w3c/csswg-drafts/issues/1088">w3c/csswg-drafts issue #1088</a> to track
  progress on that front.</p>


  <h4 id="xml">XML compatibility</h4>

  <p id="html-namespace">To ease migration from HTML to XML, user agents conforming to this
  specification will place elements in HTML in the <code data-x="HTML
  namespace">http://www.w3.org/1999/xhtml</code> namespace, at least for the purposes of the DOM and
  CSS. The term "<dfn export>HTML elements</dfn>" refers to any element in that namespace, even in
  XML documents.</p>

  <p>Except where otherwise stated, all elements defined or mentioned in this specification are in
  the <span>HTML namespace</span> ("<code data-x="">http://www.w3.org/1999/xhtml</code>"), and all
  attributes defined or mentioned in this specification have no namespace.</p>

  <p>The term <dfn>element type</dfn> is used to refer to the set of elements that have a given
  local name and namespace. For example, <code>button</code> elements are elements with the element
  type <code>button</code>, meaning they have the local name "<code data-x="">button</code>" and
  (implicitly as defined above) the <span>HTML namespace</span>.</p>

  <p>Attribute names are said to be <dfn>XML-compatible</dfn> if they match the <code
  data-x="xml-Name">Name</code> production defined in XML and they contain no U+003A COLON
  characters (:). <ref>XML</ref></p>


  <h4>DOM trees</h4>

  <p>When it is stated that some element or attribute is <dfn data-x="ignore">ignored</dfn>, or
  treated as some other value, or handled as if it was something else, this refers only to the
  processing of the node after it is in the DOM. <span w-nodev>A user agent must not mutate the
  DOM in such situations.</span></p>

  <p>A content attribute is said to <dfn data-x="">change</dfn> value only if its new value is
  different than its previous value; setting an attribute to a value it already has does not change
  it.</p>

  <p>The term <dfn data-x="">empty</dfn>, when used for an attribute value, <code>Text</code> node,
  or string, means that the <span>length</span> of the text is zero (i.e., not even containing <span
  data-x="control">controls</span> or U+0020 SPACE).</p>

  <p>An HTML element can have specific <dfn>HTML element insertion steps</dfn>, <dfn>HTML element
  post-connection steps</dfn>, <dfn>HTML element removing steps</dfn>, and <dfn>HTML element moving
  steps</dfn> all defined for the element's <span data-x="concept-element-local-name">local
  name</span>.</p>

  <p>The <span data-x="concept-node-insert-ext">insertion steps</span> for the HTML Standard, given
  <var>insertedNode</var>, are defined as the following:</p>

  <ol>
   <li><p>If <var>insertedNode</var> is an element whose <span
   data-x="concept-element-namespace">namespace</span> is the <span>HTML namespace</span>, and this
   standard defines <span data-x="html element insertion steps">HTML element insertion steps</span>
   for <var>insertedNode</var>'s <span data-x="concept-element-local-name">local name</span>, then
   run the corresponding <span>HTML element insertion steps</span> given
   <var>insertedNode</var>.</p></li>

   <li>
    <p>If <var>insertedNode</var> is a <span>form-associated element</span> or the ancestor of a
    <span>form-associated element</span>, then:</p>

    <ol>
     <li><p>If the <span>form-associated element</span>'s <span>parser inserted flag</span> is set,
     then return.</p></li>

     <li><p><span>Reset the form owner</span> of the <span>form-associated element</span>.</p></li>
    </ol>
   </li>

   <li><p>If <var>insertedNode</var> is an <code>Element</code> that is not on the
   <span>stack of open elements</span> of an <span>HTML parser</span>, then
   <span>process internal resource links</span> given <var>insertedNode</var>'s
   <span>node document</span>.</p></li>
  </ol>

  <p>The <span data-x="concept-node-post-connection-ext">post-connection steps</span> for the HTML
  Standard, given <var>insertedNode</var>, are defined as the following:</p>

  <ol>
   <li><p>If <var>insertedNode</var> is an element whose <span
   data-x="concept-element-namespace">namespace</span> is the <span>HTML namespace</span>, and this
   standard defines <span data-x="html element post-connection steps">HTML element post-connection
   steps</span> for <var>insertedNode</var>'s <span data-x="concept-element-local-name">local
   name</span>, then run the corresponding <span>HTML element post-connection steps</span> given
   <var>insertedNode</var>.</p></li>
  </ol>

  <p>The <span data-x="concept-node-remove-ext">removing steps</span> for the HTML Standard, given
  <var>removedNode</var> and <var>oldParent</var>, are defined as the following:</p>

  <ol>
   <li><p>Let <var>document</var> be <var>removedNode</var>'s <span>node document</span>.</p></li>

   <li id="node-remove-focus-fixup">
    <p>If <var>document</var>'s <span data-x="focused area of the document">focused area</span> is
    <var>removedNode</var>, then set <var>document</var>'s <span data-x="focused area of the
    document">focused area</span> to <var>document</var>'s <span>viewport</span>, and set
    <var>document</var>'s <span>relevant global object</span>'s <span
    data-x="window-navigation-api">navigation API</span>'s <span>focus changed during ongoing
    navigation</span> to false.</p>

    <p class="note">This does <em>not</em> perform the <span>unfocusing steps</span>,
    <span>focusing steps</span>, or <span>focus update steps</span>, and thus no <code
    data-x="event-blur">blur</code> or <code data-x="event-change">change</code> events are
    fired.</p>
   </li>

   <li><p>If <var>removedNode</var> is an element whose <span
   data-x="concept-element-namespace">namespace</span> is the <span>HTML namespace</span>, and this
   standard defines <span data-x="html element removing steps">HTML element removing steps</span>
   for <var>removedNode</var>'s <span data-x="concept-element-local-name">local name</span>, then
   run the corresponding <span>HTML element removing steps</span> given <var>removedNode</var> and
   <var>oldParent</var>.</p></li>

   <li>
    <p>If <var>removedNode</var> is a <span>form-associated element</span> or the ancestor of a
    <span>form-associated element</span>, then:</p>

    <ol>
     <li><p>If the <span>form-associated element</span> has a <span>form owner</span> and the
     <span>form-associated element</span> and its <span>form owner</span> are no longer in the same
     <span>tree</span>, then <span>reset the form owner</span> of the <span>form-associated
     element</span>.</p></li>
    </ol>
   </li>

   <li><p>If <var>removedNode</var>'s <code data-x="attr-popover">popover</code> attribute is not in
   the <span data-x="attr-popover-none-state">no popover state</span>, then run the <span>hide
   popover algorithm</span> given <var>removedNode</var>, false, false, and false.</p></li>
  </ol>

  <p>The <span data-x="concept-node-move-ext">moving steps</span> for the HTML Standard, given
  <var>movedNode</var>, are defined as the following:</p>

  <ol>
   <li><p>If <var>movedNode</var> is an element whose <span
   data-x="concept-element-namespace">namespace</span> is the <span>HTML namespace</span>, and this
   standard defines <span>HTML element moving steps</span> for <var>movedNode</var>'s <span
   data-x="concept-element-local-name">local name</span>, then run the corresponding <span>HTML
   element moving steps</span> given <var>movedNode</var>.</p></li>

   <li>
    <p>If <var>movedNode</var> is a <span>form-associated element</span> or the ancestor of a
    <span>form-associated element</span>, then:</p>

    <ol>
     <li><p>If the <span>form-associated element</span> has a <span>form owner</span> and the
     <span>form-associated element</span> and its <span>form owner</span> are no longer in the same
     <span>tree</span>, then <span>reset the form owner</span> of the <span>form-associated
     element</span>.</p></li>
    </ol>
   </li>
  </ol>

  <p>A <dfn id="insert-an-element-into-a-document" data-x="node is inserted into a document"
  data-lt="inserted into a document|node is inserted into a document" export>node is inserted into a
  document</dfn> when the <span data-x="concept-node-insert-ext">insertion steps</span> are invoked
  with it as the argument and it is now <span>in a document tree</span>.  Analogously, a <dfn
  id="remove-an-element-from-a-document" data-x="node is removed from a document" data-lt="removed
  from a document|node is removed from a document" export>node is removed from a document</dfn> when
  the <span data-x="concept-node-remove-ext">removing steps</span> are invoked with it as the
  argument and it is now no longer <span>in a document tree</span>.</p>

  <p>A node <dfn export>becomes connected</dfn> when the <span
  data-x="concept-node-insert-ext">insertion steps</span> are invoked with it as the argument and it
  is now <span>connected</span>. Analogously, a node <dfn data-lt="become disconnected"
  export>becomes disconnected</dfn> when the <span data-x="concept-node-remove-ext">removing
  steps</span> are invoked with it as the argument and it is now no longer
  <span>connected</span>.</p>

  <p>A node is <dfn export>browsing-context connected</dfn> when it is <span>connected</span> and
  its <span data-x="shadow-including root">shadow-including root</span>'s <span
  data-x="concept-document-bc">browsing context</span> is non-null. A node <dfn data-lt="become
  browsing-context connected" export>becomes browsing-context connected</dfn> when the <span
  data-x="concept-node-insert-ext">insertion steps</span> are invoked with it as the argument and it
  is now <span>browsing-context connected</span>. A node <dfn data-lt="become browsing-context
  disconnected" export>becomes browsing-context disconnected</dfn> either when the <span
  data-x="concept-node-remove-ext">removing steps</span> are invoked with it as the argument and it
  is now no longer <span>browsing-context connected</span>, or when its <span>shadow-including
  root</span>'s <span data-x="concept-document-bc">browsing context</span> becomes null.


  <h4>Scripting</h4>

  <p>The construction "a <code data-x="">Foo</code> object", where <code data-x="">Foo</code> is
  actually an interface, is sometimes used instead of the more accurate "an object implementing the
  interface <code data-x="">Foo</code>".</p>

  <p>An IDL attribute is said to be <dfn data-x="">getting</dfn> when its value is being retrieved
  (e.g. by author script), and is said to be <dfn data-x="">setting</dfn> when a new value is
  assigned to it.</p>

  <p>If a DOM object is said to be <dfn>live</dfn>, then the attributes and methods on that object
  <span w-nodev>must</span> operate on the actual underlying data, not a snapshot of the
  data.</p>


  <h4>Plugins</h4>

  <p>The term <dfn>plugin</dfn> refers to an <span>implementation-defined</span> set of content
  handlers used by the user agent that can take part in the user agent's rendering of a
  <code>Document</code> object, but that neither act as <span data-x="child navigable">child
  navigables</span> of the <code>Document</code> nor introduce any <code>Node</code> objects to the
  <code>Document</code>'s DOM.</p>

  <p>Typically such content handlers are provided by third parties, though a user agent can also
  designate built-in content handlers as plugins.</p>

  <div w-nodev>

  <p>A user agent must not consider the types <code>text/plain</code> and
  <code>application/octet-stream</code> as having a registered <span>plugin</span>.</p> <!-- because
  of the way <object> elements handles those types, if nothing else (it also doesn't make any sense
  to have a plugin registered for those types, of course) -->

  </div>

  <p class="example">One example of a plugin would be a PDF viewer that is instantiated in a
  <span>navigable</span> when the user navigates to a PDF file. This would count as a plugin
  regardless of whether the party that implemented the PDF viewer component was the same as that
  which implemented the user agent itself. However, a PDF viewer application that launches separate
  from the user agent (as opposed to using the same interface) is not a plugin by this
  definition.</p>

  <p class="note">This specification does not define a mechanism for interacting with plugins, as it
  is expected to be user-agent- and platform-specific. Some UAs might opt to support a plugin
  mechanism such as the Netscape Plugin API; others might use remote content converters or have
  built-in support for certain types. Indeed, this specification doesn't require user agents to
  support plugins at all. <ref>NPAPI</ref></p>

  <div w-nodev>

  <p class="warning">Browsers should take extreme care when interacting with external content
  intended for <span data-x="plugin">plugins</span>. When third-party software is run with the same
  privileges as the user agent itself, vulnerabilities in the third-party software become as
  dangerous as those in the user agent.</p>

  <p>
  <!--INSERT TRACKING-->
  Since different users having different sets of <span data-x="plugin">plugins</span> provides a
  tracking vector that increases the chances of users being uniquely identified, user agents are
  encouraged to support the exact same set of <span data-x="plugin">plugins</span> for each
  user.</p>

  </div>



  <h4 id="encoding-terminology">Character encodings</h4>

  <p>A <dfn data-x="encoding" data-x-href="https://encoding.spec.whatwg.org/#encoding">character
  encoding</dfn>, or just <i>encoding</i> where that is not ambiguous, is a defined way to convert
  between byte streams and Unicode strings, as defined in <cite>Encoding</cite>. An
  <span>encoding</span> has an <dfn data-x-href="https://encoding.spec.whatwg.org/#name">encoding
  name</dfn> and one or more <dfn data-x="encoding label"
  data-x-href="https://encoding.spec.whatwg.org/#label">encoding labels</dfn>, referred to as the
  encoding's <i>name</i> and <i>labels</i> in the Encoding standard. <ref>ENCODING</ref></p>



  <div w-nodev>

  <h4>Conformance classes</h4>

  <p>This specification describes the conformance criteria for <span w-nodev>user agents
  (relevant to implementers) and</span> documents<span w-nodev> (relevant to authors and
  authoring tool implementers)</span>.</p>

  <p><dfn>Conforming documents</dfn> are those that comply with all the conformance criteria for
  documents. For readability, some of these conformance requirements are phrased as conformance
  requirements on authors; such requirements are implicitly requirements on documents: by
  definition, all documents are assumed to have had an author. (In some cases, that author may
  itself be a user agent &mdash; such user agents are subject to additional rules, as explained
  below.)</p>

  <p class="example">For example, if a requirement states that "authors <!--non-normative-->must not
  use the <code data-x="">foobar</code> element", it would imply that documents are not allowed to
  contain elements named <code data-x="">foobar</code>.</p>

  <p class="note impl">There is no implied relationship between document conformance requirements
  and implementation conformance requirements. User agents are not free to handle non-conformant
  documents as they please; the processing model described in this specification applies to
  implementations regardless of the conformity of the input documents.</p>

  <p>User agents fall into several (overlapping) categories with different conformance
  requirements.</p>

  <dl>
   <dt id="interactive">Web browsers and other interactive user agents</dt>

   <dd>
    <p>Web browsers that support <span>the XML syntax</span> must process elements and attributes
    from the <span>HTML namespace</span> found in XML documents as described in this specification,
    so that users can interact with them, unless the semantics of those elements have been
    overridden by other specifications.</p>

    <p class="example">A conforming web browser would, upon finding a <code>script</code> element in
    an XML document, execute the script contained in that element. However, if the element is found
    within a transformation expressed in XSLT (assuming the user agent also supports XSLT), then the
    processor would instead treat the <code>script</code> element as an opaque element that forms
    part of the transform.</p>

    <p>Web browsers that support <span>the HTML syntax</span> must process documents labeled with an
    <span>HTML MIME type</span> as described in this specification, so that users can interact with
    them.</p>

    <p>User agents that support scripting must also be conforming implementations of the IDL
    fragments in this specification, as described in <cite>Web IDL</cite>. <ref>WEBIDL</ref></p>

    <p class="note">Unless explicitly stated, specifications that override the semantics of HTML
    elements do not override the requirements on DOM objects representing those elements. For
    example, the <code>script</code> element in the example above would still implement the
    <code>HTMLScriptElement</code> interface.</p>
   </dd>

   <dt id="non-interactive">Non-interactive presentation user agents</dt>

   <dd>
    <p>User agents that process HTML and XML documents purely to render non-interactive versions of
    them must comply to the same conformance criteria as web browsers, except that they are exempt
    from requirements regarding user interaction.</p>

    <p class="note">Typical examples of non-interactive presentation user agents are printers
    (static UAs) and overhead displays (dynamic UAs). It is expected that most static
    non-interactive presentation user agents will also opt to <a href="#non-scripted">lack scripting
    support</a>.</p>

    <p class="example">A non-interactive but dynamic presentation UA would still execute scripts,
    allowing forms to be dynamically submitted, and so forth. However, since the concept of "focus"
    is irrelevant when the user cannot interact with the document, the UA would not need to support
    any of the focus-related DOM APIs.</p>
   </dd>

   <dt id="renderingUA">Visual user agents that support the suggested default rendering</dt>

   <dd>
    <p>User agents, whether interactive or not, may be designated (possibly as a user option) as
    supporting the suggested default rendering defined by this specification.</p>

    <p>This is not required. In particular, even user agents that do implement the suggested default
    rendering are encouraged to offer settings that override this default to improve the experience
    for the user, e.g. changing the color contrast, using different focus styles, or otherwise
    making the experience more accessible and usable to the user.</p>

    <p>User agents that are designated as supporting the suggested default rendering must, while so
    designated, implement the rules <a href="#rendering">the Rendering section</a> defines as the
    behavior that user agents are <em>expected</em> to implement.</p>
   </dd>

   <dt id="non-scripted">User agents with no scripting support</dt>

   <dd>
    <p>Implementations that do not support scripting (or which have their scripting features
    disabled entirely) are exempt from supporting the events and DOM interfaces mentioned in this
    specification. For the parts of this specification that are defined in terms of an events model
    or in terms of the DOM, such user agents must still act as if events and the DOM were
    supported.</p>

    <p class="note">Scripting can form an integral part of an application. Web browsers that do not
    support scripting, or that have scripting disabled, might be unable to fully convey the author's
    intent.</p>
   </dd>


   <dt>Conformance checkers</dt>

   <dd id="conformance-checkers">

    <p>Conformance checkers must verify that a document conforms to the applicable conformance
    criteria described in this specification. Automated conformance checkers are exempt from
    detecting errors that require interpretation of the author's intent (for example, while a
    document is non-conforming if the content of a <code>blockquote</code> element is not a quote,
    conformance checkers running without the input of human judgement do not have to check that
    <code>blockquote</code> elements only contain quoted material).</p>

    <p>Conformance checkers must check that the input document conforms when parsed without a
    <span data-x="concept-document-bc">browsing context</span> (meaning that no scripts are run, and
    that the parser's <span>scripting flag</span> is disabled), and should also check that the input
    document conforms when parsed with a <span data-x="concept-document-bc">browsing context</span>
    in which scripts execute, and that the scripts never cause non-conforming states to occur other
    than transiently during script execution itself. (This is only a "SHOULD" and not a "MUST"
    requirement because it has been proven to be impossible. <ref>COMPUTABLE</ref>)</p>

    <p>The term "HTML validator" can be used to refer to a conformance checker that itself conforms
    to the applicable requirements of this specification.</p>

    <div class="note">

     <p>XML DTDs cannot express all the conformance requirements of this specification. Therefore, a
     validating XML processor and a DTD cannot constitute a conformance checker. Also, since neither
     of the two authoring formats defined in this specification are applications of SGML, a
     validating SGML system cannot constitute a conformance checker either.</p>

     <p>To put it another way, there are three types of conformance criteria:</p>

     <ol>
      <li>Criteria that can be expressed in a DTD.</li>

      <li>Criteria that cannot be expressed by a DTD, but can still be checked by a machine.</li>

      <li>Criteria that can only be checked by a human.</li>
     </ol>

     <p>A conformance checker must check for the first two. A simple DTD-based validator only checks
     for the first class of errors and is therefore not a conforming conformance checker according
     to this specification.</p>

    </div>
   </dd>


   <dt>Data mining tools</dt>

   <dd id="data-mining">

    <p>Applications and tools that process HTML and XML documents for reasons other than to either
    render the documents or check them for conformance should act in accordance with the semantics
    of the documents that they process.</p>

    <p class="example">A tool that generates <span data-x="outline">document outlines</span> but
    increases the nesting level for each paragraph and does not increase the nesting level for
    <span data-x="concept-heading">headings</span> would not be conforming.</p>
   </dd>


   <dt id="editors">Authoring tools and markup generators</dt>

   <dd>
    <p>Authoring tools and markup generators must generate <span>conforming documents</span>.
    Conformance criteria that apply to authors also apply to authoring tools, where appropriate.</p>

    <p>Authoring tools are exempt from the strict requirements of using elements only for their
    specified purpose, but only to the extent that authoring tools are not yet able to determine
    author intent. However, authoring tools must not automatically misuse elements or encourage
    their users to do so.</p>

    <p class="example">For example, it is not conforming to use an <code>address</code> element for
    arbitrary contact information; that element can only be used for marking up contact information
    for its nearest <code>article</code> or <code>body</code> element ancestor. However, since an
    authoring tool is likely unable to determine the difference, an authoring tool is exempt from
    that requirement. This does not mean, though, that authoring tools can use <code>address</code>
    elements for any block of italics text (for instance); it just means that the authoring tool
    doesn't have to verify that when the user uses a tool for inserting contact information for an
    <code>article</code> element, that the user really is doing that and not inserting something
    else instead.</p>

    <p class="note">In terms of conformance checking, an editor has to output documents that conform
    to the same extent that a conformance checker will verify.</p>

    <p>When an authoring tool is used to edit a non-conforming document, it may preserve the
    conformance errors in sections of the document that were not edited during the editing session
    (i.e. an editing tool is allowed to round-trip erroneous content). However, an authoring tool
    must not claim that the output is conformant if errors have been so preserved.</p>

    <p>Authoring tools are expected to come in two broad varieties: tools that work from structure
    or semantic data, and tools that work on a What-You-See-Is-What-You-Get media-specific editing
    basis (WYSIWYG).</p>

    <p>The former is the preferred mechanism for tools that author HTML, since the structure in the
    source information can be used to make informed choices regarding which HTML elements and
    attributes are most appropriate.</p>

    <p>However, WYSIWYG tools are legitimate. WYSIWYG tools should use elements they know are
    appropriate, and should not use elements that they do not know to be appropriate. This might in
    certain extreme cases mean limiting the use of flow elements to just a few elements, like
    <code>div</code>, <code>b</code>, <code>i</code>, and <code>span</code> and making liberal use
    of the <code data-x="attr-style">style</code> attribute.</p>

    <p>All authoring tools, whether WYSIWYG or not, should make a best effort attempt at enabling
    users to create well-structured, semantically rich, media-independent content.</p>
   </dd>
  </dl>

  <p>For compatibility with existing content and prior specifications, this specification describes
  two authoring formats: one based on <span data-x="the XML syntax">XML</span>, and one using a <a
  href="#writing">custom format</a> inspired by SGML (referred to as <span>the HTML syntax</span>).
  Implementations must support at least one of these two formats, although supporting both is
  encouraged.</p>

  <p>Some conformance requirements are phrased as requirements on elements, attributes, methods or
  objects. Such requirements fall into two categories: those describing content model restrictions,
  and those describing implementation behavior. Those in the former category are requirements on
  documents and authoring tools. Those in the second category are requirements on user agents.
  Similarly, some conformance requirements are phrased as requirements on authors; such requirements
  are to be interpreted as conformance requirements on the documents that authors produce. (In other
  words, this specification does not distinguish between conformance criteria on authors and
  conformance criteria on documents.)</p>

  </div>


  <h4>Dependencies</h4>

  <!--
    data-noexport="" is used here to avoid the definition scraper (Shepherd) from picking up any of the terms found
    within this section. This is *not* used in accordance to the Bikeshed data model, which requires data-noexport=""
    to exist only on <dfn> or <hN> elements. (That is why we use data-noexport="" instead of noexport=""; Wattsi will
    only translate noexport="" according to the Bikeshed data model.)

    We may be able to remove this once all definitions are sufficiently marked up and Shepherd stops trying to infer
    definitions by scraping HTML.
  -->
  <div data-noexport="">

  <p>This specification relies on several other underlying specifications.</p>

  <dl>
   <dt>Infra</dt>

   <dd>
    <p>The following terms are defined in <cite>Infra</cite>: <ref>INFRA</ref></p>

    <ul class="brief">
     <li>The general iteration terms <dfn data-x-href="https://infra.spec.whatwg.org/#iteration-while">while</dfn>,
             <dfn data-x-href="https://infra.spec.whatwg.org/#iteration-continue">continue</dfn>, and
             <dfn data-x-href="https://infra.spec.whatwg.org/#iteration-break">break</dfn>.</li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#assert">Assert</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#implementation-defined">implementation-defined</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#willful-violation">willful violation</dfn></li>
     <li id="fingerprint"><span id="fingerprinting-vector"></span>
     <!-- INSERT TRACKING -->
     <dfn data-x-href="https://infra.spec.whatwg.org/#tracking-vector">tracking vector</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#code-point">code point</dfn> and its synonym
         <dfn data-x-href="https://infra.spec.whatwg.org/#code-point">character</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#surrogate">surrogate</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#scalar-value">scalar value</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#tuple">tuple</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#noncharacter">noncharacter</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#string">string</dfn>,
         <dfn data-x-href="https://infra.spec.whatwg.org/#code-unit">code unit</dfn>,
         <dfn data-x-href="https://infra.spec.whatwg.org/#code-unit-prefix">code unit prefix</dfn>,
         <dfn data-x-href="https://infra.spec.whatwg.org/#code-unit-less-than">code unit less than</dfn>,
         <dfn data-x-href="https://infra.spec.whatwg.org/#string-starts-with">starts with</dfn>,
         <dfn data-x-href="https://infra.spec.whatwg.org/#string-ends-with">ends with</dfn>,
         <dfn data-x-href="https://infra.spec.whatwg.org/#string-length">length</dfn>, and
         <dfn data-x-href="https://infra.spec.whatwg.org/#string-code-point-length">code point length</dfn></li>
     <li id="case-sensitive">The string equality operations <dfn data-x-href="https://infra.spec.whatwg.org/#string-is">is</dfn> and
         <dfn data-x-href="https://infra.spec.whatwg.org/#string-is">identical to</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#scalar-value-string">scalar value string</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#javascript-string-convert">convert</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#ascii-string">ASCII string</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#ascii-tab-or-newline">ASCII tab or newline</dfn></li>
     <li><dfn id="space-characters" data-x-href="https://infra.spec.whatwg.org/#ascii-whitespace">ASCII whitespace</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#control">control</dfn></li>
     <li><dfn data-x="ASCII digits" data-x-href="https://infra.spec.whatwg.org/#ascii-digit">ASCII digit</dfn></li>
     <li><dfn id="uppercase-ascii-hex-digits" data-x-href="https://infra.spec.whatwg.org/#ascii-upper-hex-digit">ASCII upper hex digit</dfn></li>
     <li><dfn id="lowercase-ascii-hex-digits" data-x-href="https://infra.spec.whatwg.org/#ascii-lower-hex-digit">ASCII lower hex digit</dfn></li>
     <li><dfn data-x="ASCII hex digits" data-x-href="https://infra.spec.whatwg.org/#ascii-hex-digit">ASCII hex digit</dfn></li>
     <li><dfn id="uppercase-ascii-letters" data-x-href="https://infra.spec.whatwg.org/#ascii-upper-alpha">ASCII upper alpha</dfn></li>
     <li><dfn id="lowercase-ascii-letters" data-x-href="https://infra.spec.whatwg.org/#ascii-lower-alpha">ASCII lower alpha</dfn></li>
     <li><dfn id="ascii-letters" data-x-href="https://infra.spec.whatwg.org/#ascii-alpha">ASCII alpha</dfn></li>
     <li><dfn id="alphanumeric-ascii-characters" data-x-href="https://infra.spec.whatwg.org/#ascii-alphanumeric">ASCII alphanumeric</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#isomorphic-decode">isomorphic decode</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#isomorphic-encode">isomorphic encode</dfn></li>
     <li><dfn data-x="converted to ASCII lowercase" data-x-href="https://infra.spec.whatwg.org/#ascii-lowercase">ASCII lowercase</dfn></li>
     <li><dfn data-x="converted to ASCII uppercase" data-x-href="https://infra.spec.whatwg.org/#ascii-uppercase">ASCII uppercase</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#ascii-case-insensitive">ASCII case-insensitive</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#strip-newlines">strip newlines</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#normalize-newlines">normalize newlines</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#strip-leading-and-trailing-ascii-whitespace">strip leading and trailing ASCII whitespace</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#strip-and-collapse-ascii-whitespace">strip and collapse ASCII whitespace</dfn></li>
     <li><dfn id="split-a-string-on-spaces" data-x-href="https://infra.spec.whatwg.org/#split-on-ascii-whitespace">split a string on ASCII whitespace</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#split-on-commas">split a string on commas</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#collect-a-sequence-of-code-points">collect a sequence of code points</dfn> and its associated
         <dfn data-x-href="https://infra.spec.whatwg.org/#string-position-variable">position variable</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#skip-ascii-whitespace">skip ASCII whitespace</dfn></li>
     <li>The <dfn data-x-href="https://infra.spec.whatwg.org/#ordered-map">ordered map</dfn> data structure and the associated definitions for
             <dfn data-x="map key" data-x-href="https://infra.spec.whatwg.org/#map-key">key</dfn>,
             <dfn data-x="map value" data-x-href="https://infra.spec.whatwg.org/#map-value">value</dfn>,
             <dfn data-x="map empty" data-x-href="https://infra.spec.whatwg.org/#map-is-empty">empty</dfn>,
             <dfn data-x="map entry" data-x-href="https://infra.spec.whatwg.org/#map-entry">entry</dfn>,
             <dfn data-x="map exists" data-x-href="https://infra.spec.whatwg.org/#map-exists">exists</dfn>,
             <dfn data-x="map get" data-x-href="https://infra.spec.whatwg.org/#map-get">getting the value of an entry</dfn>,
             <dfn data-x="map set" data-x-href="https://infra.spec.whatwg.org/#map-set">setting the value of an entry</dfn>,
             <dfn data-x="map remove" data-x-href="https://infra.spec.whatwg.org/#map-remove">removing an entry</dfn>,
             <dfn data-x="map clear" data-x-href="https://infra.spec.whatwg.org/#map-clear">clear</dfn>,
             <dfn data-x="map get the keys" data-x-href="https://infra.spec.whatwg.org/#map-getting-the-keys">getting the keys</dfn>,
             <dfn data-x="map get the values" data-x-href="https://infra.spec.whatwg.org/#map-getting-the-values">getting the values</dfn>,
             <dfn data-x="map sort descending" data-x-href="https://infra.spec.whatwg.org/#map-sort-in-descending-order">sorting in descending order</dfn>,
             <dfn data-x="map size" data-x-href="https://infra.spec.whatwg.org/#map-size">size</dfn>, and
             <dfn data-x="map iterate" data-x-href="https://infra.spec.whatwg.org/#map-iterate">iterate</dfn></li>
     <li>The <dfn data-x-href="https://infra.spec.whatwg.org/#list">list</dfn> data structure and the associated definitions for
             <dfn data-x="list append" data-x-href="https://infra.spec.whatwg.org/#list-append">append</dfn>,
             <dfn data-x="list extend" data-x-href="https://infra.spec.whatwg.org/#list-extend">extend</dfn>,
             <dfn data-x="list prepend" data-x-href="https://infra.spec.whatwg.org/#list-prepend">prepend</dfn>,
             <dfn data-x="list replace" data-x-href="https://infra.spec.whatwg.org/#list-replace">replace</dfn>,
             <dfn data-x="list remove" data-x-href="https://infra.spec.whatwg.org/#list-remove">remove</dfn>,
             <dfn data-x="list empty" data-x-href="https://infra.spec.whatwg.org/#list-empty">empty</dfn>,
             <dfn data-x="list contains" data-x-href="https://infra.spec.whatwg.org/#list-contain">contains</dfn>,
             <dfn data-x="list size" data-x-href="https://infra.spec.whatwg.org/#list-size">size</dfn>,
             <dfn data-x-href="https://infra.spec.whatwg.org/#list-get-the-indices">indices</dfn>,
             <dfn data-x="list is empty" data-x-href="https://infra.spec.whatwg.org/#list-is-empty">is empty</dfn>,
             <dfn data-x="list item" data-x-href="https://infra.spec.whatwg.org/#list-item">item</dfn>,
             <dfn data-x="list iterate" data-x-href="https://infra.spec.whatwg.org/#list-iterate">iterate</dfn>, and
             <dfn data-x="list clone" data-x-href="https://infra.spec.whatwg.org/#list-clone">clone</dfn>
             <dfn data-x="list sort" data-x-href="https://infra.spec.whatwg.org/#list-sort-in-ascending-order">sort in ascending order</dfn>
             <dfn data-x="list sort descending" data-x-href="https://infra.spec.whatwg.org/#list-sort-in-descending-order">sort in descending order</dfn></li>
     <li>The <dfn data-x-href="https://infra.spec.whatwg.org/#stack">stack</dfn> data structure and the associated definitions for
             <dfn data-x="stack push" data-x-href="https://infra.spec.whatwg.org/#stack-push">push</dfn> and
             <dfn data-x="stack pop" data-x-href="https://infra.spec.whatwg.org/#stack-pop">pop</dfn></li>
     <li>The <dfn data-x-href="https://infra.spec.whatwg.org/#queue">queue</dfn> data structure and the associated definitions for
             <dfn data-x-href="https://infra.spec.whatwg.org/#queue-enqueue">enqueue</dfn> and
             <dfn data-x-href="https://infra.spec.whatwg.org/#queue-dequeue">dequeue</dfn></li>
     <li>The <dfn data-x="set" data-x-href="https://infra.spec.whatwg.org/#ordered-set">ordered set</dfn> data structure and the associated definition for
             <dfn data-x="set append" data-x-href="https://infra.spec.whatwg.org/#set-append">append</dfn> and
             <dfn data-x="set union" data-x-href="https://infra.spec.whatwg.org/#set-union">union</dfn></li>
     <li>The <dfn data-x-href="https://infra.spec.whatwg.org/#struct">struct</dfn> specification type and the associated definition for
             <dfn data-x="struct item" data-x-href="https://infra.spec.whatwg.org/#struct-item">item</dfn></li>
     <li>The <dfn data-x-href="https://infra.spec.whatwg.org/#byte-sequence">byte sequence</dfn> data structure</li>
     <li>The <dfn data-x-href="https://infra.spec.whatwg.org/#forgiving-base64-encode">forgiving-base64 encode</dfn> and
             <dfn data-x-href="https://infra.spec.whatwg.org/#forgiving-base64-decode">forgiving-base64 decode</dfn> algorithms</li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#the-exclusive-range">exclusive range</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#parse-a-json-string-to-an-infra-value">parse a JSON string to an Infra value</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#html-namespace">HTML namespace</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#mathml-namespace">MathML namespace</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#svg-namespace">SVG namespace</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#xlink-namespace">XLink namespace</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#xml-namespace">XML namespace</dfn></li>
     <li><dfn data-x-href="https://infra.spec.whatwg.org/#xmlns-namespace">XMLNS namespace</dfn></li>
    </ul>
   </dd>

   <dt>Unicode and Encoding</dt>

   <dd>
    <p>The Unicode character set is used to represent textual data, and <cite>Encoding</cite>
    defines requirements around <span data-x="encoding">character encodings</span>.
    <ref>UNICODE</ref></p>

    <p class="note">This specification <a href="#encoding-terminology">introduces terminology</a>
    based on the terms defined in those specifications, as described earlier.</p>

    <p>The following terms are used as defined in <cite>Encoding</cite>: <ref>ENCODING</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://encoding.spec.whatwg.org/#concept-encoding-get">Getting an
     encoding</dfn></li>

     <li><dfn data-x-href="https://encoding.spec.whatwg.org/#get-an-output-encoding">Get an output
     encoding</dfn></li>

     <li>The generic <dfn data-x-href="https://encoding.spec.whatwg.org/#decode">decode</dfn>
     algorithm which takes a byte stream and an encoding and returns a character stream</li>

     <li>The <dfn data-x-href="https://encoding.spec.whatwg.org/#utf-8-decode">UTF-8 decode</dfn>
     algorithm which takes a byte stream and returns a character stream, additionally stripping one
     leading UTF-8 Byte Order Mark (BOM), if any</li>

     <li>The <dfn data-x-href="https://encoding.spec.whatwg.org/#utf-8-decode-without-bom">UTF-8
     decode without BOM</dfn> algorithm which is identical to <span>UTF-8 decode</span> except that
     it does not strip one leading UTF-8 Byte Order Mark (BOM)</li>

     <li>The <dfn data-x-href="https://encoding.spec.whatwg.org/#encode">encode</dfn> algorithm
     which takes a character stream and an encoding and returns a byte stream</li>

     <li>The <dfn data-x-href="https://encoding.spec.whatwg.org/#utf-8-encode">UTF-8 encode</dfn>
     algorithm which takes a character stream and returns a byte stream</li>

     <li>The <dfn data-x-href="https://encoding.spec.whatwg.org/#bom-sniff">BOM sniff</dfn>
     algorithm which takes a byte stream and returns an encoding or null.</li>
    </ul>
   </dd>


   <dt>XML and related specifications</dt>

   <dd>
    <p>Implementations that support <span>the XML syntax</span> for HTML must support some version
    of XML, as well as its corresponding namespaces specification, because that syntax uses an XML
    serialization with namespaces. <ref>XML</ref> <ref>XMLNS</ref></p>

    <p>Data mining tools and other user agents that perform operations on content without running
    scripts, evaluating CSS or XPath expressions, or otherwise exposing the resulting DOM to
    arbitrary content, may "support namespaces" by just asserting that their DOM node analogues are
    in certain namespaces, without actually exposing the namespace strings.</p>

    <p class="note">In <span>the HTML syntax</span>, namespace prefixes and namespace declarations
    do not have the same effect as in XML. For instance, the colon has no special meaning in HTML
    element names.</p>

    <hr>

    <p>The attribute with the name <dfn
    data-x-href="https://www.w3.org/TR/xml/#sec-white-space"><code
    data-x="attr-xml-space">space</code></dfn> in the <span>XML namespace</span> is defined by
    <cite>Extensible Markup Language</cite> (<cite>XML</cite>). <ref>XML</ref></p>

    <p>The <dfn data-x-href="https://www.w3.org/TR/xml/#NT-Name"><code
    data-x="xml-Name">Name</code></dfn> production is defined in <cite>XML</cite>.
    <ref>XML</ref></p>

    <p>This specification also references the <dfn
    data-x-href="https://www.w3.org/TR/xml-stylesheet/#the-xml-stylesheet-processing-instruction"><code>&lt;?xml-stylesheet?></code></dfn>
    processing instruction, defined in <cite>Associating Style Sheets with XML documents</cite>.
    <ref>XMLSSPI</ref></p>

    <p>This specification also non-normatively mentions the <dfn><code>XSLTProcessor</code></dfn>
    interface and its <dfn><code
    data-x="dom-XSLTProcessor-transformToFragment">transformToFragment()</code></dfn> and <dfn><code
    data-x="dom-XSLTProcessor-transformToDocument">transformToDocument()</code></dfn> methods.
    <ref>XSLTP</ref></p>
   </dd>


   <dt>URLs</dt>

   <dd>
    <p>The following terms are defined in <cite>URL</cite>: <ref>URL</ref></p>

    <ul class="brief">
     <li><dfn data-x="concept-host" data-x-href="https://url.spec.whatwg.org/#concept-host">host</dfn></li>
     <li><dfn data-x-href="https://url.spec.whatwg.org/#host-public-suffix">public suffix</dfn></li>
     <li><dfn data-x="concept-domain" data-x-href="https://url.spec.whatwg.org/#concept-domain">domain</dfn></li>
     <li><dfn data-x-href="https://url.spec.whatwg.org/#ip-address">IP address</dfn></li>
     <li><dfn data-x-href="https://url.spec.whatwg.org/#concept-url">URL</dfn></li>
     <li><dfn data-x="concept-url-origin" data-x-href="https://url.spec.whatwg.org/#concept-url-origin">Origin</dfn> of URLs</li>
     <li><dfn data-x-href="https://url.spec.whatwg.org/#syntax-url-absolute">Absolute URL</dfn></li>
     <li><dfn data-x-href="https://url.spec.whatwg.org/#syntax-url-relative">Relative URL</dfn></li>
     <li><dfn data-x-href="https://url.spec.whatwg.org/#host-registrable-domain">registrable domain</dfn></li>
     <li>The <dfn data-x-href="https://url.spec.whatwg.org/#concept-url-parser">URL parser</dfn>
     <li>The <dfn data-x-href="https://url.spec.whatwg.org/#concept-basic-url-parser">basic URL parser</dfn> and its
      <dfn data-x="basic url parser url" data-x-href="https://url.spec.whatwg.org/#basic-url-parser-url"><var>url</var></dfn> and
      <dfn data-x="basic url parser state override" data-x-href="https://url.spec.whatwg.org/#basic-url-parser-state-override"><var>state override</var></dfn>
      arguments, as well as these parser states:
      <ul class="brief">
       <li><dfn data-x-href="https://url.spec.whatwg.org/#scheme-start-state">scheme start state</dfn></li>
       <li><dfn data-x-href="https://url.spec.whatwg.org/#host-state">host state</dfn></li>
       <li><dfn data-x-href="https://url.spec.whatwg.org/#hostname-state">hostname state</dfn></li>
       <li><dfn data-x-href="https://url.spec.whatwg.org/#port-state">port state</dfn></li>
       <li><dfn data-x-href="https://url.spec.whatwg.org/#path-start-state">path start state</dfn></li>
       <li><dfn data-x-href="https://url.spec.whatwg.org/#query-state">query state</dfn></li>
       <li><dfn data-x-href="https://url.spec.whatwg.org/#fragment-state">fragment state</dfn></li>
      </ul>
     <li>
      <dfn data-x-href="https://url.spec.whatwg.org/#concept-url">URL record</dfn>, as well as its
      individual components:
      <ul class="brief">
       <li><dfn data-x="concept-url-scheme" data-x-href="https://url.spec.whatwg.org/#concept-url-scheme">scheme</dfn></li>
       <li><dfn data-x="concept-url-username" data-x-href="https://url.spec.whatwg.org/#concept-url-username">username</dfn></li>
       <li><dfn data-x="concept-url-password" data-x-href="https://url.spec.whatwg.org/#concept-url-password">password</dfn></li>
       <li><dfn data-x="concept-url-host" data-x-href="https://url.spec.whatwg.org/#concept-url-host">host</dfn></li>
       <li><dfn data-x="concept-url-port" data-x-href="https://url.spec.whatwg.org/#concept-url-port">port</dfn></li>
       <li><dfn data-x="concept-url-path" data-x-href="https://url.spec.whatwg.org/#concept-url-path">path</dfn></li>
       <li><dfn data-x="concept-url-query" data-x-href="https://url.spec.whatwg.org/#concept-url-query">query</dfn></li>
       <li><dfn data-x="concept-url-fragment" data-x-href="https://url.spec.whatwg.org/#concept-url-fragment">fragment</dfn></li>
       <li><dfn data-x="concept-url-blob-entry" data-x-href="https://url.spec.whatwg.org/#concept-url-blob-entry">blob URL entry</dfn></li>
      </ul>
     </li>
     <li><dfn data-x-href="https://url.spec.whatwg.org/#valid-url-string">valid URL string</dfn></li>
     <li>The <dfn data-x-href="https://url.spec.whatwg.org/#cannot-have-a-username-password-port">cannot have a username/password/port</dfn> concept</li>
     <li>The <dfn data-x-href="https://url.spec.whatwg.org/#url-opaque-path">opaque path</dfn> concept</li>
     <li><dfn data-x="concept-url-serializer" data-x-href="https://url.spec.whatwg.org/#concept-url-serializer">URL serializer</dfn> and its
         <dfn data-x="url serializer exclude fragment" data-x-href="https://url.spec.whatwg.org/#url-serializer-exclude-fragment">exclude fragment</dfn> argument</li>
     <li><dfn data-x-href="https://url.spec.whatwg.org/#url-path-serializer">URL path serializer</dfn></li>
     <li>The <dfn data-x-href="https://url.spec.whatwg.org/#concept-host-parser">host parser</dfn></li>
     <li>The <dfn data-x="host serializer" data-x-href="https://url.spec.whatwg.org/#concept-host-serializer">host serializer</dfn></li>
     <li><dfn data-x-href="https://url.spec.whatwg.org/#concept-host-equals">Host equals</dfn></li>
     <li><dfn data-x="concept-url-equals" data-x-href="https://url.spec.whatwg.org/#concept-url-equals">URL equals</dfn> and its
         <dfn data-x="url equals exclude fragments" data-x-href="https://url.spec.whatwg.org/#url-equals-exclude-fragments">exclude fragments</dfn> argument</li>
     <li><dfn data-x="serialize an integer" data-x-href="https://url.spec.whatwg.org/#serialize-an-integer">serialize an integer</dfn></li>
     <li><dfn data-x-href="https://url.spec.whatwg.org/#default-encode-set">Default encode set</dfn></li>
     <li><dfn data-x-href="https://url.spec.whatwg.org/#component-percent-encode-set">component percent-encode set</dfn></li>
     <li><dfn data-x-href="https://url.spec.whatwg.org/#string-utf-8-percent-encode">UTF-8 percent-encode</dfn></li>
     <li><dfn data-x-href="https://url.spec.whatwg.org/#string-percent-decode">percent-decode</dfn></li>
     <li><dfn data-x-href="https://url.spec.whatwg.org/#set-the-username">set the username</dfn></li>
     <li><dfn data-x-href="https://url.spec.whatwg.org/#set-the-password">set the password</dfn></li>
     <li>The <dfn data-x-href="https://url.spec.whatwg.org/#concept-urlencoded"><code>application/x-www-form-urlencoded</code></dfn> format</li>
     <li>The <dfn data-x-href="https://url.spec.whatwg.org/#concept-urlencoded-serializer"><code>application/x-www-form-urlencoded</code> serializer</dfn></li>
     <li><dfn data-x-href="https://url.spec.whatwg.org/#is-special">is special</dfn></li>
    </ul>

    <p>A number of schemes and protocols are referenced by this specification also:</p>

    <ul class="brief">
     <li>The <dfn data-x="about protocol"
     data-x-href="https://www.rfc-editor.org/rfc/rfc6694#section-2"><code>about:</code></dfn> scheme
     <ref>ABOUT</ref></li>
     <li>The <dfn data-x="blob protocol"
     data-x-href="https://w3c.github.io/FileAPI/#DefinitionOfScheme"><code>blob:</code></dfn> scheme
     <ref>FILEAPI</ref></li>
     <li>The <dfn data-x="data protocol"
     data-x-href="https://www.rfc-editor.org/rfc/rfc2397#section-2"><code>data:</code></dfn> scheme
     <ref>RFC2397</ref></li>
     <li>The <dfn data-x="http protocol"
     data-x-href="https://httpwg.org/specs/rfc7230.html#http.uri"><code>http:</code></dfn> scheme
     <ref>HTTP</ref></li>
     <li>The <dfn data-x="https protocol"
     data-x-href="https://httpwg.org/specs/rfc7230.html#https.uri"><code>https:</code></dfn> scheme
     <ref>HTTP</ref></li>
     <li>The <dfn data-x="mailto protocol"
     data-x-href="https://www.rfc-editor.org/rfc/rfc6068#section-2"><code>mailto:</code></dfn>
     scheme <ref>MAILTO</ref></li>
     <li>The <dfn data-x="sms protocol"
     data-x-href="https://www.rfc-editor.org/rfc/rfc5724#section-2"><code>sms:</code></dfn> scheme
     <ref>SMS</ref></li>
     <li>The <dfn data-x="urn protocol"
     data-x-href="https://www.rfc-editor.org/rfc/rfc2141#section-2"><code>urn:</code></dfn> scheme
     <ref>URN</ref></li>
    </ul>

    <p><dfn data-x-href="https://www.w3.org/TR/media-frags/#media-fragment-syntax">Media fragment
    syntax</dfn> is defined in <cite>Media Fragments URI</cite>. <ref>MEDIAFRAG</ref></p>
   </dd>


   <dt>HTTP and related specifications</dt>

   <dd>
    <p>The following terms are defined in the HTTP specifications: <ref>HTTP</ref></p>

    <ul class="brief">
     <li>`<dfn data-x="http-accept" data-x-href="https://httpwg.org/specs/rfc7231.html#header.accept"><code>Accept</code></dfn>` header</li>
     <li>`<dfn data-x="http-accept-language" data-x-href="https://httpwg.org/specs/rfc7231.html#header.accept-language"><code>Accept-Language</code></dfn>` header</li>
     <li>`<dfn data-x="http-cache-control" data-x-href="https://httpwg.org/specs/rfc7234.html#header.cache-control"><code>Cache-Control</code></dfn>` header</li>
     <li>`<dfn data-x="http-content-disposition" data-x-href="https://httpwg.org/specs/rfc6266.html"><code>Content-Disposition</code></dfn>` header</li>
     <li>`<dfn data-x="http-content-language" data-x-href="https://httpwg.org/specs/rfc7231.html#header.content-language"><code>Content-Language</code></dfn>` header</li>
     <li>`<dfn data-x="http-content-range" data-x-href="https://httpwg.org/specs/rfc7233.html#header.content-range"><code>Content-Range</code></dfn>` header</li>
     <li>`<dfn data-x="http-last-modified" data-x-href="https://httpwg.org/specs/rfc7232.html#header.last-modified"><code>Last-Modified</code></dfn>` header</li>
     <li>`<dfn data-x="http-range" data-x-href="https://httpwg.org/specs/rfc7233.html#header.range"><code>Range</code></dfn>` header</li>
     <li>`<dfn data-x="http-referer" data-x-href="https://httpwg.org/specs/rfc7231.html#header.referer"><code>Referer</code></dfn>` header</li>
    </ul>

    <p>The following terms are defined in <cite>HTTP State Management Mechanism</cite>:
    <ref>COOKIES</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://httpwg.org/specs/rfc6265.html#sane-cookie-syntax">cookie-string</dfn></li>
     <li><dfn data-x-href="https://httpwg.org/specs/rfc6265.html#storage-model">receives a set-cookie-string</dfn></li>
     <li>`<dfn data-x="http-cookie" data-x-href="https://httpwg.org/specs/rfc6265.html#cookie"><code>Cookie</code></dfn>` header</li>
    </ul>

    <p>The following term is defined in <cite>Web Linking</cite>: <ref>WEBLINK</ref></p>

    <ul class="brief">
     <li>`<dfn data-x="http-link" data-x-href="https://httpwg.org/specs/rfc8288.html#header"><code>Link</code></dfn>` header</li>
     <li><dfn data-x="parsing a link field value" data-x-href="https://httpwg.org/specs/rfc8288.html#parse-fv">Parsing a `<code>Link</code>` field value</dfn></li>
    </ul>

    <p>The following terms are defined in <cite>Structured Field Values for HTTP</cite>:
    <ref>STRUCTURED-FIELDS</ref></p>

    <ul class="brief">
     <li><dfn data-x="http-structured-header" data-x-href="https://httpwg.org/specs/rfc8941.html">structured header</dfn></li>
     <li><dfn data-x="http-structured-header-boolean" data-x-href="https://httpwg.org/specs/rfc8941.html#boolean">boolean</dfn></li>
     <li><dfn data-x="http-structured-header-token" data-x-href="https://httpwg.org/specs/rfc8941.html#token">token</dfn></li>
     <li><dfn data-x="http-structured-header-parameters" data-x-href="https://httpwg.org/specs/rfc8941.html#param">parameters</dfn></li>
    </ul>

    <p>The following terms are defined in <cite>MIME Sniffing</cite>: <ref>MIMESNIFF</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://mimesniff.spec.whatwg.org/#mime-type">MIME type</dfn></li>
     <li><dfn data-x-href="https://mimesniff.spec.whatwg.org/#mime-type-essence">MIME type essence</dfn></li>
     <li><dfn data-x-href="https://mimesniff.spec.whatwg.org/#valid-mime-type">valid MIME type string</dfn></li>
     <li><dfn data-x-href="https://mimesniff.spec.whatwg.org/#valid-mime-type-with-no-parameters">valid MIME type string with no parameters</dfn></li>
     <li><dfn data-x-href="https://mimesniff.spec.whatwg.org/#html-mime-type">HTML MIME type</dfn></li>
     <li><dfn data-x-href="https://mimesniff.spec.whatwg.org/#javascript-mime-type">JavaScript MIME type</dfn> and
         <dfn data-x-href="https://mimesniff.spec.whatwg.org/#javascript-mime-type-essence-match">JavaScript MIME type essence match</dfn></li>
     <li><dfn data-x-href="https://mimesniff.spec.whatwg.org/#json-mime-type">JSON MIME type</dfn></li>
     <li><dfn data-x-href="https://mimesniff.spec.whatwg.org/#xml-mime-type">XML MIME type</dfn></li>
     <li><dfn data-x-href="https://mimesniff.spec.whatwg.org/#image-mime-type">image MIME type</dfn></li>
     <li><dfn data-x-href="https://mimesniff.spec.whatwg.org/#audio-or-video-mime-type">audio or video MIME type</dfn></li>
     <li><dfn data-x-href="https://mimesniff.spec.whatwg.org/#font-mime-type">font MIME type</dfn></li>
     <li><dfn data-x-href="https://mimesniff.spec.whatwg.org/#parse-a-mime-type">parse a MIME type</dfn></li>
     <li><dfn data-x-href="https://mimesniff.spec.whatwg.org/#supported-by-the-user-agent">is MIME type supported by the user agent?</dfn></li>
    </ul>
   </dd>


   <dt>Fetch</dt>

   <dd>
    <p>The following terms are defined in <cite>Fetch</cite>: <ref>FETCH</ref></p>

    <ul class="brief">
     <li><dfn data-x="header-abnf" data-x-href="https://fetch.spec.whatwg.org/#abnf">ABNF</dfn></li>
     <li><dfn><code>about:blank</code></dfn></li>
     <li>An <dfn data-x-href="https://fetch.spec.whatwg.org/#http-scheme">HTTP(S) scheme</dfn></li>
     <li>A URL which <dfn data-x="is-local" data-x-href="https://fetch.spec.whatwg.org/#is-local">is local</dfn></li>
     <li>A <dfn data-x-href="https://fetch.spec.whatwg.org/#local-scheme">local scheme</dfn></li>
     <li>A <dfn data-x-href="https://fetch.spec.whatwg.org/#fetch-scheme">fetch scheme</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#http-cors-protocol">CORS protocol</dfn></li>
     <li><dfn data-x="default-user-agent-value" data-x-href="https://fetch.spec.whatwg.org/#default-user-agent-value">default `<code>User-Agent</code>` value</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#concept-header-extract-mime-type">extract a MIME type</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#legacy-extract-an-encoding">legacy extract an encoding</dfn></li>
     <li><dfn data-x="concept-fetch" data-x-href="https://fetch.spec.whatwg.org/#concept-fetch">fetch</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#fetch-controller">fetch controller</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#fetch-controller-process-the-next-manual-redirect">process the next manual redirect</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#ok-status">ok status</dfn></li>
     <li><dfn data-x="navigation-request" data-x-href="https://fetch.spec.whatwg.org/#navigation-request">navigation request</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#concept-network-error">network error</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#concept-aborted-network-error">aborted network error</dfn></li>
     <li>`<dfn data-x="http-origin" data-x-href="https://fetch.spec.whatwg.org/#http-origin"><code>Origin</code></dfn>` header</li>
     <li>`<dfn data-x-href="https://fetch.spec.whatwg.org/#http-cross-origin-resource-policy"><code>Cross-Origin-Resource-Policy</code></dfn>` header</li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#concept-header-list-get-structured-header">getting a structured field value</dfn></li>
     <li><dfn data-x="concept-header-list" data-x-href="https://fetch.spec.whatwg.org/#concept-header-list">header list</dfn></li>
     <li><dfn data-x="concept-header-list-set" data-x-href="https://fetch.spec.whatwg.org/#concept-header-list-set">set</dfn></li>
     <li><dfn data-x="concept-header-list-get-decode-split" data-x-href="https://fetch.spec.whatwg.org/#concept-header-list-get-decode-split">get, decode, and split</dfn></li>
     <li><dfn data-x="fetch-controller-abort" data-x-href="https://fetch.spec.whatwg.org/#fetch-controller-abort">abort</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#cross-origin-resource-policy-check">cross-origin resource policy check</dfn></li>
     <li>the <dfn data-x-href="https://fetch.spec.whatwg.org/#requestcredentials"><code>RequestCredentials</code></dfn> enumeration</li>
     <li>the <dfn data-x-href="https://fetch.spec.whatwg.org/#requestdestination"><code>RequestDestination</code></dfn> enumeration</li>
     <li>the <dfn data-x-href="https://fetch.spec.whatwg.org/#dom-global-fetch"><code>fetch()</code></dfn> method</li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#finalize-and-report-timing">report timing</dfn></li>
     <li><dfn data-x="serialize-a-response-url-for-reporting" data-x-href="https://fetch.spec.whatwg.org/#serialize-a-response-url-for-reporting">serialize a response URL for reporting</dfn></li>
     <li><dfn data-x="body safely extract" data-x-href="https://fetch.spec.whatwg.org/#bodyinit-safely-extract">safely extracting a body</dfn></li>
     <li><dfn data-x="body-incrementally-read" data-x-href="https://fetch.spec.whatwg.org/#body-incrementally-read">incrementally reading a body</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#process-response-end-of-body">processResponseConsumeBody</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#fetch-processresponseendofbody">processResponseEndOfBody</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#process-response">processResponse</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#fetch-useparallelqueue">useParallelQueue</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#fetch-processearlyhintsresponse">processEarlyHintsResponse</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#concept-connection-pool">connection pool</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#concept-connection-obtain">obtain a connection</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#determine-the-network-partition-key">determine the network partition key</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#extract-full-timing-info">extract full timing info</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#byte-sequence-as-a-body">as a body</dfn></li>
     <li><dfn data-x="response-body-info" data-x-href="https://fetch.spec.whatwg.org/#response-body-info">response body info</dfn></li>
     <li><dfn data-x-href="https://fetch.spec.whatwg.org/#resolve-an-origin">resolve an origin</dfn></li>
     <li>
      <dfn data-x="concept-response"
      data-x-href="https://fetch.spec.whatwg.org/#concept-response">response</dfn> and its
      associated:
      <ul class="brief">
       <li><dfn data-x="concept-response-type" data-x-href="https://fetch.spec.whatwg.org/#concept-response-type">type</dfn></li>
       <li><dfn data-x="concept-response-url" data-x-href="https://fetch.spec.whatwg.org/#concept-response-url">URL</dfn></li>
       <li><dfn data-x="concept-response-url-list" data-x-href="https://fetch.spec.whatwg.org/#concept-response-url-list">URL list</dfn></li>
       <li><dfn data-x="concept-response-status" data-x-href="https://fetch.spec.whatwg.org/#concept-response-status">status</dfn></li>
       <li><dfn data-x="concept-response-header-list" data-x-href="https://fetch.spec.whatwg.org/#concept-response-header-list">header list</dfn></li>
       <li><dfn data-x="concept-response-body" data-x-href="https://fetch.spec.whatwg.org/#concept-response-body">body</dfn></li>
       <li><dfn data-x="concept-response-body-info" data-x-href="https://fetch.spec.whatwg.org/#concept-response-body-info">body info</dfn></li>
       <li><dfn data-x="concept-internal-response" data-x-href="https://fetch.spec.whatwg.org/#concept-internal-response">internal response</dfn></li>
       <li><dfn data-x="concept-response-location-url" data-x-href="https://fetch.spec.whatwg.org/#concept-response-location-url">location URL</dfn></li>
       <li><dfn data-x="concept-response-timing-info" data-x-href="https://fetch.spec.whatwg.org/#concept-response-timing-info">timing info</dfn></li>
       <li><dfn data-x="concept-response-service-worker-timing-info" data-x-href="https://fetch.spec.whatwg.org/#response-service-worker-timing-info">service worker timing info</dfn></li>
       <li><dfn data-x="concept-response-has-cross-origin-redirects" data-x-href="https://fetch.spec.whatwg.org/#response-has-cross-origin-redirects">has-cross-origin-redirects</dfn></li>
       <li><dfn data-x="concept-response-timing-allow-passed" data-x-href="https://fetch.spec.whatwg.org/#concept-response-timing-allow-passed">timing allow passed</dfn></li>
       <li>
        <dfn data-x-href="https://wicg.github.io/background-fetch/#extract-content-range-values">extract content-range values</dfn>
        <!-- TODO: move this to FETCH -->
       </li>
      </ul>
     </li>
     <li>
      <dfn data-x="concept-request"
      data-x-href="https://fetch.spec.whatwg.org/#concept-request">request</dfn> and its associated:
      <ul class="brief">
       <li><dfn data-x="concept-request-url" data-x-href="https://fetch.spec.whatwg.org/#concept-request-url">URL</dfn></li>
       <li><dfn data-x="concept-request-method" data-x-href="https://fetch.spec.whatwg.org/#concept-request-method">method</dfn></li>
       <li><dfn data-x="concept-request-header-list" data-x-href="https://fetch.spec.whatwg.org/#concept-request-header-list">header list</dfn></li>
       <li><dfn data-x="concept-request-body" data-x-href="https://fetch.spec.whatwg.org/#concept-request-body">body</dfn></li>
       <li><dfn data-x="concept-request-client" data-x-href="https://fetch.spec.whatwg.org/#concept-request-client">client</dfn></li>
       <li><dfn data-x="concept-request-url-list" data-x-href="https://fetch.spec.whatwg.org/#concept-request-url-list">URL list</dfn></li>
       <li><dfn data-x="concept-request-current-url" data-x-href="https://fetch.spec.whatwg.org/#concept-request-current-url">current URL</dfn></li>
       <li><dfn data-x="concept-request-reserved-client" data-x-href="https://fetch.spec.whatwg.org/#concept-request-reserved-client">reserved client</dfn></li>
       <li><dfn data-x="concept-request-replaces-client-id" data-x-href="https://fetch.spec.whatwg.org/#concept-request-replaces-client-id">replaces client id</dfn></li>
       <li><dfn data-x="concept-request-initiator" data-x-href="https://fetch.spec.whatwg.org/#concept-request-initiator">initiator</dfn></li>
       <li><dfn data-x="concept-request-destination" data-x-href="https://fetch.spec.whatwg.org/#concept-request-destination">destination</dfn></li>
       <li><dfn data-x="concept-potential-destination" data-x-href="https://fetch.spec.whatwg.org/#concept-potential-destination">potential destination</dfn></li>
       <li><dfn data-x="concept-potential-destination-translate" data-x-href="https://fetch.spec.whatwg.org/#concept-potential-destination-translate">translating</dfn> a <span data-x="concept-potential-destination">potential destination</span></li>
       <li><dfn data-x="concept-script-like-destination" data-x-href="https://fetch.spec.whatwg.org/#request-destination-script-like">script-like</dfn> <span data-x="concept-request-destination">destinations</span></li>
       <li><dfn data-x="concept-request-priority" data-x-href="https://fetch.spec.whatwg.org/#request-priority">priority</dfn></li>
       <li><dfn data-x="concept-request-origin" data-x-href="https://fetch.spec.whatwg.org/#concept-request-origin">origin</dfn></li>
       <li><dfn data-x="concept-request-referrer" data-x-href="https://fetch.spec.whatwg.org/#concept-request-referrer">referrer</dfn></li>
       <li><dfn data-x-href="https://fetch.spec.whatwg.org/#synchronous-flag">synchronous flag</dfn></li>
       <li><dfn data-x="concept-request-mode" data-x-href="https://fetch.spec.whatwg.org/#concept-request-mode">mode</dfn></li>
       <li><dfn data-x="concept-request-credentials-mode" data-x-href="https://fetch.spec.whatwg.org/#concept-request-credentials-mode">credentials mode</dfn></li>
       <li><dfn data-x-href="https://fetch.spec.whatwg.org/#concept-request-use-url-credentials-flag">use-URL-credentials flag</dfn></li>
       <li><dfn data-x-href="https://fetch.spec.whatwg.org/#unsafe-request-flag">unsafe-request flag</dfn></li>
       <li><dfn data-x="concept-request-cache-mode" data-x-href="https://fetch.spec.whatwg.org/#concept-request-cache-mode">cache mode</dfn></li>
       <li><dfn data-x="concept-request-redirect-count" data-x-href="https://fetch.spec.whatwg.org/#concept-request-redirect-count">redirect count</dfn></li>
       <li><dfn data-x="concept-request-redirect-mode" data-x-href="https://fetch.spec.whatwg.org/#concept-request-redirect-mode">redirect mode</dfn></li>
       <li><dfn data-x="concept-request-policy-container" data-x-href="https://fetch.spec.whatwg.org/#concept-request-policy-container">policy container</dfn></li>
       <li><dfn data-x="concept-request-referrer-policy" data-x-href="https://fetch.spec.whatwg.org/#concept-request-referrer-policy">referrer policy</dfn></li>
       <li><dfn data-x="concept-request-nonce-metadata" data-x-href="https://fetch.spec.whatwg.org/#concept-request-nonce-metadata">cryptographic nonce metadata</dfn></li>
       <li><dfn data-x="concept-request-integrity-metadata" data-x-href="https://fetch.spec.whatwg.org/#concept-request-integrity-metadata">integrity metadata</dfn></li>
       <li><dfn data-x="concept-request-parser-metadata" data-x-href="https://fetch.spec.whatwg.org/#concept-request-parser-metadata">parser metadata</dfn></li>
       <li><dfn data-x="concept-request-reload-navigation-flag" data-x-href="https://fetch.spec.whatwg.org/#concept-request-reload-navigation-flag">reload-navigation flag</dfn></li>
       <li><dfn data-x="concept-request-history-navigation-flag" data-x-href="https://fetch.spec.whatwg.org/#concept-request-history-navigation-flag">history-navigation flag</dfn></li>
       <li><dfn data-x="concept-request-user-activation" data-x-href="https://fetch.spec.whatwg.org/#request-user-activation">user-activation</dfn></li>
       <li><dfn data-x="concept-request-render-blocking" data-x-href="https://fetch.spec.whatwg.org/#request-render-blocking">render-blocking</dfn></li>
       <li><dfn data-x="concept-request-initiator-type" data-x-href="https://fetch.spec.whatwg.org/#request-initiator-type">initiator type</dfn></li>
       <li><dfn data-x-href="https://fetch.spec.whatwg.org/#concept-request-add-range-header">add a range header</dfn></li>
      </ul>
     </li>
     <li>
      <dfn data-x="fetch-timing-info"
      data-x-href="https://fetch.spec.whatwg.org/#fetch-timing-info">fetch timing info</dfn> and its
      associated:
      <ul class="brief">
       <li><dfn data-x="fetch-timing-info-start-time" data-x-href="https://fetch.spec.whatwg.org/#fetch-timing-info-start-time">start time</dfn></li>
       <li><dfn data-x="fetch-timing-info-end-time" data-x-href="https://fetch.spec.whatwg.org/#fetch-timing-info-end-time">end time</dfn></li>
      </ul>
     </li>
    </ul>

    <p>The following terms are defined in <cite>Referrer Policy</cite>:
    <ref>REFERRERPOLICY</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/webappsec-referrer-policy/#referrer-policy">referrer policy</dfn></li>
     <li>The `<dfn data-x="http-referrer-policy" data-x-href="https://w3c.github.io/webappsec-referrer-policy/#referrer-policy-header-dfn"><code>Referrer-Policy</code></dfn>` HTTP header</li>
     <li>The <dfn data-x="parse-referrer-policy-header" data-x-href="https://w3c.github.io/webappsec-referrer-policy/#parse-referrer-policy-from-header">parse a referrer policy from a `<code>Referrer-Policy</code>` header</dfn> algorithm</li>
     <li>The "<dfn data-x="referrer-policy-no-referrer" data-x-href="https://w3c.github.io/webappsec-referrer-policy/#referrer-policy-no-referrer"><code>no-referrer</code></dfn>",
             "<dfn data-x="referrer-policy-no-referrer-when-downgrade" data-x-href="https://w3c.github.io/webappsec-referrer-policy/#referrer-policy-no-referrer-when-downgrade"><code>no-referrer-when-downgrade</code></dfn>",
             "<dfn data-x="referrer-policy-origin-when-cross-origin" data-x-href="https://w3c.github.io/webappsec-referrer-policy/#referrer-policy-origin-when-cross-origin"><code>origin-when-cross-origin</code></dfn>", and
             "<dfn data-x="referrer-policy-unsafe-url" data-x-href="https://w3c.github.io/webappsec-referrer-policy/#referrer-policy-unsafe-url"><code>unsafe-url</code></dfn>" referrer policies</li>
     <li>The <dfn data-x-href="https://w3c.github.io/webappsec-referrer-policy/#default-referrer-policy">default referrer policy</dfn></li>
    </ul>

    <p>The following terms are defined in <cite>Mixed Content</cite>: <ref>MIX</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/webappsec-mixed-content/#a-priori-authenticated-url"><i>a priori</i> authenticated URL</dfn></li>
    </ul>

    <p>The following terms are defined in <cite>Subresource Integrity</cite>: <ref>SRI</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/webappsec-subresource-integrity/#parse-metadata">parse integrity metadata</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webappsec-subresource-integrity/#the-integrity-attribute">the requirements of the integrity attribute</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webappsec-subresource-integrity/#get-the-strongest-metadata">get the strongest metadata from set</dfn></li>
    </ul>
   </dd>

   <dt>Paint Timing</dt>

   <dd>
    <p>The following terms are defined in <cite>Paint Timing</cite>: <ref>PAINTTIMING</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/paint-timing/#mark-paint-timing">mark paint timing</dfn></li>
    </ul>
   </dd>

   <dt>Navigation Timing</dt>

   <dd>
    <p>The following terms are defined in <cite>Navigation Timing</cite>:
    <ref>NAVIGATIONTIMING</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/navigation-timing/#dfn-create-the-navigation-timing-entry">create the navigation timing entry</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/navigation-timing/#dfn-queue-the-navigation-timing-entry">queue the navigation timing entry</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/navigation-timing/#dom-navigationtimingtype"><code>NavigationTimingType</code></dfn> and its
         "<dfn data-x="dom-navigationtimingtype-navigate" data-x-href="https://w3c.github.io/navigation-timing/#dom-navigationtimingtype-navigate"><code>navigate</code></dfn>",
         "<dfn data-x="dom-navigationtimingtype-reload" data-x-href="https://w3c.github.io/navigation-timing/#dom-navigationtimingtype-reload"><code>reload</code></dfn>", and
         "<dfn data-x="dom-navigationtimingtype-back_forward" data-x-href="https://w3c.github.io/navigation-timing/#dom-navigationtimingtype-back_forward"><code>back_forward</code></dfn>" values.</li>
    </ul>
   </dd>

   <dt>Resource Timing</dt>

   <dd>
    <p>The following terms are defined in <cite>Resource Timing</cite>:
    <ref>RESOURCETIMING</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/resource-timing/#dfn-mark-resource-timing">Mark resource timing</dfn></li>
    </ul>

   <dt>Performance Timeline</dt>

   <dd>
    <p>The following terms are defined in <cite>Performance Timeline</cite>:
    <ref>PERFORMANCETIMELINE</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/performance-timeline/#dom-performanceentry"><code>PerformanceEntry</code></dfn> and its
         <dfn data-x="PerformanceEntry-name" data-x-href="https://w3c.github.io/performance-timeline/#dom-performanceentry-name"><code>name</code></dfn>,
         <dfn data-x="PerformanceEntry-entryType" data-x-href="https://w3c.github.io/performance-timeline/#dom-performanceentry-entrytype"><code>entryType</code></dfn>,
         <dfn data-x="PerformanceEntry-startTime" data-x-href="https://w3c.github.io/performance-timeline/#dom-performanceentry-starttime"><code>startTime</code></dfn>, and
         <dfn data-x="PerformanceEntry-duration" data-x-href="https://w3c.github.io/performance-timeline/#dom-performanceentry-duration"><code>duration</code></dfn> attributes.</li>

     <li><dfn data-x-href="https://w3c.github.io/performance-timeline/#queue-a-performanceentry">Queue a performance entry</dfn></li>
     </ul>
   </dd>

   <dt>Long Animation Frames</dt>

   <dd>
    <p>The following terms are defined in <cite>Long Animation Frames</cite>: <ref>LONGANIMATIONFRAMES</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/long-animation-frames/#record-task-start-time">record task start time</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/long-animation-frames/#record-task-end-time">record task end time</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/long-animation-frames/#record-rendering-time">record rendering time</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/long-animation-frames/#record-classic-script-creation-time">record classic script creation time</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/long-animation-frames/#record-classic-script-execution-start-time">record classic script execution start time</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/long-animation-frames/#record-module-script-execution-start-time">record module script execution start time</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/long-animation-frames/#record-pause-duration">Record pause duration</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/long-animation-frames/#record-timing-info-for-timer-handler">record timing info for timer handler</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/long-animation-frames/#record-timing-info-for-microtask-checkpoint">record timing info for microtask checkpoint</dfn></li>
    </ul>
   </dd>

   <dt>Long Tasks</dt>

   <dd>
    <p>The following terms are defined in <cite>Long Tasks</cite>: <ref>LONGTASKS</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/longtasks/#report-long-tasks">report long tasks</dfn></li>
    </ul>
   </dd>

   <dt>Web IDL</dt>

   <dd>
    <p>The IDL fragments in this specification must be interpreted as required for conforming IDL
    fragments, as described in <cite>Web IDL</cite>. <ref>WEBIDL</ref></p>

    <p>The following terms are defined in <cite>Web IDL</cite>:</p>

    <ul class="brief">
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#this">this</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-extended-attribute">extended attribute</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-named-constructor">named constructor</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#idl-constructors">constructor operation</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#overridden-constructor-steps">overridden constructor steps</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#internally-create-a-new-object-implementing-the-interface">internally create a new object implementing the interface</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-array-index-property-name">array index property name</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-support-indexed-properties">supports indexed properties</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-supported-property-indices">supported property indices</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-determine-the-value-of-an-indexed-property">determine the value of an indexed property</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-set-the-value-of-an-existing-indexed-property">set the value of an existing indexed property</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-set-the-value-of-a-new-indexed-property">set the value of a new indexed property</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-support-named-properties">support named properties</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-supported-property-names">supported property names</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-determine-the-value-of-a-named-property">determine the value of a named property</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-set-the-value-of-an-existing-named-property">set the value of an existing named property</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-set-the-value-of-a-new-named-property">set the value of a new named property</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-delete-an-existing-named-property">delete an existing named property</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-perform-a-security-check">perform a security check</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-platform-object">platform object</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-legacy-platform-object">legacy platform object</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-primary-interface">primary interface</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-interface-object">interface object</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-named-properties-object">named properties object</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#include">include</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-inherit">inherit</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-interface-prototype-object">interface prototype object</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#implements">implements</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-associated-realm">associated realm</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#es-platform-objects">[[Realm]] field of a platform object</dfn></li>
     <li><dfn data-x="named-properties-object-getownproperty" data-x-href="https://webidl.spec.whatwg.org/#named-properties-object-getownproperty">[[GetOwnProperty]] internal method of a named properties object</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-callback-context">callback context</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-frozen-array-type">frozen array</dfn> and
         <dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-create-frozen-array">creating a frozen array</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#new" data-x="new">create a new object implementing the interface</dfn></li>
     <li><dfn data-x="dfn-callback-this-value" data-x-href="https://webidl.spec.whatwg.org/#dfn-callback-this-value">callback this value</dfn></li>
     <li><dfn data-x="concept-idl-convert" data-x-href="https://webidl.spec.whatwg.org/#es-type-mapping">converting</dfn> between Web IDL types and JS types</li>
     <li><dfn data-x="es-invoking-callback-functions" data-x-href="https://webidl.spec.whatwg.org/#invoke-a-callback-function">invoking</dfn> and
         <dfn data-x="es-constructing-callback-functions" data-x-href="https://webidl.spec.whatwg.org/#construct-a-callback-function">constructing</dfn> callback functions</li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-overload-resolution-algorithm">overload resolution algorithm</dfn></li>
     <li><dfn data-x="idl-exposed" data-x-href="https://webidl.spec.whatwg.org/#dfn-exposed">exposed</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#a-promise-resolved-with">a promise resolved with</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#a-promise-rejected-with">a promise rejected with</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#wait-for-all">wait for all</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#upon-rejection">upon rejection</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#upon-fulfillment">upon fulfillment</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#mark-a-promise-as-handled">mark as handled</dfn></li>
     <li><dfn data-x="Global" data-x-href="https://webidl.spec.whatwg.org/#Global"><code>[Global]</code></dfn></li>
     <li><dfn data-x="LegacyFactoryFunction" data-x-href="https://webidl.spec.whatwg.org/#LegacyFactoryFunction"><code>[LegacyFactoryFunction]</code></dfn></li>
     <li><dfn data-x="LegacyLenientThis" data-x-href="https://webidl.spec.whatwg.org/#LegacyLenientThis"><code>[LegacyLenientThis]</code></dfn></li>
     <li><dfn data-x="LegacyNullToEmptyString" data-x-href="https://webidl.spec.whatwg.org/#LegacyNullToEmptyString"><code>[LegacyNullToEmptyString]</code></dfn></li>
     <li><dfn data-x="LegacyOverrideBuiltIns" data-x-href="https://webidl.spec.whatwg.org/#LegacyOverrideBuiltIns"><code>[LegacyOverrideBuiltIns]</code></dfn></li>
     <li><dfn data-x="LegacyPlatformObjectGetOwnProperty" data-x-href="https://webidl.spec.whatwg.org/#LegacyPlatformObjectGetOwnProperty">LegacyPlatformObjectGetOwnProperty</dfn></li>
     <li><dfn data-x="LegacyTreatNonObjectAsNull" data-x-href="https://webidl.spec.whatwg.org/#LegacyTreatNonObjectAsNull"><code>[LegacyTreatNonObjectAsNull]</code></dfn></li>
     <li><dfn data-x="LegacyUnenumerableNamedProperties" data-x-href="https://webidl.spec.whatwg.org/#LegacyUnenumerableNamedProperties"><code>[LegacyUnenumerableNamedProperties]</code></dfn></li>
     <li><dfn data-x="LegacyUnforgeable" data-x-href="https://webidl.spec.whatwg.org/#LegacyUnforgeable"><code>[LegacyUnforgeable]</code></dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-set-entries">set entries</dfn></li>
    </ul>

    <p><cite>Web IDL</cite> also defines the following types that are used in Web IDL fragments in
    this specification:</p>

    <ul class="brief">
     <li><dfn data-x="idl-ArrayBuffer" data-x-href="https://webidl.spec.whatwg.org/#idl-ArrayBuffer"><code>ArrayBuffer</code></dfn></li>
     <li><dfn data-x="idl-ArrayBufferView" data-x-href="https://webidl.spec.whatwg.org/#common-ArrayBufferView"><code>ArrayBufferView</code></dfn></li>
     <li><dfn data-x="idl-boolean" data-x-href="https://webidl.spec.whatwg.org/#idl-boolean"><code>boolean</code></dfn></li>
     <li><dfn data-x="idl-DOMString" data-x-href="https://webidl.spec.whatwg.org/#idl-DOMString"><code>DOMString</code></dfn></li>
     <li><dfn data-x="idl-double" data-x-href="https://webidl.spec.whatwg.org/#idl-double"><code>double</code></dfn></li>
     <li><dfn data-x="IDL enumeration" data-x-href="https://webidl.spec.whatwg.org/#idl-enums">enumeration</dfn></li>
     <li><dfn data-x="idl-Function" data-x-href="https://webidl.spec.whatwg.org/#common-Function"><code>Function</code></dfn></li>
     <li><dfn data-x="idl-long" data-x-href="https://webidl.spec.whatwg.org/#idl-long"><code>long</code></dfn></li>
     <li><dfn data-x="idl-object" data-x-href="https://webidl.spec.whatwg.org/#idl-object"><code>object</code></dfn></li>
     <li><dfn data-x="idl-Promise" data-x-href="https://webidl.spec.whatwg.org/#idl-promise"><code>Promise</code></dfn></li>
     <li><dfn data-x="idl-Uint8ClampedArray" data-x-href="https://webidl.spec.whatwg.org/#idl-Uint8ClampedArray"><code>Uint8ClampedArray</code></dfn></li>
     <li><dfn data-x="idl-unrestricted-double" data-x-href="https://webidl.spec.whatwg.org/#idl-unrestricted-double"><code>unrestricted double</code></dfn></li>
     <li><dfn data-x="idl-unsigned-long" data-x-href="https://webidl.spec.whatwg.org/#idl-unsigned-long"><code>unsigned long</code></dfn></li>
     <li><dfn data-x="idl-USVString" data-x-href="https://webidl.spec.whatwg.org/#idl-USVString"><code>USVString</code></dfn></li>
     <li><dfn data-x="idl-VoidFunction" data-x-href="https://webidl.spec.whatwg.org/#VoidFunction"><code>VoidFunction</code></dfn></li>
    </ul>

    <p>The term <dfn data-x-href="https://webidl.spec.whatwg.org/#dfn-throw">throw</dfn> in this
    specification is used as defined in <cite>Web IDL</cite>. The <dfn
    data-x-href="https://webidl.spec.whatwg.org/#dfn-DOMException"><code>DOMException</code></dfn>
    type and the following exception names are defined by Web IDL and used by this
    specification:</p>

    <ul class="brief">
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#indexsizeerror">"<code>IndexSizeError</code>"</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#hierarchyrequesterror">"<code>HierarchyRequestError</code>"</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#invalidcharactererror">"<code>InvalidCharacterError</code>"</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#nomodificationallowederror">"<code>NoModificationAllowedError</code>"</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#notfounderror">"<code>NotFoundError</code>"</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#notsupportederror">"<code>NotSupportedError</code>"</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#invalidstateerror">"<code>InvalidStateError</code>"</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#syntaxerror">"<code>SyntaxError</code>"</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#invalidaccesserror">"<code>InvalidAccessError</code>"</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#securityerror">"<code>SecurityError</code>"</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#networkerror">"<code>NetworkError</code>"</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#aborterror">"<code>AbortError</code>"</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#quotaexceedederror">"<code>QuotaExceededError</code>"</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#datacloneerror">"<code>DataCloneError</code>"</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#encodingerror">"<code>EncodingError</code>"</dfn></li>
     <li><dfn data-x-href="https://webidl.spec.whatwg.org/#notallowederror">"<code>NotAllowedError</code>"</dfn></li>
    </ul>

    <p>When this specification requires a user agent to <dfn>create a <code>Date</code> object</dfn>
    representing a particular time (which could be the special value Not-a-Number), the milliseconds
    component of that time, if any, must be truncated to an integer, and the time value of the newly
    created <code>Date</code> object must represent the resulting truncated time.</p>

    <p class="example">For instance, given the time 23045 millionths of a second after 01:00 UTC on
    January 1st 2000, i.e. the time 2000-01-01T00:00:00.023045Z, then the <code>Date</code> object
    created representing that time would represent the same time as that created representing the
    time 2000-01-01T00:00:00.023Z, 45 millionths earlier. If the given time is NaN, then the result
    is a <code>Date</code> object that represents a time value NaN (indicating that the object does
    not represent a specific instant of time).</p>
   </dd>


   <dt>JavaScript</dt>

   <dd>
    <p>Some parts of the language described by this specification only support JavaScript as the
    underlying scripting language. <ref>JAVASCRIPT</ref></p>

    <p class="note">The term "JavaScript" is used to refer to ECMA-262, rather than the official
    term ECMAScript, since the term JavaScript is more widely known.</p>

    <p>The following terms are defined in the JavaScript specification and used in this
    specification:</p>

    <ul class="brief">
     <li><dfn data-x-href="https://tc39.es/ecma262/#active-function-object">active function object</dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#sec-agents">agent</dfn> and
         <dfn data-x-href="https://tc39.es/ecma262/#sec-agent-clusters">agent cluster</dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#sec-automatic-semicolon-insertion">automatic semicolon insertion</dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#sec-candidate-executions">candidate execution</dfn></li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#current-realm">current realm</dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#clamping">clamping</dfn> a mathematical value</li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#early-error-rule">early error</dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#sec-forward-progress">forward progress</dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#sec-invariants-of-the-essential-internal-methods">invariants of the essential internal methods</dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#sec-execution-contexts">JavaScript execution context</dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#execution-context-stack">JavaScript execution context stack</dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#sec-code-realms">realm</dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#sec-jobcallback-records">JobCallback Record</dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#sec-built-in-function-objects">NewTarget</dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#running-execution-context">running JavaScript execution context</dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#surrounding-agent">surrounding agent</dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#sec-abstract-closure">abstract closure</dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#immutable-prototype-exotic-object">immutable prototype exotic object</dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#sec-well-known-symbols">Well-Known Symbols</dfn>, including
         <dfn>%Symbol.hasInstance%</dfn>,
         <dfn>%Symbol.isConcatSpreadable%</dfn>,
         <dfn>%Symbol.toPrimitive%</dfn>, and
         <dfn>%Symbol.toStringTag%</dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#sec-well-known-intrinsic-objects">Well-Known Intrinsic Objects</dfn>, including
         <dfn data-x-href="https://tc39.es/ecma262/#sec-properties-of-the-array-prototype-object">%Array.prototype%</dfn>,
         <dfn data-x-href="https://tc39.es/ecma262/#sec-properties-of-the-error-prototype-object">%Error.prototype%</dfn>,
         <dfn>%EvalError.prototype%</dfn>,
         <dfn data-x-href="https://tc39.es/ecma262/#sec-properties-of-the-function-prototype-object">%Function.prototype%</dfn>,
         <dfn data-x-href="https://tc39.es/ecma262/#sec-properties-of-the-object-prototype-object">%Object.prototype%</dfn>,
         <dfn data-x-href="https://tc39.es/ecma262/#sec-object.prototype.valueof">%Object.prototype.valueOf%</dfn>,
         <dfn>%RangeError.prototype%</dfn>,
         <dfn>%ReferenceError.prototype%</dfn>,
         <dfn>%SyntaxError.prototype%</dfn>,
         <dfn>%TypeError.prototype%</dfn>, and
         <dfn>%URIError.prototype%</dfn></li>

     <li>The <dfn data-x="js-prod-FunctionBody" data-x-href="https://tc39.es/ecma262/#prod-FunctionBody"><i>FunctionBody</i></dfn> production</li>
     <li>The <dfn data-x="js-prod-Module" data-x-href="https://tc39.es/ecma262/#prod-Module"><i>Module</i></dfn> production</li>
     <li>The <dfn data-x="js-prod-Pattern" data-x-href="https://tc39.es/ecma262/#prod-Pattern"><i>Pattern</i></dfn> production</li>
     <li>The <dfn data-x="js-prod-Script" data-x-href="https://tc39.es/ecma262/#prod-Script"><i>Script</i></dfn> production</li>

     <li>The
     <dfn data-x="js-BigInt" data-x-href="https://tc39.es/ecma262/#sec-ecmascript-language-types-bigint-type">BigInt</dfn>,
     <dfn data-x="js-Boolean" data-x-href="https://tc39.es/ecma262/#sec-ecmascript-language-types-boolean-type">Boolean</dfn>,
     <dfn data-x="js-Number" data-x-href="https://tc39.es/ecma262/#sec-ecmascript-language-types-number-type">Number</dfn>,
     <dfn data-x="js-String" data-x-href="https://tc39.es/ecma262/#sec-ecmascript-language-types-string-type">String</dfn>,
     <dfn data-x="js-Symbol" data-x-href="https://tc39.es/ecma262/#sec-ecmascript-language-types-symbol-type">Symbol</dfn>,
     and <dfn data-x="js-Object" data-x-href="https://tc39.es/ecma262/#sec-object-type">Object</dfn> ECMAScript language types</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-completion-record-specification-type">Completion Record</dfn> specification type</li>
     <li>The <dfn data-x="js-List" data-x-href="https://tc39.es/ecma262/#sec-list-and-record-specification-type">List</dfn> and
             <dfn data-x-href="https://tc39.es/ecma262/#sec-list-and-record-specification-type">Record</dfn> specification types</li>
     <li>The <dfn data-x="PropertyDescriptor" data-x-href="https://tc39.es/ecma262/#sec-property-descriptor-specification-type">Property Descriptor</dfn> specification type</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#modulerequest-record">ModuleRequest Record</dfn> specification type</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-script-records">Script Record</dfn> specification type</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-synthetic-module-records">Synthetic Module Record</dfn> specification type</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-cyclic-module-records">Cyclic Module Record</dfn> specification type</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-source-text-module-records">Source Text Module Record</dfn> specification type and its
             <dfn data-x="js-Evaluate" data-x-href="https://tc39.es/ecma262/#sec-moduleevaluation">Evaluate</dfn>,
             <dfn data-x="js-Link" data-x-href="https://tc39.es/ecma262/#sec-moduledeclarationlinking">Link</dfn> and
             <dfn data-x="js-LoadRequestedModules" data-x-href="https://tc39.es/ecma262/#sec-LoadRequestedModules">LoadRequestedModules</dfn> methods</li>

     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-arraycreate">ArrayCreate</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-call">Call</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-clear-kept-objects">ClearKeptObjects</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-cleanup-finalization-registry">CleanupFinalizationRegistry</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-construct">Construct</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-copydatablockbytes">CopyDataBlockBytes</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-createbuiltinfunction">CreateBuiltinFunction</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-createbytedatablock">CreateByteDataBlock</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-createdataproperty">CreateDataProperty</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-create-default-export-synthetic-module">CreateDefaultExportSyntheticModule</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-definepropertyorthrow">DefinePropertyOrThrow</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-detacharraybuffer">DetachArrayBuffer</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-enumerableownproperties">EnumerableOwnProperties</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-FinishLoadingImportedModule">FinishLoadingImportedModule</dfn> abstract operation</li>
     <li>The <dfn data-x="js-OrdinaryFunctionCreate" data-x-href="https://tc39.es/ecma262/#sec-ordinaryfunctioncreate">OrdinaryFunctionCreate</dfn> abstract operation</li>
     <li>The <dfn data-x="js-Get" data-x-href="https://tc39.es/ecma262/#sec-get-o-p">Get</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-getactivescriptormodule">GetActiveScriptOrModule</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-getfunctionrealm">GetFunctionRealm</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-hasownproperty">HasOwnProperty</dfn> abstract operation</li>
     <li>The <dfn data-x="js-HostCallJobCallback" data-x-href="https://tc39.es/ecma262/#sec-hostcalljobcallback">HostCallJobCallback</dfn> abstract operation</li>
     <li>The <dfn data-x="js-HostEnqueueFinalizationRegistryCleanupJob" data-x-href="https://tc39.es/ecma262/#sec-host-cleanup-finalization-registry">HostEnqueueFinalizationRegistryCleanupJob</dfn> abstract operation</li>
     <li>The <dfn data-x="js-HostEnqueueGenericJob" data-x-href="https://tc39.es/ecma262/#sec-hostenqueuegenericjob">HostEnqueueGenericJob</dfn> abstract operation</li>
     <li>The <dfn data-x="js-HostEnqueuePromiseJob" data-x-href="https://tc39.es/ecma262/#sec-hostenqueuepromisejob">HostEnqueuePromiseJob</dfn> abstract operation</li>
     <li>The <dfn data-x="js-HostEnqueueTimeoutJob" data-x-href="https://tc39.es/ecma262/#sec-hostenqueuetimeoutjob">HostEnqueueTimeoutJob</dfn> abstract operation</li>
     <li>The <dfn data-x="js-HostEnsureCanAddPrivateElement" data-x-href="https://tc39.es/ecma262/#sec-hostensurecanaddprivateelement">HostEnsureCanAddPrivateElement</dfn> abstract operation</li>
     <li>The <dfn data-x="js-HostGetSupportedImportAttributes" data-x-href="https://tc39.es/ecma262/#sec-hostgetsupportedimportattributes">HostGetSupportedImportAttributes</dfn> abstract operation</li>
     <li>The <dfn data-x="js-HostLoadImportedModule" data-x-href="https://tc39.es/ecma262/#sec-HostLoadImportedModule">HostLoadImportedModule</dfn> abstract operation</li>
     <li>The <dfn data-x="js-HostMakeJobCallback" data-x-href="https://tc39.es/ecma262/#sec-hostmakejobcallback">HostMakeJobCallback</dfn> abstract operation</li>
     <li>The <dfn data-x="js-HostPromiseRejectionTracker" data-x-href="https://tc39.es/ecma262/#sec-host-promise-rejection-tracker">HostPromiseRejectionTracker</dfn> abstract operation</li>
     <li>The <dfn data-x="js-InitializeHostDefinedRealm" data-x-href="https://tc39.es/ecma262/#sec-initializehostdefinedrealm">InitializeHostDefinedRealm</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-isarraybufferviewoutofbounds">IsArrayBufferViewOutOfBounds</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-isaccessordescriptor">IsAccessorDescriptor</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-iscallable">IsCallable</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-isconstructor">IsConstructor</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-isdatadescriptor">IsDataDescriptor</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-isdetachedbuffer">IsDetachedBuffer</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-issharedarraybuffer">IsSharedArrayBuffer</dfn> abstract operation</li>
     <li>The <dfn data-x="js-NewObjectEnvironment" data-x-href="https://tc39.es/ecma262/#sec-newobjectenvironment">NewObjectEnvironment</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-normalcompletion">NormalCompletion</dfn> abstract operation</li>
     <!-- the next set of Ordinary* abstract operations are in order of appearance -->
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-ordinarygetprototypeof">OrdinaryGetPrototypeOf</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-ordinarysetprototypeof">OrdinarySetPrototypeOf</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-ordinaryisextensible">OrdinaryIsExtensible</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-ordinarypreventextensions">OrdinaryPreventExtensions</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-ordinarygetownproperty">OrdinaryGetOwnProperty</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-ordinarydefineownproperty">OrdinaryDefineOwnProperty</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-ordinaryget">OrdinaryGet</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-ordinaryset">OrdinarySet</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-ordinarydelete">OrdinaryDelete</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-ordinaryownpropertykeys">OrdinaryOwnPropertyKeys</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-objectcreate">OrdinaryObjectCreate</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-parse-json-module">ParseJSONModule</dfn> abstract operation</li>
     <li>The <dfn data-x="js-ParseModule" data-x-href="https://tc39.es/ecma262/#sec-parsemodule">ParseModule</dfn> abstract operation</li>
     <li>The <dfn data-x="js-ParseScript" data-x-href="https://tc39.es/ecma262/#sec-parse-script">ParseScript</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-newpromisereactionjob">NewPromiseReactionJob</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-newpromiseresolvethenablejob">NewPromiseResolveThenableJob</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-regexpbuiltinexec">RegExpBuiltinExec</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-regexpcreate">RegExpCreate</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-runjobs">RunJobs</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-samevalue">SameValue</dfn> abstract operation</li>
     <li>The <dfn data-x="js-ScriptEvaluation" data-x-href="https://tc39.es/ecma262/#sec-runtime-semantics-scriptevaluation">ScriptEvaluation</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-setsyntheticmoduleexport">SetSyntheticModuleExport</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-set-immutable-prototype">SetImmutablePrototype</dfn> abstract operation</li>
     <li>The <dfn data-x="js-ToBoolean" data-x-href="https://tc39.es/ecma262/#sec-toboolean">ToBoolean</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-tostring">ToString</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-touint32">ToUint32</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#typedarray-create">TypedArrayCreate</dfn> abstract operation</li>

     <li>The <dfn data-x="js-abstract-equality" data-x-href="https://tc39.es/ecma262/#sec-islooselyequal">IsLooselyEqual</dfn> abstract operation</li>
     <li>The <dfn data-x="js-strict-equality" data-x-href="https://tc39.es/ecma262/#sec-isstrictlyequal">IsStrictlyEqual</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-atomics-object"><code>Atomics</code></dfn> object</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-atomics.waitasync"><code>Atomics.waitAsync</code></dfn> object</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-date-objects"><code>Date</code></dfn> class</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-finalization-registry-objects"><code>FinalizationRegistry</code></dfn> class</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-regexp-regular-expression-objects"><code>RegExp</code></dfn> class</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-sharedarraybuffer-objects"><code>SharedArrayBuffer</code></dfn> class</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-native-error-types-used-in-this-standard-syntaxerror"><code>SyntaxError</code></dfn> class</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-native-error-types-used-in-this-standard-typeerror"><code>TypeError</code></dfn> class</li>
     <li>The <dfn data-x="js-RangeError" data-x-href="https://tc39.es/ecma262/#sec-native-error-types-used-in-this-standard-rangeerror"><code>RangeError</code></dfn> class</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-weak-ref-objects"><code>WeakRef</code></dfn> class</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-eval-x"><code>eval()</code></dfn> function</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-weak-ref.prototype.deref"><code>WeakRef.prototype.deref()</code></dfn> function</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-IsHTMLDDA-internal-slot">[[IsHTMLDDA]]</dfn> internal slot</li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#sec-import-calls"><code>import()</code></dfn></li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#sec-meta-properties"><code>import.meta</code></dfn></li>
     <li>The <dfn data-x="js-HostGetImportMetaProperties" data-x-href="https://tc39.es/ecma262/#sec-hostgetimportmetaproperties">HostGetImportMetaProperties</dfn> abstract operation</li>
     <li>The <dfn data-x="js-typeof" data-x-href="https://tc39.es/ecma262/#sec-typeof-operator"><code>typeof</code></dfn> operator</li>
     <li>The <dfn data-x-href="https://tc39.es/ecma262/#sec-delete-operator"><code>delete</code></dfn> operator</li>
     <li><dfn data-x-href="https://tc39.es/ecma262/#table-49">The <var>TypedArray</var> Constructors</dfn> table</li>
    </ul>

    <p>Users agents that support JavaScript must also implement the <cite>Dynamic Code Brand
    Checks</cite> proposal. The following terms are defined there, and used in this specification:
    <ref>JSDYNAMICCODEBRANDCHECKS</ref></p>

    <ul class="brief">
     <li>The <dfn data-x="js-HostEnsureCanCompileStrings" data-x-href="https://tc39.es/proposal-dynamic-code-brand-checks/#sec-hostensurecancompilestrings">HostEnsureCanCompileStrings</dfn> abstract operation</li>
     <li>The <dfn data-x="js-HostGetCodeForEval" data-x-href="https://tc39.es/proposal-dynamic-code-brand-checks/#sec-hostgetcodeforeval">HostGetCodeForEval</dfn> abstract operation</li>
    </ul>

    <p>Users agents that support JavaScript must also implement <cite>ECMAScript
    Internationalization API</cite>. <ref>JSINTL</ref></p>

    <p>User agents that support JavaScript must also implement the <cite>Temporal</cite> proposal.
    The following terms are defined there, and used in this specification: <ref>JSTEMPORAL</ref></p>

    <ul class="brief">
     <li>The <dfn data-x="js-HostSystemUTCEpochNanoseconds" data-x-href="https://tc39.es/proposal-temporal/#sec-hostsystemutcepochnanoseconds">HostSystemUTCEpochNanoseconds</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://tc39.es/proposal-temporal/#eqn-nsMaxInstant">nsMaxInstant</dfn> and
         <dfn data-x-href="https://tc39.es/proposal-temporal/#eqn-nsMinInstant">nsMinInstant</dfn> values</li>
    </ul>
   </dd>

   <dt>WebAssembly</dt>

   <dd>
    <p>The following term is defined in <cite>WebAssembly JavaScript Interface</cite>:
    <ref>WASMJS</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://webassembly.github.io/spec/js-api/#module"><code>WebAssembly.Module</code></dfn></li>
    </ul>
   </dd>

   <dt>DOM</dt>

   <dd>
    <p>The Document Object Model (DOM) is a representation &mdash; a model &mdash; of a document and
    its content. The DOM is not just an API; the conformance criteria of HTML implementations are
    defined, in this specification, in terms of operations on the DOM. <ref>DOM</ref></p>

    <p>Implementations must support DOM and the events defined in UI Events, because this
    specification is defined in terms of the DOM, and some of the features are defined as extensions
    to the DOM interfaces. <ref>DOM</ref> <ref>UIEVENTS</ref></p>

    <p>In particular, the following features are defined in <cite>DOM</cite>: <ref>DOM</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-attr"><code>Attr</code></dfn> interface</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-characterdata"><code>CharacterData</code></dfn> interface</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-comment"><code>Comment</code></dfn> interface</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-domimplementation"><code>DOMImplementation</code></dfn> interface</li>
     <li><dfn data-x="DOM Document" data-x-href="https://dom.spec.whatwg.org/#interface-document"><code>Document</code></dfn> interface and its
         <dfn data-x="dom-Document-doctype" data-x-href="https://dom.spec.whatwg.org/#dom-document-doctype"><code>doctype</code></dfn> attribute
     </li>
     <li><dfn data-x="DOM DocumentOrShadowRoot" data-x-href="https://dom.spec.whatwg.org/#documentorshadowroot"><code>DocumentOrShadowRoot</code></dfn> interface</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-documentfragment"><code>DocumentFragment</code></dfn> interface</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-documenttype"><code>DocumentType</code></dfn> interface</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-childnode"><code>ChildNode</code></dfn> interface</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-element"><code>Element</code></dfn> interface</li>
     <li><dfn data-x="dom-Element-attachShadow" data-x-href="https://dom.spec.whatwg.org/#dom-element-attachshadow"><code>attachShadow()</code></dfn> method.</li>
     <li>An element's <dfn data-x="concept-element-shadow-root" data-x-href="https://dom.spec.whatwg.org/#concept-element-shadow-root">shadow root</dfn></li>
     <li>A <span data-x="concept-element-shadow-root">shadow root</span>'s <dfn data-x="concept-shadow-root-mode" data-x-href="https://dom.spec.whatwg.org/#shadowroot-mode">mode</dfn></li>
     <li>A <span data-x="concept-element-shadow-root">shadow root</span>'s <dfn data-x="concept-shadow-root-declarative" data-x-href="https://dom.spec.whatwg.org/#shadowroot-declarative">declarative</dfn> member</li>
     <li>The <dfn data-x="concept-attach-a-shadow-root" data-x-href="https://dom.spec.whatwg.org/#concept-attach-a-shadow-root">attach a shadow root</dfn> algorithm</li>
     <li>The <dfn data-x="dom-retarget" data-x-href="https://dom.spec.whatwg.org/#retarget">retargeting algorithm</dfn></li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-node"><code>Node</code></dfn> interface</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-nodelist"><code>NodeList</code></dfn> interface</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-processinginstruction"><code>ProcessingInstruction</code></dfn> interface</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-shadowroot"><code>ShadowRoot</code></dfn> interface</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-text"><code>Text</code></dfn> interface</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-range"><code>Range</code></dfn> interface</li>

     <li><dfn data-x-href="https://dom.spec.whatwg.org/#concept-node-document">node document</dfn> concept</li>
     <li><dfn data-x="concept-document-type" data-x-href="https://dom.spec.whatwg.org/#concept-document-type">document type</dfn> concept</li>
     <li><dfn data-x="concept-DocumentFragment-host" data-x-href="https://dom.spec.whatwg.org/#concept-documentfragment-host">host</dfn> concept</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#concept-shadow-root">shadow root</dfn> concept, and its <dfn data-x-href="https://dom.spec.whatwg.org/#shadowroot-delegates-focus">delegates focus</dfn>, <dfn data-x-href="https://dom.spec.whatwg.org/#shadowroot-available-to-element-internals">available to element internals</dfn>, <dfn data-x-href="https://dom.spec.whatwg.org/#shadowroot-clonable">clonable</dfn>, and <dfn data-x="shadow-serializable" data-x-href="https://dom.spec.whatwg.org/#shadowroot-serializable">serializable</dfn>.</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#element-shadow-host">shadow host</dfn> concept</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-htmlcollection"><code>HTMLCollection</code></dfn> interface, its
         <dfn data-x="dom-HTMLCollection-length" data-x-href="https://dom.spec.whatwg.org/#dom-htmlcollection-length"><code>length</code></dfn> attribute, and its
         <dfn data-x="dom-HTMLCollection-item" data-x-href="https://dom.spec.whatwg.org/#dom-htmlcollection-item"><code>item()</code></dfn> and
         <dfn data-x="dom-HTMLCollection-namedItem" data-x-href="https://dom.spec.whatwg.org/#dom-htmlcollection-nameditem"><code>namedItem()</code></dfn> methods</li>
     <li>The terms <dfn data-x="concept-collection" data-x-href="https://dom.spec.whatwg.org/#concept-collection">collection</dfn> and <dfn data-x-href="https://dom.spec.whatwg.org/#represented-by-the-collection">represented by the collection</dfn></li>

     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-domtokenlist"><code>DOMTokenList</code></dfn> interface, and its
         <dfn data-x="dom-DOMTokenList-value" data-x-href="https://dom.spec.whatwg.org/#dom-domtokenlist-value"><code>value</code></dfn> attribute and
         <dfn data-x="dom-DOMTokenList-supports" data-x-href="https://dom.spec.whatwg.org/#dom-domtokenlist-supports"><code>supports</code></dfn> operation</li>

     <li><dfn data-x="dom-DOMImplementation-createDocument" data-x-href="https://dom.spec.whatwg.org/#dom-domimplementation-createdocument"><code>createDocument()</code></dfn> method</li>
     <li><dfn data-x="dom-DOMImplementation-createHTMLDocument" data-x-href="https://dom.spec.whatwg.org/#dom-domimplementation-createhtmldocument"><code>createHTMLDocument()</code></dfn> method</li>
     <li><dfn data-x="dom-Document-createElement" data-x-href="https://dom.spec.whatwg.org/#dom-document-createelement"><code>createElement()</code></dfn> method</li>
     <li><dfn data-x="dom-Document-createElementNS" data-x-href="https://dom.spec.whatwg.org/#dom-document-createelementns"><code>createElementNS()</code></dfn> method</li>
     <li><dfn data-x="dom-Document-getElementById" data-x-href="https://dom.spec.whatwg.org/#dom-nonelementparentnode-getelementbyid"><code>getElementById()</code></dfn> method</li>
     <li><dfn data-x="dom-document-getElementsByClassName" data-x-href="https://dom.spec.whatwg.org/#dom-document-getelementsbyclassname"><code>getElementsByClassName()</code></dfn> method</li>
     <li><dfn data-x="dom-Node-append" data-x-href="https://dom.spec.whatwg.org/#dom-node-append"><code>append()</code></dfn> method</li>
     <li><dfn data-x="dom-Node-appendChild" data-x-href="https://dom.spec.whatwg.org/#dom-node-appendchild"><code>appendChild()</code></dfn> method</li>
     <li><dfn data-x="dom-Node-cloneNode" data-x-href="https://dom.spec.whatwg.org/#dom-node-clonenode"><code>cloneNode()</code></dfn> method</li>
     <li><dfn data-x="dom-ParentNode-moveBefore" data-x-href="https://dom.spec.whatwg.org/#dom-parentnode-movebefore"><code>moveBefore()</code></dfn> method</li>
     <li><dfn data-x="dom-Document-importNode" data-x-href="https://dom.spec.whatwg.org/#dom-document-importnode"><code>importNode()</code></dfn> method</li>
     <li><dfn data-x="dom-Event-preventDefault" data-x-href="https://dom.spec.whatwg.org/#dom-event-preventdefault"><code>preventDefault()</code></dfn> method</li>
     <li><dfn data-x="dom-Element-id" data-x-href="https://dom.spec.whatwg.org/#dom-element-id"><code>id</code></dfn> attribute</li>
     <li><dfn data-x="dom-Element-setAttribute" data-x-href="https://dom.spec.whatwg.org/#dom-element-setattribute"><code>setAttribute()</code></dfn> method</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#dom-node-textcontent"><code>textContent</code></dfn> attribute</li>

     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#concept-tree">tree</dfn>, <dfn data-x-href="https://dom.spec.whatwg.org/#concept-shadow-tree">shadow tree</dfn>, and <dfn data-x-href="https://dom.spec.whatwg.org/#concept-node-tree">node tree</dfn> concepts</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#concept-tree-order">tree order</dfn> and <dfn data-x-href="https://dom.spec.whatwg.org/#concept-shadow-including-tree-order">shadow-including tree order</dfn> concepts</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#concept-element" data-x="concept-element">element</dfn> concept</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#concept-tree-child" data-x="concept-tree-child">child</dfn> concept</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#concept-tree-root">root</dfn> and <dfn data-x-href="https://dom.spec.whatwg.org/#concept-shadow-including-root">shadow-including root</dfn> concepts</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#concept-tree-inclusive-ancestor">inclusive ancestor</dfn>,
             <dfn data-x-href="https://dom.spec.whatwg.org/#concept-tree-descendant">descendant</dfn>,
             <dfn data-x="concept-shadow-including-ancestor" data-x-href="https://dom.spec.whatwg.org/#concept-shadow-including-ancestor">shadow-including ancestor</dfn>,
             <dfn data-x-href="https://dom.spec.whatwg.org/#concept-shadow-including-descendant">shadow-including descendant</dfn>,
             <dfn data-x-href="https://dom.spec.whatwg.org/#concept-shadow-including-inclusive-descendant">shadow-including inclusive descendant</dfn>, and
             <dfn data-x-href="https://dom.spec.whatwg.org/#concept-shadow-including-inclusive-ancestor">shadow-including inclusive ancestor</dfn> concepts</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#concept-tree-first-child">first child</dfn>,
             <dfn data-x-href="https://dom.spec.whatwg.org/#concept-tree-next-sibling">next sibling</dfn>,
             <dfn data-x-href="https://dom.spec.whatwg.org/#concept-tree-previous-sibling">previous sibling</dfn>, and
             <dfn data-x-href="https://dom.spec.whatwg.org/#concept-tree-parent">parent</dfn> concepts</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#parent-element">parent element</dfn> concept</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#document-element">document element</dfn> concept</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#in-a-document-tree">in a document tree</dfn>, <dfn data-x-href="https://dom.spec.whatwg.org/#in-a-document">in a document</dfn> (legacy), and <dfn data-x-href="https://dom.spec.whatwg.org/#connected">connected</dfn> concepts</li>
     <li>The <dfn data-x="concept-slot" data-x-href="https://dom.spec.whatwg.org/#concept-slot">slot</dfn> concept, and its <dfn data-x="slot-name" data-x-href="https://dom.spec.whatwg.org/#slot-name">name</dfn> and <dfn data-x-href="https://dom.spec.whatwg.org/#slot-assigned-nodes">assigned nodes</dfn></li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#slotable-assigned-slot">assigned slot</dfn> concept</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#dom-shadowroot-slot-assignment">slot assignment</dfn> concept</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#concept-slotable">slottable</dfn> concept</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#assign-slotables-for-a-tree">assign slottables for a tree</dfn> algorithm</li>
     <li>The <dfn data-x="event-slotchange" data-x-href="https://dom.spec.whatwg.org/#eventdef-htmlslotelement-slotchange"><code>slotchange</code></dfn> event</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#concept-tree-inclusive-descendant">inclusive descendant</dfn> concept</li>
     <li>The <dfn data-x="finding flattened slottables" data-x-href="https://dom.spec.whatwg.org/#find-flattened-slotables">find flattened slottables</dfn> algorithm</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#slottable-manual-slot-assignment">manual slot assignment</dfn> concept</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#assign-a-slot">assign a slot</dfn> algorithm</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#concept-node-pre-insert">pre-insert</dfn>, <dfn data-x="concept-node-insert" data-x-href="https://dom.spec.whatwg.org/#concept-node-insert">insert</dfn>, <dfn data-x="concept-node-append" data-x-href="https://dom.spec.whatwg.org/#concept-node-append">append</dfn>, <dfn data-x="concept-node-replace" data-x-href="https://dom.spec.whatwg.org/#concept-node-replace">replace</dfn>, <dfn data-x="concept-node-replace-all" data-x-href="https://dom.spec.whatwg.org/#concept-node-replace-all">replace all</dfn>, <dfn data-x-href="https://dom.spec.whatwg.org/#string-replace-all">string replace all</dfn>, <dfn data-x="concept-node-remove" data-x-href="https://dom.spec.whatwg.org/#concept-node-remove">remove</dfn>, and <dfn data-x="concept-node-adopt" data-x-href="https://dom.spec.whatwg.org/#concept-node-adopt">adopt</dfn> algorithms for nodes</li>
     <li>The <dfn data-x="concept-tree-descendant" data-x-href="https://dom.spec.whatwg.org/#concept-tree-descendant">descendant</dfn> concept</li>
     <li>The <dfn data-x="concept-node-insert-ext" data-x-href="https://dom.spec.whatwg.org/#concept-node-insert-ext">insertion steps</dfn>,
     <li>The <dfn data-x="concept-node-post-connection-ext" data-x-href="https://dom.spec.whatwg.org/#concept-node-post-connection-ext">post-connection steps</dfn>,
             <dfn data-x="concept-node-remove-ext" data-x-href="https://dom.spec.whatwg.org/#concept-node-remove-ext">removing steps</dfn>,
             <dfn data-x="concept-node-move-ext" data-x-href="https://dom.spec.whatwg.org/#concept-node-move-ext">moving steps</dfn>,
             <dfn data-x="concept-node-adopt-ext" data-x-href="https://dom.spec.whatwg.org/#concept-node-adopt-ext">adopting steps</dfn>, and
             <dfn data-x-href="https://dom.spec.whatwg.org/#concept-node-children-changed-ext">children changed steps</dfn> hooks for elements</li>
     <li>The <dfn data-x="concept-element-attributes-change" data-x-href="https://dom.spec.whatwg.org/#concept-element-attributes-change">change</dfn>, <dfn data-x="concept-element-attributes-append" data-x-href="https://dom.spec.whatwg.org/#concept-element-attributes-append">append</dfn>, <dfn data-x="concept-element-attributes-remove" data-x-href="https://dom.spec.whatwg.org/#concept-element-attributes-remove">remove</dfn>, <dfn data-x="concept-element-attributes-replace" data-x-href="https://dom.spec.whatwg.org/#concept-element-attributes-replace">replace</dfn>, <dfn data-x="concept-element-attributes-get-by-namespace" data-x-href="https://dom.spec.whatwg.org/#concept-element-attributes-get-by-namespace">get an attribute by namespace and local name</dfn>, <dfn data-x="concept-element-attributes-set-value" data-x-href="https://dom.spec.whatwg.org/#concept-element-attributes-set-value">set value</dfn>, and <dfn data-x="concept-element-attributes-remove-by-namespace" data-x-href="https://dom.spec.whatwg.org/#concept-element-attributes-remove-by-namespace">remove an attribute by namespace and local name</dfn> algorithms for attributes</li>
     <li>The <dfn data-x="concept-element-attributes-change-ext" data-x-href="https://dom.spec.whatwg.org/#concept-element-attributes-change-ext">attribute change steps</dfn> hook for attributes</li>
     <li>The <dfn data-x="concept-attribute-value" data-x-href="https://dom.spec.whatwg.org/#concept-attribute-value">value</dfn> concept for attributes</li>
     <li>The <dfn data-x="concept-attribute-local-name" data-x-href="https://dom.spec.whatwg.org/#concept-attribute-local-name">local name</dfn> concept for attributes</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#concept-element-attribute">attribute list</dfn> concept</li>
     <li>The <dfn data-x="concept-cd-data" data-x-href="https://dom.spec.whatwg.org/#concept-cd-data">data</dfn> of a <code>CharacterData</code> node and its
             <dfn data-x-href="https://dom.spec.whatwg.org/#concept-cd-replace">replace data</dfn> algorithm</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#concept-child-text-content">child text content</dfn> of a node</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#concept-descendant-text-content">descendant text content</dfn> of a node</li>
     <li>The <dfn data-x="concept-doctype-name" data-x-href="https://dom.spec.whatwg.org/#concept-doctype-name">name</dfn>,
             <dfn data-x="concept-doctype-publicid" data-x-href="https://dom.spec.whatwg.org/#concept-doctype-publicid">public ID</dfn>, and
             <dfn data-x="concept-doctype-systemid" data-x-href="https://dom.spec.whatwg.org/#concept-doctype-systemid">system ID</dfn> of a doctype</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-event"><code>Event</code></dfn> interface</li>
     <li><dfn data-x="dom-Event-constructor" data-x-href="https://dom.spec.whatwg.org/#concept-event-constructor"><code>Event</code> and derived interfaces constructor behavior</dfn></li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#interface-eventtarget"><code>EventTarget</code></dfn> interface</li>
     <li>The <!--en-GB--><dfn id="activation-behaviour" data-x-href="https://dom.spec.whatwg.org/#eventtarget-activation-behavior">activation behavior</dfn> hook</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#eventtarget-legacy-pre-activation-behavior">legacy-pre-activation behavior</dfn> hook</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#eventtarget-legacy-canceled-activation-behavior">legacy-canceled-activation behavior</dfn> hook</li>
     <li>The <dfn data-x="creating an event" data-x-href="https://dom.spec.whatwg.org/#concept-event-create">create an event</dfn> algorithm</li>
     <li>The <dfn data-x="concept-event-fire" data-x-href="https://dom.spec.whatwg.org/#concept-event-fire">fire an event</dfn> algorithm</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#canceled-flag">canceled flag</dfn></li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#dispatch-flag">dispatch flag</dfn></li>
     <li>The <dfn data-x="concept-event-dispatch" data-x-href="https://dom.spec.whatwg.org/#concept-event-dispatch">dispatch</dfn> algorithm</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#dictdef-eventinit"><code>EventInit</code></dfn> dictionary type</li>
     <li><dfn data-x="dom-Event-type" data-x-href="https://dom.spec.whatwg.org/#dom-event-type"><code>type</code></dfn> attribute</li>
     <li>An event's <dfn data-x="concept-event-target" data-x-href="https://dom.spec.whatwg.org/#concept-event-target">target</dfn></li>
     <li><dfn data-x="dom-Event-currentTarget" data-x-href="https://dom.spec.whatwg.org/#dom-event-currenttarget"><code>currentTarget</code></dfn> attribute</li>
     <li><dfn data-x="dom-Event-bubbles" data-x-href="https://dom.spec.whatwg.org/#dom-event-bubbles"><code>bubbles</code></dfn> attribute</li>
     <li><dfn data-x="dom-Event-cancelable" data-x-href="https://dom.spec.whatwg.org/#dom-event-cancelable"><code>cancelable</code></dfn> attribute</li>
     <li><dfn data-x="dom-Event-composed" data-x-href="https://dom.spec.whatwg.org/#dom-event-composed"><code>composed</code></dfn> attribute</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#composed-flag">composed flag</dfn></li>
     <li><dfn data-x="dom-Event-isTrusted" data-x-href="https://dom.spec.whatwg.org/#dom-event-istrusted"><code>isTrusted</code></dfn> attribute</li>
     <li><dfn data-x="dom-Event-initEvent" data-x-href="https://dom.spec.whatwg.org/#dom-event-initevent"><code>initEvent()</code></dfn> method</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#add-an-event-listener">add an event listener</dfn></li>
     <li><dfn data-x="dom-EventTarget-addEventListener" data-x-href="https://dom.spec.whatwg.org/#dom-eventtarget-addeventlistener"><code>addEventListener()</code></dfn> method</li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#remove-an-event-listener">remove an event listener</dfn> and
             <dfn data-x-href="https://dom.spec.whatwg.org/#remove-all-event-listeners">remove all event listeners</dfn> algorithms</li>
     <li><dfn data-x="dom-EventListener" data-x-href="https://dom.spec.whatwg.org/#callbackdef-eventlistener"><code>EventListener</code></dfn> callback interface</li>
     <li>The <dfn data-x="concept-event-type" data-x-href="https://dom.spec.whatwg.org/#dom-event-type">type</dfn> of an event</li>
     <li>An <dfn data-x-href="https://dom.spec.whatwg.org/#concept-event-listener">event listener</dfn> and its
            <dfn data-x="event listener type" data-x-href="https://dom.spec.whatwg.org/#event-listener-type">type</dfn> and
            <dfn data-x="event listener callback" data-x-href="https://dom.spec.whatwg.org/#event-listener-callback">callback</dfn></li>

     <li>The <dfn data-x="document's character encoding" data-x-href="https://dom.spec.whatwg.org/#concept-document-encoding">encoding</dfn> (herein the <i>character encoding</i>),
             <dfn data-x="concept-document-mode" data-x-href="https://dom.spec.whatwg.org/#concept-document-mode">mode</dfn>,
             <dfn data-x="concept-document-allow-declarative-shadow-roots" data-x-href="https://dom.spec.whatwg.org/#concept-document-allow-declarative-shadow-roots">allow declarative shadow roots</dfn>, and
             <dfn data-x="concept-document-content-type" data-x-href="https://dom.spec.whatwg.org/#concept-document-content-type">content type</dfn> of a <code>Document</code></li>
     <li>The distinction between <dfn data-x-href="https://dom.spec.whatwg.org/#xml-document">XML documents</dfn> and
                                 <dfn data-x-href="https://dom.spec.whatwg.org/#html-document">HTML documents</dfn></li>
     <li>The terms <dfn data-x-href="https://dom.spec.whatwg.org/#concept-document-quirks">quirks mode</dfn>,
                   <dfn data-x-href="https://dom.spec.whatwg.org/#concept-document-limited-quirks">limited-quirks mode</dfn>, and
                   <dfn data-x-href="https://dom.spec.whatwg.org/#concept-document-no-quirks">no-quirks mode</dfn></li>
     <li>The algorithm <dfn data-x="concept-node-clone" data-x-href="https://dom.spec.whatwg.org/#concept-node-clone">clone a node</dfn> with its arguments
                       <dfn data-x="concept-node-clone-document" data-x-href="https://dom.spec.whatwg.org/#clone-a-node-document"><var>document</var></dfn>,
                       <dfn data-x="concept-node-clone-subtree" data-x-href="https://dom.spec.whatwg.org/#clone-a-node-subtree"><var>subtree</var></dfn>, and
                       <dfn data-x="concept-node-clone-parent" data-x-href="https://dom.spec.whatwg.org/#clone-a-node-parent"><var>parent</var></dfn>, and the concept of
                       <dfn data-x="concept-node-clone-ext" data-x-href="https://dom.spec.whatwg.org/#concept-node-clone-ext">cloning steps</dfn></li>
     <li>The concept of <dfn>base URL change steps</dfn> and the definition of what happens when an element is <dfn>affected by a base URL change</dfn></li>
     <li>The concept of an element's <dfn data-x="concept-id" data-x-href="https://dom.spec.whatwg.org/#concept-id">unique identifier (ID)</dfn></li>
     <li>The concept of an element's <dfn data-x="concept-class" data-x-href="https://dom.spec.whatwg.org/#concept-class">classes</dfn></li>
     <li>The term <dfn data-x="concept-supported-tokens" data-x-href="https://dom.spec.whatwg.org/#concept-supported-tokens">supported tokens</dfn></li>

     <li>The concept of a DOM <dfn data-x="concept-range" data-x-href="https://dom.spec.whatwg.org/#concept-range">range</dfn>, and the terms
                              <dfn data-x="concept-range-start-node" data-x-href="https://dom.spec.whatwg.org/#concept-range-start-node">start node</dfn>,
                              <dfn data-x="concept-range-start" data-x-href="https://dom.spec.whatwg.org/#concept-range-start">start</dfn>,
                              <dfn data-x="concept-range-end" data-x-href="https://dom.spec.whatwg.org/#concept-range-end">end</dfn>, and
                              <dfn data-x="concept-range-bp" data-x-href="https://dom.spec.whatwg.org/#concept-range-bp">boundary point</dfn> as applied to ranges.</li>

     <li>The <dfn data-x="create an element" data-x-href="https://dom.spec.whatwg.org/#concept-create-element">create an element</dfn> algorithm</li>
     <li>The <dfn data-x="element interface" data-x-href="https://dom.spec.whatwg.org/#concept-element-interface">element interface</dfn> concept</li>
     <li>The concepts of <dfn data-x-href="https://dom.spec.whatwg.org/#concept-element-custom-element-state">custom element state</dfn>, and of
                         <dfn data-x="concept-element-defined" data-x-href="https://dom.spec.whatwg.org/#concept-element-defined">defined</dfn> and
                         <dfn data-x="concept-element-custom" data-x-href="https://dom.spec.whatwg.org/#concept-element-custom">custom</dfn> elements</li>
     <li>An element's <dfn data-x="concept-element-namespace" data-x-href="https://dom.spec.whatwg.org/#concept-element-namespace">namespace</dfn>,
                      <dfn data-x="concept-element-namespace-prefix" data-x-href="https://dom.spec.whatwg.org/#concept-element-namespace-prefix">namespace prefix</dfn>,
                      <dfn data-x="concept-element-local-name" data-x-href="https://dom.spec.whatwg.org/#concept-element-local-name">local name</dfn>,
                      <dfn data-x="concept-element-custom-element-definition" data-x-href="https://dom.spec.whatwg.org/#concept-element-custom-element-definition">custom element definition</dfn>, and
                      <dfn data-x="concept-element-is-value" data-x-href="https://dom.spec.whatwg.org/#concept-element-is-value"><code data-x="">is</code> value</dfn></li>

     <li><dfn data-x-href="https://dom.spec.whatwg.org/#mutationobserver"><code>MutationObserver</code></dfn> interface and <dfn data-x-href="https://dom.spec.whatwg.org/#mutation-observers">mutation observers</dfn> in general</li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#abortcontroller"><code>AbortController</code></dfn> and its
         <dfn data-x="concept-AbortController-signal" data-x-href="https://dom.spec.whatwg.org/#abortcontroller-signal">signal</dfn></li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#abortsignal"><code>AbortSignal</code></dfn></li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#abortsignal-aborted" data-x="AbortSignal-aborted">aborted</dfn></li>
     <li><dfn data-x-href="https://dom.spec.whatwg.org/#abortcontroller-signal-abort">signal abort</dfn></li>
     <li><dfn data-x="AbortSignal-add" data-x-href="https://dom.spec.whatwg.org/#abortsignal-add">add</dfn></li>
     <li>The <dfn data-x-href="https://dom.spec.whatwg.org/#concept-element-attributes-get-by-name">get an attribute by name</dfn> algorithm</li>
    </ul>

    <p>The following features are defined in <cite>UI Events</cite>: <ref>UIEVENTS</ref></p>

    <ul class="brief">
     <li>The <dfn data-x-href="https://w3c.github.io/uievents/#mouseevent"><code>MouseEvent</code></dfn> interface</li>
     <li>The <code>MouseEvent</code> interface's <dfn data-x="dom-MouseEvent-relatedTarget" data-x-href="https://w3c.github.io/uievents/#dom-mouseevent-relatedtarget"><code>relatedTarget</code></dfn> attribute</li>
     <li><dfn data-x-href="https://w3c.github.io/uievents/#dictdef-mouseeventinit"><code>MouseEventInit</code></dfn> dictionary type</li>

     <li>The <dfn data-x-href="https://w3c.github.io/uievents/#focusevent"><code>FocusEvent</code></dfn> interface</li>
     <li>The <code>FocusEvent</code> interface's <dfn data-x="dom-FocusEvent-relatedTarget" data-x-href="https://w3c.github.io/uievents/#dom-focusevent-relatedtarget"><code>relatedTarget</code></dfn> attribute</li>

     <li>The <dfn data-x-href="https://w3c.github.io/uievents/#uievent"><code>UIEvent</code></dfn> interface</li>
     <li>The <code>UIEvent</code> interface's <dfn data-x="dom-UIEvent-view" data-x-href="https://w3c.github.io/uievents/#dom-uievent-view"><code>view</code></dfn> attribute</li>

     <li><dfn data-x="event-auxclick" data-x-href="https://w3c.github.io/uievents/#event-type-auxclick"><code>auxclick</code></dfn> event</li>
     <li><dfn data-x="event-beforeinput" data-x-href="https://w3c.github.io/uievents/#event-type-beforeinput"><code>beforeinput</code></dfn> event</li>
     <li><dfn data-x="event-click" data-x-href="https://w3c.github.io/uievents/#event-type-click"><code>click</code></dfn> event</li>
     <li><dfn data-x="event-contextmenu" data-x-href="https://w3c.github.io/uievents/#event-type-contextmenu"><code>contextmenu</code></dfn> event</li>
     <li><dfn data-x="event-dblclick" data-x-href="https://w3c.github.io/uievents/#event-type-dblclick"><code>dblclick</code></dfn> event</li>
     <li><dfn data-x="event-input" data-x-href="https://w3c.github.io/uievents/#event-type-input"><code>input</code></dfn> event</li>
     <li><dfn data-x="event-mousedown" data-x-href="https://w3c.github.io/uievents/#event-type-mousedown"><code>mousedown</code></dfn> event</li>
     <li><dfn data-x="event-mouseenter" data-x-href="https://w3c.github.io/uievents/#event-type-mouseenter"><code>mouseenter</code></dfn> event</li>
     <li><dfn data-x="event-mouseleave" data-x-href="https://w3c.github.io/uievents/#event-type-mouseleave"><code>mouseleave</code></dfn> event</li>
     <li><dfn data-x="event-mousemove" data-x-href="https://w3c.github.io/uievents/#event-type-mousemove"><code>mousemove</code></dfn> event</li>
     <li><dfn data-x="event-mouseout" data-x-href="https://w3c.github.io/uievents/#event-type-mouseout"><code>mouseout</code></dfn> event</li>
     <li><dfn data-x="event-mouseover" data-x-href="https://w3c.github.io/uievents/#event-type-mouseover"><code>mouseover</code></dfn> event</li>
     <li><dfn data-x="event-mouseup" data-x-href="https://w3c.github.io/uievents/#event-type-mouseup"><code>mouseup</code></dfn> event</li>
     <li><dfn data-x="event-wheel" data-x-href="https://w3c.github.io/uievents/#event-type-wheel"><code>wheel</code></dfn> event</li>

     <li><dfn data-x="event-keydown" data-x-href="https://w3c.github.io/uievents/#event-type-keydown"><code>keydown</code></dfn> event</li>
     <li><dfn data-x="event-keypress" data-x-href="https://w3c.github.io/uievents/#event-type-keypress"><code>keypress</code></dfn> event</li>
     <li><dfn data-x="event-keyup" data-x-href="https://w3c.github.io/uievents/#event-type-keyup"><code>keyup</code></dfn> event</li>
    </ul>

    <p>The following features are defined in <cite>Touch Events</cite>: <ref>TOUCH</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/touch-events/#touch-interface"><code>Touch</code></dfn> interface</li>
     <li><dfn data-x-href="https://w3c.github.io/touch-events/#dfn-touch-point">Touch point</dfn> concept</li>
     <li><dfn data-x="event-touchend" data-x-href="https://w3c.github.io/touch-events/#event-touchend"><code>touchend</code></dfn> event</li>
    </ul>

    <p>The following features are defined in <cite>Pointer Events</cite>:
    <ref>POINTEREVENTS</ref></p>

    <ul class="brief">
     <li>The <dfn data-x-href="https://w3c.github.io/pointerevents/#pointerevent-interface"><code>PointerEvent</code></dfn> interface</li>
     <li>The <code>PointerEvent</code> interface's <dfn data-x-href="https://w3c.github.io/pointerevents/#dom-pointerevent-pointertype"><code>pointerType</code></dfn> attribute</li>
     <li><dfn data-x-href="https://w3c.github.io/pointerevents/#dfn-fire-a-pointer-event">fire a pointer event</dfn></li>
     <li><dfn data-x="event-pointerdown" data-x-href="https://w3c.github.io/pointerevents/#the-pointerdown-event"><code>pointerdown</code></dfn> event</li>
     <li><dfn data-x="event-pointerup" data-x-href="https://w3c.github.io/pointerevents/#the-pointerup-event"><code>pointerup</code></dfn> event</li>
     <li><dfn data-x="event-pointercancel" data-x-href="https://w3c.github.io/pointerevents/#the-pointercancel-event"><code>pointercancel</code></dfn> event</li>
    </ul>

    <p>The following events are defined in <cite>Clipboard API and events</cite>:
    <ref>CLIPBOARD-APIS</ref></p>

    <ul class="brief">
     <li><dfn data-x="event-copy" data-x-href="https://w3c.github.io/clipboard-apis/#clipboard-event-copy"><code>copy</code></dfn> event</li>
     <li><dfn data-x="event-cut" data-x-href="https://w3c.github.io/clipboard-apis/#clipboard-event-cut"><code>cut</code></dfn> event</li>
     <li><dfn data-x="event-paste" data-x-href="https://w3c.github.io/clipboard-apis/#clipboard-event-paste"><code>paste</code></dfn> event</li>
    </ul>

    <p>This specification sometimes uses the term <dfn data-x="">name</dfn> to refer to the event's
    <span data-x="concept-event-type">type</span>; as in, "an event named <code
    data-x="">click</code>" or "if the event name is <code data-x="">keypress</code>". The terms
    "name" and "type" for events are synonymous.</p>

    <p>The following features are defined in <cite>DOM Parsing and Serialization</cite>:
    <ref>DOMPARSING</ref></p>

    <ul class="brief">
     <li><dfn data-x="xml-serialization" data-x-href="https://w3c.github.io/DOM-Parsing/#dfn-xml-serialization"><code>XML serialization</code></dfn></li>
    </ul>

    <p>The following features are defined in <cite>Selection API</cite>: <ref>SELECTION</ref></p>

    <ul class="brief">
     <li><dfn data-x="document-selection" data-x-href="https://w3c.github.io/selection-api/#dfn-selection">selection</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/selection-api/#selection-interface"><code>Selection</code></dfn></li>
    </ul>

    <p class="note">User agents are encouraged to implement the features described in
    <cite>execCommand</cite>. <ref>EXECCOMMAND</ref></p>

    <p>The following features are defined in <cite>Fullscreen API</cite>: <ref>FULLSCREEN</ref></p>

    <ul class="brief">
     <li><dfn data-x="dom-element-requestFullscreen" data-x-href="https://fullscreen.spec.whatwg.org/#dom-element-requestfullscreen"><code>requestFullscreen()</code></dfn></li>
     <li><dfn data-x="event-fullscreenchange" data-x-href="https://fullscreen.spec.whatwg.org/#eventdef-document-fullscreenchange"><code>fullscreenchange</code></dfn></li>
     <li><dfn data-x-href="https://fullscreen.spec.whatwg.org/#run-the-fullscreen-steps">run the fullscreen steps</dfn></li>
     <li><dfn data-x-href="https://fullscreen.spec.whatwg.org/#fully-exit-fullscreen">fully exit fullscreen</dfn></li>
     <li><dfn data-x-href="https://fullscreen.spec.whatwg.org/#fullscreen-element">fullscreen element</dfn></li>
     <li><dfn data-x-href="https://fullscreen.spec.whatwg.org/#fullscreen-flag">fullscreen flag</dfn></li>
    </ul>

    <p><cite>High Resolution Time</cite> provides the following features: <ref>HRT</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/hr-time/#dfn-current-high-resolution-time">current high resolution time</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/hr-time/#dfn-relative-high-resolution-time">relative high resolution time</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/hr-time/#dfn-unsafe-shared-current-time">unsafe shared current time</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/hr-time/#dfn-shared-monotonic-clock">shared monotonic clock</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/hr-time/#dfn-unsafe-moment">unsafe moment</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/hr-time/#dfn-duration-from">duration from</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/hr-time/#dfn-coarsen-time">coarsen time</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/hr-time/#dfn-current-wall-time">current wall time</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/hr-time/#dfn-unix-epoch">Unix epoch</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/hr-time/#dom-domhighrestimestamp"><code>DOMHighResTimeStamp</code></dfn></li>
    </ul>
   </dd>



   <dt>File API</dt>

   <dd>
    <p>This specification uses the following features defined in <cite>File API</cite>:
    <ref>FILEAPI</ref></p>

    <ul class="brief">
     <li>The <dfn data-x-href="https://w3c.github.io/FileAPI/#dfn-Blob"><code>Blob</code></dfn> interface and its
             <dfn data-x="dom-Blob-type" data-x-href="https://w3c.github.io/FileAPI/#dfn-type"><code>type</code></dfn> attribute</li>
     <li>The <dfn data-x-href="https://w3c.github.io/FileAPI/#dfn-file"><code>File</code></dfn> interface and its
             <dfn data-x="dom-File-name" data-x-href="https://w3c.github.io/FileAPI/#dfn-name"><code>name</code></dfn> and
             <dfn data-x="dom-File-lastModified" data-x-href="https://w3c.github.io/FileAPI/#dfn-lastModified"><code>lastModified</code></dfn> attributes</li>
     <li>The <dfn data-x-href="https://w3c.github.io/FileAPI/#filelist-section"><code>FileList</code></dfn> interface</li>
     <li>The concept of a <code>Blob</code>'s <dfn data-x-href="https://w3c.github.io/FileAPI/#snapshot-state">snapshot state</dfn></li>
     <li>The concept of <dfn data-x="file-error-read">read errors</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/FileAPI/#BlobURLStore">Blob URL Store</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/FileAPI/#blob-url-entry">blob URL entry</dfn> and its
         <dfn data-x="blob-url-entry-environment" data-x-href="https://w3c.github.io/FileAPI/#blob-url-entry-environment">environment</dfn></li>
     <li>The <dfn data-x="blob-url-obtain-object" data-x-href="https://w3c.github.io/FileAPI/#blob-url-obtain-object">obtain a blob object</dfn> algorithm</li>
    </ul>
   </dd>



   <dt>Indexed Database API</dt>

   <dd>
    <p>The following terms are defined in <cite>Indexed Database API</cite>:
    <ref>INDEXEDDB</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/IndexedDB/#cleanup-indexed-database-transactions">cleanup Indexed Database transactions</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/IndexedDB/#idbversionchangeevent"><code>IDBVersionChangeEvent</code></dfn></li>
    </ul>
   </dd>


   <dt>Media Source Extensions</dt>

   <dd>
    <p>The following terms are defined in <cite>Media Source Extensions</cite>:
    <ref>MEDIASOURCE</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/media-source/#idl-def-mediasource"><code>MediaSource</code></dfn> interface</li>
     <li><dfn data-x-href="https://w3c.github.io/media-source/#mediasource-detach">detaching from a media element</dfn></li>
    </ul>
   </dd>

   <dt>Media Capture and Streams</dt>

   <dd>
    <p>The following terms are defined in <cite>Media Capture and Streams</cite>:
    <ref>MEDIASTREAM</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/mediacapture-main/getusermedia.html#idl-def-mediastream"><code>MediaStream</code></dfn> interface</li>
     <li><dfn data-x-href="https://w3c.github.io/mediacapture-main/getusermedia.html#mediastreamtrack"><code>MediaStreamTrack</code></dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/mediacapture-main/getusermedia.html#idl-def-MediaStreamTrackState.live">live state</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/mediacapture-main/getusermedia.html#dom-mediadevices-getusermedia"><code>getUserMedia()</code></dfn></li>
    </ul>
   </dd>


   <dt>Reporting</dt>

   <dd>
    <p>The following terms are defined in <cite>Reporting</cite>: <ref>REPORTING</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/reporting/#queue-report">Queue a report</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/reporting/#report-type">report type</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/reporting/#visible-to-reportingobservers">visible to <code>ReportingObserver</code>s</dfn></li>
    </ul>
   </dd>


   <dt>XMLHttpRequest</dt>

   <dd>
    <p>The following features and terms are defined in <cite>XMLHttpRequest</cite>:
    <ref>XHR</ref></p>

    <ul class="brief">
     <li>The <dfn data-x-href="https://xhr.spec.whatwg.org/#xmlhttprequest"><code>XMLHttpRequest</code></dfn> interface, and its
             <dfn data-x="dom-XMLHttpRequest-responseXML" data-x-href="https://xhr.spec.whatwg.org/#dom-xmlhttprequest-responsexml"><code>responseXML</code></dfn> attribute</li>
     <li>The <dfn data-x-href="https://xhr.spec.whatwg.org/#interface-progressevent"><code>ProgressEvent</code></dfn> interface, and its
             <dfn data-x="dom-ProgressEvent-lengthComputable" data-x-href="https://xhr.spec.whatwg.org/#dom-progressevent-lengthcomputable"><code>lengthComputable</code></dfn>,
             <dfn data-x="dom-ProgressEvent-loaded" data-x-href="https://xhr.spec.whatwg.org/#dom-progressevent-loaded"><code>loaded</code></dfn>, and
             <dfn data-x="dom-ProgressEvent-total" data-x-href="https://xhr.spec.whatwg.org/#dom-progressevent-total"><code>total</code></dfn> attributes</li>
     <li>The <dfn data-x-href="https://xhr.spec.whatwg.org/#formdata"><code>FormData</code></dfn> interface, and its associated
             <dfn data-x="FormData entry list" data-x-href="https://xhr.spec.whatwg.org/#concept-formdata-entry-list">entry list</dfn></li>
    </ul>
   </dd>

   <dt>Battery Status</dt>

   <dd>
    <p>The following features are defined in <cite>Battery Status API</cite>: <ref>BATTERY</ref></p>

    <ul class="brief">
     <li><dfn data-x="dom-navigator-getBattery" data-x-href="https://w3c.github.io/battery/#widl-Navigator-getBattery-Promise-BatteryManager"><code>getBattery()</code></dfn> method</li>
    </ul>
   </dd>


   <dt>Media Queries</dt>

   <dd>
    <p>Implementations must support <cite>Media Queries</cite>. The <dfn
    data-x-href="https://drafts.csswg.org/mediaqueries/#typedef-media-condition">&lt;media-condition></dfn>
    feature is defined therein. <ref>MQ</ref></p>
   </dd>


   <dt>CSS modules</dt>

   <dd>
    <p>While support for CSS as a whole is not required of implementations of this specification
    (though it is encouraged, at least for web browsers), some features are defined in terms of
    specific CSS requirements.</p>

    <p>When this specification requires that something be <dfn data-x="parse something according to
    a CSS grammar" data-x-href="https://drafts.csswg.org/css-syntax/#parse-grammar">parsed according
    to a particular CSS grammar</dfn>, the relevant algorithm in <cite>CSS Syntax</cite> must be
    followed, including error handling rules. <ref>CSSSYNTAX</ref></p>

    <p class="example">For example, user agents are required to close all open constructs upon
    finding the end of a style sheet unexpectedly. Thus, when parsing the string "<code
    data-x="">rgb(0,0,0</code>" (with a missing close-parenthesis) for a color value, the close
    parenthesis is implied by this error handling rule, and a value is obtained (the color 'black').
    However, the similar construct "<code data-x="">rgb(0,0,</code>" (with both a missing
    parenthesis and a missing "blue" value) cannot be parsed, as closing the open construct does not
    result in a viable value.</p>

    <p>The following terms and features are defined in <cite>Cascading Style Sheets</cite>
    (<cite>CSS</cite>): <ref>CSS</ref></p>

    <ul class="brief">
      <li><dfn data-x-href="https://drafts.csswg.org/css2/#viewport">viewport</dfn></li>
      <li><dfn data-x-href="https://drafts.csswg.org/css2/#line-box">line box</dfn></li>
      <li><dfn data-x-href="https://drafts.csswg.org/css2/#out-of-flow">out-of-flow</dfn></li>
      <li><dfn data-x-href="https://drafts.csswg.org/css2/#in-flow">in-flow</dfn></li>
      <li><dfn data-x-href="https://drafts.csswg.org/css2/#collapsing-margins">collapsing margins</dfn></li>
      <li><dfn data-x-href="https://drafts.csswg.org/css2/#containing-block-details">containing block</dfn></li>
      <li><dfn data-x-href="https://drafts.csswg.org/css2/#inline-box">inline box</dfn></li>
      <li><dfn data-x-href="https://drafts.csswg.org/css2/#block-boxes%E2%91%A0">block box</dfn></li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css2/#propdef-top">'top'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css2/#propdef-bottom">'bottom'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css2/#propdef-left">'left'</dfn>, and
              <dfn data-x-href="https://drafts.csswg.org/css2/#propdef-right">'right'</dfn> properties</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css2/#float-position">'float'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css2/#flow-control">'clear'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css2/#the-width-property">'width'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css2/#the-height-property">'height'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css2/#min-max-widths">'min-width'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css2/#min-max-heights">'min-height'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css2/#min-max-widths">'max-width'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css2/#min-max-heights">'max-height'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css2/#propdef-line-height">'line-height'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css2/#propdef-vertical-align">'vertical-align'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css2/#content%E2%91%A0">'content'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css2/#value-def-inline-block">'inline-block'</dfn> value of the <span>'display'</span> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css2/#propdef-visibility">'visibility'</dfn> property</li>
    </ul>

    <p>The basic version of the <dfn
    data-x-href="https://drafts.csswg.org/css2/#display-prop">'display'</dfn> property
    is defined in <cite>CSS</cite>, and the property is extended by other CSS modules.
    <ref>CSS</ref> <ref>CSSRUBY</ref> <ref>CSSTABLE</ref></p>

    <p>The following terms and features are defined in <cite>CSS Box Model</cite>:
    <ref>CSSBOX</ref></p>

    <ul>
      <li><dfn data-x-href="https://drafts.csswg.org/css-box/#content-area">content area</dfn></li>
      <li><dfn data-x-href="https://drafts.csswg.org/css-box/#content-box">content box</dfn></li>
      <li><dfn data-x-href="https://drafts.csswg.org/css-box/#border-box">border box</dfn></li>
      <li><dfn data-x-href="https://drafts.csswg.org/css-box/#margin-box">margin box</dfn></li>
      <li><dfn data-x-href="https://drafts.csswg.org/css-box/#border-edge">border edge</dfn></li>
      <li><dfn data-x-href="https://drafts.csswg.org/css-box/#margin-edge">margin edge</dfn></li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-box/#propdef-margin-top">'margin-top'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-box/#propdef-margin-bottom">'margin-bottom'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-box/#propdef-margin-left">'margin-left'</dfn>, and
              <dfn data-x-href="https://drafts.csswg.org/css-box/#propdef-margin-right">'margin-right'</dfn> properties</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-box/#propdef-padding-top">'padding-top'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-box/#propdef-padding-bottom">'padding-bottom'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-box/#propdef-padding-left">'padding-left'</dfn>, and
              <dfn data-x-href="https://drafts.csswg.org/css-box/#propdef-padding-right">'padding-right'</dfn> properties</li>
    </ul>

    <p>The following features are defined in <cite>CSS Logical Properties</cite>:
    <ref>CSSLOGICAL</ref></p>

    <ul class="brief">
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-margin-block">'margin-block'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-margin-block-start">'margin-block-start'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-margin-block-end">'margin-block-end'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-margin-inline">'margin-inline'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-margin-inline-start">'margin-inline-start'</dfn>, and
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-margin-inline-end">'margin-inline-end'</dfn> properties</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-padding-block">'padding-block'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-padding-block-start">'padding-block-start'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-padding-block-end">'padding-block-end'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-padding-inline">'padding-inline'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-padding-inline-start">'padding-inline-start'</dfn>, and
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-padding-inline-end">'padding-inline-end'</dfn> properties</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-block-width">'border-block-width'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-block-start-width">'border-block-start-width'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-block-end-width">'border-block-end-width'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-inline-width">'border-inline-width'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-inline-start-width">'border-inline-start-width'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-inline-end-width">'border-inline-end-width'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-block-style">'border-block-style'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-block-start-style">'border-block-start-style'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-block-end-style">'border-block-end-style'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-inline-style">'border-inline-style'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-inline-start-style">'border-inline-start-style'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-inline-end-style">'border-inline-end-style'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-block-start-color">'border-block-start-color'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-block-end-color">'border-block-end-color'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-inline-start-color">'border-inline-start-color'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-inline-end-color">'border-inline-end-color'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-start-start-radius">'border-start-start-radius'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-start-end-radius">'border-start-end-radius'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-end-start-radius">'border-end-start-radius'</dfn>, and
              <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-border-end-end-radius">'border-end-end-radius'</dfn>
              properties</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-block-size">'block-size'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-inline-size">'inline-size'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-inset-block-start">'inset-block-start'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-logical/#propdef-inset-block-end">'inset-block-end'</dfn> property</li>
    </ul>

    <p>The following terms and features are defined in <cite>CSS Color</cite>:
    <ref>CSSCOLOR</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.csswg.org/css-color/#named-color">named color</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-color/#typedef-color">&lt;color></dfn></li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-color/#the-color-property">'color'</dfn> property</li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-color/#valdef-color-currentcolor" id="currentcolor">'currentcolor'</dfn> value</li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-color/#opaque-black">opaque black</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-color/#transparent-black">transparent black</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-color/#valdef-color-srgb">'srgb'</dfn> color space</li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-color/#valdef-color-display-p3">'display-p3'</dfn> color space</li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-color-5/#valdef-color-profile-rendering-intent-relative-colorimetric">'relative-colorimetric'</dfn> rendering intent</li>
     <li><dfn id="parsed-as-a-css-color-value" data-x-href="https://drafts.csswg.org/css-color/#parse-a-css-color-value">parse a CSS &lt;color> value</dfn></li>
     <li><!--en-GB--><dfn id="serialisation-of-a-color" data-x-href="https://drafts.csswg.org/css-color/#serializing-color-values">serialize a CSS &lt;color> value</dfn> including <dfn data-x-href="https://drafts.csswg.org/css-color/#color-serialization-html-compatible-serialization-is-requested">HTML-compatible serialization is requested</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-color/#color-conversion">Converting Colors</dfn></li>
     <li><dfn id="color-function" data-x-href="https://drafts.csswg.org/css-color/#color-function">'color()'</dfn></li>
    </ul>

    <p>The following terms are defined in <cite>CSS Images</cite>: <ref>CSSIMAGES</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.csswg.org/css-images/#default-object-size">default object size</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-images/#concrete-object-size">concrete object size</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-images/#natural-dimensions">natural dimensions</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-images/#natural-height">natural height</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-images/#natural-width">natural width</dfn></li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-images-3/#the-image-orientation">'image-orientation'</dfn> property</li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-images-4/#funcdef-conic-gradient">'conic-gradient'</dfn></li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-images/#the-object-fit">'object-fit'</dfn> property</li>
    </ul>

    <p>The term <dfn data-x-href="https://drafts.csswg.org/css-images-4/#paint-source">paint
    source</dfn> is used as defined in <cite>CSS Images Level 4</cite>
    to define the interaction of certain HTML elements with the CSS 'element()' function.
    <ref>CSSIMAGES4</ref></p>

    <p>The following features are defined in <cite>CSS Backgrounds and Borders</cite>:
    <ref>CSSBG</ref></p>

    <ul class="brief">
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-background-color">'background-color'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-background-image">'background-image'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-background-repeat">'background-repeat'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-background-attachment">'background-attachment'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-background-position">'background-position'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-background-clip">'background-clip'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-background-origin">'background-origin'</dfn>, and
              <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-background-size">'background-size'</dfn>
              properties</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-radius">'border-radius'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-top-left-radius">'border-top-left-radius'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-top-right-radius">'border-top-right-radius'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-bottom-right-radius">'border-bottom-right-radius'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-bottom-left-radius">'border-bottom-left-radius'</dfn>
              properties</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-image-source">'border-image-source'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-image-slice">'border-image-slice'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-image-width">'border-image-width'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-image-outset">'border-image-outset'</dfn>, and
              <dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-image-repeat">'border-image-repeat'</dfn>
              properties</li>
    </ul>

    <p><cite>CSS Backgrounds and Borders</cite> also defines the following border properties:
    <ref>CSSBG</ref></p>

    <table>
     <caption>Border properties
     <tbody>
      <tr>
       <td>
       <th>Top
       <th>Bottom
       <th>Left
       <th>Right
      <tr>
       <th>Width
       <td><dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-top-width">'border-top-width'</dfn>
       <td><dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-bottom-width">'border-bottom-width'</dfn>
       <td><dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-left-width">'border-left-width'</dfn>
       <td><dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-right-width">'border-right-width'</dfn>
      <tr>
       <th>Style
       <td><dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-top-style">'border-top-style'</dfn>
       <td><dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-bottom-style">'border-bottom-style'</dfn>
       <td><dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-left-style">'border-left-style'</dfn>
       <td><dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-right-style">'border-right-style'</dfn>
      <tr>
       <th>Color
       <td><dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-top-color">'border-top-color'</dfn>
       <td><dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-bottom-color">'border-bottom-color'</dfn>
       <td><dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-left-color">'border-left-color'</dfn>
       <td><dfn data-x-href="https://drafts.csswg.org/css-backgrounds/#propdef-border-right-color">'border-right-color'</dfn>
    </table>

    <p>The following features are defined in <cite>CSS Box Alignment</cite>: <ref>CSSALIGN</ref></p>

    <ul class="brief">
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-align/#propdef-align-content">'align-content'</dfn> property</li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-align/#propdef-align-items">'align-items'</dfn> property</li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-align/#propdef-align-self">'align-self'</dfn> property</li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-align/#propdef-justify-self">'justify-self'</dfn> property</li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-align/#propdef-propdef-justify-content">'justify-content'</dfn> property</li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-align/#propdef-propdef-justify-items">'justify-items'</dfn> property</li>
    </ul>

    <p>The following terms and features are defined in <cite>CSS Display</cite>:
    <ref>CSSDISPLAY</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.csswg.org/css-display/#outer-display-type">outer display type</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-display/#inner-display-type">inner display type</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-display/#block-level">block-level</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-display/#block-container">block container</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-display/#formatting-context">formatting context</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-display/#block-formatting-context">block formatting context</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-display/#inline-formatting-context">inline formatting context</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-display/#replaced-element">replaced element</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-display/#css-box">CSS box</dfn></li>
    </ul>

    <p>The following features are defined in <cite>CSS Flexible Box Layout</cite>:
    <ref>CSSFLEXBOX</ref></p>

    <ul class="brief">
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-flexbox/#propdef-flex-direction">'flex-direction'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-flexbox/#propdef-flex-wrap">'flex-wrap'</dfn> property</li>
    </ul>

    <p>The following terms and features are defined in <cite>CSS Fonts</cite>:
    <ref>CSSFONTS</ref></p>

    <ul class="brief">
      <li><dfn data-x-href="https://drafts.csswg.org/css-fonts/#first-available-font">first available font</dfn></li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#font-family-prop">'font-family'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#font-weight-prop">'font-weight'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#font-size-prop">'font-size'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#font-prop">'font'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#propdef-font-kerning">'font-kerning'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#propdef-font-stretch">'font-stretch'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#propdef-font-variant-caps">'font-variant-caps'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#valdef-font-variant-caps-small-caps">'small-caps'</dfn> value</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#valdef-font-variant-caps-all-small-caps">'all-small-caps'</dfn> value</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#valdef-font-variant-caps-petite-caps">'petite-caps'</dfn> value</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#valdef-font-variant-caps-all-petite-caps">'all-petite-caps'</dfn> value</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#valdef-font-variant-caps-unicase">'unicase'</dfn> value</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#valdef-font-variant-caps-titling-caps">'titling-caps'</dfn> value</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#valdef-font-stretch-ultra-condensed">'ultra-condensed'</dfn> value</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#valdef-font-stretch-extra-condensed">'extra-condensed'</dfn> value</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#valdef-font-stretch-condensed">'condensed'</dfn> value</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#valdef-font-stretch-semi-condensed">'semi-condensed'</dfn> value</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#valdef-font-stretch-semi-expanded">'semi-expanded'</dfn> value</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#valdef-font-stretch-expanded">'expanded'</dfn> value</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#valdef-font-stretch-extra-expanded">'extra-expanded'</dfn> value</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-fonts/#valdef-font-stretch-ultra-expanded">'ultra-expanded'</dfn> value</li>
    </ul>

    <p>The following features are defined in <cite>CSS Grid Layout</cite>: <ref>CSSGRID</ref></p>

    <ul class="brief">
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-grid/#propdef-grid-auto-columns">'grid-auto-columns'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-grid/#propdef-grid-auto-flow">'grid-auto-flow'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-grid/#propdef-grid-auto-rows">'grid-auto-rows'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-grid/#propdef-grid-column-gap">'grid-column-gap'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-grid/#propdef-grid-row-gap">'grid-row-gap'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-grid/#propdef-grid-template-areas">'grid-template-areas'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-grid/#propdef-grid-template-columns">'grid-template-columns'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-grid/#propdef-grid-template-rows">'grid-template-rows'</dfn> property</li>
    </ul>

    <p>The following terms are defined in <cite>CSS Inline Layout</cite>: <ref>CSSINLINE</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.csswg.org/css-inline/#alphabetic-baseline">alphabetic baseline</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-inline/#ascent-metric">ascent metric</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-inline/#descent-metric">descent metric</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-inline/#em-over-baseline">em-over baseline</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-inline/#em-under-baseline">em-under baseline</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-inline/#hanging-baseline">hanging baseline</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-inline/#ideographic-under-baseline">ideographic-under baseline</dfn></li>
    </ul>

    <p>The following terms and features are defined in <cite>CSS Box Sizing</cite>:
    <ref>CSSSIZING</ref></p>

    <ul class="brief">
      <li><dfn data-x-href="https://drafts.csswg.org/css-sizing/#fit-content-inline-size">fit-content inline size</dfn></li>
      <li><dfn data-x-href="https://drafts.csswg.org/css-sizing-4/#aspect-ratio">'aspect-ratio'</dfn> property</li>
      <li><dfn data-x-href="https://drafts.csswg.org/css-sizing/#intrinsic-size">intrinsic size</dfn></li>
    </ul>

    <p>The following features are defined in <cite>CSS Lists and Counters</cite>.
    <ref>CSSLISTS</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.csswg.org/css-lists/#list-item" data-x="css-list-item">list item</dfn></li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-lists/#propdef-counter-reset">'counter-reset'</dfn> property</li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-lists/#propdef-counter-set">'counter-set'</dfn> property</li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-lists/#propdef-list-style-type">'list-style-type'</dfn> property</li>
    </ul>

    <p>The following features are defined in <cite>CSS Overflow</cite>. <ref>CSSOVERFLOW</ref></p>

    <ul class="brief">
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-overflow/#propdef-overflow">'overflow'</dfn> property and its <dfn data-x-href="https://drafts.csswg.org/css-overflow/#valdef-overflow-hidden">'hidden'</dfn> value</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-overflow/#propdef-text-overflow">'text-overflow'</dfn> property</li>
      <li>The term <dfn data-x-href="https://drafts.csswg.org/css-overflow/#scroll-container">scroll container</dfn>
    </ul>

    <p>The following terms and features are defined in <cite>CSS Positioned Layout</cite>:
    <ref>CSSPOSITION</ref></p>

    <ul class="brief">
      <li><dfn data-x-href="https://drafts.csswg.org/css-position/#absolute-position">absolutely-positioned</dfn></li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-position/#position-property">'position'</dfn> property and its
              <dfn data-x-href="https://drafts.csswg.org/css-position/#valdef-position-static">'static'</dfn> value</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-position-4/#document-top-layer">top layer</dfn>
              (an <span data-x="set">ordered set</span>)</li>
      <li><dfn data-x-href="https://drafts.csswg.org/css-position-4/#add-an-element-to-the-top-layer">add an element to the top layer</dfn></li>
      <li><dfn data-x-href="https://drafts.csswg.org/css-position-4/#request-an-element-to-be-removed-from-the-top-layer">request an element to be removed from the top layer</dfn></li>
      <li><dfn data-x-href="https://drafts.csswg.org/css-position-4/#remove-an-element-from-the-top-layer-immediately">remove an element from the top layer immediately</dfn></li>
      <li><dfn data-x-href="https://drafts.csswg.org/css-position-4/#process-top-layer-removals">process top layer removals</dfn></li>
    </ul>

    <p>The following features are defined in <cite>CSS Multi-column Layout</cite>.
    <ref>CSSMULTICOL</ref></p>

    <ul class="brief">
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-multicol/#propdef-column-count">'column-count'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-multicol/#propdef-column-fill">'column-fill'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-multicol/#propdef-column-gap">'column-gap'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-multicol/#propdef-column-rule">'column-rule'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-multicol/#propdef-column-width">'column-width'</dfn> property</li>
    </ul>

    <p>The <dfn
    data-x-href="https://drafts.csswg.org/css-ruby/#valdef-display-ruby-base">'ruby-base'</dfn>
    value of the <span>'display'</span> property is defined in <cite>CSS Ruby Layout</cite>.
    <ref>CSSRUBY</ref></p>

    <p>The following features are defined in <cite>CSS Table</cite>: <ref>CSSTABLE</ref></p>

    <ul class="brief">
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-tables/#propdef-border-spacing">'border-spacing'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-tables/#border-collapse-property">'border-collapse'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-tables/#table-cell">'table-cell'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-tables/#table-row">'table-row'</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-tables/#table-caption">'table-caption'</dfn>, and
              <dfn data-x-href="https://drafts.csswg.org/css-tables/#table">'table'</dfn> values of the <span>'display'</span> property</li>
    </ul>

    <p>The following features are defined in <cite>CSS Text</cite>: <ref>CSSTEXT</ref></p>

    <ul class="brief">
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-text-4/#content-language">content language</dfn> concept</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-text/#text-transform-property">'text-transform'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-text/#white-space-property">'white-space'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-text/#text-align-property">'text-align'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-text/#letter-spacing-property">'letter-spacing'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-text/#propdef-word-spacing">'word-spacing'</dfn> property</li>
    </ul>

    <p>The following features are defined in <cite>CSS Writing Modes</cite>: <ref>CSSWM</ref></p>

    <ul class="brief">
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-writing-modes/#direction">'direction'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-writing-modes/#unicode-bidi">'unicode-bidi'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-writing-modes/#propdef-writing-mode">'writing-mode'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-writing-modes/#block-flow-direction">block flow direction</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-writing-modes/#block-axis">block axis</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-writing-modes/#inline-axis">inline axis</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-writing-modes/#block-size">block size</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-writing-modes/#inline-size">inline size</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-writing-modes/#block-start">block-start</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-writing-modes/#block-end">block-end</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-writing-modes/#inline-start">inline-start</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-writing-modes/#inline-end">inline-end</dfn>,
              <dfn data-x-href="https://drafts.csswg.org/css-writing-modes/#line-left">line-left</dfn>, and
              <dfn data-x-href="https://drafts.csswg.org/css-writing-modes/#line-right">line-right</dfn> concepts</li>
    </ul>

    <p>The following features are defined in <cite>CSS Basic User Interface</cite>:
    <ref>CSSUI</ref></p>

    <ul class="brief">
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-ui/#outline">'outline'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-ui/#cursor">'cursor'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-ui/#appearance-switching">'appearance'</dfn> property, its
              <dfn data-x-href="https://drafts.csswg.org/css-ui/#typedef-appearance-compat-auto">&lt;compat-auto></dfn> non-terminal value type, its
              <dfn data-x-href="https://drafts.csswg.org/css-ui/#valdef-appearance-textfield">'textfield'</dfn> value, and its
              <dfn data-x-href="https://drafts.csswg.org/css-ui/#valdef-appearance-menulist-button">'menulist-button'</dfn> value.</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-ui/#field-sizing">'field-sizing'</dfn> property, and its
              <dfn data-x-href="https://drafts.csswg.org/css-ui/#valdef-field-sizing-content" data-x="field-sizing-content">'content'</dfn> value.</li>
      <li>The concept <dfn data-x-href="https://drafts.csswg.org/css-ui/#widget">widget</dfn></li>
      <li>The concept <dfn data-x-href="https://drafts.csswg.org/css-ui/#native-appearance">native appearance</dfn></li>
      <li>The concept <dfn data-x-href="https://drafts.csswg.org/css-ui/#primitive-appearance">primitive appearance</dfn></li>
      <li>The concept <dfn data-x-href="https://drafts.csswg.org/css-ui/#element-with-default-preferred-size">element with default preferred size</dfn></li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-ui/#non-devolvable">non-devolvable widget</dfn> and
              <dfn data-x-href="https://drafts.csswg.org/css-ui/#devolvable">devolvable widget</dfn> classification, and the related
              <dfn data-x-href="https://drafts.csswg.org/css-ui/#devolved">devolved widget</dfn> state.</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-ui-4/#pointer-events-control">'pointer-events'</dfn> property</li>
      <li>The <dfn data-x-href="https://drafts.csswg.org/css-ui-4/#content-selection">'user-select'</dfn> property</li>
    </ul>

    <p>The algorithm to <dfn
    data-x-href="https://drafts.csswg.org/web-animations-1/#update-animations-and-send-events">update
    animations and send events</dfn> is defined in <cite>Web Animations</cite>.
    <ref>WEBANIMATIONS</ref></p>

    <p>Implementations that support scripting must support the CSS Object Model. The following
    features and terms are defined in the CSSOM specifications: <ref>CSSOM</ref>
    <ref>CSSOMVIEW</ref>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.csswg.org/cssom-view/#the-screen-interface"><code>Screen</code></dfn> interface</li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom/#the-linkstyle-interface"><code>LinkStyle</code></dfn> interface</li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom/#the-cssstyledeclaration-interface"><code>CSSStyleDeclaration</code></dfn> interface</li>
     <li><dfn data-x="dom-style" data-x-href="https://drafts.csswg.org/cssom/#dom-elementcssinlinestyle-style"><code>style</code></dfn> IDL attribute</li>
     <li><dfn data-x="dom-CSSStyleDeclaration-cssText" data-x-href="https://drafts.csswg.org/cssom/#dom-cssstyledeclaration-csstext"><code>cssText</code></dfn> attribute of <code>CSSStyleDeclaration</code></li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom/#the-stylesheet-interface"><code>StyleSheet</code></dfn> interface</li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom/#the-cssstylesheet-interface"><code>CSSStyleSheet</code></dfn> interface</li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom/#create-a-css-style-sheet">create a CSS style sheet</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom/#remove-a-css-style-sheet">remove a CSS style sheet</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom/#associated-css-style-sheet">associated CSS style sheet</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom/#create-a-constructed-cssstylesheet">create a constructed <code>CSSStyleSheet</code></dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom/#synchronously-replace-the-rules-of-a-cssstylesheet">synchronously replace the rules of a <code>CSSStyleSheet</code></dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom/#disable-a-css-style-sheet">disable a CSS style sheet</dfn></li>
     <li>
      <dfn data-x="CSS style sheet"
      data-x-href="https://drafts.csswg.org/cssom/#css-style-sheet">CSS style sheets</dfn> and their
      properties:
      <ul class="brief">
       <li><dfn data-x="concept-css-style-sheet-type" data-x-href="https://drafts.csswg.org/cssom/#concept-css-style-sheet-type">type</dfn></li>
       <li><dfn data-x="concept-css-style-sheet-location" data-x-href="https://drafts.csswg.org/cssom/#concept-css-style-sheet-location">location</dfn></li>
       <li><dfn data-x="concept-css-style-sheet-parent-CSS-style-sheet" data-x-href="https://drafts.csswg.org/cssom/#concept-css-style-sheet-parent-css-style-sheet">parent CSS style sheet</dfn></li>
       <li><dfn data-x="concept-css-style-sheet-owner-node" data-x-href="https://drafts.csswg.org/cssom/#concept-css-style-sheet-owner-node">owner node</dfn></li>
       <li><dfn data-x="concept-css-style-sheet-owner-CSS-rule" data-x-href="https://drafts.csswg.org/cssom/#concept-css-style-sheet-owner-css-rule">owner CSS rule</dfn></li>
       <li><dfn data-x="concept-css-style-sheet-media" data-x-href="https://drafts.csswg.org/cssom/#concept-css-style-sheet-media">media</dfn></li>
       <li><dfn data-x="concept-css-style-sheet-title" data-x-href="https://drafts.csswg.org/cssom/#concept-css-style-sheet-title">title</dfn></li>
       <li><dfn data-x="concept-css-style-sheet-alternate-flag" data-x-href="https://drafts.csswg.org/cssom/#concept-css-style-sheet-alternate-flag">alternate flag</dfn></li>
       <li><dfn data-x="concept-css-style-sheet-disabled-flag" data-x-href="https://drafts.csswg.org/cssom/#concept-css-style-sheet-disabled-flag">disabled flag</dfn></li>
       <li><dfn data-x="concept-css-style-sheet-CSS-rules" data-x-href="https://drafts.csswg.org/cssom/#concept-css-style-sheet-css-rules">CSS rules</dfn></li>
       <li><dfn data-x="concept-css-style-sheet-origin-clean-flag" data-x-href="https://drafts.csswg.org/cssom/#concept-css-style-sheet-origin-clean-flag">origin-clean flag</dfn></li>
      </ul>
     </li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom/#css-style-sheet-set">CSS style sheet set</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom/#css-style-sheet-set-name">CSS style sheet set name</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom/#preferred-css-style-sheet-set-name">preferred CSS style sheet set name</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom/#change-the-preferred-css-style-sheet-set-name">change the preferred CSS style sheet set name</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom/#serialize-a-css-value">Serializing a CSS value</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom-view/#document-run-the-resize-steps">run the resize steps</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom-view/#document-run-the-scroll-steps">run the scroll steps</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom-view/#evaluate-media-queries-and-report-changes">evaluate media queries and report changes</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom-view/#scroll-a-target-into-view">Scroll a target into view</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom-view/#scroll-to-the-beginning-of-the-document">Scroll to the beginning of the document</dfn></li>
     <li>The <dfn data-x="event-resize" data-x-href="https://drafts.csswg.org/cssom-view/#eventdef-window-resize"><code>resize</code></dfn> event</li>
     <li>The <dfn data-x="event-scroll" data-x-href="https://drafts.csswg.org/cssom-view/#eventdef-document-scroll"><code>scroll</code></dfn> event</li>
     <li>The <dfn data-x="event-scrollend" data-x-href="https://drafts.csswg.org/cssom-view/#eventdef-document-scrollend"><code>scrollend</code></dfn> event</li>
     <li><dfn data-x-href="https://drafts.csswg.org/cssom-view/#set-up-browsing-context-features">set up browsing context features</dfn></li>
     <li>The <dfn data-x="mouseevent-clientx" data-x-href="https://drafts.csswg.org/cssom-view/#dom-mouseevent-clientx">clientX</dfn> and <dfn data-x="mouseevent-clienty" data-x-href="https://drafts.csswg.org/cssom-view/#dom-mouseevent-clienty">clientY</dfn> extension attributes of the <span>MouseEvent</span> interface</li>
    </ul>

    <p>The following features and terms are defined in <cite>CSS Syntax</cite>:
    <ref>CSSSYNTAX</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.csswg.org/css-syntax/#conform-classes">conformant style sheet</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-syntax/#parse-a-list-of-component-values">parse a list of component values</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-syntax/#parse-a-comma-separated-list-of-component-values">parse a comma-separated list of component values</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-syntax/#component-value">component value</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-syntax/#environment-encoding">environment encoding</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-syntax/#typedef-whitespace-token">&lt;whitespace-token></dfn></li>
    </ul>

    <p>The following terms are defined in <cite>Selectors</cite>: <ref>SELECTORS</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.csswg.org/selectors/#type-selector">type selector</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/selectors/#attribute-selector">attribute selector</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/selectors/#pseudo-class">pseudo-class</dfn></li>
     <li><dfn
     data-x-href="https://drafts.csswg.org/selectors/#the-focus-visible-pseudo"><code>:focus-visible</code></dfn>
     pseudo-class</li>
     <li><dfn data-x-href="https://drafts.csswg.org/selectors/#indicate-focus">indicate focus</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/selectors/#pseudo-element">pseudo-element</dfn></li>
    </ul>

    <p>The following features are defined in <cite>CSS Values and Units</cite>:
    <ref>CSSVALUES</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.csswg.org/css-values/#lengths">&lt;length></dfn></li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-values/#em">'em'</dfn> unit</li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-values/#ex">'ex'</dfn> unit</li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-values/#vw">'vw'</dfn> unit</li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-values/#in">'in'</dfn> unit</li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-values/#px">'px'</dfn> unit</li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-values/#pt">'pt'</dfn> unit</li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-values/#funcdef-attr">'attr()'</dfn> function</li>
     <li>The <dfn data-x-href="https://drafts.csswg.org/css-values/#math-function">math functions</dfn></li>
    </ul>

    <p>The following features are defined in <cite>CSS View Transitions</cite>:
    <ref>CSSVIEWTRANSITIONS</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.csswg.org/css-view-transitions/#perform-pending-transition-operations">perform pending transition operations</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-view-transitions/#document-rendering-suppression-for-view-transitions">rendering suppression for view transitions</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-view-transitions/#activate-view-transition">activate view transition</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-view-transitions/#viewtransition"><code>ViewTransition</code></dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-view-transitions/#view-transition-page-visibility-change-steps">view transition page visibility change steps</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-view-transitions-2/#resolve-inbound-cross-document-view-transition">resolving inbound cross-document view-transition</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-view-transitions-2/#setup-cross-document-view-transition">setting up a cross-document view-transition</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-view-transitions-2/#can-navigation-trigger-a-cross-document-view-transition">can navigation trigger a cross-document view-transition?</dfn>
    </ul>

    <p>The term <dfn data-x="css-styling-attribute"
    data-x-href="https://drafts.csswg.org/css-style-attr/#style-attribute">style attribute</dfn> is
    defined in <cite>CSS Style Attributes</cite>. <ref>CSSATTR</ref></p>

    <p>The following terms are defined in the <cite>CSS Cascading and Inheritance</cite>:
    <ref>CSSCASCADE</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.csswg.org/css-cascade/#cascaded-value">cascaded value</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-cascade/#specified-value">specified value</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-cascade/#computed-value">computed value</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-cascade/#used-value">used value</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-cascade/#origin">cascade origin</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-cascade/#cascade-origin-author">Author Origin</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-cascade/#cascade-origin-user">User Origin</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-cascade/#cascade-origin-ua">User Agent Origin</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-cascade/#cascade-origin-animation">Animation Origin</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-cascade/#cascade-origin-transition">Transition Origin</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-cascade/#initial-value">initial value</dfn></li>
    </ul>

    <p>The <code>CanvasRenderingContext2D</code> object's use of fonts depends on the features
    described in the CSS <cite>Fonts</cite> and <cite>Font Loading</cite> specifications, including
    in particular <dfn><code>FontFace</code></dfn> objects and the <dfn
    data-x-href="https://drafts.csswg.org/css-font-loading/#font-source">font source</dfn> concept.
    <ref>CSSFONTS</ref> <ref>CSSFONTLOAD</ref></p>

    <p>The following interfaces and terms are defined in <cite>Geometry Interfaces</cite>:
    <ref>GEOMETRY</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.fxtf.org/geometry/#dommatrix"><code>DOMMatrix</code></dfn> interface, and associated
         <dfn data-x-href="https://drafts.fxtf.org/geometry/#matrix-m11-element">m11 element</dfn>,
         <dfn data-x-href="https://drafts.fxtf.org/geometry/#matrix-m12-element">m12 element</dfn>,
         <dfn data-x-href="https://drafts.fxtf.org/geometry/#matrix-m21-element">m21 element</dfn>,
         <dfn data-x-href="https://drafts.fxtf.org/geometry/#matrix-m22-element">m22 element</dfn>,
         <dfn data-x-href="https://drafts.fxtf.org/geometry/#matrix-m41-element">m41 element</dfn>, and
         <dfn data-x-href="https://drafts.fxtf.org/geometry/#matrix-m42-element">m42 element</dfn></li>
     <li><dfn data-x-href="https://drafts.fxtf.org/geometry/#dictdef-dommatrix2dinit"><code>DOMMatrix2DInit</code></dfn> and
         <dfn data-x-href="https://drafts.fxtf.org/geometry/#dictdef-dommatrixinit"><code>DOMMatrixInit</code></dfn> dictionaries</li>
     <li>The <dfn data-x-href="https://drafts.fxtf.org/geometry/#create-a-dommatrix-from-the-dictionary">create a <code>DOMMatrix</code> from a dictionary</dfn>
         and <dfn data-x-href="https://drafts.fxtf.org/geometry/#create-a-dommatrix-from-the-2d-dictionary">create a <code>DOMMatrix</code> from a 2D dictionary</dfn>
         algorithms for <code>DOMMatrix2DInit</code> or <code>DOMMatrixInit</code></li>

     <li>The <dfn data-x-href="https://drafts.fxtf.org/geometry/#dictdef-dompointinit"><code>DOMPointInit</code></dfn> dictionary, and associated
         <dfn data-x="DOMPointInit-x" data-x-href="https://drafts.fxtf.org/geometry/#dom-dompointinit-x">x</dfn> and
         <dfn data-x="DOMPointInit-y" data-x-href="https://drafts.fxtf.org/geometry/#dom-dompointinit-y">y</dfn> members</li>
      <li><dfn data-x-href="https://drafts.fxtf.org/geometry/#matrix-multiply">Matrix multiplication</dfn></li>
    </ul>

    <p>The following terms are defined in the <cite>CSS Scoping</cite>: <ref>CSSSCOPING</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.csswg.org/css-scoping/#flat-tree">flat tree</dfn></li>
    </ul>

    <p>The following terms and features are defined in <cite>CSS Color Adjustment</cite>:
    <ref>CSSCOLORADJUST</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.csswg.org/css-color-adjust/#color-scheme-prop">'color-scheme'</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-color-adjust/#pages-supported-color-schemes">page's supported color-schemes</dfn></li>
    </ul>

    <p>The following terms are defined in <cite>CSS Pseudo-Elements</cite>: <ref>CSSPSEUDO</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.csswg.org/css-pseudo/#details-content-pseudo">'::details-content'</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-pseudo/#file-selector-button-pseudo">'::file-selector-button'</dfn></li>
    </ul>

    <p>The following terms are defined in <cite>CSS Containment</cite>: <ref>CSSCONTAIN</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.csswg.org/css-contain/#skips-its-contents">skips its contents</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-contain/#relevant-to-the-user">relevant to the user</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-contain/#proximity-to-the-viewport">proximity to the viewport</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-contain/#containment-layout">layout containment</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-contain/#content-visibility">'content-visibility'</dfn> property</li>
     <li><dfn data-x-href="https://drafts.csswg.org/css-contain/#propdef-content-visibility" data-x="content-visibility-auto">'auto'</dfn> value for <span>'content-visibility'</span></li>
    </ul>

    <p>The following terms are defined in <cite>CSS Anchor Positioning</cite>: <ref>CSSANCHOR</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.csswg.org/css-anchor-position/#implicit-anchor-element">implicit anchor element</dfn></li>
    </ul>
   </dd>


   <dt>Intersection Observer</dt>

   <dd>
    <p>The following term is defined in <cite>Intersection Observer</cite>:
    <ref>INTERSECTIONOBSERVER</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/IntersectionObserver/#run-the-update-intersection-observations-steps">run the update intersection observations steps</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/IntersectionObserver/#intersectionobserver"><code>IntersectionObserver</code></dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/IntersectionObserver/#dictdef-intersectionobserverinit"><code>IntersectionObserverInit</code></dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/IntersectionObserver/#dom-intersectionobserver-observe" data-x="dom-IntersectionObserver-observe"><code>observe</code></dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/IntersectionObserver/#dom-intersectionobserver-unobserve" data-x="dom-IntersectionObserver-unobserve"><code>unobserve</code></dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/IntersectionObserver/#dom-intersectionobserverentry-isintersecting" data-x="dom-IntersectionObserverEntry-isIntersecting"><code>isIntersecting</code></dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/IntersectionObserver/#dom-intersectionobserverentry-target" data-x="dom-IntersectionObserverEntry-target"><code>target</code></dfn></li>
    </ul>
   </dd>


   <dt>Resize Observer</dt>

   <dd>
    <p>The following terms are defined in <cite>Resize Observer</cite>:
    <ref>RESIZEOBSERVER</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.csswg.org/resize-observer-1/#gather-active-observations-h">gather active resize observations at depth</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/resize-observer-1/#has-active-observations-h">has active resize observations</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/resize-observer-1/#has-skipped-observations-h">has skipped resize observations</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/resize-observer-1/#broadcast-resize-notifications-h">broadcast active resize observations</dfn></li>
     <li><dfn data-x-href="https://drafts.csswg.org/resize-observer-1/#deliver-resize-error">deliver resize loop error</dfn></li>
    </ul>
   </dd>


   <dt>WebGL</dt>

   <dd>
    <p>The following interfaces are defined in the WebGL specifications: <ref>WEBGL</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://www.khronos.org/registry/webgl/specs/latest/1.0/#WebGLRenderingContext"><code>WebGLRenderingContext</code></dfn> interface</li>
     <li><dfn data-x-href="https://www.khronos.org/registry/webgl/specs/latest/2.0/#WebGL2RenderingContext"><code>WebGL2RenderingContext</code></dfn> interface</li>
     <li><dfn data-x-href="https://www.khronos.org/registry/webgl/specs/latest/1.0/#WebGLContextAttributes"><code>WebGLContextAttributes</code></dfn> dictionary</li>
    </ul>
   </dd>



   <dt>WebGPU</dt>

   <dd>
    <p>The following interfaces are defined in <cite>WebGPU</cite>: <ref>WEBGPU</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://gpuweb.github.io/gpuweb/#canvas-context"><code>GPUCanvasContext</code></dfn> interface</li>
    </ul>
   </dd>



   <dt>WebVTT</dt>

   <dd>
    <p>Implementations may support WebVTT as a text track format for subtitles, captions, metadata,
    etc., for media resources. <ref>WEBVTT</ref></p>

    <p>The following terms, used in this specification, are defined in <cite>WebVTT</cite>:</p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/webvtt/#webvtt-file">WebVTT file</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webvtt/#webvtt-file-using-cue-text">WebVTT file using cue text</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webvtt/#webvtt-file-using-only-nested-cues">WebVTT file using only nested cues</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webvtt/#webvtt-parser">WebVTT parser</dfn></li>
     <li>The <dfn data-x-href="https://w3c.github.io/webvtt/#rules-for-updating-the-display-of-webvtt-text-tracks">rules for updating the display of WebVTT text tracks</dfn></li>
     <li>The WebVTT <dfn data-x-href="https://w3c.github.io/webvtt/#webvtt-cue-writing-direction">text track cue writing direction</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webvtt/#vttcue"><code>VTTCue</code></dfn> interface</li>
    </ul>
   </dd>



   <dt>ARIA</dt>

   <dd>
    <p>The <dfn element-attr><code data-x="attr-aria-role">role</code></dfn> attribute is defined in
    <cite>Accessible Rich Internet Applications</cite> (<cite>ARIA</cite>), as are the following
    roles: <ref>ARIA</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/aria/#button"><code data-x="attr-aria-role-button">button</code></dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/aria/#presentation"><code data-x="attr-aria-role-presentation">presentation</code></dfn></li>
    </ul>

    <p>In addition, the following <dfn><code data-x="attr-aria-*">aria-*</code></dfn> content
    attributes are defined in <cite>ARIA</cite>: <ref>ARIA</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/aria/#aria-checked"><code data-x="attr-aria-checked">aria-checked</code></dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/aria/#aria-describedby"><code data-x="attr-aria-describedby">aria-describedby</code></dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/aria/#aria-disabled"><code data-x="attr-aria-disabled">aria-disabled</code></dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/aria/#aria-label"><code data-x="attr-aria-label">aria-label</code></dfn></li>
    </ul>

    <p>Finally, the following terms are defined in <cite>ARIA</cite>: <ref>ARIA</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/aria/#dfn-role">role</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/aria/#dfn-accessible-name" data-x="concept-accessible-name">accessible name</dfn></li>
     <li>The <dfn data-x-href="https://w3c.github.io/aria/#ARIAMixin"><code>ARIAMixin</code></dfn> interface, with its associated
             <dfn data-x-href="https://w3c.github.io/aria/#dfn-ariamixin-getter-steps"><code>ARIAMixin</code> getter steps</dfn> and
             <dfn data-x-href="https://w3c.github.io/aria/#dfn-ariamixin-setter-steps"><code>ARIAMixin</code> setter steps</dfn> hooks, and its
             <dfn data-x="dom-ARIAMixin-role" data-x-href="https://w3c.github.io/aria/#idl-def-ariamixin-role"><code>role</code></dfn> and
             <dfn data-x="dom-ARIAMixin-aria*" data-x-href="https://w3c.github.io/aria/#idl-def-ariamixin-ariaactivedescendantelement"><code>aria*</code></dfn> attributes</li>
    </ul>
   </dd>


   <dt>Content Security Policy</dt>

   <dd>
    <p>The following terms are defined in <cite>Content Security Policy</cite>: <ref>CSP</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/webappsec-csp/#content-security-policy-object">Content Security Policy</dfn></li>
     <li><dfn data-x="csp-disposition" data-x-href="https://w3c.github.io/webappsec-csp/#policy-disposition">disposition</dfn></li>
     <li><dfn data-x="csp-directive-set" data-x-href="https://w3c.github.io/webappsec-csp/#policy-directive-set">directive set</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webappsec-csp/#directives">Content Security Policy directive</dfn></li>
     <li><dfn data-x="concept-csp-list" data-x-href="https://w3c.github.io/webappsec-csp/#csp-list">CSP list</dfn></li>
     <li>The <dfn data-x-href="https://w3c.github.io/webappsec-csp/#grammardef-serialized-policy">Content Security Policy syntax</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webappsec-csp/#enforced">enforce the policy</dfn></li>
     <li>The <dfn data-x-href="https://w3c.github.io/webappsec-csp/#parse-serialized-policy">parse a serialized Content Security Policy</dfn> algorithm</li>
     <li>The <dfn data-x-href="https://w3c.github.io/webappsec-csp/#run-document-csp-initialization">Run CSP initialization for a Document</dfn> algorithm</li>
     <li>The <dfn data-x-href="https://w3c.github.io/webappsec-csp/#run-global-object-csp-initialization">Run CSP initialization for a global object</dfn> algorithm</li>
     <li>The <dfn data-x-href="https://w3c.github.io/webappsec-csp/#should-block-inline">Should element's inline behavior be blocked by Content Security Policy?</dfn> algorithm</li>
     <li>The <dfn data-x-href="https://w3c.github.io/webappsec-csp/#should-block-navigation-request">Should navigation request of type be blocked by Content Security Policy?</dfn> algorithm</li>
     <li>The <dfn data-x-href="https://w3c.github.io/webappsec-csp/#should-block-navigation-response">Should navigation response to navigation request of type in target be blocked by Content Security Policy?</dfn> algorithm</li>
     <li>The <dfn data-x-href="https://w3c.github.io/webappsec-csp/#report-uri"><code data-x="">report-uri</code> directive</dfn></li>
     <li>The <dfn data-x="csp-EnsureCSPDoesNotBlockStringCompilation" data-x-href="https://w3c.github.io/webappsec-csp/#can-compile-strings">EnsureCSPDoesNotBlockStringCompilation</dfn> abstract operation</li>
     <li>The <dfn data-x-href="https://w3c.github.io/webappsec-csp/#allow-base-for-document">Is base allowed for Document?</dfn> algorithm</li>
     <li>The <dfn data-x-href="https://w3c.github.io/webappsec-csp/#frame-ancestors"><code data-x="">frame-ancestors</code> directive</dfn></li>
     <li>The <dfn data-x-href="https://w3c.github.io/webappsec-csp/#sandbox"><code data-x="">sandbox</code> directive</dfn></li>
     <li>The <dfn data-x-href="https://w3c.github.io/webappsec-csp/#contains-a-header-delivered-content-security-policy">contains a header-delivered Content Security Policy</dfn> property.</li>
     <li>The <dfn data-x="parse-response-csp" data-x-href="https://w3c.github.io/webappsec-csp/#parse-response-csp">Parse a response's Content Security Policies</dfn> algorithm.</li>
     <li><dfn data-x-href="https://w3c.github.io/webappsec-csp/#securitypolicyviolationevent"><code>SecurityPolicyViolationEvent</code></dfn> interface</li>
     <li>The <dfn data-x="event-securitypolicyviolation" data-x-href="https://w3c.github.io/webappsec-csp/#eventdef-globaleventhandlers-securitypolicyviolation"><code>securitypolicyviolation</code></dfn> event</li>
    </ul>
   </dd>

   <dt>Service Workers</dt>

   <dd>
    <p>The following terms are defined in <cite>Service Workers</cite>: <ref>SW</ref></p>

    <ul class="brief">
     <li><dfn data-x="dfn-active-worker" data-x-href="https://w3c.github.io/ServiceWorker/#dfn-active-worker">active worker</dfn></li>
     <li><dfn data-x="dfn-client-message-queue" data-x-href="https://w3c.github.io/ServiceWorker/#dfn-client-message-queue">client message queue</dfn></li>
     <li><dfn data-x="dfn-control" data-x-href="https://w3c.github.io/ServiceWorker/#dfn-control">control</dfn></li>
     <li><dfn data-x="on-fetch-request-algorithm" data-x-href="https://w3c.github.io/ServiceWorker/#on-fetch-request-algorithm">handle fetch</dfn></li>
     <li><dfn data-x="scope-match-algorithm" data-x-href="https://w3c.github.io/ServiceWorker/#scope-match-algorithm">match service worker registration</dfn></li>
     <li><dfn data-x="dfn-service-worker" data-x-href="https://w3c.github.io/ServiceWorker/#dfn-service-worker">service worker</dfn></li>
     <li><dfn data-x="serviceworkercontainer-service-worker-client" data-x-href="https://w3c.github.io/ServiceWorker/#serviceworkercontainer-service-worker-client">service worker client</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/service-workers/#dfn-service-worker-registration">service worker registration</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/ServiceWorker/#serviceworker"><code>ServiceWorker</code></dfn> interface</li>
     <li><dfn data-x-href="https://w3c.github.io/ServiceWorker/#serviceworkercontainer"><code>ServiceWorkerContainer</code></dfn> interface</li>
     <li><dfn data-x-href="https://w3c.github.io/ServiceWorker/#serviceworkerglobalscope"><code>ServiceWorkerGlobalScope</code></dfn> interface</li>
     <li><dfn data-x="service-worker-unregister" data-x-href="https://w3c.github.io/service-workers/#navigator-service-worker-unregister"><code>unregister</code></dfn></li>
    </ul>
   </dd>

   <dt>Secure Contexts</dt>

   <dd>
    <p>The following algorithms are defined in <cite>Secure Contexts</cite>:
    <ref>SECURE-CONTEXTS</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/webappsec-secure-contexts/#potentially-trustworthy-url">Is url potentially trustworthy?</dfn></li>
    </ul>
   </dd>

   <dt>Permissions Policy</dt>

   <dd>
    <p>The following terms are defined in <cite>Permissions Policy</cite>:
    <ref>PERMISSIONSPOLICY</ref></p>

    <ul class="brief">
     <li><dfn data-x="concept-permissions-policy" data-x-href="https://w3c.github.io/webappsec-feature-policy/#permissions-policy">permissions policy</dfn></li>
     <li><dfn data-x="concept-policy-controlled-feature" data-x-href="https://w3c.github.io/webappsec-feature-policy/#policy-controlled-feature">policy-controlled feature</dfn></li>
     <li><dfn data-x="concept-container-policy" data-x-href="https://w3c.github.io/webappsec-feature-policy/#container-policy">container policy</dfn></li>
     <li><dfn data-x="concept-serialized-permissions-policy" data-x-href="https://w3c.github.io/webappsec-feature-policy/#serialized-permissions-policy">serialized permissions policy</dfn></li>
     <li><dfn data-x="concept-default-allowlist" data-x-href="https://w3c.github.io/webappsec-feature-policy/#default-allowlist">default allowlist</dfn></li>
     <li>The <dfn data-x-href="https://w3c.github.io/webappsec-feature-policy/#create-for-navigable">creating a permissions policy</dfn> algorithm</li>
     <li>The <dfn data-x-href="https://w3c.github.io/webappsec-feature-policy/#create-from-response">creating a permissions policy from a response</dfn> algorithm</li>
     <li>The <dfn data-x="is-feature-enabled" data-x-href="https://w3c.github.io/webappsec-feature-policy/#is-feature-enabled">is feature enabled by policy for origin</dfn> algorithm</li>
     <li>The <dfn data-x="process-permissions-policy-attributes" data-x-href="https://w3c.github.io/webappsec-feature-policy/#process-permissions-policy-attributes">process permissions policy attributes</dfn> algorithm</li>
    </ul>
   </dd>


   <dt>Payment Request API</dt>

   <dd>
    <p>The following feature is defined in <cite>Payment Request API</cite>:
    <ref>PAYMENTREQUEST</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/payment-request/#dom-paymentrequest"><code>PaymentRequest</code></dfn> interface</li>
    </ul>
   </dd>


   <dt>MathML</dt>

   <dd>
    <p>While support for MathML as a whole is not required by this specification (though it is
    encouraged, at least for web browsers), certain features depend upon small parts of MathML being
    implemented. <ref>MATHML</ref></p>

    <p>The following features are defined in <cite>Mathematical Markup Language</cite>
    (<cite>MathML</cite>):</p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/mathml-core/#dfn-annotation-xml">MathML <code>annotation-xml</code></dfn> element</li>
     <li><dfn data-x-href="https://w3c.github.io/mathml-core/#the-top-level-math-element">MathML <code>math</code></dfn> element</li>
     <li><dfn data-x-href="https://w3c.github.io/mathml-core/#error-message-merror">MathML <code>merror</code></dfn> element</li>
     <li><dfn data-x-href="https://w3c.github.io/mathml-core/#the-mi-element">MathML <code>mi</code></dfn> element</li>
     <li><dfn data-x-href="https://w3c.github.io/mathml-core/#number-mn">MathML <code>mn</code></dfn> element</li>
     <li><dfn data-x-href="https://w3c.github.io/mathml-core/#operator-fence-separator-or-accent-mo">MathML <code>mo</code></dfn> element</li>
     <li><dfn data-x-href="https://w3c.github.io/mathml-core/#string-literal-ms">MathML <code>ms</code></dfn> element</li>
     <li><dfn data-x-href="https://w3c.github.io/mathml-core/#text-mtext">MathML <code>mtext</code></dfn> element</li>
    </ul>
   </dd>


   <dt>SVG</dt>

   <dd>
    <p>While support for SVG as a whole is not required by this specification (though it is
    encouraged, at least for web browsers), certain features depend upon parts of SVG being
    implemented.</p>

    <p>User agents that implement SVG must implement the <cite>SVG 2</cite> specification, and not
    any earlier revisions.</p>

    <p>The following features are defined in the <cite>SVG 2</cite> specification:
    <ref>SVG</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://svgwg.org/svg2-draft/types.html#InterfaceSVGElement"><code>SVGElement</code></dfn> interface</li>
     <li><dfn data-x-href="https://svgwg.org/svg2-draft/embedded.html#InterfaceSVGImageElement"><code>SVGImageElement</code></dfn> interface</li>
     <li><dfn data-x-href="https://svgwg.org/svg2-draft/interact.html#InterfaceSVGScriptElement"><code>SVGScriptElement</code></dfn> interface</li>
     <li><dfn data-x-href="https://svgwg.org/svg2-draft/struct.html#InterfaceSVGSVGElement"><code>SVGSVGElement</code></dfn> interface</li>
     <li><dfn data-x-href="https://svgwg.org/svg2-draft/linking.html#AElement">SVG <code>a</code></dfn> element</li>
     <li><dfn data-x-href="https://svgwg.org/svg2-draft/struct.html#DescElement">SVG <code>desc</code></dfn> element</li>
     <li><dfn data-x-href="https://svgwg.org/svg2-draft/embedded.html#ForeignObjectElement">SVG <code>foreignObject</code></dfn> element</li>
     <li><dfn data-x-href="https://svgwg.org/svg2-draft/embedded.html#ImageElement">SVG <code>image</code></dfn> element</li>
     <li><dfn data-x-href="https://svgwg.org/svg2-draft/interact.html#ScriptElement">SVG <code>script</code></dfn> element</li>
     <li><dfn data-x-href="https://svgwg.org/svg2-draft/struct.html#SVGElement">SVG <code>svg</code></dfn> element</li>
     <li><dfn data-x-href="https://svgwg.org/svg2-draft/struct.html#TitleElement">SVG <code>title</code></dfn> element</li>
     <li><dfn data-x-href="https://svgwg.org/svg2-draft/struct.html#UseElement">SVG <code>use</code></dfn> element</li>
     <li><dfn data-x-href="https://svgwg.org/svg2-draft/painting.html#TextRendering">SVG <code>text-rendering</code></dfn> property</li>
    </ul>
   </dd>


   <dt>Filter Effects</dt>

   <dd>
    <p>The following features are defined in <cite>Filter Effects</cite>: <ref>FILTERS</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.fxtf.org/filter-effects/#typedef-filter-value-list">&lt;filter-value-list&gt;</dfn></li>
    </ul>
   </dd>

   <dt>Compositing</dt>

   <dd>
    <p>The following features are defined in <cite>Compositing and Blending</cite>:
    <ref>COMPOSITE</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://drafts.fxtf.org/compositing/#ltblendmodegt">&lt;blend-mode&gt;</dfn></li>
     <li><dfn data-x-href="https://drafts.fxtf.org/compositing/#compositemode">&lt;composite-mode&gt;</dfn></li>
     <li><dfn data-x-href="https://drafts.fxtf.org/compositing/#porterduffcompositingoperators_srcover" data-x="gcop-source-over">source-over</dfn></li>
     <li><dfn data-x-href="https://drafts.fxtf.org/compositing/#porterduffcompositingoperators_src" data-x="gcop-copy">copy</dfn></li>
    </ul>
   </dd>
   <dt>Cooperative Scheduling of Background Tasks</dt>

   <dd>
    <p>The following features are defined in <cite>Cooperative Scheduling of Background
    Tasks</cite>: <ref>REQUESTIDLECALLBACK</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/requestidlecallback/#the-requestidlecallback-method"><code>requestIdleCallback()</code></dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/requestidlecallback/#start-an-idle-period-algorithm">start an idle period algorithm</dfn></li>
    </ul>
   </dd>

   <dt>Screen Orientation</dt>
   <dd>
    <p>The following terms are defined in <cite>Screen Orientation</cite>:
    <ref>SCREENORIENTATION</ref></p>

    <ul>
     <li><dfn data-x-href="https://w3c.github.io/screen-orientation/#dfn-screen-orientation-change-steps">screen orientation change steps</dfn></li>
    </ul>
   </dd>

   <dt>Storage</dt>
   <dd>
    <p>The following terms are defined in <cite>Storage</cite>: <ref>STORAGE</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://storage.spec.whatwg.org/#obtain-a-local-storage-bottle-map">obtain a local storage bottle map</dfn></li>
     <li><dfn data-x-href="https://storage.spec.whatwg.org/#obtain-a-session-storage-bottle-map">obtain a session storage bottle map</dfn></li>
     <li><dfn data-x-href="https://storage.spec.whatwg.org/#obtain-a-storage-key-for-non-storage-purposes">obtain a storage key for non-storage purposes</dfn></li>
     <li><dfn data-x-href="https://storage.spec.whatwg.org/#storage-key-equal">storage key equal</dfn></li>
     <li><dfn data-x-href="https://storage.spec.whatwg.org/#storage-proxy-map">storage proxy map</dfn></li>
     <li><dfn data-x-href="https://storage.spec.whatwg.org/#legacy-clone-a-traversable-storage-shed">legacy-clone a traversable storage shed</dfn></li>
    </ul>
   </dd>

   <dt>Web App Manifest</dt>

   <dd>
    <p>The following features are defined in <cite>Web App Manifest</cite>: <ref>MANIFEST</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/manifest/#dfn-manifest">application manifest</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/manifest/#dfn-installed-web-application">installed web application</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/manifest/#dfn-processing-a-manifest">process the manifest</dfn></li>
    </ul>
   </dd>

   <dt>WebAssembly JavaScript Interface: ESM Integration</dt>

   <dd>
    <p>The following terms are defined in <cite>WebAssembly JavaScript Interface: ESM Integration</cite>: <ref>WASMESM</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://webassembly.github.io/esm-integration/js-api/index.html#webassembly-module-record">WebAssembly Module Record</dfn></li>
     <li><dfn data-x-href="https://webassembly.github.io/esm-integration/js-api/index.html#parse-a-webassembly-module">parse a WebAssembly module</dfn></li>
    </ul>
   </dd>

   <dt>WebCodecs</dt>

   <dd>
    <p>The following features are defined in <cite>WebCodecs</cite>: <ref>WEBCODECS</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/webcodecs/#videoframe-interface"><code>VideoFrame</code></dfn> interface.</li>
     <li><dfn data-x-href="https://w3c.github.io/webcodecs/#dom-videoframe-display-width-slot">[[display width]]</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webcodecs/#dom-videoframe-display-height-slot">[[display height]]</dfn></li>
    </ul>
   </dd>

   <dt>WebDriver</dt>

   <dd>
    <p>The following terms are defined in <cite>WebDriver</cite>: <ref>WEBDRIVER</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/webdriver/#dfn-extension-commands">extension command</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver/#dfn-remote-end-steps">remote end steps</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver/#dfn-errors">WebDriver error</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver/#dfn-error-code">WebDriver error code</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver/#dfn-invalid-argument">invalid argument</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver/#dfn-getting-properties">getting a property</dfn></li>
     <li><dfn data-x="success-value" data-x-href="https://w3c.github.io/webdriver/#dfn-success">success</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver/#security">WebDriver's security considerations</dfn></li>
     <li><dfn data-x="webdriver-current-browsing-context" data-x-href="https://w3c.github.io/webdriver/#dfn-current-browsing-context">current browsing context</dfn></li>
    </ul>
   </dd>

   <dt>WebDriver BiDi</dt>

   <dd>
    <p>The following terms are defined in <cite>WebDriver BiDi</cite>: <ref>WEBDRIVERBIDI</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/webdriver-bidi/#webdriver-bidi-navigation-status">WebDriver BiDi navigation status</dfn></li>
     <li><dfn data-x="navigation-status-id" data-x-href="https://w3c.github.io/webdriver-bidi/#navigation-status-id">navigation status id</dfn></li>
     <li><dfn data-x="navigation-status-status" data-x-href="https://w3c.github.io/webdriver-bidi/#navigation-status-status">navigation status status</dfn></li>
     <li><dfn data-x="navigation-status-canceled" data-x-href="https://w3c.github.io/webdriver-bidi/#navigation-status-canceled">navigation status canceled</dfn></li>
     <li><dfn data-x="navigation-status-committed" data-x-href="https://w3c.github.io/webdriver-bidi/#navigation-status-committed">navigation status committed</dfn></li>
     <li><dfn data-x="navigation-status-pending" data-x-href="https://w3c.github.io/webdriver-bidi/#navigation-status-pending">navigation status pending</dfn></li>
     <li><dfn data-x="navigation-status-complete" data-x-href="https://w3c.github.io/webdriver-bidi/#navigation-status-complete">navigation status complete</dfn></li>
     <li><dfn data-x="navigation-status-url" data-x-href="https://w3c.github.io/webdriver-bidi/#navigation-status-url">navigation status url</dfn></li>
     <li><dfn data-x="navigation-status-suggested-filename" data-x-href="https://w3c.github.io/webdriver-bidi/#navigation-status-suggested-filename">navigation status suggested filename</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver-bidi/#webdriver-bidi-navigation-aborted">WebDriver BiDi navigation aborted</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver-bidi/#webdriver-bidi-navigation-committed">WebDriver BiDi navigation committed</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver-bidi/#webdriver-bidi-navigation-failed">WebDriver BiDi navigation failed</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver-bidi/#webdriver-bidi-navigation-started">WebDriver BiDi navigation started</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver-bidi/#webdriver-bidi-download-started">WebDriver BiDi download started</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver-bidi/#webdriver-bidi-fragment-navigated">WebDriver BiDi fragment navigated</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver-bidi/#webdriver-bidi-dom-content-loaded">WebDriver BiDi DOM content loaded</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver-bidi/#webdriver-bidi-load-complete">WebDriver BiDi load complete</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver-bidi/#webdriver-bidi-history-updated">WebDriver BiDi history updated</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver-bidi/#webdriver-bidi-navigable-created">WebDriver BiDi navigable created</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver-bidi/#webdriver-bidi-navigable-destroyed">WebDriver BiDi navigable destroyed</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver-bidi/#webdriver-bidi-user-prompt-closed">WebDriver BiDi user prompt closed</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver-bidi/#webdriver-bidi-user-prompt-opened">WebDriver BiDi user prompt opened</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webdriver-bidi/#webdriver-bidi-file-dialog-opened">WebDriver BiDi file dialog opened</dfn></li>
    </ul>
   </dd>

   <dt>Web Cryptography API</dt>

   <dd>
    <p>The following terms are defined in <cite>Web Cryptography API</cite>:
    <ref>WEBCRYPTO</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/webcrypto/#dfn-generate-a-random-uuid">generating a random UUID</dfn></li>
    </ul>
   </dd>

   <dt>WebSockets</dt>

   <dd>
    <p>The following terms are defined in <cite>WebSockets</cite>: <ref>WEBSOCKETS</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://websockets.spec.whatwg.org/#websocket"><code>WebSocket</code></dfn></li>
     <li><dfn data-x-href="https://websockets.spec.whatwg.org/#make-disappear" for="WebSocket">make disappear</dfn></li>
    </ul>
   </dd>

   <dt>WebTransport</dt>

   <dd>
    <p>The following terms are defined in <cite>WebTransport</cite>: <ref>WEBTRANSPORT</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/webtransport/#webtransport"><code>WebTransport</code></dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webtransport/#context-cleanup-steps"><code>context cleanup steps</code></dfn></li>
    </ul>
   </dd>

   <dt>Web Authentication: An API for accessing Public Key Credentials</dt>

   <dd>
    <p>The following terms are defined in <cite>Web Authentication: An API for accessing Public Key
    Credentials</cite>: <ref>WEBAUTHN</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/webauthn/#public-key-credential">public key credential</dfn></li>
    </ul>
   </dd>

   <dt>Credential Management</dt>

   <dd>
    <p>The following terms are defined in <cite>Credential Management</cite>: <ref>CREDMAN</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/webappsec-credential-management/#dom-credentialmediationrequirement-conditional">conditional mediation</dfn></li>
     <li><dfn data-x="credman-credential" data-x-href="https://w3c.github.io/webappsec-credential-management/#credential">credential</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webappsec-credential-management/#dom-credentialscontainer-get"><code>navigator.credentials.get()</code></dfn></li>
    </ul>
   </dd>

   <dt>Console</dt>

   <dd>
    <p>The following terms are defined in <cite>Console</cite>: <ref>CONSOLE</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://console.spec.whatwg.org/#report-a-warning-to-the-console">report a warning to the console</dfn></li>
    </ul>
   </dd>

   <dt>Web Locks API</dt>

   <dd>
    <p>The following terms are defined in <cite>Web Locks API</cite>: <ref>WEBLOCKS</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/web-locks/#lock-concept">locks</dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/web-locks/#lock-request">lock requests</dfn></li>
    </ul>
   </dd>

   <dt>Trusted Types</dt>

   <dd>
    <p>This specification uses the following features defined in <cite>Trusted Types</cite>:
    <ref>TRUSTED-TYPES</ref></p>

    <ul class="brief">
     <li><dfn data-x="tt-trustedhtml" data-x-href="https://w3c.github.io/trusted-types/dist/spec/#trustedhtml"><code>TrustedHTML</code></dfn></li>
     <li><dfn data-x="tt-trustedhtml-data" data-x-href="https://w3c.github.io/trusted-types/dist/spec/#trustedhtml-data">data</dfn></li>
     <li><dfn data-x="tt-trustedscript" data-x-href="https://w3c.github.io/trusted-types/dist/spec/#trusted-script"><code>TrustedScript</code></dfn></li>
     <li><dfn data-x="tt-trustedscript-data" data-x-href="https://w3c.github.io/trusted-types/dist/spec/#trustedscript-data"><code>data</code></dfn></li>
     <li><dfn data-x="tt-trustedscripturl" data-x-href="https://w3c.github.io/trusted-types/dist/spec/#trustedscripturl"><code>TrustedScriptURL</code></dfn></li>
     <li><dfn data-x="tt-getcompliantstring" data-x-href="https://w3c.github.io/trusted-types/dist/spec/#get-trusted-type-compliant-string-algorithm">Get Trusted Type compliant string</dfn></li>
    </ul>
   </dd>

   <dt>WebRTC API</dt>

   <dd>
    <p>The following terms are defined in <cite>WebRTC API</cite>: <ref>WEBRTC</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/webrtc-pc/#dom-rtcdatachannel"><code>RTCDataChannel</code></dfn></li>
     <li><dfn data-x-href="https://w3c.github.io/webrtc-pc/#dom-rtcpeerconnection"><code>RTCPeerConnection</code></dfn></li>
    </ul>
   </dd>

   <dt>Picture-in-Picture API</dt>

   <dd>
    <p>The following terms are defined in <cite>Picture-in-Picture API</cite>: <ref>PICTUREINPICTURE</ref></p>

    <ul class="brief">
     <li><dfn data-x-href="https://w3c.github.io/picture-in-picture/#pictureinpicturewindow"><code>PictureInPictureWindow</code></dfn></li>
    </ul>
   </dd>

   <dt>Idle Detection API</dt>

   <dd>
    <p>The following terms are defined in <cite>Idle Detection API</cite>:</p>

    <ul class="brief">
     <li><dfn data-x-href="https://wicg.github.io/idle-detection/#idledetector"><code>IdleDetector</code></dfn></li>
    </ul>
   </dd>

   <dt>Web Speech API</dt>

   <dd>
    <p>The following terms are defined in <cite>Web Speech API</cite>:</p>

    <ul class="brief">
     <li><dfn data-x-href="https://wicg.github.io/speech-api/#speechrecognition"><code>SpeechRecognition</code></dfn></li>
    </ul>
   </dd>

   <dt>WebOTP API</dt>

   <dd>
    <p>The following terms are defined in <cite>WebOTP API</cite>:</p>

    <ul class="brief">
     <li><dfn data-x-href="https://wicg.github.io/web-otp/#otpcredential"><code>OTPCredential</code></dfn></li>
    </ul>
   </dd>

  <dt>Web Share API</dt>

  <dd>
   <p>The following terms are defined in <cite>Web Share API</cite>:</p>

   <ul class="brief">
    <li><dfn data-x-href="https://w3c.github.io/web-share/#share-method" data-x="dom-Navigator-share">share()</dfn></li>
   </ul>
  </dd>

  <dt>Web Smart Card API</dt>

  <dd>
   <p>The following terms are defined in <cite>Web Smart Card API</cite>:</p>

   <ul class="brief">
    <li><dfn data-x-href="https://wicg.github.io/web-smart-card/#dom-smartcardconnection"><code>SmartCardConnection</code></dfn></li>
   </ul>
  </dd>

  <dt>Web Background Synchronization</dt>

  <dd>
   <p>The following terms are defined in <cite>Web Background Synchronization</cite>:</p>

   <ul class="brief">
    <li><dfn data-x-href="https://wicg.github.io/background-sync/spec/#syncmanager"><code>SyncManager</code></dfn></li>
    <li><dfn data-x-href="https://wicg.github.io/background-sync/spec/#dom-syncmanager-register" data-x="dom-SyncManager-register"><code>register()</code></dfn></li>
   </ul>
  </dd>

  <dt>Web Periodic Background Synchronization</dt>

  <dd>
   <p>The following terms are defined in <cite>Web Periodic Background Synchronization</cite>:</p>

   <ul class="brief">
    <li><dfn data-x-href="https://wicg.github.io/periodic-background-sync/#periodicsyncmanager"><code>PeriodicSyncManager</code></dfn></li>
    <li><dfn data-x-href="https://wicg.github.io/periodic-background-sync/#dom-periodicsyncmanager-register" data-x="dom-PeriodicSyncManager-register"><code>register()</code></dfn></li>
   </ul>
  </dd>

  <dt>Web Background Fetch</dt>

  <dd>
   <p>The following terms are defined in <cite>Background Fetch</cite>:</p>

   <ul class="brief">
    <li><dfn data-x-href="https://wicg.github.io/background-fetch/#backgroundfetchmanager"><code>BackgroundFetchManager</code></dfn></li>
    <li><dfn data-x-href="https://wicg.github.io/background-fetch/#dom-backgroundfetchmanager-fetch" data-x="dom-BackgroundFetchManager-fetch"><code>fetch()</code></dfn></li>
   </ul>
  </dd>

  <dt>Keyboard Lock</dt>

  <dd>
   <p>The following terms are defined in <cite>Keyboard Lock</cite>:</p>

   <ul class="brief">
    <li><dfn data-x-href="https://wicg.github.io/keyboard-lock/#keyboard"><code>Keyboard</code></dfn></li>
    <li><dfn data-x-href="https://wicg.github.io/keyboard-lock/#dom-keyboard-lock" data-x="dom-Keyboard-lock"><code>lock()</code></dfn></li>
   </ul>
  </dd>

  <dt>Web MIDI API</dt>

  <dd>
   <p>The following terms are defined in <cite>Web MIDI API</cite>:</p>

   <ul class="brief">
    <li><dfn data-x-href="https://webaudio.github.io/web-midi-api/#dom-navigator-requestmidiaccess" data-x="dom-Navigator-requestMIDIAccess"><code>requestMIDIAccess()</code></dfn></li>
   </ul>
  </dd>

  <dt>Generic Sensor API</dt>

  <dd>
   <p>The following terms are defined in <cite>Generic Sensor API</cite>:</p>

   <ul class="brief">
    <li><dfn data-x="request-sensor-access" data-x-href="https://w3c.github.io/sensors/#request-sensor-access">request sensor access</dfn></li>
   </ul>
  </dd>

  <dt>WebHID API</dt>

  <dd>
   <p>The following terms are defined in <cite>WebHID API</cite>:</p>

   <ul class="brief">
    <li><dfn data-x="request-device" data-x-href="https://wicg.github.io/webhid/#requestdevice-method"><code>requestDevice</code></dfn></li>
   </ul>
  </dd>

  <dt>WebXR Device API</dt>

  <dd>
   <p>The following terms are defined in <cite>WebXR Device API</cite>:</p>

   <ul class="brief">
    <li><dfn data-x-href="https://immersive-web.github.io/webxr/#xrsystem"><code>XRSystem</code></dfn></li>
   </ul>
  </dd>

 </dl>

  <hr>

  <p>This specification does not <em>require</em> support of any particular network protocol, style
  sheet language, scripting language, or any of the DOM specifications beyond those required in the
  list above. However, the language described by this specification is biased towards CSS as the
  styling language, JavaScript as the scripting language, and HTTP as the network protocol, and
  several features assume that those languages and protocols are in use.</p>

  <p>A user agent that implements the HTTP protocol must implement <cite>HTTP State Management
  Mechanism</cite> (Cookies) as well. <ref>HTTP</ref> <ref>COOKIES</ref></p>

  <p class="note">This specification might have certain additional requirements on character
  encodings, image formats, audio formats, and video formats in the respective sections.</p>

  </div>




  <h4>Extensibility</h4>

  <p>Vendor-specific proprietary user agent extensions to this specification are strongly
  discouraged. Documents must not use such extensions, as doing so reduces interoperability and
  fragments the user base, allowing only users of specific user agents to access the content in
  question.</p>

  <div w-nodev>

  <p>All extensions must be defined so that the use of extensions neither contradicts nor causes the
  non-conformance of functionality defined in the specification.</p> <!-- thanks to QA Framework -->

  <div class="example">

   <p>For example, while strongly discouraged from doing so, an implementation could add a new IDL
   attribute "<code data-x="">typeTime</code>" to a control that returned the time it took the user
   to select the current value of a control (say). On the other hand, defining a new control that
   appears in a form's <code data-x="dom-form-elements">elements</code> array would be in violation
   of the above requirement, as it would violate the definition of <code
   data-x="dom-form-elements">elements</code> given in this specification.</p>

  </div>

  </div>

  <hr>

  <p>When vendor-neutral extensions to this specification are needed, either this specification can
  be updated accordingly, or an extension specification can be written that overrides the
  requirements in this specification. When someone applying this specification to their activities
  decides that they will recognize the requirements of such an extension specification, it becomes
  an <dfn data-x="other applicable specifications">applicable specification</dfn> for the purposes
  of conformance requirements in this specification.</p>

  <p class="note">Someone could write a specification that defines any arbitrary byte stream as
  conforming, and then claim that their random junk is conforming. However, that does not mean that
  their random junk actually is conforming for everyone's purposes: if someone else decides that
  that specification does not apply to their work, then they can quite legitimately say that the
  aforementioned random junk is just that, junk, and not conforming at all. As far as conformance
  goes, what matters in a particular community is what that community <em>agrees</em> is
  applicable.</p>

  <div w-nodev>

  <hr>

  <p>User agents must treat elements and attributes that they do not understand as semantically
  neutral; leaving them in the DOM (for DOM processors), and styling them according to CSS (for CSS
  processors), but not inferring any meaning from them.</p>

  <p>When support for a feature is disabled (e.g. as an emergency measure to mitigate a security
  problem, or to aid in development, or for performance reasons), user agents must act as if they
  had no support for the feature whatsoever, and as if the feature was not mentioned in this
  specification. For example, if a particular feature is accessed via an attribute in a Web IDL
  interface, the attribute itself would be omitted from the objects that implement that interface
  &mdash; leaving the attribute on the object but making it return null or throw an exception is
  insufficient.</p>

  </div>


  <div w-nodev>

  <h4>Interactions with XPath and XSLT</h4>

  <p id="xpath-1.0-processors">Implementations of XPath 1.0 that operate on <span>HTML
  documents</span> parsed or created in the manners described in this specification (e.g. as part of
  the <code data-x="">document.evaluate()</code> API) must act as if the following edit was applied
  to the XPath 1.0 specification.</p>

  <p>First, remove this paragraph:</p>

  <blockquote cite="https://www.w3.org/TR/1999/REC-xpath-19991116/#node-tests">

   <p>A <a href="https://www.w3.org/TR/REC-xml-names/#NT-QName">QName</a> in the node test is expanded
   into an <a href="https://www.w3.org/TR/1999/REC-xpath-19991116/#dt-expanded-name">expanded-name</a>
   using the namespace declarations from the expression context. This is the same way expansion is
   done for element type names in start and end-tags except that the default namespace declared with
   <code data-x="">xmlns</code> is not used: if the <a
   href="https://www.w3.org/TR/REC-xml-names/#NT-QName">QName</a> does not have a prefix, then the
   namespace URI is null (this is the same way attribute names are expanded). It is an error if the
   <a href="https://www.w3.org/TR/REC-xml-names/#NT-QName">QName</a> has a prefix for which there is
   no namespace declaration in the expression context.</p>

  </blockquote>

  <p>Then, insert in its place the following:</p>

  <blockquote cite="https://www.w3.org/Bugs/Public/show_bug.cgi?id=7059#c37">

   <p>A QName in the node test is expanded into an expanded-name using the namespace declarations
   from the expression context. If the QName has a prefix, then there must be a<!-- added 2009-10-27
   - https://www.w3.org/Bugs/Public/show_bug.cgi?id=8062 --> namespace declaration for this prefix in
   the expression context, and the corresponding<!-- typo fixed 2009-10-27 -
   https://www.w3.org/Bugs/Public/show_bug.cgi?id=8063 --> namespace URI is the one that is
   associated with this prefix. It is an error if the QName has a prefix for which there is no
   namespace declaration in the expression context. </p>

   <p>If the QName has no prefix and the principal node type of the axis is element, then the
   default element namespace is used. Otherwise, if the QName has no prefix, the namespace URI is
   null. The default element namespace is a member of the context for the XPath expression. The
   value of the default element namespace when executing an XPath expression through the DOM3 XPath
   API is determined in the following way:</p>

   <ol>
    <li>If the context node is from an HTML DOM, the default element namespace is
    "http://www.w3.org/1999/xhtml".</li>

    <li>Otherwise, the default element namespace URI is null.</li>
   </ol>

   <p class="note">This is equivalent to adding the default element namespace feature of XPath 2.0
   to XPath 1.0, and using the HTML namespace as the default element namespace for HTML documents.
   It is motivated by the desire to have implementations be compatible with legacy HTML content
   while still supporting the changes that this specification introduces to HTML regarding the
   namespace used for HTML elements, and by the desire to use XPath 1.0 rather than XPath 2.0.</p>

  </blockquote>

  <p class="note">This change is a <span>willful violation</span> of the XPath 1.0 specification,
  motivated by desire to have implementations be compatible with legacy content while still
  supporting the changes that this specification introduces to HTML regarding which namespace is
  used for HTML elements. <ref>XPATH10</ref></p> <!-- note: version matters for
  this ref -->

  <hr>

  <p id="dom-based-xslt-1.0-processors">XSLT 1.0 processors outputting to a DOM when the output
  method is "html" (either explicitly or via the defaulting rule in XSLT 1.0) are affected as
  follows:</p>

  <p>If the transformation program outputs an element in no namespace, the processor must, prior to
  constructing the corresponding DOM element node, change the namespace of the element to the
  <span>HTML namespace</span>, <span data-x="converted to ASCII lowercase">ASCII-lowercase</span> the
  element's local name, and <span data-x="converted to ASCII lowercase">ASCII-lowercase</span> the
  names of any non-namespaced attributes on the element.</p>

  <p class="note">This requirement is a <span>willful violation</span> of the XSLT 1.0
  specification, required because this specification changes the namespaces and case-sensitivity
  rules of HTML in a manner that would otherwise be incompatible with DOM-based XSLT
  transformations. (Processors that serialize the output are unaffected.) <ref>XSLT10</ref></p> <!--
  note: version matters for this ref -->

  <hr>

  <p>This specification does not specify precisely how XSLT processing interacts with the <span>HTML
  parser</span> infrastructure (for example, whether an XSLT processor acts as if it puts any
  elements into a <span>stack of open elements</span>). However, XSLT processors must <span>stop
  parsing</span> if they successfully complete, and must <span>update the current document
  readiness</span> first to "<code data-x="">interactive</code>" and then to "<code
  data-x="">complete</code>" if they are aborted.</p>

  <hr>

  <p>This specification does not specify how XSLT interacts with the <span
  data-x="navigate">navigation</span> algorithm, how it fits in with the <span>event loop</span>, nor
  how error pages are to be handled (e.g. whether XSLT errors are to replace an incremental XSLT
  output, or are rendered inline, etc.).</p>

  <p class="note">There are also additional non-normative comments regarding the interaction of XSLT
  and HTML <a href="#scriptTagXSLT">in the <code>script</code> element section</a>, and of
  XSLT, XPath, and HTML <a href="#template-XSLT-XPath">in the <code>template</code> element
  section</a>.</p>

  </div>



  <h3 id="policy-controlled-features">Policy-controlled features</h3>

  <p>This document defines the following <span
  data-x="concept-policy-controlled-feature">policy-controlled features</span>:</p>

  <ul>
    <li>"<dfn data-x="autoplay-feature"><code data-x="">autoplay</code></dfn>", which has a <span
    data-x="concept-default-allowlist">default allowlist</span> of <code
    data-x="">'self'</code>.</li>
    <li>"<dfn data-x="cross-origin-isolated-feature"><code
    data-x="">cross-origin-isolated</code></dfn>", which has a <span
    data-x="concept-default-allowlist">default allowlist</span> of <code
    data-x="">'self'</code>.</li>
    <li>"<dfn data-x="focus-without-user-activation-feature"><code
    data-x="">focus-without-user-activation</code></dfn>", which has a <span
    data-x="concept-default-allowlist">default allowlist</span> of <code
    data-x="">'self'</code>.</li>
  </ul>


  <h3 split-filename="common-microsyntaxes">Common microsyntaxes</h3>

  <p>There are various places in HTML that accept particular data types, such as dates or numbers.
  This section describes what the conformance criteria for content in those formats is, and how to
  parse them.</p>

  <div w-nodev>

  <p class="note">Implementers are strongly urged to carefully examine any third-party libraries
  they might consider using to implement the parsing of syntaxes described below. For example, date
  libraries are likely to implement error handling behavior that differs from what is required in
  this specification, since error-handling behavior is often not defined in specifications that
  describe date syntaxes similar to those used in this specification, and thus implementations tend
  to vary greatly in how they handle errors.</p>

  </div>


  <div w-nodev>

  <h4>Common parser idioms</h4>

  <p>Some of the micro-parsers described below follow the pattern of having an <var>input</var>
  variable that holds the string being parsed, and having a <var>position</var> variable pointing at
  the next character to parse in <var>input</var>.</p>

  </div>



  <h4>Boolean attributes</h4>

  <p>A number of attributes are <dfn data-x="boolean attribute">boolean attributes</dfn>. The
  presence of a boolean attribute on an element represents the true value, and the absence of the
  attribute represents the false value.</p>

  <p>If the attribute is present, its value must either be the empty string or a value that is an
  <span>ASCII case-insensitive</span> match for the attribute's canonical name, with no leading or
  trailing whitespace.</p>

  <p class="note">The values "true" and "false" are not allowed on boolean attributes. To represent
  a false value, the attribute has to be omitted altogether.</p>

  <div class="example">

   <p>Here is an example of a checkbox that is checked and disabled. The <code
   data-x="attr-input-checked">checked</code> and <code data-x="attr-fe-disabled">disabled</code>
   attributes are the boolean attributes.</p>

   <pre><code class="html">&lt;label>&lt;input type=checkbox checked name=cheese disabled> Cheese&lt;/label></code></pre>

   <p>This could be equivalently written as this:

   <pre><code class="html">&lt;label>&lt;input type=checkbox checked=checked name=cheese disabled=disabled> Cheese&lt;/label></code></pre>

   <p>You can also mix styles; the following is still equivalent:</p>

   <pre><code class="html">&lt;label>&lt;input type='checkbox' checked name=cheese disabled=""> Cheese&lt;/label></code></pre>

  </div>



  <h4>Keywords and enumerated attributes</h4>

  <p>Some attributes, called <dfn data-x="enumerated attribute" data-lt="enumerated attribute"
  export>enumerated attributes</dfn>, take on a finite set of states. The state for such an
  attribute is derived by combining the attribute's value, a set of keyword/state mappings given in
  the specification of each attribute, and two possible special states that can also be given in the
  specification of the attribute. These special states are the <dfn><i>invalid value
  default</i></dfn> and the <dfn><i>missing value default</i></dfn>.</p>

  <p class="note">Multiple keywords can map to the same state.</p>

  <p class="note">The empty string can be a valid keyword. Note that the <i data-x="missing value
  default">missing value default</i> applies only when the attribute is <em>missing</em>, not when
  it is present with an empty string value.</p>

  <p>To determine the state of an attribute, use the following steps:</p>

  <ol>
   <li>
    <p>If the attribute is not specified:</p>

    <ol>
     <li><p>If the attribute has a <i data-x="missing value default">missing value default</i>
     state defined, then return that <i data-x="missing value default">missing value default</i>
     state.</p></li>

     <li><p>Otherwise, return no state.</p></li>
    </ol>
   </li>

   <li><p>If the attribute's value is an <span>ASCII case-insensitive</span> match for one of the
   keywords defined for the attribute, then return the state represented by that keyword.</p></li>

   <li><p>If the attribute has an <i data-x="invalid value default">invalid value default</i> state
   defined, then return that <i data-x="invalid value default">invalid value default</i>
   state.</p></li>

   <li><p>Return no state.</p></li>
  </ol>

  <p>For authoring conformance purposes, if an enumerated attribute is specified, the attribute's
  value must be an <span>ASCII case-insensitive</span> match for one of the conforming keywords for
  that attribute, with no leading or trailing whitespace.</p>

  <p>For <span data-x="reflect">reflection</span> purposes, states which have any keywords mapping
  to them are said to have a <dfn>canonical keyword</dfn>. This is determined as follows:</p>

  <ul>
   <li><p>If there is only one keyword mapping to the given state, then it is that keyword.</p></li>

   <li><p>If there is only one <em>conforming</em> keyword mapping to the given state, then it is
   that conforming keyword.</p></li>

   <li><p>Otherwise, the canonical keyword for the state will be explicitly given in the
   specification for the attribute.</p></li>
  </ul>



  <h4>Numbers</h4>

  <h5>Signed integers</h5>

  <p>A string is a <dfn>valid integer</dfn> if it consists of one or more <span>ASCII digits</span>,
  optionally prefixed with a U+002D HYPHEN-MINUS character (-).</p>

  <p>A <span>valid integer</span> without a U+002D HYPHEN-MINUS (-) prefix represents the number
  that is represented in base ten by that string of digits. A <span>valid integer</span>
  <em>with</em> a U+002D HYPHEN-MINUS (-) prefix represents the number represented in base ten by
  the string of digits that follows the U+002D HYPHEN-MINUS, subtracted from zero.</p>

  <div w-nodev>

  <p>The <dfn>rules for parsing integers</dfn> are as given in the following algorithm. When
  invoked, the steps must be followed in the order given, aborting at the first step that returns a
  value. This algorithm will return either an integer or an error.</p>

  <ol>
   <li><p>Let <var>input</var> be the string being parsed.</p></li>

   <li><p>Let <var>position</var> be a pointer into <var>input</var>, initially pointing at the
   start of the string.</p></li>

   <li><p>Let <var>sign</var> have the value "positive".</p></li>

   <li><p><span>Skip ASCII whitespace</span> within <var>input</var> given
   <var>position</var>.</p></li>

   <li><p>If <var>position</var> is past the end of <var>input</var>, return an error.</p></li>

   <li>
    <p>If the character indicated by <var>position</var> (the first character) is a U+002D
    HYPHEN-MINUS character (-):</p>

    <ol>
     <li>Let <var>sign</var> be "negative".</li>

     <li>Advance <var>position</var> to the next character.</li>

     <li>If <var>position</var> is past the end of <var>input</var>, return an error.</li>
    </ol>

    <p>Otherwise, if the character indicated by <var>position</var> (the first character) is a
    U+002B PLUS SIGN character (+):</p>

    <ol>
     <li>Advance <var>position</var> to the next character. (The "<code data-x="">+</code>" is
     ignored, but it is not conforming.)</li>

     <li>If <var>position</var> is past the end of <var>input</var>, return an error.</li>
    </ol>
   </li>

   <li><p>If the character indicated by <var>position</var> is not an <span data-x="ASCII
   digits">ASCII digit</span>, then return an error.</p></li>

   <!-- Ok. At this point we know we have a number. It might have trailing garbage which we'll
   ignore, but it's a number, and we won't return an error. -->

   <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
   <var>input</var> given <var>position</var>, and interpret the resulting sequence as a base-ten
   integer. Let <var>value</var> be that integer.</p></li>

   <li><p>If <var>sign</var> is "positive", return <var>value</var>, otherwise return the result of
   subtracting <var>value</var> from zero.</p></li>
  </ol>

  </div>


  <h5>Non-negative integers</h5>

  <p>A string is a <dfn>valid non-negative integer</dfn> if it consists of one or more <span>ASCII
  digits</span>.</p>

  <p>A <span>valid non-negative integer</span> represents the number that is represented in base ten
  by that string of digits.</p>

  <div w-nodev>

  <p>The <dfn>rules for parsing non-negative integers</dfn> are as given in the following algorithm.
  When invoked, the steps must be followed in the order given, aborting at the first step that
  returns a value. This algorithm will return either zero, a positive integer, or an error.</p>

  <ol>
   <li><p>Let <var>input</var> be the string being parsed.</p></li>

   <li><p>Let <var>value</var> be the result of parsing <var>input</var> using the
   <span>rules for parsing integers</span>.</p></li>

   <li><p>If <var>value</var> is an error, return an error.</p></li>

   <li><p>If <var>value</var> is less than zero, return an error.</p></li>

   <li><p>Return <var>value</var>.</p></li>
  </ol>

  <!-- Implications: A leading + is ignored. A leading - is ignored if the value is zero. -->

  </div>


  <h5>Floating-point numbers</h5>

  <p>A string is a <dfn>valid floating-point number</dfn> if it consists of:</p>

  <ol>
   <li><p>Optionally, a U+002D HYPHEN-MINUS character (-).</p></li>

   <li>
    <p>One or both of the following, in the given order:</p>

    <ol>
     <li><p>A series of one or more <span>ASCII digits</span>.</p></li>

     <li>
      <p>Both of the following, in the given order:</p>

      <ol>
       <li><p>A single U+002E FULL STOP character (.).</p></li>

       <li><p>A series of one or more <span>ASCII digits</span>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li>
    <p>Optionally:</p>

    <ol>
     <li><p>Either a U+0065 LATIN SMALL LETTER E character (e) or a U+0045 LATIN CAPITAL LETTER E
     character (E).</p></li>

     <li><p>Optionally, a U+002D HYPHEN-MINUS character (-) or U+002B PLUS SIGN character
     (+).</p></li>

     <li><p>A series of one or more <span>ASCII digits</span>.</p></li>
    </ol>
   </li>
  </ol>

  <p>A <span>valid floating-point number</span> represents the number obtained by multiplying the
  significand by ten raised to the power of the exponent, where the significand is the first number,
  interpreted as base ten (including the decimal point and the number after the decimal point, if
  any, and interpreting the significand as a negative number if the whole string starts with a
  U+002D HYPHEN-MINUS character (-) and the number is not zero), and where the exponent is the
  number after the E, if any (interpreted as a negative number if there is a U+002D HYPHEN-MINUS
  character (-) between the E and the number and the number is not zero, or else ignoring a U+002B
  PLUS SIGN character (+) between the E and the number if there is one). If there is no E, then the
  exponent is treated as zero.</p>

  <p class="note">The Infinity and Not-a-Number (NaN) values are not <span data-x="valid
  floating-point number">valid floating-point numbers</span>.</p>

  <div w-nodev>

  <p class="note">The <span>valid floating-point number</span> concept is typically only used to
  restrict what is allowed for authors, while the user agent requirements use the <span>rules for
  parsing floating-point number values</span> below (e.g., the <code
  data-x="attr-progress-max">max</code> attribute of the <code>progress</code> element). However, in
  some cases the user agent requirements include checking if a string is a <span>valid
  floating-point number</span> (e.g., the <span>value sanitization algorithm</span> for the <span
  data-x="attr-input-type-number">Number</span> state of the <code>input</code> element, or the
  <span>parse a srcset attribute</span> algorithm).</p>

  <p>The <dfn data-x="best representation of the number as a floating-point number">best
  representation of the number <var>n</var> as a floating-point number</dfn> is the string
  obtained from running <span>ToString</span>(<var>n</var>). The abstract operation
  <span>ToString</span> is not uniquely determined. When there are multiple possible strings that
  could be obtained from <span>ToString</span> for a particular value, the user agent must always
  return the same string for that value (though it may differ from the value used by other user
  agents).</p>

  <p>The <dfn export>rules for parsing floating-point number values</dfn> are as given in the
  following algorithm. This algorithm must be aborted at the first step that returns something.
  This algorithm will return either a number or an error.</p>

  <ol>
   <li><p>Let <var>input</var> be the string being parsed.</p></li>

   <li><p>Let <var>position</var> be a pointer into <var>input</var>, initially pointing at the
   start of the string.</p></li>

   <li><p>Let <var>value</var> have the value 1.</li>

   <li><p>Let <var>divisor</var> have the value 1.</p></li>

   <li><p>Let <var>exponent</var> have the value 1.</p></li>

   <li><p><span>Skip ASCII whitespace</span> within <var>input</var> given
   <var>position</var>.</p></li>

   <li><p>If <var>position</var> is past the end of <var>input</var>, return an error.</p></li>

   <li>
    <p>If the character indicated by <var>position</var> is a U+002D HYPHEN-MINUS character (-):</p>

    <ol>
     <li>Change <var>value</var> and <var>divisor</var> to &#x2212;1.</li>

     <li>Advance <var>position</var> to the next character.</li>

     <li>If <var>position</var> is past the end of <var>input</var>, return an error.</li>
    </ol>

    <p>Otherwise, if the character indicated by <var>position</var> (the first character) is a
    U+002B PLUS SIGN character (+):</p>

    <ol>
     <li>Advance <var>position</var> to the next character. (The "<code data-x="">+</code>"
     is ignored, but it is not conforming.)</li>

     <li>If <var>position</var> is past the end of <var>input</var>, return an error.</li>
    </ol>
   </li>

   <li><p>If the character indicated by <var>position</var> is a U+002E FULL STOP (.), and that is
   not the last character in <var>input</var>, and the character after the character indicated by
   <var>position</var> is an <span data-x="ASCII digits">ASCII digit</span>, then set
   <var>value</var> to zero and jump to the step labeled <i>fraction</i>.</p> <!-- we have to check
   there's a number so that ".e1" fails to parse but ".0" does not -->

   <li><p>If the character indicated by <var>position</var> is not an <span data-x="ASCII
   digits">ASCII digit</span>, then return an error.</p></li>

   <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
   <var>input</var> given <var>position</var>, and interpret the resulting sequence as a base-ten
   integer. Multiply <var>value</var> by that integer.</p></li>

   <li>If <var>position</var> is past the end of <var>input</var>, jump to the step labeled
   <i>conversion</i>.</li>

   <li><p><i>Fraction</i>: If the character indicated by <var>position</var> is a U+002E
   FULL STOP (.), run these substeps:</p>

    <ol>
     <li><p>Advance <var>position</var> to the next character.</p></li>

     <li><p>If <var>position</var> is past the end of <var>input</var>, or if the character
     indicated by <var>position</var> is not an <span data-x="ASCII digits">ASCII digit</span>,
     U+0065 LATIN SMALL LETTER E (e), or U+0045 LATIN CAPITAL LETTER E (E), then jump to the step
     labeled <i>conversion</i>.</li>

     <li><p>If the character indicated by <var>position</var> is a U+0065 LATIN SMALL
     LETTER E character (e) or a U+0045 LATIN CAPITAL LETTER E character (E), skip the remainder of
     these substeps.</p>

     <li><p><i>Fraction loop</i>: Multiply <var>divisor</var> by ten.</p></li>

     <li>Add the value of the character indicated by <var>position</var>, interpreted as a
     base-ten digit (0..9) and divided by <var>divisor</var>, to <var>value</var>.</li>

     <li><p>Advance <var>position</var> to the next character.</p></li>

     <li><p>If <var>position</var> is past the end of <var>input</var>, then jump to the step
     labeled <i>conversion</i>.</li>

     <li><p>If the character indicated by <var>position</var> is an <span data-x="ASCII
     digits">ASCII digit</span>, jump back to the step labeled <i>fraction loop</i> in these
     substeps.</p></li>
    </ol>
   </li>

   <li><p>If the character indicated by <var>position</var> is U+0065 (e) or a U+0045 (E), then:</p>

    <ol>
     <li><p>Advance <var>position</var> to the next character.</p></li>

     <li><p>If <var>position</var> is past the end of <var>input</var>, then jump to the step
     labeled <i>conversion</i>.</li>

     <li>
      <p>If the character indicated by <var>position</var> is a U+002D HYPHEN-MINUS character
      (-):</p>

      <ol>
       <li>Change <var>exponent</var> to &#x2212;1.</li>

       <li>Advance <var>position</var> to the next character.</li>

       <li><p>If <var>position</var> is past the end of <var>input</var>, then jump to the step
       labeled <i>conversion</i>.</li>
      </ol>

      <p>Otherwise, if the character indicated by <var>position</var> is a U+002B PLUS SIGN
      character (+):</p>

      <ol>
       <li>Advance <var>position</var> to the next character.</li>

       <li><p>If <var>position</var> is past the end of <var>input</var>, then jump to the step
       labeled <i>conversion</i>.</li>
      </ol>
     </li>

     <li><p>If the character indicated by <var>position</var> is not an <span data-x="ASCII
     digits">ASCII digit</span>, then jump to the step labeled <i>conversion</i>.</li>

     <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
     <var>input</var> given <var>position</var>, and interpret the resulting sequence as a base-ten
     integer. Multiply <var>exponent</var> by that integer.</p></li>

     <li><p>Multiply <var>value</var> by ten raised to the <var>exponent</var>th power.</p></li>
    </ol>
   </li>

   <li><p><i>Conversion</i>: Let <var>S</var> be the set of finite IEEE 754 double-precision
   floating-point values except &#x2212;0, but with two special values added: 2<sup>1024</sup> and &#x2212;2<sup>1024</sup>.</p></li>

   <li><p>Let <var>rounded-value</var> be the number in <var>S</var> that is closest to
   <var>value</var>, selecting the number with an even significand if there are two equally close
   values. (The two special values 2<sup>1024</sup> and &#x2212;2<sup>1024</sup>
   are considered to have even significands for this purpose.)</p></li>

   <li><p>If <var>rounded-value</var> is 2<sup>1024</sup> or &#x2212;2<sup>1024</sup>, return an error.</p></li>

   <li><p>Return <var>rounded-value</var>.</p></li>
  </ol>

  </div>


<div w-nodev>
  <h5 id="percentages-and-dimensions">Percentages and lengths</h5>

  <p>The <dfn>rules for parsing dimension values</dfn> are as given in the following algorithm. When
  invoked, the steps must be followed in the order given, aborting at the first step that returns a
  value. This algorithm will return either a number greater than or equal to 0.0, or failure; if a
  number is returned, then it is further categorized as either a percentage or a length.</p>

  <ol>
   <li><p>Let <var>input</var> be the string being parsed.</p></li>

   <li><p>Let <var>position</var> be a <span>position variable</span> for <var>input</var>,
   initially pointing at the start of <var>input</var>.</p></li>

   <li><p><span>Skip ASCII whitespace</span> within <var>input</var> given
   <var>position</var>.</p></li>

   <li><p>If <var>position</var> is past the end of <var>input</var> or the code point at
   <var>position</var> within <var>input</var> is not an <span data-x="ASCII digits">ASCII
   digit</span>, then return failure.</p></li>

   <!-- Ok. At this point we know we have a number. It might have trailing garbage which we'll
   ignore, but it's a number, and we won't return an error. -->

   <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
   <var>input</var> given <var>position</var>, and interpret the resulting sequence as a base-ten
   integer. Let <var>value</var> be that number.</p></li>

   <li><p>If <var>position</var> is past the end of <var>input</var>, then return <var>value</var>
   as a length.</p></li>

   <li>
    <p>If the code point at <var>position</var> within <var>input</var> is U+002E (.), then:</p>

    <ol>
     <li><p>Advance <var>position</var> by 1.</p></li>

     <li><p>If <var>position</var> is past the end of <var>input</var> or the code point at
     <var>position</var> within <var>input</var> is not an <span data-x="ASCII digits">ASCII
     digit</span>, then return the <span>current dimension value</span> with <var>value</var>,
     <var>input</var>, and <var>position</var>.</p></li>

     <li><p>Let <var>divisor</var> have the value 1.</p></li>

     <li>
      <p>While true:</p>

      <ol>
       <li><p>Multiply <var>divisor</var> by ten.</p></li>

       <li><p>Add the value of the code point at <var>position</var> within <var>input</var>,
       interpreted as a base-ten digit (0..9) and divided by <var>divisor</var>, to
       <var>value</var>.</p></li>

       <li><p>Advance <var>position</var> by 1.</p></li>

       <li><p>If <var>position</var> is past the end of <var>input</var>, then return
       <var>value</var> as a length.</p></li>

       <li><p>If the code point at <var>position</var> within <var>input</var> is not an <span
       data-x="ASCII digits">ASCII digit</span>, then <span>break</span>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Return the <span>current dimension value</span> with <var>value</var>, <var>input</var>,
   and <var>position</var>.</p></li>
  </ol>

  <p>The <dfn>current dimension value</dfn>, given <var>value</var>, <var>input</var>, and
  <var>position</var>, is determined as follows:</p>

  <ol>
   <li><p>If <var>position</var> is past the end of <var>input</var>, then return <var>value</var>
   as a length.</p></li>

   <li><p>If the code point at <var>position</var> within <var>input</var> is U+0025 (%), then
   return <var>value</var> as a percentage.</p></li>

   <li><p>Return <var>value</var> as a length.</p></li>
  </ol>

  <h5 id="nonzero-percentages-and-lengths">Nonzero percentages and lengths</h5>

  <p>The <dfn id="rules-for-parsing-non-zero-dimension-values">rules for parsing nonzero dimension
  values</dfn> are as given in the following algorithm. When invoked, the steps must be followed in
  the order given, aborting at the first step that returns a value. This algorithm will return
  either a number greater than 0.0, or an error; if a number is returned, then it is further
  categorized as either a percentage or a length.</p>

  <ol>
   <li><p>Let <var>input</var> be the string being parsed.</p></li>

   <li><p>Let <var>value</var> be the result of parsing <var>input</var> using the <span>rules for
   parsing dimension values</span>.</p></li>

   <li><p>If <var>value</var> is an error, return an error.</p></li>

   <li><p>If <var>value</var> is zero, return an error.</p></li>

   <li><p>If <var>value</var> is a percentage, return <var>value</var> as a percentage.</p></li>

   <li><p>Return <var>value</var> as a length.</p></li>
  </ol>

  </div>


  <h5>Lists of floating-point numbers</h5>

  <p>A <dfn>valid list of floating-point numbers</dfn> is a number of <span data-x="valid
  floating-point number">valid floating-point numbers</span> separated by U+002C COMMA characters,
  with no other characters (e.g. no <span>ASCII whitespace</span>). In addition, there might be
  restrictions on the number of floating-point numbers that can be given, or on the range of values
  allowed.</p>

  <div w-nodev>

  <p>The <dfn>rules for parsing a list of floating-point numbers</dfn> are as follows:</p>

  <ol>
   <li><p>Let <var>input</var> be the string being parsed.</p></li>

   <li><p>Let <var>position</var> be a pointer into <var>input</var>, initially pointing at the
   start of the string.</p></li>

   <li><p>Let <var>numbers</var> be an initially empty list of floating-point numbers. This list
   will be the result of this algorithm.</p></li>

   <li><p><span>Collect a sequence of code points</span> that are <span>ASCII whitespace</span>,
   U+002C COMMA, or U+003B SEMICOLON characters from <var>input</var> given <var>position</var>.
   This skips past any leading delimiters.</p></li>

   <li><p>While <var>position</var> is not past the end of <var>input</var>:</p>

    <ol>
     <li><p><span>Collect a sequence of code points</span> that are not <span>ASCII
     whitespace</span>, U+002C COMMA, U+003B SEMICOLON, <span>ASCII digits</span>, U+002E FULL STOP,
     or U+002D HYPHEN-MINUS characters from <var>input</var> given <var>position</var>. This skips
     past leading garbage.</p></li>

     <li><p><span>Collect a sequence of code points</span> that are not <span>ASCII
     whitespace</span>, U+002C COMMA, or U+003B SEMICOLON characters from <var>input</var> given
     <var>position</var>, and let <var>unparsed number</var> be the result.</p></li>

     <li><p>Let <var>number</var> be the result of parsing <var>unparsed number</var> using the
     <span>rules for parsing floating-point number values</span>.</p></li>

     <li><p>If <var>number</var> is an error, set <var>number</var> to zero.</p></li>

     <li><p>Append <var>number</var> to <var>numbers</var>.</p></li>

     <li><p><span>Collect a sequence of code points</span> that are <span>ASCII whitespace</span>,
     U+002C COMMA, or U+003B SEMICOLON characters from <var>input</var> given <var>position</var>.
     This skips past the delimiter.</p></li>
    </ol>
   </li>

   <li><p>Return <var>numbers</var>.</p></li>
  </ol>

  </div>

  <div w-nodev>

  <h5>Lists of dimensions</h5>

  <!-- no definition of a type since no conforming feature uses this syntax (it's only used in
  cols="" and rows="" on <frameset> elements -->

  <p>The <dfn>rules for parsing a list of dimensions</dfn> are as follows. These rules return a list
  of zero or more pairs consisting of a number and a unit, the unit being one of <i>percentage</i>,
  <i>relative</i>, and <i>absolute</i>.</p>

  <ol>
   <li><p>Let <var>raw input</var> be the string being parsed.</p></li>

   <li><p>If the last character in <var>raw input</var> is a U+002C COMMA character (,),
   then remove that character from <var>raw input</var>.</p></li>

   <li><p><span data-x="split a string on commas">Split the string <var>raw input</var> on
   commas</span>. Let <var>raw tokens</var> be the resulting list of tokens.</p></li>

   <li><p>Let <var>result</var> be an empty list of number/unit pairs.</p></li>

   <li>
    <p>For each token in <var>raw tokens</var>, run the following substeps:</p>

    <ol>
     <li><p>Let <var>input</var> be the token.</p></li>

     <li><p>Let <var>position</var> be a pointer into <var>input</var>,
     initially pointing at the start of the string.</p></li>

     <li><p>Let <var>value</var> be the number 0.</p></li>

     <li><p>Let <var>unit</var> be <i>absolute</i>.</p></li>

     <li><p>If <var>position</var> is past the end of <var>input</var>, set <var>unit</var> to <i>relative</i> and jump to the last substep.</p></li>

     <li><p>If the character at <var>position</var> is an <span data-x="ASCII
     digits">ASCII digit</span>, <span>collect a sequence of code points</span> that are <span>ASCII
     digits</span> from <var>input</var> given <var>position</var>, interpret the resulting sequence
     as an integer in base ten, and increment <var>value</var> by that integer.</p></li>

     <li>
      <p>If the character at <var>position</var> is U+002E (.), then:</p>

      <ol>
       <li><p><span>Collect a sequence of code points</span> consisting of <span>ASCII
       whitespace</span> and <span>ASCII digits</span> from <var>input</var> given
       <var>position</var>. Let <var>s</var> be the resulting sequence.</p></li>

       <li><p>Remove all <span>ASCII whitespace</span> in <var>s</var>.</p></li>

       <li>
        <p>If <var>s</var> is not the empty string, then:</p>

        <ol>
         <li><p>Let <var>length</var> be the number of characters in <var>s</var> (after the spaces were removed).</p></li>

         <li><p>Let <var>fraction</var> be the result of interpreting <var>s</var> as a base-ten integer, and then dividing that number by <span
         data-x="">10<sup><var>length</var></sup></span>.</li>

         <li><p>Increment <var>value</var> by <var>fraction</var>.</p></li>
        </ol>
       </li>
      </ol>
     </li>

     <li><p><span>Skip ASCII whitespace</span> within <var>input</var> given
     <var>position</var>.</p></li>

     <li>
      <p>If the character at <var>position</var> is a U+0025 PERCENT SIGN character (%),
      then set <var>unit</var> to <i>percentage</i>.</p>

      <p>Otherwise, if the character at <var>position</var> is a U+002A ASTERISK character
      (*), then set <var>unit</var> to <i>relative</i>.</p>
     </li>

     <!-- the remaining characters in /input/ are ignored -->

     <li><p>Add an entry to <var>result</var> consisting of the number given by <var>value</var> and the unit given by <var>unit</var>.</p></li>
    </ol>
   </li>

   <li><p>Return the list <var>result</var>.</p></li>
  </ol>

  </div>


  <h4>Dates and times</h4>

  <p>In the algorithms below, the <dfn>number of days in month <var>month</var> of year
  <var>year</var></dfn> is: <em>31</em> if <var>month</var> is 1, 3, 5, 7, 8,
  10, or 12; <em>30</em> if <var>month</var> is 4, 6, 9, or 11; <em>29</em> if <var>month</var> is 2
  and <var>year</var> is a number divisible by 400, or if <var>year</var> is a number divisible by 4
  but not by 100; and <em>28</em> otherwise. This takes into account leap years in the Gregorian
  calendar. <ref>GREGORIAN</ref></p>

  <p>When <span>ASCII digits</span> are used in the date and time syntaxes defined in this section,
  they express numbers in base ten.</p>

  <div w-nodev>

  <p class="note">While the formats described here are intended to be subsets of the corresponding
  ISO8601 formats, this specification defines parsing rules in much more detail than ISO8601.
  Implementers are therefore encouraged to carefully examine any date parsing libraries before using
  them to implement the parsing rules described below; ISO8601 libraries might not parse dates and
  times in exactly the same manner. <ref>ISO8601</ref></p>

  </div>

  <p>Where this specification refers to the <dfn>proleptic Gregorian calendar</dfn>, it means the
  modern Gregorian calendar, extrapolated backwards to year 1. A date in the <span>proleptic
  Gregorian calendar</span>, sometimes explicitly referred to as a <dfn>proleptic-Gregorian
  date</dfn>, is one that is described using that calendar even if that calendar was not in use at
  the time (or place) in question. <ref>GREGORIAN</ref></p>

  <p class="note">The use of the Gregorian calendar as the wire format in this specification is an
  arbitrary choice resulting from the cultural biases of those involved in the decision. See also
  the section discussing <a href="#input-author-notes">date, time, and number formats</a> in forms
  <span w-nodev>(for authors), <a href="#input-impl-notes">implementation notes regarding
  localization of form controls</a>,</span> and the <code>time</code> element.</p>


  <h5>Months</h5>

  <p>A <dfn data-x="concept-month">month</dfn> consists of a specific <span>proleptic-Gregorian
  date</span> with no time-zone information and no date information beyond a year and a month.
  <ref>GREGORIAN</ref></p>

  <p>A string is a <dfn>valid month string</dfn> representing a year <var>year</var> and
  month <var>month</var> if it consists of the following components in the given order:</p>

  <ol>
   <li>Four or more <span>ASCII digits</span>, representing <var>year</var>, where <var>year</var>&nbsp;&gt;&nbsp;0</li>

   <li>A U+002D HYPHEN-MINUS character (-)</li>

   <li>Two <span>ASCII digits</span>, representing the month <var>month</var>, in the range
   1&nbsp;&le;&nbsp;<var>month</var>&nbsp;&le;&nbsp;12</li>
  </ol>

  <div w-nodev>

  <p>The rules to <dfn>parse a month string</dfn> are as follows. This will return either a year and
  month, or nothing. If at any point the algorithm says that it "fails", this means that it is
  aborted at that point and returns nothing.</p>

  <ol>
   <li><p>Let <var>input</var> be the string being parsed.</p></li>

   <li><p>Let <var>position</var> be a pointer into <var>input</var>, initially
   pointing at the start of the string.</p></li>

   <li><p><span>Parse a month component</span> to obtain <var>year</var> and <var>month</var>. If this returns nothing, then fail.</p>

   <li><p>If <var>position</var> is <em>not</em> beyond the
   end of <var>input</var>, then fail.</p></li>

   <li><p>Return <var>year</var> and <var>month</var>.</p></li>
  </ol>

  <p>The rules to <dfn>parse a month component</dfn>, given an <var>input</var> string and
  a <var>position</var>, are as follows. This will return either a year and a month, or
  nothing. If at any point the algorithm says that it "fails", this means that it is aborted at that
  point and returns nothing.</p>

  <ol>
   <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
   <var>input</var> given <var>position</var>. If the collected sequence is not at least four
   characters long, then fail. Otherwise, interpret the resulting sequence as a base-ten integer.
   Let that number be the <var>year</var>.</p></li>

   <li><p>If <var>year</var> is not a number greater than zero, then fail.</p></li>

   <li><p>If <var>position</var> is beyond the end of <var>input</var> or if the
   character at <var>position</var> is not a U+002D HYPHEN-MINUS character, then fail.
   Otherwise, move <var>position</var> forwards one character.</p></li>

   <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
   <var>input</var> given <var>position</var>. If the collected sequence is not exactly two
   characters long, then fail. Otherwise, interpret the resulting sequence as a base-ten integer.
   Let that number be the <var>month</var>.</p></li>

   <li><p>If <var>month</var> is not a number in the range 1&nbsp;&le;&nbsp;<var>month</var>&nbsp;&le;&nbsp;12, then fail.</p></li>

   <li><p>Return <var>year</var> and <var>month</var>.</p></li>
  </ol>

  </div>


  <h5>Dates</h5>

  <p>A <dfn data-x="concept-date">date</dfn> consists of a specific <span>proleptic-Gregorian
  date</span> with no time-zone information, consisting of a year, a month, and a day.
  <ref>GREGORIAN</ref></p>

  <p>A string is a <dfn>valid date string</dfn> representing a year <var>year</var>, month
  <var>month</var>, and day <var>day</var> if it consists of the following
  components in the given order:</p>

  <ol>
   <li>A <span>valid month string</span>, representing <var>year</var> and <var>month</var></li>

   <li>A U+002D HYPHEN-MINUS character (-)</li>

   <li>Two <span>ASCII digits</span>, representing <var>day</var>, in the range
   1&nbsp;&le;&nbsp;<var>day</var>&nbsp;&le;&nbsp;<var>maxday</var> where <var>maxday</var> is the <span data-x="number of days in month month of year year">number of
   days in the month <var>month</var> and year <var>year</var></span></li>
  </ol>

  <div w-nodev>

  <p>The rules to <dfn>parse a date string</dfn> are as follows. This will return either a date, or
  nothing. If at any point the algorithm says that it "fails", this means that it is aborted at that
  point and returns nothing.</p>

  <ol>
   <li><p>Let <var>input</var> be the string being parsed.</p></li>

   <li><p>Let <var>position</var> be a pointer into <var>input</var>, initially
   pointing at the start of the string.</p></li>

   <li><p><span>Parse a date component</span> to obtain <var>year</var>, <var>month</var>, and <var>day</var>. If this returns nothing, then fail.</p>

   <li><p>If <var>position</var> is <em>not</em> beyond the end of <var>input</var>, then fail.</p></li>

   <li><p>Let <var>date</var> be the date with year <var>year</var>, month <var>month</var>, and day <var>day</var>.</p></li>

   <li><p>Return <var>date</var>.</p></li>
  </ol>

  <p>The rules to <dfn>parse a date component</dfn>, given an <var>input</var> string and a
  <var>position</var>, are as follows. This will return either a year, a month, and a day,
  or nothing. If at any point the algorithm says that it "fails", this means that it is aborted at
  that point and returns nothing.</p>

  <ol>
   <li><p><span>Parse a month component</span> to obtain <var>year</var> and <var>month</var>. If this returns nothing, then fail.</li>

   <li><p>Let <var>maxday</var> be the <span>number of days in month <var>month</var> of year <var>year</var></span>.</p></li>

   <li><p>If <var>position</var> is beyond the end of <var>input</var> or if the
   character at <var>position</var> is not a U+002D HYPHEN-MINUS character, then fail.
   Otherwise, move <var>position</var> forwards one character.</p></li>

   <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
   <var>input</var> given <var>position</var>. If the collected sequence is not exactly two
   characters long, then fail. Otherwise, interpret the resulting sequence as a base-ten integer.
   Let that number be the <var>day</var>.</p></li>

   <li><p>If <var>day</var> is not a number in the range 1&nbsp;&le;&nbsp;<var>day</var>&nbsp;&le;&nbsp;<var>maxday</var>, then fail.</li>

   <li><p>Return <var>year</var>, <var>month</var>, and <var>day</var>.</p></li>
  </ol>

  </div>


  <h5>Yearless dates</h5>

  <p>A <dfn data-x="concept-yearless-date">yearless date</dfn> consists of a Gregorian month and a
  day within that month, but with no associated year. <ref>GREGORIAN</ref></p>

  <p>A string is a <dfn>valid yearless date string</dfn> representing a month <var>month</var> and a day <var>day</var> if it consists of the following components
  in the given order:</p>

  <ol>
   <li>Optionally, two U+002D HYPHEN-MINUS characters (-)</li>

   <li>Two <span>ASCII digits</span>, representing the month <var>month</var>, in the range
   1&nbsp;&le;&nbsp;<var>month</var>&nbsp;&le;&nbsp;12</li>

   <li>A U+002D HYPHEN-MINUS character (-)</li>

   <li>Two <span>ASCII digits</span>, representing <var>day</var>, in the range
   1&nbsp;&le;&nbsp;<var>day</var>&nbsp;&le;&nbsp;<var>maxday</var> where <var>maxday</var> is the <span data-x="number of days in month month of year year">number of
   days</span> in the month <var>month</var> and any arbitrary leap year (e.g. 4 or
   2000)</li>
  </ol>

  <p class="note">In other words, if the <var>month</var> is "<code data-x="">02</code>",
  meaning February, then the day can be 29, as if the year was a leap year.</p>

  <div w-nodev>

  <p>The rules to <dfn>parse a yearless date string</dfn> are as follows. This will return either a
  month and a day, or nothing. If at any point the algorithm says that it "fails", this means that
  it is aborted at that point and returns nothing.</p>

  <ol>
   <li><p>Let <var>input</var> be the string being parsed.</p></li>

   <li><p>Let <var>position</var> be a pointer into <var>input</var>, initially
   pointing at the start of the string.</p></li>

   <li><p><span>Parse a yearless date component</span> to obtain <var>month</var> and <var>day</var>. If this returns nothing, then fail.</p>

   <li><p>If <var>position</var> is <em>not</em> beyond the end of <var>input</var>, then fail.</p></li>

   <li><p>Return <var>month</var> and <var>day</var>.</p></li>
  </ol>

  <p>The rules to <dfn>parse a yearless date component</dfn>, given an <var>input</var>
  string and a <var>position</var>, are as follows. This will return either a month and a
  day, or nothing. If at any point the algorithm says that it "fails", this means that it is aborted
  at that point and returns nothing.</p>

  <ol>
   <li><p><span>Collect a sequence of code points</span> that are U+002D HYPHEN-MINUS characters (-)
   from <var>input</var> given <var>position</var>. If the collected sequence is not exactly zero or
   two characters long, then fail.</p></li>

   <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
   <var>input</var> given <var>position</var>. If the collected sequence is not exactly two
   characters long, then fail. Otherwise, interpret the resulting sequence as a base-ten integer.
   Let that number be the <var>month</var>.</p></li>

   <li><p>If <var>month</var> is not a number in the range 1&nbsp;&le;&nbsp;<var>month</var>&nbsp;&le;&nbsp;12, then fail.</p></li>

   <li><p>Let <var>maxday</var> be the <span data-x="number of days in month month of year
   year">number of days</span> in month <var>month</var> of any arbitrary leap year (e.g. 4
   or 2000).</p></li>

   <li><p>If <var>position</var> is beyond the end of <var>input</var> or if the
   character at <var>position</var> is not a U+002D HYPHEN-MINUS character, then fail.
   Otherwise, move <var>position</var> forwards one character.</p></li>

   <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
   <var>input</var> given <var>position</var>. If the collected sequence is not exactly two
   characters long, then fail. Otherwise, interpret the resulting sequence as a base-ten integer.
   Let that number be the <var>day</var>.</p></li>

   <li><p>If <var>day</var> is not a number in the range 1&nbsp;&le;&nbsp;<var>day</var>&nbsp;&le;&nbsp;<var>maxday</var>, then fail.</li>

   <li><p>Return <var>month</var> and <var>day</var>.</p></li>
  </ol>

  </div>


  <h5>Times</h5>

  <p>A <dfn data-x="concept-time">time</dfn> consists of a specific time with no time-zone
  information, consisting of an hour, a minute, a second, and a fraction of a second.</p>

  <p>A string is a <dfn>valid time string</dfn> representing an hour <var>hour</var>, a
  minute <var>minute</var>, and a second <var>second</var> if it consists of the
  following components in the given order:</p>

  <ol>
   <li>Two <span>ASCII digits</span>, representing <var>hour</var>, in the range
   0&nbsp;&le;&nbsp;<var>hour</var>&nbsp;&le;&nbsp;23</li>

   <li>A U+003A COLON character (:)</li>

   <li>Two <span>ASCII digits</span>, representing <var>minute</var>, in the range
   0&nbsp;&le;&nbsp;<var>minute</var>&nbsp;&le;&nbsp;59</li>

   <li>If <var>second</var> is nonzero, or optionally if <var>second</var> is
   zero:

    <ol>
     <li>A U+003A COLON character (:)</li>

     <li>Two <span>ASCII digits</span>, representing the integer part of <var>second</var>,
     in the range 0&nbsp;&le;&nbsp;<var>s</var>&nbsp;&le;&nbsp;59</li>

     <li>If <var>second</var> is not an integer, or optionally if <var>second</var> is an integer:

      <ol>
       <li>A U+002E FULL STOP character (.)</li>

       <li>One, two, or three <span>ASCII digits</span>, representing the fractional part of <var>second</var></li>
      </ol>
     </li>
    </ol>
   </li>
  </ol>

  <p class="note">The <var>second</var> component cannot be 60 or 61; leap seconds cannot
  be represented.</p>

  <div w-nodev>

  <p>The rules to <dfn>parse a time string</dfn> are as follows. This will return either a time, or
  nothing. If at any point the algorithm says that it "fails", this means that it is aborted at that
  point and returns nothing.</p>

  <ol>
   <li><p>Let <var>input</var> be the string being parsed.</p></li>

   <li><p>Let <var>position</var> be a pointer into <var>input</var>, initially
   pointing at the start of the string.</p></li>

   <li><p><span>Parse a time component</span> to obtain <var>hour</var>, <var>minute</var>, and <var>second</var>. If this returns nothing, then fail.</p>

   <li><p>If <var>position</var> is <em>not</em> beyond the end of <var>input</var>, then fail.</p></li>

   <li><p>Let <var>time</var> be the time with hour <var>hour</var>, minute <var>minute</var>, and second <var>second</var>.</p></li>

   <li><p>Return <var>time</var>.</p></li>
  </ol>

  <p>The rules to <dfn>parse a time component</dfn>, given an <var>input</var> string and a
  <var>position</var>, are as follows. This will return either an hour, a minute, and a
  second, or nothing. If at any point the algorithm says that it "fails", this means that it is
  aborted at that point and returns nothing.</p>

  <ol>
   <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
   <var>input</var> given <var>position</var>. If the collected sequence is not exactly two
   characters long, then fail. Otherwise, interpret the resulting sequence as a base-ten integer.
   Let that number be the <var>hour</var>.</p></li>

   <li>If <var>hour</var> is not a number in the range 0&nbsp;&le;&nbsp;<var>hour</var>&nbsp;&le;&nbsp;23, then fail.</li>

   <li><p>If <var>position</var> is beyond the end of <var>input</var> or if the
   character at <var>position</var> is not a U+003A COLON character, then fail. Otherwise,
   move <var>position</var> forwards one character.</p></li>

   <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
   <var>input</var> given <var>position</var>. If the collected sequence is not exactly two
   characters long, then fail. Otherwise, interpret the resulting sequence as a base-ten integer.
   Let that number be the <var>minute</var>.</p></li>

   <li>If <var>minute</var> is not a number in the range 0&nbsp;&le;&nbsp;<var>minute</var>&nbsp;&le;&nbsp;59, then fail.</li>

   <li><p>Let <var>second</var> be 0.</p></li>

   <li>
    <p>If <var>position</var> is not beyond the end of <var>input</var> and the character at
    <var>position</var> is U+003A (:), then:</p>

    <ol>
     <li><p>Advance <var>position</var> to the next character in <var>input</var>.</p></li>

     <li><p>If <var>position</var> is beyond the end of <var>input</var>, or at
     the last character in <var>input</var>, or if the next <em>two</em> characters in <var>input</var> starting at <var>position</var> are not both <span>ASCII
     digits</span>, then fail.</p></li>

     <li><p><span>Collect a sequence of code points</span> that are either <span>ASCII digits</span>
     or U+002E FULL STOP characters from <var>input</var> given <var>position</var>. If the
     collected sequence is three characters long, or if it is longer than three characters long and
     the third character is not a U+002E FULL STOP character, or if it has more than one U+002E FULL
     STOP character, then fail. Otherwise, interpret the resulting sequence as a base-ten number
     (possibly with a fractional part). Set <var>second</var> to that number.</p></li>

     <li><p>If <var>second</var> is not a number in the range
     0&nbsp;&le;&nbsp;<var>second</var>&nbsp;&lt;&nbsp;60, then fail.</p></li>
    </ol>
   </li>

   <li><p>Return <var>hour</var>, <var>minute</var>, and <var>second</var>.</p></li>
  </ol>

  </div>


  <h5>Local dates and times</h5>

  <p>A <dfn data-x="concept-datetime-local">local date and time</dfn> consists of a specific
  <span>proleptic-Gregorian date</span>, consisting of a year, a month, and a day, and a time,
  consisting of an hour, a minute, a second, and a fraction of a second, but expressed without a
  time zone. <ref>GREGORIAN</ref></p>

  <p>A string is a <dfn>valid local date and time string</dfn> representing a date and time if it
  consists of the following components in the given order:</p>

  <ol>
   <li>A <span>valid date string</span> representing the date</li>

   <li>A U+0054 LATIN CAPITAL LETTER T character (T) or a U+0020 SPACE character</li>

   <li>A <span>valid time string</span> representing the time</li>
  </ol>

  <p>A string is a <!--en-GB--><dfn id="valid-normalised-local-date-and-time-string">valid
  normalized local date and time string</dfn> representing a date and time if it consists of the
  following components in the given order:</p>

  <ol>
   <li>A <span>valid date string</span> representing the date</li>

   <li>A U+0054 LATIN CAPITAL LETTER T character (T)</li>

   <li>A <span>valid time string</span> representing the time, expressed as the shortest possible
   string for the given time (e.g. omitting the seconds component entirely if the given time is zero
   seconds past the minute)</li>
  </ol>

  <div w-nodev>

  <p>The rules to <dfn>parse a local date and time string</dfn> are as follows. This will return
  either a date and time, or nothing. If at any point the algorithm says that it "fails", this means
  that it is aborted at that point and returns nothing.</p>

  <ol>
   <li><p>Let <var>input</var> be the string being parsed.</p></li>

   <li><p>Let <var>position</var> be a pointer into <var>input</var>, initially
   pointing at the start of the string.</p></li>

   <li><p><span>Parse a date component</span> to obtain <var>year</var>, <var>month</var>, and <var>day</var>. If this returns nothing, then fail.</p>

   <li><p>If <var>position</var> is beyond the end of <var>input</var> or if the
   character at <var>position</var> is neither a U+0054 LATIN CAPITAL LETTER T character
   (T) nor a U+0020 SPACE character, then fail. Otherwise, move <var>position</var>
   forwards one character.</p></li>

   <li><p><span>Parse a time component</span> to obtain <var>hour</var>, <var>minute</var>, and <var>second</var>. If this returns nothing, then fail.</p>

   <li><p>If <var>position</var> is <em>not</em> beyond the end of <var>input</var>, then fail.</p></li>

   <li><p>Let <var>date</var> be the date with year <var>year</var>, month <var>month</var>, and day <var>day</var>.</p></li>

   <li><p>Let <var>time</var> be the time with hour <var>hour</var>, minute <var>minute</var>, and second <var>second</var>.</p></li>

   <li><p>Return <var>date</var> and <var>time</var>.</p></li>
  </ol>

  </div>


  <h5>Time zones</h5>

  <p>A <dfn data-x="concept-timezone">time-zone offset</dfn> consists of a signed number of hours and
  minutes.</p>

  <p>A string is a <dfn>valid time-zone offset string</dfn> representing a time-zone offset if it
  consists of either:</p>


  <ul>
   <li><p>A U+005A LATIN CAPITAL LETTER Z character (Z), allowed only if the time zone is
   UTC</p></li>

   <li>
    <p>Or, the following components, in the given order:</p>

    <ol>
     <li>Either a U+002B PLUS SIGN character (+) or, if the time-zone offset is not zero, a U+002D
     HYPHEN-MINUS character (-), representing the sign of the time-zone offset</li>
     <!-- the -00:00 offset is disallowed because RFC3339 gives it a special semantic -->

     <li>Two <span>ASCII digits</span>, representing the hours component <var>hour</var> of
     the time-zone offset, in the range 0&nbsp;&le;&nbsp;<var>hour</var>&nbsp;&le;&nbsp;23</li>

     <li>Optionally, a U+003A COLON character (:)</li>

     <li>Two <span>ASCII digits</span>, representing the minutes component <var>minute</var> of the time-zone offset, in the range 0&nbsp;&le;&nbsp;<var>minute</var>&nbsp;&le;&nbsp;59</li>
    </ol>
   </li>
  </ul>

  <p class="note">This format allows for time-zone offsets from -23:59 to +23:59. Right now, in
  practice, the range of offsets of actual time zones is -12:00 to +14:00, and the minutes component
  of offsets of actual time zones is always either 00, 30, or 45. There is no guarantee that this
  will remain so forever, however, since time zones are used as political footballs and are thus
  subject to very whimsical policy decisions.</p>

  <p class="note">See also the usage notes and examples in the <span data-x="concept-datetime">global
  date and time</span> section below for details on using time-zone offsets with historical times
  that predate the formation of formal time zones.</p>

  <div w-nodev>

  <p>The rules to <dfn>parse a time-zone offset string</dfn> are as follows. This will return either
  a time-zone offset, or nothing. If at any point the algorithm says that it "fails", this means
  that it is aborted at that point and returns nothing.</p>

  <ol>
   <li><p>Let <var>input</var> be the string being parsed.</p></li>

   <li><p>Let <var>position</var> be a pointer into <var>input</var>, initially
   pointing at the start of the string.</p></li>

   <li><p><span>Parse a time-zone offset component</span> to obtain <var>timezone<sub>hours</sub></var> and <var>timezone<sub>minutes</sub></var>. If this
   returns nothing, then fail.</p>

   <li><p>If <var>position</var> is <em>not</em> beyond the end of <var>input</var>, then fail.</p></li>

   <li><p>Return the time-zone offset that is <var>timezone<sub>hours</sub></var>
   hours and <var>timezone<sub>minutes</sub></var> minutes from UTC.</p></li>
  </ol>

  <p>The rules to <dfn>parse a time-zone offset component</dfn>, given an <var>input</var>
  string and a <var>position</var>, are as follows. This will return either time-zone hours
  and time-zone minutes, or nothing. If at any point the algorithm says that it "fails", this means
  that it is aborted at that point and returns nothing.</p>

  <ol>
   <li>
    <p>If the character at <var>position</var> is a U+005A LATIN CAPITAL LETTER Z character
    (Z), then:</p>

    <ol>
     <li><p>Let <var>timezone<sub>hours</sub></var> be 0.</p></li>

     <li><p>Let <var>timezone<sub>minutes</sub></var> be 0.</p></li>

     <li><p>Advance <var>position</var> to the next character in <var>input</var>.</p></li>
    </ol>

    <p>Otherwise, if the character at <var>position</var> is either a U+002B PLUS SIGN (+)
    or a U+002D HYPHEN-MINUS (-), then:</p>

    <ol>
     <li><p>If the character at <var>position</var> is a U+002B PLUS SIGN (+), let <var>sign</var> be "positive". Otherwise, it's a U+002D HYPHEN-MINUS (-); let <var>sign</var> be "negative".</p></li>

     <li><p>Advance <var>position</var> to the next character in <var>input</var>.</p></li>

     <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
     <var>input</var> given <var>position</var>. Let <var>s</var> be the collected
     sequence.</p></li>

     <li>
      <p>If <var>s</var> is exactly two characters long, then:</p>

      <ol>
       <li><p>Interpret <var>s</var> as a base-ten integer. Let that number be the <var>timezone<sub>hours</sub></var>.</p></li>

       <li><p>If <var>position</var> is beyond the end of <var>input</var> or if
       the character at <var>position</var> is not a U+003A COLON character, then fail.
       Otherwise, move <var>position</var> forwards one character.</p></li>

       <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
       <var>input</var> given <var>position</var>. If the collected sequence is not exactly two
       characters long, then fail. Otherwise, interpret the resulting sequence as a base-ten
       integer. Let that number be the <var>timezone<sub>minutes</sub></var>.</p></li>
      </ol>

      <p>If <var>s</var> is exactly four characters long, then:</p>

      <ol>
       <li><p>Interpret the first two characters of <var>s</var> as a base-ten integer. Let
       that number be the <var>timezone<sub>hours</sub></var>.</p></li>

       <li><p>Interpret the last two characters of <var>s</var> as a base-ten integer. Let
       that number be the <var>timezone<sub>minutes</sub></var>.</p></li>
      </ol>

      <p>Otherwise, fail.</p>
     </li>

     <li>If <var>timezone<sub>hours</sub></var> is not a number in the range
     0&nbsp;&le;&nbsp;<var>timezone<sub>hours</sub></var>&nbsp;&le;&nbsp;23, then
     fail.</li>

     <li>If <var>sign</var> is "negative", then negate <var>timezone<sub>hours</sub></var>.</li>

     <li>If <var>timezone<sub>minutes</sub></var> is not a number in the range
     0&nbsp;&le;&nbsp;<var>timezone<sub>minutes</sub></var>&nbsp;&le;&nbsp;59,
     then fail.</li>

     <li>If <var>sign</var> is "negative", then negate <var>timezone<sub>minutes</sub></var>.</li>
    </ol>

    <p>Otherwise, fail.</p>
   </li>

   <li><p>Return <var>timezone<sub>hours</sub></var> and <var>timezone<sub>minutes</sub></var>.</p></li>
  </ol>

  </div>


  <h5>Global dates and times</h5>

  <p>A <dfn data-x="concept-datetime">global date and time</dfn> consists of a specific
  <span>proleptic-Gregorian date</span>, consisting of a year, a month, and a day, and a time,
  consisting of an hour, a minute, a second, and a fraction of a second, expressed with a time-zone
  offset, consisting of a signed number of hours and minutes. <ref>GREGORIAN</ref></p>

  <p>A string is a <dfn>valid global date and time string</dfn> representing a date, time, and a
  time-zone offset if it consists of the following components in the given order:</p>

  <ol>
   <li>A <span>valid date string</span> representing the date</li>

   <li>A U+0054 LATIN CAPITAL LETTER T character (T) or a U+0020 SPACE character</li>

   <li>A <span>valid time string</span> representing the time</li>

   <li>A <span>valid time-zone offset string</span> representing the time-zone offset</li>
  </ol>

  <p>Times in dates before the formation of UTC in the mid-twentieth century must be expressed and
  interpreted in terms of UT1 (contemporary Earth solar time at the 0&deg; longitude), not UTC (the
  approximation of UT1 that ticks in SI seconds). Time before the formation of time zones must be
  expressed and interpreted as UT1 times with explicit time zones that approximate the contemporary
  difference between the appropriate local time and the time observed at the location of Greenwich,
  London.</p>

  <div class="example">

   <p>The following are some examples of dates written as <span data-x="valid global date and time
   string">valid global date and time strings</span>.</p>

   <dl>
    <dt>"<code data-x="">0037-12-13 00:00Z</code>"</dt>

    <dd>Midnight in areas using London time on the birthday of Nero (the Roman Emperor). See below
    for further discussion on which date this actually corresponds to.</dd>

    <dt>"<code data-x="">1979-10-14T12:00:00.001-04:00</code>"</dt>

    <dd>One millisecond after noon on October 14th 1979, in the time zone in use on the east coast
    of the USA during daylight saving time.</dd>

    <dt>"<code data-x="">8592-01-01T02:09+02:09</code>"</dt>

    <dd>Midnight UTC on the 1st of January, 8592. The time zone associated with that time is two
    hours and nine minutes ahead of UTC, which is not currently a real time zone, but is nonetheless
    allowed.</dd>
   </dl>

   <p>Several things are notable about these dates:</p>

   <ul>
    <li>Years with fewer than four digits have to be zero-padded. The date "37-12-13" would not be a
    valid date.</li>

    <li>If the "<code data-x="">T</code>" is replaced by a space, it must be a single space
    character. The string "<code data-x="">2001-12-21&nbsp;&nbsp;12:00Z</code>" (with two spaces
    between the components) would not be parsed successfully.</li>

    <li>To unambiguously identify a moment in time prior to the introduction of the Gregorian
    calendar (insofar as moments in time before the formation of UTC can be unambiguously
    identified), the date has to be first converted to the Gregorian calendar from the calendar in
    use at the time (e.g. from the Julian calendar). The date of Nero's birth is the 15th of
    December 37, in the Julian Calendar, which is the 13th of December 37 in the <span>proleptic
    Gregorian calendar</span>.</li> <!-- This might not be true. I can't find a reference that gives
    his birthday with an explicit statement about the calendar being used. However, it seems
    unlikely that it would be given in the Gregorian calendar, so I assume sites use the Julian one.
    -->

    <li>The time and time-zone offset components are not optional.</li>

    <li>Dates before the year one can't be represented as a datetime in this version of HTML.</li>

    <li>Times of specific events in ancient times are, at best, approximations, since time was not
    well coordinated or measured until relatively recent decades.</li>

    <li>Time-zone offsets differ based on daylight saving time.</li>
   </ul>

  </div>

  <div w-nodev>

  <p>The rules to <dfn>parse a global date and time string</dfn> are as follows. This will return
  either a time in UTC, with associated time-zone offset information for round-tripping or display
  purposes, or nothing. If at any point the algorithm says that it "fails", this means that it is
  aborted at that point and returns nothing.</p>

  <ol>
   <li><p>Let <var>input</var> be the string being parsed.</p></li>

   <li><p>Let <var>position</var> be a pointer into <var>input</var>, initially
   pointing at the start of the string.</p></li>

   <li><p><span>Parse a date component</span> to obtain <var>year</var>, <var>month</var>, and <var>day</var>. If this returns nothing, then fail.</p>

   <li><p>If <var>position</var> is beyond the end of <var>input</var> or if the
   character at <var>position</var> is neither a U+0054 LATIN CAPITAL LETTER T character
   (T) nor a U+0020 SPACE character, then fail. Otherwise, move <var>position</var>
   forwards one character.</p></li>

   <li><p><span>Parse a time component</span> to obtain <var>hour</var>, <var>minute</var>, and <var>second</var>. If this returns nothing, then fail.</p>

   <li><p>If <var>position</var> is beyond the end of <var>input</var>, then
   fail.</p></li>

   <li><p><span>Parse a time-zone offset component</span> to obtain <var>timezone<sub>hours</sub></var> and <var>timezone<sub>minutes</sub></var>. If this
   returns nothing, then fail.</p>

   <li><p>If <var>position</var> is <em>not</em> beyond the end of <var>input</var>, then fail.</p></li>

   <li><p>Let <var>time</var> be the moment in time at year <var>year</var>, month
   <var>month</var>, day <var>day</var>, hours <var>hour</var>, minute
   <var>minute</var>, second <var>second</var>, subtracting <var>timezone<sub>hours</sub></var> hours and <var>timezone<sub>minutes</sub></var> minutes. That moment in time is a moment in the UTC time
   zone.</p></li>

   <li><p>Let <var>timezone</var> be <var>timezone<sub>hours</sub></var>
   hours and <var>timezone<sub>minutes</sub></var> minutes from UTC.</p></li>

   <li><p>Return <var>time</var> and <var>timezone</var>.</p></li>
  </ol>

  </div>


  <h5>Weeks</h5>

  <p>A <dfn data-x="concept-week">week</dfn> consists of a week-year number and a week number
  representing a seven-day period starting on a Monday. Each week-year in this calendaring system
  has either 52 or 53 such seven-day periods, as defined below. The seven-day period starting on the
  Gregorian date Monday December 29th 1969 (1969-12-29) is defined as week number 1 in week-year
  1970. Consecutive weeks are numbered sequentially. The week before the number 1 week in a
  week-year is the last week in the previous week-year, and vice versa. <ref>GREGORIAN</ref></p>

  <p>A week-year with a number <var>year</var> has 53 weeks if it corresponds to either a
  year <var>year</var> in the <span>proleptic Gregorian calendar</span> that has a Thursday
  as its first day (January 1st), or a year <var>year</var> in the <span>proleptic
  Gregorian calendar</span> that has a Wednesday as its first day (January 1st) and where <var>year</var> is a number divisible by 400, or a number divisible by 4 but not by 100. All
  other week-years have 52 weeks.</p>

  <p>The <dfn>week number of the last day</dfn> of a week-year with 53 weeks is 53; the week number
  of the last day of a week-year with 52 weeks is 52.</p>

  <p class="note">The week-year number of a particular day can be different than the number of the
  year that contains that day in the <span>proleptic Gregorian calendar</span>. The first week in a
  week-year <var>y</var> is the week that contains the first Thursday of the Gregorian year
  <var>y</var>.</p>

  <p class="note">For modern purposes, a <span data-x="concept-week">week</span> as defined here is
  equivalent to ISO weeks as defined in ISO 8601. <ref>ISO8601</ref></p>

  <p>A string is a <dfn>valid week string</dfn> representing a week-year <var>year</var>
  and week <var>week</var> if it consists of the following components in the given
  order:</p>

  <ol>
   <li>Four or more <span>ASCII digits</span>, representing <var>year</var>, where <var>year</var>&nbsp;&gt;&nbsp;0</li>

   <li>A U+002D HYPHEN-MINUS character (-)</li>

   <li>A U+0057 LATIN CAPITAL LETTER W character (W)</li>

   <li>Two <span>ASCII digits</span>, representing the week <var>week</var>, in the range
   1&nbsp;&le;&nbsp;<var>week</var>&nbsp;&le;&nbsp;<var>maxweek</var>, where <var>maxweek</var> is the <span>week number of the last day</span> of week-year <var>year</var></li>
  </ol>

  <div w-nodev>

  <p>The rules to <dfn>parse a week string</dfn> are as follows. This will return either a week-year
  number and week number, or nothing. If at any point the algorithm says that it "fails", this means
  that it is aborted at that point and returns nothing.</p>

  <ol>
   <li><p>Let <var>input</var> be the string being parsed.</p></li>

   <li><p>Let <var>position</var> be a pointer into <var>input</var>, initially
   pointing at the start of the string.</p></li>

   <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
   <var>input</var> given <var>position</var>. If the collected sequence is not at least four
   characters long, then fail. Otherwise, interpret the resulting sequence as a base-ten integer.
   Let that number be the <var>year</var>.</p></li>

   <li><p>If <var>year</var> is not a number greater than zero, then fail.</p></li>

   <li><p>If <var>position</var> is beyond the end of <var>input</var> or if the
   character at <var>position</var> is not a U+002D HYPHEN-MINUS character, then fail.
   Otherwise, move <var>position</var> forwards one character.</p></li>

   <li><p>If <var>position</var> is beyond the end of <var>input</var> or if the
   character at <var>position</var> is not a U+0057 LATIN CAPITAL LETTER W character (W),
   then fail. Otherwise, move <var>position</var> forwards one character.</p></li>

   <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
   <var>input</var> given <var>position</var>. If the collected sequence is not exactly two
   characters long, then fail. Otherwise, interpret the resulting sequence as a base-ten integer.
   Let that number be the <var>week</var>.</p></li>

   <li><p>Let <var>maxweek</var> be the <span>week number of the last day</span> of year
   <var>year</var>.</p></li>

   <li><p>If <var>week</var> is not a number in the range 1&nbsp;&le;&nbsp;<var>week</var>&nbsp;&le;&nbsp;<var>maxweek</var>, then fail.</p></li>

   <li><p>If <var>position</var> is <em>not</em> beyond the end of <var>input</var>, then fail.</p></li>

   <li><p>Return the week-year number <var>year</var> and the week number <var>week</var>.</p></li>
  </ol>

  </div>


  <h5>Durations</h5>

  <!-- if you add support for year and month durations, then search for MONTHS throughout the spec
  (not just this section) for things that have to change -->

  <p>A <dfn data-x="concept-duration">duration</dfn> consists of <!--MONTHS: either a number of
  months or--> a number of seconds.</p>

  <p class="note">Since months and seconds are not comparable (a month is not a precise number of
  seconds, but is instead a period whose exact length depends on the precise day from which it is
  measured) a <span data-x="concept-duration">duration</span> as defined in this specification cannot
  <!--MONTHS: consist of a mixture of both--> include months (or years, which are equivalent to
  twelve months). Only durations that describe a specific number of seconds can be described.</p>

  <p>A string is a <dfn>valid duration string</dfn> representing a <span
  data-x="concept-duration">duration</span> <var>t</var> if it consists of either of the
  following:</p>

  <ul>
   <li>
    <!-- The ISO format -->

    <!-- NOTE: ISO durations also have a number of formats we do not
         consider conforming:

         - a "week" format (e.g. P4W). This one is actually supported
           by the parser.

         - P<date>T<time>, with or without hyphens

         - All these formats, with commas instead of periods for the
           seconds decimal

    -->

    <p>A literal U+0050 LATIN CAPITAL LETTER P character followed by one or more of the following
    subcomponents, in the order given, where <!--the number of years, if any, multiplied by twelve,
    plus the number of months, if any, equals the number of months in <var>t</var>, if the
    duration is in months; or, if it is in seconds, -->the number of days, hours, minutes, and
    seconds corresponds to the same number of seconds as in <var>t</var>:</p>

    <ol>
<!--MONTHS:
     <li><p>One or more <span>ASCII digits</span> followed by a U+0059 LATIN CAPITAL LETTER Y
     character, representing a number of years.</p></li>

     <li><p>One or more <span>ASCII digits</span> followed by a U+004D LATIN CAPITAL LETTER M
     character, representing a number of months.</p></li>
-->

     <li><p>One or more <span>ASCII digits</span> followed by a U+0044 LATIN CAPITAL LETTER D
     character, representing a number of days.</p></li>

     <li>
      <p>A U+0054 LATIN CAPITAL LETTER T character followed by one or more of the following
      subcomponents, in the order given:</p>

      <ol>
       <li><p>One or more <span>ASCII digits</span> followed by a U+0048 LATIN CAPITAL LETTER H
       character, representing a number of hours.</p></li>

       <li><p>One or more <span>ASCII digits</span> followed by a U+004D LATIN CAPITAL LETTER M
       character, representing a number of minutes.</p></li>

       <li>
        <p>The following components:</p>

        <ol>
         <li><p>One or more <span>ASCII digits</span>, representing a number of seconds.</p></li>

         <li><p>Optionally, a U+002E FULL STOP character (.) followed by one, two, or three
         <span>ASCII digits</span>, representing a fraction of a second.</p></li>

         <li><p>A U+0053 LATIN CAPITAL LETTER S character.</p></li>
        </ol>
       </li>
      </ol>
     </li>
    </ol>

    <p class="note">This, as with a number of other date- and time-related microsyntaxes defined in
    this specification, is based on one of the formats defined in ISO 8601. <ref>ISO8601</ref></p>
   </li>

   <li>
    <p>One or more <span data-x="duration time component">duration time components</span>, each with
    a different <span>duration time component scale</span>, in any order; the sum of the represented
    seconds being equal to the number of seconds in <var>t</var>.</p>

    <p>A <dfn>duration time component</dfn> is a string consisting of the following components:</p>

    <ol>
     <li><p>Zero or more <span>ASCII whitespace</span>.</p></li>

     <li><p>One or more <span>ASCII digits</span>, representing a number of time units, scaled by
     the <span>duration time component scale</span> specified (see below) to represent a number of
     seconds.</p></li>

     <li><p>If the <span>duration time component scale</span> specified is 1 (i.e. the units are
     seconds), then, optionally, a U+002E FULL STOP character (.) followed by one, two, or three
     <span>ASCII digits</span>, representing a fraction of a second.</p></li>

     <li><p>Zero or more <span>ASCII whitespace</span>.</p></li>

     <li>
      <p>One of the following characters, representing the <dfn>duration time component scale</dfn>
      of the time unit used in the numeric part of the <span>duration time component</span>:</p>

      <dl>
       <dt>U+0057 LATIN CAPITAL LETTER W character</dt>
       <dt>U+0077 LATIN SMALL LETTER W character</dt>

       <dd>Weeks. The scale is 604800.</dd>

       <dt>U+0044 LATIN CAPITAL LETTER D character</dt>
       <dt>U+0064 LATIN SMALL LETTER D character</dt>

       <dd>Days. The scale is 86400.</dd>

       <dt>U+0048 LATIN CAPITAL LETTER H character</dt>
       <dt>U+0068 LATIN SMALL LETTER H character</dt>

       <dd>Hours. The scale is 3600.</dd>

       <dt>U+004D LATIN CAPITAL LETTER M character</dt>
       <dt>U+006D LATIN SMALL LETTER M character</dt>

       <dd>Minutes. The scale is 60.</dd>

       <dt>U+0053 LATIN CAPITAL LETTER S character</dt>
       <dt>U+0073 LATIN SMALL LETTER S character</dt>

       <dd>Seconds. The scale is 1.</dd>
      </dl>
     </li>

     <li><p>Zero or more <span>ASCII whitespace</span>.</p></li>
    </ol>

    <p class="note">This is not based on any of the formats in ISO 8601. It is intended to be a more
    human-readable alternative to the ISO 8601 duration format.</p>
   </li>
  </ul>

  <div w-nodev>

  <p>The rules to <dfn>parse a duration string</dfn> are as follows. This will return either a <span
  data-x="concept-duration">duration</span> or nothing. If at any point the algorithm says that it
  "fails", this means that it is aborted at that point and returns nothing.</p>

  <ol>
   <!--MONTHS: this algorithm actually already parses months adequately, though it would need
   tweaking if we introduced a "mo" unit for the non-ISO variant. See other "MONTHS" annotations
   below. -->

   <li><p>Let <var>input</var> be the string being parsed.</p></li>

   <li><p>Let <var>position</var> be a pointer into <var>input</var>, initially
   pointing at the start of the string.</p></li>

   <li><p>Let <var>months</var>, <var>seconds</var>, and <var>component
   count</var> all be zero.</p></li>

   <li>
    <p>Let <var>M-disambiguator</var> be <i>minutes</i>.</p>

    <!--MONTHS: this note would change-->
    <p class="note">This flag's other value is <i>months</i>. It is used to disambiguate the "M"
    unit in ISO8601 durations, which use the same unit for months and minutes. Months are not
    allowed, but are parsed for future compatibility and to avoid misinterpreting ISO8601 durations
    that would be valid in other contexts.</p>
   </li>

   <li><p><span>Skip ASCII whitespace</span> within <var>input</var> given
   <var>position</var>.</p></li>

   <li><p>If <var>position</var> is past the end of <var>input</var>, then
   fail.</p></li>

   <li><p>If the character in <var>input</var> pointed to by <var>position</var> is a U+0050 LATIN
   CAPITAL LETTER P character, then advance <var>position</var> to the next character, set
   <var>M-disambiguator</var> to <i>months</i>, and <span>skip ASCII whitespace</span> within
   <var>input</var> given <var>position</var>.</p></li>

   <li>
    <p>While true:</p>

    <ol>
     <li><p>Let <var>units</var> be undefined. It will be assigned one of the following
     values: <i>years</i>, <i>months</i>, <i>weeks</i>, <i>days</i>, <i>hours</i>, <i>minutes</i>,
     and <i>seconds</i>.</p></li>

     <li><p>Let <var>next character</var> be undefined. It is used to process characters
     from the <var>input</var>.</p></li>

     <li><p>If <var>position</var> is past the end of <var>input</var>, then break.</p></li>

     <li><p>If the character in <var>input</var> pointed to by <var>position</var> is a U+0054 LATIN
     CAPITAL LETTER T character, then advance <var>position</var> to the next character, set
     <var>M-disambiguator</var> to <i>minutes</i>, <span>skip ASCII whitespace</span> within
     <var>input</var> given <var>position</var>, and <span>continue</span>.</p></li>

     <li><p>Set <var>next character</var> to the character in <var>input</var>
     pointed to by <var>position</var>.</p></li>

     <li>
      <p>If <var>next character</var> is a U+002E FULL STOP character (.), then let <var>N</var> equal zero. (Do not advance <var>position</var>. That is taken care
      of below.)</p>

      <p>Otherwise, if <var>next character</var> is an <span data-x="ASCII digits">ASCII
      digit</span>, then <span>collect a sequence of code points</span> that are <span>ASCII
      digits</span> from <var>input</var> given <var>position</var>, interpret the resulting
      sequence as a base-ten integer, and let <var>N</var> be that number.</p>

      <p>Otherwise, <var>next character</var> is not part of a number; fail.</p>
     </li>

     <li><p>If <var>position</var> is past the end of <var>input</var>, then
     fail.</p></li>

     <li><p>Set <var>next character</var> to the character in <var>input</var>
     pointed to by <var>position</var>, and this time advance <var>position</var>
     to the next character. (If <var>next character</var> was a U+002E FULL STOP character
     (.) before, it will still be that character this time.)</p></li>

     <li>
      <p>If <var>next character</var> is U+002E (.), then:</p>

      <ol>
       <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
       <var>input</var> given <var>position</var>. Let <var>s</var> be the resulting
       sequence.</p></li>

       <li><p>If <var>s</var> is the empty string, then fail.</p></li>

       <li><p>Let <var>length</var> be the number of characters in <var>s</var>.</p></li>

       <li><p>Let <var>fraction</var> be the result of interpreting <var>s</var>
       as a base-ten integer, and then dividing that number by <span data-x="">10<sup><var>length</var></sup></span>.</li>

       <li><p>Increment <var>N</var> by <var>fraction</var>.</p></li>

       <li><p><span>Skip ASCII whitespace</span> within <var>input</var> given
       <var>position</var>.</p></li>

       <li><p>If <var>position</var> is past the end of <var>input</var>, then
       fail.</p></li>

       <li><p>Set <var>next character</var> to the character in <var>input</var>
       pointed to by <var>position</var>, and advance <var>position</var> to the
       next character.</p></li>

       <li><p>If <var>next character</var> is neither a U+0053 LATIN CAPITAL LETTER S
       character nor a U+0073 LATIN SMALL LETTER S character, then fail.</p></li>

       <li><p>Set <var>units</var> to <i>seconds</i>.</p></li>
      </ol>

      <p>Otherwise:</p>

      <ol>
       <li><p>If <var>next character</var> is <span>ASCII whitespace</span>, then <span>skip ASCII
       whitespace</span> within <var>input</var> given <var>position</var>, set <var>next
       character</var> to the character in <var>input</var> pointed to by <var>position</var>, and
       advance <var>position</var> to the next character.</p></li>

       <li>
        <!--MONTHS: this would probably be where we would add more
        logic to support a new unit like 'mo' -->

        <p>If <var>next character</var> is a U+0059 LATIN CAPITAL LETTER Y character, or a
        U+0079 LATIN SMALL LETTER Y character, set <var>units</var> to <i>years</i> and set
        <var>M-disambiguator</var> to <i>months</i>.</p>

        <p>If <var>next character</var> is a U+004D LATIN CAPITAL LETTER M character or a
        U+006D LATIN SMALL LETTER M character, and <var>M-disambiguator</var> is
        <i>months</i>, then set <var>units</var> to <i>months</i>.</p>

        <p>If <var>next character</var> is a U+0057 LATIN CAPITAL LETTER W character or a
        U+0077 LATIN SMALL LETTER W character, set <var>units</var> to <i>weeks</i> and set
        <var>M-disambiguator</var> to <i>minutes</i>.</p>

        <p>If <var>next character</var> is a U+0044 LATIN CAPITAL LETTER D character or a
        U+0064 LATIN SMALL LETTER D character, set <var>units</var> to <i>days</i> and set
        <var>M-disambiguator</var> to <i>minutes</i>.</p>

        <p>If <var>next character</var> is a U+0048 LATIN CAPITAL LETTER H character or a
        U+0068 LATIN SMALL LETTER H character, set <var>units</var> to <i>hours</i> and set
        <var>M-disambiguator</var> to <i>minutes</i>.</p>

        <p>If <var>next character</var> is a U+004D LATIN CAPITAL LETTER M character or a
        U+006D LATIN SMALL LETTER M character, and <var>M-disambiguator</var> is
        <i>minutes</i>, then set <var>units</var> to <i>minutes</i>.</p>

        <p>If <var>next character</var> is a U+0053 LATIN CAPITAL LETTER S character or a
        U+0073 LATIN SMALL LETTER S character, set <var>units</var> to <i>seconds</i> and
        set <var>M-disambiguator</var> to <i>minutes</i>.</p>

        <p>Otherwise, if <var>next character</var> is none of the above characters, then
        fail.</p>
       </li>
      </ol>
     </li>

     <li><p>Increment <var>component count</var>.</p></li>

     <li><p>Let <var>multiplier</var> be 1.</p></li>

     <li><p>If <var>units</var> is <i>years</i>, multiply <var>multiplier</var> by
     12 and set <var>units</var> to <i>months</i>.</p></li>

     <li>
      <p>If <var>units</var> is <i>months</i>, add the product of <var>N</var> and
      <var>multiplier</var> to <var>months</var>.</p>

      <p>Otherwise:</p>

      <ol>
       <li><p>If <var>units</var> is <i>weeks</i>, multiply <var>multiplier</var>
       by 7 and set <var>units</var> to <i>days</i>.</p></li>

       <li><p>If <var>units</var> is <i>days</i>, multiply <var>multiplier</var>
       by 24 and set <var>units</var> to <i>hours</i>.</p></li>

       <li><p>If <var>units</var> is <i>hours</i>, multiply <var>multiplier</var>
       by 60 and set <var>units</var> to <i>minutes</i>.</p></li>

       <li><p>If <var>units</var> is <i>minutes</i>, multiply <var>multiplier</var> by 60 and set <var>units</var> to <i>seconds</i>.</p></li>

       <li><p>Forcibly, <var>units</var> is now <i>seconds</i>. Add the product of <var>N</var> and <var>multiplier</var> to <var>seconds</var>.</p></li>
      </ol>
     </li>

     <li><p><span>Skip ASCII whitespace</span> within <var>input</var> given
     <var>position</var>.</p></li>
    </ol>
   </li>

   <li><p>If <var>component count</var> is zero,
   fail.</p></li>

   <!--MONTHS: if we add month support this is where you'd return them-->
   <li><p>If <var>months</var> is not zero, fail.</p></li>

   <li><p>Return the <span data-x="concept-duration">duration</span> consisting of <var>seconds</var> seconds.</p></li>
  </ol>

  </div>


  <h5>Vaguer moments in time</h5>

  <p>A string is a <dfn>valid date string with optional time</dfn> if
  it is also one of the following:</p>

  <ul>
   <li>A <span>valid date string</span></li>

   <li>A <span>valid global date and time string</span></li>
  </ul>

  <div w-nodev>

  <hr>

  <p>The rules to <dfn>parse a date or time string</dfn> are as follows. The algorithm will return
  either a <span data-x="concept-date">date</span>, a <span data-x="concept-time">time</span>, a <span
  data-x="concept-datetime">global date and time</span>, or nothing. If at any point the algorithm
  says that it "fails", this means that it is aborted at that point and returns nothing.</p>

  <ol>
   <li><p>Let <var>input</var> be the string being parsed.</p></li>

   <li><p>Let <var>position</var> be a pointer into <var>input</var>, initially
   pointing at the start of the string.</p></li>

   <li><p>Set <var>start position</var> to the same position as <var>position</var>.</p></li>

   <li><p>Set the <var>date present</var> and <var>time present</var> flags to
   true.</p></li>

   <li><p><span>Parse a date component</span> to obtain <var>year</var>, <var>month</var>, and <var>day</var>. If this fails, then set the <var>date
   present</var> flag to false.</p>

   <li>
    <p>If <var>date present</var> is true, and <var>position</var> is not beyond
    the end of <var>input</var>, and the character at <var>position</var> is
    either a U+0054 LATIN CAPITAL LETTER T character (T) or a U+0020 SPACE character, then advance
    <var>position</var> to the next character in <var>input</var>.</p>

    <p>Otherwise, if <var>date present</var> is true, and either <var>position</var> is beyond the end of <var>input</var> or the character at <var>position</var> is neither a U+0054 LATIN CAPITAL LETTER T character (T) nor a U+0020
    SPACE character, then set <var>time present</var> to false.</p>

    <p>Otherwise, if <var>date present</var> is false, set <var>position</var>
    back to the same position as <var>start position</var>.</p>
   </li>

   <li><p>If the <var>time present</var> flag is true, then <span>parse a time
   component</span> to obtain <var>hour</var>, <var>minute</var>, and <var>second</var>. If this returns nothing, then fail.</p></li>

   <li><p>If the <var>date present</var> and <var>time present</var> flags are
   both true, but <var>position</var> is beyond the end of <var>input</var>, then
   fail.</p></li>

   <li><p>If the <var>date present</var> and <var>time present</var> flags are
   both true, <span>parse a time-zone offset component</span> to obtain <var>timezone<sub>hours</sub></var> and <var>timezone<sub>minutes</sub></var>. If this
   returns nothing, then fail.</p>

   <li><p>If <var>position</var> is <em>not</em> beyond the end of <var>input</var>, then fail.</p></li>

   <li>
    <p>If the <var>date present</var> flag is true and the <var>time present</var>
    flag is false, then let <var>date</var> be the date with year <var>year</var>,
    month <var>month</var>, and day <var>day</var>, and return <var>date</var>.</p>

    <p>Otherwise, if the <var>time present</var> flag is true and the <var>date
    present</var> flag is false, then let <var>time</var> be the time with hour <var>hour</var>, minute <var>minute</var>, and second <var>second</var>,
    and return <var>time</var>.</p>

    <p>Otherwise, let <var>time</var> be the moment in time at year <var>year</var>, month <var>month</var>, day <var>day</var>, hours <var>hour</var>, minute <var>minute</var>, second <var>second</var>,
    subtracting <var>timezone<sub>hours</sub></var> hours and <var>timezone<sub>minutes</sub></var> minutes, that moment in time being a moment
    in the UTC time zone; let <var>timezone</var> be <var>timezone<sub>hours</sub></var> hours and <var>timezone<sub>minutes</sub></var>
    minutes from UTC; and return <var>time</var> and <var>timezone</var>.</p>
   </li>
  </ol>

  </div>


  <div w-nodev>

  <!--en-GB--><h4 id="colours">Legacy colors</h4>

  <p>Some obsolete legacy attributes parse colors using the
  <!--en-GB--><dfn id="rules-for-parsing-a-legacy-colour-value">rules for parsing a legacy color
  value</dfn>, given a string <var>input</var>. They will return either a CSS color or failure.</p>

  <ol>
   <li><p>If <var>input</var> is the empty string, then return failure.</p></li>

   <li><p><span>Strip leading and trailing ASCII whitespace</span> from <var>input</var>.</p></li>

   <li><p>If <var>input</var> is an <span>ASCII case-insensitive</span> match for "<code
   data-x="">transparent</code>", then return failure.</p></li>

   <li>
    <p>If <var>input</var> is an <span>ASCII case-insensitive</span> match for one of the <span
    data-x="named color">named colors</span>, then return the CSS color corresponding to that
    keyword. <ref>CSSCOLOR</ref></p>

    <p class="note"><a
    href="https://www.w3.org/TR/css3-color/#css2-system">CSS2 System
    Colors</a> are not recognized.</p>
   </li>

   <li>
    <p>If <var>input</var>'s <span>code point length</span> is four, and the first character in
    <var>input</var> is U+0023 (#), and the last three characters of <var>input</var> are all
    <span>ASCII hex digits</span>, then:</p>

    <ol>
     <li><p>Let <var>result</var> be a CSS color.</p>

     <li><p>Interpret the second character of <var>input</var> as a hexadecimal digit; let
     the red component of <var>result</var> be the resulting number multiplied by 17.</p>

     <li><p>Interpret the third character of <var>input</var> as a hexadecimal digit; let
     the green component of <var>result</var> be the resulting number multiplied by 17.</p>

     <li><p>Interpret the fourth character of <var>input</var> as a hexadecimal digit; let
     the blue component of <var>result</var> be the resulting number multiplied by 17.</p>

     <li><p>Return <var>result</var>.</p>
    </ol>
   </li>

   <li><p>Replace any <span data-x="code point">code points</span> greater than U+FFFF in
   <var>input</var> (i.e., any characters that are not in the basic multilingual plane) with "<code
   data-x="">00</code>".</p></li>

   <li><p>If <var>input</var>'s <span>code point length</span> is greater than 128, truncate
   <var>input</var>, leaving only the first 128 characters.</p></li>

   <li><p>If the first character in <var>input</var> is U+0023 (#), then remove it.</p></li>

   <li><p>Replace any character in <var>input</var> that is not an <span data-x="ASCII hex
   digits">ASCII hex digit</span> with U+0030 (0).</p></li>

   <li><p>While <var>input</var>'s <span>code point length</span> is zero or not a multiple of
   three, append U+0030 (0) to <var>input</var>.</p></li>

   <li><p>Split <var>input</var> into three strings of equal <span>code point length</span>, to
   obtain three components. Let <var>length</var> be the <span>code point length</span> that all of
   those components have (one third the <span>code point length</span> of <var>input</var>).</p></li>

   <li><p>If <var>length</var> is greater than 8, then remove the leading <span
   data-x=""><var>length</var>-8</span> characters in each component, and let <var>length</var> be 8.</p></li>

   <li><p>While <var>length</var> is greater than two and the first character in each
   component is U+0030 (0), remove that character and reduce <var>length</var> by one.</p></li>

   <li><p>If <var>length</var> is <em>still</em> greater than two, truncate each component,
   leaving only the first two characters in each.</p></li>

   <li><p>Let <var>result</var> be a CSS color.</p>

   <li><p>Interpret the first component as a hexadecimal number; let the red component of <var>result</var> be the resulting number.</p></li>

   <li><p>Interpret the second component as a hexadecimal number; let the green component of <var>result</var> be the resulting number.</p></li>

   <li><p>Interpret the third component as a hexadecimal number; let the blue component of <var>result</var> be the resulting number.</p></li>

   <li><p>Return <var>result</var>.</p>
  </ol>

  </div>



  <h4>Space-separated tokens</h4>

  <p>A <dfn export>set of space-separated tokens</dfn> is a string containing zero or more words
  (known as tokens) separated by one or more <span>ASCII whitespace</span>, where words consist of
  any string of one or more characters, none of which are <span>ASCII whitespace</span>.</p>

  <p>A string containing a <span>set of space-separated tokens</span> may have leading or trailing
  <span>ASCII whitespace</span>.</p>

  <p>An <dfn export>unordered set of unique space-separated tokens</dfn> is a <span>set of
  space-separated tokens</span> where none of the tokens are duplicated.</p>

  <p>An <dfn>ordered set of unique space-separated tokens</dfn> is a <span>set of space-separated
  tokens</span> where none of the tokens are duplicated but where the order of the tokens is
  meaningful.</p>

  <p><span data-x="set of space-separated tokens">Sets of space-separated tokens</span> sometimes
  have a defined set of allowed values. When a set of allowed values is defined, the tokens must all
  be from that list of allowed values; other values are non-conforming. If no such set of allowed
  values is provided, then all values are conforming.</p>

  <p class="note">How tokens in a <span>set of space-separated tokens</span> are to be compared
  (e.g. case-sensitively or not) is defined on a per-set basis.</p>



  <h4>Comma-separated tokens</h4>

  <p>A <dfn>set of comma-separated tokens</dfn> is a string containing zero or more tokens each
  separated from the next by a single U+002C COMMA character (,), where tokens consist of any string
  of zero or more characters, neither beginning nor ending with <span>ASCII whitespace</span>, nor
  containing any U+002C COMMA characters (,), and optionally surrounded by <span>ASCII
  whitespace</span>.</p>

  <p class="example">For instance, the string "<code
  data-x="">&nbsp;a&nbsp;,b,,d&nbsp;d&nbsp;</code>" consists of four tokens: "a", "b", the empty
  string, and "d&nbsp;d". Leading and trailing whitespace around each token doesn't count as part of
  the token, and the empty string can be a token.</p>

  <p><span data-x="set of comma-separated tokens">Sets of comma-separated tokens</span> sometimes
  have further restrictions on what consists a valid token. When such restrictions are defined, the
  tokens must all fit within those restrictions; other values are non-conforming. If no such
  restrictions are specified, then all values are conforming.</p>



  <h4 id="syntax-references">References</h4>

  <p>A <dfn>valid hash-name reference</dfn> to an element of type <var>type</var> is a
  string consisting of a U+0023 NUMBER SIGN character (#) followed by a string which exactly matches
  the value of the <code data-x="">name</code> attribute of an element with type <var>type</var> in
  the same <span>tree</span>.</p>

  <div w-nodev>

  <p>The <dfn>rules for parsing a hash-name reference</dfn> to an element of type <var>type</var>,
  given a context node <var>scope</var>, are as follows:</p>

  <ol>
   <li><p>If the string being parsed does not contain a U+0023 NUMBER SIGN character, or if the
   first such character in the string is the last character in the string, then return null.</p></li>

   <li><p>Let <var>s</var> be the string from the character immediately after the first
   U+0023 NUMBER SIGN character in the string being parsed up to the end of that string.</p></li>

   <li>
    <p>Return the first element of type <var>type</var> in <var>scope</var>'s <span>tree</span>, in
    <span>tree order</span>, that has an <code data-x="attr-id">id</code> or <code
    data-x="">name</code> attribute whose value is <var>s</var>, or null if there is no such
    element.</p>

    <p class="note">Although <code data-x="attr-id">id</code> attributes are accounted for when
    parsing, they are not used in determining whether a value is a <span><em>valid</em> hash-name
    reference</span>. That is, a hash-name reference that refers to an element based on <code
    data-x="attr-id">id</code> is a conformance error (unless that element also has a <code
    data-x="">name</code> attribute with the same value).</p>
   </li>

   <!-- History behind case-sensitive matching above: https://github.com/whatwg/html/issues/1666 -->
  </ol>

  </div>


  <h4 id="mq">Media queries</h4>

  <p>A string is a <dfn>valid media query list</dfn> if it matches the <code
  data-x="">&lt;media-query-list&gt;</code> production of <cite>Media Queries</cite>.
  <ref>MQ</ref></p>

  <p>A string <dfn>matches the environment</dfn> of the user if it is the empty string, a string
  consisting of only <span>ASCII whitespace</span>, or is a media query list that matches the user's
  environment according to the definitions given in <cite>Media Queries</cite>. <ref>MQ</ref></p>


  <h4 id="unique-values">Unique internal values</h4>

  <p>A <dfn>unique internal value</dfn> is a value that is serializable, comparable by value, and
  never exposed to script.</p>

  <p>To create a <dfn>new unique internal value</dfn>, return a <span>unique internal value</span>
  that has never previously been returned by this algorithm.</p>


  <h3 split-filename="urls-and-fetching">URLs</h3>

  <h4>Terminology</h4>

  <p>A string is a <dfn>valid non-empty URL</dfn> if it is a <span>valid URL string</span> but it is
  not the empty string.</p>

  <p>A string is a <dfn>valid URL potentially surrounded by spaces</dfn> if, after <span
  data-x="strip leading and trailing ASCII whitespace">stripping leading and trailing ASCII
  whitespace</span> from it, it is a <span>valid URL string</span>.</p>

  <p>A string is a <dfn>valid non-empty URL potentially surrounded by spaces</dfn> if, after <span
  data-x="strip leading and trailing ASCII whitespace">stripping leading and trailing ASCII
  whitespace</span> from it, it is a <span>valid non-empty URL</span>.</p>

  <p>This specification defines the URL <dfn><code>about:legacy-compat</code></dfn> as a reserved,
  though unresolvable, <code data-x="about protocol">about:</code> URL, for use in <span
  data-x="syntax-doctype">DOCTYPE</span>s in <span>HTML documents</span> when needed for
  compatibility with XML tools. <ref>ABOUT</ref></p>

  <p>This specification defines the URL <dfn><code>about:html-kind</code></dfn> as a reserved,
  though unresolvable, <code data-x="about protocol">about:</code> URL, that is used as an
  identifier for kinds of media tracks. <ref>ABOUT</ref></p>

  <p>This specification defines the URL <dfn><code>about:srcdoc</code></dfn> as a reserved, though
  unresolvable, <code data-x="about protocol">about:</code> URL, that is used as the <span
  data-x="concept-document-url">URL</span> of <span data-x="an iframe srcdoc
  document"><code>iframe</code> <code data-x="attr-iframe-srcdoc">srcdoc</code> documents</span>.
  <ref>ABOUT</ref></p>

  <p>The <dfn>fallback base URL</dfn> of a <code>Document</code> object <var>document</var> is the
  <span>URL record</span> obtained by running these steps:</p>

  <ol>
   <li>
    <p>If <var>document</var> is <span data-x="an iframe srcdoc document">an <code>iframe</code>
    <code data-x="attr-iframe-srcdoc">srcdoc</code> document</span>, then:</p>

    <ol>
     <li><p><span>Assert</span>: <var>document</var>'s <span
     data-x="concept-document-about-base-url">about base URL</span> is non-null.</p></li>

     <li><p>Return <var>document</var>'s <span data-x="concept-document-about-base-url">about base
     URL</span>.</p></li>
    </ol>
   </li>

   <li><p>If <var>document</var>'s <span data-x="concept-document-url">URL</span> <span>matches
   <code>about:blank</code></span> and <var>document</var>'s <span
   data-x="concept-document-about-base-url">about base URL</span> is non-null, then return
   <var>document</var>'s <span data-x="concept-document-about-base-url">about base URL</span>.</p></li>

   <!-- https://www.hixie.ch/tests/adhoc/dom/level0/history/pushState/001/ -->

   <li><p>Return <var>document</var>'s <span data-x="concept-document-url">URL</span>.</p></li>
  </ol>

  <p>The <dfn export>document base URL</dfn> of a <code>Document</code> object is the
  <span>URL record</span> obtained by running these steps:</p>

  <ol>
   <li><p>If there is no <code>base</code> element that has an <code
   data-x="attr-base-href">href</code> attribute in the <code>Document</code>, then return the
   <code>Document</code>'s <span>fallback base URL</span>.</p></li>

   <li><p>Otherwise, return the <span>frozen base URL</span> of the first <code>base</code> element
   in the <code>Document</code> that has an <code data-x="attr-base-href">href</code> attribute, in
   <span>tree order</span>.</p></li>
  </ol>

  <hr>

  <p>A <span>URL</span> <dfn>matches <code>about:blank</code></dfn> if its <span
  data-x="concept-url-scheme">scheme</span> is "<code data-x="">about</code>", its <span
  data-x="concept-url-path">path</span> contains a single string "<code data-x="">blank</code>", its
  <span data-x="concept-url-username">username</span> and <span
  data-x="concept-url-password">password</span> are the empty string, and its <span
  data-x="concept-url-host">host</span> is null.</p>

  <p class="note">Such a URL's <span data-x="concept-url-query">query</span> and <span
  data-x="concept-url-fragment">fragment</span> can be non-null. For example, the <span>URL
  record</span> created by <span data-x="URL parser">parsing</span> "<code
  data-x="">about:blank?foo#bar</code>" <span>matches <code>about:blank</code></span>.</p>

  <p>A <span>URL</span> <dfn>matches <code>about:srcdoc</code></dfn> if its <span
  data-x="concept-url-scheme">scheme</span> is "<code data-x="">about</code>", its <span
  data-x="concept-url-path">path</span> contains a single string "<code data-x="">srcdoc</code>",
  its <span data-x="concept-url-query">query</span> is null, its <span
  data-x="concept-url-username">username</span> and <span
  data-x="concept-url-password">password</span> are the empty string, and its <span
  data-x="concept-url-host">host</span> is null.</p>

  <p class="note">The reason that <span>matches <code>about:srcdoc</code></span> ensures that the
  <span>URL</span>'s <span data-x="concept-url-query">query</span> is null is because it is not
  possible to create <span data-x="an iframe srcdoc document">an <code>iframe</code> <code
  data-x="attr-iframe-srcdoc">srcdoc</code> document</span> whose <span
  data-x="concept-document-url">URL</span> has a non-null <span
  data-x="concept-url-query">query</span>, unlike <code>Document</code>s whose <span
  data-x="concept-document-url">URL</span> <span>matches <code>about:blank</code></span>. In other
  words, the set of all <span>URL</span>s that <span data-x="matches about:srcdoc">match
  <code>about:srcdoc</code></span> only vary in their <span
  data-x="concept-url-fragment">fragment</span>.</p>


  <div w-nodev>

  <h4 id="resolving-urls">Parsing URLs</h4>

  <p>Parsing a URL is the process of taking a string and obtaining the <span>URL record</span> that
  it represents. While this process is defined in <cite>URL</cite>, the HTML standard defines
  several wrappers to abstract base URLs and encodings. <ref>URL</ref></p>

  <p class="note">Most new APIs are to use <span>parse a URL</span>. Older APIs and HTML elements
  might have reason to use <span data-x="encoding-parsing a URL">encoding-parse a URL</span>. When a
  custom base URL is needed or no base URL is desired, the <span>URL parser</span> can of course be
  used directly as well.</p>

  <p>To <dfn export>parse a URL</dfn>, given a string <var>url</var>, relative to a
  <code>Document</code> object or <span>environment settings object</span> <var>environment</var>,
  run these steps. They return failure or a <span>URL</span>.</p>

  <ol>
   <li><p>Let <var>baseURL</var> be <var>environment</var>'s <span data-x="document base URL">base
   URL</span>, if <var>environment</var> is a <code>Document</code> object; otherwise
   <var>environment</var>'s <span>API base URL</span>.</p></li>

   <li><p>Return the result of applying the <span>URL parser</span> to <var>url</var>, with
   <var>baseURL</var>.</p></li>
  </ol>

  <p id="resolve-a-url">To <dfn export data-x="encoding-parsing a URL">encoding-parse a URL</dfn>,
  given a string <var>url</var>, relative to a <code>Document</code> object or <span>environment
  settings object</span> <var>environment</var>, run these steps. They return failure or a
  <span>URL</span>.</p>

  <ol>
   <li><p>Let <var>encoding</var> be <span>UTF-8</span>.</p></li>

   <li><p>If <var>environment</var> is a <code>Document</code> object, then set <var>encoding</var>
   to <var>environment</var>'s <span data-x="document's character encoding">character
   encoding</span>.</p></li>

   <li><p>Otherwise, if <var>environment</var>'s <span>relevant global object</span> is a
   <code>Window</code> object, set <var>encoding</var> to <var>environment</var>'s <span>relevant
   global object</span>'s <span data-x="concept-document-window">associated
   <code>Document</code></span>'s <span data-x="document's character encoding">character
   encoding</span>.</p></li>

   <li><p>Let <var>baseURL</var> be <var>environment</var>'s <span data-x="document base URL">base
   URL</span>, if <var>environment</var> is a <code>Document</code> object; otherwise
   <var>environment</var>'s <span>API base URL</span>.</p></li>

   <li><p>Return the result of applying the <span>URL parser</span> to <var>url</var>, with
   <var>baseURL</var> and <var>encoding</var>.</p></li>
  </ol>

  <p>To <dfn data-x="encoding-parsing-and-serializing a URL">encoding-parse-and-serialize a
  URL</dfn>, given a string <var>url</var>, relative to a <code>Document</code> object or
  <span>environment settings object</span> <var>environment</var>, run these steps. They return
  failure or a string.</p>

  <ol>
   <li><p>Let <var>url</var> be the result of <span>encoding-parsing a URL</span> given
   <var>url</var>, relative to <var>environment</var>.</p></li>

   <li><p>If <var>url</var> is failure, then return failure.</p></li>

   <li><p>Return the result of applying the <span data-x="concept-url-serializer">URL
   serializer</span> to <var>url</var>.</p></li>
  </ol>

  </div>



  <div w-nodev>

  <h4>Dynamic changes to base URLs</h4>

  <p>When a document's <span>document base URL</span> changes, all elements in that document are
  <span>affected by a base URL change</span>.</p>

  <p>The following are <span>base URL change steps</span>, which run when an element is
  <span>affected by a base URL change</span> (as defined by <cite>DOM</cite>):</p>

  <dl class="switch">

   <dt>If the element creates a <span>hyperlink</span></dt>

   <dd>
    <p>If the <span>URL</span> identified by the hyperlink is being shown to the user, or if any
    data derived from that <span>URL</span> is affecting the display, then the <code
    data-x="attr-hyperlink-href">href</code> attribute's value should be <span
    data-x="encoding-parsing a URL">reparsed</span>, relative to the element's <span>node
    document</span> and the UI updated appropriately.</p>

    <p class="example">For example, the CSS <code data-x="selector-link">:link</code>/<code
    data-x="selector-visited">:visited</code> <span data-x="pseudo-class">pseudo-classes</span>
    might have been affected.</p>

    <p>If the hyperlink has a <code data-x="attr-hyperlink-ping">ping</code> attribute and its
    <span data-x="URL">URL(s)</span> are being shown to the user, then the <code
    data-x="attr-hyperlink-ping">ping</code> attribute's tokens should be <span
    data-x="encoding-parsing a URL">reparsed</span>, relative to the element's <span>node
    document</span> and the UI updated appropriately.</p>
   </dd>

   <dt>If the element is a <code>q</code>, <code>blockquote</code>, <code>ins</code>, or
   <code>del</code> element with a <code data-x="">cite</code> attribute</dt>

   <dd>
    <p>If the <span>URL</span> identified by the <code data-x="">cite</code> attribute is being
    shown to the user, or if any data derived from that <span>URL</span> is affecting the display,
    then the <code data-x="">cite</code> attribute's value should be <span data-x="encoding-parsing
    a URL">reparsed</span>, relative to the element's <span>node document</span> and the UI updated
    appropriately.</p>
   </dd>

   <dt>Otherwise</dt>

   <dd>
    <p>The element is not directly affected.</p>

    <p class="example">For instance, changing the base URL doesn't affect the image displayed by
    <code>img</code> elements, although subsequent accesses of the <code
    data-x="dom-img-src">src</code> IDL attribute from script will return a new <span>absolute
    URL</span> that might no longer correspond to the image being shown.</p>
   </dd>
  </dl>

  </div>


  <div w-nodev>

  <h3>Fetching resources</h3>

  <h4>Terminology</h4>

  <p>A <span data-x="concept-response">response</span> whose <span
  data-x="concept-response-type">type</span> is "<code data-x="">basic</code>", "<code
  data-x="">cors</code>", or "<code data-x="">default</code>" is <dfn export>CORS-same-origin</dfn>.
  <ref>FETCH</ref></p>

  <p>A <span data-x="concept-response">response</span> whose <span
  data-x="concept-response-type">type</span> is "<code data-x="">opaque</code>" or "<code
  data-x="">opaqueredirect</code>" is <dfn>CORS-cross-origin</dfn>.</p>

  <p>A <span data-x="concept-response">response</span>'s <dfn export>unsafe response</dfn> is its
  <span data-x="concept-internal-response">internal response</span> if it has one, and the <span
  data-x="concept-response">response</span> itself otherwise.</p>

  <p>To <dfn>create a potential-CORS request</dfn>, given a <var>url</var>, <var>destination</var>,
  <var>corsAttributeState</var>, and an optional <i>same-origin fallback flag</i>, run these
  steps:</p>

  <ol>
   <li><p>Let <var>mode</var> be "<code data-x="">no-cors</code>" if <var>corsAttributeState</var>
   is <span data-x="attr-crossorigin-none">No CORS</span>, and "<code data-x="">cors</code>"
   otherwise.</p></li>

   <li><p>If <i>same-origin fallback flag</i> is set and <var>mode</var> is "<code
   data-x="">no-cors</code>", set <var>mode</var> to "<code data-x="">same-origin</code>".</p></li>

   <li><p>Let <var>credentialsMode</var> be "<code data-x="">include</code>".</p></li>

   <li><p>If <var>corsAttributeState</var> is <span
   data-x="attr-crossorigin-anonymous">Anonymous</span>, set <var>credentialsMode</var> to "<code
   data-x="">same-origin</code>".</p></li>

   <li><p>Return a new <span data-x="concept-request">request</span> whose
   <span data-x="concept-request-url">URL</span> is <var>url</var>, <span
   data-x="concept-request-destination">destination</span> is <var>destination</var>,
   <span data-x="concept-request-mode">mode</span> is <var>mode</var>, <span
   data-x="concept-request-credentials-mode">credentials mode</span> is
   <var>credentialsMode</var>, and whose <span>use-URL-credentials flag</span> is set.</p></li>
  </ol>


  <h4 id="content-type-sniffing">Determining the type of a resource</h4>

  <p>The <dfn data-x="Content-Type">Content-Type metadata</dfn> of a resource must be obtained and
  interpreted in a manner consistent with the requirements of <cite>MIME Sniffing</cite>.
  <ref>MIMESNIFF</ref></p>

  <p>The <dfn data-x="Content-Type sniffing"
  data-x-href="https://mimesniff.spec.whatwg.org/#computed-mime-type">computed MIME type</dfn> of a
  resource must be found in a manner consistent with the requirements given in <cite>MIME
  Sniffing</cite>. <ref>MIMESNIFF</ref></p>

  <p>The <dfn data-x="Content-Type sniffing: image"
  data-x-href="https://mimesniff.spec.whatwg.org/#rules-for-sniffing-images-specifically">rules for
  sniffing images specifically</dfn>, the <dfn data-x="Content-Type sniffing: text or binary"
  data-x-href="https://mimesniff.spec.whatwg.org/#rules-for-text-or-binary">rules for distinguishing
  if a resource is text or binary</dfn>, and the <dfn data-x="Content-Type sniffing: video"
  data-x-href="https://mimesniff.spec.whatwg.org/#rules-for-sniffing-audio-and-video-specifically">rules
  for sniffing audio and video specifically</dfn> are also defined in <cite>MIME Sniffing</cite>.
  These rules return a <span>MIME type</span> as their result. <ref>MIMESNIFF</ref></p>

  <p class="warning">It is imperative that the rules in <cite>MIME Sniffing</cite> be
  followed exactly. When a user agent uses different heuristics for content type detection than the
  server expects, security problems can occur. For more details, see <cite>MIME Sniffing</cite>.
  <ref>MIMESNIFF</ref></p>


  <h4>Extracting character encodings from <code>meta</code> elements</h4>

  <p>The <dfn>algorithm for extracting a character encoding from a <code>meta</code> element</dfn>,
  given a string <var>s</var>, is as follows. It returns either a character encoding or
  nothing.</p>

  <ol> <!-- https://www.hixie.ch/tests/adhoc/html/parsing/encoding/all.html -->

   <li><p>Let <var>position</var> be a pointer into <var>s</var>, initially
   pointing at the start of the string.</p></li>

   <li><p><i>Loop</i>: Find the first seven characters in <var>s</var> after <var>position</var> that are an <span>ASCII case-insensitive</span> match for the word "<code
   data-x="">charset</code>". If no such match is found, return nothing.</p></li>

   <li><p>Skip any <span>ASCII whitespace</span> that immediately follow the word "<code
   data-x="">charset</code>" (there might not be any).</p></li>

   <li><p>If the next character is not a U+003D EQUALS SIGN (=), then move <var>position</var> to point just before that next character, and jump back to the step
   labeled <i>loop</i>.</p></li>

   <li><p>Skip any <span>ASCII whitespace</span> that immediately follow the equals sign (there
   might not be any).</p></li>

   <li>
    <p>Process the next character as follows:</p>

    <dl class="switch">

     <dt>If it is a U+0022 QUOTATION MARK character (") and there is a later U+0022 QUOTATION MARK character (") in <var>s</var></dt>
     <dt>If it is a U+0027 APOSTROPHE character (') and there is a later U+0027 APOSTROPHE character (') in <var>s</var></dt>
     <dd>Return the result of <span>getting an encoding</span> from the substring that is between
     this character and the next earliest occurrence of this character.</dd>

     <dt>If it is an unmatched U+0022 QUOTATION MARK character (")</dt>
     <dt>If it is an unmatched U+0027 APOSTROPHE character (')</dt>
     <dt>If there is no next character</dt>
     <dd>Return nothing.</dd>

     <dt>Otherwise</dt>
     <dd>Return the result of <span>getting an encoding</span> from the substring that consists of
     this character up to but not including the first <span>ASCII whitespace</span> or U+003B
     SEMICOLON character (;), or the end of <var>s</var>, whichever comes first.</dd>
    </dl>
   </li>
  </ol>

  <p class="note">This algorithm is distinct from those in the HTTP specifications (for example,
  HTTP doesn't allow the use of single quotes and requires supporting a backslash-escape mechanism
  that is not supported by this algorithm<!-- not to mention not having any rules for
  error-handling, which is of course why we're having to define it ourselves -->). While the
  algorithm is used in contexts that, historically, were related to HTTP, the syntax as supported by
  implementations diverged some time ago. <ref>HTTP</ref></p>

  </div>


  <h4>CORS settings attributes</h4>

  <p>A <dfn>CORS settings attribute</dfn> is an <span>enumerated attribute</span> with the following
  keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th> Keyword
     <th> State
     <th> Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="audio/crossorigin,video/crossorigin,img/crossorigin,link/crossorigin,script/crossorigin"><code data-x="attr-crossorigin-anonymous-keyword">anonymous</code></dfn>
     <td rowspan=2><dfn data-x="attr-crossorigin-anonymous">Anonymous</dfn>
     <td rowspan=2><span data-x="concept-request">Requests</span> for the element will have their
     <span data-x="concept-request-mode">mode</span> set to "<code data-x="">cors</code>" and their
     <span data-x="concept-request-credentials-mode">credentials mode</span> set to "<code
     data-x="">same-origin</code>".
    <tr>
     <td>(the empty string)
    <tr>
     <td><dfn attr-value for="audio/crossorigin,video/crossorigin,img/crossorigin,link/crossorigin,script/crossorigin"><code data-x="attr-crossorigin-use-credentials-keyword">use-credentials</code></dfn>
     <td><dfn data-x="attr-crossorigin-use-credentials">Use Credentials</dfn>
     <td><span data-x="concept-request">Requests</span> for the element will have their <span
     data-x="concept-request-mode">mode</span> set to "<code data-x="">cors</code>" and their <span
     data-x="concept-request-credentials-mode">credentials mode</span> set to "<code
     data-x="">include</code>".
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> is the <dfn
  data-x="attr-crossorigin-none">No CORS</dfn> state, and its <i data-x="invalid value
  default">invalid value default</i> is the <span
  data-x="attr-crossorigin-anonymous">Anonymous</span> state. For the purposes of <span
  data-x="reflect">reflection</span>, the <span>canonical keyword</span> for the <span
  data-x="attr-crossorigin-anonymous">Anonymous</span> state is the <code
  data-x="attr-crossorigin-anonymous-keyword">anonymous</code> keyword.</p>

  <p>The majority of fetches governed by <span data-x="CORS settings attribute">CORS settings
  attributes</span> will be done via the <span>create a potential-CORS request</span> algorithm.</p>

  <p id="module-script-credentials-mode">For more modern features, where the request's <span
  data-x="concept-request-mode">mode</span> is always "<code data-x="">cors</code>", certain <span
  data-x="CORS settings attribute">CORS settings attributes</span> have been repurposed to have a
  slightly different meaning, wherein they only impact the <span
  data-x="concept-request">request</span>'s <span
  data-x="concept-request-credentials-mode">credentials mode</span>. To perform this translation, we
  define the <dfn export>CORS settings attribute credentials mode</dfn> for a given <span>CORS
  settings attribute</span> to be determined by switching on the attribute's state:</p>

  <dl class="switch">
   <dt><span data-x="attr-crossorigin-none">No CORS</span></dt>
   <dt><span data-x="attr-crossorigin-anonymous">Anonymous</span></dt>
   <dd>"<code data-x="">same-origin</code>"</dd>

   <dt><span data-x="attr-crossorigin-none">Use Credentials</span></dt>
   <dd>"<code data-x="">include</code>"</dd>
  </dl>


  <h4>Referrer policy attributes</h4>

  <p>A <dfn export>referrer policy attribute</dfn> is an <span>enumerated attribute</span>.  Each
  <span>referrer policy</span>, including the empty string, is a keyword for this attribute, mapping
  to a state of the same name.</p>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the empty string state.</p>

  <p>The impact of these states on the processing model of various <span
  data-x="concept-fetch">fetches</span> is defined in more detail throughout this specification, in
  <cite>Fetch</cite>, and in <cite>Referrer Policy</cite>. <ref>FETCH</ref>
  <ref>REFERRERPOLICY</ref></p>

  <div class="note">
   <p>Several signals can contribute to which processing model is used for a given <span
   data-x="concept-fetch">fetch</span>; a <span>referrer policy attribute</span> is only one of
   them. In general, the order in which these signals are processed are:</p>

   <ol>
    <li><p>First, the presence of a <code data-x="rel-noreferrer">noreferrer</code> link
    type;</p></li>
    <li><p>Then, the value of a <span>referrer policy attribute</span>;</p></li>
    <li><p>Then, the presence of any <code>meta</code> element with <code
    data-x="attr-meta-name">name</code> attribute set to <code
    data-x="meta-referrer">referrer</code>.</p></li>
    <li><p>Finally, the `<code data-x="http-referrer-policy">Referrer-Policy</code>` HTTP
    header.</p></li>
   </ol>
  </div>

  <h4>Nonce attributes</h4>

  <p>A <dfn element-attr for="htmlsvg-global" data-x="attr-nonce"><code>nonce</code></dfn> content
  attribute represents a cryptographic nonce ("number used once") which can be used by <cite>Content
  Security Policy</cite> to determine whether or not a given fetch will be allowed to proceed. The
  value is text. <ref>CSP</ref></p>

  <p>Elements that have a <code data-x="attr-nonce">nonce</code> content attribute ensure that the
  cryptographic nonce is only exposed to script (and not to side-channels like CSS attribute
  selectors) by taking the value from the content attribute, moving it into an internal slot
  named <dfn for="HTMLOrSVGElement" attribute>[[CryptographicNonce]]</dfn>, exposing it to script
  via the <code>HTMLOrSVGElement</code> interface mixin, and setting the content attribute to the
  empty string. Unless otherwise specified, the slot's value is the empty string.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-HTMLOrSVGElement-nonce">nonce</span></code></dt>
   <dd><p>Returns the value set for <var>element</var>'s cryptographic nonce. If the setter was not
   used, this will be the value originally found in the <code data-x="attr-nonce">nonce</code>
   content attribute.</p></dd>

   <dt><code data-x=""><var>element</var>.<span data-x="dom-HTMLOrSVGElement-nonce">nonce</span> = <var>value</var></code></dt>
   <dd><p>Updates <var>element</var>'s cryptographic nonce value.</p></dd>
  </dl>

  <p>The <dfn id="dom-noncedelement-nonce" attribute for="HTMLOrSVGElement"><code
  data-x="dom-HTMLOrSVGElement-nonce">nonce</code></dfn> IDL attribute must, on getting, return the
  value of this element's <span>[[CryptographicNonce]]</span>; and on setting, set this element's
  <span>[[CryptographicNonce]]</span> to the given value.</p>

  <p class="note" id="nonce-does-not-update-dom">Note how the setter for the <code
  data-x="dom-HTMLOrSVGElement-nonce">nonce</code> IDL attribute does not update the corresponding
  content attribute. This, as well as the below setting of the <code
  data-x="attr-nonce">nonce</code> content attribute to the empty string when an element
  <span>becomes browsing-context connected</span>, is meant to prevent exfiltration of the nonce
  value through mechanisms that can easily read content attributes, such as selectors. Learn more in
  <a href="https://github.com/whatwg/html/issues/2369">issue #2369</a>, where this behavior was
  introduced.</p>

  <p>The following <span data-x="concept-element-attributes-change-ext">attribute change
  steps</span> are used for the <code data-x="attr-nonce">nonce</code> content attribute:

  <ol>
   <li><p>If <var>element</var> does not <span>include</span> <code>HTMLOrSVGElement</code>, then
   return.</p></li>

   <li><p>If <var>localName</var> is not <code data-x="attr-nonce">nonce</code> or
   <var>namespace</var> is not null, then return.</p></li>

   <li><p>If <var>value</var> is null, then set <var>element</var>'s
   <span>[[CryptographicNonce]]</span> to the empty string.</p></li>

   <li><p>Otherwise, set <var>element</var>'s <span>[[CryptographicNonce]]</span> to
   <var>value</var>.</p></li>
  </ol>

  <p>Whenever an element <span data-x="include">including</span> <code>HTMLOrSVGElement</code>
  <span>becomes browsing-context connected</span>, the user agent must execute the following steps
  on the <var>element</var>:</p>

  <ol>
   <li><p>Let <var>CSP list</var> be <var>element</var>'s <span data-x="shadow-including
   root">shadow-including root</span>'s <span data-x="concept-document-policy-container">policy
   container</span>'s <span data-x="policy-container-csp-list">CSP list</span>.</p></li>

   <li>
    <p>If <var>CSP list</var> <span>contains a header-delivered Content Security Policy</span>, and
    <var>element</var> has a <code data-x="attr-nonce">nonce</code> content attribute
    whose value is not the empty string, then:</p>

    <ol>
     <li><p>Let <var>nonce</var> be <var>element</var>'s
     <span>[[CryptographicNonce]]</span>.</p></li>

     <li><p><span data-x="concept-element-attributes-set-value">Set an attribute value</span> for
     <var>element</var> using "<code data-x="attr-nonce">nonce</code>" and the empty
     string.</p></li>

     <li><p>Set <var>element</var>'s <span>[[CryptographicNonce]]</span> to
     <var>nonce</var>.</p></li>
    </ol>

    <p class="note">If <var>element</var>'s <span>[[CryptographicNonce]]</span> were not restored it
    would be the empty string at this point.</p>
   </li>
  </ol>

  <p>The <span data-x="concept-node-clone-ext">cloning steps</span> for elements that
  <span>include</span> <code>HTMLOrSVGElement</code> given <var>node</var>, <var>copy</var>, and
  <var>subtree</var> are to set <var>copy</var>'s <span>[[CryptographicNonce]]</span> to
  <var>node</var>'s <span>[[CryptographicNonce]]</span>.</p>

  <h4>Lazy loading attributes</h4>

  <p>A <dfn>lazy loading attribute</dfn> is an <span>enumerated attribute</span> with the following
  keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="img/loading,iframe/loading"><code data-x="attr-loading-lazy">lazy</code></dfn>
     <td><dfn data-x="attr-loading-lazy-state">Lazy</dfn>
     <td>Used to defer fetching a resource until some conditions are met.
    <tr>
     <td><dfn attr-value for="img/loading,iframe/loading"><code data-x="attr-loading-eager">eager</code></dfn>
     <td><dfn data-x="attr-loading-eager-state">Eager</dfn>
     <td>Used to fetch a resource immediately; the default state.
  </table>

  <p>The attribute directs the user agent to fetch a resource immediately or to defer fetching until
  some conditions associated with the element are met, according to the attribute's current
  state.</p>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <span
  data-x="attr-loading-eager-state">Eager</span> state.</p>

  <hr>

  <p>The <dfn>will lazy load element steps</dfn>, given an element <var>element</var>,
  are as follows:</p>

  <ol>
   <li>
    <p>If <span data-x="concept-n-noscript">scripting is disabled</span> for <var>element</var>,
    then return false.</p>

    <p class="note">This is an anti-tracking measure, because if a user agent supported lazy loading
    when scripting is disabled, it would still be possible for a site to track a user's approximate
    scroll position throughout a session, by strategically placing images in a page's markup such
    that a server can track how many images are requested and when.</p>
   </li>

   <li><p>If <var>element</var>'s <span>lazy loading attribute</span> is in the <span
   data-x="attr-loading-lazy-state">Lazy</span> state, then return true.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  <p>Each <code>img</code> and <code>iframe</code> element has associated <dfn>lazy load resumption
  steps</dfn>, initially null.</p>

  <p class="note">For <code>img</code> and <code>iframe</code> elements that <span data-x="will lazy
  load element steps">will lazy load</span>, these steps are run from the <span>lazy load
  intersection observer</span>'s callback or when their <span>lazy loading attribute</span> is set
  to the <span data-x="attr-loading-eager-state">Eager</span> state. This causes the element to
  continue loading.</p>

  <p>Each <code>Document</code> has a <dfn>lazy load intersection observer</dfn>, initially set to
  null but can be set to an <code>IntersectionObserver</code> instance.</p>

  <p>To <dfn>start intersection-observing a lazy loading element</dfn> <var>element</var>, run these
  steps:</p>

  <ol>
   <li><p>Let <var>doc</var> be <var>element</var>'s <span>node document</span>.</p></li>

   <li>
    <p>If <var>doc</var>'s <span>lazy load intersection observer</span> is null, set it to a new
    <code>IntersectionObserver</code> instance, initialized as follows:</p>

    <p class="XXX">The intention is to use the original value of the
    <code>IntersectionObserver</code> constructor. However, we're forced to use the
    JavaScript-exposed constructor in this specification, until <cite>Intersection Observer</cite>
    exposes low-level hooks for use in specifications. See bug <a
    href="https://github.com/w3c/IntersectionObserver/issues/464">w3c/IntersectionObserver#464</a>
    which tracks this. <ref>INTERSECTIONOBSERVER</ref></p>

    <ul>
     <li>
      <p>The <var>callback</var> is these steps, with arguments <var>entries</var> and
      <var>observer</var>:</p>

      <ol>
       <li>
        <p>For each <var>entry</var> in <var>entries</var> <span class="XXX">using a method of
        iteration which does not trigger developer-modifiable array accessors or
        iteration hooks</span>:</p>

        <ol>
         <li><p>Let <var>resumptionSteps</var> be null.</p></li>

         <li><p>If <var>entry</var>.<code
         data-x="dom-IntersectionObserverEntry-isIntersecting">isIntersecting</code> is true, then
         set <var>resumptionSteps</var> to <var>entry</var>.<code
         data-x="dom-IntersectionObserverEntry-target">target</code>'s
         <span>lazy load resumption steps</span>.</p></li>

         <li><p>If <var>resumptionSteps</var> is null, then return.</p></li>

         <li><p><span>Stop intersection-observing a lazy loading element</span> for
         <var>entry</var>.<code data-x="dom-IntersectionObserverEntry-target">target</code>.</p></li>

         <li><p>Set <var>entry</var>.<code
         data-x="dom-IntersectionObserverEntry-target">target</code>'s <span>lazy load resumption
         steps</span> to null.</p></li>

         <li><p>Invoke <var>resumptionSteps</var>.</p></li>
        </ol>

        <p class="XXX">The intention is to use the original value of the
        <code data-x="dom-IntersectionObserverEntry-isIntersecting">isIntersecting</code> and
        <code data-x="dom-IntersectionObserverEntry-target">target</code> getters. See <a
        href="https://github.com/w3c/IntersectionObserver/issues/464">w3c/IntersectionObserver#464</a>.
        <ref>INTERSECTIONOBSERVER</ref></p>
       </li>
      </ol>
     </li>

     <li>
      <p>The <var>options</var> is an <code>IntersectionObserverInit</code> dictionary with the
      following dictionary members: «[ "<code data-x="">scrollMargin</code>" → <span>lazy load scroll
      margin</span> ]»</p>

      <p class="note">This allows for fetching the image during scrolling, when it does not yet —
      but is about to — intersect the viewport.</p>

      <p class="XXX">The <span>lazy load scroll margin</span> suggestions imply dynamic changes to the
      value, but the <code>IntersectionObserver</code> API does not support changing the scroll
      margin. See issue <a
      href="https://github.com/w3c/IntersectionObserver/issues/428">w3c/IntersectionObserver#428</a>.</p>
     </li>
    </ul>
   </li>

   <li>
    <p>Call <var>doc</var>'s <span>lazy load intersection observer</span>'s <code
    data-x="dom-IntersectionObserver-observe">observe</code> method with <var>element</var> as the
    argument.</p>

    <p class="XXX">The intention is to use the original value of the <code
    data-x="dom-IntersectionObserver-observe">observe</code> method. See <a
    href="https://github.com/w3c/IntersectionObserver/issues/464">w3c/IntersectionObserver#464</a>.
    <ref>INTERSECTIONOBSERVER</ref></p>
   </li>
  </ol>

  <p>To <dfn>stop intersection-observing a lazy loading element</dfn> <var>element</var>, run these
  steps:</p>

  <ol>
   <li><p>Let <var>doc</var> be <var>element</var>'s <span>node document</span>.</p></li>

   <li><p><span>Assert</span>: <var>doc</var>'s <span>lazy load intersection observer</span> is not
   null.</p></li>

   <li>
    <p>Call <var>doc</var>'s <span>lazy load intersection observer</span>'s <code
    data-x="dom-IntersectionObserver-unobserve">unobserve</code> method with <var>element</var> as
    the argument.</p>

    <p class="XXX">The intention is to use the original value of the <code
    data-x="dom-IntersectionObserver-unobserve">unobserve</code> method. See <a
    href="https://github.com/w3c/IntersectionObserver/issues/464">w3c/IntersectionObserver#464</a>.
    <ref>INTERSECTIONOBSERVER</ref></p>
   </li>
  </ol>

  <p>
  <!--INSERT TRACKING-->
  The <dfn id="lazy-load-root-margin">lazy load scroll margin</dfn> is an
  <span>implementation-defined</span> value, but with the following suggestions to consider:</p>

  <ul>
   <li><p>Set a minimum value that most often results in the resources being loaded before they
   intersect the viewport under normal usage patterns for the given device.</p></li>

   <li><p>The typical scrolling speed: increase the value for devices with faster typical scrolling
   speeds.</p></li>

   <li><p>The current scrolling speed or momentum: the UA can attempt to predict where the scrolling
   will likely stop, and adjust the value accordingly.</p></li>

   <li><p>The network quality: increase the value for slow or high-latency connections.</p></li>

   <li><p>User preferences can influence the value.</p></li>
  </ul>

  <p class="note">It is important <span data-x="tracking vector">for privacy</span> that the
  <span>lazy load scroll margin</span> not leak additional information. For example, the typical
  scrolling speed on the current device could be imprecise so as to not introduce a new
  fingerprinting vector.</p>

  <h4>Blocking attributes</h4>

  <p>A <dfn>blocking attribute</dfn> explicitly indicates that certain operations should be blocked
  on the fetching of an external resource. The operations that can be blocked are represented by
  <dfn data-x="possible blocking token">possible blocking tokens</dfn>, which are strings listed by
  the following table:</p>

  <table>
   <thead>
    <tr>
     <th>Possible blocking token
     <th>Description
   <tbody>
    <tr>
     <td>"<dfn><code data-x="blocking-token-render">render</code></dfn>"
     <td>The element is <span>potentially render-blocking</span>.
  </table>


  <p class="note" id="future-blocking-tokens">In the future, there might be more <span
  data-x="possible blocking token">possible blocking tokens</span>.</p>

  <p>A <span>blocking attribute</span> must have a value that is an <span>unordered set of unique
  space-separated tokens</span>, each of which are <span data-x="possible blocking token">possible
  blocking tokens</span>. The <span data-x="concept-supported-tokens">supported tokens</span> of a
  <span>blocking attribute</span> are the <span data-x="possible blocking token">possible blocking
  tokens</span>. Any element can have at most one <span>blocking attribute</span>.</p>

  <p>The <dfn>blocking tokens set</dfn> for an element <var>el</var> are the result of the
  following steps:</p>

  <ol>
   <li><p>Let <var>value</var> be the value of <var>el</var>'s <span>blocking attribute</span>, or
   the empty string if no such attribute exists.</p>

   <li><p>Set <var>value</var> to <var>value</var>, <span>converted to ASCII lowercase</span>.</p>

   <li><p>Let <var>rawTokens</var> be the result of <span data-x="split a string on ASCII
   whitespace">splitting <var>value</var> on ASCII whitespace</span>.</p>

   <li><p>Return a set containing the elements of <var>rawTokens</var> that are <span
   data-x="possible blocking token">possible blocking tokens</span>.</p>
  </ol>

  <p>An element is <dfn>potentially render-blocking</dfn> if its <span>blocking tokens set</span>
  contains "<code data-x="blocking-token-render">render</code>", or if it is
  <dfn>implicitly potentially render-blocking</dfn>, which will be defined at the individual
  elements. By default, an element is not <span>implicitly potentially render-blocking</span>.</p>

  <h4>Fetch priority attributes</h4>

  <p>A <dfn>fetch priority attribute</dfn> is an <span>enumerated attribute</span> with the
  following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value
     for="img/fetchpriority,script/fetchpriority,link/fetchpriority"
     data-x="attr-fetchpriority-high"><code>high</code></dfn>
     <td><dfn data-x="attr-fetchpriority-high-state">high</dfn>
     <td>Signals a high-priority <span data-x="concept-fetch">fetch</span> relative to other
     resources with the same <span data-x="concept-request-destination">destination</span>.
    <tr>
     <td><dfn attr-value
     for="img/fetchpriority,script/fetchpriority,link/fetchpriority"
     data-x="attr-fetchpriority-low"><code>low</code></dfn>
     <td><dfn data-x="attr-fetchpriority-low-state">low</dfn>
     <td>Signals a low-priority <span data-x="concept-fetch">fetch</span> relative to other
     resources with the same <span data-x="concept-request-destination">destination</span>.
    <tr>
     <td><dfn attr-value
     for="img/fetchpriority,script/fetchpriority,link/fetchpriority"
     data-x="attr-fetchpriority-auto"><code>auto</code></dfn>
     <td><dfn data-x="attr-fetchpriority-auto-state">auto</dfn>
     <td>Signals automatic determination of <span data-x="concept-fetch">fetch</span> priority
     relative to other resources with the same <span
     data-x="concept-request-destination">destination</span>.
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and
  <i data-x="invalid value default">invalid value default</i> are both the
  <span data-x="attr-fetchpriority-auto-state">auto</span> state.</p>

  <h3 split-filename="common-dom-interfaces">Common DOM interfaces</h3>

  <h4>Reflecting content attributes in IDL attributes</h4>

  <p>The building blocks for reflecting are as follows:</p>

  <ul>
   <li><p>A <dfn>reflected target</dfn> is an element or <code>ElementInternals</code>
   object. It is typically clear from context and typically identical to the interface of the
   <span>reflected IDL attribute</span>. It is always identical to that interface when it is an
   <code>ElementInternals</code> object.</p></li>

   <li><p>A <dfn>reflected IDL attribute</dfn> is an attribute interface member.</p></li>

   <li><p>A <dfn>reflected content attribute name</dfn> is a string. When the <span>reflected
   target</span> is an element, it represents the local name of a content attribute whose namespace
   is null. When the <span>reflected target</span> is an <code>ElementInternals</code> object, it
   represents a key of the <span>reflected target</span>'s <span data-x="internals-target">target
   element</span>'s <span>internal content attribute map</span>.</p></li>
  </ul>

  <p>A <span>reflected IDL attribute</span> can be defined to <dfn export>reflect</dfn> a
  <span>reflected content attribute name</span> of a <span>reflected target</span>. In general this
  means that the IDL attribute getter returns the current value of the content attribute, and the
  setter changes the value of the content attribute to the given value.</p>

  <p>If the <span>reflected target</span> is an element, then the <span>reflected IDL
  attribute</span> can additionally declare to <dfn>support <code>ElementInternals</code></dfn>.
  This means that the <code>ElementInternals</code> interface also has a <span>reflected IDL
  attribute</span>, with the same identifier, and that <span>reflected IDL attribute</span> <span
  data-x="reflect">reflects</span> the same <span>reflected content attribute name</span>.</p>

  <p class="example" id="example-reflect-incantation">The <code data-x=""><var>fooBar</var></code>
  IDL attribute <!--non-normative-->must <span>reflect</span> the <code
  data-x=""><var>foobar</var></code> content attribute and <span>support
  <code>ElementInternals</code></span>.</p>

  <div w-nodev>

  <p><span data-x="reflected target">Reflected targets</span> have these associated algorithms:

  <ul class="brief">
   <li><dfn>get the element</dfn>: takes no arguments; returns an element.</li>
   <li><dfn>get the content attribute</dfn>: takes no arguments; returns null or a string.</li>
   <li><dfn>set the content attribute</dfn>: takes a string <var>value</var>; returns nothing.</li>
   <li><dfn>delete the content attribute</dfn>: takes no arguments; returns nothing.</li>
  </ul>

  <p>For a <span>reflected target</span> that is an element <var>element</var>, these are defined as
  follows:</p>

  <dl>
   <dt><span>get the element</span></dt>
   <dd><ol><li><p>Return <var>element</var>.</p></li></ol></dd>

   <dt><span>get the content attribute</span></dt>
   <dd>
    <ol>
     <li><p>Let <var>attribute</var> be the result of running <span
     data-x="concept-element-attributes-get-by-namespace">get an attribute by namespace and local
     name</span> given null, the <span>reflected content attribute name</span>, and
     <var>element</var>.</p></li>

     <li><p>If <var>attribute</var> is null, then return null.</p></li>

     <li><p>Return <var>attribute</var>'s <span
     data-x="concept-attribute-value">value</span>.</p></li>
    </ol>
   </dd>

   <dt><span>set the content attribute</span> with a string <var>value</var></dt>
   <dd><ol><li><p><span data-x="concept-element-attributes-set-value">Set an attribute value</span>
   given <var>element</var>, the <span>reflected content attribute name</span>, and
   <var>value</var>.</p></li></ol></dd>

   <dt><span>delete the content attribute</span></dt>
   <dd><ol><li><p><span data-x="concept-element-attributes-remove-by-namespace">Remove an attribute
   by namespace and local name</span> given null, the <span>reflected content attribute name</span>,
   and <var>element</var>.</p></li></ol></dd>
  </dl>

  <p>For a <span>reflected target</span> that is an <code>ElementInternals</code> object
  <var>elementInternals</var>, they are defined as follows:</p>

  <dl>
   <dt><span>get the element</span></dt>
   <dd><ol><li><p>Return <var>elementInternals</var>'s <span data-x="internals-target">target
   element</span>.</p></li></ol></dd>

   <dt><span>get the content attribute</span></dt>
   <dd>
    <ol>
     <li><p>If <var>elementInternals</var>'s <span data-x="internals-target">target element</span>'s
     <span>internal content attribute map</span>[the <span>reflected content attribute name</span>]
     <span data-x="map exists">does not exist</span>, then return null.

     <li><p>Return <var>elementInternals</var>'s <span data-x="internals-target">target
     element</span>'s <span>internal content attribute map</span>[the <span>reflected content
     attribute name</span>].</p></li>
    </ol>
   </dd>

   <dt><span>set the content attribute</span> with a string <var>value</var></dt>
   <dd><ol><li><p><span data-x="map set">Set</span> <var>elementInternals</var>'s <span
   data-x="internals-target">target element</span>'s <span>internal content attribute map</span>[the
   <span>reflected content attribute name</span>] to <var>value</var>.</p></li></ol></dd>

   <dt><span>delete the content attribute</span></dt>
   <dd><ol><li><p><span data-x="map remove">Remove</span> <var>elementInternals</var>'s <span
   data-x="internals-target">target element</span>'s <span>internal content attribute map</span>[the
   <span>reflected content attribute name</span>].</p></li></ol></dd>
  </dl>

  <p class="note">This results in somewhat redundant data structures for
  <code>ElementInternals</code> objects as their <span data-x="internals-target">target
  element</span>'s <span>internal content attribute map</span> cannot be directly manipulated and as
  such reflection is only happening in a single direction. This approach was nevertheless chosen to
  make it less error-prone to define IDL attributes that are shared between <span data-x="reflected
  target">reflected targets</span> and benefit from common API semantics.</p>

  <hr>

  <p>IDL attributes of type <code data-x="idl-DOMString">DOMString</code> or <code data-x=""><span
  data-x="idl-DOMString">DOMString</span>?</code> that <span>reflect</span> <span data-x="enumerated
  attribute">enumerated</span> content attributes can be <dfn>limited to only known values</dfn>.
  Per the processing models below, those will cause the getters for such IDL attributes to only
  return keywords for those enumerated attributes, or the empty string or null.</p>

  <p>If a <span>reflected IDL attribute</span> has the type <code
  data-x="idl-DOMString">DOMString</code>:</p>

  <ul>
   <li>
    <p>The getter steps are:</p>

    <ol>
     <li><p>Let <var>element</var> be the result of running <span>this</span>'s <span>get the
     element</span>.</p></li>

     <li><p>Let <var>contentAttributeValue</var> be the result of running <span>this</span>'s
     <span>get the content attribute</span>.</p></li>

     <li><p>Let <var>attributeDefinition</var> be the attribute definition of <var>element</var>'s
     content attribute whose namespace is null and local name is the <span>reflected content
     attribute name</span>.

     <li>
      <p>If <var>attributeDefinition</var> indicates it is an <span>enumerated attribute</span> and
      the <span>reflected IDL attribute</span> is defined to be <span>limited to only known
      values</span>:</p>

      <ol>
       <li><p>If <var>contentAttributeValue</var> does not correspond to any state of
       <var>attributeDefinition</var> (e.g., it is null and there is no <i data-x="missing value
       default">missing value default</i>), or if it is in a state of
       <var>attributeDefinition</var> with no associated keyword value, then return the empty
       string.</p></li>

       <li><p>Return the <span>canonical keyword</span> for the state of
       <var>attributeDefinition</var> that <var>contentAttributeValue</var> corresponds to.</p></li>
      </ol>
     </li>

     <li><p>If <var>contentAttributeValue</var> is null, then return the empty string.</p></li>

     <li><p>Return <var>contentAttributeValue</var>.</p></li>
    </ol>
   </li>

   <li><p>The setter steps are to run <span>this</span>'s <span>set the content attribute</span>
   with the given value.</p></li>
  </ul>

  <p>If a <span>reflected IDL attribute</span> has the type <code data-x=""><span
  data-x="idl-DOMString">DOMString</span>?</code>:</p>

  <ul>
   <li>
    <p>The getter steps are:</p>

    <ol>
     <li><p>Let <var>element</var> be the result of running <span>this</span>'s <span>get the
     element</span>.</p></li>

     <li><p>Let <var>contentAttributeValue</var> be the result of running <span>this</span>'s
     <span>get the content attribute</span>.</p></li>

     <li><p>Let <var>attributeDefinition</var> be the attribute definition of <var>element</var>'s
     content attribute whose namespace is null and local name is the <span>reflected content
     attribute name</span>.

     <li>
      <p>If <var>attributeDefinition</var> indicates it is an <span>enumerated attribute</span>:</p>

      <ol>
       <li><p><span>Assert</span>: the <span>reflected IDL attribute</span> is <span>limited to only
       known values</span>.</p></li>

       <li><p><span>Assert</span>: <var>contentAttributeValue</var> corresponds to a state of
       <var>attributeDefinition</var>.</p></li>

       <li><p>If <var>contentAttributeValue</var> corresponds to a state of
       <var>attributeDefinition</var> with no associated keyword value, then return null.</p></li>

       <li><p>Return the <span>canonical keyword</span> for the state of
       <var>attributeDefinition</var> that <var>contentAttributeValue</var> corresponds to.</p></li>
      </ol>
     </li>

     <li><p>Return <var>contentAttributeValue</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>The setter steps are:</p>

    <ol>
     <li><p>If the given value is null, then run <span>this</span>'s <span>delete the content
     attribute</span>.</p></li>

     <li><p>Otherwise, run <span>this</span>'s <span>set the content attribute</span> with the given
     value.</p></li>
    </ol>
   </li>
  </ul>

  <p>If a <span>reflected IDL attribute</span> has the type <code
  data-x="idl-USVString">USVString</code>:</p>

  <ul>
   <li>
    <p>The getter steps are:</p>

    <ol>
     <li><p>Let <var>element</var> be the result of running <span>this</span>'s <span>get the
     element</span>.</p></li>

     <li><p>Let <var>contentAttributeValue</var> be the result of running <span>this</span>'s
     <span>get the content attribute</span>.</p></li>

     <li><p>Let <var>attributeDefinition</var> be the attribute definition of <var>element</var>'s
     content attribute whose namespace is null and local name is the <span>reflected content
     attribute name</span>.

     <li>
      <p>If <var>attributeDefinition</var> indicates it contains a <span>URL</span>:</p>

      <ol>
       <li><p>If <var>contentAttributeValue</var> is null, then return the empty string.</p></li>

       <li><p>Let <var>urlString</var> be the result of <span>encoding-parsing-and-serializing a
       URL</span> given <var>contentAttributeValue</var>, relative to <var>element</var>'s
       <span>node document</span>.</p></li>

       <li><p>If <var>urlString</var> is not failure, then return <var>urlString</var>.</p></li>
      </ol>
     </li>

     <li><p>Return <var>contentAttributeValue</var>, <span
     data-x="convert">converted to a scalar value string</span>.</p></li>
    </ol>
   </li>

   <li><p>The setter steps are to run <span>this</span>'s <span>set the content attribute</span>
   with the given value.</p></li>
  </ul>

  <p>If a <span>reflected IDL attribute</span> has the type <code
  data-x="idl-boolean">boolean</code>:</p>

  <ul>
   <li>
    <p>The getter steps are:</p>

    <ol>
     <li><p>Let <var>contentAttributeValue</var> be the result of running <span>this</span>'s
     <span>get the content attribute</span>.</p></li>

     <li><p>If <var>contentAttributeValue</var> is null, then return false.</p></li>

     <li><p>Return true.</p></li>
    </ol>
   </li>

   <li>
    <p>The setter steps are:</p>

    <ol>
     <li><p>If the given value is false, then run <span>this</span>'s <span>delete the content
     attribute</span>.</p></li>

     <li><p>If the given value is true, then run <span>this</span>'s <span>set the content
     attribute</span> with the empty string.
    </ol>
   </li>
  </ul>

  <p class="note">This corresponds to the rules for <span data-x="boolean attribute">boolean content
  attributes</span>.

  <p>If a <span>reflected IDL attribute</span> has the type <code data-x="idl-long">long</code>,
  optionally <dfn>limited to only non-negative numbers</dfn> and optionally with a <dfn>default
  value</dfn> <var>defaultValue</var>:

  <ul>
   <li>
    <p>The getter steps are:</p>

    <ol>
     <li><p>Let <var>contentAttributeValue</var> be the result of running <span>this</span>'s
     <span>get the content attribute</span>.</p></li>

     <li>
      <p>If <var>contentAttributeValue</var> is not null:

      <ol>
       <li><p>Let <var>parsedValue</var> be the result of <span data-x="rules for parsing
       integers">integer parsing</span> <var>contentAttributeValue</var> if the <span>reflected IDL
       attribute</span> is not <span>limited to only non-negative numbers</span>; otherwise the
       result of <span data-x="rules for parsing non-negative integers">non-negative integer
       parsing</span> <var>contentAttributeValue</var>.</p></li>

       <li><p>If <var>parsedValue</var> is not an error and is within the <code
       data-x="idl-long">long</code> range, then return <var>parsedValue</var>.</p></li>
      </ol>
     </li>

     <li><p>If the <span>reflected IDL attribute</span> has a <span>default value</span>, then
     return <var>defaultValue</var>.</p></li>

     <li><p>If the <span>reflected IDL attribute</span> is <span>limited to only non-negative
     numbers</span>, then return &#x2212;1.</p></li>

     <li><p>Return 0.</p></li>
    </ol>
   </li>

   <li>
    <p>The setter steps are:</p>

    <ol>
     <li><p>If the <span>reflected IDL attribute</span> is <span>limited to only non-negative
     numbers</span> and the given value is negative, then throw an
     <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>Run <span>this</span>'s <span>set the content attribute</span> with the given value
     converted to the shortest possible string representing the number as a <span>valid
     integer</span>.</p></li>
    </ol>
   </li>
  </ul>

  <p>If a <span>reflected IDL attribute</span> has the type <code
  data-x="idl-unsigned-long">unsigned long</code>, optionally <dfn
  id="limited-to-only-non-negative-numbers-greater-than-zero">limited to only positive
  numbers</dfn>, <dfn
  id="limited-to-only-non-negative-numbers-greater-than-zero-with-fallback">limited to only positive
  numbers with fallback</dfn>, or <dfn>clamped to the range</dfn> [<var>clampedMin</var>,
  <var>clampedMax</var>], and optionally with a <span>default value</span> <var>defaultValue</var>:

  <ul>
   <li>
    <p>The getter steps are:</p>

    <ol>
     <li><p>Let <var>contentAttributeValue</var> be the result of running <span>this</span>'s
     <span>get the content attribute</span>.</p></li>

     <li><p>Let <var>minimum</var> be 0.</p></li>

     <li><p>If the <span>reflected IDL attribute</span> is <span>limited to only positive
     numbers</span> or <span>limited to only positive numbers with fallback</span>, then set
     <var>minimum</var> to 1.</p></li>

     <li><p>If the <span>reflected IDL attribute</span> is <span>clamped to the range</span>, then
     set <var>minimum</var> to <var>clampedMin</var>.</p></li>

     <li><p>Let <var>maximum</var> be 2147483647 if the <span>reflected IDL attribute</span> is not
     <span>clamped to the range</span>; otherwise <var>clampedMax</var>.</p></li>

     <li>
      <p>If <var>contentAttributeValue</var> is not null:

      <ol>
       <li><p>Let <var>parsedValue</var> be the result of <span data-x="rules for parsing
       non-negative integers">non-negative integer parsing</span>
       <var>contentAttributeValue</var>.</p></li>

       <li><p>If <var>parsedValue</var> is not an error and is in the range <var>minimum</var> to
       <var>maximum</var>, inclusive, then return <var>parsedValue</var>.</p></li>

       <li>
        <p>If <var>parsedValue</var> is not an error and the <span>reflected IDL attribute</span> is
        <span>clamped to the range</span>:

        <ol>
         <li><p>If <var>parsedValue</var> is less than <var>minimum</var>, then return
         <var>minimum</var>.</p></li>

         <li><p>Return <var>maximum</var>.</p></li>
        </ol>
       </li>
      </ol>
     </li>

     <li><p>If the <span>reflected IDL attribute</span> has a <span>default value</span>, then
     return <var>defaultValue</var>.</p></li>

     <li><p>Return <var>minimum</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>The setter steps are:</p>

    <ol>
     <li><p>If the <span>reflected IDL attribute</span> is <span>limited to only positive
     numbers</span> and the given value is 0, then throw an
     <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>Let <var>minimum</var> be 0.</p></li>

     <li><p>If the <span>reflected IDL attribute</span> is <span>limited to only positive
     numbers</span> or <span>limited to only positive numbers with fallback</span>, then set
     <var>minimum</var> to 1.</p></li>

     <li><p>Let <var>newValue</var> be <var>minimum</var>.</p></li>

     <li><p>If the <span>reflected IDL attribute</span> has a <span>default value</span>, then set
     <var>newValue</var> to <var>defaultValue</var>.</p></li>

     <li><p>If the given value is in the range <var>minimum</var> to 2147483647, inclusive,
     then set <var>newValue</var> to it.</p></li>

     <li><p>Run <span>this</span>'s <span>set the content attribute</span> with <var>newValue</var>
     converted to the shortest possible string representing the number as a <span>valid non-negative
     integer</span>.</p></li>
    </ol>

    <p class="note"><span>Clamped to the range</span> has no effect on the setter steps.</p>
   </li>
  </ul>
  <!-- see https://www.w3.org/Bugs/Public/show_bug.cgi?id=10352
           https://bugzilla.mozilla.org/show_bug.cgi?id=586118
           https://www.w3.org/Bugs/Public/show_bug.cgi?id=17705
       ...for why the range is clamped to 2147483647 rather the normal unsigned long maximum value
       of 4294967295 -->

  <p>If a <span>reflected IDL attribute</span> has the type <code data-x="idl-double">double</code>,
  optionally <span id="limited-to-numbers-greater-than-zero">limited to only positive numbers</span>
  and optionally with a <span>default value</span> <var>defaultValue</var>:

  <ul>
   <li>
    <p>The getter steps are:</p>

    <ol>
     <li><p>Let <var>contentAttributeValue</var> be the result of running <span>this</span>'s
     <span>get the content attribute</span>.</p></li>

     <li>
      <p>If <var>contentAttributeValue</var> is not null:

      <ol>
       <li><p>Let <var>parsedValue</var> be the result of <span data-x="rules for parsing
       floating-point number values">floating-point number parsing</span>
       <var>contentAttributeValue</var>.</p></li>

       <li><p>If <var>parsedValue</var> is not an error and is greater than 0, then return
       <var>parsedValue</var>.</p></li>

       <li><p>If <var>parsedValue</var> is not an error and the <span>reflected IDL attribute</span>
       is not <span>limited to only positive numbers</span>, then return
       <var>parsedValue</var>.</p></li>
      </ol>
     </li>

     <li><p>If the <span>reflected IDL attribute</span> has a <span>default value</span>, then
     return <var>defaultValue</var>.</p></li>

     <li><p>Return 0.</p></li>
    </ol>
   </li>


   <li>
    <p>The setter steps are:</p>

    <ol>
     <li><p>If the <span>reflected IDL attribute</span> is <span>limited to only positive
     numbers</span> and the given value is not greater than 0, then return.</p></li>

     <li><p>Run <span>this</span>'s <span>set the content attribute</span> with the given value,
     converted to the <span>best representation of the number as a floating-point
     number</span>.</p></li>
    </ol>
   </li>
  </ul>

  <p class="note">The values Infinity and Not-a-Number (NaN) values throw an exception on setting,
  as defined in <cite>Web IDL</cite>. <ref>WEBIDL</ref></p>

  <p>If a <span>reflected IDL attribute</span> has the type <code>DOMTokenList</code>, then its
  getter steps are to return a <code>DOMTokenList</code> object whose associated element is
  <span>this</span> and associated attribute's local name is the <span>reflected content
  attribute name</span>. Specification authors cannot use <span>support
  <code>ElementInternals</code></span> for IDL attributes of this type.</p>
  <!-- Supporting ElementInternals is doable in theory, but would require corresponding changes to
       DOMTokenList. -->

  <p>If a <span>reflected IDL attribute</span> has the type <code data-x=""><var>T</var>?</code>,
  where <var>T</var> is either <code>Element</code> or an interface that inherits from
  <code>Element</code>, then with <var>attr</var> being the <span>reflected content attribute
  name</span>:</p>

  <ul>
   <li><p>Its <span>reflected target</span> has an <dfn>explicitly set
   <var>attr</var>-element</dfn>, which is a weak reference to an element or null. It is initially
   null.</p></li>

   <li>
    <p>Its <span>reflected target</span> <var>reflectedTarget</var> has a <dfn
    for="Element,ElementInternals" export id="attr-associated-element">get the
    <var>attr</var>-associated element</dfn> algorithm, that runs these steps:</p>

    <ol>
     <li><p>Let <var>element</var> be the result of running <var>reflectedTarget</var>'s <span>get
     the element</span>.</p></li>

     <li><p>Let <var>contentAttributeValue</var> be the result of running
     <var>reflectedTarget</var>'s <span>get the content attribute</span>.</p></li>

     <li>
      <p>If <var>reflectedTarget</var>'s <span>explicitly set <var>attr</var>-element</span> is not
      null:</p>

      <ol>
       <li><p>If <var>reflectedTarget</var>'s <span>explicitly set <var>attr</var>-element</span> is
       a <span>descendant</span> of any of <var>element</var>'s <span
       data-x="concept-shadow-including-ancestor">shadow-including ancestors</span>, then return
       <var>reflectedTarget</var>'s <span>explicitly set <var>attr</var>-element</span>.</p></li>

       <li><p>Return null.</p></li>
      </ol>
     </li>

     <li>
      <p>Otherwise, if <var>contentAttributeValue</var> is not null, return the first element
      <var>candidate</var>, in <span>tree order</span>, that meets the following criteria:</p>

      <ul>
       <li><p><var>candidate</var>'s <span>root</span> is the same as <var>element</var>'s
       <span>root</span>;</p></li>

       <li><p><var>candidate</var>'s <span data-x="concept-ID">ID</span> is
       <var>contentAttributeValue</var>; and</p></li>

       <li><p><var>candidate</var> <span>implements</span> <var>T</var>.</p></li>
      </ul>

      <p>If no such element exists, then return null.</p>
     </li>

     <li><p>Return null.</p></li>
    </ol>
   </li>

   <li><p>The getter steps are to return the result of running <span>this</span>'s <span>get the
   <var>attr</var>-associated element</span>.</p></li>

   <li>
    <p>The setter steps are:</p>
    <ol>
     <li>
      <p>If the given value is null, then:</p>

      <ol>
       <li><p>Set <span>this</span>'s <span>explicitly set <var>attr</var>-element</span> to
       null.</p></li>

       <li><p>Run <span>this</span>'s <span>delete the content attribute</span>.</p></li>

       <li><p>Return.</p></li>
      </ol>
     </li>

     <li><p>Run <span>this</span>'s <span>set the content attribute</span> with the empty
     string.</p></li>

     <li><p>Set <span>this</span>'s <span>explicitly set <var>attr</var>-element</span> to a weak
     reference to the given value.</p></li>
    </ol>
   </li>

   <li>
    <p>For element <span data-x="reflected target">reflected targets</span> only: the following
    <span data-x="concept-element-attributes-change-ext">attribute change steps</span>, given
    <var>element</var>, <var>localName</var>, <var>oldValue</var>, <var>value</var>, and
    <var>namespace</var>, are used to synchronize between the content attribute and the IDL
    attribute:</p>

    <ol>
     <li><p>If <var>localName</var> is not <var>attr</var> or <var>namespace</var> is not null, then
     return.</p></li>

     <li><p>Set <var>element</var>'s <span>explicitly set <var>attr</var>-element</span> to
     null.</p></li>
    </ol>
   </li>
  </ul>

  <p class="note"><span data-x="reflected IDL attribute">Reflected IDL attributes</span> of this
  type are strongly encouraged to have their identifier end in "<code data-x="">Element</code>" for
  consistency.</p>

  <p>If a <span>reflected IDL attribute</span> has the type <code
  data-x="">FrozenArray&lt;<var>T</var>&gt;?</code>, where <var>T</var> is either
  <code>Element</code> or an interface that inherits from <code>Element</code>, then with
  <var>attr</var> being the <span>reflected content attribute name</span>:</p>

  <ul>
   <li><p>Its <span>reflected target</span> has an <dfn>explicitly set
   <var>attr</var>-elements</dfn>, which is either a <span>list</span> of weak references to
   elements or null. It is initially null.</p></li>

   <li><p>Its <span>reflected target</span> has a <dfn>cached <var>attr</var>-associated
   elements</dfn>, which is a <span>list</span> of elements. It is initially « ».</p></li>

   <li><p>Its <span>reflected target</span> has a <dfn>cached <var>attr</var>-associated
   elements object</dfn>, which is a <code data-x="">FrozenArray&lt;<var>T</var>&gt;?</code>. It is
   initially null.</p></li>

   <li>
    <p>Its <span>reflected target</span> <var>reflectedTarget</var> has a <dfn
    for="Element,ElementInternals" export id="attr-associated-elements">get the
    <var>attr</var>-associated elements</dfn> algorithm, which runs these steps:</p>

    <ol>
     <li><p>Let <var>elements</var> be an empty <span>list</span>.</p></li>

     <li><p>Let <var>element</var> be the result of running <var>reflectedTarget</var>'s <span>get
     the element</span>.</p></li>

     <li>
      <p>If <var>reflectedTarget</var>'s <span>explicitly set <var>attr</var>-elements</span> is not
      null:</p>

      <ol>
       <li>
        <p><span data-x="list iterate">For each</span> <var>attrElement</var> in
        <var>reflectedTarget</var>'s <span>explicitly set <var>attr</var>-elements</span>:</p>

        <ol>
         <li><p>If <var>attrElement</var> is not a <span>descendant</span> of any of
         <var>element</var>'s <span data-x="concept-shadow-including-ancestor">shadow-including
         ancestors</span>, then <span>continue</span>.</p></li>

         <li><p><span data-x="list append">Append</span> <var>attrElement</var> to
         <var>elements</var>.</p></li>
        </ol>
       </li>
      </ol>
     </li>

     <li>
      <p>Otherwise:</p>

      <ol>
       <li><p>Let <var>contentAttributeValue</var> be the result of running
       <var>reflectedTarget</var>'s <span>get the content attribute</span>.

       <li><p>If <var>contentAttributeValue</var> is null, then return null.</p></li>

       <li><p>Let <var>tokens</var> be <var>contentAttributeValue</var>, <span data-x="split a
       string on ASCII whitespace">split on ASCII whitespace</span>.

       <li>
        <p><span data-x="list iterate">For each</span> <var>id</var> of <var>tokens</var>:</p>

        <ol>
         <li>
          <p>Let <var>candidate</var> be the first element, in <span>tree order</span>, that meets
          the following criteria:</p>

          <ul>
           <li><p><var>candidate</var>'s <span>root</span> is the same as <var>element</var>'s
           <span>root</span>;</p></li>

           <li><p><var>candidate</var>'s <span data-x="concept-ID">ID</span> is <var>id</var>;
           and</p></li>

           <li><p><var>candidate</var> <span>implements</span> <var>T</var>.</p></li>
          </ul>

          <p>If no such element exists, then <span>continue</span>.</p>
         </li>

         <li><p><span data-x="list append">Append</span> <var>candidate</var> to
         <var>elements</var>.</p></li>
        </ol>
       </li>
      </ol>
     </li>

     <li><p>Return <var>elements</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>The getter steps are:</p>

    <ol>
     <li><p>Let <var>elements</var> be the result of running <span>this</span>'s <span>get the
     <var>attr</var>-associated elements</span>.</p></li>

     <li><p>If the contents of <var>elements</var> is equal to the contents of <span>this</span>'s
     <span>cached <var>attr</var>-associated elements</span>, then return <span>this</span>'s
     <span>cached <var>attr</var>-associated elements object</span>.</p></li>

     <li><p>Let <var>elementsAsFrozenArray</var> be <var>elements</var>, <span
     data-x="concept-idl-convert">converted</span> to a <code
     data-x="">FrozenArray&lt;<var>T</var>&gt;?</code>.</p></li>

     <li><p>Set <span>this</span>'s <span>cached <var>attr</var>-associated elements</span> to
     <var>elements</var>.</p></li>

     <li><p>Set <span>this</span>'s <span>cached <var>attr</var>-associated elements object</span>
     to <var>elementsAsFrozenArray</var>.</p></li>

     <li><p>Return <var>elementsAsFrozenArray</var>.</p></li>
    </ol>

    <p class="note">This extra caching layer is necessary to preserve the invariant that <code
    data-x="">element.reflectedElements === element.reflectedElements</code>.</p>
   </li>

   <li>
    <p>The setter steps are:</p>
    <ol>
     <li>
      <p>If the given value is null:</p>

      <ol>
       <li><p>Set <span>this</span>'s <span>explicitly set <var>attr</var>-elements</span> to
       null.</p></li>

       <li><p>Run <span>this</span>'s <span>delete the content attribute</span>.</p></li>

       <li><p>Return.</p></li>
      </ol>
     </li>

     <li><p>Run <span>this</span>'s <span>set the content attribute</span> with the empty
     string.</p></li>

     <li><p>Let <var>elements</var> be an empty <span>list</span>.</p></li>

     <li>
      <p><span data-x="list iterate">For each</span> <var>element</var> in the given value:</p>

      <ol>
       <li><p><span data-x="list append">Append</span> a weak reference to <var>element</var> to
       <var>elements</var>.</p></li>
      </ol>
     </li>

     <li><p>Set <span>this</span>'s <span>explicitly set <var>attr</var>-elements</span> to
     <var>elements</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>For element <span data-x="reflected target">reflected targets</span> only: the following
    <span data-x="concept-element-attributes-change-ext">attribute change steps</span>, given
    <var>element</var>, <var>localName</var>, <var>oldValue</var>, <var>value</var>, and
    <var>namespace</var>, are used to synchronize between the content attribute and the IDL
    attribute:</p>

    <ol>
     <li><p>If <var>localName</var> is not <var>attr</var> or <var>namespace</var> is not null, then
     return.</p></li>

     <li><p>Set <var>element</var>'s <span>explicitly set <var>attr</var>-elements</span> to
     null.</p></li>
    </ol>
   </li>
  </ul>

  <p class="note"><span data-x="reflected IDL attribute">Reflected IDL attributes</span> of this
  type are strongly encouraged to have their identifier end in "<code data-x="">Elements</code>" for
  consistency.</p>

  <h4>Using reflect in specifications</h4>

  <p><span data-x="reflect">Reflection</span> is primarily about improving web developer ergonomics
  by giving them typed access to content attributes through <span data-x="reflected IDL
  attribute">reflected IDL attributes</span>. The ultimate source of truth, which the web platform
  builds upon, is the content attributes themselves. That is, specification authors must not use the
  <span>reflected IDL attribute</span> getter or setter steps, but instead must use the content
  attribute presence and value. (Or an abstraction on top, such as the state of an <span>enumerated
  attribute</span>.)</p>

  <p>Two important exceptions to this are <span data-x="reflected IDL attribute">reflected IDL
  attributes</span> whose type is one of the following:</p>

  <ul>
   <li><p><code data-x=""><var>T</var>?</code>, where <var>T</var> is either <code>Element</code> or
   an interface that inherits from <code>Element</code></p></li>

   <li><p><code data-x="">FrozenArray&lt;<var>T</var>&gt;?</code>, where <var>T</var> is either
   <code>Element</code> or an interface that inherits from <code>Element</code></p></li>
  </ul>

  <p>For those, specification authors must use the <span>reflected target</span>'s
  <span>get the <var>attr</var>-associated element</span> and <span>get the
  <var>attr</var>-associated elements</span>, respectively. The content attribute presence and value
  must not be used as they cannot be fully synchronized with the <span>reflected IDL
  attribute</span>.</p>

  <p>A <span>reflected target</span>'s <span>explicitly set <var>attr</var>-element</span>,
  <span>explicitly set <var>attr</var>-elements</span>, <span>cached <var>attr</var>-associated
  elements</span>, and <span>cached <var>attr</var>-associated elements object</span> are to be
  treated as internal implementation details and not to be built upon.</p>

  </div>

  <h4>Collections</h4>

  <p>The <code>HTMLFormControlsCollection</code> and <code>HTMLOptionsCollection</code> interfaces
  are <span data-x="concept-collection">collections</span> derived from the
  <code>HTMLCollection</code> interface. The <code>HTMLAllCollection</code> interface is a <span
  data-x="concept-collection">collection</span>, but is not so derived.</p>


  <h5>The <code>HTMLAllCollection</code> interface</h5>

  <p>The <code>HTMLAllCollection</code> interface is used for the legacy <code
  data-x="dom-document-all">document.all</code> attribute. It operates similarly to
  <code>HTMLCollection</code>; the main differences are that it allows a staggering variety of
  different (ab)uses of its methods to all end up returning something, and that it can be called as
  a function as an alternative to property access.</p>

  <p class="note">All <code>HTMLAllCollection</code> objects are rooted at a <code>Document</code>
  and have a filter that matches all elements, so the elements <span>represented by the
  collection</span> of an <code>HTMLAllCollection</code> object consist of all the descendant
  elements of the root <code>Document</code>.</p>

  <p w-nodev>Objects that implement the <code>HTMLAllCollection</code> interface are <span
  data-x="legacy platform object">legacy platform objects</span> with an additional [[Call]] internal
  method described in the <a href="#HTMLAllCollection-call">section below</a>. They also have an
  <span>[[IsHTMLDDA]]</span> internal slot.</p>

  <div class="note">
   <p>Objects that implement the <code>HTMLAllCollection</code> interface have several unusual
   behaviors, due of the fact that they have an <span>[[IsHTMLDDA]]</span> internal slot:</p>

   <ul>
    <li><p>The <span data-x="js-ToBoolean">ToBoolean</span> abstract operation in JavaScript returns
    false when given objects implementing the <code>HTMLAllCollection</code> interface.</p></li>

    <li><p>The <span data-x="js-abstract-equality">IsLooselyEqual</span> abstract operation,
    when given objects implementing the <code>HTMLAllCollection</code> interface, returns true when
    compared to the <code data-x="">undefined</code> and <code data-x="">null</code> values.
    (Comparisons using the <span data-x="js-strict-equality">IsStrictlyEqual</span> abstract
    operation, and IsLooselyEqual comparisons to other values such as strings or objects, are
    unaffected.)</p></li>

    <li><p>The <code data-x="js-typeof">typeof</code> operator in JavaScript returns the string
    <code data-x="">"undefined"</code> when applied to objects implementing the
    <code>HTMLAllCollection</code> interface.</p></li>
   </ul>

   <p>These special behaviors are motivated by a desire for compatibility with two classes of legacy
   content: one that uses the presence of <code data-x="dom-document-all">document.all</code> as a
   way to detect legacy user agents, and one that only supports those legacy user agents and uses
   the <code data-x="dom-document-all">document.all</code> object without testing for its presence
   first. <ref>JAVASCRIPT</ref></p>
  </div>

  <pre><code class="idl">[Exposed=Window,
 <span>LegacyUnenumerableNamedProperties</span>]
interface <dfn interface>HTMLAllCollection</dfn> {
  readonly attribute unsigned long <span data-x="dom-HTMLAllCollection-length">length</span>;
  getter <span>Element</span> (unsigned long index);
  getter (<span>HTMLCollection</span> or <span>Element</span>)? <span data-x="dom-HTMLAllCollection-namedItem">namedItem</span>(DOMString name);
  (<span>HTMLCollection</span> or <span>Element</span>)? <span data-x="dom-HTMLAllCollection-item">item</span>(optional DOMString nameOrIndex);

  // Note: HTMLAllCollection objects have a <a href="#HTMLAllCollection-call">custom [[Call]] internal method</a> and an <span>[[IsHTMLDDA]]</span> internal slot.
};</code></pre>

  <div w-nodev>

  <p>The object's <span>supported property indices</span> are as defined for
  <code>HTMLCollection</code> objects.</p>

  <p>The <span>supported property names</span> consist of the non-empty values of all the <code
  data-x="attr-id">id</code> attributes of all the elements <span>represented by the
  collection</span>, and the non-empty values of all the <code data-x="">name</code> attributes of
  all the <span>"all"-named elements</span> <span>represented by the collection</span>, in
  <span>tree order</span>, ignoring later duplicates, with the <code data-x="attr-id">id</code> of
  an element preceding its <code data-x="">name</code> if it contributes both, they differ from
  each other, and neither is the duplicate of an earlier entry.</p>

  <p>The <dfn attribute for="HTMLAllCollection"><code
  data-x="dom-HTMLAllCollection-length">length</code></dfn> getter steps are to return the number
  of nodes <span>represented by the collection</span>.</p>

  <p>The indexed property getter must return the result of <span
  data-x="concept-get-all-indexed">getting the "all"-indexed element</span> from <span>this</span>
  given the passed index.</p>

  <p>The <dfn method for="HTMLAllCollection"><code
  data-x="dom-HTMLAllCollection-namedItem">namedItem(<var>name</var>)</code></dfn> method steps are
  to return the result of <span data-x="concept-get-all-named">getting the "all"-named
  element(s)</span> from <span>this</span> given <var>name</var>.</p>

  <p>The <dfn method for="HTMLAllCollection"><code
  data-x="dom-HTMLAllCollection-item">item(<var>nameOrIndex</var>)</code></dfn> method  steps
  are:</p>

  <ol>
   <li><p>If <var>nameOrIndex</var> was not provided, return null.</p></li>

   <li><p>Return the result of <span data-x="concept-get-all-indexed-or-named">getting the
   "all"-indexed or named element(s)</span> from <span>this</span>, given
   <var>nameOrIndex</var>.</p></li>
  </ol>

  <hr>

  <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2837 -->
  <p>The following elements are <dfn>"all"-named elements</dfn>:
   <code>a</code>,
   <code>button</code>,
   <code>embed</code>,
   <code>form</code>,
   <code>frame</code>,
   <code>frameset</code>,
   <code>iframe</code>,
   <code>img</code>,
   <code>input</code>,
   <code>map</code>,
   <code>meta</code>,
   <code>object</code>,
   <code>select</code>, and
   <code>textarea</code>
  </p>

  <p>To <dfn data-x="concept-get-all-indexed">get the "all"-indexed element</dfn> from an
  <code>HTMLAllCollection</code> <var>collection</var> given an index <var>index</var>, return the
  <var>index</var><sup>th</sup> element in <var>collection</var>, or null if there is no such
  <var>index</var><sup>th</sup> element.</p>

  <p>To <dfn data-x="concept-get-all-named">get the "all"-named element(s)</dfn> from an
  <code>HTMLAllCollection</code> <var>collection</var> given a name <var>name</var>, perform the
  following steps:</p>

  <ol>
   <li><p>If <var>name</var> is the empty string, return null.</p></li>

   <li>
    <p>Let <var>subCollection</var> be an <code>HTMLCollection</code> object rooted at the same
    <code>Document</code> as <var>collection</var>, whose filter matches only elements that are
    either:</p>

    <ul>
     <li><p><span>"all"-named elements</span> with a <code data-x="">name</code> attribute equal to
     <var>name</var>, or,</p></li>

     <li><p>elements with an <span data-x="concept-id">ID</span> equal to <var>name</var>.</p></li>
    </ul>
   </li>

   <li><p>If there is exactly one element in <var>subCollection</var>, then return that
   element.</p></li>

   <li><p>Otherwise, if <var>subCollection</var> is empty, return null.</p></li>

   <li><p>Otherwise, return <var>subCollection</var>.</p></li>
  </ol>

  <p>To <dfn data-x="concept-get-all-indexed-or-named">get the "all"-indexed or named
  element(s)</dfn> from an <code>HTMLAllCollection</code> <var>collection</var> given
  <var>nameOrIndex</var>:

  <ol>
   <li><p>If <var>nameOrIndex</var>, <span data-x="concept-idl-convert">converted</span> to a
   JavaScript String value, is an <span>array index property name</span>, return the result of <span
   data-x="concept-get-all-indexed">getting the "all"-indexed element</span> from
   <var>collection</var> given the number represented by <var>nameOrIndex</var>.</p></li>

   <li><p>Return the result of <span data-x="concept-get-all-named">getting the "all"-named
   element(s)</span> from <var>collection</var> given <var>nameOrIndex</var>.</p></li>
  </ol>

  <h6 id="HTMLAllCollection-call">[[Call]] ( <var>thisArgument</var>,  <var>argumentsList</var> )</h6>

  <ol>
   <li><p>If <var>argumentsList</var>'s <span data-x="list size">size</span> is zero, or if
   <var>argumentsList</var>[0] is undefined, return null.</p></li>

   <li><p>Let <var>nameOrIndex</var> be the result of <span
   data-x="concept-idl-convert">converting</span> <var>argumentsList</var>[0] to a <code
   data-x="idl-DOMString">DOMString</code>.</p></li>

   <li><p>Let <var>result</var> be the result of <span
   data-x="concept-get-all-indexed-or-named">getting the "all"-indexed or named element(s)</span>
   from this <code>HTMLAllCollection</code> given <var>nameOrIndex</var>.</p></li>

   <li><p>Return the result of <span data-x="concept-idl-convert">converting</span>
   <var>result</var> to an ECMAScript value.</p></li>
  </ol>

  <p class="note">The <var>thisArgument</var> is ignored, and thus code such as <code
  data-x="">Function.prototype.call.call(document.all, null, "x")</code> will still search for
  elements. (<code data-x="">document.all.call</code> does not exist, since <code
  data-x="">document.all</code> does not inherit from <code
  data-x="">Function.prototype</code>.)</p>

  </div>


  <h5>The <code>HTMLFormControlsCollection</code> interface</h5>

  <p>The <code>HTMLFormControlsCollection</code> interface is used for
  <span data-x="concept-collection">collections</span> of <span data-x="category-listed">listed
  elements</span> in <code>form</code> elements.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLFormControlsCollection</dfn> : <span>HTMLCollection</span> {
  // inherits <span data-x="dom-HTMLCollection-length">length</span> and <span data-x="dom-HTMLCollection-item">item</span>()
  getter (<span>RadioNodeList</span> or <span>Element</span>)? <span data-x="dom-HTMLFormControlsCollection-namedItem">namedItem</span>(DOMString name); // shadows inherited <span data-x="dom-HTMLCollection-namedItem">namedItem()</span>
};

[Exposed=Window]
interface <dfn interface>RadioNodeList</dfn> : <span>NodeList</span> {
  attribute DOMString <span data-x="dom-RadioNodeList-value">value</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>collection</var>.<span subdfn data-x="dom-HTMLCollection-length">length</span></code></dt>
   <dd><p>Returns the number of elements in <var>collection</var>.</p></dd>

   <dt><code data-x=""><var>element</var> = <var>collection</var>.<span subdfn data-x="dom-HTMLCollection-item">item</span>(<var>index</var>)</code></dt>
   <dt><code data-x=""><var>element</var> = <var>collection</var>[<var>index</var>]</code></dt>
   <dd><p>Returns the item at index <var>index</var> in <var>collection</var>. The items are sorted
   in <span>tree order</span>.</p></dd>

   <dt><code data-x=""><var>element</var> = <var>collection</var>.<span subdfn data-x="dom-HTMLFormControlsCollection-namedItem">namedItem</span>(<var>name</var>)</code></dt>
   <dt><code data-x=""><var>radioNodeList</var> = <var>collection</var>.<span data-x="dom-HTMLFormControlsCollection-namedItem">namedItem</span>(<var>name</var>)</code></dt>
   <dt><code data-x=""><var>element</var> = <var>collection</var>[<var>name</var>]</code></dt>
   <dt><code data-x=""><var>radioNodeList</var> = <var>collection</var>[<var>name</var>]</code></dt>
   <dd>
    <p>Returns the item with <span data-x="concept-id">ID</span> or <code
    data-x="attr-fe-name">name</code> <var>name</var> from <var>collection</var>.</p>

    <p>If there are multiple matching items, then a <code>RadioNodeList</code> object containing all
    those elements is returned.</p>
   </dd>

   <dt><code data-x=""><var>radioNodeList</var>.<span subdfn data-x="dom-radionodelist-value">value</span></code></dt>
   <dd><p>Returns the value of the first checked radio button represented by
   <var>radioNodeList</var>.</p></dd>

   <dt><code data-x=""><var>radioNodeList</var>.<span data-x="dom-radionodelist-value">value</span> = <var>value</var></code></dt>
   <dd><p>Checks the first radio button represented by <var>radioNodeList</var> that has value
   <var>value</var>.</p></dd>
  </dl>

  <div w-nodev>

  <p>The object's <span>supported property indices</span> are as defined for
  <code>HTMLCollection</code> objects.</p>

  <p>The <span>supported property names</span> consist of the non-empty values of all the <code
  data-x="attr-id">id</code> and <code data-x="attr-fe-name">name</code> attributes of all the
  elements <span>represented by the collection</span>, in <span>tree order</span>, ignoring later
  duplicates, with the <code data-x="attr-id">id</code> of an element preceding its <code
  data-x="attr-fe-name">name</code> if it contributes both, they differ from each other, and neither is the
  duplicate of an earlier entry.</p>

  <p>The <dfn method for="HTMLFormControlsCollection"><code
  data-x="dom-HTMLFormControlsCollection-namedItem">namedItem(<var>name</var>)</code></dfn> method
  must act according to the following algorithm:</p>

  <ol>
   <li>If <var>name</var> is the empty string, return null and stop the algorithm.</li>

   <li>If, at the time the method is called, there is exactly one node in the collection that has
   either an <code data-x="attr-id">id</code> attribute or a <code data-x="attr-fe-name">name</code>
   attribute equal to <var>name</var>, then return that node and stop the algorithm.</li>

   <li>Otherwise, if there are no nodes in the collection that have either an <code
   data-x="attr-id">id</code> attribute or a <code data-x="attr-fe-name">name</code> attribute equal
   to <var>name</var>, then return null and stop the algorithm.</li>

   <li>Otherwise, create a new <code>RadioNodeList</code> object representing a <span>live</span>
   view of the <code>HTMLFormControlsCollection</code> object, further filtered so that the only
   nodes in the <code>RadioNodeList</code> object are those that have either an <code
   data-x="attr-id">id</code> attribute or a <code data-x="attr-fe-name">name</code> attribute equal
   to <var>name</var>. The nodes in the <code>RadioNodeList</code> object must be sorted in
   <span>tree order</span>.</li>

   <li>Return that <code>RadioNodeList</code> object.</li>
  </ol>

  <hr>

  <p>Members of the <code>RadioNodeList</code> interface inherited from the <code>NodeList</code>
  interface must behave as they would on a <code>NodeList</code> object.</p>

  <p>The <dfn attribute for="RadioNodeList"><code
  data-x="dom-RadioNodeList-value">value</code></dfn> IDL attribute on the
  <code>RadioNodeList</code> object, on getting, must return the value returned by running the
  following steps:</p>

  <ol>
   <li><p>Let <var>element</var> be the first element in <span>tree order</span>
   represented by the <code>RadioNodeList</code> object that is an <code>input</code> element whose
   <code data-x="attr-input-type">type</code> attribute is in the <span
   data-x="attr-input-type-radio">Radio Button</span> state and whose <span
   data-x="concept-fe-checked">checkedness</span> is true. Otherwise, let it be null.</p></li>

   <li><p>If <var>element</var> is null, return the empty string.</p></li>

   <li><p>If <var>element</var> is an element with no <code
   data-x="attr-input-value">value</code> attribute, return the string "<code data-x="">on</code>".</p></li>

   <li><p>Otherwise, return the value of <var>element</var>'s <code
   data-x="attr-input-value">value</code> attribute.</p></li>
  </ol>

  <p>On setting, the <code data-x="dom-RadioNodeList-value">value</code> IDL attribute must run the
  following steps:</p>

  <ol>
   <li><p>If the new value is the string "<code data-x="">on</code>": let <var>element</var> be the first element in <span>tree order</span>
   represented by the <code>RadioNodeList</code> object that is an <code>input</code> element whose
   <code data-x="attr-input-type">type</code> attribute is in the <span
   data-x="attr-input-type-radio">Radio Button</span> state and whose <code
   data-x="attr-input-value">value</code> content attribute is either absent, or present and equal to the new value, if any. If no such element exists, then instead let <var>element</var> be null.</p>

   <p>Otherwise: let <var>element</var> be the first element in <span>tree order</span>
   represented by the <code>RadioNodeList</code> object that is an <code>input</code> element whose
   <code data-x="attr-input-type">type</code> attribute is in the <span
   data-x="attr-input-type-radio">Radio Button</span> state and whose <code
   data-x="attr-input-value">value</code> content attribute is present and equal to the new value, if
   any. If no such element exists, then instead let <var>element</var> be null.</p></li>

   <li><p>If <var>element</var> is not null, then set its <span
   data-x="concept-fe-checked">checkedness</span> to true.</p>
  </ol>

<!--
https://software.hixie.ch/utilities/js/live-dom-viewer/?%3C%21DOCTYPE%20html%3E...%0A%3Cform%20name%3D%22a%22%3E%3Cinput%20id%3D%22x%22%20name%3D%22y%22%3E%3Cinput%20name%3D%22x%22%20id%3D%22y%22%3E%3C/form%3E%0A%3Cscript%3E%0A%20%20var%20x%3B%0A%20%20w%28x%20%3D%20document.forms%5B%27a%27%5D%5B%27x%27%5D%29%3B%0A%20%20w%28x.length%29%3B%0A%20%20x%5B0%5D.parentNode.removeChild%28x%5B0%5D%29%3B%0A%20%20w%28x.length%29%3B%0A%20%20w%28x%20%3D%3D%20document.forms%5B%27a%27%5D%5B%27x%27%5D%29%3B%0A%3C/script%3E%0A
-->

  </div>


  <h5>The <code>HTMLOptionsCollection</code> interface</h5>

  <p>The <code>HTMLOptionsCollection</code> interface is used for <span
  data-x="concept-collection">collections</span> of <code>option</code> elements. It is always
  rooted on a <code>select</code> element and has attributes and methods that manipulate that
  element's descendants.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLOptionsCollection</dfn> : <span>HTMLCollection</span> {
  // inherits <span data-x="dom-HTMLCollection-item">item</span>(), <span data-x="dom-HTMLCollection-namedItem">namedItem</span>()
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-HTMLOptionsCollection-length">length</span>; // shadows inherited <span data-x="dom-HTMLCollection-length">length</span>
  [<span>CEReactions</span>] <a href="#dom-htmloptionscollection-setter">setter</a> undefined (unsigned long index, <span>HTMLOptionElement</span>? option);
  [<span>CEReactions</span>] undefined <span data-x="dom-HTMLOptionsCollection-add">add</span>((<span>HTMLOptionElement</span> or <span>HTMLOptGroupElement</span>) element, optional (<span>HTMLElement</span> or long)? before = null);
  [<span>CEReactions</span>] undefined <span data-x="dom-HTMLOptionsCollection-remove">remove</span>(long index);
  attribute long <span data-x="dom-HTMLOptionsCollection-selectedIndex">selectedIndex</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>collection</var>.<span subdfn data-x="dom-HTMLOptionsCollection-length">length</span></code></dt>
   <dd><p>Returns the number of elements in <var>collection</var>.</p></dd>

   <dt><code data-x=""><var>collection</var>.<span data-x="dom-HTMLOptionsCollection-length">length</span> = <var>value</var></code></dt>
   <dd>
    <p>When set to a smaller number than the existing length, truncates the number of
    <code>option</code> elements in the container corresponding to <var>collection</var>.</p>

    <p>When set to a greater number than the existing length, if that number is less than or equal
    to 100000, adds new blank <code>option</code> elements to the container corresponding to
    <var>collection</var>.</p>
   </dd>

   <dt><code data-x=""><var>element</var> = <var>collection</var>.<span data-x="dom-HTMLCollection-item">item</span>(<var>index</var>)</code></dt>
   <dt><code data-x=""><var>element</var> = <var>collection</var>[<var>index</var>]</code></dt>
   <dd><p>Returns the item at index <var>index</var> in <var>collection</var>. The items are sorted
   in <span>tree order</span>.</p></dd>

   <dt><code data-x=""><var>collection</var>[<var>index</var>] = <var>element</var></code></dt>
   <dd>
    <p>When <var>index</var> is a greater number than the number of items in <var>collection</var>,
    adds new blank <code>option</code> elements in the corresponding container.</p>

    <p>When set to null, removes the item at index <var>index</var> from <var>collection</var>.</p>

    <p>When set to an <code>option</code> element, adds or replaces it at index <var>index</var> in
    <var>collection</var>.</p>
   </dd>

   <dt><code data-x=""><var>element</var> = <var>collection</var>.<span subdfn data-x="dom-HTMLCollection-namedItem">namedItem</span>(<var>name</var>)</code></dt>
   <dt><code data-x=""><var>element</var> = <var>collection</var>[<var>name</var>]</code></dt>
   <dd>
    <p>Returns the item with <span data-x="concept-id">ID</span> or <code
    data-x="attr-option-name">name</code> <var>name</var> from <var>collection</var>.</p>

    <p>If there are multiple matching items, then the first is returned.</p>
   </dd>

   <dt><code data-x=""><var>collection</var>.<span subdfn data-x="dom-HTMLOptionsCollection-add">add</span>(<var>element</var>[, <var>before</var>])</code></dt>
   <dd>
    <p>Inserts <var>element</var> before the node given by <var>before</var>.</p>

    <p>The <var>before</var> argument can be a number, in which case <var>element</var> is inserted
    before the item with that number, or an element from <var>collection</var>, in which case
    <var>element</var> is inserted before that element.</p>

    <p>If <var>before</var> is omitted, null, or a number out of range, then <var>element</var> will
    be added at the end of the list.</p>

    <p>Throws a <span>"<code>HierarchyRequestError</code>"</span> <code>DOMException</code> if
    <var>element</var> is an ancestor of the element into which it is to be inserted.</p>
   </dd>

   <dt><code data-x=""><var>collection</var>.<span subdfn data-x="dom-HTMLOptionsCollection-remove">remove</span>(<var>index</var>)</code></dt>
   <dd><p>Removes the item with index <var>index</var> from <var>collection</var>.</p></dd>

   <dt><code data-x=""><var>collection</var>.<span data-x="dom-HTMLOptionsCollection-selectedIndex">selectedIndex</span></code></dt>
   <dd><p>Returns the index of the first selected item, if any, or &#x2212;1 if there is no selected
   item.</p></dd>

   <dt><code data-x=""><var>collection</var>.<span subdfn data-x="dom-HTMLOptionsCollection-selectedIndex">selectedIndex</span> = <var>index</var></code></dt>
   <dd><p>Changes the selection to the <code>option</code> element at index <var>index</var> in
   <var>collection</var>.</p></dd>
  </dl>

  <div w-nodev>

  <p>The object's <span>supported property indices</span> are as defined for
  <code>HTMLCollection</code> objects.</p>

  <p>The <dfn attribute for="HTMLOptionsCollection"><code
  data-x="dom-HTMLOptionsCollection-length">length</code></dfn> getter steps are to return the
  number of nodes <span>represented by the collection</span>.</p>

  <p>The <code data-x="dom-HTMLOptionsCollection-length">length</code> setter steps are:</p>

  <ol>
   <li><p>Let <var>current</var> be the number of nodes <span>represented by the
   collection</span>.</p></li>

   <li>
    <p>If the given value is greater than <var>current</var>, then:</p>

    <ol>
     <li><p>If the given value is greater than 100,000, then return.</p></li>

     <li><p>Let <var>n</var> be <var>value</var> &minus; <var>current</var>.</p></li>

     <li><p>Append <var>n</var> new <code>option</code> elements with no attributes and no child
     nodes to the <code>select</code> element on which <span>this</span> is rooted.</p></li>
    </ol>
   </li>

   <li>
    <p>If the given value is less than <var>current</var>, then:</p>

    <ol>
     <li><p>Let <var>n</var> be <var>current</var> &minus; <var>value</var>.</p></li>

     <li><p>Remove the last <var>n</var> nodes in the collection from their parent nodes.</p></li>
    </ol>
   </li>
  </ol>

  <p class="note">Setting <code data-x="dom-HTMLOptionsCollection-length">length</code> never removes
  or adds any <code>optgroup</code> elements, and never adds new children to existing
  <code>optgroup</code> elements (though it can remove children from them).</p>

  <p>The <span>supported property names</span> consist of the non-empty values of all the <code
  data-x="attr-id">id</code> and <code data-x="attr-option-name">name</code> attributes of all the
  elements <span>represented by the collection</span>, in <span>tree order</span>, ignoring later
  duplicates, with the <code data-x="attr-id">id</code> of an element preceding its <code
  data-x="attr-option-name">name</code> if it contributes both, they differ from each other, and neither is
  the duplicate of an earlier entry.</p>

  <p id="dom-htmloptionscollection-setter">When the user agent is to <span>set the value of a new
  indexed property</span> or <span>set the value of an existing indexed property</span> for a given
  property index <var>index</var> to a new value <var>value</var>, it must run the following
  algorithm:</p>

  <ol>
   <li><p>If <var>value</var> is null, invoke the steps for the <code
   data-x="dom-HTMLOptionsCollection-remove">remove</code> method with <var>index</var> as
   the argument, and return.</p></li>

   <li><p>Let <var>length</var> be the number of nodes <span>represented by the
   collection</span>.</p></li>

   <li><p>Let <var>n</var> be <var>index</var> minus <var>length</var>.</p></li>

   <li><p>If <var>n</var> is greater than zero, then <span
   data-x="concept-node-append">append</span> a <code>DocumentFragment</code> consisting of <span
   data-x=""><var>n</var>-1</span> new <code>option</code> elements with no attributes and
   no child nodes to the <code>select</code> element on which the <code>HTMLOptionsCollection</code>
   is rooted.</p></li>

   <li><p>If <var>n</var> is greater than or equal to zero, <span
   data-x="concept-node-append">append</span> <var>value</var> to the <code>select</code>
   element. Otherwise, <span data-x="concept-node-replace">replace</span> the <var>index</var>th element in the collection by <var>value</var>.</p></li>
  </ol>

  <p>The <dfn method for="HTMLOptionsCollection"><code
  data-x="dom-HTMLOptionsCollection-add">add(<var>element</var>, <var>before</var>)</code></dfn>
  method must act according to the following algorithm:</p>

  <ol>
   <li><p>If <var>element</var> is an ancestor of the <code>select</code> element on which
   the <code>HTMLOptionsCollection</code> is rooted, then throw a
   <span>"<code>HierarchyRequestError</code>"</span> <code>DOMException</code>.</li>

   <li><p>If <var>before</var> is an element, but that element isn't a descendant of the
   <code>select</code> element on which the <code>HTMLOptionsCollection</code> is rooted, then throw
   a <span>"<code>NotFoundError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <var>element</var> and <var>before</var> are the same element, then
   return.</li>

   <li><p>If <var>before</var> is a node, then let <var>reference</var> be that
   node. Otherwise, if <var>before</var> is an integer, and there is a <var>before</var>th node in the collection, let <var>reference</var> be that node.
   Otherwise, let <var>reference</var> be null.</p></li>

   <li><p>If <var>reference</var> is not null, let <var>parent</var> be the parent
   node of <var>reference</var>. Otherwise, let <var>parent</var> be the
   <code>select</code> element on which the <code>HTMLOptionsCollection</code> is rooted.</p></li>

   <li><p><span>Pre-insert</span> <var>element</var> into <var>parent</var> node before
   <var>reference</var>.</p>
  </ol>

  <p>The <dfn method for="HTMLOptionsCollection"><code
  data-x="dom-HTMLOptionsCollection-remove">remove(<var>index</var>)</code></dfn> method must act
  according to the following algorithm:</p>

  <ol>
   <li><p>If the number of nodes <span>represented by the collection</span> is zero, return.</li>

   <li><p>If <var>index</var> is not a number greater than or equal to 0 and less than the
   number of nodes <span>represented by the collection</span>, return.</p></li>
   <!-- note that Web IDL converts 'a', NaN, -0.5, 0.5, most objects, and null to 0, and numbers in
   general truncate. so what we're checking against here in a way has very little to do with what's
   going on in the JS -->
   <!-- objects get converted by calling ToNumber which calls ToPrimitive which eventually calls valueOf:
         https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2828
         https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2833
    -->
   <!-- same for select.remove(), which calls this algorithm also:
         https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2826
         https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2825
         https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2827
     -->

   <li><p>Let <var>element</var> be the <var>index</var>th element in the
   collection.</p></li>

   <li><p>Remove <var>element</var> from its parent node.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLOptionsCollection"><code
  data-x="dom-HTMLOptionsCollection-selectedIndex">selectedIndex</code></dfn> IDL attribute must act
  like the identically named attribute on the <code>select</code> element on which the
  <code>HTMLOptionsCollection</code> is rooted</p>

  <!-- see also https://ln.hixie.ch/?start=1161042744&count=1 -->

  </div>


  <h4>The <code>DOMStringList</code> interface</h4>

  <p>The <code>DOMStringList</code> interface is a non-fashionable retro way of representing a list
  of strings.</p>

  <pre><code class="idl">[Exposed=(Window,Worker)]
interface <dfn interface>DOMStringList</dfn> {
  readonly attribute unsigned long <span data-x="dom-DOMStringList-length">length</span>;
  getter DOMString? <span data-x="dom-DOMStringList-item">item</span>(unsigned long index);
  boolean <span data-x="dom-DOMStringList-contains">contains</span>(DOMString string);
};</code></pre>

  <p w-nodev class="warning">New APIs must use <code data-x="">sequence&lt;DOMString></code> or
  equivalent rather than <code>DOMStringList</code>.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>strings</var>.<span subdfn data-x="dom-DOMStringList-length">length</span></code></dt>
   <dd><p>Returns the number of strings in <var>strings</var>.</p></dd>

   <dt><code data-x=""><var>strings</var>[<var>index</var>]</code></dt>
   <dt><code data-x=""><var>strings</var>.<span subdfn data-x="dom-DOMStringList-item">item</span>(<var>index</var>)</code></dt>
   <dd><p>Returns the string with index <var>index</var> from <var>strings</var>.</p></dd>

   <dt><code data-x=""><var>strings</var>.<span subdfn data-x="dom-DOMStringList-contains">contains</span>(<var>string</var>)</code></dt>
   <dd><p>Returns true if <var>strings</var> contains <var>string</var>, and false
   otherwise.</p></dd>
  </dl>

  <div w-nodev>

  <p>Each <code>DOMStringList</code> object has an associated <span>list</span>.</p>

  <p>The <code>DOMStringList</code> interface <span>supports indexed properties</span>. The
  <span>supported property indices</span> are the <span>indices</span> of <span>this</span>'s
  associated list.</p>

  <p>The <dfn attribute for="DOMStringList"
  data-x="dom-DOMStringList-length"><code>length</code></dfn> getter steps are to return
  <span>this</span>'s associated list's <span data-x="list size">size</span>.</p>

  <p>The <dfn method for="DOMStringList"
  data-x="dom-DOMStringList-item"><code>item(<var>index</var>)</code></dfn> method steps are to
  return the <var>index</var>th item in <span>this</span>'s associated list, or null if
  <var>index</var> plus one is greater than <span>this</span>'s associated list's <span data-x="list
  size">size</span>.</p>

  <p>The <dfn method for="DOMStringList"
  data-x="dom-DOMStringList-contains"><code>contains(<var>string</var>)</code></dfn> method steps
  are to return true if <span>this</span>'s associated list <span data-x="list
  contains">contains</span> <var>string</var>, and false otherwise.</p>

  </div>


  <h3 split-filename="structured-data">Safe passing of structured data</h3>

  <p id="structuredclone"><span id="structured-clone"></span>To support passing JavaScript objects,
  including <span data-x="platform object">platform objects</span>, across <span>realm</span>
  boundaries, this specification defines <span w-nodev>the following </span>infrastructure for
  serializing and deserializing objects, including in some cases transferring the underlying data
  instead of copying it. Collectively this serialization/deserialization process is known as
  "structured cloning", although most APIs perform separate serialization and deserialization steps.
  (With the notable exception being the <code data-x="dom-structuredClone">structuredClone()</code>
  method.)</p>

  <div w-nodev>

  <p>This section uses the terminology and typographic conventions from the JavaScript
  specification. <ref>JAVASCRIPT</ref></p>

  </div>

  <h4 export data-lt="serializable object"><dfn>Serializable objects</dfn></h4>

  <p><span>Serializable objects</span> support being serialized, and later deserialized, in a way
  that is independent of any given <span>realm</span>. This allows them to be stored on disk and
  later restored, or cloned across <span>agent</span> and even <span>agent cluster</span>
  boundaries.</p>

  <p>Not all objects are <span>serializable objects</span>, and not all aspects of objects that are
  <span>serializable objects</span> are necessarily preserved when they are serialized.</p>

  <div w-nodev>

  <p><span data-x="platform object">Platform objects</span> can be <span>serializable objects</span>
  if their <span>primary interface</span> is decorated with the <dfn extended-attribute
  data-lt="Serializable" data-x="Serializable"><code>[Serializable]</code></dfn> IDL <span>extended
  attribute</span>. Such interfaces must also define the following algorithms:</p>

  <dl>
   <dt><dfn export>serialization steps</dfn>, taking a <span>platform object</span>
   <var>value</var>, a <span>Record</span> <var>serialized</var>, and a boolean
   <var>forStorage</var></dt>
   <dd>
    <p>A set of steps that serializes the data in <var>value</var> into fields of
    <var>serialized</var>. The resulting data serialized into <var>serialized</var> must be
    independent of any <span>realm</span>.</p>

    <p>These steps may throw an exception if serialization is not possible.</p>

    <p>These steps may perform a <span>sub-serialization</span> to serialize nested data
    structures. They should not call <span>StructuredSerialize</span> directly, as doing so will
    omit the important <var>memory</var> argument.</p>

    <p>The introduction of these steps should omit mention of the <var>forStorage</var> argument if
    it is not relevant to the algorithm.</p>
   </dd>

   <dt><dfn export>deserialization steps</dfn>, taking a <span>Record</span> <var>serialized</var>,
   a <span>platform object</span> <var>value</var>, and a <span>realm</span>
   <var>targetRealm</var></dt>
   <dd>
    <p>A set of steps that deserializes the data in <var>serialized</var>, using it to set up
    <var>value</var> as appropriate. <var>value</var> will be a newly-created instance of the
    <span>platform object</span> type in question, with none of its internal data set up; setting
    that up is the job of these steps.</p>

    <p>These steps may throw an exception if deserialization is not possible.</p>

    <p>These steps may perform a <span>sub-deserialization</span> to deserialize nested data
    structures. They should not call <span>StructuredDeserialize</span> directly, as doing so will
    omit the important <var>targetRealm</var> and <var>memory</var> arguments.</p>
   </dd>
  </dl>

  <p>It is up to the definition of individual platform objects to determine what data is serialized
  and deserialized by these steps. Typically the steps are very symmetric.</p>

  <p>The <code data-x="Serializable">[Serializable]</code> extended attribute must take no
  arguments, and must only appear on an interface. It must not appear more than once on an
  interface.</p>

  <p>For a given <span>platform object</span>, only the object's <span>primary interface</span> is
  considered during the (de)serialization process. Thus, if inheritance is involved in defining the
  interface, each <code data-x="Serializable">[Serializable]</code>-annotated interface in the
  inheritance chain needs to define standalone <span>serialization steps</span> and
  <span>deserialization steps</span>, including taking into account any important data that might
  come from inherited interfaces.</p>

  <div class="example">
   <p>Let's say we were defining a platform object <code data-x="">Person</code>, which had
   associated with it two pieces of associated data:</p>

   <ul>
    <li><p>a name value, which is a string; and</p></li>

    <li><p>a best friend value, which is either another <code data-x="">Person</code> instance
    or null.</p></li>
   </ul>

   <p>We could then define <code data-x="">Person</code> instances to be <span>serializable
   objects</span> by annotating the <code data-x="">Person</code> interface with the <code
   data-x="Serializable">[Serializable]</code> <span>extended attribute</span>, and defining the
   following accompanying algorithms:</p>

   <dl>
    <dt><span>serialization steps</span></dt>
    <dd>
     <ol>
      <li><p>Set <var>serialized</var>.[[Name]] to <var>value</var>'s associated name
      value.</p></li>

      <li><p>Let <var>serializedBestFriend</var> be the <span>sub-serialization</span> of
      <var>value</var>'s associated best friend value.</p></li>

      <li><p>Set <var>serialized</var>.[[BestFriend]] to <var>serializedBestFriend</var>.</p></li>
     </ol>
    </dd>

    <dt><span>deserialization steps</span></dt>
    <dd>
     <ol>
      <li><p>Set <var>value</var>'s associated name value to
      <var>serialized</var>.[[Name]].</p></li>

      <li><p>Let <var>deserializedBestFriend</var> be the <span>sub-deserialization</span> of
      <var>serialized</var>.[[BestFriend]].</p></li>

      <li><p>Set <var>value</var>'s associated best friend value to
      <var>deserializedBestFriend</var>.</p></li>
     </ol>
    </dd>
   </dl>
  </div>

  <p>Objects defined in the JavaScript specification are handled by the
  <span>StructuredSerialize</span> abstract operation directly.</p>

  <p class="note" id="cloneable-objects">Originally, this specification defined the concept of
  "cloneable objects", which could be cloned from one <span>realm</span> to another. However, to
  better specify the behavior of certain more complex situations, the model was updated to make the
  serialization and deserialization explicit.</p>

  </div>

  <h4 export data-lt="transferable object"><dfn>Transferable objects</dfn></h4>

  <p><span>Transferable objects</span> support being transferred across <span
  data-x="agent">agents</span>. Transferring is effectively recreating the object while sharing a
  reference to the underlying data and then detaching the object being transferred. This is useful
  to transfer ownership of expensive resources. Not all objects are <span>transferable
  objects</span> and not all aspects of objects that are <span>transferable objects</span> are
  necessarily preserved when transferred.</p>

  <p class="note">Transferring is an irreversible and non-idempotent operation. Once an object has
  been transferred, it cannot be transferred, or indeed used, again.</p>

  <div w-nodev>

  <p><span data-x="platform object">Platform objects</span> can be <span>transferable objects</span>
  if their <span>primary interface</span> is decorated with the <dfn extended-attribute
  data-lt="Transferable" data-x="Transferable"><code>[Transferable]</code></dfn> IDL <span>extended
  attribute</span>. Such interfaces must also define the following algorithms:</p>

  <dl>
   <dt><dfn export>transfer steps</dfn>, taking a <span>platform object</span>
   <var>value</var> and a <span>Record</span> <var>dataHolder</var></dt>
   <dd>
    <p>A set of steps that transfers the data in <var>value</var> into fields of
    <var>dataHolder</var>. The resulting data held in <var>dataHolder</var> must be
    independent of any <span>realm</span>.</p>

    <p>These steps may throw an exception if transferral is not possible.</p>
   </dd>

   <dt><dfn export>transfer-receiving steps</dfn>, taking a <span>Record</span>
   <var>dataHolder</var> and a <span>platform object</span> <var>value</var></dt>
   <dd>
    <p>A set of steps that receives the data in <var>dataHolder</var>, using it to set up
    <var>value</var> as appropriate. <var>value</var> will be a newly-created instance of the
    <span>platform object</span> type in question, with none of its internal data set up; setting
    that up is the job of these steps.</p>

    <p>These steps may throw an exception if it is not possible to receive the transfer.</p>
   </dd>
  </dl>

  <p>It is up to the definition of individual platform objects to determine what data is transferred
  by these steps. Typically the steps are very symmetric.</p>

  <p>The <code data-x="Transferable">[Transferable]</code> extended attribute must take no
  arguments, and must only appear on an interface. It must not appear more than once on an
  interface.</p>

  <p>For a given <span>platform object</span>, only the object's <span>primary interface</span> is
  considered during the transferring process. Thus, if inheritance is involved in defining the
  interface, each <code data-x="Transferable">[Transferable]</code>-annotated interface in the
  inheritance chain needs to define standalone <span>transfer steps</span> and
  <span>transfer-receiving steps</span>, including taking into account any important data that might
  come from inherited interfaces.</p>

  <p><span data-x="platform object">Platform objects</span> that are <span>transferable
  objects</span> have a <dfn for="platform object" attribute>[[Detached]]</dfn> internal slot. This
  is used to ensure that once a platform object has been transferred, it cannot be transferred
  again.</p>

  <p>Objects defined in the JavaScript specification are handled by the
  <span>StructuredSerializeWithTransfer</span> abstract operation directly.</p>

  <h4 id="structuredserializeinternal" noexport data-lt="StructuredSerializeInternal"
  abstract-op><dfn>StructuredSerializeInternal</dfn> ( <var>value</var>, <var>forStorage</var> [ ,
  <var>memory</var> ] )</h4>

  <p>The <span>StructuredSerializeInternal</span> abstract operation takes as input a JavaScript
  value <var>value</var> and serializes it to a <span>realm</span>-independent form, represented
  here as a <span>Record</span>. This serialized form has all the information necessary to later
  deserialize into a new JavaScript value in a different realm.</p>

  <p>This process can throw an exception, for example when trying to serialize un-serializable
  objects.</p>

  <ol>
   <li>
    <p>If <var>memory</var> was not supplied, let <var>memory</var> be an empty <span
    data-x="ordered map">map</span>.</p>

    <p class="note">The purpose of the <var>memory</var> map is to avoid serializing objects twice.
    This ends up preserving cycles and the identity of duplicate objects in graphs.</p>
   </li>

   <li><p>If <var>memory</var>[<var>value</var>] <span data-x="map exists">exists</span>, then
   return <var>memory</var>[<var>value</var>].</p></li>

   <li><p>Let <var>deep</var> be false.</p></li>

   <li><p>If <var>value</var> is undefined, null, <span data-x="js-Boolean">a Boolean</span>, <span
   data-x="js-Number">a Number</span>, <span data-x="js-BigInt">a BigInt</span>, or <span
   data-x="js-String">a String</span>, then return { [[Type]]: "primitive", [[Value]]:
   <var>value</var> }.</p></li>

   <li><p>If <var>value</var> <span data-x="js-Symbol">is a Symbol</span>, then throw a
   <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>serialized</var> be an uninitialized value.</p></li>

   <li><p>If <var>value</var> has a [[BooleanData]] internal slot, then set <var>serialized</var> to
   { [[Type]]: "Boolean", [[BooleanData]]: <var>value</var>.[[BooleanData]] }.</p></li>

   <li><p>Otherwise, if <var>value</var> has a [[NumberData]] internal slot, then set
   <var>serialized</var> to { [[Type]]: "Number", [[NumberData]]: <var>value</var>.[[NumberData]]
   }.</p></li>

   <li><p>Otherwise, if <var>value</var> has a [[BigIntData]]
   internal slot, then set <var>serialized</var> to { [[Type]]: "BigInt", [[BigIntData]]:
   <var>value</var>.[[BigIntData]] }.</p></li>

   <li><p>Otherwise, if <var>value</var> has a [[StringData]] internal slot, then set
   <var>serialized</var> to { [[Type]]: "String", [[StringData]]: <var>value</var>.[[StringData]]
   }.</p></li>

   <li><p>Otherwise, if <var>value</var> has a [[DateValue]] internal slot, then set
   <var>serialized</var> to { [[Type]]: "Date", [[DateValue]]: <var>value</var>.[[DateValue]]
   }.</p></li>

   <li><p>Otherwise, if <var>value</var> has a [[RegExpMatcher]] internal slot, then set
   <var>serialized</var> to { [[Type]]: "RegExp", [[RegExpMatcher]]:
   <var>value</var>.[[RegExpMatcher]], [[OriginalSource]]: <var>value</var>.[[OriginalSource]],
   [[OriginalFlags]]: <var>value</var>.[[OriginalFlags]] }.</p></li>

   <li>
    <p>Otherwise, if <var>value</var> has an [[ArrayBufferData]] internal slot, then:</p>

    <ol>
     <li>
      <p>If <span>IsSharedArrayBuffer</span>(<var>value</var>) is true, then:

      <ol>
       <li>
        <p>If the <span>current settings object</span>'s <span
        data-x="concept-settings-object-cross-origin-isolated-capability">cross-origin isolated
        capability</span> is false, then throw a <span>"<code>DataCloneError</code>"</span>
        <code>DOMException</code>.</p>

        <p class="note">This check is only needed when serializing (and not when deserializing) as
        the <span data-x="concept-settings-object-cross-origin-isolated-capability">cross-origin
        isolated capability</span> cannot change over time and a <code>SharedArrayBuffer</code>
        cannot leave an <span>agent cluster</span>.</p>
       </li>

       <li><p>If <var>forStorage</var> is true, then throw a
       <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p></li>

       <li><p>If <var>value</var> has an [[ArrayBufferMaxByteLength]] internal slot, then set
       <var>serialized</var> to { [[Type]]: "GrowableSharedArrayBuffer", [[ArrayBufferData]]:
       <var>value</var>.[[ArrayBufferData]], [[ArrayBufferByteLengthData]]:
       <var>value</var>.[[ArrayBufferByteLengthData]], [[ArrayBufferMaxByteLength]]:
       <var>value</var>.[[ArrayBufferMaxByteLength]], [[AgentCluster]]: the <span>surrounding
       agent</span>'s <span>agent cluster</span> }.</p></li>

       <li><p>Otherwise, set <var>serialized</var> to { [[Type]]: "SharedArrayBuffer",
       [[ArrayBufferData]]: <var>value</var>.[[ArrayBufferData]], [[ArrayBufferByteLength]]:
       <var>value</var>.[[ArrayBufferByteLength]], [[AgentCluster]]: the <span>surrounding
       agent</span>'s <span>agent cluster</span> }.</p></li>
      </ol>
     </li>

     <li>
      <p>Otherwise:</p>

      <ol>
       <li><p>If <span>IsDetachedBuffer</span>(<var>value</var>) is true, then throw a
       <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p></li>

       <li><p>Let <var>size</var> be <var>value</var>.[[ArrayBufferByteLength]].</li>

       <li>
        <p>Let <var>dataCopy</var> be ? <span>CreateByteDataBlock</span>(<var>size</var>).</p>

        <p class="note">This can throw a <code data-x="js-RangeError">RangeError</code> exception
        upon allocation failure.</p>
       </li>

       <li><p>Perform <span>CopyDataBlockBytes</span>(<var>dataCopy</var>, 0,
       <var>value</var>.[[ArrayBufferData]], 0, <var>size</var>).</p></li>

       <li><p>If <var>value</var> has an [[ArrayBufferMaxByteLength]] internal slot, then set
       <var>serialized</var> to { [[Type]]: "ResizableArrayBuffer", [[ArrayBufferData]]:
       <var>dataCopy</var>, [[ArrayBufferByteLength]]: <var>size</var>,
       [[ArrayBufferMaxByteLength]]: <var>value</var>.[[ArrayBufferMaxByteLength]] }.</p></li>

       <li><p>Otherwise, set <var>serialized</var> to { [[Type]]: "ArrayBuffer",
       [[ArrayBufferData]]: <var>dataCopy</var>, [[ArrayBufferByteLength]]: <var>size</var>
       }.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if <var>value</var> has a [[ViewedArrayBuffer]] internal slot, then:</p>

    <ol>
     <li><p>If <span>IsArrayBufferViewOutOfBounds</span>(<var>value</var>) is true, then throw a
     <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>Let <var>buffer</var> be the value of <var>value</var>'s [[ViewedArrayBuffer]] internal
     slot.</p></li>

     <li><p>Let <var>bufferSerialized</var> be ?
     <span>StructuredSerializeInternal</span>(<var>buffer</var>, <var>forStorage</var>,
     <var>memory</var>).</p></li>

     <li><p><span>Assert</span>: <var>bufferSerialized</var>.[[Type]] is "ArrayBuffer",
     "ResizableArrayBuffer", "SharedArrayBuffer", or "GrowableSharedArrayBuffer".</p></li>

     <li><p>If <var>value</var> has a [[DataView]] internal slot, then set <var>serialized</var> to
     { [[Type]]: "ArrayBufferView", [[Constructor]]: "DataView", [[ArrayBufferSerialized]]:
     <var>bufferSerialized</var>, [[ByteLength]]: <var>value</var>.[[ByteLength]], [[ByteOffset]]:
     <var>value</var>.[[ByteOffset]] }.</p></li>

     <li>
      <p>Otherwise:</p>

      <ol>
       <li><p><span>Assert</span>: <var>value</var> has a [[TypedArrayName]] internal slot.</p></li>

       <li><p>Set <var>serialized</var> to { [[Type]]: "ArrayBufferView", [[Constructor]]:
       <var>value</var>.[[TypedArrayName]], [[ArrayBufferSerialized]]: <var>bufferSerialized</var>,
       [[ByteLength]]: <var>value</var>.[[ByteLength]], [[ByteOffset]]:
       <var>value</var>.[[ByteOffset]], [[ArrayLength]]: <var>value</var>.[[ArrayLength]] }.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if <var>value</var> has [[MapData]] internal slot, then:</p>

    <ol>
     <li><p>Set <var>serialized</var> to { [[Type]]: "Map", [[MapData]]: a new empty <span
     data-x="js-List">List</span> }.</p></li>

     <li><p>Set <var>deep</var> to true.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if <var>value</var> has [[SetData]] internal slot, then:</p>

    <ol>
     <li><p>Set <var>serialized</var> to { [[Type]]: "Set", [[SetData]]: a new empty <span
     data-x="js-List">List</span> }.</p></li>

     <li><p>Set <var>deep</var> to true.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if <var>value</var> has an [[ErrorData]] internal slot and <var>value</var> is not
    a <span>platform object</span>, then:</p>
    <!-- "is not a platform object" check is needed because DOMExceptions can have [[ErrorData]]
         too -->

    <ol>
     <li><p>Let <var>name</var> be ? <span data-x="js-Get">Get</span>(<var>value</var>,
     "name").</p></li>

     <li><p>If <var>name</var> is not one of "Error", "EvalError", "RangeError", "ReferenceError",
     "SyntaxError", "TypeError", or "URIError", then set <var>name</var> to "Error".</p></li>

     <li><p>Let <var>valueMessageDesc</var> be ? <var>value</var>.[[GetOwnProperty]]("<code
     data-x="">message</code>").</p></li>

     <li><p>Let <var>message</var> be undefined if
     <span>IsDataDescriptor</span>(<var>valueMessageDesc</var>) is false, and
     ? <span>ToString</span>(<var>valueMessageDesc</var>.[[Value]]) otherwise.</p></li>

     <li><p>Set <var>serialized</var> to { [[Type]]: "Error", [[Name]]: <var>name</var>,
     [[Message]]: <var>message</var> }.</p></li>

     <li>
      <p>User agents should attach a serialized representation of any interesting accompanying
      data which are not yet specified, notably the <code data-x="">stack</code> property, to
      <var>serialized</var>.</p>

      <p class="note">See the <cite>Error Stacks</cite> proposal for in-progress work on specifying
      this data. <ref>JSERRORSTACKS</ref></p>
     </li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if <var>value</var> is an Array exotic object, then:</p>
    <!-- IsArray supports proxies too, which we cannot -->

    <ol>
     <li><p>Let <var>valueLenDescriptor</var> be ?
     <span>OrdinaryGetOwnProperty</span>(<var>value</var>, "<code data-x="">length</code>").</p></li>

     <li><p>Let <var>valueLen</var> be <var>valueLenDescriptor</var>.[[Value]].</p></li>

     <li><p>Set <var>serialized</var> to { [[Type]]: "Array", [[Length]]: <var>valueLen</var>,
     [[Properties]]: a new empty <span data-x="js-List">List</span> }.</p></li>

     <li><p>Set <var>deep</var> to true.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if <var>value</var> is a <span>platform object</span> that is a <span
    data-x="serializable objects">serializable object</span>:</p>

    <ol>
     <li><p>If <var>value</var> has a <span>[[Detached]]</span> internal slot whose value is true,
     then throw a <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>Let <var>typeString</var> be the identifier of the <span>primary interface</span> of
     <var>value</var>.</p></li>

     <li><p>Set <var>serialized</var> to { [[Type]]: <var>typeString</var> }.</p></li>

     <li><p>Set <var>deep</var> to true.</p></li>
    </ol>
   </li>

   <li><p>Otherwise, if <var>value</var> is a <span>platform object</span>, then throw a
   <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Otherwise, if <span>IsCallable</span>(<var>value</var>) is true, then throw a
   <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p></li>

   <li>
    <p>Otherwise, if <var>value</var> has any internal slot other than [[Prototype]],
    [[Extensible]], or [[PrivateElements]], then throw a <span>"<code>DataCloneError</code>"</span>
    <code>DOMException</code>.</p>

    <p class="example">For instance, a [[PromiseState]] or [[WeakMapData]] internal slot.</p>
   </li>

   <li>
    <p>Otherwise, if <var>value</var> is an exotic object and <var>value</var> is not the
    <span>%Object.prototype%</span> intrinsic object associated with any <span>realm</span>, then
    throw a <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p>

    <p class="example">For instance, a proxy object.</p>
   </li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p>Set <var>serialized</var> to { [[Type]]: "Object", [[Properties]]: a new empty <span
     data-x="js-List">List</span> }.</p></li>

     <li><p>Set <var>deep</var> to true.</p></li>
    </ol>

    <p class="note"><span>%Object.prototype%</span> will end up being handled via this step and
    subsequent steps. The end result is that its exoticness is ignored, and after deserialization
    the result will be an empty object (not an <span>immutable prototype exotic object</span>).
   </li>

   <li><p><span data-x="map set">Set</span> <var>memory</var>[<var>value</var>] to
   <var>serialized</var>.</p></li>

   <li>
    <p>If <var>deep</var> is true, then:</p>

    <ol>
     <li>
      <p>If <var>value</var> has a [[MapData]] internal slot, then:

      <ol>
       <li><p>Let <var>copiedList</var> be a new empty <span data-x="js-List">List</span>.

       <li>
        <p><span data-x="list iterate">For each</span> <span>Record</span> { [[Key]], [[Value]] }
        <var>entry</var> of <var>value</var>.[[MapData]]:</p>

        <ol>
         <li><p>Let <var>copiedEntry</var> be a new <span>Record</span> { [[Key]]:
         <var>entry</var>.[[Key]], [[Value]]: <var>entry</var>.[[Value]] }.</p></li>

         <li><p>If <var>copiedEntry</var>.[[Key]] is not the special value <i>empty</i>, <span
         data-x="list append">append</span> <var>copiedEntry</var> to <var>copiedList</var>.</p></li>
        </ol>
       </li>

       <li>
        <p><span data-x="list iterate">For each</span> <span>Record</span> { [[Key]], [[Value]] }
        <var>entry</var> of <var>copiedList</var>:</p>

        <ol>
         <li><p>Let <var>serializedKey</var> be ?
         <span>StructuredSerializeInternal</span>(<var>entry</var>.[[Key]], <var>forStorage</var>,
         <var>memory</var>).</p></li>

         <li><p>Let <var>serializedValue</var> be ?
         <span>StructuredSerializeInternal</span>(<var>entry</var>.[[Value]], <var>forStorage</var>,
         <var>memory</var>).</p></li>

         <li><p><span data-x="list append">Append</span> { [[Key]]: <var>serializedKey</var>,
         [[Value]]: <var>serializedValue</var> } to <var>serialized</var>.[[MapData]].</p></li>
        </ol>
       </li>
      </ol>
     </li>

     <li>
      <p>Otherwise, if <var>value</var> has a [[SetData]] internal slot, then:</p>

      <ol>
       <li><p>Let <var>copiedList</var> be a new empty <span data-x="js-List">List</span>.

       <li>
        <p><span data-x="list iterate">For each</span> <var>entry</var> of <var>value</var>.[[SetData]]:</p>

        <ol>
         <li><p>If <var>entry</var> is not the special value <i>empty</i>, <span data-x="list
         append">append</span> <var>entry</var> to <var>copiedList</var>.</p></li>
        </ol>
       </li>

       <li>
        <p><span data-x="list iterate">For each</span> <var>entry</var> of <var>copiedList</var>:</p>

        <ol>
         <li><p>Let <var>serializedEntry</var> be ?
         <span>StructuredSerializeInternal</span>(<var>entry</var>, <var>forStorage</var>,
         <var>memory</var>).</p></li>

         <li><p><span data-x="list append">Append</span> <var>serializedEntry</var> to
         <var>serialized</var>.[[SetData]].</p></li>
        </ol>
       </li>
      </ol>
     </li>

     <li>
      <p>Otherwise, if <var>value</var> is a <span>platform object</span> that is a <span
      data-x="serializable objects">serializable object</span>, then perform the <span>serialization
      steps</span> for <var>value</var>'s <span>primary interface</span>, given <var>value</var>,
      <var>serialized</var>, and <var>forStorage</var>.</p>

      <p>The <span>serialization steps</span> may need to perform a <dfn
      export>sub-serialization</dfn>. This is an operation which takes as input a value
      <var>subValue</var>, and returns <span>StructuredSerializeInternal</span>(<var>subValue</var>,
      <var>forStorage</var>, <var>memory</var>). (In other words, a <span>sub-serialization</span>
      is a specialization of <span>StructuredSerializeInternal</span> to be consistent within this
      invocation.)</p>
     </li>

     <li>
      <p>Otherwise, for each <var>key</var> in !
      <span>EnumerableOwnProperties</span>(<var>value</var>, key):</p>

      <ol>
       <li>
        <p>If ! <span>HasOwnProperty</span>(<var>value</var>, <var>key</var>) is true, then:</p>

        <ol>
         <li><p>Let <var>inputValue</var> be ? <var>value</var>.[[Get]](<var>key</var>,
         <var>value</var>).</p></li>

         <li><p>Let <var>outputValue</var> be ?
         <span>StructuredSerializeInternal</span>(<var>inputValue</var>, <var>forStorage</var>,
         <var>memory</var>).</p></li>

         <li><p><span data-x="list append">Append</span> { [[Key]]: <var>key</var>, [[Value]]:
         <var>outputValue</var> } to <var>serialized</var>.[[Properties]].</p></li>
        </ol>
       </li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Return <var>serialized</var>.</p></li>
  </ol>

  <div class="example">
   <p>It's important to realize that the <span data-x="Record">Records</span>
   produced by <span>StructuredSerializeInternal</span> might contain "pointers" to other records
   that create circular references. For example, when we pass the following JavaScript object into
   <span>StructuredSerializeInternal</span>:</p>

  <pre><code class="js">const o = {};
o.myself = o;</code></pre>

  <p>it produces the following result:</p>

  <pre><samp>{
  [[Type]]: "Object",
  [[Properties]]: «
    {
      [[Key]]: "myself",
      [[Value]]: <i>&lt;a pointer to this whole structure></i>
    }
  »
}</samp></pre>
  </div>


  <h4 id="structuredserialize" data-lt="StructuredSerialize"
  abstract-op><dfn>StructuredSerialize</dfn> ( <var>value</var> )</h4>

  <ol>
   <li><p>Return ? <span>StructuredSerializeInternal</span>(<var>value</var>, false).</p></li>
  </ol>


  <h4 id="structuredserializeforstorage" data-lt="StructuredSerializeForStorage"
  abstract-op><dfn>StructuredSerializeForStorage</dfn> ( <var>value</var> )</h4>

  <ol>
   <li><p>Return ? <span>StructuredSerializeInternal</span>(<var>value</var>, true).</p></li>
  </ol>


  <h4 id="structureddeserialize" data-lt="StructuredDeserialize"
  abstract-op><dfn>StructuredDeserialize</dfn> ( <var>serialized</var>, <var>targetRealm</var> [ ,
  <var>memory</var> ] )</h4>

  <p>The <span>StructuredDeserialize</span> abstract operation takes as input a <span>Record</span>
  <var>serialized</var>, which was previously produced by <span>StructuredSerialize</span> or
  <span>StructuredSerializeForStorage</span>, and deserializes it into a new JavaScript value,
  created in <var>targetRealm</var>.</p>

  <p>This process can throw an exception, for example when trying to allocate memory for the new
  objects (especially <code data-x="">ArrayBuffer</code> objects).</p>

  <ol>
   <li>
    <p>If <var>memory</var> was not supplied, let <var>memory</var> be an empty <span
    data-x="ordered map">map</span>.</p>

    <p class="note">The purpose of the <var>memory</var> map is to avoid deserializing objects
    twice. This ends up preserving cycles and the identity of duplicate objects in graphs.</p>
   </li>

   <li><p>If <var>memory</var>[<var>serialized</var>] <span data-x="map exists">exists</span>, then
   return <var>memory</var>[<var>serialized</var>].</p></li>

   <li><p>Let <var>deep</var> be false.</p></li>

   <li><p>Let <var>value</var> be an uninitialized value.</p></li>

   <li><p>If <var>serialized</var>.[[Type]] is "primitive", then set <var>value</var> to
   <var>serialized</var>.[[Value]].</p>

   <li><p>Otherwise, if <var>serialized</var>.[[Type]] is "Boolean", then set <var>value</var> to a
   new Boolean object in <var>targetRealm</var> whose [[BooleanData]] internal slot value is
   <var>serialized</var>.[[BooleanData]].</p></li>

   <li><p>Otherwise, if <var>serialized</var>.[[Type]] is "Number", then set <var>value</var> to a
   new Number object in <var>targetRealm</var> whose [[NumberData]] internal slot value is
   <var>serialized</var>.[[NumberData]].</p></li>

   <li><p>Otherwise, if <var>serialized</var>.[[Type]] is "BigInt", then set <var>value</var> to a
   new BigInt object in <var>targetRealm</var> whose [[BigIntData]] internal slot value is
   <var>serialized</var>.[[BigIntData]].</p></li>

   <li><p>Otherwise, if <var>serialized</var>.[[Type]] is "String", then set <var>value</var> to a
   new String object in <var>targetRealm</var> whose [[StringData]] internal slot value is
   <var>serialized</var>.[[StringData]].</p></li>

   <li><p>Otherwise, if <var>serialized</var>.[[Type]] is "Date", then set <var>value</var> to a new
   Date object in <var>targetRealm</var> whose [[DateValue]] internal slot value is
   <var>serialized</var>.[[DateValue]].</p></li>

   <li><p>Otherwise, if <var>serialized</var>.[[Type]] is "RegExp", then set <var>value</var> to a
   new RegExp object in <var>targetRealm</var> whose [[RegExpMatcher]] internal slot value is
   <var>serialized</var>.[[RegExpMatcher]], whose [[OriginalSource]] internal slot value is
   <var>serialized</var>.[[OriginalSource]], and whose [[OriginalFlags]] internal slot value is
   <var>serialized</var>.[[OriginalFlags]].</p></li>

   <li>
    <p>Otherwise, if <var>serialized</var>.[[Type]] is "SharedArrayBuffer", then:</p>

    <ol>
     <li><p>If <var>targetRealm</var>'s corresponding <span>agent cluster</span> is not
     <var>serialized</var>.[[AgentCluster]], then throw a
     <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>Otherwise, set <var>value</var> to a new SharedArrayBuffer object in
     <var>targetRealm</var> whose [[ArrayBufferData]] internal slot value is
     <var>serialized</var>.[[ArrayBufferData]] and whose [[ArrayBufferByteLength]] internal slot
     value is <var>serialized</var>.[[ArrayBufferByteLength]].</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if <var>serialized</var>.[[Type]] is "GrowableSharedArrayBuffer", then:</p>

    <ol>
     <li><p>If <var>targetRealm</var>'s corresponding <span>agent cluster</span> is not
     <var>serialized</var>.[[AgentCluster]], then throw a
     <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>Otherwise, set <var>value</var> to a new SharedArrayBuffer object in
     <var>targetRealm</var> whose [[ArrayBufferData]] internal slot value is
     <var>serialized</var>.[[ArrayBufferData]], whose [[ArrayBufferByteLengthData]] internal slot
     value is <var>serialized</var>.[[ArrayBufferByteLengthData]], and whose
     [[ArrayBufferMaxByteLength]] internal slot value is
     <var>serialized</var>.[[ArrayBufferMaxByteLength]].</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if <var>serialized</var>.[[Type]] is "ArrayBuffer", then set <var>value</var> to a
    new ArrayBuffer object in <var>targetRealm</var> whose [[ArrayBufferData]] internal slot value
    is <var>serialized</var>.[[ArrayBufferData]], and whose [[ArrayBufferByteLength]] internal slot
    value is <var>serialized</var>.[[ArrayBufferByteLength]].</p>

    <p>If this throws an exception, catch it, and then throw a
    <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p>

    <p class="note">This step might throw an exception if there is not enough memory available to
    create such an ArrayBuffer object.</p>
   </li>

   <li>
    <p>Otherwise, if <var>serialized</var>.[[Type]] is "ResizableArrayBuffer", then set
    <var>value</var> to a new ArrayBuffer object in <var>targetRealm</var> whose [[ArrayBufferData]]
    internal slot value is <var>serialized</var>.[[ArrayBufferData]], whose
    [[ArrayBufferByteLength]] internal slot value is
    <var>serialized</var>.[[ArrayBufferByteLength]], and whose [[ArrayBufferMaxByteLength]] internal
    slot value is <var>serialized</var>.[[ArrayBufferMaxByteLength]].</p>

    <p>If this throws an exception, catch it, and then throw a
    <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p>

    <p class="note">This step might throw an exception if there is not enough memory available to
    create such an ArrayBuffer object.</p>
   </li>

   <li>
    <p>Otherwise, if <var>serialized</var>.[[Type]] is "ArrayBufferView", then:</p>

    <ol>
     <li><p>Let <var>deserializedArrayBuffer</var> be ?
     <span>StructuredDeserialize</span>(<var>serialized</var>.[[ArrayBufferSerialized]],
     <var>targetRealm</var>, <var>memory</var>).</p></li>

     <li><p>If <var>serialized</var>.[[Constructor]] is "DataView", then set <var>value</var> to a
     new DataView object in <var>targetRealm</var> whose [[ViewedArrayBuffer]] internal slot value
     is <var>deserializedArrayBuffer</var>, whose [[ByteLength]] internal slot value is
     <var>serialized</var>.[[ByteLength]], and whose [[ByteOffset]] internal slot value is
     <var>serialized</var>.[[ByteOffset]].</p></li>

     <li><p>Otherwise, set <var>value</var> to a new typed array object in <var>targetRealm</var>,
     using the constructor given by <var>serialized</var>.[[Constructor]], whose
     [[ViewedArrayBuffer]] internal slot value is <var>deserializedArrayBuffer</var>, whose
     [[TypedArrayName]] internal slot value is <var>serialized</var>.[[Constructor]], whose
     [[ByteLength]] internal slot value is <var>serialized</var>.[[ByteLength]], whose
     [[ByteOffset]] internal slot value is <var>serialized</var>.[[ByteOffset]], and whose
     [[ArrayLength]] internal slot value is <var>serialized</var>.[[ArrayLength]].</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if <var>serialized</var>.[[Type]] is "Map", then:</p>

    <ol>
     <li><p>Set <var>value</var> to a new Map object in <var>targetRealm</var> whose [[MapData]]
     internal slot value is a new empty <span data-x="js-List">List</span>.</p></li>

     <li><p>Set <var>deep</var> to true.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if <var>serialized</var>.[[Type]] is "Set", then:</p>

    <ol>
     <li><p>Set <var>value</var> to a new Set object in <var>targetRealm</var> whose [[SetData]]
     internal slot value is a new empty <span data-x="js-List">List</span>.</p></li>

     <li><p>Set <var>deep</var> to true.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if <var>serialized</var>.[[Type]] is "Array", then:</p>

    <ol>
     <li><p>Let <var>outputProto</var> be
     <var>targetRealm</var>.[[Intrinsics]].[[<span>%Array.prototype%</span>]].</p></li>

     <li><p>Set <var>value</var> to ! <span>ArrayCreate</span>(<var>serialized</var>.[[Length]],
     <var>outputProto</var>).</p></li>

     <li><p>Set <var>deep</var> to true.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if <var>serialized</var>.[[Type]] is "Object", then:</p>

    <ol>
     <li><p>Set <var>value</var> to a new Object in <var>targetRealm</var>.</p></li>

     <li><p>Set <var>deep</var> to true.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if <var>serialized</var>.[[Type]] is "Error", then:</p>

    <ol>
     <li><p>Let <var>prototype</var> be <span>%Error.prototype%</span>.</p></li>

     <li><p>If <var>serialized</var>.[[Name]] is "EvalError", then set <var>prototype</var> to
     <span>%EvalError.prototype%</span>.</p></li>

     <li><p>If <var>serialized</var>.[[Name]] is "RangeError", then set <var>prototype</var>
     to <span>%RangeError.prototype%</span>.</p></li>

     <li><p>If <var>serialized</var>.[[Name]] is "ReferenceError", then set
     <var>prototype</var> to <span>%ReferenceError.prototype%</span>.</p></li>

     <li><p>If <var>serialized</var>.[[Name]] is "SyntaxError", then set <var>prototype</var>
     to <span>%SyntaxError.prototype%</span>.</p></li>

     <li><p>If <var>serialized</var>.[[Name]] is "TypeError", then set <var>prototype</var> to
     <span>%TypeError.prototype%</span>.</p></li>

     <li><p>If <var>serialized</var>.[[Name]] is "URIError", then set <var>prototype</var> to
     <span>%URIError.prototype%</span>.</p></li>

     <li><p>Let <var>message</var> be <var>serialized</var>.[[Message]].</p></li>

     <li><p>Set <var>value</var> to <span>OrdinaryObjectCreate</span>(<var>prototype</var>, «
     [[ErrorData]] »).</p></li>

     <li><p>Let <var>messageDesc</var> be <span>PropertyDescriptor</span>{ [[Value]]:
     <var>message</var>, [[Writable]]: true, [[Enumerable]]: false, [[Configurable]]: true
     }.</p></li>

     <li><p>If <var>message</var> is not undefined, then perform !
     <span>OrdinaryDefineOwnProperty</span>(<var>value</var>, "<code data-x="">message</code>",
     <var>messageDesc</var>).</p></li>

     <li><p>Any interesting accompanying data attached to <var>serialized</var> should be
     deserialized and attached to <var>value</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p>Let <var>interfaceName</var> be <var>serialized</var>.[[Type]].</p></li>

     <li><p>If the interface identified by <var>interfaceName</var> is not
     <span data-x="idl-exposed">exposed</span> in <var>targetRealm</var>, then throw a
     <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>Set <var>value</var> to a new instance of the interface identified by
     <var>interfaceName</var>, created in <var>targetRealm</var>.</p>

     <li><p>Set <var>deep</var> to true.</p></li>
    </ol>
   </li>

   <li><p><span data-x="map set">Set</span> <var>memory</var>[<var>serialized</var>] to
   <var>value</var>.</p></li>

   <li>
    <p>If <var>deep</var> is true, then:</p>

    <ol>
     <li>
      <p>If <var>serialized</var>.[[Type]] is "Map", then:</p>

      <ol>
       <li>
        <p><span data-x="list iterate">For each</span> <span>Record</span> { [[Key]], [[Value]] }
        <var>entry</var> of <var>serialized</var>.[[MapData]]:</p>

        <ol>
         <li><p>Let <var>deserializedKey</var> be ?
         <span>StructuredDeserialize</span>(<var>entry</var>.[[Key]], <var>targetRealm</var>,
         <var>memory</var>).</p></li>

         <li><p>Let <var>deserializedValue</var> be ?
         <span>StructuredDeserialize</span>(<var>entry</var>.[[Value]], <var>targetRealm</var>,
         <var>memory</var>).</p></li>

         <li><p><span data-x="list append">Append</span> { [[Key]]: <var>deserializedKey</var>,
         [[Value]]: <var>deserializedValue</var> } to <var>value</var>.[[MapData]].</p></li>
        </ol>
       </li>
      </ol>
     </li>

     <li>
      <p>Otherwise, if <var>serialized</var>.[[Type]] is "Set", then:</p>

      <ol>
       <li>
        <p><span data-x="list iterate">For each</span> <var>entry</var> of
        <var>serialized</var>.[[SetData]]:</p>

        <ol>
         <li><p>Let <var>deserializedEntry</var> be ?
         <span>StructuredDeserialize</span>(<var>entry</var>, <var>targetRealm</var>,
         <var>memory</var>).</p></li>

         <li><p><span data-x="list append">Append</span> <var>deserializedEntry</var> to
         <var>value</var>.[[SetData]].</p></li>
        </ol>
       </li>
      </ol>
     </li>

     <li>
      <p>Otherwise, if <var>serialized</var>.[[Type]] is "Array" or "Object", then:</p>

      <ol>
       <li>
        <p><span data-x="list iterate">For each</span> <span>Record</span> { [[Key]], [[Value]] }
        <var>entry</var> of <var>serialized</var>.[[Properties]]:</p>

        <ol>
         <li><p>Let <var>deserializedValue</var> be ?
         <span>StructuredDeserialize</span>(<var>entry</var>.[[Value]], <var>targetRealm</var>,
         <var>memory</var>).</p></li>

         <li><p>Let <var>result</var> be ! <span>CreateDataProperty</span>(<var>value</var>,
         <var>entry</var>.[[Key]], <var>deserializedValue</var>).</p></li>

         <li><p><span>Assert</span>: <var>result</var> is true.</p></li>
        </ol>
       </li>
      </ol>
     </li>

     <li>
      <p>Otherwise:</p>

      <ol>
       <li>
        <p>Perform the appropriate <span>deserialization steps</span> for the interface identified
        by <var>serialized</var>.[[Type]], given <var>serialized</var>, <var>value</var>, and
        <var>targetRealm</var>.</p>

        <p>The <span>deserialization steps</span> may need to perform a <dfn
        export>sub-deserialization</dfn>. This is an operation which takes as input a
        previously-serialized <span>Record</span> <var>subSerialized</var>, and returns
        <span>StructuredDeserialize</span>(<var>subSerialized</var>, <var>targetRealm</var>,
        <var>memory</var>). (In other words, a <span>sub-deserialization</span> is a specialization
        of <span>StructuredDeserialize</span> to be consistent within this invocation.)</p>
       </li>
      </ol>
    </ol>
   </li>

   <li><p>Return <var>value</var>.</p></li>
  </ol>

  <h4 id="structuredserializewithtransfer" data-lt="StructuredSerializeWithTransfer"
  abstract-op><dfn>StructuredSerializeWithTransfer</dfn> ( <var>value</var>, <var>transferList</var>
  )</h4>

  <ol>
   <li>
    <p>Let <var>memory</var> be an empty <span data-x="ordered map">map</span>.</p>

    <p class="note">In addition to how it is used normally by
    <span>StructuredSerializeInternal</span>, in this algorithm <var>memory</var> is also used to
    ensure that <span>StructuredSerializeInternal</span> ignores items in <var>transferList</var>,
    and let us do our own handling instead.</p>
   </li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>transferable</var> of
    <var>transferList</var>:</p>

    <ol>
     <li><p>If <var>transferable</var> has neither an [[ArrayBufferData]] internal slot nor a
     <span>[[Detached]]</span> internal slot, then throw a
     <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>If <var>transferable</var> has an [[ArrayBufferData]] internal slot and
     <span>IsSharedArrayBuffer</span>(<var>transferable</var>) is true, then throw a
     <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>If <var>memory</var>[<var>transferable</var>] <span data-x="map exists">exists</span>,
     then throw a <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p></li>

     <li>
      <p><span data-x="map set">Set</span> <var>memory</var>[<var>transferable</var>] to {
      [[Type]]: an uninitialized value }.</p>

      <p class="note"><var>transferable</var> is not transferred yet as transferring has side
      effects and <span>StructuredSerializeInternal</span> needs to be able to throw first.</p>
     </li>
    </ol>
   </li>

   <li><p>Let <var>serialized</var> be ? <span>StructuredSerializeInternal</span>(<var>value</var>,
   false, <var>memory</var>).</p></li>

   <li><p>Let <var>transferDataHolders</var> be a new empty <span
   data-x="js-List">List</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>transferable</var> of
    <var>transferList</var>:</p>

    <ol>
     <li><p>If <var>transferable</var> has an [[ArrayBufferData]] internal slot and
     <span>IsDetachedBuffer</span>(<var>transferable</var>) is true, then throw a
     <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>If <var>transferable</var> has a <span>[[Detached]]</span> internal slot and
     <var>transferable</var>.<span>[[Detached]]</span> is true, then throw a
     <span>"<code>DataCloneError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>Let <var>dataHolder</var> be <var>memory</var>[<var>transferable</var>].</p></li>

     <li>
      <p>If <var>transferable</var> has an [[ArrayBufferData]] internal slot, then:</p>

      <ol>
       <li>
        <p>If <var>transferable</var> has an [[ArrayBufferMaxByteLength]] internal slot, then:</p>

        <ol>
         <li><p>Set <var>dataHolder</var>.[[Type]] to "ResizableArrayBuffer".</p></li>

         <li><p>Set <var>dataHolder</var>.[[ArrayBufferData]] to
         <var>transferable</var>.[[ArrayBufferData]].</p></li>

         <li><p>Set <var>dataHolder</var>.[[ArrayBufferByteLength]] to
         <var>transferable</var>.[[ArrayBufferByteLength]].</p></li>

         <li><p>Set <var>dataHolder</var>.[[ArrayBufferMaxByteLength]] to
         <var>transferable</var>.[[ArrayBufferMaxByteLength]].</p></li>
        </ol>
       </li>

       <li>
        <p>Otherwise:</p>

        <ol>
         <li><p>Set <var>dataHolder</var>.[[Type]] to "ArrayBuffer".</p></li>

         <li><p>Set <var>dataHolder</var>.[[ArrayBufferData]] to
         <var>transferable</var>.[[ArrayBufferData]].</p></li>

         <li><p>Set <var>dataHolder</var>.[[ArrayBufferByteLength]] to
         <var>transferable</var>.[[ArrayBufferByteLength]].</p></li>
        </ol>
       </li>

       <li>
        <p>Perform ? <span>DetachArrayBuffer</span>(<var>transferable</var>).</p>

        <p class="note">Specifications can use the [[ArrayBufferDetachKey]] internal slot to prevent
        <code data-x="idl-ArrayBuffer">ArrayBuffer</code>s from being detached. This is used in
        <cite>WebAssembly JavaScript Interface</cite>, for example. <ref>WASMJS</ref></p>
       </li>
      </ol>
     </li>

     <li>
      <p>Otherwise:</p>

      <ol>
       <li><p><span>Assert</span>: <var>transferable</var> is a <span>platform object</span> that is
       a <span data-x="transferable objects">transferable object</span>.</p></li>

       <li><p>Let <var>interfaceName</var> be the identifier of the <span>primary interface</span>
       of <var>transferable</var>.</p></li>

       <li><p>Set <var>dataHolder</var>.[[Type]] to <var>interfaceName</var>.</p></li>

       <li><p>Perform the appropriate <span>transfer steps</span> for the interface identified by
       <var>interfaceName</var>, given <var>transferable</var> and <var>dataHolder</var>.</p></li>

       <li><p>Set <var>transferable</var>.<span>[[Detached]]</span> to true.</p></li>
      </ol>
     </li>

     <li><p><span data-x="list append">Append</span> <var>dataHolder</var> to
     <var>transferDataHolders</var>.</p></li>
    </ol>
   </li>

   <li><p>Return { [[Serialized]]: <var>serialized</var>, [[TransferDataHolders]]:
   <var>transferDataHolders</var> }.</p></li>
  </ol>

  <h4 id="structureddeserializewithtransfer" data-lt="StructuredDeserializeWithTransfer"
  abstract-op><dfn>StructuredDeserializeWithTransfer</dfn> ( <var>serializeWithTransferResult</var>,
  <var>targetRealm</var> )</h4>

  <ol>
   <li>
    <p>Let <var>memory</var> be an empty <span data-x="ordered map">map</span>.</p>

    <p class="note">Analogous to <span>StructuredSerializeWithTransfer</span>, in addition to how it
    is used normally by <span>StructuredDeserialize</span>, in this algorithm <var>memory</var> is
    also used to ensure that <span>StructuredDeserialize</span> ignores items in
    <var>serializeWithTransferResult</var>.[[TransferDataHolders]], and let us do our own handling
    instead.</p>
   </li>

   <li><p>Let <var>transferredValues</var> be a new empty <span
   data-x="js-List">List</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>transferDataHolder</var> of
    <var>serializeWithTransferResult</var>.[[TransferDataHolders]]:</p>

    <ol>
     <li><p>Let <var>value</var> be an uninitialized value.</p></li>

     <li>
      <p>If <var>transferDataHolder</var>.[[Type]] is "ArrayBuffer", then set <var>value</var> to a
      new ArrayBuffer object in <var>targetRealm</var> whose [[ArrayBufferData]] internal slot value
      is <var>transferDataHolder</var>.[[ArrayBufferData]], and whose [[ArrayBufferByteLength]]
      internal slot value is <var>transferDataHolder</var>.[[ArrayBufferByteLength]].</p>

      <p class="note">In cases where the original memory occupied by [[ArrayBufferData]] is
      accessible during the deserialization, this step is unlikely to throw an exception, as no new
      memory needs to be allocated: the memory occupied by [[ArrayBufferData]] is instead just
      getting transferred into the new ArrayBuffer. This could be true, for example, when both the
      source and target realms are in the same process.</p>
     </li>

     <li>
      <p>Otherwise, if <var>transferDataHolder</var>.[[Type]] is "ResizableArrayBuffer", then set
      <var>value</var> to a new ArrayBuffer object in <var>targetRealm</var> whose
      [[ArrayBufferData]] internal slot value is <var>transferDataHolder</var>.[[ArrayBufferData]],
      whose [[ArrayBufferByteLength]] internal slot value is
      <var>transferDataHolder</var>.[[ArrayBufferByteLength]], and whose
      [[ArrayBufferMaxByteLength]] internal slot value is
      <var>transferDataHolder</var>.[[ArrayBufferMaxByteLength]].</p>

      <p class="note">For the same reason as the previous step, this step is also unlikely to throw
      an exception.</p>
     </li>

     <li>
      <p>Otherwise:</p>

      <ol>
       <li><p>Let <var>interfaceName</var> be <var>transferDataHolder</var>.[[Type]].</p></li>

       <li><p>If the interface identified by <var>interfaceName</var> is not exposed in
       <var>targetRealm</var>, then throw a <span>"<code>DataCloneError</code>"</span>
       <code>DOMException</code>.</p></li>

       <li><p>Set <var>value</var> to a new instance of the interface identified by
       <var>interfaceName</var>, created in <var>targetRealm</var>.</p></li>

       <li><p>Perform the appropriate <span>transfer-receiving steps</span> for the interface
       identified by <var>interfaceName</var> given <var>transferDataHolder</var> and
       <var>value</var>.</p></li>
      </ol>
     </li>

     <li><p><span data-x="map set">Set</span> <var>memory</var>[<var>transferDataHolder</var>] to
     <var>value</var>.</p></li>

     <li><p><span data-x="list append">Append</span> <var>value</var> to
     <var>transferredValues</var>.</p></li>
    </ol>
   </li>

   <li><p>Let <var>deserialized</var> be ?
   <span>StructuredDeserialize</span>(<var>serializeWithTransferResult</var>.[[Serialized]],
   <var>targetRealm</var>, <var>memory</var>).</p></li>

   <li><p>Return { [[Deserialized]]: <var>deserialized</var>, [[TransferredValues]]:
   <var>transferredValues</var> }.</p></li>
  </ol>


  <h4 id="performing-structured-clones-from-other-specifications">Performing serialization and
  transferring from other specifications</h4>

  <p>Other specifications may use the abstract operations defined here. The following provides
  some guidance on when each abstract operation is typically useful, with examples.</p>

  <dl>
   <dt><span>StructuredSerializeWithTransfer</span></dt>
   <dt><span>StructuredDeserializeWithTransfer</span></dt>
   <dd>
    <p>Cloning a value to another <span>realm</span>, with a transfer list, but where the target
    realm is not known ahead of time. In this case the serialization step can be performed
    immediately, with the deserialization step delayed until the target realm becomes known.</p>

    <p class="example"><code data-x="dom-MessagePort-postMessage">messagePort.postMessage()</code>
    uses this pair of abstract operations, as the destination realm is not known until the
    <code>MessagePort</code> <span>has been shipped</span>.</p>
   </dd>

   <dt><span>StructuredSerialize</span></dt>
   <dt><span>StructuredSerializeForStorage</span></dt>
   <dt><span>StructuredDeserialize</span></dt>
   <dd>
    <p>Creating a <span>realm</span>-independent snapshot of a given value which can be saved for an
    indefinite amount of time, and then reified back into a JavaScript value later, possibly
    multiple times.</p>

    <p><span>StructuredSerializeForStorage</span> can be used for situations where the serialization
    is anticipated to be stored in a persistent manner, instead of passed between realms. It throws
    when attempting to serialize <code>SharedArrayBuffer</code> objects, since storing shared memory
    does not make sense. Similarly, it can throw or possibly have different behavior when given a
    <span>platform object</span> with custom <span>serialization steps</span> when the
    <var>forStorage</var> argument is true.</p>

    <p class="example"><code data-x="dom-history-pushState">history.pushState()</code> and <code
    data-x="dom-history-replaceState">history.replaceState()</code> use
    <span>StructuredSerializeForStorage</span> on author-supplied state objects, storing them as
    <span>serialized state</span> in the appropriate <span>session history entry</span>. Then,
    <span>StructuredDeserialize</span> is used so that the <code
    data-x="dom-history-state">history.state</code> property can return a clone of the
    originally-supplied state object.</p>

    <p class="example"><code
    data-x="dom-BroadcastChannel-postMessage">broadcastChannel.postMessage()</code> uses
    <span>StructuredSerialize</span> on its input, then uses <span>StructuredDeserialize</span>
    multiple times on the result to produce a fresh clone for each destination being broadcast
    to. Note that transferring does not make sense in multi-destination situations.</p>

    <p class="example">Any API for persisting JavaScript values to the filesystem would also use
    <span>StructuredSerializeForStorage</span> on its input and <span>StructuredDeserialize</span>
    on its output.</p>
   </dd>
  </dl>

  <p>In general, call sites may pass in Web IDL values instead of JavaScript values; this is to be
  understood to perform an implicit <span data-x="concept-idl-convert">conversion</span> to the
  JavaScript value before invoking these algorithms.</p>

  <hr>

  <p>Call sites that are not invoked as a result of author code synchronously calling into a user
  agent method must take care to properly <span>prepare to run script</span> and <span>prepare to
  run a callback</span> before invoking <span>StructuredSerialize</span>,
  <span>StructuredSerializeForStorage</span>, or <span>StructuredSerializeWithTransfer</span>
  abstract operations, if they are being performed on arbitrary objects. This is necessary because
  the serialization process can invoke author-defined accessors as part of its final
  deep-serialization steps, and these accessors could call into operations that rely on the <span
  data-x="concept-entry-everything">entry</span> and <span
  data-x="concept-incumbent-everything">incumbent</span> concepts being properly set up.</p>

  <p class="example"><code data-x="dom-window-postMessage">window.postMessage()</code> performs
  <span>StructuredSerializeWithTransfer</span> on its arguments, but is careful to do so
  immediately, inside the synchronous portion of its algorithm. Thus it is able to use the
  algorithms without needing to <span>prepare to run script</span> and <span>prepare to run a
  callback</span>.</p>

  <p class="example">In contrast, a hypothetical API that used <span>StructuredSerialize</span> to
  serialize some author-supplied object periodically, directly from a <span
  data-x="concept-task">task</span> on the <span>event loop</span>, would need to ensure it performs
  the appropriate preparations beforehand. As of this time, we know of no such APIs on the platform;
  usually it is simpler to perform the serialization ahead of time, as a synchronous consequence of
  author code.</p>

  </div>

  <h4 id="structured-cloning">Structured cloning API</h4>

  <dl class="domintro">
   <dt><code data-x=""><var>result</var> = self.<span subdfn data-x="dom-structuredClone">structuredClone</span>(<var>value</var>[, { <span data-x="dom-StructuredSerializeOptions-transfer">transfer</span> }])</code></dt>
   <dd>
    <p>Takes the input value and returns a deep copy by performing the structured clone algorithm.
    <span>Transferable objects</span> listed in the <code
    data-x="dom-StructuredSerializeOptions-transfer">transfer</code> array are transferred, not
    just cloned, meaning that they are no longer usable in the input value.</p>

    <p>Throws a <span>"<code>DataCloneError</code>"</span> <code>DOMException</code> if any part of
    the input value is not <span data-x="serializable objects">serializable</span>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="WindowOrWorkerGlobalScope"
  data-x="dom-structuredClone"><code>structuredClone(<var>value</var>,
  <var>options</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>serialized</var> be ?
   <span>StructuredSerializeWithTransfer</span>(<var>value</var>, <var>options</var>["<code
   data-x="dom-StructuredSerializeOptions-transfer">transfer</code>"]).</p></li>

   <li><p>Let <var>deserializeRecord</var> be ?
   <span>StructuredDeserializeWithTransfer</span>(<var>serialized</var>, <span>this</span>'s
   <span data-x="concept-relevant-realm">relevant realm</span>).</p></li>

   <li><p>Return <var>deserializeRecord</var>.[[Deserialized]].</p></li>
  </ol>

  </div>


  <h2 split-filename="dom" id="dom">Semantics, structure, and APIs of HTML documents</h2>

  <h3>Documents</h3>

  <p>Every XML and HTML document in an HTML UA is represented by a <code>Document</code> object.
  <ref>DOM</ref></p>

  <p>The <code>Document</code> object's <dfn id="the-document's-address"
  data-x="concept-document-url"
  data-x-href="https://dom.spec.whatwg.org/#concept-document-url">URL</dfn> is defined in
  <cite>DOM</cite>. It is initially set when the <code>Document</code> object is created, but can
  change during the lifetime of the <code>Document</code> object; for example, it changes when the
  user <span data-x="navigate">navigates</span> to a <span data-x="navigate-fragid">fragment</span>
  on the page and when the <code data-x="dom-history-pushState">pushState()</code> method is called
  with a new <span>URL</span>. <ref>DOM</ref></p>
  <!--
    https://www.hixie.ch/tests/adhoc/dom/level0/history/pushState/002/
    https://www.hixie.ch/tests/adhoc/html/navigation/fragids/016.html
  -->

  <p class="warning">Interactive user agents typically expose the <code>Document</code> object's
  <span data-x="concept-document-url">URL</span> in their user interface. This is the primary
  mechanism by which a user can tell if a site is attempting to impersonate another.</p>

  <p>The <code>Document</code> object's <dfn data-x="concept-document-origin"
  data-x-href="https://dom.spec.whatwg.org/#concept-document-origin">origin</dfn> is defined in
  <cite>DOM</cite>. It is initially set when the <code>Document</code> object is created, and can
  change during the lifetime of the <code>Document</code> only upon setting <code
  data-x="dom-document-domain">document.domain</code>. A <code>Document</code>'s <span
  data-x="concept-document-origin">origin</span> can differ from the <span
  data-x="concept-url-origin">origin</span> of its <span data-x="concept-document-url">URL</span>;
  for example when a <span>child navigable</span> is <span data-x="create a new child
  navigable">created</span>, its <span data-x="nav-document">active document</span>'s <span
  data-x="concept-document-origin">origin</span> is inherited from its <span
  data-x="nav-parent">parent</span>'s <span data-x="nav-document">active document</span>'s <span
  data-x="concept-document-origin">origin</span>, even though its <span
  data-x="nav-document">active document</span>'s <span data-x="concept-document-url">URL</span> is
  <code>about:blank</code>. <ref>DOM</ref></p>

  <p>When a <code>Document</code> is created by a <span data-x="concept-script">script</span> using
  the <code data-x="dom-DOMImplementation-createDocument">createDocument()</code> or <code
  data-x="dom-DOMImplementation-createHTMLDocument">createHTMLDocument()</code> methods, the
  <code>Document</code> is <span>ready for post-load tasks</span> immediately.</p>

  <p><dfn export>The document's referrer</dfn> is a string (representing a <span>URL</span>) that
  can be set when the <code>Document</code> is created. If it is not explicitly set, then its value
  is the empty string.</p>


  <h4>The <code>Document</code> object</h4>

  <p><cite>DOM</cite> defines a <code data-x="DOM Document">Document</code> interface, which
  this specification extends significantly.</p>

  <pre><code class="idl">enum <dfn enum>DocumentReadyState</dfn> { "loading", "interactive", "complete" };
enum <dfn enum>DocumentVisibilityState</dfn> { "visible", "hidden" };
typedef (<span>HTMLScriptElement</span> or <span>SVGScriptElement</span>) <dfn typedef>HTMLOrSVGScriptElement</dfn>;

[<span>LegacyOverrideBuiltIns</span>]
partial interface <dfn id="document" data-lt="">Document</dfn> {
  static <code>Document</code> <span data-x="dom-parseHTMLUnsafe">parseHTMLUnsafe</span>((<code data-x="tt-trustedhtml">TrustedHTML</code> or DOMString) html);

  // <span>resource metadata management</span>
  [PutForwards=<span data-x="dom-location-href">href</span>, <span>LegacyUnforgeable</span>] readonly attribute <span>Location</span>? <span data-x="dom-document-location">location</span>;
  attribute USVString <span data-x="dom-document-domain">domain</span>;
  readonly attribute USVString <span data-x="dom-document-referrer">referrer</span>;
  attribute USVString <span data-x="dom-document-cookie">cookie</span>;
  readonly attribute DOMString <span data-x="dom-document-lastModified">lastModified</span>;
  readonly attribute <span>DocumentReadyState</span> <span data-x="dom-document-readyState">readyState</span>;

  // <span>DOM tree accessors</span>
  <a href="#dom-document-nameditem">getter</a> object (DOMString name);
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-document-title">title</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-document-dir">dir</span>;
  [<span>CEReactions</span>] attribute <span>HTMLElement</span>? <span data-x="dom-document-body">body</span>;
  readonly attribute <span>HTMLHeadElement</span>? <span data-x="dom-document-head">head</span>;
  [SameObject] readonly attribute <span>HTMLCollection</span> <span data-x="dom-document-images">images</span>;
  [SameObject] readonly attribute <span>HTMLCollection</span> <span data-x="dom-document-embeds">embeds</span>;
  [SameObject] readonly attribute <span>HTMLCollection</span> <span data-x="dom-document-plugins">plugins</span>;
  [SameObject] readonly attribute <span>HTMLCollection</span> <span data-x="dom-document-links">links</span>;
  [SameObject] readonly attribute <span>HTMLCollection</span> <span data-x="dom-document-forms">forms</span>;
  [SameObject] readonly attribute <span>HTMLCollection</span> <span data-x="dom-document-scripts">scripts</span>;
  NodeList <span data-x="dom-document-getElementsByName">getElementsByName</span>(DOMString elementName);
  readonly attribute <span>HTMLOrSVGScriptElement</span>? <span data-x="dom-document-currentScript">currentScript</span>; // classic scripts in a document tree only

  // <span>dynamic markup insertion</span>
  [<span>CEReactions</span>] <span>Document</span> <span data-x="dom-document-open">open</span>(optional DOMString unused1, optional DOMString unused2); // both arguments are <a href="#dom-document-open-unused-arguments">ignored</a>
  <span>WindowProxy</span>? <span data-x="dom-document-open-window">open</span>(USVString url, DOMString name, DOMString features);
  [<span>CEReactions</span>] undefined <span data-x="dom-document-close">close</span>();
  [<span>CEReactions</span>] undefined <span data-x="dom-document-write">write</span>((<code data-x="tt-trustedhtml">TrustedHTML</code> or DOMString)... text);
  [<span>CEReactions</span>] undefined <span data-x="dom-document-writeln">writeln</span>((<code data-x="tt-trustedhtml">TrustedHTML</code> or DOMString)... text);

  // <span>user interaction</span>
  readonly attribute <span>WindowProxy</span>? <span data-x="dom-document-defaultView">defaultView</span>;
  boolean <span data-x="dom-document-hasFocus">hasFocus</span>();
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-document-designMode">designMode</span>;
  [<span>CEReactions</span>] boolean <span data-x="dom-document-execCommand">execCommand</span>(DOMString commandId, optional boolean showUI = false, optional DOMString value = "");
  boolean <span data-x="dom-document-queryCommandEnabled">queryCommandEnabled</span>(DOMString commandId);
  boolean <span data-x="dom-document-queryCommandIndeterm">queryCommandIndeterm</span>(DOMString commandId);
  boolean <span data-x="dom-document-queryCommandState">queryCommandState</span>(DOMString commandId);
  boolean <span data-x="dom-document-queryCommandSupported">queryCommandSupported</span>(DOMString commandId);
  DOMString <span data-x="dom-document-queryCommandValue">queryCommandValue</span>(DOMString commandId);
  readonly attribute boolean <span data-x="dom-document-hidden">hidden</span>;
  readonly attribute <span>DocumentVisibilityState</span> <span data-x="dom-document-visibilityState">visibilityState</span>;

  // special <span>event handler IDL attributes</span> that only apply to Document objects
  [<span>LegacyLenientThis</span>] attribute <span>EventHandler</span> <span data-x="handler-onreadystatechange">onreadystatechange</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onvisibilitychange">onvisibilitychange</span>;

  // <a href="#Document-partial">also has obsolete members</a>
};
<span>Document</span> includes <span>GlobalEventHandlers</span>;</code></pre>

  <p>Each <code>Document</code> has a <dfn data-x="concept-document-policy-container" export
  for="Document">policy container</dfn> (a <span>policy container</span>), initially a new policy
  container, which contains policies which apply to the <code>Document</code>.</p>

  <p id="concept-document-feature-policy">Each <code>Document</code> has a <dfn
  data-x="concept-document-permissions-policy" export for="Document">permissions policy</dfn>, which
  is a <span data-x="concept-permissions-policy">permissions policy</span>, which is initially
  empty.</p>

  <p>Each <code>Document</code> has a <dfn data-x="concept-document-module-map">module map</dfn>,
  which is a <span>module map</span>, initially empty.</p>

  <p>Each <code>Document</code> has an <dfn data-x="concept-document-coop">opener policy</dfn>,
  which is an <span>opener policy</span>, initially a new opener policy.</p>

  <p>Each <code>Document</code> has an <dfn>is initial <code>about:blank</code></dfn>, which is a
  boolean, initially false.</p>

  <p>Each <code>Document</code> has a <dfn data-x="concept-document-navigation-id">during-loading
  navigation ID for WebDriver BiDi</dfn>, which is a <span>navigation ID</span> or null, initially
  null.</p>

  <p class="note">As the name indicates, this is used for interfacing with the <cite>WebDriver
  BiDi</cite> specification, which needs to be informed about certain occurrences during the early
  parts of the <code>Document</code>'s lifecycle, in a way that ties them to the original
  <span>navigation ID</span> used when the navigation that created this <code>Document</code> was
  the <span>ongoing navigation</span>. This eventually gets set back to null, after <cite>WebDriver
  BiDi</cite> considers the loading process to be finished. <ref>BIDI</ref></p>

  <p>Each <code>Document</code> has an <dfn data-x="concept-document-about-base-url">about base
  URL</dfn>, which is a <span>URL</span> or null, initially null.</p>

  <p class="note">This is only populated for "<code data-x="">about:</code>"-schemed
  <code>Document</code>s.</p>

  <p>Each <code>Document</code> has a <dfn
  data-x="concept-document-bfcache-blocking-details">bfcache blocking details</dfn>, which is a
  <span>set</span> of <span data-x="nrr-details-struct">not restored reason details</span>,
  initially empty.</p>

  <p>Each <code>Document</code> has an <dfn>open dialogs list</dfn>, which is a <span>list</span> of
  <code>dialog</code> elements, initially empty.</p>

  <h4>The <code>DocumentOrShadowRoot</code> interface</h4>

  <p><cite>DOM</cite> defines the <code data-x="DOM
  DocumentOrShadowRoot">DocumentOrShadowRoot</code> mixin, which this specification
  extends.</p>

  <pre><code class="idl">partial interface mixin <dfn data-lt="">DocumentOrShadowRoot</dfn> {
  readonly attribute <span>Element</span>? <span data-x="dom-documentorshadowroot-activeElement">activeElement</span>;
};</code></pre>

  <h4><dfn>Resource metadata management</dfn></h4>

  <dl class="domintro">
   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-referrer">referrer</span></code></dt>
   <dd>
    <p>Returns the <span data-x="concept-document-url">URL</span> of the <code>Document</code> from
    which the user navigated to this one, unless it was blocked or there was no such document, in
    which case it returns the empty string.</p>

    <p>The <code data-x="rel-noreferrer">noreferrer</code> link type can be used to block the
    referrer.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="Document"><code data-x="dom-document-referrer">referrer</code></dfn>
  attribute must return <span>the document's referrer</span>.</p>

  </div>

  <hr>

  <dl class="domintro">
   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-cookie">cookie</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the HTTP cookies that apply to the <code>Document</code>. If there are no cookies or
    cookies can't be applied to this resource, the empty string will be returned.</p>

    <p>Can be set, to add a new cookie to the element's set of HTTP cookies.</p>

    <p>If the contents are <span data-x="sandboxed origin browsing context flag">sandboxed into an
    opaque origin</span> (e.g., in an <code>iframe</code> with the <code
    data-x="attr-iframe-sandbox">sandbox</code> attribute), a
    <span>"<code>SecurityError</code>"</span> <code>DOMException</code> will be thrown on getting
    and setting.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="Document"><code data-x="dom-document-cookie">cookie</code></dfn>
  attribute represents the cookies of the resource identified by the document's <span
  data-x="concept-document-url">URL</span>.</p>

  <p>A <code>Document</code> object that falls into one of the following conditions is a
  <dfn>cookie-averse <code>Document</code> object</dfn>:</p>

  <ul>
   <li>A <code>Document</code> object whose <span data-x="concept-document-bc">browsing
   context</span> is null.</li>

   <li>A <code>Document</code> whose <span data-x="concept-document-url">URL</span>'s <span
   data-x="concept-url-scheme">scheme</span> is not an <span>HTTP(S) scheme</span>.</li>
  </ul>

  <!--Other specifications can also define <code>Document</code> objects as being <span
  data-x="cookie-averse Document object">cookie-averse <code>Document</code> objects</span>.-->

  <p id="sandboxCookies">
  <!--INSERT TRACKING-->
  On getting, if the document is a <span>cookie-averse <code>Document</code> object</span>, then the
  user agent must return the empty string. Otherwise, if the <code>Document</code>'s <span
  data-x="concept-document-origin">origin</span> is an <span data-x="concept-origin-opaque">opaque
  origin</span>, the user agent must throw a <span>"<code>SecurityError</code>"</span>
  <code>DOMException</code>. Otherwise, the user agent must return the <span>cookie-string</span>
  for the document's <span data-x="concept-document-url">URL</span> for a "non-HTTP" API, decoded
  using <span>UTF-8 decode without BOM</span>. <ref>COOKIES</ref></p>

  <p>On setting, if the document is a <span>cookie-averse <code>Document</code> object</span>, then
  the user agent must do nothing. Otherwise, if the <code>Document</code>'s <span
  data-x="concept-document-origin">origin</span> is an <span data-x="concept-origin-opaque">opaque
  origin</span>, the user agent must throw a <span>"<code>SecurityError</code>"</span>
  <code>DOMException</code>. Otherwise, the user agent must act as it would when <span
  data-x="receives a set-cookie-string">receiving a set-cookie-string</span> for the document's
  <span data-x="concept-document-url">URL</span> via a "non-HTTP" API, consisting of the new value
  <span data-x="utf-8 encode">encoded as UTF-8</span>.  <ref>COOKIES</ref> <ref>ENCODING</ref></p>

  <p class="note">Since the <code data-x="dom-document-cookie">cookie</code> attribute is accessible
  across frames, the path restrictions on cookies are only a tool to help manage which cookies are
  sent to which parts of the site, and are not in any way a security feature.</p>

  <p class="warning">The <code data-x="dom-document-cookie">cookie</code> attribute's getter and
  setter synchronously access shared state. Since there is no locking mechanism, other browsing
  contexts in a multiprocess user agent can modify cookies while scripts are running. A site could,
  for instance, try to read a cookie, increment its value, then write it back out, using the new
  value of the cookie as a unique identifier for the session; if the site does this twice in two
  different browser windows at the same time, it might end up using the same "unique" identifier for
  both sessions, with potentially disastrous effects.</p>

  <hr>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-lastmodified">lastModified</span></code></dt>
   <dd>
    <p>Returns the date of the last modification to the document, as reported by the server, in the
    form "<code data-x="">MM/DD/YYYY&nbsp;hh:mm:ss</code>", in the user's local time zone.</p>

    <p>If the last modification date is not known, the current time is returned instead.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="Document"><code
  data-x="dom-document-lastModified">lastModified</code></dfn> attribute, on getting, must return
  the date and time of the <code>Document</code>'s source file's last modification, in the user's
  local time zone, in the following format:</p>

  <ol>
   <li><p>The month component of the date.</p></li>

   <li><p>A U+002F SOLIDUS character (/).</p></li>

   <li><p>The day component of the date.</p></li>

   <li><p>A U+002F SOLIDUS character (/).</p></li>

   <li><p>The year component of the date.</p></li>

   <li><p>A U+0020 SPACE character.</p></li>

   <li><p>The hours component of the time.</p></li>

   <li><p>A U+003A COLON character (:).</p></li>

   <li><p>The minutes component of the time.</p></li>

   <li><p>A U+003A COLON character (:).</p></li>

   <li><p>The seconds component of the time.</p></li>
  </ol>

  <p>All the numeric components above, other than the year, must be given as two <span>ASCII
  digits</span> representing the number in base ten, zero-padded if necessary. The year must be
  given as the shortest possible string of four or more <span>ASCII digits</span> representing the
  number in base ten, zero-padded if necessary.</p>

  <p>The <code>Document</code>'s source file's last modification date and time must be derived from
  relevant features of the networking protocols used, e.g. from the value of the HTTP `<code
  data-x="http-last-modified">Last-Modified</code>` header of the document, or from metadata in the
  file system for local files. If the last modification date and time are not known, the attribute
  must return the current date and time in the above format.</p>

  </div>


  <h4>Reporting document loading status</h4>

  <dl class="domintro">
   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-readyState">readyState</span></code></dt>
   <dd>
    <p>Returns "<code data-x="">loading</code>" while the <code>Document</code> is loading, "<code
    data-x="">interactive</code>" once it is finished parsing but still loading subresources, and
    "<code data-x="">complete</code>" once it has loaded.</p>

    <p>The <code data-x="event-readystatechange">readystatechange</code> event fires on the
    <code>Document</code> object when this value changes.</p>

    <p><span w-dev subdfn data-x="stop parsing"></span>The <code
    data-x="event-DOMContentLoaded">DOMContentLoaded</code> event fires after the transition to
    "<code data-x="">interactive</code>" but before the transition to "<code
    data-x="">complete</code>", at the point where all subresources apart from <code
    data-x="attr-script-async">async</code> <code>script</code> elements have loaded.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>Each <code>Document</code> has a <dfn>current document readiness</dfn>, a string, initially
  "<code data-x="">complete</code>".</p>

  <p class="note">For <code>Document</code> objects created via the <span
  data-x="create-the-document-object">create and initialize a <code>Document</code> object</span>
  algorithm, this will be immediately reset to "<code data-x="">loading</code>" before any script
  can observe the value of <code data-x="dom-document-readyState">document.readyState</code>. This
  default applies to other cases such as <span data-x="is initial about:blank">initial
  <code>about:blank</code></span> <code>Document</code>s or <code>Document</code>s without a
  <span data-x="concept-document-bc">browsing context</span>.</p>

  <p>The <dfn attribute for="Document"><code
  data-x="dom-document-readyState">readyState</code></dfn> getter steps are to return
  <span>this</span>'s <span>current document readiness</span>.</p>

  <p>To <dfn>update the current document readiness</dfn> for <code>Document</code>
  <var>document</var> to <var>readinessValue</var>:</p>

  <ol>
   <li><p>If <var>document</var>'s <span>current document readiness</span> equals
   <var>readinessValue</var>, then return.</p></li>

   <li><p>Set <var>document</var>'s <span>current document readiness</span> to
   <var>readinessValue</var>.</p></li>

   <li>
    <p>If <var>document</var> is associated with an <span>HTML parser</span>, then:</p>

    <ol>
     <li><p>Let <var>now</var> be the <span>current high resolution time</span> given
     <var>document</var>'s <span>relevant global object</span>.</p></li>

     <li><p>If <var>readinessValue</var> is "<code data-x="">complete</code>", and
     <var>document</var>'s <span>load timing info</span>'s <span>DOM complete time</span> is 0, then
     set <var>document</var>'s <span>load timing info</span>'s <span>DOM complete time</span> to
     <var>now</var>.</p></li>

     <li><p>Otherwise, if <var>readinessValue</var> is "<code data-x="">interactive</code>", and
     <var>document</var>'s <span>load timing info</span>'s <span>DOM interactive time</span> is 0,
     then set <var>document</var>'s <span>load timing info</span>'s <span>DOM interactive
     time</span> to <var>now</var>.</p></li>
    </ol>
   </li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
   data-x="event-readystatechange">readystatechange</code> at <var>document</var>.</p></li>
  </ol>

  <hr>

  <p>A <code>Document</code> is said to have an <dfn>active parser</dfn> if it is associated with an
  <span>HTML parser</span> or an <span>XML parser</span> that has not yet been <span data-x="stop
  parsing">stopped</span> or <span data-x="abort a parser">aborted</span>.</p>

  <hr>

  <p>A <code>Document</code> has a <span>document load timing info</span> <dfn export
  for="Document">load timing info</dfn>.</p>

  <p>A <code>Document</code> has a <span>document unload timing info</span> <dfn export
  for="Document">previous document unload timing</dfn>.</p>

  <p>A <code>Document</code> has a boolean <dfn export>was created via cross-origin redirects</dfn>,
  initially false.</p>

  <p>The <dfn export>document load timing info</dfn> <span>struct</span> has the following <span
  data-x="struct item">items</span>:</p>

  <dl>
   <dt><dfn export for="document load timing info">navigation start time</dfn> (default 0)</dt>
   <dd>A number</dd>

   <dt><dfn export for="document load timing info">DOM interactive time</dfn> (default 0)</dt>
   <dt><dfn export for="document load timing info">DOM content loaded event start time</dfn>
   (default 0)</dt>
   <dt><dfn export for="document load timing info">DOM content loaded event end time</dfn> (default
   0)</dt>
   <dt><dfn export for="document load timing info">DOM complete time</dfn> (default 0)</dt>
   <dt><dfn export for="document load timing info">load event start time</dfn> (default 0)</dt>
   <dt><dfn export for="document load timing info">load event end time</dfn> (default 0)</dt>
   <dd><code>DOMHighResTimeStamp</code> values</dd>
  </dl>

  <p>The <dfn export>document unload timing info</dfn> <span>struct</span> has the following <span
  data-x="struct item">items</span>:</p>

  <dl>
   <dt><dfn export for="document unload timing info">unload event start time</dfn> (default 0)</dt>
   <dt><dfn export for="document unload timing info">unload event end time</dfn> (default 0)</dt>
   <dd><code>DOMHighResTimeStamp</code> values</dd>
  </dl>

  </div>

  <h4><dfn>Render-blocking mechanism</dfn></h4>

  <p>Each <code>Document</code> has a <dfn>render-blocking element set</dfn>, a <span>set</span> of
  elements, initially the empty set.</p>

  <p>A <code>Document</code> <var>document</var> <dfn>allows adding render-blocking elements</dfn>
  if <var>document</var>'s <span data-x="concept-document-content-type">content type</span> is
  "<code>text/html</code>" and <span>the body element</span> of <var>document</var> is null.</p>

  <p>A <code>Document</code> <var>document</var> is <dfn>render-blocked</dfn> if both of the
  following are true:</p>

  <ul>
   <li><p><var>document</var>'s <span>render-blocking element set</span> is non-empty, or
   <var>document</var> <span>allows adding render-blocking elements</span>.</p>

   <li><p>The <span>current high resolution time</span> given <var>document</var>'s <span>relevant
   global object</span> has not exceeded an <span>implementation-defined</span> timeout value.</p>
  </ul>

  <p>An element <var>el</var> is <dfn>render-blocking</dfn> if <var>el</var>'s
  <span>node document</span> <var>document</var> is <span>render-blocked</span>, and <var>el</var>
  is in <var>document</var>'s <span>render-blocking element set</span>.</p>

  <p>To <dfn>block rendering</dfn> on an element <var>el</var>:</p>

  <ol>
   <li><p>Let <var>document</var> be <var>el</var>'s <span>node document</span>.</p>

   <li><p>If <var>document</var> <span>allows adding render-blocking elements</span>, then
   <span data-x="set append">append</span> <var>el</var> to <var>document</var>'s
   <span>render-blocking element set</span>.</p>
  </ol>

  <p>To <dfn>unblock rendering</dfn> on an element <var>el</var>:</p>

  <ol>
   <li><p>Let <var>document</var> be <var>el</var>'s <span>node document</span>.</p>

   <li><p><span data-x="list remove">Remove</span> <var>el</var> from <var>document</var>'s
   <span>render-blocking element set</span>.</p>
  </ol>

  <p>Whenever a <span>render-blocking</span> element <var>el</var>
  <span>becomes browsing-context disconnected</span>, or <var>el</var>'s
  <span>blocking attribute</span>'s value is changed so that <var>el</var> is no longer
  <span>potentially render-blocking</span>, then <span>unblock rendering</span> on
  <var>el</var>.</p>

  <h4><dfn>DOM tree accessors</dfn></h4>

  <p><dfn>The <code>html</code> element</dfn> of a document is its <span>document element</span>,
  if it's an <code>html</code> element, and null otherwise.</p>

  <hr>

  <dl class="domintro">
   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-head">head</span></code></dt>
   <dd><p>Returns <span>the <code>head</code> element</span>.</p></dd>
  </dl>

  <p><dfn>The <code>head</code> element</dfn> of a document is the first <code>head</code> element
  that is a child of <span>the <code>html</code> element</span>, if there is one, or null
  otherwise.</p>

  <div w-nodev>

  <p>The <dfn attribute for="Document"><code data-x="dom-document-head">head</code></dfn> attribute,
  on getting, must return <span>the <code>head</code> element</span> of the document (a
  <code>head</code> element or null).</p>

  </div>

  <hr>

  <dl class="domintro">
   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-title">title</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the document's title, as given by <span>the <code>title</code> element</span> for
    HTML and as given by the <span>SVG <code>title</code></span> element for SVG.</p>

    <p>Can be set, to update the document's title. If there is no appropriate element to update, the
    new value is ignored.</p>
   </dd>
  </dl>

  <p><dfn>The <code>title</code> element</dfn> of a document is the first <code>title</code> element
  in the document (in <span>tree order</span>), if there is one, or null otherwise.</p>

  <div w-nodev>

  <p>The <dfn attribute for="Document" id="document.title"
  data-x="dom-document-title"><code>title</code></dfn> attribute must, on getting, run the following
  algorithm:</p>

  <ol>
   <li><p>If the <span>document element</span> is an <span>SVG <code>svg</code></span> element, then
   let <var>value</var> be the <span>child text content</span> of the first <span>SVG
   <code>title</code></span> element that is a child of the <span>document element</span>.</p></li>

   <li><p>Otherwise, let <var>value</var> be the <span>child text content</span> of <span>the
   <code>title</code> element</span>, or the empty string if <span>the <code>title</code>
   element</span> is null.</p></li>

   <li><p><span>Strip and collapse ASCII whitespace</span> in <var>value</var>.</p></li>

   <li><p>Return <var>value</var>.</p></li>
  </ol>

  <p>On setting, the steps corresponding to the first matching condition in the following list must
  be run:</p>

  <dl class="switch">
   <dt>If the <span>document element</span> is an <span>SVG <code>svg</code></span> element</dt>
   <dd>
    <ol>
     <li><p>If there is an <span>SVG <code>title</code></span> element that is a child of the
     <span>document element</span>, let <var>element</var> be the first such element.</p></li>

     <li>
      <p>Otherwise:</p>

      <ol>
       <li><p>Let <var>element</var> be the result of <span data-x="create an element">creating an
       element</span> given the <span>document element</span>'s <span>node document</span>, "<code
       data-x="">title</code>", and the <span>SVG namespace</span>.</p>

       <li><p>Insert <var>element</var> as the <span>first child</span> of the <span>document
       element</span>.</p></li>
      </ol>
     </li>

     <li><p><span>String replace all</span> with the given value within <var>element</var>.</p></li>
    </ol>
   </dd>

   <dt>If the <span>document element</span> is in the <span>HTML namespace</span></dt>
   <dd>
    <ol>
     <li><p>If <span>the <code>title</code> element</span> is null and <span>the <code>head</code>
     element</span> is null, then return.</p></li>

     <li><p>If <span>the <code>title</code> element</span> is non-null, let <var>element</var> be
     <span>the <code>title</code> element</span>.</p></li>

     <li>
      <p>Otherwise:</p>

      <ol>
       <li><p>Let <var>element</var> be the result of <span data-x="create an element">creating an
       element</span> given the <span>document element</span>'s <span>node document</span>, "<code
       data-x="">title</code>", and the <span>HTML namespace</span>.</p>

       <li><p><span data-x="concept-node-append">Append</span> <var>element</var> to <span>the
       <code>head</code> element</span>.</p></li>
      </ol>
     </li>

     <li><p><span>String replace all</span> with the given value within <var>element</var>.</p></li>
    </ol>
   </dd>

   <dt>Otherwise</dt>
   <dd>
    <p>Do nothing.</p>
   </dd>
  </dl>

  </div>

  <hr>

  <dl class="domintro">
   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-body">body</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns <span>the body element</span>.</p>

    <p>Can be set, to replace <span>the body element</span>.</p>

    <p>If the new value is not a <code>body</code> or <code>frameset</code> element, this will throw
    a <span>"<code>HierarchyRequestError</code>"</span> <code>DOMException</code>.</p>
   </dd>
  </dl>

  <p><dfn export>The body element</dfn> of a document is the first of <span>the <code>html</code>
  element</span>'s children that is either a <code>body</code> element or a <code>frameset</code>
  element, or null if there is no such element.</p>

  <div w-nodev>

  <p>The <dfn attribute for="Document"><code data-x="dom-document-body">body</code></dfn> attribute,
  on getting, must return <span>the body element</span> of the document (either a <code>body</code>
  element, a <code>frameset</code> element, or null). On setting, the following algorithm must be
  run:</p>

  <ol>
   <!-- if changes are requested:
    http://searchfox.org/mozilla-central/source/dom/html/nsHTMLDocument.cpp
    search for ::GetBody ::SetBody
    https://trac.webkit.org/browser/trunk/Source/WebCore/html/HTMLDocument.cpp
    search for ::setBody
    https://trac.webkit.org/browser/trunk/Source/WebCore/dom/Document.cpp
    search for ::body
   -->

   <li>If the new value is not a <code>body</code> or <code>frameset</code> element, then throw a
   <span>"<code>HierarchyRequestError</code>"</span> <code>DOMException</code>.</li>

   <li>Otherwise, if the new value is the same as <span>the body element</span>, return.</li>

   <li>Otherwise, if <span>the body element</span> is not null, then <span
   data-x="concept-node-replace">replace</span> <span>the body element</span> with the new value
   within <span>the body element</span>'s parent and return.</li>

   <li>Otherwise, if there is no <span>document element</span>, throw a
   <span>"<code>HierarchyRequestError</code>"</span> <code>DOMException</code>.</li>

   <li>Otherwise, <span>the body element</span> is null, but there's a
   <span>document element</span>. <span data-x="concept-node-append">Append</span> the new value to
   the <span>document element</span>.</li>
  </ol>

  <p class="note">The value returned by the <code data-x="dom-document-body">body</code> getter is
  not always the one passed to the setter.</p>

  <div class="example">
   <p>In this example, the setter successfully inserts a <code>body</code> element (though this is
   non-conforming since SVG does not allow a <code>body</code> as child of <span>SVG
   <code>svg</code></span>). However the getter will return null because the document element is not
   <code>html</code>.</p>

   <pre><code class="html">&lt;svg xmlns="http://www.w3.org/2000/svg">
 &lt;script>
  document.body = document.createElementNS("http://www.w3.org/1999/xhtml", "body");
  console.assert(document.body === null);
 &lt;/script>
&lt;/svg></code></pre>
  </div>

  </div>

  <hr>

  <dl class="domintro">
   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-images">images</span></code></dt>
   <dd>
    <p>Returns an <code>HTMLCollection</code> of the <code>img</code> elements in the
    <code>Document</code>.</p>
   </dd>

   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-embeds">embeds</span></code></dt>
   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-plugins">plugins</span></code></dt>
   <dd>
    <p>Returns an <code>HTMLCollection</code> of the <code>embed</code> elements in the
    <code>Document</code>.</p>
   </dd>

   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-links">links</span></code></dt>
   <dd>
    <p>Returns an <code>HTMLCollection</code> of the <code>a</code> and <code>area</code> elements
    in the <code>Document</code> that have <code data-x="attr-hyperlink-href">href</code>
    attributes.</p>
   </dd>

   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-forms">forms</span></code></dt>
   <dd>
    <p>Returns an <code>HTMLCollection</code> of the <code>form</code> elements in the
    <code>Document</code>.</p>
   </dd>

   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-scripts">scripts</span></code></dt>
   <dd>
    <p>Returns an <code>HTMLCollection</code> of the <code>script</code> elements in the
    <code>Document</code>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <!-- these all return the same object each time because of a rule in the collection section -->

  <p>The <dfn attribute for="Document"><code data-x="dom-document-images">images</code></dfn>
  attribute must return an <code>HTMLCollection</code> rooted at the <code>Document</code> node,
  whose filter matches only <code>img</code> elements.</p>

  <p>The <dfn attribute for="Document"><code data-x="dom-document-embeds">embeds</code></dfn>
  attribute must return an <code>HTMLCollection</code> rooted at the <code>Document</code> node,
  whose filter matches only <code>embed</code> elements.</p>

  <p>The <dfn attribute for="Document"><code data-x="dom-document-plugins">plugins</code></dfn>
  attribute must return the same object as that returned by the <code
  data-x="dom-document-embeds">embeds</code> attribute.</p>

  <p>The <dfn attribute for="Document"><code data-x="dom-document-links">links</code></dfn>
  attribute must return an <code>HTMLCollection</code> rooted at the <code>Document</code> node,
  whose filter matches only <code>a</code> elements with <code
  data-x="attr-hyperlink-href">href</code> attributes and <code>area</code> elements with <code
  data-x="attr-hyperlink-href">href</code> attributes.</p>

  <p>The <dfn attribute for="Document"><code data-x="dom-document-forms">forms</code></dfn>
  attribute must return an <code>HTMLCollection</code> rooted at the <code>Document</code> node,
  whose filter matches only <code>form</code> elements.</p>

  <p>The <dfn attribute for="Document"><code data-x="dom-document-scripts">scripts</code></dfn>
  attribute must return an <code>HTMLCollection</code> rooted at the <code>Document</code> node,
  whose filter matches only <code>script</code> elements.</p>

  <hr>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>collection</var> = <var>document</var>.<span subdfn data-x="dom-document-getElementsByName">getElementsByName</span>(<var>name</var>)</code></dt>
   <dd><p>Returns a <code>NodeList</code> of elements in the <code>Document</code> that have a <code
   data-x="">name</code> attribute with the value <var>name</var>.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="Document"><code
  data-x="dom-document-getElementsByName">getElementsByName(<var>elementName</var>)</code></dfn> method
  steps are to return a <span>live</span> <code>NodeList</code> containing all the <span>HTML
  elements</span> in that document that have a <code data-x="">name</code> attribute whose value is
  <span>identical to</span> the <var>elementName</var> argument, in <span>tree order</span>. When the
  method is invoked on a <code>Document</code> object again with the same argument, the user agent
  may return the same as the object returned by the earlier call. In other cases, a new
  <code>NodeList</code> object must be returned.</p>

  </div>

  <hr>

  <dl class="domintro">
   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-currentScript">currentScript</span></code></dt>
   <dd>
    <p>Returns the <code>script</code> element, or the <span>SVG <code>script</code></span> element,
    that is currently executing, as long as the element represents a <span>classic script</span>. In
    the case of reentrant script execution, returns the one that most recently started executing
    amongst those that have not yet finished executing.</p>

    <p>Returns null if the <code>Document</code> is not currently executing a <code>script</code> or
    <span>SVG <code>script</code></span> element (e.g., because the running script is an event
    handler, or a timeout), or if the currently executing <code>script</code> or <span>SVG
    <code>script</code></span> element represents a <span>module script</span>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="Document"><code
  data-x="dom-document-currentScript">currentScript</code></dfn> attribute, on getting, must return
  the value to which it was most recently set. When the <code>Document</code> is created, the <code
  data-x="dom-document-currentScript">currentScript</code> must be initialized to null.</p>

  </div>

  <p class="note">This API has fallen out of favor in the implementer and standards community, as
  it globally exposes <code>script</code> or <span>SVG <code>script</code></span> elements. As such,
  it is not available in newer contexts, such as when running <span data-x="module script">module
  scripts</span> or when running scripts in a <span>shadow tree</span>. We are looking into creating
  a new solution for identifying the running script in such contexts, which does not make it
  globally available: see <a href="https://github.com/whatwg/html/issues/1013">issue #1013</a>.</p>

  <div w-nodev>

  <hr>

  <p id="dom-document-namedItem-which">The <code>Document</code> interface <span data-x="support
  named properties">supports named properties</span>. The <span>supported property names</span> of a
  <code>Document</code> object <var>document</var> at any moment consist of the following, in
  <span>tree order</span> according to the element that contributed them, ignoring later duplicates,
  and with values from <code data-x="attr-id">id</code> attributes coming before values from <code
  data-x="">name</code> attributes when the same element contributes both:</p>

  <!-- KEEP THIS LIST IN SYNC WITH "NAMED ELEMENTS" DEFINITION BELOW -->
  <ul>
   <li><p>the value of the <code data-x="">name</code> content attribute for all
   <span>exposed</span> <code>embed</code>, <code>form</code>, <code>iframe</code>,
   <code>img</code>, and <span>exposed</span> <code>object</code> elements that have a non-empty
   <code data-x="">name</code> content attribute and are <span>in a document tree</span> with
   <var>document</var> as their <span>root</span>;</p></li>

   <li><p>the value of the <code data-x="attr-id">id</code> content attribute for all
   <span>exposed</span> <code>object</code> elements that have a non-empty
   <code data-x="attr-id">id</code> content attribute and are <span>in a document tree</span> with
   <var>document</var> as their <span>root</span>; and</p></li>

   <li><p>the value of the <code data-x="attr-id">id</code> content attribute for all
   <code>img</code> elements that have both a non-empty <code data-x="attr-id">id</code> content
   attribute and a non-empty <code data-x="">name</code> content attribute, and are <span>in a
   document tree</span> with <var>document</var> as their <span>root</span>.</p></li>
  </ul>

  <p id="dom-document-nameditem">To <span>determine the value of a named property</span>
  <var>name</var> for a <code>Document</code>, the user agent must return the value obtained using
  the following steps:</p>

  <ol>
   <li>
    <p>Let <var>elements</var> be the list of <span data-x="dom-document-namedItem-filter">named
    elements</span> with the name <var>name</var> that are <span>in a document tree</span> with the
    <code>Document</code> as their <span>root</span>.</p>

    <p class="note">There will be at least one such element, since the algorithm would otherwise not
    have been <span data-x="LegacyPlatformObjectGetOwnProperty">invoked by Web IDL</span>.</p>
   </li>

   <li><p>If <var>elements</var> has only one element, and that element is an <code>iframe</code>
   element, and that <code>iframe</code> element's <span>content navigable</span> is not null, then
   return the <span data-x="nav-wp">active <code>WindowProxy</code></span> of the element's
   <span>content navigable</span>.</p></li>

   <li>
    <p>Otherwise, if <var>elements</var> has only one element, return that element.</p>
   </li>

   <li>
    <p>Otherwise, return an <code>HTMLCollection</code> rooted at the <code>Document</code> node,
    whose filter matches only <span data-x="dom-document-namedItem-filter">named elements</span> with
    the name <var>name</var>.</p> <!-- the same one each time is returned, because of the
    rule under collections -->
   </li>

  <!--
  Note that this named getter overrides built-in properties, as in:
     https://software.hixie.ch/utilities/js/live-dom-viewer/?%3C!DOCTYPE%20html%3E%0D%0A...%3Ciframe%20name%3Dbody%3E%3C%2Fiframe%3E%3Cscript%3Ew(document.body)%3C%2Fscript%3E
  This is what the "LegacyOverrideBuiltIns" bit means in the IDL.
  -->
  </ol>

  <p><dfn data-x="dom-document-nameditem-filter">Named elements</dfn> with the name <var>name</var>, for the purposes of the above algorithm, are those that are either:</p>

  <!-- KEEP THIS LIST IN SYNC WITH SUPPORTED PROPERTY VALUES ABOVE -->
  <ul>
   <li><span>Exposed</span> <code>embed</code>, <code>form</code>, <code>iframe</code>,
   <code>img</code>, or <span>exposed</span> <code>object</code> elements that have a <code
   data-x="">name</code> content attribute whose value is <var>name</var>, or</li>

   <li><span>Exposed</span> <code>object</code> elements that have an <code
   data-x="attr-id">id</code> content attribute whose value is <var>name</var>, or</li>

   <li><code>img</code> elements that have an <code data-x="attr-id">id</code> content attribute
   whose value is <var>name</var>, and that have a non-empty <code data-x="">name</code>
   content attribute present also.</li>
  </ul>

  <p>An <code>embed</code> or <code>object</code> element is said to be <dfn>exposed</dfn> if it has
  no <span>exposed</span> <code>object</code> ancestor, and, for <code>object</code> elements, is
  additionally either not showing its <span>fallback content</span> or has no <code>object</code> or
  <code>embed</code> descendants.</p>

  </div>

  <hr>

  <p class="note">The <code data-x="dom-document-dir">dir</code> attribute on the
  <code>Document</code> interface is defined along with the <code data-x="attr-dir">dir</code>
  content attribute.</p>



  <h3>Elements</h3>

  <h4>Semantics</h4>

  <p>Elements, attributes, and attribute values in HTML are defined (by this specification) to have
  certain meanings (semantics). For example, the <code>ol</code> element represents an ordered list,
  and the <code data-x="attr-lang">lang</code> attribute represents the language of the content.</p>

  <p>These definitions allow HTML processors, such as web browsers or search engines, to present and
  use documents and applications in a wide variety of contexts that the author might not have
  considered.</p>

  <div class="example">

   <p>As a simple example, consider a web page written by an author who only considered desktop
   computer web browsers:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang="en">
 &lt;head>
  &lt;title>My Page&lt;/title>
 &lt;/head>
 &lt;body>
  &lt;h1>Welcome to my page&lt;/h1>
  &lt;p>I like cars and lorries and have a big Jeep!&lt;/p>
  &lt;h2>Where I live&lt;/h2>
  &lt;p>I live in a small hut on a mountain!&lt;/p>
 &lt;/body>
&lt;/html></code></pre>

   <p>Because HTML conveys <em>meaning</em>, rather than presentation, the same
   page can also be used by a small browser on a mobile phone, without any change to the page.
   Instead of headings being in large letters as on the desktop, for example, the browser on the
   mobile phone might use the same size text for the whole page, but with the headings in bold.</p>

   <p>But it goes further than just differences in screen size: the same page could equally be used
   by a blind user using a browser based around speech synthesis, which instead of displaying the
   page on a screen, reads the page to the user, e.g. using headphones. Instead of large text for
   the headings, the speech browser might use a different volume or a slower voice.</p>

   <p>That's not all, either. Since the browsers know which parts of the page are the headings, they
   can create a document outline that the user can use to quickly navigate around the document,
   using keys for "jump to next heading" or "jump to previous heading". Such features are especially
   common with speech browsers, where users would otherwise find quickly navigating a page quite
   difficult.</p>

   <p>Even beyond browsers, software can make use of this information. Search engines can use the
   headings to more effectively index a page, or to provide quick links to subsections of the page
   from their results. Tools can use the headings to create a table of contents (that is in fact how
   this very specification's table of contents is generated).</p>

   <p>This example has focused on headings, but the same principle applies to all of the semantics
   in HTML.</p>

  </div>

  <p>Authors must not use elements, attributes, or attribute values for purposes other than their
  appropriate intended semantic purpose, as doing so prevents software from correctly processing the
  page.</p>

  <div class="example">

   <p>For example, the following snippet, intended to represent the heading of a
   corporate site, is non-conforming because the second line is not intended to
   be a heading of a subsection, but merely a subheading or subtitle (a
   subordinate heading for the same section).</p>

   <pre class="bad"><code class="html">&lt;body&gt;
 &lt;h1&gt;ACME Corporation&lt;/h1&gt;
 &lt;h2&gt;The leaders in arbitrary fast delivery since 1920&lt;/h2&gt;
 ...</code></pre>

   <p>The <code>hgroup</code> element can be used for these kinds of situations:</p>

   <pre><code class="html">&lt;body&gt;
 &lt;hgroup&gt;
  &lt;h1&gt;ACME Corporation&lt;/h1&gt;
  &lt;p&gt;The leaders in arbitrary fast delivery since 1920&lt;/p&gt;
 &lt;/hgroup&gt;
 ...</code></pre>

  </div>

  <div class="example">

   <p>The document in this next example is similarly non-conforming, despite
   being syntactically correct, because the data placed in the cells is clearly
   not tabular data, and the <code>cite</code> element mis-used:</p>

   <pre class="bad" lang="en-GB"><code class="html">&lt;!DOCTYPE HTML&gt;
&lt;html lang="en-GB"&gt;
 &lt;head&gt; &lt;title&gt; Demonstration &lt;/title&gt; &lt;/head&gt;
 &lt;body&gt;
  &lt;table&gt;
   &lt;tr&gt; &lt;td&gt; <!--en-GB-->My favourite animal is the cat. &lt;/td&gt; &lt;/tr&gt;
   &lt;tr&gt;
    &lt;td&gt;
     &mdash;&lt;a href="https://example.org/~ernest/"&gt;&lt;cite&gt;Ernest&lt;/cite&gt;&lt;/a&gt;,
     in an essay from 1992
    &lt;/td&gt;
   &lt;/tr&gt;
  &lt;/table&gt;
 &lt;/body&gt;
&lt;/html&gt;</code></pre>

   <p>This would make software that relies on these semantics fail: for example,
   a speech browser that allowed a blind user to navigate tables in the document
   would report the quote above as a table, confusing the user; similarly, a
   tool that extracted titles of works from pages would extract "Ernest" as the
   title of a work, even though it's actually a person's name, not a title.</p>

   <p>A corrected version of this document might be:</p>

   <pre lang="en-GB"><code class="html">&lt;!DOCTYPE HTML&gt;
&lt;html lang="en-GB"&gt;
 &lt;head&gt; &lt;title&gt; Demonstration &lt;/title&gt; &lt;/head&gt;
 &lt;body&gt;
  &lt;blockquote&gt;
   &lt;p&gt; <!--en-GB-->My favourite animal is the cat. &lt;/p&gt;
  &lt;/blockquote&gt;
  &lt;p&gt;
   &mdash;&lt;a href="https://example.org/~ernest/"&gt;Ernest&lt;/a&gt;,
   in an essay from 1992
  &lt;/p&gt;
 &lt;/body&gt;
&lt;/html&gt;</code></pre>

  </div>

  <p>Authors must not use elements, attributes, or attribute values that are not permitted by this
  specification or <span>other applicable specifications</span>, as doing so makes it significantly
  harder for the language to be extended in the future.</p>

  <div class="example">

   <p>In the next example, there is a non-conforming attribute value ("carpet") and a non-conforming
   attribute ("texture"), which is not permitted by this specification:</p>

   <pre class="bad"><code class="html">&lt;label&gt;Carpet: &lt;input type="carpet" name="c" texture="deep pile"&gt;&lt;/label&gt;</code></pre>

   <p>Here would be an alternative and correct way to mark this up:</p>

   <pre><code class="html">&lt;label&gt;Carpet: &lt;input type="text" class="carpet" name="c" data-texture="deep pile"&gt;&lt;/label&gt;</code></pre>

  </div>

  <p id="no-browsing-context">DOM nodes whose <span>node document</span>'s <span
  data-x="concept-document-bc">browsing context</span> is null are exempt from all document
  conformance requirements other than the <a href="#writing">HTML syntax</a> requirements and <a
  href="#writing-xhtml-documents">XML syntax</a> requirements.</p>

  <div class="example">
   <p>In particular, the <code>template</code> element's <span>template contents</span>'s <span>node
   document</span>'s <span data-x="concept-document-bc">browsing context</span> is null. For
   example, the <span data-x="concept-element-content-model">content model</span> requirements and
   attribute value microsyntax requirements do not apply to a <code>template</code> element's
   <span>template contents</span>. In this example an <code>img</code> element has attribute values
   that are placeholders that would be invalid outside a <code>template</code> element.</p>

   <pre><code class="html">&lt;template>
 &lt;article>
  &lt;img <mark>src="{{src}}" alt="{{alt}}"</mark>>
  &lt;h1>&lt;/h1>
 &lt;/article>
&lt;/template></code></pre>

   <p>However, if the above markup were to omit the <code data-x="">&lt;/h1></code> end tag, that
   would be a violation of the <a href="#writing">HTML syntax</a>, and would thus be flagged as an
   error by conformance checkers.</p>
  </div>

  <p>Through scripting and using other mechanisms, the values of attributes, text, and indeed the
  entire structure of the document may change dynamically while a user agent is processing it. The
  semantics of a document at an instant in time are those represented by the state of the document
  at that instant in time, and the semantics of a document can therefore change over time. User
  agents <span w-nodev>must</span> update their presentation of the document as this
  occurs.</p>

  <p class="example">HTML has a <code>progress</code> element that describes a progress bar. If its
  "value" attribute is dynamically updated by a script, the UA would update the rendering to show
  the progress changing.</p>



  <h4>Elements in the DOM</h4>

  <p>The nodes representing <span>HTML elements</span> in the DOM <span w-nodev>must</span>
  implement, and expose to scripts, the interfaces listed for them in the relevant sections of this
  specification. This includes <span>HTML elements</span> in <span>XML documents</span>, even when
  those documents are in another context (e.g. inside an XSLT transform).</p>

  <p>Elements in the DOM <dfn data-x="represents">represent</dfn> things; that is, they have
  intrinsic <em>meaning</em>, also known as semantics.</p>

  <p class="example">For example, an <code>ol</code> element represents an ordered list.</p>

  <p>Elements can be <dfn data-x="referenced">referenced</dfn> (referred to) in some way, either
  explicitly or implicitly. One way that an element in the DOM can be explicitly referenced is by
  giving an <code data-x="attr-id">id</code> attribute to the element, and then creating a
  <span>hyperlink</span> with that <code data-x="attr-id">id</code> attribute's value as the <span
  data-x="navigate-fragid">fragment</span> for the <span>hyperlink</span>'s <code
  data-x="attr-hyperlink-href">href</code> attribute value. Hyperlinks are not necessary for a
  reference, however; any manner of referring to the element in question will suffice.</p>

  <div class="example">
   <p>Consider the following <code>figure</code> element, which is given an <code
   data-x="attr-id">id</code> attribute:</p>

   <pre><code class="html">&lt;figure id="module-script-graph">
  &lt;img src="module-script-graph.svg"
       alt="Module A depends on module B, which depends
            on modules C and D.">
  &lt;figcaption>Figure 27: a simple module graph&lt;/figcaption>
&lt;/figure></code></pre>

   <p>A <span>hyperlink</span>-based <span data-x="referenced">reference</span> could be created
   using the <code>a</code> element, like so:</p>

   <pre><code class="html">As we can see in &lt;a href="#module-script-graph">figure 27&lt;/a>, ...</code></pre>

   <p>However, there are many other ways of <span data-x="referenced">referencing</span> the
   <code>figure</code> element, such as:</p>

   <ul>
    <li><p>"As depicted in the figure of modules A, B, C, and D..."</p></li>

    <li><p>"In Figure 27..." (without a hyperlink)</p></li>

    <li><p>"From the contents of the 'simple module graph' figure..."</p></li>

    <li><p>"In the figure below..." (but <a href="#figure-note-about-references">this is
    discouraged</a>)</p></li>
   </ul>
  </div>

  <p>The basic interface, from which all the <span>HTML elements</span>' interfaces inherit, <span
  w-nodev>and which must be used by elements that have no additional requirements,</span> is
  the <code>HTMLElement</code> interface.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLElement</dfn> : <span>Element</span> {
  [<span>HTMLConstructor</span>] constructor();

  // metadata attributes
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-title">title</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-lang">lang</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-translate">translate</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-dir">dir</span>;

  // <span>user interaction</span>
  [<span>CEReactions</span>] attribute (boolean or unrestricted double or DOMString)? <span data-x="dom-hidden">hidden</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-inert">inert</span>;
  undefined <span data-x="dom-click">click</span>();
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-accessKey">accessKey</span>;
  readonly attribute DOMString <span data-x="dom-accessKeyLabel">accessKeyLabel</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-draggable">draggable</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-spellcheck">spellcheck</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-writingSuggestions">writingSuggestions</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-autocapitalize">autocapitalize</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-autocorrect">autocorrect</span>;

  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-innerText">innerText</span>;
  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-outerText">outerText</span>;

  <span>ElementInternals</span> <span data-x="dom-attachInternals">attachInternals</span>();

  // The popover API
  undefined <span data-x="dom-showPopover">showPopover</span>(optional <span>ShowPopoverOptions</span> options = {});
  undefined <span data-x="dom-hidePopover">hidePopover</span>();
  boolean <span data-x="dom-togglePopover">togglePopover</span>(optional (<span>TogglePopoverOptions</span> or boolean) options = {});
  [<span>CEReactions</span>] attribute DOMString? <span data-x="dom-popover">popover</span>;
};

dictionary <dfn dictionary>ShowPopoverOptions</dfn> {
  <span>HTMLElement</span> <dfn dict-member for="ShowPopoverOptions" data-x="dom-ShowPopoverOptions-source">source</dfn>;
};

dictionary <dfn dictionary>TogglePopoverOptions</dfn> : <span>ShowPopoverOptions</span> {
  boolean <dfn dict-member for="TogglePopoverOptions" data-x="dom-TogglePopoverOptions-force">force</dfn>;
};

<span>HTMLElement</span> includes <span>GlobalEventHandlers</span>;
<span>HTMLElement</span> includes <span>ElementContentEditable</span>;
<span>HTMLElement</span> includes <span>HTMLOrSVGElement</span>;

[Exposed=Window]
interface <dfn interface>HTMLUnknownElement</dfn> : <span>HTMLElement</span> {
  // Note: <a href="#customized-built-in-element-restrictions">intentionally</a> no [<span>HTMLConstructor</span>]
};</code></pre>

  <p>The <code>HTMLElement</code> interface holds methods and attributes related to a number of
  disparate features, and the members of this interface are therefore described in various different
  sections of this specification.</p>

  <div w-nodev>

  <hr>

  <p>The <span>element interface</span> for an element with name <var>name</var> in the
  <span>HTML namespace</span> is determined as follows:</p>

  <ol>
   <li><p>If <var>name</var> is <code>applet</code>, <code>bgsound</code>, <code>blink</code>,
   <code>isindex</code>, <code>keygen</code>, <code>multicol</code>, <code>nextid</code>, or
   <code>spacer</code>, then return <code>HTMLUnknownElement</code>.</p></li>

   <li><p>If <var>name</var> is <code>acronym</code>, <code>basefont</code>, <code>big</code>,
   <code>center</code>, <code>nobr</code>, <code>noembed</code>, <code>noframes</code>,
   <code>plaintext</code>, <code>rb</code>, <code>rtc</code>, <code>strike</code>, or
   <code>tt</code>, then return <code>HTMLElement</code>.</p></li>

   <li><p>If <var>name</var> is <code>listing</code> or <code>xmp</code>, then return
   <code>HTMLPreElement</code>.</p></li>

   <li><p>Otherwise, if this specification defines an interface appropriate for the <span>element
   type</span> corresponding to the local name <var>name</var>, then return that interface.</p></li>

   <li><p>If <span>other applicable specifications</span> define an appropriate interface for
   <var>name</var>, then return the interface they define.</p></li>

   <li><p>If <var>name</var> is a <span>valid custom element name</span>, then return
   <code>HTMLElement</code>.</p>

   <li><p>Return <code>HTMLUnknownElement</code>.</p></li>
  </ol>

  <p class="note">The use of <code>HTMLElement</code> instead of <code>HTMLUnknownElement</code> in
  the case of <span data-x="valid custom element name">valid custom element names</span> is done to
  ensure that any potential future <span data-x="custom-element-upgrades">upgrades</span> only cause
  a linear transition of the element's prototype chain, from <code>HTMLElement</code> to a subclass,
  instead of a lateral one, from <code>HTMLUnknownElement</code> to an unrelated subclass.</p>

  <p>Features shared between HTML and SVG elements use the <code>HTMLOrSVGElement</code> interface
  mixin: <ref>SVG</ref></p>

  <pre><code class="idl">interface mixin <dfn interface>HTMLOrSVGElement</dfn> {
  [SameObject] readonly attribute <span>DOMStringMap</span> <span data-x="dom-dataset">dataset</span>;
  attribute DOMString <span data-x="dom-HTMLOrSVGElement-nonce">nonce</span>; // <a href="#nonce-does-not-update-dom">intentionally no [CEReactions]</a>

  [<span>CEReactions</span>] attribute boolean <span data-x="dom-fe-autofocus">autofocus</span>;
  [<span>CEReactions</span>] attribute long <span data-x="dom-tabindex">tabIndex</span>;
  undefined <span data-x="dom-focus">focus</span>(optional <span>FocusOptions</span> options = {});
  undefined <span data-x="dom-blur">blur</span>();
};</code></pre>

  </div>

  <div class="example">
   <p>An example of an element that is neither an HTML nor SVG element is one created as
   follows:</p>

   <pre><code class="html">const el = document.createElementNS("some namespace", "example");
console.assert(el.constructor === Element);</code></pre>
  </div>

  <div w-nodev>

  <h4 id="html-element-constructors">HTML element constructors</h4>

  <p>To support the <a href="#custom-elements">custom elements</a> feature, all HTML elements have
  special constructor behavior. This is indicated via the <dfn extended-attribute
  data-lt="HTMLConstructor" data-x="HTMLConstructor"><code>[HTMLConstructor]</code></dfn> IDL
  <span>extended attribute</span>.  It indicates that the interface object for the given interface
  will have a specific behavior when called, as defined in detail below.</p>

  <p>The <code data-x="HTMLConstructor">[HTMLConstructor]</code> extended attribute must take no
  arguments, and must only appear on <span data-x="constructor operation">constructor
  operations</span>. It must appear only once on a constructor operation, and the interface must
  contain only the single, annotated constructor operation, and no others. The annotated
  constructor operation must be declared to take no arguments.</p>

  <p>Interfaces declared with constructor operations that are annotated with the <code
  data-x="HTMLConstructor">[HTMLConstructor]</code> extended attribute have the following
  <span>overridden constructor steps</span>:</p>

  <ol>
   <li><p>Let <var>registry</var> be <span>current global object</span>'s <span
   data-x="global-custom-element-registry">custom element registry</span>.</p></li>

   <li>
    <p>If <span>NewTarget</span> is equal to the <span>active function
    object</span>, then throw a <code>TypeError</code>.</p>

    <div class="example">
     <p>This can occur when a custom element is defined using an <span>element interface</span> as
     its constructor:</p>

     <pre><code class="js">customElements.define("bad-1", HTMLButtonElement);
new HTMLButtonElement();          // (1)
document.createElement("bad-1");  // (2)</code></pre>

     <p>In this case, during the execution of <code>HTMLButtonElement</code> (either explicitly, as
     in (1), or implicitly, as in (2)), both the <span>active function object</span> and
     <span>NewTarget</span> are <code>HTMLButtonElement</code>. If this check was not present, it
     would be possible to create an instance of <code>HTMLButtonElement</code> whose local name was
     <code data-x="">bad-1</code>.</p>
    </div>
   </li>

   <li>
    <p>Let <var>definition</var> be the item in <var>registry</var>'s <span>custom element
    definition set</span> with <span
    data-x="concept-custom-element-definition-constructor">constructor</span> equal to
    <span>NewTarget</span>. If there is no such item, then throw a <code>TypeError</code>.</p>

    <p class="note">Since there can be no item in <var>registry</var>'s <span>custom element
    definition set</span> with a <span
    data-x="concept-custom-element-definition-constructor">constructor</span> of undefined, this
    step also prevents HTML element constructors from being called as functions (since in that case
    <span>NewTarget</span> will be undefined).</p>
   </li>

   <li><p>Let <var>isValue</var> be null.</p></li>

   <li>
    <p>If <var>definition</var>'s <span data-x="concept-custom-element-definition-local-name">local
    name</span> is equal to <var>definition</var>'s <span
    data-x="concept-custom-element-definition-name">name</span> (i.e., <var>definition</var> is for
    an <span>autonomous custom element</span>):</p>

    <ol>
     <li>
      <p>If the <span>active function object</span> is not <code>HTMLElement</code>, then throw a
      <code>TypeError</code>.</p>

      <div class="example">
       <p>This can occur when a custom element is defined to not extend any local names, but
       inherits from a non-<code>HTMLElement</code> class:</p>

       <pre><code class="js">customElements.define("bad-2", class Bad2 extends HTMLParagraphElement {});</code></pre>

       <p>In this case, during the (implicit) <code data-x="">super()</code> call that occurs when
       constructing an instance of <code data-x="">Bad2</code>, the <span>active function
       object</span> is <code>HTMLParagraphElement</code>, not <code>HTMLElement</code>.</p>
      </div>
     </li>
    </ol>
   </li>

   <li>
    <p>Otherwise (i.e., if <var>definition</var> is for a <span>customized built-in
    element</span>):</p>

    <ol>
     <li><p>Let <var>valid local names</var> be the list of local names for elements defined in this
     specification or in <span>other applicable specifications</span> that use the <span>active
     function object</span> as their <span>element interface</span>.</p></li>

     <li>
      <p>If <var>valid local names</var> does not contain <var>definition</var>'s <span
      data-x="concept-custom-element-definition-local-name">local name</span>, then throw a
      <code>TypeError</code>.</p>

      <div class="example">
       <p>This can occur when a custom element is defined to extend a given local name but inherits
       from the wrong class:</p>

       <pre><code class="js">customElements.define("bad-3", class Bad3 extends HTMLQuoteElement {}, { extends: "p" });</code></pre>

       <p>In this case, during the (implicit) <code data-x="">super()</code> call that occurs when
       constructing an instance of <code data-x="">Bad3</code>, <var>valid local names</var> is the
       list containing <code>q</code> and <code>blockquote</code>, but <var>definition</var>'s <span
       data-x="concept-custom-element-definition-local-name">local name</span> is <code>p</code>,
       which is not in that list.</p>
      </div>
     </li>

     <li><p>Set <var>isValue</var> to <var>definition</var>'s <span
     data-x="concept-custom-element-definition-name">name</span>.</p></li>
    </ol>
   </li>

   <li>
    <p>If <var>definition</var>'s <span
    data-x="concept-custom-element-definition-construction-stack">construction stack</span> is
    empty:</p>

    <ol>
     <li><p>Let <var>element</var> be the result of <span data-x="internally create a new object
     implementing the interface">internally creating a new object implementing the interface</span>
     to which the <span>active function object</span> corresponds, given the <span>current
     realm</span> and <span>NewTarget</span>.</p></li>

     <li><p>Set <var>element</var>'s <span>node document</span> to the <span>current global
     object</span>'s <span data-x="concept-document-window">associated
     <code>Document</code></span>.</p></li>

     <li><p>Set <var>element</var>'s <span data-x="concept-element-namespace">namespace</span> to
     the <span>HTML namespace</span>.</p></li>

     <li><p>Set <var>element</var>'s <span data-x="concept-element-namespace-prefix">namespace
     prefix</span> to null.</p></li>

     <li><p>Set <var>element</var>'s <span data-x="concept-element-local-name">local name</span> to
     <var>definition</var>'s <span data-x="concept-custom-element-definition-local-name">local
     name</span>.</p></li>

     <li><p>Set <var>element</var>'s <span>custom element state</span> to "<code
     data-x="">custom</code>".</p></li>

     <li><p>Set <var>element</var>'s <span data-x="concept-element-custom-element-definition">custom
     element definition</span> to <var>definition</var>.</p></li>

     <li><p>Set <var>element</var>'s <span data-x="concept-element-is-value"><code
     data-x="">is</code> value</span> to <var>isValue</var>.</p></li>

     <li><p>Return <var>element</var>.</p></li>
    </ol>

    <p class="note">This occurs when author script constructs a new custom element directly, e.g.
    via <code data-x="">new MyCustomElement()</code>.</p>
   </li>

   <li><p>Let <var>prototype</var> be ? <span data-x="js-Get">Get</span>(<span>NewTarget</span>,
   "prototype").</p></li>

   <li>
    <p>If <var>prototype</var> <span data-x="js-Object">is not an Object</span>, then:</p>

    <ol>
     <li><p>Let <var>realm</var> be ? <span>GetFunctionRealm</span>(<span>NewTarget</span>).</p></li>

     <li><p>Set <var>prototype</var> to the <span>interface prototype object</span> of
     <var>realm</var> whose interface is the same as the interface of the <span>active function
     object</span>.</p></li>
    </ol>

    <p class="note">The realm of the <span>active function object</span> might not be
    <var>realm</var>, so we are using the more general concept of "the same interface" across
    realms; we are not looking for equality of <span data-x="interface object">interface
    objects</span>. This fallback behavior, including using the realm of <span>NewTarget</span> and
    looking up the appropriate prototype there, is designed to match analogous behavior for the
    JavaScript built-ins and Web IDL's <span>internally create a new object implementing the
    interface</span> algorithm.</p>
   </li>

   <li><p>Let <var>element</var> be the last entry in <var>definition</var>'s <span
   data-x="concept-custom-element-definition-construction-stack">construction stack</span>.</li>

   <li>
    <p>If <var>element</var> is an <span data-x="concept-already-constructed-marker"><i>already
    constructed</i> marker</span>, then throw a <code>TypeError</code>.</p>

    <div class="example">
     <p>This can occur when the author code inside the <span>custom element
     constructor</span> <a href="#custom-element-conformance">non-conformantly</a> creates another
     instance of the class being constructed, before calling <code
     data-x="">super()</code>:</p>

     <pre><code class="js">let doSillyThing = true;

class DontDoThis extends HTMLElement {
  constructor() {
    if (doSillyThing) {
      doSillyThing = false;
      new DontDoThis();
      // Now the construction stack will contain an <i>already constructed</i> marker.
    }

    // This will then fail with a TypeError:
    super();
  }
}</code></pre>
    </div>

    <div class="example">
     <p>This can also occur when author code inside the <span>custom element constructor</span> <a
     href="#custom-element-conformance">non-conformantly</a> calls <code data-x="">super()</code>
     twice, since per the JavaScript specification, this actually executes the superclass
     constructor (i.e. this algorithm) twice, before throwing an error:

     <pre><code class="js">class DontDoThisEither extends HTMLElement {
  constructor() {
    super();

    // This will throw, but not until it has already called into the HTMLElement constructor
    super();
  }
}</code></pre>
    </div>
   </li>

   <li><p>Perform ? <var>element</var>.[[SetPrototypeOf]](<var>prototype</var>).</p></li>

   <li><p>Replace the last entry in <var>definition</var>'s <span
   data-x="concept-custom-element-definition-construction-stack">construction stack</span> with an
   <span data-x="concept-already-constructed-marker"><i>already constructed</i>
   marker</span>.</p></li>

   <li>
    <p>Return <var>element</var>.</p>

    <p class="note">This step is normally reached when <span
    data-x="custom-element-upgrades">upgrading</span> a custom element; the existing element is
    returned, so that the <code data-x="">super()</code> call inside the <span>custom element
    constructor</span> assigns that existing element to <b>this</b>.</p>
   </li>
  </ol>

  <hr>

  <p>In addition to the constructor behavior implied by <code
  data-x="HTMLConstructor">[HTMLConstructor]</code>, some elements also have <span data-x="named
  constructor">named constructors</span> (which are really factory functions with a modified <code
  data-x="">prototype</code> property).

  <div class="example">
   <p>Named constructors for HTML elements can also be used in an <code
   data-x="dom-ElementDefinitionOptions-extends">extends</code> clause when defining a <span>custom
   element constructor</span>:</p>

   <pre><code class="js">class AutoEmbiggenedImage extends Image {
  constructor(width, height) {
    super(width * 10, height * 10);
  }
}

customElements.define("auto-embiggened", AutoEmbiggenedImage, { extends: "img" });

const image = new AutoEmbiggenedImage(15, 20);
console.assert(image.width === 150);
console.assert(image.height === 200);</code></pre>
  </div>

  </div>



  <h4>Element definitions</h4>

  <p>Each element in this specification has a definition that includes the following
  information:</p>

  <dl>
   <dt><dfn data-x="concept-element-categories">Categories</dfn></dt>

   <dd><p>A list of <span data-x="content categories">categories</span> to which the element belongs.
   These are used when defining the <span>content models</span> for each element.</p></dd>


   <dt><dfn data-x="concept-element-contexts">Contexts in which this element can be used</dfn></dt>

   <dd>
    <p>A <em>non-normative</em> description of where the element can be used. This information is
    redundant with the content models of elements that allow this one as a child, and is provided
    only as a convenience.</p>

    <div class="note">
     <p>For simplicity, only the most specific expectations are listed.</p>

     <p>For example, all <span>phrasing content</span> is <span>flow content</span>. Thus, elements
     that are <span>phrasing content</span> will only be listed as "where <span>phrasing
     content</span> is expected", since this is the more-specific expectation. Anywhere that expects
     <span>flow content</span> also expects <span>phrasing content</span>, and thus also meets this
     expectation.</p>
    </div>
   </dd>


   <dt><dfn data-x="concept-element-content-model">Content model</dfn></dt>

   <dd><p>A normative description of what content must be included as children and descendants of
   the element.</p></dd>


   <dt><dfn data-x="concept-element-tag-omission">Tag omission in text/html</dfn></dt>

   <dd><p>A <em>non-normative</em> description of whether, in the <code>text/html</code> syntax, the
   <span data-x="syntax-start-tag">start</span> and <span data-x="syntax-end-tag">end</span> tags can
   be omitted. This information is redundant with the normative requirements given in the <span
   data-x="syntax-tag-omission">optional tags</span> section, and is provided in the element
   definitions only as a convenience.</p></dd>


   <dt><dfn data-x="concept-element-attributes">Content attributes</dfn></dt>

   <dd><p>A normative list of attributes that may be specified on the element (except where
   otherwise disallowed), along with non-normative descriptions of those attributes. (The content to
   the left of the dash is normative, the content to the right of the dash is not.)</p></dd>


   <dt><dfn
   data-x="concept-element-accessibility-considerations">Accessibility considerations</dfn></dt>

   <dd>
    <p>For authors: Conformance requirements for use of <cite>ARIA</cite> <code
    data-x="attr-aria-role">role</code> and <code data-x="attr-aria-*">aria-*</code> attributes are
    defined in <cite>ARIA in HTML</cite>. <ref>ARIA</ref> <ref>ARIAHTML</ref></p>

    <p>For implementers: User agent requirements for implementing accessibility API semantics are
    defined in <cite>HTML Accessibility API Mappings</cite>. <ref>HTMLAAM</ref></p>
   </dd>


   <dt><dfn data-x="concept-element-dom">DOM interface</dfn></dt>

   <dd><p>A normative definition of a DOM interface that such elements must implement.</p></dd>
  </dl>

  <p>This is then followed by a description of what the element <span>represents</span>, along with
  any additional normative conformance criteria that may apply to authors<span data-x=""
  w-nodev> and implementations</span>. Examples are sometimes also included.</p>


  <h5>Attributes</h5>

  <p id="attribute-text">An attribute value is a string. Except where otherwise specified,
  attribute values on <span>HTML elements</span> may be any string value, including the empty
  string, and there is no restriction on what text can be specified in such attribute values.</p>



  <h4><dfn>Content models</dfn></h4>

  <p>Each element defined in this specification has a content model: a description of the element's
  expected <span data-x="concept-html-contents">contents</span>. An <span data-x="HTML elements">HTML
  element</span> must have contents that match the requirements described in the element's content
  model. The <dfn data-x="concept-html-contents">contents</dfn> of an element are its children in the
  DOM.</p>

  <p><span>ASCII whitespace</span> is always allowed between elements. User agents represent these
  characters between elements in the source markup as <code>Text</code> nodes in the DOM.<!-- not a
  conf criteria since the parser now requires this --> Empty <code>Text</code> nodes and
  <code>Text</code> nodes consisting of just sequences of those characters are considered
  <dfn>inter-element whitespace</dfn>.</p>

  <p><span>Inter-element whitespace</span>, comment nodes, and processing instruction nodes must be
  ignored when establishing whether an element's contents match the element's content model or not,
  and must be ignored when following algorithms that define document and element semantics.</p>

  <p class="note">Thus, an element <var>A</var> is said to be <i>preceded or followed</i>
  by a second element <var>B</var> if <var>A</var> and <var>B</var> have
  the same parent node and there are no other element nodes or <code>Text</code> nodes (other than
  <span>inter-element whitespace</span>) between them. Similarly, a node is the <i>only child</i> of
  an element if that element contains no other nodes other than <span>inter-element
  whitespace</span>, comment nodes, and processing instruction nodes.</p>

  <p>Authors must not use <span>HTML elements</span> anywhere except where they are explicitly
  allowed, as defined for each element, or as explicitly required by other specifications. For XML
  compound documents, these contexts could be inside elements from other namespaces, if those
  elements are defined as providing the relevant contexts.</p>

  <p class="example"><cite>The Atom Syndication Format</cite> defines a <code
  data-x="">content</code> element. When its <code data-x="">type</code> attribute has the value
  <code data-x="">xhtml</code>, <cite>The Atom Syndication Format</cite> requires that it contain a
  single HTML <code>div</code> element. Thus, a <code>div</code> element is allowed in that context,
  even though this is not explicitly normatively stated by this specification. <ref>ATOM</ref></p>

  <p>In addition, <span>HTML elements</span> may be orphan nodes (i.e. without a parent node).</p>

  <div class="example">

   <p>For example, creating a <code>td</code> element and storing it in a global variable in a
   script is conforming, even though <code>td</code> elements are otherwise only supposed to be used
   inside <code>tr</code> elements.</p>

   <pre><code class="js">var data = {
  name: "Banana",
  cell: document.createElement('td'),
};</code></pre>

  </div>


  <h5>The "nothing" content model</h5>

  <p>When an element's content model is <dfn data-x="concept-content-nothing">nothing</dfn>, the
  element must contain no <code>Text</code> nodes (other than <span>inter-element whitespace</span>)
  and no element nodes.</p>

  <p class="note">Most HTML elements whose content model is "nothing" are also, for convenience,
  <span>void elements</span> (elements that have no <span data-x="syntax-end-tag">end tag</span> in
  the <a href="#syntax">HTML syntax</a>). However, these are entirely separate concepts.</p>


  <h5>Kinds of content</h5>

  <p>Each element in HTML falls into zero or more <dfn data-x="content categories">categories</dfn>
  that group elements with similar characteristics together. The following broad categories are used
  in this specification:</p>

  <ul class="brief">
   <li><span>Metadata content</span></li>
   <li><span>Flow content</span></li>
   <li><span>Sectioning content</span></li>
   <li><span>Heading content</span></li>
   <li><span>Phrasing content</span></li>
   <li><span>Embedded content</span></li>
   <li><span>Interactive content</span></li>
  </ul>

  <p class="note">Some elements also fall into other categories, which are defined in other parts of
  this specification.</p>

  <p>These categories are related as follows:</p>

  <p><iframe width="1000" height="288" src="/images/content-venn.svg"></iframe></p>

  <p>Sectioning content, heading content, phrasing content, embedded content, and interactive
  content are all types of flow content. Metadata is sometimes flow content. Metadata and
  interactive content are sometimes phrasing content. Embedded content is also a type of phrasing
  content, and sometimes is interactive content.</p>

  <p>Other categories are also used for specific purposes, e.g. form controls are specified using a
  number of categories to define common requirements. Some elements have unique requirements and do
  not fit into any particular category.</p>


  <h6>Metadata content</h6>

  <p><dfn export>Metadata content</dfn> is content that sets up the presentation or behavior of the
  rest of the content, or that sets up the relationship of the document with other documents, or
  that conveys other "out of band" information.</p>

  <!-- when updating this also update the category index -->
  <ul class="brief category-list">
   <li><code>base</code></li>
   <li><code>link</code></li>
   <li><code>meta</code></li>
   <li><code>noscript</code></li>
   <li><code>script</code></li>
   <li><code>style</code></li>
   <li><code>template</code></li>
   <li><code>title</code></li>
  </ul>

  <p>Elements from other namespaces whose semantics are primarily metadata-related (e.g. RDF) are
  also <span>metadata content</span>.</p>

  <div class="example">

   <p>Thus, in the XML serialization, one can use RDF, like this:</p>

   <pre><code class="html">&lt;html xmlns="http://www.w3.org/1999/xhtml"
      xmlns:r="http://www.w3.org/1999/02/22-rdf-syntax-ns#" xml:lang="en">
 &lt;head>
  &lt;title>Hedral's Home Page&lt;/title>
  &lt;r:RDF>
   &lt;Person xmlns="http://www.w3.org/2000/10/swap/pim/contact#"
           r:about="https://hedral.example.com/#">
    &lt;fullName>Cat Hedral&lt;/fullName>
    &lt;mailbox r:resource="mailto:hedral@damowmow.com"/>
    &lt;personalTitle>Sir&lt;/personalTitle>
   &lt;/Person>
  &lt;/r:RDF>
 &lt;/head>
 &lt;body>
  &lt;h1>My home page&lt;/h1>
  &lt;p>I like playing with string, I guess. Sister says squirrels are fun
  too so sometimes I follow her to play with them.&lt;/p>
 &lt;/body>
&lt;/html></code></pre>

   <p>This isn't possible in the HTML serialization, however.</p>

  </div>


  <h6>Flow content</h6>

  <p>Most elements that are used in the body of documents and applications are categorized as
  <dfn export>flow content</dfn>.</p>

  <!-- when updating this also update the category index -->
  <ul class="brief category-list">
   <li><code>a</code></li>
   <li><code>abbr</code></li>
   <li><code>address</code></li>
   <li><code>area</code> (if it is a descendant of a <code>map</code> element)</li>
   <li><code>article</code></li>
   <li><code>aside</code></li>
   <li><code>audio</code></li>
   <li><code>b</code></li>
   <li><code>bdi</code></li>
   <li><code>bdo</code></li>
   <li><code>blockquote</code></li>
   <li><code>br</code></li>
   <li><code>button</code></li>
   <li><code>canvas</code></li>
   <li><code>cite</code></li>
   <li><code>code</code></li>
   <li><code>data</code></li>
   <li><code>datalist</code></li>
   <li><code>del</code></li>
   <li><code>details</code></li>
   <li><code>dfn</code></li>
   <li><code>dialog</code></li>
   <li><code>div</code></li>
   <li><code>dl</code></li>
   <li><code>em</code></li>
   <li><code>embed</code></li>
   <li><code>fieldset</code></li>
   <li><code>figure</code></li>
   <li><code>footer</code></li>
   <li><code>form</code></li>
   <li><code>h1</code></li>
   <li><code>h2</code></li>
   <li><code>h3</code></li>
   <li><code>h4</code></li>
   <li><code>h5</code></li>
   <li><code>h6</code></li>
   <li><code>header</code></li>
   <li><code>hgroup</code></li>
   <li><code>hr</code></li>
   <li><code>i</code></li>
   <li><code>iframe</code></li>
   <li><code>img</code></li>
   <li><code>input</code></li>
   <li><code>ins</code></li>
   <li><code>kbd</code></li>
   <li><code>label</code></li>
   <li><code>link</code> (if it is <span>allowed in the body</span>)</li>
   <li><code>main</code> (if it is a <span>hierarchically correct <code>main</code> element</span>)</li>
   <li><code>map</code></li>
   <li><code>mark</code></li>
   <li><span>MathML <code>math</code></span></li>
   <li><code>menu</code></li>
   <li><code>meta</code> (if the <code data-x="attr-itemprop">itemprop</code> attribute is present)</li>
   <li><code>meter</code></li>
   <li><code>nav</code></li>
   <li><code>noscript</code></li>
   <li><code>object</code></li>
   <li><code>ol</code></li>
   <li><code>output</code></li>
   <li><code>p</code></li>
   <li><code>picture</code></li>
   <li><code>pre</code></li>
   <li><code>progress</code></li>
   <li><code>q</code></li>
   <li><code>ruby</code></li>
   <li><code>s</code></li>
   <li><code>samp</code></li>
   <li><code>script</code></li>
   <li><code>search</code></li>
   <li><code>section</code></li>
   <li><code>select</code></li>
   <li><code>slot</code></li>
   <li><code>small</code></li>
   <li><code>span</code></li>
   <li><code>strong</code></li>
   <li><code>sub</code></li>
   <li><code>sup</code></li>
   <li><span>SVG <code>svg</code></span></li>
   <li><code>table</code></li>
   <li><code>template</code></li>
   <li><code>textarea</code></li>
   <li><code>time</code></li>
   <li><code>u</code></li>
   <li><code>ul</code></li>
   <li><code>var</code></li>
   <li><code>video</code></li>
   <li><code>wbr</code></li>
   <li><span data-x="autonomous custom element">autonomous custom elements</span></li>
   <li><span data-x="text content">text</span></li>
  </ul>


  <h6>Sectioning content</h6>

  <p><dfn export>Sectioning content</dfn> is content that defines the scope of <code>header</code>
  and <code>footer</code> elements.</p>

  <!-- when updating this also update the category index -->
  <ul class="brief category-list">
   <li><code>article</code></li>
   <li><code>aside</code></li>
   <li><code>nav</code></li>
   <li><code>section</code></li>
  </ul>


  <h6>Heading content</h6>

  <p><dfn export>Heading content</dfn> defines the heading of a section (whether explicitly marked
  up using <span>sectioning content</span> elements, or implied by the heading content itself).</p>

  <!-- when updating this also update the category index -->
  <ul class="brief category-list">
   <li><code>h1</code></li>
   <li><code>h2</code></li>
   <li><code>h3</code></li>
   <li><code>h4</code></li>
   <li><code>h5</code></li>
   <li><code>h6</code></li>
   <li><code>hgroup</code> (if it has a descendant <code>h1</code> to <code>h6</code> element)</li>
  </ul>


  <h6>Phrasing content</h6>

  <p><dfn export>Phrasing content</dfn> is the text of the document, as well as elements that mark
  up that text at the intra-paragraph level. Runs of <span>phrasing content</span> form <span
  data-x="paragraph">paragraphs</span>.</p>

  <!-- when updating this also update the category index -->
  <ul class="brief category-list">
   <li><code>a</code></li>
   <li><code>abbr</code></li>
   <li><code>area</code> (if it is a descendant of a <code>map</code> element)</li>
   <li><code>audio</code></li>
   <li><code>b</code></li>
   <li><code>bdi</code></li>
   <li><code>bdo</code></li>
   <li><code>br</code></li>
   <li><code>button</code></li>
   <li><code>canvas</code></li>
   <li><code>cite</code></li>
   <li><code>code</code></li>
   <li><code>data</code></li>
   <li><code>datalist</code></li>
   <li><code>del</code></li>
   <li><code>dfn</code></li>
   <li><code>em</code></li>
   <li><code>embed</code></li>
   <li><code>i</code></li>
   <li><code>iframe</code></li>
   <li><code>img</code></li>
   <li><code>input</code></li>
   <li><code>ins</code></li>
   <li><code>kbd</code></li>
   <li><code>label</code></li>
   <li><code>link</code> (if it is <span>allowed in the body</span>)</li>
   <li><code>map</code></li>
   <li><code>mark</code></li>
   <li><span>MathML <code>math</code></span></li>
   <li><code>meta</code> (if the <code data-x="attr-itemprop">itemprop</code> attribute is present)</li>
   <li><code>meter</code></li>
   <li><code>noscript</code></li>
   <li><code>object</code></li>
   <li><code>output</code></li>
   <li><code>picture</code></li>
   <li><code>progress</code></li>
   <li><code>q</code></li>
   <li><code>ruby</code></li>
   <li><code>s</code></li>
   <li><code>samp</code></li>
   <li><code>script</code></li>
   <li><code>select</code></li>
   <li><code>slot</code></li>
   <li><code>small</code></li>
   <li><code>span</code></li>
   <li><code>strong</code></li>
   <li><code>sub</code></li>
   <li><code>sup</code></li>
   <li><span>SVG <code>svg</code></span></li>
   <li><code>template</code></li>
   <li><code>textarea</code></li>
   <li><code>time</code></li>
   <li><code>u</code></li>
   <li><code>var</code></li>
   <li><code>video</code></li>
   <li><code>wbr</code></li>
   <li><span data-x="autonomous custom element">autonomous custom elements</span></li>
   <li><span data-x="text content">text</span></li>
  </ul>

  <p class="note">Most elements that are categorized as phrasing content can only contain elements
  that are themselves categorized as phrasing content, not any flow content.</p>

  <p><dfn data-x="text content">Text</dfn>, in the context of content models, means either nothing,
  or <code>Text</code> nodes. <span data-x="text content">Text</span> is sometimes used as a content
  model on its own, but is also <span>phrasing content</span>, and can be <span>inter-element
  whitespace</span> (if the <code>Text</code> nodes are empty or contain just <span>ASCII
  whitespace</span>).</p>

  <p><code>Text</code> nodes and attribute values must consist of <span data-x="scalar value">scalar
  values</span>, excluding <span data-x="noncharacter">noncharacters</span>, and <span
  data-x="control">controls</span> other than <span>ASCII whitespace</span>.

  <!--<code>Text</code> nodes and attribute values may begin with an <i>isolated combining
  character</i>.--> <!-- commented out since nothing disallows it currently, so it's implicit;
  however, if we ever make charmod a normative reference, this will be needed to override it -->
  <!-- see also https://www.w3.org/Bugs/Public/show_bug.cgi?id=13502#c22 -->

  This specification includes extra constraints on the exact value of <code>Text</code> nodes and
  attribute values depending on their precise context.

  <!-- <ref>UNICODE</ref>-->

  </p>



  <h6>Embedded content</h6>

  <p><dfn id="embedded-content-category" export>Embedded content</dfn> is content that imports
  another resource into the document, or content from another vocabulary that
  is inserted into the document.</p>

  <!-- when updating this also update the category index -->
  <ul class="brief category-list">
   <li><code>audio</code></li>
   <li><code>canvas</code></li>
   <li><code>embed</code></li>
   <li><code>iframe</code></li>
   <li><code>img</code></li>
   <li><span>MathML <code>math</code></span></li>
   <li><code>object</code></li>
   <li><code>picture</code></li>
   <li><span>SVG <code>svg</code></span></li>
   <li><code>video</code></li>
  </ul>

  <p>Elements that are from namespaces other than the <span>HTML namespace</span> and that convey
  content but not metadata, are <span>embedded content</span> for the purposes of the content models
  defined in this specification. (For example, MathML or SVG.)</p>

  <p>Some embedded content elements can have <dfn>fallback content</dfn>: content that is to be used
  when the external resource cannot be used (e.g. because it is of an unsupported format). The
  element definitions state what the fallback is, if any.</p>
  <!-- we might not need to define fallback content at all. consider removing it if we don't end up
  using it anywhere (e.g. Selection) -->


  <h6>Interactive content</h6>

<!--
TESTS:
https://software.hixie.ch/utilities/js/live-dom-viewer/?%3C%21DOCTYPE%20html%3E%0A%3Cp%20tabindex%3D1%3Etest%20%3Ca%20href%3D%22%22%3E%20%3Cem%3Etest%3C/em%3E%20%3C/a%3E%0A%3Cscript%3E%0A%20function%20test%20%28e%29%20%7B%20w%28e.type%20+%20%27%20on%20%27%20+%20e.target.tagName%20+%20%27%20through%20%27%20+%20e.currentTarget.tagName%29%3B%20%7D%0A%20document.getElementsByTagName%28%27a%27%29%5B0%5D.addEventListener%28%27click%27%2C%20test%2C%20false%29%3B%0A%20document.getElementsByTagName%28%27a%27%29%5B0%5D.addEventListener%28%27DOMActivate%27%2C%20test%2C%20false%29%3B%0A%20document.getElementsByTagName%28%27p%27%29%5B0%5D.addEventListener%28%27click%27%2C%20test%2C%20false%29%3B%0A%20document.getElementsByTagName%28%27p%27%29%5B0%5D.addEventListener%28%27DOMActivate%27%2C%20test%2C%20false%29%3B%0A%3C/script%3E%0A
https://software.hixie.ch/utilities/js/live-dom-viewer/?%3C%21DOCTYPE%20HTML%3E%0A%3Ca%20href%3Dhttp%3A//google.com/%20target%3Da%3EA%3C/a%3E%3Ca%20href%3Dhttp%3A//yahoo.com/%20target%3Db%3EB%3C/a%3E%3Cbr%3E%0A%3Ciframe%20name%3Da%3E%3C/iframe%3E%3Ciframe%20name%3Db%3E%3C/iframe%3E%0A%3Cscript%3E%0A%20var%20a%20%3D%20document.getElementsByTagName%28%27a%27%29%5B0%5D%3B%0A%20var%20b%20%3D%20document.getElementsByTagName%28%27a%27%29%5B1%5D%3B%0A%20a.appendChild%28b%29%3B%0A%3C/script%3E
https://software.hixie.ch/utilities/js/live-dom-viewer/?%3C%21DOCTYPE%20HTML%3E%0A%3Cform%20action%3D%22http%3A//google.com/%22%20onsubmit%3D%22w%28%27onsubmit%27%29%22%3E%3Cem%3EA%3C/em%3E%3C/form%3E%0A%3Cscript%3E%0Adocument.getElementsByTagName%28%27form%27%29%5B0%5D.attachEvent%28%27onsubmit%27%2C%20function%20%28%29%20%7B%20w%28%27submit%20fired%27%29%20%7D%29%3B%0Adocument.getElementsByTagName%28%27form%27%29%5B0%5D.fireEvent%28%27onsubmit%27%29%3B%0A%3C/script%3E
https://software.hixie.ch/utilities/js/live-dom-viewer/?%3C%21DOCTYPE%20HTML%3E%0A%3Cform%20action%3D%22http%3A//google.com/%22%3EX%3C/form%3E%0A%3Cscript%3E%0Avar%20evt%20%3D%20document.createEvent%28%22Events%22%29%3B%0Aevt.initEvent%28%22submit%22%2C%20true%2C%20true%29%3B%0Adocument.getElementsByTagName%28%27form%27%29%5B0%5D.dispatchEvent%28evt%29%3B%0A%3C/script%3E
-->

  <p><dfn export>Interactive content</dfn> is content that is specifically intended for user
  interaction.</p>

  <!-- when updating this also update the category index -->
  <ul class="brief category-list">
   <li><code>a</code> (if the <code data-x="attr-hyperlink-href">href</code> attribute is present)</li>
   <li><code>audio</code> (if the <code data-x="attr-media-controls">controls</code> attribute is present)</li>
   <li><code>button</code></li>
   <li><code>details</code></li>
   <li><code>embed</code></li>
   <li><code>iframe</code></li>
   <li><code>img</code> (if the <code data-x="attr-hyperlink-usemap">usemap</code> attribute is present)</li>
   <li><code>input</code> (if the <code data-x="attr-input-type">type</code> attribute is <em>not</em> in the <span data-x="attr-input-type-hidden">Hidden</span> state)</li>
   <li><code>label</code></li>
   <li><code>select</code></li>
   <li><code>textarea</code></li>
   <li><code>video</code> (if the <code data-x="attr-media-controls">controls</code> attribute is present)</li>
  </ul>



  <h6 id="palpable-content">Palpable content</h6>

  <p>As a general rule, elements whose content model allows any <span>flow content</span> or
  <span>phrasing content</span> should have at least one node in its <span
  data-x="concept-html-contents">contents</span> that is <dfn export>palpable content</dfn> and
  that does not have the <code data-x="attr-hidden">hidden</code> attribute specified.</p>

  <p class="note"><span>Palpable content</span> makes an element non-empty by providing either
  some descendant non-empty <span data-x="text content">text</span>, or else something users can
  hear (<code>audio</code> elements) or view (<code>video</code>, <code>img</code>, or
  <code>canvas</code> elements) or otherwise interact with (for example, interactive form
  controls).</p>

  <p>This requirement is not a hard requirement, however, as there are many cases where an element
  can be empty legitimately, for example when it is used as a placeholder which will later be filled
  in by a script, or when the element is part of a template and would on most pages be filled in but
  on some pages is not relevant.</p>

  <p>Conformance checkers are encouraged to provide a mechanism for authors to find elements that
  fail to fulfill this requirement, as an authoring aid.</p>

  <p>The following elements are palpable content:</p>

  <!-- when updating this also update the category index -->
  <ul class="brief category-list">
   <li><code>a</code></li>
   <li><code>abbr</code></li>
   <li><code>address</code></li>
   <li><code>article</code></li>
   <li><code>aside</code></li>
   <li><code>audio</code> (if the <code data-x="attr-media-controls">controls</code> attribute is present)</li>
   <li><code>b</code></li>
   <li><code>bdi</code></li>
   <li><code>bdo</code></li>
   <li><code>blockquote</code></li>
   <li><code>button</code></li>
   <li><code>canvas</code></li>
   <li><code>cite</code></li>
   <li><code>code</code></li>
   <li><code>data</code></li>
   <li><code>del</code></li>
   <li><code>details</code></li>
   <li><code>dfn</code></li>
   <li><code>div</code></li>
   <li><code>dl</code> (if the element's children include at least one name-value group)</li>
   <li><code>em</code></li>
   <li><code>embed</code></li>
   <li><code>fieldset</code></li>
   <li><code>figure</code></li>
   <li><code>footer</code></li>
   <li><code>form</code></li>
   <li><code>h1</code></li>
   <li><code>h2</code></li>
   <li><code>h3</code></li>
   <li><code>h4</code></li>
   <li><code>h5</code></li>
   <li><code>h6</code></li>
   <li><code>header</code></li>
   <li><code>hgroup</code></li>
   <li><code>i</code></li>
   <li><code>iframe</code></li>
   <li><code>img</code></li>
   <li><code>input</code> (if the <code data-x="attr-input-type">type</code> attribute is <em>not</em> in the <span data-x="attr-input-type-hidden">Hidden</span> state)</li>
   <li><code>ins</code></li>
   <li><code>kbd</code></li>
   <li><code>label</code></li>
   <li><code>main</code></li>
   <li><code>map</code></li>
   <li><code>mark</code></li>
   <li><span>MathML <code>math</code></span></li>
   <li><code>menu</code> (if the element's children include at least one <code>li</code> element)</li>
   <li><code>meter</code></li>
   <li><code>nav</code></li>
   <li><code>object</code></li>
   <li><code>ol</code> (if the element's children include at least one <code>li</code> element)</li>
   <li><code>output</code></li>
   <li><code>p</code></li>
   <li><code>picture</code></li>
   <li><code>pre</code></li>
   <li><code>progress</code></li>
   <li><code>q</code></li>
   <li><code>ruby</code></li>
   <li><code>s</code></li>
   <li><code>samp</code></li>
   <li><code>search</code></li>
   <li><code>section</code></li>
   <li><code>select</code></li>
   <li><code>small</code></li>
   <li><code>span</code></li>
   <li><code>strong</code></li>
   <li><code>sub</code></li>
   <li><code>sup</code></li>
   <li><span>SVG <code>svg</code></span></li>
   <li><code>table</code></li>
   <li><code>textarea</code></li>
   <li><code>time</code></li>
   <li><code>u</code></li>
   <li><code>ul</code> (if the element's children include at least one <code>li</code> element)</li>
   <li><code>var</code></li>
   <li><code>video</code></li>
   <li><span data-x="autonomous custom element">autonomous custom elements</span></li>
   <li><span data-x="text content">text</span> that is not <span>inter-element whitespace</span></li>
  </ul>


  <h6>Script-supporting elements</h6>

  <p><dfn>Script-supporting elements</dfn> are those that do not <span
  data-x="represents">represent</span> anything themselves (i.e. they are not rendered), but are
  used to support scripts, e.g. to provide functionality for the user.</p>

  <p>The following elements are script-supporting elements:</p>

  <!-- when updating this also update the category index -->
  <ul class="brief category-list">
   <li><code>script</code></li>
   <li><code>template</code></li>
  </ul>



  <h5>Transparent content models</h5>

  <p>Some elements are described as <dfn>transparent</dfn>; they have "transparent" in the
  description of their content model. The content model of a <span>transparent</span> element is
  derived from the content model of its parent element: the elements required in the part of the
  content model that is "transparent" are the same elements as required in the part of the content
  model of the parent of the transparent element in which the transparent element finds itself.</p>

  <div class="example">

   <p>For instance, an <code>ins</code> element inside a <code>ruby</code> element cannot contain an
   <code>rt</code> element, because the part of the <code>ruby</code> element's content model that
   allows <code>ins</code> elements is the part that allows <span>phrasing content</span>, and the
   <code>rt</code> element is not <span>phrasing content</span>.</p>

  </div>

<!--(as far as I can tell, there's no way to violate this requirement
     without first violating the earlier requirement)
  <p>Furthermore, when a content model includes a part that is "transparent", those parts must not
  contain content that would not be conformant if all transparent elements in the tree were
  replaced, in their parent element, by the <span data-x="concept-html-contents">contents</span> in
  the "transparent" part of their content model, retaining order.</p>

  <div class="example">

   <p>Consider the following markup fragment:</p>

   <pre><code class="html">&lt;p>Hello &lt;a href="world.html">&lt;em>wonderful&lt;/em> world&lt;/a>!&lt;/p></code></pre>

   <p>Its DOM looks like the following:</p>

   <ul class="domTree"><li class="t1"><code>p</code><ul><li class="t3"><code>#text</code>: <span data-x="">Hello </span></li><li class="t1"><code>a</code> <span class="t2" data-x=""><code class="attribute name" data-x="attr-hyperlink-href">href</code>="<code class="attribute value" data-x="">world.html</code>"</span><ul><li class="t1"><code>em</code><ul><li class="t3"><code>#text</code>: <span data-x="">wonderful</span></li></ul></li><li class="t3"><code>#text</code>: <span data-x=""> world</span></li></ul></li><li class="t3"><code>#text</code>: <span data-x="">!</span></li></ul></li></ul>

   <p>The content model of the <code>a</code> element is <span>transparent</span>. To see if its
   contents are conforming, therefore, the element is replaced by its contents:</p>

   <ul class="domTree"><li class="t1"><code>p</code><ul><li class="t3"><code>#text</code>: <span data-x="">Hello </span></li><li class="t1"><code>em</code><ul><li class="t3"><code>#text</code>: <span data-x="">wonderful</span></li></ul></li><li class="t3"><code>#text</code>: <span data-x=""> world</span></li><li class="t3"><code>#text</code>: <span data-x="">!</span></li></ul></li></ul>

   <p>Since that is conforming, the contents of the <code>a</code> are conforming in the original
   fragment (there is no complication regarding the various parts of the <code>p</code> element's
   content model, since that element's content model has only one part).</p>

  </div>
-->

  <p class="note">In some cases, where transparent elements are nested in each other, the process
  has to be applied iteratively.</p>

  <div class="example">
   <p>Consider the following markup fragment:</p>

   <pre><code class="html">&lt;p>&lt;object>&lt;ins>&lt;map>&lt;a href="/">Apples&lt;/a>&lt;/map>&lt;/ins>&lt;/object>&lt;/p></code></pre>

   <p>To check whether "Apples" is allowed inside the <code>a</code> element, the content models are
   examined. The <code>a</code> element's content model is transparent, as is the <code>map</code>
   element's, as is the <code>ins</code> element's, as is the <code>object</code> element's. The
   <code>object</code> element is found in the <code>p</code> element, whose content model is
   <span>phrasing content</span>. Thus, "Apples" is allowed, as text is phrasing content.</p>
  </div>

  <p>When a transparent element has no parent, then the part of its content model that is
  "transparent" must instead be treated as accepting any <span>flow content</span>.</p>


  <h5>Paragraphs</h5>

  <p class="note">The term <span>paragraph</span> as defined in this section is used for more than
  just the definition of the <code>p</code> element. The <span>paragraph</span> concept defined here
  is used to describe how to interpret documents. The <code>p</code> element is merely one of
  several ways of marking up a <span>paragraph</span>.</p>

  <!-- note: the actual definition is below -->
  <p>A <dfn>paragraph</dfn> is typically a run of <span>phrasing content</span> that forms a block
  of text with one or more sentences that discuss a particular topic, as in typography, but can also
  be used for more general thematic grouping. For instance, an address is also a paragraph, as is a
  part of a form, a byline, or a stanza in a poem.</p>

  <div class="example">

   <p>In the following example, there are two paragraphs in a section. There is also a heading,
   which contains phrasing content that is not a paragraph. Note how the comments and
   <span>inter-element whitespace</span> do not form paragraphs.</p>

   <pre><code class="html">&lt;section>
  &lt;h2>Example of paragraphs&lt;/h2>
  This is the &lt;em>first&lt;/em> paragraph in this example.
  &lt;p>This is the second.&lt;/p>
  &lt;!-- This is not a paragraph. -->
&lt;/section></code></pre>

  </div>

  <p>Paragraphs in <span>flow content</span> are defined relative to what the document looks like
  without the <code>a</code>, <code>ins</code>, <code>del</code>, and <code>map</code> elements
  complicating matters, since those elements, with their hybrid content models, can straddle
  paragraph boundaries, as shown in the first two examples below.</p>

  <p class="note">Generally, having elements straddle paragraph boundaries is best avoided.
  Maintaining such markup can be difficult.</p>

  <div class="example">

   <p>The following example takes the markup from the earlier example and puts <code>ins</code> and
   <code>del</code> elements around some of the markup to show that the text was changed (though in
   this case, the changes admittedly don't make much sense). Notice how this example has exactly the
   same paragraphs as the previous one, despite the <code>ins</code> and <code>del</code> elements
   &mdash; the <code>ins</code> element straddles the heading and the first paragraph, and the
   <code>del</code> element straddles the boundary between the two paragraphs.</p>

   <pre><code class="html">&lt;section>
  &lt;ins>&lt;h2>Example of paragraphs&lt;/h2>
  This is the &lt;em>first&lt;/em> paragraph in&lt;/ins> this example&lt;del>.
  &lt;p>This is the second.&lt;/p>&lt;/del>
  &lt;!-- This is not a paragraph. -->
&lt;/section></code></pre>

  </div>

  <div w-nodev>

  <p>Let <var>view</var> be a view of the DOM that replaces all <code>a</code>,
  <code>ins</code>, <code>del</code>, and <code>map</code> elements in the document with their <span
  data-x="concept-html-contents">contents</span>. Then, in <var>view</var>, for each run
  of sibling <span>phrasing content</span> nodes uninterrupted by other types of content, in an
  element that accepts content other than <span>phrasing content</span> as well as <span>phrasing
  content</span>, let <var>first</var> be the first node of the run, and let <var>last</var> be the last node of the run. For each such run that consists of at least one
  node that is neither <span>embedded content</span> nor <span>inter-element whitespace</span>, a
  paragraph exists in the original DOM from immediately before <var>first</var> to
  immediately after <var>last</var>. (Paragraphs can thus span across <code>a</code>,
  <code>ins</code>, <code>del</code>, and <code>map</code> elements.)</p>

  <p>Conformance checkers may warn authors of cases where they have paragraphs that overlap each
  other (this can happen with <code>object</code>, <code>video</code>, <code>audio</code>, and
  <code>canvas</code> elements, and indirectly through elements in other namespaces that allow HTML
  to be further embedded therein, like <span>SVG <code>svg</code></span> or <span>MathML
  <code>math</code></span>).</p>
  <!-- example below -->

  </div>

  <p>A <span>paragraph</span> is also formed explicitly by <code>p</code> elements.</p>

  <p class="note">The <code>p</code> element can be used to wrap individual paragraphs when there
  would otherwise not be any content other than phrasing content to separate the paragraphs from
  each other.</p>

  <div class="example">

   <p>In the following example, the link spans half of the first paragraph, all of the heading
   separating the two paragraphs, and half of the second paragraph. It straddles the paragraphs and
   the heading.</p>

   <pre><code class="html">&lt;header>
 Welcome!
 &lt;a href="about.html">
  This is home of...
  &lt;h1>The Falcons!&lt;/h1>
  The Lockheed Martin multirole jet fighter aircraft!
 &lt;/a>
 This page discusses the F-16 Fighting Falcon's innermost secrets.
&lt;/header></code></pre>

   <p>Here is another way of marking this up, this time showing the paragraphs explicitly, and
   splitting the one link element into three:</p>

   <pre><code class="html">&lt;header>
 &lt;p>Welcome! &lt;a href="about.html">This is home of...&lt;/a>&lt;/p>
 &lt;h1>&lt;a href="about.html">The Falcons!&lt;/a>&lt;/h1>
 &lt;p>&lt;a href="about.html">The Lockheed Martin multirole jet
 fighter aircraft!&lt;/a> This page discusses the F-16 Fighting
 Falcon's innermost secrets.&lt;/p>
&lt;/header></code></pre>

  </div>

  <div class="example">

   <!-- I don't know if there's a better way to deal with this, but if there is, let me know... -->

   <p>It is possible for paragraphs to overlap when using certain elements that define fallback
   content. For example, in the following section:</p>

   <pre><code class="html">&lt;section>
 &lt;h2>My Cats&lt;/h2>
 You can play with my cat simulator.
 &lt;object data="cats.sim">
  To see the cat simulator, use one of the following links:
  &lt;ul>
   &lt;li>&lt;a href="cats.sim">Download simulator file&lt;/a>
   &lt;li>&lt;a href="https://sims.example.com/watch?v=LYds5xY4INU">Use online simulator&lt;/a>
  &lt;/ul>
  Alternatively, upgrade to the Mellblom Browser.
 &lt;/object>
 I'm quite proud of it.
&lt;/section></code></pre>

   <p>There are five paragraphs:</p>

   <ol class="brief">

    <li>The paragraph that says "You can play with my cat simulator. <i>object</i> I'm
    quite proud of it.", where <i>object</i> is the <code>object</code> element.</li>

    <li>The paragraph that says "To see the cat simulator, use one of the following links:".</li>

    <li>The paragraph that says "Download simulator file".</li>

    <li>The paragraph that says "Use online simulator".</li>

    <li>The paragraph that says "Alternatively, upgrade to the Mellblom Browser.".</li>
   </ol>

   <p>The first paragraph is overlapped by the other four. A user agent that supports the "cats.sim"
   resource will only show the first one, but a user agent that shows the fallback will confusingly
   show the first sentence of the first paragraph as if it was in the same paragraph as the second
   one, and will show the last paragraph as if it was at the start of the second sentence of the
   first paragraph.</p>

   <p>To avoid this confusion, explicit <code>p</code> elements can be used. For example:</p>

   <!-- My Fish -->
   <pre><code class="html">&lt;section>
 &lt;h2>My Cats&lt;/h2>
 &lt;p>You can play with my cat simulator.&lt;/p>
 &lt;object data="cats.sim">
  &lt;p>To see the cat simulator, use one of the following links:&lt;/p>
  &lt;ul>
   &lt;li>&lt;a href="cats.sim">Download simulator file&lt;/a>
   &lt;li>&lt;a href="https://sims.example.com/watch?v=LYds5xY4INU">Use online simulator&lt;/a>
  &lt;/ul>
  &lt;p>Alternatively, upgrade to the Mellblom Browser.&lt;/p>
 &lt;/object>
 &lt;p>I'm quite proud of it.&lt;/p>
&lt;/section></code></pre>

  </div>


  <h4><dfn>Global attributes</dfn></h4>

  <p>The following attributes are common to and may be specified on all <span>HTML
  elements</span><span w-nodev> (even those not defined in this specification)</span>:</p>

  <ul class="brief">
   <li><code data-x="attr-accesskey">accesskey</code></li>
   <li><code data-x="attr-autocapitalize">autocapitalize</code></li>
   <li><code data-x="attr-autocorrect">autocorrect</code></li>
   <li><code data-x="attr-fe-autofocus">autofocus</code></li>
   <li><code data-x="attr-contenteditable">contenteditable</code></li>
   <li><code data-x="attr-dir">dir</code></li>
   <li><code data-x="attr-draggable">draggable</code></li>
   <li><code data-x="attr-enterkeyhint">enterkeyhint</code></li>
   <li><code data-x="attr-hidden">hidden</code></li>
   <li><code data-x="attr-inert">inert</code></li>
   <li><code data-x="attr-inputmode">inputmode</code></li>
   <li><code data-x="attr-is">is</code></li>
   <li><code data-x="attr-itemid">itemid</code></li>
   <li><code data-x="attr-itemprop">itemprop</code></li>
   <li><code data-x="attr-itemref">itemref</code></li>
   <li><code data-x="attr-itemscope">itemscope</code></li>
   <li><code data-x="attr-itemtype">itemtype</code></li>
   <li><code data-x="attr-lang">lang</code></li>
   <li><code data-x="attr-nonce">nonce</code></li>
   <li><code data-x="attr-popover">popover</code></li>
   <li><code data-x="attr-spellcheck">spellcheck</code></li>
   <li><code data-x="attr-style">style</code></li>
   <li><code data-x="attr-tabindex">tabindex</code></li>
   <li><code data-x="attr-title">title</code></li>
   <li><code data-x="attr-translate">translate</code></li>
   <li><code data-x="attr-writingsuggestions">writingsuggestions</code></li>
  </ul>

  <div w-nodev>

  <p>These attributes are only defined by this specification as attributes for <span>HTML
  elements</span>. When this specification refers to elements having these attributes, elements from
  namespaces that are not defined as having these attributes must not be considered as being
  elements with these attributes.</p>

  <div class="example">

   <p>For example, in the following XML fragment, the "<code data-x="">bogus</code>" element does not
   have a <code data-x="attr-dir">dir</code> attribute as defined in this specification, despite
   having an attribute with the literal name "<code data-x="">dir</code>". Thus, <span>the
   directionality</span> of the inner-most <code>span</code> element is '<span
   data-x="concept-rtl">rtl</span>', inherited from the <code>div</code> element indirectly through
   the "<code data-x="">bogus</code>" element.</p>

   <pre class="bad"><code class="html">&lt;div xmlns="http://www.w3.org/1999/xhtml" dir="rtl">
 &lt;bogus xmlns="https://example.net/ns" dir="ltr">
  &lt;span xmlns="http://www.w3.org/1999/xhtml">
  &lt;/span>
 &lt;/bogus>
&lt;/div></code></pre>

  </div>

  </div>

  <hr>

  <p><cite>DOM</cite> defines the user agent requirements for the <dfn element-attr for="global"
  data-x="attr-class" id="classes"><code>class</code></dfn>, <dfn element-attr for="global"
  data-x="attr-id" id="the-id-attribute"><code>id</code></dfn>, and <dfn element-attr for="global"
  data-x="attr-slot"><code>slot</code></dfn> attributes for any element in any namespace.
  <ref>DOM</ref></p>

  <p>The <code data-x="attr-class">class</code>, <code data-x="attr-id">id</code>, and <code
  data-x="attr-slot">slot</code> attributes may be specified on all <span>HTML elements</span>.</p>

  <p>When specified on <span>HTML elements</span>, the <code data-x="attr-class">class</code>
  attribute must have a value that is a <span>set of space-separated tokens</span> representing the
  various classes that the element belongs to.</p>

  <div class="note">

  <p>Assigning classes to an element affects class matching in selectors in CSS, the <code
  data-x="dom-document-getElementsByClassName">getElementsByClassName()</code> method in the DOM,
  and other such features.</p>

  <p>There are no additional restrictions on the tokens authors can use in the <code
  data-x="attr-class">class</code> attribute, but authors are encouraged to use values that describe
  the nature of the content, rather than values that describe the desired presentation of the
  content.</p>

  </div>

  <p>When specified on <span>HTML elements</span>, the <code data-x="attr-id">id</code> attribute
  value must be unique amongst all the <span data-x="concept-id">IDs</span> in the element's
  <span>tree</span> and must contain at least one character. The value must not contain any
  <span>ASCII whitespace</span>.</p>

  <!-- ASCII whitespace is disallowed because space-separated lists of IDs otherwise would not be
  able to reach all valid IDs -->

  <div class="note">

  <p>The <code data-x="attr-id">id</code> attribute specifies its element's <span
  data-x="concept-id">unique identifier (ID)</span>.</p>

  <p>There are no other restrictions on what form an ID can take; in particular, IDs can consist of
  just digits, start with a digit, start with an underscore, consist of just punctuation, etc.</p>

  <p>An element's <span data-x="concept-id">unique identifier</span> can be used for a variety of
  purposes, most notably as a way to link to specific parts of a document using <span
  data-x="concept-url-fragment">fragments</span>, as a way to target an element when scripting, and
  as a way to style a specific element from CSS.</p>

  </div>

  <div w-nodev>

  <p>Identifiers are opaque strings. Particular meanings should not be derived from the value of the
  <code data-x="attr-id">id</code> attribute.</p>

  </div>

  <p>There are no conformance requirements for the <code data-x="attr-slot">slot</code> attribute
  specific to <span>HTML elements</span>.</p>

  <p class="note">The <code data-x="attr-slot">slot</code> attribute is used to <span>assign a
  slot</span> to an element: an element with a <code data-x="attr-slot">slot</code> attribute is
  <span data-x="assign a slot">assigned</span> to the <span data-x="concept-slot">slot</span>
  created by the <code>slot</code> element whose <span data-x="attr-slot-name">name</span>
  attribute's value matches that <code data-x="attr-slot">slot</code> attribute's value — but only
  if that <code>slot</code> element finds itself in the <span>shadow tree</span> whose
  <span>root</span>'s <span data-x="concept-DocumentFragment-host">host</span> has the corresponding
  <code data-x="attr-slot">slot</code> attribute value.</p>

  <hr>

  <p>To enable assistive technology products to expose a more fine-grained interface than is
  otherwise possible with HTML elements and attributes, a set of <a href="#wai-aria">annotations
  for assistive technology products</a> can be specified (the ARIA <code
  data-x="attr-aria-role">role</code> and <code data-x="attr-aria-*">aria-*</code> attributes).
  <ref>ARIA</ref></p>

  <hr>

  <p>The following <span>event handler content attributes</span> may be specified on any <span
  data-x="HTML elements">HTML element</span>:</p>

  <ul class="brief">
   <li><code data-x="handler-onauxclick">onauxclick</code></li>
   <li><code data-x="handler-onbeforeinput">onbeforeinput</code></li>
   <li><code data-x="handler-onbeforematch">onbeforematch</code></li>
   <li><code data-x="handler-onbeforetoggle">onbeforetoggle</code></li>
   <li><code data-x="handler-onblur">onblur</code>*</li>
   <li><code data-x="handler-oncancel">oncancel</code></li>
   <li><code data-x="handler-oncanplay">oncanplay</code></li>
   <li><code data-x="handler-oncanplaythrough">oncanplaythrough</code></li>
   <li><code data-x="handler-onchange">onchange</code></li>
   <li><code data-x="handler-onclick">onclick</code></li>
   <li><code data-x="handler-onclose">onclose</code></li>
   <li><code data-x="handler-oncommand">oncommand</code></li>
   <li><code data-x="handler-oncontextlost">oncontextlost</code></li>
   <li><code data-x="handler-oncontextmenu">oncontextmenu</code></li>
   <li><code data-x="handler-oncontextrestored">oncontextrestored</code></li>
   <li><code data-x="handler-oncopy">oncopy</code></li>
   <li><code data-x="handler-oncuechange">oncuechange</code></li>
   <li><code data-x="handler-oncut">oncut</code></li>
   <li><code data-x="handler-ondblclick">ondblclick</code></li>
   <li><code data-x="handler-ondrag">ondrag</code></li>
   <li><code data-x="handler-ondragend">ondragend</code></li>
   <li><code data-x="handler-ondragenter">ondragenter</code></li>
   <li><code data-x="handler-ondragleave">ondragleave</code></li>
   <li><code data-x="handler-ondragover">ondragover</code></li>
   <li><code data-x="handler-ondragstart">ondragstart</code></li>
   <li><code data-x="handler-ondrop">ondrop</code></li>
   <li><code data-x="handler-ondurationchange">ondurationchange</code></li>
   <li><code data-x="handler-onemptied">onemptied</code></li>
   <li><code data-x="handler-onended">onended</code></li>
   <li><code data-x="handler-onerror">onerror</code>*</li>
   <li><code data-x="handler-onfocus">onfocus</code>*</li>
   <li><code data-x="handler-onformdata">onformdata</code></li>
   <li><code data-x="handler-oninput">oninput</code></li>
   <li><code data-x="handler-oninvalid">oninvalid</code></li>
   <li><code data-x="handler-onkeydown">onkeydown</code></li>
   <li><code data-x="handler-onkeypress">onkeypress</code></li>
   <li><code data-x="handler-onkeyup">onkeyup</code></li>
   <li><code data-x="handler-onload">onload</code>*</li>
   <li><code data-x="handler-onloadeddata">onloadeddata</code></li>
   <li><code data-x="handler-onloadedmetadata">onloadedmetadata</code></li>
   <li><code data-x="handler-onloadstart">onloadstart</code></li>
   <li><code data-x="handler-onmousedown">onmousedown</code></li>
   <li><code data-x="handler-onmouseenter">onmouseenter</code></li>
   <li><code data-x="handler-onmouseleave">onmouseleave</code></li>
   <li><code data-x="handler-onmousemove">onmousemove</code></li>
   <li><code data-x="handler-onmouseout">onmouseout</code></li>
   <li><code data-x="handler-onmouseover">onmouseover</code></li>
   <li><code data-x="handler-onmouseup">onmouseup</code></li>
   <li><code data-x="handler-onpaste">onpaste</code></li>
   <li><code data-x="handler-onpause">onpause</code></li>
   <li><code data-x="handler-onplay">onplay</code></li>
   <li><code data-x="handler-onplaying">onplaying</code></li>
   <li><code data-x="handler-onprogress">onprogress</code></li>
   <li><code data-x="handler-onratechange">onratechange</code></li>
   <li><code data-x="handler-onreset">onreset</code></li>
   <li><code data-x="handler-onresize">onresize</code>*</li>
   <li><code data-x="handler-onscroll">onscroll</code>*</li>
   <li><code data-x="handler-onscrollend">onscrollend</code>*</li>
   <li><code data-x="handler-onsecuritypolicyviolation">onsecuritypolicyviolation</code></li>
   <li><code data-x="handler-onseeked">onseeked</code></li>
   <li><code data-x="handler-onseeking">onseeking</code></li>
   <li><code data-x="handler-onselect">onselect</code></li>
   <li><code data-x="handler-onslotchange">onslotchange</code></li>
   <li><code data-x="handler-onstalled">onstalled</code></li>
   <li><code data-x="handler-onsubmit">onsubmit</code></li>
   <li><code data-x="handler-onsuspend">onsuspend</code></li>
   <li><code data-x="handler-ontimeupdate">ontimeupdate</code></li>
   <li><code data-x="handler-ontoggle">ontoggle</code></li>
   <li><code data-x="handler-onvolumechange">onvolumechange</code></li>
   <li><code data-x="handler-onwaiting">onwaiting</code></li>
   <li><code data-x="handler-onwheel">onwheel</code></li>
  </ul>

  <p class="note">The attributes marked with an asterisk have a different meaning when specified on
  <code>body</code> elements as those elements expose <span>event handlers</span> of the
  <code>Window</code> object with the same names.</p>

  <p class="note">While these attributes apply to all elements, they are not useful on all elements.
  For example, only <span data-x="media element">media elements</span> will ever receive a <code
  data-x="event-media-volumechange">volumechange</code> event fired by the user agent.</p>

  <hr>

  <p><span data-x="custom data attribute">Custom data attributes</span> (e.g. <code
  data-x="">data-foldername</code> or <code data-x="">data-msgid</code>) can be specified on any
  <span data-x="HTML elements">HTML element</span>, to store custom data, state, annotations, and
  similar, specific to the page.</p>

  <hr>

  <p>In <span>HTML documents</span>, elements in the <span>HTML namespace</span> may have an <code
  data-x="">xmlns</code> attribute specified, if, and only if, it has the exact value "<code
  data-x="">http://www.w3.org/1999/xhtml</code>". This does not apply to <span>XML
  documents</span>.</p>

  <p class="note">In HTML, the <code data-x="">xmlns</code> attribute has absolutely no effect. It
  is basically a talisman. It is allowed merely to make migration to and from XML mildly easier.
  When parsed by an <span>HTML parser</span>, the attribute ends up in no namespace, not the "<code
  data-x="">http://www.w3.org/2000/xmlns/</code>" namespace like namespace declaration attributes in
  XML do.</p>

  <p class="note">In XML, an <code data-x="">xmlns</code> attribute is part of the namespace
  declaration mechanism, and an element cannot actually have an <code data-x="">xmlns</code>
  attribute in no namespace specified.</p>

  <hr>

  <p><cite>XML</cite> also allows the use of the <code data-x="attr-xml-space">xml:space</code>
  attribute in the <span>XML namespace</span> on any element in an <span data-x="XML documents">XML
  document</span>. This attribute has no effect on <span>HTML elements</span>, as the default
  behavior in HTML is to preserve whitespace. <ref>XML</ref></p>

  <p class="note">There is no way to serialize the <code data-x="attr-xml-space">xml:space</code>
  attribute on <span>HTML elements</span> in the <code>text/html</code> syntax.</p>



  <h5>The <code data-x="attr-title">title</code> attribute</h5>

  <p>The <dfn element-attr for="html-global"><code data-x="attr-title">title</code></dfn> attribute
  <span>represents</span> advisory information for the element, such as would be appropriate for a
  tooltip. On a link, this could be the title or a description of the target resource; on an image,
  it could be the image credit or a description of the image; on a paragraph, it could be a footnote
  or commentary on the text; on a citation, it could be further information about the source; on
  <span>interactive content</span>, it could be a label for, or instructions for, use of the
  element; and so forth.  The value is text.</p>

  <!-- search for title-warning if modifying this paragraph -->
  <p class="note">Relying on the <code data-x="attr-title">title</code> attribute is currently
  discouraged as many user agents do not expose the attribute in an accessible manner as required by
  this specification (e.g., requiring a pointing device such as a mouse to cause a tooltip to
  appear, which excludes keyboard-only users and touch-only users, such as anyone with a modern
  phone or tablet).</p>

  <p>If this attribute is omitted from an element, then it implies that the <code
  data-x="attr-title">title</code> attribute of the nearest ancestor <span data-x="HTML
  elements">HTML element</span> with a <code data-x="attr-title">title</code> attribute set is also
  relevant to this element. Setting the attribute overrides this, explicitly stating that the
  advisory information of any ancestors is not relevant to this element. Setting the attribute to
  the empty string indicates that the element has no advisory information.</p>

  <p>If the <code data-x="attr-title">title</code> attribute's value contains U+000A LINE FEED (LF)
  characters, the content is split into multiple lines. Each U+000A LINE FEED (LF) character
  represents a line break.</p>

  <div class="example">
   <p>Caution is advised with respect to the use of newlines in <code
   data-x="attr-title">title</code> attributes.</p>

   <p>For instance, the following snippet actually defines an abbreviation's expansion <em>with a
   line break in it</em>:</p>

   <pre class="bad"><code class="html">&lt;p>My logs show that there was some interest in &lt;abbr title="Hypertext
Transport Protocol">HTTP&lt;/abbr> today.&lt;/p></code></pre> <!-- DO NOT REWRAP THIS LINE -->
  </div>

  <p>Some elements, such as <code>link</code>, <code>abbr</code>, and <code>input</code>, define
  additional semantics for the <code data-x="attr-title">title</code> attribute beyond the semantics
  described above.</p> <!-- the other two are <code>style</code> and <code>dfn</code> -->

  <div w-nodev>

  <p>The <dfn>advisory information</dfn> of an element is the value that the following algorithm
  returns, with the algorithm being aborted once a value is returned. When the algorithm returns the
  empty string, then there is no advisory information.</p>

  <ol>
   <li><p>If the element has a <code data-x="attr-title">title</code> attribute, then return the
   result of running <span>normalize newlines</span> on its value.</p></li>

   <li><p>If the element has a parent element, then return the parent element's <span>advisory
   information</span>.</p></li>

   <li><p>Return the empty string.</p></li>
  </ol>

  <p>User agents should inform the user when elements have <span>advisory information</span>,
  otherwise the information would not be discoverable.</p>

  <hr>

  <p>The <dfn attribute for="HTMLElement"><code data-x="dom-title">title</code></dfn> IDL attribute
  must <span>reflect</span> the <code data-x="attr-title">title</code> content attribute.</p>

  </div>



  <h5>The <code data-x="attr-lang">lang</code> and <code data-x="attr-xml-lang">xml:lang</code>
  attributes</h5>

  <p>The <dfn element-attr for="html-global"><code data-x="attr-lang">lang</code></dfn> attribute
  (in no namespace) specifies the primary language for the element's contents and for any of the
  element's attributes that contain text. Its value must be a valid BCP 47 language tag, or the
  empty string. Setting the attribute to the empty string indicates that the primary language is
  unknown<!-- UA conformance requirements for this are below -->. <ref>BCP47</ref></p>

  <p>The <dfn data-x="attr-xml-lang" data-x-href="https://www.w3.org/TR/xml/#sec-lang-tag"><code
  data-x="">lang</code></dfn> attribute in the <span>XML namespace</span> is defined in XML.
  <ref>XML</ref></p>

  <p>If these attributes are omitted from an element, then the language of this element is the same
  as the language of its parent element, if any (except for <code>slot</code> elements in a
  <span>shadow tree</span>).</p>

  <p>The <code data-x="attr-lang">lang</code> attribute in no namespace may be used on any <span
  data-x="HTML elements">HTML element</span>.</p>

  <p>The <span data-x="attr-xml-lang"><code data-x="">lang</code> attribute in the <span>XML
  namespace</span></span> may be used on <span>HTML elements</span> in <span>XML documents</span>,
  as well as elements in other namespaces if the relevant specifications allow it (in particular,
  MathML and SVG allow <span data-x="attr-xml-lang"><code data-x="">lang</code> attributes in the
  <span>XML namespace</span></span> to be specified on their elements). If both the <code
  data-x="attr-lang">lang</code> attribute in no namespace and the <span data-x="attr-xml-lang"><code
  data-x="">lang</code> attribute in the <span>XML namespace</span></span> are specified on the same
  element, they must have exactly the same value when compared in an <span>ASCII
  case-insensitive</span> manner.</p>

  <p>Authors must not use the <span data-x="attr-xml-lang"><code data-x="">lang</code> attribute in
  the <span>XML namespace</span></span> on <span>HTML elements</span> in <span>HTML
  documents</span>. To ease migration to and from XML, authors may specify an attribute in no
  namespace with no prefix and with the literal localname "<code data-x="">xml:lang</code>" on
  <span>HTML elements</span> in <span>HTML documents</span>, but such attributes must only be
  specified if a <code data-x="attr-lang">lang</code> attribute in no namespace is also specified,
  and both attributes must have the same value when compared in an <span>ASCII
  case-insensitive</span> manner.</p>

  <p class="note">The attribute in no namespace with no prefix and with the literal localname "<code
  data-x="">xml:lang</code>" has no effect on language processing.</p>

  <div w-nodev>

  <hr>

  <p>To determine the <dfn export for="Node">language</dfn> of a node, user agents must use the
  first appropriate step in the following list:</p>

  <dl class="switch">
   <dt>If the node is an element that has a <span data-x="attr-xml-lang"><code data-x="">lang</code>
   attribute in the <span>XML namespace</span></span> set</dt>
   <dd><p>Use the value of that attribute.</p></dd>

   <dt>If the node is an <span data-x="HTML elements">HTML element</span> or an element in the
   <span>SVG namespace</span>, and it has a <code data-x="attr-lang">lang</code> in no namespace
   attribute set</dt>
   <dd><p>Use the value of that attribute.</p></dd>

   <dt>If the node's parent is a <span>shadow root</span></dt>
   <dd><p>Use the <span>language</span> of that <span>shadow root</span>'s <span
   data-x="concept-DocumentFragment-host">host</span>.</p></dd>

   <!-- this assumes that if there's an ancestor element, the parent must be an element -->
   <dt>If the node's <span>parent element</span> is not null</dt>
   <dd><p>Use the <span>language</span> of that <span>parent element</span>.</p></dd>

   <dt>Otherwise</dt>
   <dd><p>If there is a <span>pragma-set default language</span> set, then that is the language of
   the node. If there is no <span>pragma-set default language</span> set, then language information
   from a higher-level protocol (such as HTTP), if any, must be used as the final fallback language
   instead. In the absence of any such language information, and in cases where the higher-level
   protocol reports multiple languages, the language of the node is unknown, and the corresponding
   language tag is the empty string.</p></dd>
  </dl>

  <p>If the resulting value is not a recognized language tag, then it must be treated as an unknown
  language having the given language tag, distinct from all other languages. For the purposes of
  round-tripping or communicating with other services that expect language tags, user agents should
  pass unknown language tags through unmodified, and tagged as being BCP 47 language tags, so that
  subsequent services do not interpret the data as another type of language description.
  <ref>BCP47</ref></p>

  <p class="example">Thus, for instance, an element with <code data-x="">lang="xyzzy"</code> would be
  matched by the selector <code data-x="">:lang(xyzzy)</code> (e.g. in CSS), but it would not be
  matched by <code data-x="">:lang(abcde)</code>, even though both are equally invalid. Similarly, if
  a web browser and screen reader working in unison communicated about the language of the element,
  the browser would tell the screen reader that the language was "xyzzy", even if it knew it was
  invalid, just in case the screen reader actually supported a language with that tag after all.
  Even if the screen reader supported both BCP 47 and another syntax for encoding language names,
  and in that other syntax the string "xyzzy" was a way to denote the Belarusian language, it would
  be <em>incorrect</em> for the screen reader to then start treating text as Belarusian, because
  "xyzzy" is not how Belarusian is described in BCP 47 codes (BCP 47 uses the code "be" for
  Belarusian).</p>

  <p>If the resulting value is the empty string, then it must be interpreted as meaning that the
  language of the node is <dfn data-x="concept-explicitly-unknown">explicitly unknown</dfn>.</p>

  <hr>

  <p>User agents may use the element's language to determine proper processing or rendering (e.g. in
  the selection of appropriate fonts or pronunciations, for dictionary selection, or for the user
  interfaces of form controls such as date pickers). <!--User agents must not use the element's
  language to determine text directionality. (commented out because text directionality is a
  rendering-level concern.)--></p>

  <!-- Date pickers would use the language information for determining the language of months, for
  example. They should use the page's _locale_ to figure out the format itself, not the language.
  There's currently no way to determine the page's locale except by guessing from the charset and
  language... -->

  <hr>

  <p>The <dfn attribute for="HTMLElement"><code data-x="dom-lang">lang</code></dfn> IDL attribute
  must <span>reflect</span> the <code data-x="attr-lang">lang</code> content attribute in no
  namespace.</p>

  </div>



  <h5>The <code data-x="attr-translate">translate</code> attribute</h5>

  <!-- v2: a way to translate the declensions of the word(s), but keep the actual roots the same
           for examples, see https://www.w3.org/Bugs/Public/show_bug.cgi?id=17954 -->

  <p>The <dfn element-attr for="html-global"><code data-x="attr-translate">translate</code></dfn>
  attribute is used to specify whether an element's attribute values and the values of its
  <code>Text</code> node children are to be translated when the page is localized, or whether to
  leave them unchanged. It is an attribute is an <span>enumerated attribute</span> with the
  following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="html-global/translate"><code
     data-x="attr-translate-yes-keyword">yes</code></dfn>
     <td rowspan=2><dfn data-x="attr-translate-yes-state">yes</dfn>
     <td rowspan=2>Sets <span>translation mode</span> to <span>translate-enabled</span>.
    <tr>
     <td>(the empty string)
    <tr>
     <td><dfn attr-value for="html-global/translate"><code
     data-x="attr-translate-no-keyword">no</code></dfn>
     <td><dfn data-x="attr-translate-no-state">no</dfn>
     <td>Sets <span>translation mode</span> to <span>no-translate</span>.
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <dfn
  data-x="attr-translate-inherit-state">inherit</dfn> state.</p>

  <p>Each element (even non-HTML elements) has a <dfn>translation mode</dfn>, which is in either the
  <span>translate-enabled</span> state or the <span>no-translate</span> state. If an <span
  data-x="HTML elements">HTML element</span>'s <code data-x="attr-translate">translate</code>
  attribute is in the <span data-x="attr-translate-yes-state">yes</span> state, then the element's
  <span>translation mode</span> is in the <span>translate-enabled</span> state; otherwise, if the
  element's <code data-x="attr-translate">translate</code> attribute is in the <span
  data-x="attr-translate-no-state">no</span> state, then the element's <span>translation mode</span>
  is in the <span>no-translate</span> state. Otherwise, either the element's <code
  data-x="attr-translate">translate</code> attribute is in the <span
  data-x="attr-translate-inherit-state">inherit</span> state, or the element is not an <span
  data-x="HTML elements">HTML element</span> and thus does not have a <code
  data-x="attr-translate">translate</code> attribute; in either case, the element's
  <span>translation mode</span> is in the same state as its <span>parent element</span>'s, if any,
  or in the <span>translate-enabled</span> state, if the element's <span>parent element</span> is
  null.</p>

  <p>When an element is in the <dfn>translate-enabled</dfn> state, the element's <span>translatable
  attributes</span> and the values of its <code>Text</code> node children are to be translated when
  the page is localized.</p>

  <p>When an element is in the <dfn>no-translate</dfn> state, the element's attribute values and the
  values of its <code>Text</code> node children are to be left as-is when the page is localized,
  e.g. because the element contains a person's name or a name of a computer program.</p>

  <p>The following attributes are <dfn>translatable attributes</dfn>:</p>

  <ul class="brief">
   <li><code data-x="attr-th-abbr">abbr</code> on <code>th</code> elements</li>
   <li><code data-x="">alt</code> on <code data-x="attr-area-alt">area</code>,
                                     <code data-x="attr-img-alt">img</code>, and
                                     <code data-x="attr-input-alt">input</code> elements</li>
   <li><code data-x="attr-meta-content">content</code> on <code>meta</code> elements, if the <code data-x="attr-meta-name">name</code> attribute specifies a metadata name whose value is known to be translatable</li>
   <li><code data-x="attr-hyperlink-download">download</code> on <code>a</code> and
                                                                 <code>area</code> elements</li>
   <li><code data-x="">label</code> on <code data-x="attr-optgroup-label">optgroup</code>,
                                       <code data-x="attr-option-label">option</code>, and
                                       <code data-x="attr-track-label">track</code> elements</li>
   <li><code data-x="attr-lang">lang</code> on <span>HTML elements</span>; must be "translated" to match the language used in the translation</li>
   <li><code data-x="">placeholder</code> on <code data-x="attr-input-placeholder">input</code> and
                                             <code data-x="attr-textarea-placeholder">textarea</code> elements</li>
   <li><code data-x="attr-iframe-srcdoc">srcdoc</code> on <code>iframe</code> elements; must be parsed and recursively processed</li>
   <li><code data-x="attr-style">style</code> on <span>HTML elements</span>; must be parsed and
   recursively processed (e.g. for the values of <span>'content'</span> properties)</li>
   <li><code data-x="attr-title">title</code> on all <span>HTML elements</span></li>
   <li><code data-x="attr-input-value">value</code> on <code>input</code> elements with a
       <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-button">Button</span> state
       or the <span data-x="attr-input-type-reset">Reset Button</span> state</li>
       <!-- <input value> on other types is excluded since we can't predict if it interferes with server-side processing -->
  </ul>

  <p>Other specifications may define other attributes that are also <span>translatable
  attributes</span>. For example, <cite>ARIA</cite> would define the <code
  data-x="attr-aria-label">aria-label</code> attribute as translatable.</p>

  <div w-nodev>

  <hr>

  <p>The <dfn attribute for="HTMLElement"><code data-x="dom-translate">translate</code></dfn> IDL
  attribute must, on getting, return true if the element's <span>translation mode</span> is
  <span>translate-enabled</span>, and false otherwise. On setting, it must set the content
  attribute's value to "<code data-x="">yes</code>" if the new value is true, and set the content
  attribute's value to "<code data-x="">no</code>" otherwise.</p>

  </div>

  <div class="example">

   <p>In this example, everything in the document is to be translated when the page is localized,
   except the sample keyboard input and sample program output:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang=en> &lt;!-- default on the document element is translate=yes -->
 &lt;head>
  &lt;title>The Bee Game&lt;/title> &lt;!-- implied translate=yes inherited from ancestors -->
 &lt;/head>
 &lt;body>
  &lt;p>The Bee Game is a text adventure game in English.&lt;/p>
  &lt;p>When the game launches, the first thing you should do is type
  &lt;kbd <strong>translate=no</strong>>eat honey&lt;/kbd>. The game will respond with:&lt;/p>
  &lt;pre>&lt;samp <strong>translate=no</strong>>Yum yum! That was some good honey!&lt;/samp>&lt;/pre>
 &lt;/body>
&lt;/html></code></pre>

  </div>



  <h5>The <code data-x="attr-dir">dir</code> attribute</h5>

  <p>The <dfn element-attr for="html-global"><code data-x="attr-dir">dir</code></dfn> attribute
  is an <span>enumerated attribute</span> with the following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="html-global/dir"><code data-x="attr-dir-ltr">ltr</code></dfn>
     <td><dfn data-x="attr-dir-ltr-state">ltr</dfn>
     <td>The contents of the element are explicitly directionally isolated left-to-right text.
    <tr>
     <td><dfn attr-value for="html-global/dir"><code data-x="attr-dir-rtl">rtl</code></dfn>
     <td><dfn data-x="attr-dir-rtl-state">rtl</dfn>
     <td>The contents of the element are explicitly directionally isolated right-to-left text.
    <tr>
     <td><dfn attr-value for="html-global/dir"><code data-x="attr-dir-auto">auto</code></dfn>
     <td><dfn data-x="attr-dir-auto-state">auto</dfn>
     <td>The contents of the element are explicitly directionally isolated text, but the direction
     is to be determined programmatically using the contents of the element (as described
     below).
  </table>

  <div class="note">
   <p>The heuristic used by the <span data-x="attr-dir-auto-state">auto</span> state is very crude
   (it just looks at the first character with a strong directionality, in a manner analogous to the
   Paragraph Level determination in the bidirectional algorithm). Authors are urged to only use this
   value as a last resort when the direction of the text is truly unknown and no better server-side
   heuristic can be applied. <ref>BIDI</ref></p>

   <p>For <code>textarea</code> and <code>pre</code> elements, the heuristic is applied on a
   per-paragraph level.</p>
  </div>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <dfn
  data-x="attr-dir-undefined-state">undefined</dfn> state.</p>

  <hr>

  <p>The <dfn data-x="the directionality">directionality</dfn> of an element (any element, not just
  an <span data-x="HTML elements">HTML element</span>) is either '<dfn
  data-x="concept-ltr">ltr</dfn>' or '<dfn data-x="concept-rtl">rtl</dfn>'. To compute the <span
  data-x="the directionality">directionality</span> given an element <var>element</var>, switch on
  <var>element</var>'s <code data-x="attr-dir">dir</code> attribute state:</p>

  <dl class="switch">
   <dt><span data-x="attr-dir-ltr-state">ltr</span></dt>
   <dd><p>Return '<span data-x="concept-ltr">ltr</span>'.</p></dd>

   <dt><span data-x="attr-dir-rtl-state">rtl</span></dt>
   <dd><p>Return '<span data-x="concept-rtl">rtl</span>'.</p></dd>

   <dt><span data-x="attr-dir-auto-state">auto</span></dt>
   <dd>
    <ol>
     <li><p>Let <var>result</var> be the <span>auto directionality</span> of
     <var>element</var>.</p></li>

     <li><p>If <var>result</var> is null, then return '<span
     data-x="concept-ltr">ltr</span>'.</p></li>

     <li><p>Return <var>result</var>.</p></li>
    </ol>
   </dd>

   <dt><span data-x="attr-dir-undefined-state">undefined</span></dt>
   <dd>
    <dl class="switch">
     <dt>If <var>element</var> is a <code>bdi</code> element</dt>
     <dd>
      <ol>
       <li><p>Let <var>result</var> be the <span>auto directionality</span> of
       <var>element</var>.</p></li>

       <li><p>If <var>result</var> is null, then return '<span
       data-x="concept-ltr">ltr</span>'.</p></li>

       <li><p>Return <var>result</var>.</p></li>
      </ol>
     </dd>

     <dt>If <var>element</var> is an <code>input</code> element whose <code
     data-x="attr-input-type">type</code> attribute is in the <span
     data-x="attr-input-type-tel">Telephone</span> state</dt>
     <dd><p>Return '<span data-x="concept-ltr">ltr</span>'.</p></dd>
     <!-- This looks odd, but is very much deliberate:
          https://www.w3.org/Bugs/Public/show_bug.cgi?id=17835 -->

     <dt>Otherwise</dt>
     <dd><p>Return the <span>parent directionality</span> of <var>element</var>.</p></dd>
    </dl>
   </dd>
  </dl>

  <p class="note">Since the <code data-x="attr-dir">dir</code> attribute is only defined for
  <span>HTML elements</span>, it cannot be present on elements from other namespaces. Thus, elements
  from other namespaces always end up using the <span>parent directionality</span>.</p>

  <p>The <dfn>auto-directionality form-associated elements</dfn> are:

  <ul>
   <li><p><code>input</code> elements whose <code data-x="attr-input-type">type</code> attribute is
   in the <span data-x="attr-input-type-hidden">Hidden</span>, <span
   data-x="attr-input-type-text">Text</span>, <span data-x="attr-input-type-search">Search</span>,
   <span data-x="attr-input-type-tel">Telephone</span>, <span
   data-x="attr-input-type-url">URL</span>, <span data-x="attr-input-type-email">Email</span>, <span
   data-x="attr-input-type-password">Password</span>, <span data-x="attr-input-type-submit">Submit
   Button</span>, <span data-x="attr-input-type-reset">Reset Button</span>, or <span
   data-x="attr-input-type-button">Button</span> state, and</p></li>

   <li><p><code>textarea</code> elements.</p></li>
  </ul>

  <p>To compute the <dfn>auto directionality</dfn> given an element <var>element</var>:</p>

  <ol>
   <li>
    <p>If <var>element</var> is an <span data-x="auto-directionality form-associated
    elements">auto-directionality form-associated element</span>:

    <ol>
     <li><p>If <var>element</var>'s <span data-x="concept-fe-value">value</span> contains a
     character of bidirectional character type AL or R, and there is no character of bidirectional
     character type L anywhere before it in the element's <span
     data-x="concept-fe-value">value</span>, then return '<span data-x="concept-rtl">rtl</span>'.
     <ref>BIDI</ref></p></li>

     <li><p>If <var>element</var>'s <span data-x="concept-fe-value">value</span> is not the empty
     string, then return '<span data-x="concept-ltr">ltr</span>'.</p></li>

     <li><p>Return null.</p></li>
    </ol>
   </li>

   <li>
    <p>If <var>element</var> is a <code>slot</code> element whose <span>root</span> is a
    <span>shadow root</span> and <var>element</var>'s <span>assigned nodes</span> are not empty:</p>

    <ol>
     <li>
      <p><span data-x="list iterate">For each</span> node <var>child</var> of <var>element</var>'s
      <span>assigned nodes</span>:</p>

      <ol>
       <li><p>Let <var>childDirection</var> be null.</p></li>

       <li><p>If <var>child</var> is a <code>Text</code> node, then set <var>childDirection</var> to
       the <span>text node directionality</span> of <var>child</var>.</p></li>

       <li>
        <p>Otherwise:</p>

        <ol>
         <li><p><span>Assert</span>: <var>child</var> is an <code>Element</code> node.</p></li>

         <li><p>Set <var>childDirection</var> to the <span>contained text auto directionality</span>
         of <var>child</var> with <i><span
         data-x="auto-directionality-can-exclude-root">canExcludeRoot</span></i> set to
         true.</p></li>
        </ol>
       </li>

       <li><p>If <var>childDirection</var> is not null, then return
       <var>childDirection</var>.</p></li>
      </ol>
     </li>

     <li><p>Return null.</p></li>
    </ol>
   </li>

   <li><p>Return the <span>contained text auto directionality</span> of <var>element</var> with
   <i><span data-x="auto-directionality-can-exclude-root">canExcludeRoot</span></i> set to
   false.</p></li>
  </ol>

  <p>To compute the <dfn>contained text auto directionality</dfn> of an element
  <var>element</var> with a boolean <dfn
  data-x="auto-directionality-can-exclude-root"><var>canExcludeRoot</var></dfn>:</p>

  <ol>
   <li>
    <p><span data-x="list iterate">For each</span> node <var>descendant</var> of
    <var>element</var>'s <span data-x="descendant">descendants</span>, in <span>tree
    order</span>:</p>

    <ol>
     <li>
      <p>If any of</p>

      <ul class="brief">
       <li><var>descendant</var></li>
       <li>any ancestor element of <var>descendant</var> that is a descendant of <var>element</var></li>
       <li>if <var>canExcludeRoot</var> is true, <var>element</var></li>
      </ul>

      <p>is one of</p>

      <ul class="brief">
       <li>a <code>bdi</code> element</li>
       <li>a <code>script</code> element</li>
       <li>a <code>style</code> element</li>
       <li>a <code>textarea</code> element</li>
       <li>an element whose <code data-x="attr-dir">dir</code> attribute is not in the <span
       data-x="attr-dir-undefined-state">undefined</span> state</li>
      </ul>

      <p>then <span>continue</span>.</p>
     </li>

     <li><p>If <var>descendant</var> is a <code>slot</code> element whose <span>root</span> is a
     <span>shadow root</span>, then return the <span data-x="the
     directionality">directionality</span> of that <span>shadow root</span>'s <span
     data-x="concept-DocumentFragment-host">host</span>.</p></li>

     <li><p>If <var>descendant</var> is not a <code>Text</code> node, then
     <span>continue</span>.</p></li>

     <li><p>Let <var>result</var> be the <span>text node directionality</span> of
     <var>descendant</var>.</li>

     <li><p>If <var>result</var> is not null, then return <var>result</var>.</p></li>
    </ol>
   </li>

   <li><p>Return null.</p></li>
  </ol>

  <p>To compute the <dfn>text node directionality</dfn> given a <code>Text</code> node
  <var>text</var>:</p>

  <ol>
   <li><p>If <var>text</var>'s <span data-x="concept-cd-data">data</span> does not contain a code
   point whose bidirectional character type is L, AL, or R, then return null.
   <ref>BIDI</ref></p></li>

   <li><p>Let <var>codePoint</var> be the first code point in <var>text</var>'s <span
   data-x="concept-cd-data">data</span> whose bidirectional character type is L, AL, or R.</p></li>

   <li><p>If <var>codePoint</var> is of bidirectional character type AL or R, then return '<span
   data-x="concept-rtl">rtl</span>'.</p></li>

   <li><p>If <var>codePoint</var> is of bidirectional character type L, then return '<span
   data-x="concept-ltr">ltr</span>'.</p></li>
  </ol>

  <p>To compute the <dfn>parent directionality</dfn> given an element <var>element</var>:</p>

  <ol>
   <li><p>Let <var>parentNode</var> be <var>element</var>'s parent node.</p></li>

   <li><p>If <var>parentNode</var> is a <span>shadow root</span>, then return the <span data-x="the
   directionality">directionality</span> of <var>parentNode</var>'s <span
   data-x="concept-DocumentFragment-host">host</span>.</p></li>

   <li><p>If <var>parentNode</var> is an element, then return the <span data-x="the
   directionality">directionality</span> of <var>parentNode</var>.</p></li>

   <li><p>Return '<span data-x="concept-ltr">ltr</span>'.</p></li>
  </ol>

  <div w-nodev>

  <p class="note">This attribute <a href="#bidireq">has rendering requirements involving the
  bidirectional algorithm</a>.</p>

  </div>

  <hr>

  <p>The <dfn data-x="directionality of the attribute">directionality of an attribute</dfn> of an
  <span data-x="HTML elements">HTML element</span>, which is used when the text of that attribute is
  to be included in the rendering in some manner, is determined as per the first appropriate set of
  steps from the following list:</p>

  <dl class="switch">

   <dt>If the attribute is a <span>directionality-capable attribute</span> and the element's <code
   data-x="attr-dir">dir</code> attribute is in the <span data-x="attr-dir-auto-state">auto</span>
   state</dt>

   <dd>
    <p>Find the first character (in logical order) of the attribute's value that is of bidirectional
    character type L, AL, or R. <ref>BIDI</ref></p>

    <p>If such a character is found and it is of bidirectional character type AL or R, the
    <span>directionality of the attribute</span> is '<span data-x="concept-rtl">rtl</span>'.</p>

    <p>Otherwise, the <span>directionality of the attribute</span> is '<span
    data-x="concept-ltr">ltr</span>'.</p>
   </dd>

   <dt>Otherwise</dt>

   <dd>The <span>directionality of the attribute</span> is the same as <span data-x="the
   directionality">the element's directionality</span>.</dd>
  </dl>

  <p>The following attributes are <dfn data-x="directionality-capable
  attribute">directionality-capable attributes</dfn>:</p>

  <ul class="brief">
   <li><code data-x="attr-th-abbr">abbr</code> on <code>th</code> elements</li>
   <li><code data-x="">alt</code> on <code data-x="attr-area-alt">area</code>,
                                    <code data-x="attr-img-alt">img</code>, and
                                    <code data-x="attr-input-alt">input</code> elements</li>
   <li><code data-x="attr-meta-content">content</code> on <code>meta</code> elements, if the <code data-x="attr-meta-name">name</code> attribute specifies a metadata name whose value is primarily intended to be human-readable rather than machine-readable</li>
   <li><code data-x="">label</code> on <code data-x="attr-optgroup-label">optgroup</code>,
                                       <code data-x="attr-option-label">option</code>, and
                                       <code data-x="attr-track-label">track</code> elements</li>
   <li><code data-x="">placeholder</code> on <code data-x="attr-input-placeholder">input</code> and
                                            <code data-x="attr-textarea-placeholder">textarea</code> elements</li>
   <li><code data-x="attr-title">title</code> on all <span>HTML elements</span></li>
  </ul>

  <hr>

  <dl class="domintro">
   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-dir">dir</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns <span>the <code>html</code> element</span>'s <code data-x="attr-dir">dir</code>
    attribute's value, if any.</p>

    <p>Can be set, to either "<code data-x="">ltr</code>", "<code data-x="">rtl</code>", or "<code
    data-x="">auto</code>" to replace <span>the <code>html</code> element</span>'s <code
    data-x="attr-dir">dir</code> attribute's value.</p>

    <p>If there is no <span data-x="the html element"><code>html</code> element</span>, returns the
    empty string and ignores new values.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLElement"><code data-x="dom-dir">dir</code></dfn> IDL attribute on
  an element must <span>reflect</span> the <code data-x="attr-dir">dir</code> content attribute of
  that element, <span>limited to only known values</span>.</p>

  <p>The <dfn attribute for="Document"><code data-x="dom-document-dir">dir</code></dfn> IDL
  attribute on <code>Document</code> objects must <span>reflect</span> the <code
  data-x="attr-dir">dir</code> content attribute of <span>the <code>html</code> element</span>, if
  any, <span>limited to only known values</span>. If there is no such element, then the attribute
  must return the empty string and do nothing on setting.</p>

  </div>

  <p class="note">Authors are strongly encouraged to use the <code data-x="attr-dir">dir</code>
  attribute to indicate text direction rather than using CSS, since that way their documents will
  continue to render correctly even in the absence of CSS (e.g. as interpreted by search
  engines).</p>

  <div class="example">

   <p>This markup fragment is of an IM conversation.</p>

   <pre><code class="html">&lt;p dir=auto class="u1">&lt;b>&lt;bdi>Student&lt;/bdi>:&lt;/b> How do you write "What's your name?" in Arabic?&lt;/p>
&lt;p dir=auto class="u2">&lt;b>&lt;bdi>Teacher&lt;/bdi>:&lt;/b> &#x645;&#x627; &#x627;&#x633;&#x645;&#x643;&#x61f;&lt;/p>
&lt;p dir=auto class="u1">&lt;b>&lt;bdi>Student&lt;/bdi>:&lt;/b> Thanks.&lt;/p>
&lt;p dir=auto class="u2">&lt;b>&lt;bdi>Teacher&lt;/bdi>:&lt;/b> That's written "&#x634;&#x643;&#x631;&#x64b;&#x627;".&lt;/p>
&lt;p dir=auto class="u2">&lt;b>&lt;bdi>Teacher&lt;/bdi>:&lt;/b> Do you know how to write "Please"?&lt;/p>
&lt;p dir=auto class="u1">&lt;b>&lt;bdi>Student&lt;/bdi>:&lt;/b> "&#x645;&#x646; &#x641;&#x636;&#x644;&#x643;", right?&lt;/p></code></pre>

<!--
<!DOCTYPE html>
<style>
 div { border: groove #FFFF99; padding: 0.25em; width: 22em; background: #FFFFEE; color: black; }
 p { margin: 0.25em 0; font: 1em/1 InaiMathi; color: purple; }
 .u1 { color: teal; }
</style>
<div>
<p dir=ltr class="u1"><b><bdi>Student</bdi>:</b> How do you write "What's your name?" in Arabic?</p>
<p dir=rtl class="u2"><b><bdi>Teacher</bdi>:</b> &#x645;&#x627; &#x627;&#x633;&#x645;&#x643;&#x61f;</p>
<p dir=ltr class="u1"><b><bdi>Student</bdi>:</b> Thanks.</p>
<p dir=ltr class="u2"><b><bdi>Teacher</bdi>:</b> That's written "&#x634;&#x643;&#x631;&#x64b;&#x627;".</p>
<p dir=ltr class="u2"><b><bdi>Teacher</bdi>:</b> Do you know how to write "Please"?</p>
<p dir=rtl class="u1"><b><bdi>Student</bdi>:</b> "&#x645;&#x646; &#x641;&#x636;&#x644;&#x643;", right?</p>
-->

   <p>Given a suitable style sheet and the default alignment styles for the <code>p</code> element,
   namely to align the text to the <i>start edge</i> of the paragraph, the resulting rendering could
   be as follows:</p>

   <p><img src="/images/im.png" alt="Each paragraph rendered as a separate block, with the paragraphs left-aligned except the second paragraph and the last one, which would be right aligned, with the usernames ('Student' and 'Teacher' in this example) flush right, with a colon to their left, and the text first to the left of that." width=366 height=157></p>

   <p>As noted earlier, the <code data-x="attr-dir-auto">auto</code> value is not a panacea. The
   final paragraph in this example is misinterpreted as being right-to-left text, since it begins
   with an Arabic character, which causes the "right?" to be to the left of the Arabic text.</p>

  </div>



  <h5>The <code data-x="attr-style">style</code> attribute</h5>

  <p>All <span>HTML elements</span> may have the <dfn element-attr for="html-global"><code
  data-x="attr-style">style</code></dfn> content attribute set. This is a <span
  data-x="css-styling-attribute">style attribute</span> as defined by <cite>CSS Style
  Attributes</cite>. <ref>CSSATTR</ref></p>

  <div w-nodev>

  <p>In user agents that support CSS, the attribute's value must be parsed when the attribute is
  added or has its value changed<!-- so dynamic changes to the base URL don't affect the CSS -->,
  according to the rules given for <span data-x="css-styling-attribute">style attributes</span>.
  <ref>CSSATTR</ref></p>

  <p>However, if the <span>Should element's inline behavior be blocked by Content Security
  Policy?</span> algorithm returns "<code data-x="">Blocked</code>" when executed upon the
  attribute's <span>element</span>, "<code data-x="">style attribute</code>", and the attribute's
  value, then the style rules defined in the attribute's value must not be applied to the
  <span>element</span>. <ref>CSP</ref></p>

  </div>

  <p>Documents that use <code data-x="attr-style">style</code> attributes on any of their elements
  must still be comprehensible and usable if those attributes were removed.</p>

  <p class="note">In particular, using the <code data-x="attr-style">style</code> attribute to hide
  and show content, or to convey meaning that is otherwise not included in the document, is
  non-conforming. (To hide and show content, use the <code data-x="attr-hidden">hidden</code>
  attribute.)</p>

  <hr>

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-style">style</span></code></dt>
   <dd><p>Returns a <code>CSSStyleDeclaration</code> object for the element's <code
   data-x="attr-style">style</code> attribute.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <code data-x="dom-style">style</code> IDL attribute is defined in <cite>CSS Object
  Model</cite>. <ref>CSSOM</ref></p>

  </div>

  <div class="example">

   <p>In the following example, the words that refer to colors are marked up using the
   <code>span</code> element and the <code data-x="attr-style">style</code> attribute to make those
   words show up in the relevant colors in visual media.</p>

   <pre><code class="html">&lt;p>My sweat suit is &lt;span style="color: green; background:
transparent">green&lt;/span> and my eyes are &lt;span style="color: blue;
background: transparent">blue&lt;/span>.&lt;/p></code></pre>

  </div>



  <h5><dfn>Embedding custom non-visible data</dfn> with the <code data-x="attr-data-*">data-*</code> attributes</h5>

  <p>A <dfn>custom data attribute</dfn> is an attribute in no namespace whose name starts with the
  string "<dfn><code data-x="attr-data-*">data-</code></dfn>", has at least one character after the
  hyphen, is <span>XML-compatible</span>, and contains no <span data-x="ASCII upper alpha">ASCII
  upper alphas</span>.</p>

  <p class="note">All attribute names on <span>HTML elements</span> in <span>HTML documents</span>
  get ASCII-lowercased automatically, so the restriction on ASCII uppercase letters doesn't affect
  such documents.</p>

  <p><span data-x="custom data attribute">Custom data attributes</span> are intended to store custom
  data, state, annotations, and similar, private to the page or application, for which there are no
  more appropriate attributes or elements.</p>

  <p>These attributes are not intended for use by software that is not known to the administrators
  of the site that uses the attributes. For generic extensions that are to be used by multiple
  independent tools, either this specification should be extended to provide the feature explicitly,
  or a technology like <span>microdata</span> should be used (with a standardized vocabulary).</p>

  <div class="example">

   <p>For instance, a site about music could annotate list items representing tracks in an album
   with custom data attributes containing the length of each track. This information could then be
   used by the site itself to allow the user to sort the list by track length, or to filter the list
   for tracks of certain lengths.</p>

   <pre><code class="html">&lt;ol>
 &lt;li data-length="2m11s">Beyond The Sea&lt;/li>
 ...
&lt;/ol></code></pre>

   <p>It would be inappropriate, however, for the user to use generic software not associated with
   that music site to search for tracks of a certain length by looking at this data.</p>

   <p>This is because these attributes are intended for use by the site's own scripts, and are not a
   generic extension mechanism for publicly-usable metadata.</p>

  </div>

  <div class="example">

   <p>Similarly, a page author could write markup that provides information for a translation tool
   that they are intending to use:</p>

   <pre><code class="html">&lt;p>The third &lt;span data-mytrans-de="Anspruch">claim&lt;/span> covers the case of &lt;span
translate="no">HTML&lt;/span> markup.&lt;/p></code></pre>

   <p>In this example, the "<code data-x="">data-mytrans-de</code>" attribute gives specific text
   for the MyTrans product to use when translating the phrase "claim" to German. However, the
   standard <code data-x="attr-translate">translate</code> attribute is used to tell it that in all
   languages, "HTML" is to remain unchanged. When a standard attribute is available, there is no
   need for a <span>custom data attribute</span> to be used.</p>

  </div>

  <div class="example">

   <p>In this example, custom data attributes are used to store the result of a feature detection
   for <code>PaymentRequest</code>, which could be used in CSS to style a checkout page
   differently.</p>

   <pre><code class="html">&lt;script>
 if ('PaymentRequest' in window) {
   document.documentElement.dataset.hasPaymentRequest = '';
 }
&lt;/script></code></pre>

   <p>Here, the <code data-x="">data-has-payment-request</code> attribute is effectively being used
   as a <span>boolean attribute</span>; it is enough to check the presence of the attribute.
   However, if the author so wishes, it could later be populated with some value, maybe to indicate
   limited functionality of the feature.</p>

  </div>

  <p>Every <span data-x="HTML elements">HTML element</span> may have any number of <span
  data-x="custom data attribute">custom data attributes</span> specified, with any value.</p>

  <p>Authors should carefully design such extensions so that when the attributes are ignored and any
  associated CSS dropped, the page is still usable.</p>

  <div w-nodev>

  <p>User agents must not derive any implementation behavior from these attributes or values.
  Specifications intended for user agents must not define these attributes to have any meaningful
  values.</p>

  </div>

  <p>JavaScript libraries may use the <span data-x="custom data attribute">custom data
  attributes</span>, as they are considered to be part of the page on which they are used. Authors
  of libraries that are reused by many authors are encouraged to include their name in the attribute
  names, to reduce the risk of clashes. Where it makes sense, library authors are also encouraged to
  make the exact name used in the attribute names customizable, so that libraries whose authors
  unknowingly picked the same name can be used on the same page, and so that multiple versions of a
  particular library can be used on the same page even when those versions are not mutually
  compatible.</p>

  <div class="example">

   <p>For example, a library called "DoQuery" could use attribute names like <code
   data-x="">data-doquery-range</code>, and a library called "jJo" could use attributes names like
   <code data-x="">data-jjo-range</code>. The jJo library could also provide an API to set which
   prefix to use (e.g. <code data-x="">J.setDataPrefix('j2')</code>, making the attributes have names
   like <code data-x="">data-j2-range</code>).</p>

  </div>

  <hr>

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-dataset">dataset</span></code></dt>
   <dd>
    <p>Returns a <code>DOMStringMap</code> object for the element's <code
    data-x="attr-data-*">data-*</code> attributes.</p>

    <p>Hyphenated names become camel-cased. For example, <code data-x="">data-foo-bar=""</code>
    becomes <code data-x="">element.dataset.fooBar</code>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLOrSVGElement"><code data-x="dom-dataset">dataset</code></dfn> IDL
  attribute provides convenient accessors for all the <code data-x="attr-data-*">data-*</code>
  attributes on an element. On getting, the <code data-x="dom-dataset">dataset</code> IDL attribute
  must return a <code>DOMStringMap</code> whose associated element is this element.</p>

  </div>

  <p>The <code>DOMStringMap</code> interface is used for the <code
  data-x="dom-dataset">dataset</code> attribute. Each <code>DOMStringMap</code> has an <dfn
  data-x="concept-DOMStringMap-element">associated element</dfn>.</p>

  <pre><code class="idl">[Exposed=Window,
 <span>LegacyOverrideBuiltIns</span>]
interface <dfn interface>DOMStringMap</dfn> {
  <a href="#dom-domstringmap-nameditem">getter</a> DOMString (DOMString name);
  [<span>CEReactions</span>] <a href="#dom-domstringmap-setitem">setter</a> undefined (DOMString name, DOMString value);
  [<span>CEReactions</span>] <a href="#dom-domstringmap-removeitem">deleter</a> undefined (DOMString name);
};</code></pre>

  <div w-nodev>

  <p>To <dfn data-x="concept-DOMStringMap-pairs">get a <code>DOMStringMap</code>'s name-value
  pairs</dfn>, run the following algorithm:</p>

  <ol>
   <li><p>Let <var>list</var> be an empty list of name-value pairs.</p></li>

   <li><p>For each content attribute on the <code>DOMStringMap</code>'s <span
   data-x="concept-DOMStringMap-element">associated element</span> whose first five characters are
   the string "<code data-x="">data-</code>" and whose remaining characters (if any) do not include
   any <span data-x="ASCII upper alpha">ASCII upper alphas</span>, in the order that those
   attributes are listed in the element's <span>attribute list</span>, add a name-value pair to
   <var>list</var> whose name is the attribute's name with the first five characters removed and
   whose value is the attribute's value.</p></li>

   <li><p>For each name in <var>list</var>, for each U+002D HYPHEN-MINUS character (-) in the name
   that is followed by an <span>ASCII lower alpha</span>, remove the U+002D HYPHEN-MINUS character
   (-) and replace the character that followed it by the same character <span>converted to ASCII
   uppercase</span>.</p></li>

   <li><p>Return <var>list</var>.</p></li>
  </ol>

  <p>The <span>supported property names</span> on a <code>DOMStringMap</code> object at any instant
  are the names of each pair returned from <span data-x="concept-DOMStringMap-pairs">getting the
  <code>DOMStringMap</code>'s name-value pairs</span> at that instant, in the order returned.</p>

  <p id="dom-domstringmap-nameditem">To <span>determine the value of a named property</span>
  <var>name</var> for a <code>DOMStringMap</code>, return the value component of the name-value pair
  whose name component is <var>name</var> in the list returned from <span
  data-x="concept-DOMStringMap-pairs">getting the <code>DOMStringMap</code>'s name-value
  pairs</span>.</p>

  <p id="dom-domstringmap-setitem">To <span>set the value of a new named property</span> or
  <span>set the value of an existing named property</span> for a <code>DOMStringMap</code>, given a
  property name <var>name</var> and a new value <var>value</var>, run the following steps:</p>

  <ol>
   <li><p>If <var>name</var> contains a U+002D HYPHEN-MINUS character (-) followed by an <span>ASCII
   lower alpha</span>, then throw a <span>"<code>SyntaxError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>For each <span>ASCII upper alpha</span> in <var>name</var>, insert a U+002D HYPHEN-MINUS
   character (-) before the character and replace the character with the same character
   <span>converted to ASCII lowercase</span>.</p></li>

   <li><p>Insert the string <code data-x="">data-</code> at the front of <var>name</var>.</p></li>

   <li><p>If <var>name</var> does not match the XML <code data-x="xml-Name">Name</code> production,
   throw an <span>"<code>InvalidCharacterError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p><span data-x="concept-element-attributes-set-value">Set an attribute value</span> for the
   <code>DOMStringMap</code>'s <span data-x="concept-DOMStringMap-element">associated element</span>
   using <var>name</var> and <var>value</var>.</p></li>
  </ol>

  <p id="dom-domstringmap-removeitem">To <span>delete an existing named property</span>
  <var>name</var> for a <code>DOMStringMap</code>, run the following steps:</p>

  <ol>
<!--(can't happen while the DOMStringMap deleter has no name)
   <li><p>If <var>name</var> contains a U+002D HYPHEN-MINUS character (-) followed by an <span>ASCII
   lower alpha</span>, then throw a <span>"<code>SyntaxError</code>"</span>
   <code>DOMException</code>.</p></li>
-->

   <li><p>For each <span>ASCII upper alpha</span> in <var>name</var>, insert a U+002D HYPHEN-MINUS
   character (-) before the character and replace the character with the same character
   <span>converted to ASCII lowercase</span>.</p></li>

   <li><p>Insert the string <code data-x="">data-</code> at the front of <var>name</var>.</p></li>

   <li><p><span data-x="concept-element-attributes-remove">Remove an attribute by name</span> given
   <var>name</var> and the <code>DOMStringMap</code>'s <span
   data-x="concept-DOMStringMap-element">associated element</span>.</p></li>
  </ol>

  <p class="note">This algorithm will only get invoked by <cite>Web IDL</cite> for names that
  are given by the earlier algorithm for <span data-x="concept-DOMStringMap-pairs">getting the
  <code>DOMStringMap</code>'s name-value pairs</span>. <ref>WEBIDL</ref></p>

  </div>

  <div class="example">

   <p>If a web page wanted an element to represent a space ship, e.g. as part of a game, it would
   have to use the <code data-x="attr-class">class</code> attribute along with <code
   data-x="attr-data-*">data-*</code> attributes:</p>

   <pre><code class="html">&lt;div class="spaceship" data-ship-id="92432"
     data-weapons="laser 2" data-shields="50%"
     data-<!---->x="30" data-y="10" data-z="90">
 &lt;button class="fire"
         onclick="spaceships[this.parentNode.dataset.shipId].fire()">
  Fire
 &lt;/button>
&lt;/div></code></pre>

   <p>Notice how the hyphenated attribute name becomes camel-cased in the API.</p>

  </div>

  <div class="example">

   <p>Given the following fragment and elements with similar constructions:</p>

   <pre><code class="html">&lt;img class="tower" id="tower5" data-<!---->x="12" data-y="5"
     data-ai="robotarget" data-hp="46" data-ability="flames"
     src="towers/rocket.png" alt="Rocket Tower"></code></pre>

   <p>...one could imagine a function <code
   data-x="">splashDamage()</code> that takes some arguments, the first
   of which is the element to process:</p>

   <pre><code class="js">function splashDamage(node, x, y, damage) {
  if (node.classList.contains('tower') && // checking the 'class' attribute
      node.dataset.x == x && // reading the 'data-x' attribute
      node.dataset.y == y) { // reading the 'data-y' attribute
    var hp = parseInt(node.dataset.hp); // reading the 'data-hp' attribute
    hp = hp - damage;
    if (hp &lt; 0) {
      hp = 0;
      node.dataset.ai = 'dead'; // setting the 'data-ai' attribute
      delete node.dataset.ability; // removing the 'data-ability' attribute
    }
    node.dataset.hp = hp; // setting the 'data-hp' attribute
  }
}</code></pre>

  </div>


  <h4 id="the-innertext-idl-attribute">The <code data-x="dom-innerText">innerText</code> and <code
  data-x="dom-outerText">outerText</code> properties</h4>

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-innerText">innerText</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the element's text content "as rendered".</p>

    <p>Can be set, to replace the element's children with the given value, but with line breaks
    converted to <code>br</code> elements.</p>
   </dd>

   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-outerText">outerText</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the element's text content "as rendered".</p>

    <p>Can be set, to replace the element with the given value, but with line breaks converted to
    <code>br</code> elements.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn export>get the text steps</dfn>, given an <span>HTMLElement</span>
  <var>element</var>, are:</p>

  <ol>
   <li>
    <p>If <var>element</var> is not <span>being rendered</span> or if the user agent is a non-CSS
    user agent, then return <var>element</var>'s <span>descendant text content</span>.</p>

    <p class="note">This step can produce surprising results, as when the <code
    data-x="dom-innerText">innerText</code> getter is invoked on an element not <span>being
    rendered</span>, its text contents are returned, but when accessed on an element that is
    <span>being rendered</span>, all of its children that are not <span>being rendered</span> have
    their text contents ignored.</p>
   </li>

   <li><p>Let <var>results</var> be a new empty <span>list</span>.</p></li>

   <li>
    <p>For each child node <var>node</var> of <var>element</var>:</p>

    <ol>
     <li><p>Let <var>current</var> be the <span>list</span> resulting in running the <span>rendered
     text collection steps</span> with <var>node</var>. Each item in <var>results</var> will either
     be a <span>string</span> or a positive integer (a <i>required line break count</i>).</p>

     <p class="note">Intuitively, a <i>required line break count</i> item means that a certain
     number of line breaks appear at that point, but they can be collapsed with the line breaks
     induced by adjacent <i>required line break count</i> items, reminiscent to CSS
     margin-collapsing.</p>

     <li><p>For each item <var>item</var> in <var>current</var>, append <var>item</var> to
     <var>results</var>.</p></li>
    </ol>
   </li>

   <li><p><span data-x="list remove">Remove</span> any items from <var>results</var> that are the
   empty string.</p></li>

   <li><p><span data-x="list remove">Remove</span> any runs of consecutive <i>required line break
   count</i> items at the start or end of <var>results</var>.</p></li>

   <li><p><span data-x="list replace">Replace</span> each remaining run of consecutive <i>required
   line break count</i> items with a string consisting of as many U+000A LF code points as the
   maximum of the values in the <i>required line break count</i> items.</p></li>

   <li><p>Return the concatenation of the string items in <var>results</var>.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLElement"><code data-x="dom-innerText">innerText</code></dfn> and
  <dfn attribute for="HTMLElement"><code data-x="dom-outerText">outerText</code></dfn> getter steps
  are to return the result of running <span>get the text steps</span> with <span>this</span>.</p>

  <p>The <dfn>rendered text collection steps</dfn>, given a <span>node</span> <var>node</var>,
  are as follows:</p>

  <ol>
   <li><p>Let <var>items</var> be the result of running the <span>rendered text collection
   steps</span> with each child node of <var>node</var> in <span>tree order</span>, and then
   concatenating the results to a single <span>list</span>.</p></li>

   <li><p>If <var>node</var>'s <span>computed value</span> of <span>'visibility'</span> is not
   'visible', then return <var>items</var>.</p></li>

   <li>
    <p>If <var>node</var> is not <span>being rendered</span>, then return <var>items</var>. For the
    purpose of this step, the following elements must act as described if the
    <span>computed value</span> of the <span>'display'</span> property is not 'none':</p>

    <ul>
     <li><p><code>select</code> elements have an associated non-replaced inline <span>CSS box</span>
     whose child boxes include only those of <code>optgroup</code> and <code>option</code> element
     child nodes;</p></li>

     <li><p><code>optgroup</code> elements have an associated non-replaced block-level <span>CSS
     box</span> whose child boxes include only those of <code>option</code> element child nodes;
     and</p></li>

     <li><p><code>option</code> element have an associated non-replaced block-level <span>CSS
     box</span> whose child boxes are as normal for non-replaced block-level <span data-x="CSS
     box">CSS boxes</span>.</p></li>
    </ul>

    <p class="note"><var>items</var> can be non-empty due to 'display:contents'.</p>
   </li>

   <li><p>If <var>node</var> is a <code>Text</code> node, then for each CSS text box produced by
   <var>node</var>, in content order, compute the text of the box after application of the CSS
   <span>'white-space'</span> processing rules and <span>'text-transform'</span> rules, set
   <var>items</var> to the <span>list</span> of the resulting strings, and return <var>items</var>.
   The CSS <span>'white-space'</span> processing rules are slightly modified: collapsible spaces at
   the end of lines are always collapsed, but they are only removed if the line is the last line of
   the block, or it ends with a <code>br</code> element. Soft hyphens should be preserved.
   <ref>CSSTEXT</ref></p></li>

   <li><p>If <var>node</var> is a <code>br</code> element, then <span data-x="list
   append">append</span> a string containing a single U+000A LF code point to
   <var>items</var>.</p></li>

   <li><p>If <var>node</var>'s <span>computed value</span> of <span>'display'</span> is
   <span>'table-cell'</span>, and <var>node</var>'s <span>CSS box</span> is not the last
   <span>'table-cell'</span> box of its enclosing <span>'table-row'</span> box, then <span
   data-x="list append">append</span> a string containing a single U+0009 TAB code point to
   <var>items</var>.</p></li>

   <li><p>If <var>node</var>'s <span>computed value</span> of <span>'display'</span> is
   <span>'table-row'</span>, and <var>node</var>'s <span>CSS box</span> is not the last
   <span>'table-row'</span> box of the nearest ancestor <span>'table'</span> box, then <span
   data-x="list append">append</span> a string containing a single U+000A LF code point to
   <var>items</var>.</p></li>

   <li><p>If <var>node</var> is a <code>p</code> element, then <span data-x="list
   append">append</span> 2 (a <i>required line break count</i>) at the beginning and end of
   <var>items</var>.</p></li>

   <li>
    <p>If <var>node</var>'s <span>used value</span> of <span>'display'</span> is
    <span>block-level</span> or <span>'table-caption'</span>, then <span data-x="list
    append">append</span> 1 (a <i>required line break count</i>) at the beginning and end of
    <var>items</var>. <ref>CSSDISPLAY</ref></p>

    <p class="note">Floats and absolutely-positioned elements fall into this category.</p>
   </li>

   <li><p>Return <var>items</var>.</p></li>
  </ol>

  <p class="note">Note that descendant nodes of most replaced elements (e.g., <code>textarea</code>,
  <code>input</code>, and <code>video</code> &mdash; but not <code>button</code>) are not rendered
  by CSS, strictly speaking, and therefore have no <span data-x="CSS box">CSS boxes</span> for the
  purposes of this algorithm.</p>

  <p class="XXX">This algorithm is amenable to being generalized to work on <span
  data-x="concept-range">ranges</span>. Then we can use it as the basis for <code>Selection</code>'s
  stringifier and maybe expose it directly on <span data-x="concept-range">ranges</span>. See <a
  href="https://www.w3.org/Bugs/Public/show_bug.cgi?id=10583">Bugzilla bug 10583</a>.</p>

  <hr>

  <p>The <dfn export>set the inner text steps</dfn>, given an <span>HTMLElement</span>
  <var>element</var>, and a string <var>value</var> are:</p>

  <ol>
   <li><p>Let <var>fragment</var> be the <span>rendered text fragment</span> for <var>value</var>
   given <var>element</var>'s <span>node document</span>.</p></li>

   <li><p><span data-x="concept-node-replace-all">Replace all</span> with <var>fragment</var>
   within <var>element</var>.</p></li>
  </ol>

  <p>The <code data-x="dom-innerText">innerText</code> setter steps are to run <span>set the inner
  text steps</span>.</p>

  <p>The <code data-x="dom-outerText">outerText</code> setter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s parent is null, then throw a
   <span>"<code>NoModificationAllowedError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>next</var> be <span>this</span>'s <span>next sibling</span>.</p></li>

   <li><p>Let <var>previous</var> be <span>this</span>'s <span>previous sibling</span>.</p></li>

   <li><p>Let <var>fragment</var> be the <span>rendered text fragment</span> for the given value
   given <span>this</span>'s <span>node document</span>.</p></li>

   <li><p>If <var>fragment</var> has no <span data-x="concept-tree-child">children</span>, then
   <span data-x="concept-node-append">append</span> a new <code>Text</code> node whose <span
   data-x="concept-cd-data">data</span> is the empty string and <span>node document</span> is
   <span>this</span>'s <span>node document</span> to <var>fragment</var>.</p></li>

   <li><p><span data-x="concept-node-replace">Replace</span> <span>this</span> with
   <var>fragment</var> within <span>this</span>'s parent.</p></li>

   <li><p>If <var>next</var> is non-null and <var>next</var>'s <span>previous sibling</span> is a
   <code>Text</code> node, then <span>merge with the next text node</span> given <var>next</var>'s
   <span>previous sibling</span>.</p></li>

   <li><p>If <var>previous</var> is a <code>Text</code> node, then <span>merge with the next text
   node</span> given <var>previous</var>.</p></li>
  </ol>

  <p>The <dfn>rendered text fragment</dfn> for a string <var>input</var> given a
  <code>Document</code> <var>document</var> is the result of running the following steps:</p>

  <ol>
   <li><p>Let <var>fragment</var> be a new <code>DocumentFragment</code> whose <span>node
   document</span> is <var>document</var>.</p></li>

   <li><p>Let <var>position</var> be a <span>position variable</span> for <var>input</var>,
   initially pointing at the start of <var>input</var>.</p></li>

   <li><p>Let <var>text</var> be the empty string.</p></li>

   <li>
    <p>While <var>position</var> is not past the end of <var>input</var>:</p>

    <ol>
     <li><p><span>Collect a sequence of code points</span> that are not U+000A LF or U+000D CR from
     <var>input</var> given <var>position</var>, and set <var>text</var> to the result.</p></li>

     <li><p>If <var>text</var> is not the empty string, then <span
     data-x="concept-node-append">append</span> a new <code>Text</code> node whose <span
     data-x="concept-cd-data">data</span> is <var>text</var> and <span>node document</span> is
     <var>document</var> to <var>fragment</var>.</p></li>

     <li>
      <p>While <var>position</var> is not past the end of <var>input</var>, and the code point at
      <var>position</var> is either U+000A LF or U+000D CR:</p>

      <ol>
       <li><p>If the code point at <var>position</var> is U+000D CR and the next code point is
       U+000A LF, then advance <var>position</var> to the next code point in
       <var>input</var>.</p></li>

       <li><p>Advance <var>position</var> to the next code point in <var>input</var>.</p></li>

       <li><p><span data-x="concept-node-append">Append</span> the result of <span data-x="create an
       element">creating an element</span> given <var>document</var>, "<code data-x="">br</code>",
       and the <span>HTML namespace</span> to <var>fragment</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Return <var>fragment</var>.</p></li>
  </ol>

  <p>To <dfn>merge with the next text node</dfn> given a <code>Text</code> node <var>node</var>:</p>

  <ol>
   <li><p>Let <var>next</var> be <var>node</var>'s <span>next sibling</span>.</p></li>

   <li><p>If <var>next</var> is not a <code>Text</code> node, then return.</p></li>

   <li><p><span>Replace data</span> with <var>node</var>, <var>node</var>'s <span
   data-x="concept-cd-data">data</span>'s <span>length</span>, 0, and <var>next</var>'s <span
   data-x="concept-cd-data">data</span>.</p></li>

   <li><p><span data-x="concept-node-remove">Remove</span> <var>next</var>.</p></li>
  </ol>

  </div>

  <h4>Requirements relating to the bidirectional algorithm</h4>

  <div w-nodev>
  <h5>Authoring conformance criteria for bidirectional-algorithm formatting characters</h5>
  </div>

  <p><span>Text content</span> in <span>HTML elements</span> with <code>Text</code> nodes in their
  <span data-x="concept-html-contents">contents</span>, and text in attributes of <span>HTML
  elements</span> that allow free-form text, may contain characters in the ranges U+202A to U+202E
  and U+2066 to U+2069 (the bidirectional-algorithm formatting characters). <ref>BIDI</ref></p>

  <p class="note">Authors are encouraged to use the <code data-x="attr-dir">dir</code> attribute, the
  <code>bdo</code> element, and the <code>bdi</code> element, rather than maintaining the
  bidirectional-algorithm formatting characters manually. The bidirectional-algorithm formatting
  characters interact poorly with CSS.</p>


  <div w-nodev>

  <h5>User agent conformance criteria</h5>

  <p>User agents must implement the Unicode bidirectional algorithm to determine the proper ordering
  of characters when rendering documents and parts of documents. <ref>BIDI</ref></p>

  <p>The mapping of HTML to the Unicode bidirectional algorithm must be done in one of three ways.
  Either the user agent must implement CSS, including in particular the CSS
  <span>'unicode-bidi'</span>, <span>'direction'</span>, and <span>'content'</span> properties, and
  must have, in its user agent style sheet, the rules using those properties given in this
  specification's <a href="#rendering">rendering</a> section, or, alternatively, the user agent must
  act as if it implemented just the aforementioned properties and had a user agent style sheet that
  included all the aforementioned rules, but without letting style sheets specified in documents
  override them, or, alternatively, the user agent must implement another styling language with
  equivalent semantics. <ref>CSSGC</ref></p>

  <p id="bidireq">The following elements and attributes have requirements defined by the <a
  href="#rendering">rendering</a> section that, due to the requirements in this section, are
  requirements on all user agents (not just those that <a href="#renderingUA">support the suggested
  default rendering</a>):</p>

  <ul class="brief">
   <li><code data-x="attr-dir">dir</code> attribute</li>
   <li><code>bdi</code> element</li>
   <li><code>bdo</code> element</li>
   <li><code>br</code> element</li>
   <li><code>pre</code> element</li>
   <li><code>textarea</code> element</li>
   <li><code>wbr</code> element</li>
  </ul>

  </div>

  <div w-nodev>

  <h4 id="wai-aria">Requirements related to ARIA and to platform accessibility APIs</h4>

  <p>User agent requirements for implementing Accessibility API semantics on <span>HTML
  elements</span> are defined in <cite>HTML Accessibility API Mappings</cite>. In addition to the
  rules there, for a <span>custom element</span> <var>element</var>, the default ARIA role semantics
  are determined as follows: <ref>HTMLAAM</ref></p>

  <ol>
   <li><p>Let <var>map</var> be <var>element</var>'s <span>internal content attribute
   map</span>.</p></li>

   <li><p>If <var>map</var>["<code data-x="">role</code>"] <span data-x="map exists">exists</span>,
   then return it.</p></li>

   <li><p>Return no role.</p></li>
  </ol>

  <p>Similarly, for a <span>custom element</span> <var>element</var>, the default ARIA state and
  property semantics, for a state or property named <var>stateOrProperty</var>, are determined as
  follows:</p>

  <ol>
   <li>
    <p>If <var>element</var>'s <span>attached internals</span> is non-null:</p>

    <ol>
     <li><p>If <var>element</var>'s <span>attached internals</span>'s <span data-x="get the
     attr-associated element">get the <var>stateOrProperty</var>-associated element</span> exists,
     then return the result of running it.</p></li>

     <li><p>If <var>element</var>'s <span>attached internals</span>'s <span data-x="get the
     attr-associated elements">get the <var>stateOrProperty</var>-associated elements</span> exists,
     then return the result of running it.</p></li>
    </ol>
    <!-- We check these first as they end up being redundantly stored in element's internal content
         attribute map with an empty string value. -->

   <li><p>If <var>element</var>'s <span>internal content attribute
   map</span>[<var>stateOrProperty</var>] <span data-x="map exists">exists</span>, then return
   it.</p></li>

   <li><p>Return the default value for <var>stateOrProperty</var>.</p></li>
  </ol>

  <p class="note">The "default semantics" referred to here are sometimes also called "native",
  "implicit", or "host language" semantics in <cite>ARIA</cite>. <ref>ARIA</ref></p>

  <p class="note">One implication of these definitions is that the default semantics can change over
  time. This allows custom elements the same expressivity as built-in elements; e.g., compare to how
  the default ARIA role semantics of an <code>a</code> element change as the <code
  data-x="attr-hyperlink-href">href</code> attribute is added or removed.</p>

  <p>For an example of this in action, see <a href="#custom-elements-accessibility-example">the
  custom elements section</a>.</p>

  <hr>

  <p>Conformance checker requirements for checking use of ARIA <code
  data-x="attr-aria-role">role</code> and <code data-x="attr-aria-*">aria-*</code> attributes on
  <span>HTML elements</span> are defined in <cite>ARIA in HTML</cite>. <ref>ARIAHTML</ref></p>

  </div>



  <h2 split-filename="semantics" id="semantics">The elements of HTML</h2>

  <h3 id="the-root-element">The document element</h3>

  <h4 id="the-html-element">The <dfn element><code>html</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As document's <span>document element</span>.</dd>
   <dd>Wherever a subdocument fragment is allowed in a compound document.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>A <code>head</code> element followed by a <code>body</code> element.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-html">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-html">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLHtmlElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  // <a href="#HTMLHtmlElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLHtmlElement</code>.</dd>
  </dl>

  <p>The <code>html</code> element <span>represents</span> the root of an HTML document.</p>

  <p>Authors are encouraged to specify a <code data-x="attr-lang">lang</code> attribute on the root
  <code>html</code> element, giving the document's language. This aids speech synthesis tools to
  determine what pronunciations to use, translation tools to determine what rules to use, and so
  forth.</p>

  <div class="example">

   <p>The <code>html</code> element in the following example declares that the document's language
   is English.</p>

   <pre><code class="html">&lt;!DOCTYPE html>
<strong>&lt;html lang="en"></strong>
&lt;head>
&lt;title>Swapping Songs&lt;/title>
&lt;/head>
&lt;body>
&lt;h1>Swapping Songs&lt;/h1>
&lt;p>Tonight I swapped some of the songs I wrote with some friends, who
gave me some of the songs they wrote. I love sharing my music.&lt;/p>
&lt;/body>
<strong>&lt;/html></strong></code></pre>

  </div>




  <h3>Document metadata</h3>

  <h4 id="the-head-element">The <dfn element><code>head</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As the first element in an <code>html</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>If the document is <span>an <code>iframe</code> <code data-x="attr-iframe-srcdoc">srcdoc</code> document</span> or if title information is available from a higher-level protocol: Zero or more elements of <span>metadata content</span>, of which no more than one is a <code>title</code> element and no more than one is a <code>base</code> element.</dd>
   <dd>Otherwise: One or more elements of <span>metadata content</span>, of which exactly one is a <code>title</code> element and no more than one is a <code>base</code> element.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-head">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-head">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLHeadElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLHeadElement</code>.</dd>
  </dl>

  <p>The <code>head</code> element <span>represents</span> a collection of metadata for the
  <code>Document</code>.</p>

  <div class="example">

   <p>The collection of metadata in a <code>head</code> element can be large or small. Here is an
   example of a very short one:</p>

   <pre><code class="html">&lt;!doctype html>
&lt;html lang=en>
 &lt;head>
  &lt;title>A document with a short head&lt;/title>
 &lt;/head>
 &lt;body>
 ...</code></pre>

   <p>Here is an example of a longer one:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;HTML LANG="EN">
 &lt;HEAD>
  &lt;META CHARSET="UTF-8">
  &lt;BASE HREF="https://www.example.com/">
  &lt;TITLE>An application with a long head&lt;/TITLE>
  &lt;LINK REL="STYLESHEET" HREF="default.css">
  &lt;LINK REL="STYLESHEET ALTERNATE" HREF="big.css" TITLE="Big Text">
  &lt;SCRIPT SRC="support.js">&lt;/SCRIPT>
  &lt;META NAME="APPLICATION-NAME" CONTENT="Long headed application">
 &lt;/HEAD>
 &lt;BODY>
 ...</code></pre>

  </div>

  <p class="note">The <code>title</code> element is a required child in most situations, but when a
  higher-level protocol provides title information, e.g., in the subject line of an email when HTML
  is used as an email authoring format, the <code>title</code> element can be omitted.</p>



  <h4 id="the-title-element">The <dfn element><code>title</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Metadata content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>In a <code>head</code> element containing no other <code>title</code> elements.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="text content">Text</span> that is not <span>inter-element whitespace</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-title">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-title">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLTitleElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-title-text">text</span>;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLTitleElement</code>.</dd>
  </dl>

  <p>The <code>title</code> element <span>represents</span> the document's title or name. Authors
  should use titles that identify their documents even when they are used out of context, for
  example in a user's history or bookmarks, or in search results. The document's title is often
  different from its first heading, since the first heading does not have to stand alone when taken
  out of context.</p>

  <p>There must be no more than one <code>title</code> element per document.</p>

  <p class="note">If it's reasonable for the <code>Document</code> to have no title, then the
  <code>title</code> element is probably not required. See the <code>head</code> element's content
  model for a description of when the element is required.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>title</var>.<span subdfn data-x="dom-title-text">text</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the <span>child text content</span> of the element.</p>

    <p>Can be set, to replace the element's children with the given value.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLTitleElement"><code data-x="dom-title-text">text</code></dfn>
  attribute's getter must return this <code>title</code> element's <span>child text
  content</span>.</p>

  <p>The <code data-x="dom-title-text">text</code> attribute's setter must <span>string replace
  all</span> with the given value within this <code>title</code> element.</p>

  </div>

  <div class="example">

   <p>Here are some examples of appropriate titles, contrasted with the top-level headings that
   might be used on those same pages.</p>

   <pre><code class="html">  &lt;title>Introduction to The Mating Rituals of Bees&lt;/title>
    ...
  &lt;h1>Introduction&lt;/h1>
  &lt;p>This companion guide to the highly successful
  &lt;cite>Introduction to Medieval Bee-Keeping&lt;/cite> book is...</code></pre>

   <p>The next page might be a part of the same site. Note how the title describes the subject
   matter unambiguously, while the first heading assumes the reader knows what the context is and
   therefore won't wonder if the dances are Salsa or Waltz:</p>

   <pre><code class="html">  &lt;title>Dances used during bee mating rituals&lt;/title>
    ...
  &lt;h1>The Dances&lt;/h1></code></pre>

  </div>

  <p>The string to use as the document's title is given by the <code
  data-x="dom-document-title">document.title</code> IDL attribute.</p>

  <div w-nodev>

  <p>User agents should use the document's title when referring to the document in their user
  interface. When the contents of a <code>title</code> element are used in this way, <span>the
  directionality</span> of that <code>title</code> element should be used to set the directionality
  of the document's title in the user interface.</p>

  </div>


  <h4>The <dfn element><code>base</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Metadata content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>In a <code>head</code> element containing no other <code>base</code> elements.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="concept-content-nothing">Nothing</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-base-href">href</code></dd>
   <dd><code data-x="attr-base-target">target</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-base">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-base">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLBaseElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute USVString <span data-x="dom-base-href">href</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-base-target">target</span>;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLBaseElement</code>.</dd>
  </dl>

  <p>The <code>base</code> element allows authors to specify the <span>document base URL</span> for
  the purposes of parsing <span data-x="URL">URLs</span>, and the name of the default
  <span>navigable</span> for the purposes of <span>following hyperlinks</span>. The element does not
  <span data-x="represents">represent</span> any content beyond this information.</p>

  <p>There must be no more than one <code>base</code> element per document.</p>

  <p>A <code>base</code> element must have either an <code data-x="attr-base-href">href</code>
  attribute, a <code data-x="attr-base-target">target</code> attribute, or both.</p>

  <p>The <dfn element-attr for="base"><code data-x="attr-base-href">href</code></dfn> content
  attribute, if specified, must contain a <span>valid URL potentially surrounded by
  spaces</span>.</p>

  <p>A <code>base</code> element, if it has an <code data-x="attr-base-href">href</code> attribute,
  must come before any other elements in the tree that have attributes defined as taking <span
  data-x="URL">URLs</span>.</p>

  <div w-nodev>

  <p class="note">If there are multiple <code>base</code> elements with <code
  data-x="attr-base-href">href</code> attributes, all but the first are ignored.</p>

  </div>

  <p>The <dfn element-attr for="base"><code data-x="attr-base-target">target</code></dfn> attribute,
  if specified, must contain a <span>valid navigable target name or keyword</span>, which specifies
  which <span>navigable</span> is to be used as the default when <span
  data-x="hyperlink">hyperlinks</span> and <span data-x="form">forms</span> in the
  <code>Document</code> cause <span data-x="navigate">navigation</span>.</p>

  <p>A <code>base</code> element, if it has a <code data-x="attr-base-target">target</code>
  attribute, must come before any elements in the tree that represent <span
  data-x="hyperlink">hyperlinks</span>.</p>

  <div w-nodev>

  <p class="note">If there are multiple <code>base</code> elements with <code
  data-x="attr-base-target">target</code> attributes, all but the first are ignored.</p>

  <p>To <dfn>get an element's target</dfn>, given an <code>a</code>, <code>area</code>, or
  <code>form</code> element <var>element</var>, and an optional string-or-null <var>target</var>
  (default null), run these steps:

  <ol>
   <li>
    <p>If <var>target</var> is null, then:</p>

    <ol>
     <li><p>If <var>element</var> has a <code data-x="">target</code> attribute, then set
     <var>target</var> to that attribute's value.</p></li>

     <li><p>Otherwise, if <var>element</var>'s <span>node document</span> contains a
     <code>base</code> element with a <code data-x="attr-base-target">target</code> attribute, set
     <var>target</var> to the value of the <code data-x="attr-base-target">target</code> attribute
     of the first such <code>base</code> element.</p></li>
    </ol>
   </li>

   <li><p>If <var>target</var> is not null, and contains an <span>ASCII tab or newline</span> and a
   U+003C (&lt;), then set <var>target</var> to "<code data-x="">_blank</code>".</p></li>

   <li><p>Return <var>target</var>.</p></li>
  </ol>

  <hr>

  <p>A <code>base</code> element that is the first <code>base</code> element with an <code
  data-x="attr-base-href">href</code> content attribute <span>in a document tree</span> has a
  <dfn>frozen base URL</dfn>. The <span>frozen base URL</span> must be <span>immediately</span>
  <span data-x="set the frozen base URL">set</span> for an element whenever any of the following
  situations occur:</p>

  <ul>
   <li><p>The <code>base</code> element becomes the first <code>base</code> element in <span>tree
   order</span> with an <code data-x="attr-base-href">href</code> content attribute in its
   <code>Document</code>.</p></li>

   <li><p>The <code>base</code> element is the first <code>base</code> element in <span>tree
   order</span> with an <code data-x="attr-base-href">href</code> content attribute in its
   <code>Document</code>, and its <code data-x="attr-base-href">href</code> content attribute is
   changed.</p></li>
  </ul>

  <p>To <dfn export>set the frozen base URL</dfn> for an element <var>element</var>:</p>

  <ol>
   <li><p>Let <var>document</var> be <var>element</var>'s <span>node document</span>.

   <li><p>Let <var>urlRecord</var> be the result of <span data-x="URL parser">parsing</span> the
   value of <var>element</var>'s <code data-x="attr-base-href">href</code> content attribute with
   <var>document</var>'s <span>fallback base URL</span>, and <var>document</var>'s <span
   data-x="document's character encoding">character encoding</span>. (Thus, the <code>base</code>
   element isn't affected by itself.)</p></li>
   <!-- This uses the URL parser rather than encoding-parse a URL since otherwise we'd have to
        unnecessarily complicate the latter for two callsites. -->

   <li>
    <p>If any of the following are true:</p>

    <ul>
     <li><p><var>urlRecord</var> is failure;</p></li>

     <li><p><var>urlRecord</var>'s <span data-x="concept-url-scheme">scheme</span> is "<code
     data-x="">data</code>" or "<code data-x="">javascript</code>"; or</p></li>

     <li><p>running <span>Is base allowed for Document?</span> on <var>urlRecord</var> and
     <var>document</var> returns "<code data-x="">Blocked</code>",</p></li>
    </ul>

    <p>then set <var>element</var>'s <span>frozen base URL</span> to <var>document</var>'s
    <span>fallback base URL</span> and return.</p>
   </li>

   <li><p>Set <var>element</var>'s <span>frozen base URL</span> to <var>urlRecord</var>.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLBaseElement"><code data-x="dom-base-href">href</code></dfn> IDL
  attribute, on getting, must return the result of running the following algorithm:
  <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1715
       https://github.com/whatwg/html/issues/1060#issuecomment-211221664 -->

  <ol>
   <li><p>Let <var>document</var> be <var>element</var>'s <span>node document</span>.

   <li><p>Let <var>url</var> be the value of the <code data-x="attr-base-href">href</code>
   attribute of this element, if it has one, and the empty string otherwise.</p></li>

   <li><p>Let <var>urlRecord</var> be the result of <span data-x="URL parser">parsing</span>
   <var>url</var> with <var>document</var>'s <span>fallback base URL</span>, and
   <var>document</var>'s <span data-x="document's character encoding">character encoding</span>.
   (Thus, the <code>base</code> element isn't affected by other <code>base</code> elements or
   itself.)</p></li>
   <!-- This uses the URL parser rather than encoding-parse a URL since otherwise we'd have to
        unnecessarily complicate the latter for two callsites. -->

   <li><p>If <var>urlRecord</var> is failure, return <var>url</var>.</p></li>

   <li><p>Return the <span data-x="concept-url-serializer">serialization</span> of
   <var>urlRecord</var>.</p></li>
  </ol>

  <p>The <code data-x="dom-base-href">href</code> IDL attribute, on setting, must set the <code
  data-x="attr-base-href">href</code> content attribute to the given new value.</p>

  <p>The <dfn attribute for="HTMLBaseElement"><code data-x="dom-base-target">target</code></dfn> IDL
  attribute must <span>reflect</span> the content attribute of the same name.</p>

  </div>

  <div class="example">

   <p>In this example, a <code>base</code> element is used to set the <span>document base
   URL</span>:</p>

   <pre><code class="html">&lt;!DOCTYPE html>
&lt;html lang="en">
    &lt;head>
        &lt;title>This is an example for the &amp;lt;base&amp;gt; element&lt;/title>
        &lt;base href="https://www.example.com/news/index.html">
    &lt;/head>
    &lt;body>
        &lt;p>Visit the &lt;a href="archives.html">archives&lt;/a>.&lt;/p>
    &lt;/body>
&lt;/html></code></pre>

   <p>The link in the above example would be a link to "<code
   data-x="">https://www.example.com/news/archives.html</code>".</p>

  </div>




  <h4>The <dfn element><code>link</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Metadata content</span>.</dd>
   <dd>If the element is <span>allowed in the body</span>: <span>flow content</span>.</dd>
   <dd>If the element is <span>allowed in the body</span>: <span>phrasing content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>metadata content</span> is expected.</dd>
   <dd>In a <code>noscript</code> element that is a child of a <code>head</code> element.</dd>
   <dd>If the element is <span>allowed in the body</span>: where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="concept-content-nothing">Nothing</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-link-href">href</code></dd>
   <dd><code data-x="attr-link-crossorigin">crossorigin</code></dd>
   <dd><code data-x="attr-link-rel">rel</code></dd>
   <dd><code data-x="attr-link-media">media</code></dd>
   <dd><code data-x="attr-link-integrity">integrity</code></dd>
   <dd><code data-x="attr-link-hreflang">hreflang</code></dd>
   <dd><code data-x="attr-link-type">type</code></dd>
   <dd><code data-x="attr-link-referrerpolicy">referrerpolicy</code></dd>
   <dd><code data-x="attr-link-sizes">sizes</code></dd>
   <dd><code data-x="attr-link-imagesrcset">imagesrcset</code></dd>
   <dd><code data-x="attr-link-imagesizes">imagesizes</code></dd>
   <dd><code data-x="attr-link-as">as</code></dd>
   <dd><code data-x="attr-link-blocking">blocking</code></dd>
   <dd><code data-x="attr-link-color">color</code></dd>
   <dd><code data-x="attr-link-disabled">disabled</code></dd>
   <dd><code data-x="attr-link-fetchpriority">fetchpriority</code></dd>
   <dd>Also, the <code data-x="attr-link-title">title</code> attribute <span data-x="attr-link-title">has special semantics</span> on this element.</dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-link">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-link">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLLinkElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute USVString <span data-x="dom-link-href">href</span>;
  [<span>CEReactions</span>] attribute DOMString? <span data-x="dom-link-crossOrigin">crossOrigin</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-link-rel">rel</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-link-as">as</span>;
  [SameObject, PutForwards=<span data-x="dom-DOMTokenList-value">value</span>] readonly attribute <span>DOMTokenList</span> <span data-x="dom-link-relList">relList</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-link-media">media</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-link-integrity">integrity</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-link-hreflang">hreflang</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-link-type">type</span>;
  [SameObject, PutForwards=<span data-x="dom-DOMTokenList-value">value</span>] readonly attribute <span>DOMTokenList</span> <span data-x="dom-link-sizes">sizes</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-link-imageSrcset">imageSrcset</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-link-imageSizes">imageSizes</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-link-referrerPolicy">referrerPolicy</span>;
  [SameObject, PutForwards=<span data-x="dom-DOMTokenList-value">value</span>] readonly attribute <span>DOMTokenList</span> <span data-x="dom-link-blocking">blocking</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-link-disabled">disabled</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-link-fetchPriority">fetchPriority</span>;

  // <a href="#HTMLLinkElement-partial">also has obsolete members</a>
};
<span>HTMLLinkElement</span> includes <span>LinkStyle</span>;</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLLinkElement</code>.</dd>
  </dl>

  <p>The <code>link</code> element allows authors to link their document to other resources.</p>

  <p>The address of the link(s) is given by the <dfn element-attr for="link"><code
  data-x="attr-link-href">href</code></dfn> attribute. If the <code
  data-x="attr-link-href">href</code> attribute is present, then its value must be a <span>valid
  non-empty URL potentially surrounded by spaces</span>. One or both of the <code
  data-x="attr-link-href">href</code> or <code data-x="attr-link-imagesrcset">imagesrcset</code>
  attributes must be present.</p>

  <p w-nodev>If both the <code data-x="attr-link-href">href</code> and <code
  data-x="attr-link-imagesrcset">imagesrcset</code> attributes are absent, then the element does not
  define a link.</p>

  <p>The types of link indicated (the relationships) are given by the value of the <dfn element-attr
  for="link"><code data-x="attr-link-rel">rel</code></dfn> attribute, which, if present, must have a
  value that is a <span>unordered set of unique space-separated tokens</span>. The <a
  href="#linkTypes">allowed keywords and their meanings</a> are defined in a later section. <span
  w-nodev>If the <code data-x="attr-link-rel">rel</code> attribute is absent, has no keywords, or if
  none of the keywords used are allowed according to the definitions in this specification, then the
  element does not create any links.</span></p>

  <div w-nodev>

  <p><code data-x="attr-link-rel">rel</code>'s
  <span data-x="concept-supported-tokens">supported tokens</span> are the keywords defined in
  <a href="#linkTypes">HTML link types</a> which are allowed on <code>link</code> elements, impact
  the processing model, and are supported by the user agent. The possible <span
  data-x="concept-supported-tokens">supported tokens</span> are
  <code data-x="rel-alternate">alternate</code>,
  <code data-x="rel-dns-prefetch">dns-prefetch</code>,
  <code data-x="rel-expect">expect</code>,
  <code data-x="rel-icon">icon</code>,
  <code data-x="rel-manifest">manifest</code>,
  <code data-x="rel-modulepreload">modulepreload</code>,
  <code data-x="rel-next">next</code>,
  <code data-x="rel-pingback">pingback</code>,
  <code data-x="rel-preconnect">preconnect</code>,
  <code data-x="rel-prefetch">prefetch</code>,
  <code data-x="rel-preload">preload</code>,
  <code data-x="rel-search">search</code>, and
  <code data-x="rel-stylesheet">stylesheet</code>.
  <code data-x="attr-link-rel">rel</code>'s <span data-x="concept-supported-tokens">supported
  tokens</span> must only include the tokens from this list that the user agent implements the
  processing model for.</p>

  <p class="note">Theoretically a user agent could support the processing model for the <code
  data-x="rel-canonical">canonical</code> keyword — if it were a search engine that executed
  JavaScript. But in practice that's quite unlikely. So in most cases, <code
  data-x="rel-canonical">canonical</code> ought not be included in <code
  data-x="attr-link-rel">rel</code>'s <span data-x="concept-supported-tokens">supported
  tokens</span>.</p>

  </div>

  <p>A <code>link</code> element must have either a <code data-x="attr-link-rel">rel</code>
  attribute or an <code data-x="attr-itemprop">itemprop</code> attribute, but not both.</p>

  <p>If a <code>link</code> element has an <code data-x="attr-itemprop">itemprop</code> attribute,
  or has a <code data-x="attr-link-rel">rel</code> attribute that contains only keywords that are
  <span>body-ok</span>, then the element is said to be <dfn>allowed in the body</dfn>. This means
  that the element can be used where <span>phrasing content</span> is expected.</p>

  <p class="note">If the <code data-x="attr-link-rel">rel</code> attribute is used, the element can
  only sometimes be used in the <code>body</code> of the page. When used with the <code
  data-x="attr-itemprop">itemprop</code> attribute, the element can be used both in the
  <code>head</code> element and in the <code>body</code> of the page, subject to the constraints of
  the microdata model.</p>

  <hr>

  <p>Two categories of links can be created using the <code>link</code> element: <span
  data-x="external resource link">links to external resources</span> and <span
  data-x="hyperlink">hyperlinks</span>. The <a href="#linkTypes">link types section</a> defines
  whether a particular link type is an external resource or a hyperlink. One <code>link</code>
  element can create multiple links (of which some might be <span data-x="external resource
  link">external resource links</span> and some might be <span
  data-x="hyperlink">hyperlinks</span>); exactly which and how many links are created depends on the
  keywords given in the <code data-x="attr-link-rel">rel</code> attribute. User agents must process
  the links on a per-link basis, not a per-element basis.</p>

  <p class="note">Each link created for a <code>link</code> element is handled separately. For
  instance, if there are two <code>link</code> elements with <code data-x="">rel="stylesheet"</code>,
  they each count as a separate external resource, and each is affected by its own attributes
  independently. Similarly, if a single <code>link</code> element has a <code
  data-x="attr-link-rel">rel</code> attribute with the value <code data-x="">next stylesheet</code>,
  it creates both a <span>hyperlink</span> (for the <code data-x="rel-next">next</code> keyword) and
  an <span>external resource link</span> (for the <code data-x="rel-stylesheet">stylesheet</code>
  keyword), and they are affected by other attributes (such as <code
  data-x="attr-link-media">media</code> or <code data-x="attr-link-title">title</code>)
  differently.</p>

  <div class="example">
   <p>For example, the following <code>link</code> element creates two <span
   data-x="hyperlink">hyperlinks</span> (to the same page):</p>

   <pre><code class="html">&lt;link rel="author license" href="/about"></code></pre>

   <p>The two links created by this element are one whose semantic is that the target page has
   information about the current page's author, and one whose semantic is that the target page has
   information regarding the license under which the current page is provided.</p>
  </div>

  <p><span data-x="hyperlink">Hyperlinks</span> created with the <code>link</code> element and its
  <code data-x="attr-link-rel">rel</code> attribute apply to the whole document. This contrasts with
  the <code data-x="attr-hyperlink-rel">rel</code> attribute of <code>a</code> and <code>area</code>
  elements, which indicates the type of a link whose context is given by the link's location within
  the document.</p>

  <p>Unlike those created by <code>a</code> and <code>area</code> elements, <span
  data-x="hyperlink">hyperlinks</span> created by <code>link</code> elements are not displayed as
  part of the document by default, in user agents that <a href="#renderingUA">support the suggested
  default rendering</a>. And even if they are force-displayed using CSS, they have no
  <span>activation behavior</span>. Instead, they primarily provide semantic information which might
  be used by the page or by other software that consumes the page's contents. Additionally, the user
  agent can <a
  href="#providing-users-with-a-means-to-follow-hyperlinks-created-using-the-link-element">provide
  its own UI for following such hyperlinks</a>.</p>

  <p>The exact behavior for <span data-x="external resource link">links to external resources</span>
  depends on the exact relationship, as defined for the relevant <a href="#linkTypes">link
  type</a>.</p>

  <hr>

  <p>The <dfn element-attr for="link"><code data-x="attr-link-crossorigin">crossorigin</code></dfn>
  attribute is a <span>CORS settings attribute</span>. It is intended for use with <span
  data-x="external resource link">external resource links</span>.</p>

  <p>The <dfn element-attr for="link"><code data-x="attr-link-media">media</code></dfn> attribute
  says which media the resource applies to. The value must be a <span>valid media query
  list</span>.</p>

  <p>The <dfn for="link" element-attr><code data-x="attr-link-integrity">integrity</code></dfn>
  attribute represents the <span data-x="concept-request-integrity-metadata">integrity
  metadata</span> for requests which this element is responsible for. The value is text. The
  attribute must only be specified on <code>link</code> elements that have a <code
  data-x="attr-link-rel">rel</code> attribute that contains the <code
  data-x="rel-stylesheet">stylesheet</code>, <code data-x="rel-preload">preload</code>, or <code
  data-x="rel-modulepreload">modulepreload</code> keyword. <ref>SRI</ref></p>

  <p>The <dfn element-attr for="link"><code data-x="attr-link-hreflang">hreflang</code></dfn>
  attribute on the <code>link</code> element has the same semantics as the <span
  data-x="attr-hyperlink-hreflang"><code>hreflang</code> attribute on the <code>a</code>
  element</span>.</p>

  <p>The <dfn element-attr for="link"><code data-x="attr-link-type">type</code></dfn> attribute
  gives the <span>MIME type</span> of the linked resource. It is purely advisory. The value must be
  a <span>valid MIME type string</span>.</p>

  <p>For <span data-x="external resource link">external resource links</span>, the <code
  data-x="attr-link-type">type</code> attribute is used as a hint to user agents so that they can
  avoid fetching resources they do not support.</p>

  <p>The <dfn element-attr for="link"><code
  data-x="attr-link-referrerpolicy">referrerpolicy</code></dfn> attribute is a <span>referrer policy
  attribute</span>. It is intended for use with <span data-x="external resource link">external
  resource links</span>, where it helps set the <span>referrer policy</span> used when <span
  data-x="fetch and process the linked resource">fetching and processing the linked resource</span>.
  <ref>REFERRERPOLICY</ref></p>

  <p>The <dfn element-attr for="link"><code data-x="attr-link-title">title</code></dfn> attribute
  gives the title of the link. With one exception, it is purely advisory. The value is text. The
  exception is for style sheet links that are <span>in a document tree</span>, for which the <code
  data-x="attr-link-title">title</code> attribute defines <span data-x="CSS style sheet set">CSS
  style sheet sets</span>.</p>

  <p class="note">The <code data-x="attr-link-title">title</code> attribute on <code>link</code>
  elements differs from the global <code data-x="attr-title">title</code> attribute of most other
  elements in that a link without a title does not inherit the title of the parent element: it
  merely has no title.</p>

  <hr>

  <p>The <dfn element-attr for="link"><code data-x="attr-link-imagesrcset">imagesrcset</code></dfn>
  attribute may be present, and is a <span>srcset attribute</span>.</p>

  <p>The <code data-x="attr-link-imagesrcset">imagesrcset</code> and <code
  data-x="attr-link-href">href</code> attributes (if <span data-x="width descriptor">width
  descriptors</span> are not used) together contribute the <span data-x="image source">image
  sources</span> to the <span>source set</span>.</p>

  <p>If the <code data-x="attr-link-imagesrcset">imagesrcset</code> attribute is present and has any
  <span data-x="image candidate string">image candidate strings</span> using a <span>width
  descriptor</span>, the <dfn element-attr for="link"><code
  data-x="attr-link-imagesizes">imagesizes</code></dfn> attribute must also be present, and is a
  <span>sizes attribute</span>. The <code data-x="attr-link-imagesizes">imagesizes</code> attribute
  contributes the <span>source size</span> to the <span>source set</span>.</p>

  <p>The <code data-x="attr-link-imagesrcset">imagesrcset</code> and <code
  data-x="attr-link-imagesizes">imagesizes</code> attributes must only be specified on
  <code>link</code> elements that have both a <code data-x="attr-link-rel">rel</code> attribute that
  specifies the <code data-x="rel-preload">preload</code> keyword, as well as an <code
  data-x="attr-link-as">as</code> attribute in the "<code data-x="">image</code>" state.</p>

  <div class="example">
   <p>These attributes allow preloading the appropriate resource that is later used by an
   <code>img</code> element that has the corresponding values for its <code
   data-x="attr-img-srcset">srcset</code> and <code data-x="attr-img-sizes">sizes</code>
   attributes:</p>

   <pre><code class="html">&lt;link rel="preload" as="image"
      imagesrcset="wolf_400px.jpg 400w, wolf_800px.jpg 800w, wolf_1600px.jpg 1600w"
      imagesizes="50vw">

&lt;!-- ... later, or perhaps inserted dynamically ... -->
&lt;img src="wolf.jpg" alt="A rad wolf"
     srcset="wolf_400px.jpg 400w, wolf_800px.jpg 800w, wolf_1600px.jpg 1600w"
     sizes="50vw"></code></pre>

     <p>Note how we omit the <code data-x="attr-link-href">href</code> attribute, as it would only
     be relevant for browsers that do not support <code
     data-x="attr-link-imagesrcset">imagesrcset</code>, and in those cases it would likely cause the
     incorrect image to be preloaded.</p>
  </div>

  <div class="example">
   <p>The <code data-x="attr-link-imagesrcset">imagesrcset</code> attribute can be combined with the
   <code data-x="attr-link-media">media</code> attribute to preload the appropriate resource
   selected from a <code>picture</code> element's sources, for <span>art direction</span>:</p>

   <pre><code class="html">&lt;link rel="preload" as="image"
      imagesrcset="dog-cropped-1x.jpg, dog-cropped-2x.jpg 2x"
      media="(max-width: 800px)">
&lt;link rel="preload" as="image"
      imagesrcset="dog-wide-1x.jpg, dog-wide-2x.jpg 2x"
      media="(min-width: 801px)">

&lt;!-- ... later, or perhaps inserted dynamically ... -->
&lt;picture>
  &lt;source srcset="dog-cropped-1x.jpg, dog-cropped-2x.jpg 2x"
          media="(max-width: 800px)">
  &lt;img src="dog-wide-1x.jpg" srcset="dog-wide-2x.jpg 2x"
       alt="An awesome dog">
&lt;/picture></code></pre>
  </div>

  <hr>

  <p>The <dfn element-attr for="link"><code data-x="attr-link-sizes">sizes</code></dfn> attribute
  gives the sizes of icons for visual media. Its value, if present, is merely advisory.<span
  w-nodev> User agents may use the value to decide which icon(s) to use if multiple icons are
  available.</span> If specified, the attribute must have a value that is an <span>unordered set of
  unique space-separated tokens</span> which are <span>ASCII case-insensitive</span>. Each value
  must be either an <span>ASCII case-insensitive</span> match for the string "<code
  data-x="attr-link-sizes-any">any</code>", or a value that consists of two <span data-x="valid
  non-negative integer">valid non-negative integers</span> that do not have a leading U+0030 DIGIT
  ZERO (0) character and that are separated by a single U+0078 LATIN SMALL LETTER X or U+0058 LATIN
  CAPITAL LETTER X character. The attribute must only be specified on <code>link</code> elements
  that have a <code data-x="attr-link-rel">rel</code> attribute that specifies the <code
  data-x="rel-icon">icon</code> keyword or the <code data-x="">apple-touch-icon</code> keyword.</p>

  <p class="note">The <code data-x="">apple-touch-icon</code> keyword is a registered <span
  data-x="concept-rel-extensions">extension to the predefined set of link types</span>, but user
  agents are not required to support it in any way.</p>

  <hr>

  <p>The <dfn element-attr for="link"><code data-x="attr-link-as">as</code></dfn> attribute
  specifies the <span data-x="concept-potential-destination">potential destination</span> for a
  preload request for the resource given by the <code data-x="attr-link-href">href</code> attribute.
  It is an <span>enumerated attribute</span>. Each <span
  data-x="concept-potential-destination">potential destination</span> is a keyword for this
  attribute, mapping to a state of the same name. The attribute must be specified on
  <code>link</code> elements that have a <code data-x="attr-link-rel">rel</code> attribute that
  contains the <code data-x="rel-preload">preload</code> keyword. It may be specified on
  <code>link</code> elements that have a <code data-x="attr-link-rel">rel</code> attribute that
  contains the <code data-x="rel-modulepreload">modulepreload</code> keyword; in such cases it must
  have a value which is a <span data-x="concept-script-like-destination">script-like
  destination</span>. For other <code>link</code> elements, it must not be specified.</p>

  <p w-nodev>The processing model for how the <code data-x="attr-link-as">as</code> attribute is
  used is given in an individual link type's <span>fetch and process the linked resource</span>
  algorithm.</p>

  <p class="note">The attribute does not have a <i data-x="missing value default">missing value
  default</i> or <i data-x="invalid value default">invalid value default</i>, meaning that invalid
  or missing values for the attribute map to no state. This is accounted for in the processing
  model. For <code data-x="rel-preload">preload</code> links, both conditions are an error; for
  <code data-x="rel-modulepreload">modulepreload</code> links, a missing value will be treated as
  "<code data-x="">script</code>".</p>

  <hr>

  <p>The <dfn element-attr for="link"><code data-x="attr-link-blocking">blocking</code></dfn>
  attribute is a <span>blocking attribute</span>. It is used by link types <code
  data-x="rel-stylesheet">stylesheet</code> and <code
  data-x="rel-expect">expect</code>, and it must only be specified on link elements
  that have a <code data-x="attr-link-rel">rel</code> attribute containing those keywords.</p>

  <hr>

  <p>The <dfn element-attr for="link"><code data-x="attr-link-color">color</code></dfn> attribute is
  used with the <code data-x="">mask-icon</code> link type. The attribute must only be specified on
  <code>link</code> elements that have a <code data-x="attr-link-rel">rel</code> attribute that
  contains the <code data-x="">mask-icon</code> keyword. The value must be a string that matches the
  CSS <span>&lt;color></span> production, defining a suggested color that user agents can use to
  customize the display of the icon that the user sees when they pin your site.</p>

  <p class="note">This specification does not have any user agent requirements for the <code
  data-x="attr-link-color">color</code> attribute.</p>

  <p class="note">The <code data-x="">mask-icon</code> keyword is a registered <span
  data-x="concept-rel-extensions">extension to the predefined set of link types</span>, but user
  agents are not required to support it in any way.</p>

  <hr>

  <p><code>link</code> elements have an associated <dfn>explicitly enabled</dfn> boolean. It is
  initially false.

  <p>The <dfn element-attr for="link"><code data-x="attr-link-disabled">disabled</code></dfn>
  attribute is a <span>boolean attribute</span> that is used with the <code
  data-x="rel-stylesheet">stylesheet</code> link type.  The attribute must only be specified on
  <code>link</code> elements that have a <code data-x="attr-link-rel">rel</code> attribute that
  contains the <code data-x="rel-stylesheet">stylesheet</code> keyword.</p>

  <p>Whenever the <code data-x="attr-link-disabled">disabled</code> attribute is removed, set the
  <code>link</code> element's <span>explicitly enabled</span> attribute to true.</p>

  <div class="example">
   <p>Removing the <code data-x="attr-link-disabled">disabled</code> attribute dynamically, e.g.,
   using <code data-x="">document.querySelector("link").removeAttribute("disabled")</code>, will
   fetch and apply the style sheet:</p>

   <pre><code class="html">&lt;link disabled rel="alternate stylesheet" href="css/pooh"></code></pre>
  </div>

  <p>The <dfn element-attr for="link"
  data-x="attr-link-fetchpriority"><code>fetchpriority</code></dfn> attribute is a <span>fetch
  priority attribute</span> that is intended for use with <span
  data-x="external resource link">external resource links</span>, where it is used to set the <span
  data-x="concept-request-priority">priority</span> used when <span
  data-x="fetch and process the linked resource">fetching and processing the linked
  resource</span>.</p>

  <div w-nodev>
  <hr>

  <p>The IDL attributes
  <dfn attribute for="HTMLLinkElement"><code data-x="dom-link-href">href</code></dfn>,
  <dfn attribute for="HTMLLinkElement"><code data-x="dom-link-hreflang">hreflang</code></dfn>,
  <dfn attribute for="HTMLLinkElement"><code data-x="dom-link-integrity">integrity</code></dfn>,
  <dfn attribute for="HTMLLinkElement"><code data-x="dom-link-media">media</code></dfn>,
  <dfn attribute for="HTMLLinkElement"><code data-x="dom-link-rel">rel</code></dfn>,
  <dfn attribute for="HTMLLinkElement"><code data-x="dom-link-sizes">sizes</code></dfn>,
  <dfn attribute for="HTMLLinkElement"><code data-x="dom-link-type">type</code></dfn>,
  <dfn attribute for="HTMLLinkElement"><code data-x="dom-link-blocking">blocking</code></dfn>, and
  <dfn attribute for="HTMLLinkElement"><code data-x="dom-link-disabled">disabled</code></dfn>
  each must <span>reflect</span> the respective content attributes of the same name.</p>

  <p class="note">There is no reflecting IDL attribute for the <code
  data-x="attr-link-color">color</code> attribute, but this might be added later.</p>

  <p>The <dfn attribute for="HTMLLinkElement"><code data-x="dom-link-as">as</code></dfn> IDL
  attribute must <span>reflect</span> the <code data-x="attr-link-as">as</code> content attribute,
  <span>limited to only known values</span>.</p>

  <p>The <dfn attribute for="HTMLLinkElement"><code
  data-x="dom-link-crossOrigin">crossOrigin</code></dfn> IDL attribute must <span>reflect</span> the
  <code data-x="attr-link-crossorigin">crossorigin</code> content attribute, <span>limited to only
  known values</span>.</p>

  <p>The <dfn attribute for="HTMLLinkElement"><code
  data-x="dom-link-referrerPolicy">referrerPolicy</code></dfn> IDL attribute must
  <span>reflect</span> the <code data-x="attr-link-referrerpolicy">referrerpolicy</code> content
  attribute, <span>limited to only known values</span>.</p>

  <p>The <dfn attribute for="HTMLLinkElement"><code
  data-x="dom-link-fetchPriority">fetchPriority</code></dfn> IDL attribute must
  <span>reflect</span> the <code data-x="attr-link-fetchpriority">fetchpriority</code> content
  attribute, <span>limited to only known values</span>.</p>

  <p>The <dfn attribute for="HTMLLinkElement"><code
  data-x="dom-link-imageSrcset">imageSrcset</code></dfn> IDL attribute must <span>reflect</span> the
  <code data-x="attr-link-imagesrcset">imagesrcset</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLLinkElement"><code
  data-x="dom-link-imageSizes">imageSizes</code></dfn> IDL attribute must <span>reflect</span> the
  <code data-x="attr-link-imagesizes">imagesizes</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLLinkElement"><code data-x="dom-link-rellist">relList</code></dfn>
  IDL attribute must <span data-x="reflect">reflect</span> the <code
  data-x="attr-link-rel">rel</code> content attribute.</p>

  <p class="note">The <code data-x="dom-link-rellist">relList</code> attribute can be used for
  feature detection, by calling its <code data-x="dom-DOMTokenList-supports">supports()</code>
  method to check which <a href="#linkTypes">types of links</a> are supported.</p>

  <h5>Processing the <code data-x="attr-link-media">media</code> attribute</h5>

  <p>If the link is a <span>hyperlink</span> then the <code data-x="attr-link-media">media</code>
  attribute is purely advisory, and describes for which media the document in question was
  designed.</p>

  <p>However, if the link is an <span>external resource link</span>, then the <code
  data-x="attr-link-media">media</code> attribute is prescriptive. The user agent must apply the
  external resource when the <code data-x="attr-link-media">media</code> attribute's value
  <span>matches the environment</span> and the other relevant conditions apply, and must not apply
  it otherwise.</p><!-- note similar text in <style> element section -->

  <p id="default-media">The default, if the <code data-x="attr-link-media">media</code> attribute is
  omitted, is "<code data-x="">all</code>", meaning that by default links apply to all media.</p>

  <p class="note">The external resource might have further restrictions defined within that limit
  its applicability. For example, a CSS style sheet might have some <code data-x="">@media</code>
  blocks. This specification does not override such further restrictions or requirements.</p><!--
  note similar text in <style> element section -->

  <h5>Processing the <code data-x="attr-link-type">type</code> attribute</h5>

  <p>If the <code data-x="attr-link-type">type</code> attribute is present, then the user agent must
  assume that the resource is of the given type (even if that is not a <span>valid MIME type
  string</span>, e.g. the empty string). If the attribute is omitted, but the <span>external
  resource link</span> type has a default type defined, then the user agent must assume that the
  resource is of that type. If the UA does not support the given <span>MIME type</span> for the
  given link relationship, then the UA should not <span>fetch and process the linked
  resource</span>; if the UA does support the given <span>MIME type</span> for the given link
  relationship, then the UA should <span>fetch and process the linked resource</span> at the
  appropriate time as specified for the <span>external resource link</span>'s particular type.
  If the attribute is omitted, and the <span>external resource link</span> type does not have a
  default type defined, but the user agent would <span>fetch and process the linked resource</span>
  if the type was known and supported, then the user agent should <span>fetch and process the linked
  resource</span> under the assumption that it will be supported.</p>

  <p>User agents must not consider the <code data-x="attr-link-type">type</code> attribute
  authoritative &mdash; upon fetching the resource, user agents must not use the <code
  data-x="attr-link-type">type</code> attribute to determine its actual type. Only the actual type
  (as defined in the next paragraph) is used to determine whether to <em>apply</em> the resource,
  not the aforementioned assumed type.</p>

  <p id="concept-link-type-sniffing">If the <span>external resource link</span> type defines rules
  for processing the resource's <span data-x="Content-Type">Content-Type metadata</span>, then those
  rules apply.  Otherwise, if the resource is expected to be an image, user agents may apply the
  <span data-x="Content-Type sniffing: image">image sniffing rules</span>, with the <var>official
  type</var> being the type determined from the resource's <span data-x="Content-Type">Content-Type
  metadata</span>, and use the resulting <span data-x="Content-Type sniffing">computed type of the
  resource</span> as if it was the actual type.  Otherwise, if neither of these conditions apply or
  if the user agent opts not to apply the image sniffing rules, then the user agent must use the
  resource's <span data-x="Content-Type">Content-Type metadata</span> to determine the type of the
  resource. If there is no type metadata, but the <span>external resource link</span> type has a
  default type defined, then the user agent must assume that the resource is of that type.</p>

  <p class="note">The <code data-x="rel-stylesheet">stylesheet</code> link type defines rules for
  processing the resource's <span data-x="Content-Type">Content-Type metadata</span>.</p>

  <p>Once the user agent has established the type of the resource, the user agent must apply the
  resource if it is of a supported type and the other relevant conditions apply, and must ignore the
  resource otherwise.</p>

  <div class="example">

   <p>If a document contains style sheet links labeled as follows:</p>

   <pre><code class="html">&lt;link rel="stylesheet" href="A" type="text/plain"&gt;
&lt;link rel="stylesheet" href="B" type="text/css"&gt;
&lt;link rel="stylesheet" href="C"&gt;</code></pre>

   <p>...then a compliant UA that supported only CSS style sheets would fetch the B and C files, and
   skip the A file (since <code>text/plain</code> is not the <span>MIME type</span> for CSS style
   sheets).</p>

   <p>For files B and C, it would then check the actual types returned by the server. For those that
   are sent as <code>text/css</code>, it would apply the styles, but for those labeled as
   <code>text/plain</code>, or any other type, it would not.</p>

   <p>If one of the two files was returned without a <span>Content-Type</span> metadata, or with a
   syntactically incorrect type like <code data-x="">Content-Type:&nbsp;"null"</code>, then the
   default type for <code data-x="rel-stylesheet">stylesheet</code> links would kick in. Since that
   default type is <code>text/css</code>, the style sheet <em>would</em> nonetheless be applied.</p>

  </div>

  <h5><span id="obtaining-a-resource-from-a-link-element"></span>Fetching and processing a resource
  from a <code>link</code> element</h5>

  <p id="concept-link-obtain">All <span data-x="external resource link">external resource
  links</span> have a <dfn>fetch and process the linked resource</dfn> algorithm, which takes a
  <code>link</code> element <var>el</var>. They also have <dfn>linked resource fetch setup
  steps</dfn> which take a <code>link</code> element <var>el</var> and <span
  data-x="concept-request">request</span> <var>request</var>. Individual link types may provide
  their own <span>fetch and process the linked resource</span> algorithm, but unless explicitly
  stated, they use the <span>default fetch and process the linked resource</span> algorithm.
  Similarly, individual link types may provide their own <span>linked resource fetch setup
  steps</span>, but unless explicitly stated, these steps just return true.</p>

  <p>The <dfn>default fetch and process the linked resource</dfn>, given a <code>link</code> element
  <var>el</var>, is as follows:</p>

  <ol>
   <li><p>Let <var>options</var> be the result of <span data-x="create link options from
   element">creating link options</span> from <var>el</var>.</p></li>

   <li><p>Let <var>request</var> be the result of <span data-x="create a link request">creating a
   link request</span> given <var>options</var>.</p></li>

   <li><p>If <var>request</var> is null, then return.</p></li>

   <li><p>Set <var>request</var>'s <span>synchronous flag</span>.</p></li>

   <li><p>Run the <span>linked resource fetch setup steps</span>, given <var>el</var> and
   <var>request</var>. If the result is false, then return.</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-initiator-type">initiator
   type</span> to "<code data-x="">css</code>" if <var>el</var>'s <code
   data-x="attr-link-rel">rel</code> attribute contains the keyword <code
   data-x="rel-stylesheet">stylesheet</code>; "<code data-x="">link</code>" otherwise.</p></li>

   <li>
    <p><!--FETCH--><span data-x="concept-fetch">Fetch</span> <var>request</var> with <i
    data-x="processResponseConsumeBody">processResponseConsumeBody</i> set to the following steps
    given <span data-x="concept-response">response</span> <var>response</var> and null, failure, or
    a <span>byte sequence</span> <var>bodyBytes</var>:</p>

    <ol>
     <li><p>Let <var>success</var> be true.</p></li>

     <li>
      <p>If any of the following are true:</p>

      <ul>
       <li><p><var>bodyBytes</var> is null or failure; or</p></li>

       <li><p><var>response</var>'s <span data-x="concept-response-status">status</span> is not an
       <span>ok status</span>,</p></li>
      </ul>

      <p>then set <var>success</var> to false.</p>

      <p class="note">Note that content-specific errors, e.g., CSS parse errors or PNG decoding
      errors, do not affect <var>success</var>.</p>
     </li>

     <li>
      <p>Otherwise, wait for the <span data-x="external resource link">link
      resource</span>'s <span>critical subresources</span> to finish loading.</p>

      <p class="XXX">The specification that defines a link type's <span>critical subresources</span>
      (e.g., CSS) is expected to describe how these subresources are fetched and processed. However,
      since this is not currently explicit, this specification describes waiting for a <span
      data-x="external resource link">link resource</span>'s <span>critical subresources</span> to
      be fetched and processed, with the expectation that this will be done correctly.</p>
     </li>

     <li><p><span>Process the linked resource</span> given <var>el</var>, <var>success</var>,
     <var>response</var>, and <var>bodyBytes</var>.</li>
    </ol>
   </li>
  </ol>

  <p>To <dfn>create a link request</dfn> given a <span>link processing options</span>
  <var>options</var>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>options</var>'s <span data-x="link options href">href</span> is
   not the empty string.</p></li>

   <li><p>If <var>options</var>'s <span data-x="link options destination">destination</span> is
   null, then return null.</p></li>

   <li>
    <p>Let <var>url</var> be the result of <span>encoding-parsing a URL</span> given
    <var>options</var>'s <span data-x="link options href">href</span>, relative to
    <var>options</var>'s <span data-x="link options base url">base URL</span>.</p>

    <p class="XXX">Passing the base URL instead of a document or environment is tracked by <a
    href="https://github.com/whatwg/html/issues/9715">issue #9715</a>.</p>
   </li>

   <li><p>If <var>url</var> is failure, then return null.</p></li>

   <li><p>Let <var>request</var> be the result of <span data-x="create a potential-CORS
   request">creating a potential-CORS request</span> given <var>url</var>, <var>options</var>'s
   <span data-x="link options destination">destination</span>, and <var>options</var>'s <span
   data-x="link options crossorigin">crossorigin</span>.</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-policy-container">policy
   container</span> to <var>options</var>'s <span data-x="link options policy container">policy
   container</span>.</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-integrity-metadata">integrity
   metadata</span> to <var>options</var>'s <span data-x="link options
   integrity">integrity</span>.</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-nonce-metadata">cryptographic
   nonce metadata</span> to <var>options</var>'s <span data-x="link options nonce">cryptographic
   nonce metadata</span>.</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-referrer-policy">referrer
   policy</span> to <var>options</var>'s <span data-x="link options referrer policy">referrer
   policy</span>.</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-client">client</span> to
   <var>options</var>'s <span data-x="link options environment">environment</span>.</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-priority">priority</span> to
   <var>options</var>'s <span data-x="link options fetch priority">fetch priority</span>.</p></li>

   <li><p>Return <var>request</var>.</p></li>
  </ol>

  <p>User agents may opt to only try to <span data-x="fetch and process the linked resource">fetch
  and process</span> such resources when they are needed, instead of pro-actively fetching all the
  <span data-x="external resource link">external resources</span> that are not applied.</p>

  <!-- the next few paragraphs are similar to text in the <style> element section -->
  <!-- TODO: integrate this better with Fetch's process response definition -->
  <p>Similar to the <span>fetch and process the linked resource</span> algorithm, all <span
  data-x="external resource link">external resource links</span> have a <dfn>process the linked
  resource</dfn> algorithm which takes a <code>link</code> element <var>el</var>, boolean
  <var>success</var>, a <span data-x="concept-response">response</span> <var>response</var>, and a
  <span>byte sequence</span> <var>bodyBytes</var>. Individual link types may provide their own
  <span>process the linked resource</span> algorithm, but unless explicitly stated, that algorithm
  does nothing.</p>

  <p>Unless otherwise specified for a given <code data-x="attr-link-rel">rel</code> keyword, the
  element must <span>delay the load event</span> of the element's <span>node document</span> until
  all the attempts to <span>fetch and process the linked resource</span> and its <span>critical
  subresources</span> are complete. (Resources that the user agent has not yet attempted to fetch
  and process, e.g., because it is waiting for the resource to be needed, do not <span>delay the
  load event</span>.)</p>

  <h5>Processing `<code data-x="http-link">Link</code>` headers</h5>

  <p>All link types that can be <span data-x="external resource link">external resource
  links</span> define a <dfn>process a link header</dfn> algorithm, which takes a <span>link
  processing options</span>. This algorithm defines whether and how they react to appearing in an
  HTTP `<code data-x="http-link">Link</code>` response header.</p>

  <p class="note">For most link types, this algorithm does nothing. The <a
  href="#table-link-relations">summary table</a> is a good reference to quickly know whether a link
  type has defined <span>process a link header</span> steps.</p>

  <p>A <dfn>link processing options</dfn> is a <span>struct</span>. It has the following
  <span data-x="struct item">items</span>:</p>

  <dl>
   <dt><dfn data-x="link options href">href</dfn> (default the empty string)</dt>
   <dt><dfn data-x="link options destination">destination</dfn> (default the empty string)</dt>
   <dt><dfn data-x="link options initiator">initiator</dfn> (default "<code
   data-x="">link</code>")</dt>
   <dt><dfn data-x="link options integrity">integrity</dfn> (default the empty string)</dt>
   <dt><dfn data-x="link options type">type</dfn> (default the empty string)</dt>
   <dt><dfn data-x="link options nonce">cryptographic nonce metadata</dfn> (default the empty
   string)</dt>
   <dd>A string</dd>

   <dt><dfn data-x="link options crossorigin">crossorigin</dfn> (default <span
   data-x="attr-crossorigin-none">No CORS</span>)</dt>
   <dd>A <span>CORS settings attribute</span> state</dd>

   <dt><dfn data-x="link options referrer policy">referrer policy</dfn> (default the empty
   string)</dt>
   <dd>A <span>referrer policy</span></dd>

   <dt><dfn data-x="link options source set">source set</dfn> (default null)</dt>
   <dd>Null or a  <span>source set</span></dd>

   <dt><dfn data-x="link options base URL">base URL</dfn></dt>
   <dd>A <span>URL</span></dd>

   <dt><dfn data-x="link options origin">origin</dfn></dt>
   <dd>An <span>origin</span></dd>

   <dt><dfn data-x="link options environment">environment</dfn></dt>
   <dd>An <span>environment</span></dd>

   <dt><dfn data-x="link options policy container">policy container</dfn></dt>
   <dd>A <span>policy container</span></dd>

   <dt><dfn data-x="link options document">document</dfn> (default null)</dt>
   <dd>Null or a <code>Document</code></dd>

   <dt><dfn data-x="link options on document ready">on document ready</dfn> (default null)</dt>
   <dd>Null or an algorithm accepting a <code>Document</code></dd>

   <dt><dfn data-x="link options fetch priority">fetch priority</dfn> (default <code
   data-x="attr-fetchpriority-auto-state">auto</code>)</dt>
   <dd>A <span>fetch priority attribute</span> state</dd>
  </dl>

  <p class="note">A <span>link processing options</span> has a <span
  data-x="link options base URL">base URL</span> and an <span data-x="link options href">href</span>
  rather than a parsed URL because the URL could be a result of the options's <span
  data-x="link options source set">source set</span>.</p>

  <p>To <dfn>create link options from element</dfn> given a <code>link</code> element
  <var>el</var>:</p>

  <ol>
   <li><p>Let <var>document</var> be <var>el</var>'s <span>node document</span>.</p></li>

   <li>
    <p>Let <var>options</var> be a new <span>link processing options</span> with</p>

    <dl class="props">
     <dt><span data-x="link options destination">destination</span></dt>
     <dd>the result of <span
     data-x="translate a preload destination">translating</span> the state of <var>el</var>'s <code
     data-x="attr-link-as">as</code> attribute.</dd>

     <dt><span data-x="link options crossorigin">crossorigin</span></dt>
     <dd>the state of <var>el</var>'s <code data-x="attr-link-crossorigin">crossorigin</code>
     content attribute</dd>

     <dt><span data-x="link options referrer policy">referrer policy</span></dt>
     <dd>the state of <var>el</var>'s <code data-x="attr-link-referrerpolicy">referrerpolicy</code>
     content attribute</dd>

     <dt><span data-x="link options source set">source set</span></dt>
     <dd><var>el</var>'s <span>source set</span></dd>

     <dt><span data-x="link options base URL">base URL</span></dt>
     <dd><var>document</var>'s <span>document base URL</span></dd>

     <dt><span data-x="link options origin">origin</span></dt>
     <dd><var>document</var>'s <span data-x="concept-document-origin">origin</span></dd>

     <dt><span data-x="link options environment">environment</span></dt>
     <dd><var>document</var>'s <span>relevant settings object</span></dd>

     <dt><span data-x="link options policy container">policy container</span></dt>
     <dd><var>document</var>'s <span
     data-x="concept-document-policy-container">policy container</span></dd>

     <dt><span data-x="link options document">document</span></dt>
     <dd><var>document</var></dd>

     <dt><span data-x="link options nonce">cryptographic nonce metadata</span></dt>
     <dd>The current value of <var>el</var>'s <span>[[CryptographicNonce]]</span> internal slot</dd>

     <dt><span data-x="link options fetch priority">fetch priority</span></dt>
     <dd>the state of <var>el</var>'s <code data-x="attr-link-fetchpriority">fetchpriority</code>
     content attribute</dd>
    </dl>
   </li>

   <li><p>If <var>el</var> has an <code data-x="attr-link-href">href</code> attribute, then set
   <var>options</var>'s <span data-x="link options href">href</span> to the value of
   <var>el</var>'s <code data-x="attr-link-href">href</code> attribute.</p></li>

   <li><p>If <var>el</var> has an <code data-x="attr-link-integrity">integrity</code> attribute,
   then set <var>options</var>'s <span data-x="link options integrity">integrity</span> to the
   value of <var>el</var>'s <code data-x="attr-link-integrity">integrity</code> content
   attribute.</p></li>

   <li><p>If <var>el</var> has a <code data-x="attr-link-type">type</code> attribute, then set
   <var>options</var>'s <span data-x="link options type">type</span> to the value of
   <var>el</var>'s <code data-x="attr-link-type">type</code> attribute.</p></li>

   <li>
    <p><span>Assert</span>: <var>options</var>'s <span data-x="link options href">href</span> is not
    the empty string, or <var>options</var>'s <span data-x="link options source set">source
    set</span> is not null.</p>

    <p>A <code>link</code> element with neither an <code data-x="attr-link-href">href</code> or an
    <code data-x="attr-link-imagesrcset">imagesrcset</code> does not represent a link.</p>
   </li>

   <li><p>Return <var>options</var>.</p></li>
  </ol>

  <p>To <dfn>extract links from headers</dfn> given a <span data-x="concept-header-list">header
  list</span> <var>headers</var>:</p>

  <ol>
   <li><p>Let <var>links</var> be a new <span>list</span>.</p></li>

   <li><p>Let <var>rawLinkHeaders</var> be the result of <span
   data-x="concept-header-list-get-decode-split">getting, decoding, and splitting</span>
   `<code>Link</code>` from <var>headers</var>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>linkHeader</var> of
    <var>rawLinkHeaders</var>:</p>

    <ol>
     <li><p>Let <var>linkObject</var> be the result of <span data-x="parsing a link field
     value">parsing</span> <var>linkHeader</var>. <ref>WEBLINK</ref></p></li>

     <li><p>If <var>linkObject</var>["<code data-x="">target_uri</code>"] does not
     <span data-x="map exists">exist</span>, then <span>continue</span>.</p></li>

     <li><p><span data-x="list append">Append</span> <var>linkObject</var> to
     <var>links</var>.</p></li>
    </ol>
   </li>
   <li><p>Return <var>links</var>.</p></li>
  </ol>

  <p>To <dfn>process link headers</dfn> given a <code>Document</code> <var>doc</var>,
  a <span data-x="concept-response">response</span> <var>response</var>, and a
  "<code data-x="">pre-media</code>" or "<code data-x="">media</code>" <var>phase</var>:</p>

  <ol>
   <li><p>Let <var>links</var> be the result of <span data-x="extract links from
   headers">extracting links</span> from <var>response</var>'s <span
   data-x="concept-response-header-list">header list</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>linkObject</var> in <var>links</var>:</p>

    <ol>
     <li><p>Let <var>rel</var> be <var>linkObject</var>["<code
     data-x="">relation_type</code>"].</p></li>

     <li><p>Let <var>attribs</var> be <var>linkObject</var>["<code
     data-x="">target_attributes</code>"].</p></li>

     <li><p>Let <var>expectedPhase</var> be "<code data-x="">media</code>" if either "<code
     data-x="attr-img-srcset">srcset</code>", "<code
     data-x="attr-link-imagesrcset">imagesrcset</code>", or "<code
     data-x="attr-link-media">media</code>" <span data-x="map exists">exist</span> in
     <var>attribs</var>; otherwise "<code data-x="">pre-media</code>".</p></li>

     <li><p>If <var>expectedPhase</var> is not <var>phase</var>, then
     <span>continue</span>.</p></li>

     <li><p>If <var>attribs</var>["<code data-x="attr-link-media">media</code>"] <span data-x="map
     exists">exists</span> and <var>attribs</var>["<code data-x="attr-link-media">media</code>"]
     does not <span data-x="matches the environment">match the environment</span>, then
     <span>continue</span>.</p></li>

     <li>
      <p>Let <var>options</var> be a new <span>link processing options</span> with</p>

      <dl class="props">
       <dt><span data-x="link options href">href</span></dt>
       <dd><var>linkObject</var>["<code data-x="">target_uri</code>"]</dd>

       <dt><span data-x="link options base URL">base URL</span></dt>
       <dd><var>doc</var>'s <span>document base URL</span></dd>

       <dt><span data-x="link options origin">origin</span></dt>
       <dd><var>doc</var>'s <span data-x="concept-document-origin">origin</span></dd>

       <dt><span data-x="link options environment">environment</span></dt>
       <dd><var>doc</var>'s <span>relevant settings object</span></dd>

       <dt><span data-x="link options policy container">policy container</span></dt>
       <dd><var>doc</var>'s <span data-x="concept-document-policy-container">policy
       container</span></dd>

       <dt><span data-x="link options document">document</span></dt>
       <dd><var>doc</var></dd>
      </dl>
     </li>

     <li><p><span>Apply link options from parsed header attributes</span> to <var>options</var>
     given <var>attribs</var>.</p></li>

     <li><p>If <var>attribs</var>["<code data-x="attr-link-imagesrcset">imagesrcset</code>"] <span
     data-x="map exists">exists</span> and <var>attribs</var>["<code
     data-x="attr-link-imagesizes">imagesizes</code>"] <span data-x="map exists">exists</span>,
     then set <var>options</var>'s <span data-x="link options source set">source set</span> to the
     result of <span data-x="create a source set">creating a source set</span> given
     <var>linkObject</var>["<code data-x="">target_uri</code>"], <var>attribs</var>["<code
     data-x="attr-link-imagesrcset">imagesrcset</code>"], <var>attribs</var>["<code
     data-x="attr-link-imagesizes">imagesizes</code>"], and null.</p></li>

     <li><p>Run the <span>process a link header</span> steps for <var>rel</var> given
     <var>options</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>To <dfn>apply link options from parsed header attributes</dfn> to a <span>link processing
  options</span> <var>options</var> given <var>attribs</var>:</p>

  <ol>
   <li><p>If <var>attribs</var>["<code data-x="attr-link-as">as</code>"] <span data-x="map
   exists">exists</span>, then set <var>options</var>'s <span data-x="link options
   destination">destination</span> to the result of <span
   data-x="translate a preload destination">translating</span> <var>attribs</var>["<code
   data-x="attr-link-as">as</code>"].</p></li>

   <li><p>If <var>attribs</var>["<code data-x="attr-link-crossorigin">crossorigin</code>"] <span
   data-x="map exists">exists</span> and is an <span>ASCII case-insensitive</span> match for one of
   the <span>CORS settings attribute</span> <span data-x="enumerated attribute">keywords</span>,
   then set <var>options</var>'s <span data-x="link options crossorigin">crossorigin</span> to the
   <span>CORS settings attribute</span> state corresponding to that keyword.</p></li>

   <li><p>If <var>attribs</var>["<code data-x="attr-link-integrity">integrity</code>"] <span
   data-x="map exists">exists</span>, then set <var>options</var>'s <span data-x="link options
   integrity">integrity</span> to <var>attribs</var>["<code
   data-x="attr-link-integrity">integrity</code>"].</p></li>

   <li><p>If <var>attribs</var>["<code data-x="attr-link-referrerpolicy">referrerpolicy</code>"]
   <span data-x="map exists">exists</span> and is an <span>ASCII case-insensitive</span> match for
   some <span>referrer policy</span>, then set <var>options</var>'s <span data-x="link options
   referrer policy">referrer policy</span> to that <span>referrer policy</span>.</p></li>

   <li><p>If <var>attribs</var>["<code data-x="attr-nonce">nonce</code>"]
   <span data-x="map exists">exists</span>, then set <var>options</var>'s <span data-x="link
   options nonce">nonce</span> to <var>attribs</var>["<code
   data-x="attr-nonce">nonce</code>"].</p></li>

   <li><p>If <var>attribs</var>["<code data-x="attr-link-type">type</code>"] <span data-x="map
   exists">exists</span>, then set <var>options</var>'s <span data-x="link options
   type">type</span> to <var>attribs</var>["<code data-x="attr-link-type">type</code>"].</p></li>

   <li><p>If <var>attribs</var>["<code data-x="attr-link-fetchpriority">fetchpriority</code>"]
   <span data-x="map exists">exists</span> and is an <span>ASCII case-insensitive</span> match for
   a <span>fetch priority attribute</span> keyword, then set <var>options</var>'s <span
   data-x="link options fetch priority">fetch priority</span> to that <span>fetch priority
   attribute</span> keyword.</p></li>
  </ol>



  <h5>Early hints</h5>

  <p><dfn>Early hints</dfn> allow user-agents to perform some operations, such as to speculatively
  load resources that are likely to be used by the document, before the navigation request is fully
  handled by the server and a response code is served. Servers can indicate early hints by serving a
  <span data-x="concept-response">response</span> with a 103 status code before serving the final
  <span data-x="concept-response">response</span>.<ref>RFC8297</ref></p>

  <p class="note">For compatibility reasons <a
  href="https://httpwg.org/specs/rfc8297.html#security-considerations">early hints are typically
  delivered over HTTP/2 or above</a>, but for readability we use HTTP/1.1-style notation below.</p>

  <div class="example">
   <p>For example, given the following sequence of responses:</p>
   <pre>103 Early Hint
Link: &lt;/image.png&gt;; <span data-x="attr-link-rel">rel</span>=<span
data-x="rel-preload">preload</span>; <span data-x="attr-link-as">as</span>=image</pre>

   <pre>200 OK
Content-Type: text/html

&lt;!DOCTYPE html&gt;
...
&lt;img src="/image.png"&gt;</pre>

   <p>the image will start loading before the HTML content arrives.</p>
  </div>

  <p class="note">Only the first early hint response served during the navigation is handled, and it
  is discarded if it is succeeded by a cross-origin redirect.</p>

  <p>In addition to the `<code data-x="http-link">Link</code>` headers, it is possible that the 103
  response contains a <span>Content Security Policy</span> header, which is enforced when processing
  the early hint.</p>

  <div class="example">
   <p>For example, given the following sequence of responses:</p>
   <pre>103 Early Hint
Content-Security-Policy: style-src: self;
Link: &lt;/style.css&gt;; <span data-x="attr-link-rel">rel</span>=<span
data-x="rel-preload">preload</span>; <span data-x="attr-link-as">as</span>=style</pre>

   <pre>103 Early Hint
Link: &lt;/image.png&gt;; <span data-x="attr-link-rel">rel</span>=<span
data-x="rel-preload">preload</span>; <span data-x="attr-link-as">as</span>=image</pre>

   <pre>302 Redirect
Location: /alternate.html</pre>

   <pre>200 OK
Content-Security-Policy: style-src: none;
Link: &lt;/font.ttf&gt;; <span data-x="attr-link-rel">rel</span>=<span
data-x="rel-preload">preload</span>; <span data-x="attr-link-as">as</span>=font</pre>

   <p>The font and style would be loaded, and the image will be discarded, as only the first early
   hint response in the final redirect chain is respected. The late <span>Content Security
   Policy</span> header comes after the request to fetch the style has already been performed, but
   the style will not be accessible to the document.</p>
  </div>

  <p>To <dfn>process early hint headers</dfn> given a <span
  data-x="concept-response">response</span> <var>response</var> and an <span>environment</span>
  <var>reservedEnvironment</var>:</p>

  <p class="note">Early-hint `<code data-x="http-link">Link</code>` headers are always processed
  before `<code data-x="http-link">Link</code>` headers from the final <span
  data-x="concept-response">response</span>, followed by <code>link</code> elements. This is
  equivalent to prepending the contents of the early and final `<code
  data-x="http-link">Link</code>` headers to the <code>Document</code>'s <code>head</code> element,
  in respective order.</p>

  <ol>
   <li>
    <p>Let <var>earlyPolicyContainer</var> be the result of <span>creating a policy container from
    a fetch response</span> given <var>response</var> and <var>reservedEnvironment</var>.</p>

    <p class="note">This allows the early hint <span data-x="concept-response">response</span> to
    include a <span>Content Security Policy</span> which would be <span data-x="enforce the
    policy">enforced</span> when fetching the early hint <span
    data-x="concept-request">request</span>.</p>
   </li>

   <li><p>Let <var>links</var> be the result of <span data-x="extract links from
   headers">extracting links</span> from <var>response</var>'s <span
   data-x="concept-response-header-list">header list</span>.</p></li>

   <li><p>Let <var>earlyHints</var> be an empty <span>list</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>linkObject</var> in <var>links</var>:</p>

    <p class="note">The moment we receive the early hint link header, we begin <span
    data-x="concept-fetch">fetching</span> <var>earlyRequest</var>. If it comes back before the
    <code>Document</code> is created, we set <var>earlyResponse</var> to the <span
    data-x="concept-response">response</span> of that <span data-x="concept-fetch">fetch</span> and
    once the <code>Document</code> is created we commit it (by making it available in the <span>map
    of preloaded resources</span> as if it was a <code>link</code> element). If the
    <code>Document</code> is created first, the <span data-x="concept-response">response</span> is
    committed as soon as it becomes available.</p>

    <ol>
     <li><p>Let <var>rel</var> be <var>linkObject</var>["<code
     data-x="">relation_type</code>"].</p></li>

     <li>
      <p>Let <var>options</var> be a new <span>link processing options</span> with</p>

      <dl class="props">
       <dt><span data-x="link options href">href</span></dt>
       <dd><var>linkObject</var>["<code data-x="">target_uri</code>"]</dd>

       <dt><span data-x="link options initiator">initiator</span></dt>
       <dd>"<code data-x="">early-hint</code>"</dd>

       <dt><span data-x="link options base URL">base URL</span></dt>
       <dd><var>response</var>'s <span data-x="concept-response-url">URL</span></dd>

       <dt><span data-x="link options origin">origin</span></dt>
       <dd><var>response</var>'s <span data-x="concept-response-url">URL</span>'s <span
       data-x="concept-url-origin">origin</span></dd>

       <dt><span data-x="link options environment">environment</span></dt>
       <dd><var>reservedEnvironment</var></dd>

       <dt><span data-x="link options policy container">policy container</span></dt>
       <dd><var>earlyPolicyContainer</var></dd>
      </dl>
     </li>

     <li>
      <p>Let <var>attribs</var> be <var>linkObject</var>["<code
      data-x="">target_attributes</code>"].</p>

      <p class="note">Only the <code data-x="attr-link-as">as</code>, <code
      data-x="attr-link-crossorigin">crossorigin</code>, <code
      data-x="attr-link-integrity">integrity</code>, and <code data-x="attr-link-type">type</code>
      attributes are handled as part of early hint processing. The other ones, in particular <code
      data-x="attr-link-blocking">blocking</code>, <code
      data-x="attr-link-imagesrcset">imagesrcset</code>, <code
      data-x="attr-link-imagesizes">imagesizes</code>, and <code
      data-x="attr-link-media">media</code> are only applicable once a <code>Document</code> is
      created.</p>
     </li>

     <li><p><span>Apply link options from parsed header attributes</span> to <var>options</var>
     given <var>attribs</var>.</p></li>

     <li><p>Run the <span>process a link header</span> steps for <var>rel</var> given
     <var>options</var>.</p></li>

     <li><p><span data-x="list append">Append</span> <var>options</var> to
     <var>earlyHints</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>Return the following substeps given <code>Document</code> <var>doc</var>: <span
    data-x="list iterate">for each</span> <var>options</var> in <var>earlyHints</var>:</p>

    <ol>
     <li><p>If <var>options</var>'s <span
     data-x="link options on document ready">on document ready</span> is null, then set
     <var>options</var>'s <span data-x="link options document">document</span> to
     <var>doc</var>.</p></li>

     <li><p>Otherwise, call <var>options</var>'s <span
     data-x="link options on document ready">on document ready</span> with <var>doc</var>.</p></li>
    </ol>
   </li>
  </ol>

  </div>

  <h5>Providing users with a means to follow hyperlinks created using the <code>link</code>
  element</h5>

  <p id="linkui">Interactive user agents may provide users with a means to <span data-x="following
  hyperlinks">follow the hyperlinks</span> created using the <code>link</code> element, somewhere
  within their user interface. Such invocations of the <span data-x="following hyperlinks">follow
  the hyperlink</span> algorithm must set the <i
  data-x="following-userInvolvement">userInvolvement</i> argument to "<code
  data-x="uni-browser-ui">browser UI</code>". The exact interface is not defined by this
  specification, but it could include the following information (obtained from the element's
  attributes, again as defined below), in some form or another (possibly simplified), for each
  <span>hyperlink</span> created with each <code>link</code> element in the document:</p>

  <ul> <!-- the order here is the order that makes most sense for a UI -->

   <li>The relationship between this document and the resource (given by the <code
   data-x="attr-link-rel">rel</code> attribute)</li>

   <li>The title of the resource (given by the <code data-x="attr-link-title">title</code>
   attribute).</li>

   <li>The address of the resource (given by the <code data-x="attr-link-href">href</code>
   attribute).</li>

   <li>The language of the resource (given by the <code data-x="attr-link-hreflang">hreflang</code>
   attribute).</li>

   <li>The optimum media for the resource (given by the <code data-x="attr-link-media">media</code>
   attribute).</li>
  </ul>

  <p>User agents could also include other information, such as the type of the resource (as given by
  the <code data-x="attr-link-type">type</code> attribute).</p>




  <h4 id="the-meta-element">The <dfn element id="meta"><code>meta</code></dfn>
  element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Metadata content</span>.</dd>
   <dd>If the <code data-x="attr-itemprop">itemprop</code> attribute is present: <span>flow content</span>.</dd>
   <dd>If the <code data-x="attr-itemprop">itemprop</code> attribute is present: <span>phrasing content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>If the <code data-x="attr-meta-charset">charset</code> attribute is present, or if the element's <code data-x="attr-meta-http-equiv">http-equiv</code> attribute is in the <span data-x="attr-meta-http-equiv-content-type">Encoding declaration state</span>: in a <code>head</code> element.</dd>
   <dd>If the <code data-x="attr-meta-http-equiv">http-equiv</code> attribute is present but not in the <span data-x="attr-meta-http-equiv-content-type">Encoding declaration state</span>: in a <code>head</code> element.</dd>
   <dd>If the <code data-x="attr-meta-http-equiv">http-equiv</code> attribute is present but not in the <span data-x="attr-meta-http-equiv-content-type">Encoding declaration state</span>: in a <code>noscript</code> element that is a child of a <code>head</code> element.</dd>
   <dd>If the <code data-x="attr-meta-name">name</code> attribute is present: where <span>metadata content</span> is expected.</dd>
   <dd>If the <code data-x="attr-itemprop">itemprop</code> attribute is present: where <span>metadata content</span> is expected.</dd>
   <dd>If the <code data-x="attr-itemprop">itemprop</code> attribute is present: where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="concept-content-nothing">Nothing</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-meta-name">name</code></dd>
   <dd><code data-x="attr-meta-http-equiv">http-equiv</code></dd>
   <dd><code data-x="attr-meta-content">content</code></dd>
   <dd><code data-x="attr-meta-charset">charset</code></dd>
   <dd><code data-x="attr-meta-media">media</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-meta">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-meta">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLMetaElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-meta-name">name</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-meta-httpEquiv">httpEquiv</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-meta-content">content</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-meta-media">media</span>;

  // <a href="#HTMLMetaElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLMetaElement</code>.</dd>
  </dl>

  <p>The <code>meta</code> element <span>represents</span> various kinds of metadata that cannot be
  expressed using the <code>title</code>, <code>base</code>, <code>link</code>, <code>style</code>,
  and <code>script</code> elements.</p>

  <p>The <code>meta</code> element can represent document-level metadata with the <code
  data-x="attr-meta-name">name</code> attribute, pragma directives with the <code
  data-x="attr-meta-http-equiv">http-equiv</code> attribute, and the file's <span>character encoding
  declaration</span> when an HTML document is serialized to string form (e.g. for transmission over
  the network or for disk storage) with the <code data-x="attr-meta-charset">charset</code>
  attribute.</p>

  <p>Exactly one of the <code data-x="attr-meta-name">name</code>, <code
  data-x="attr-meta-http-equiv">http-equiv</code>, <code data-x="attr-meta-charset">charset</code>,
  and <code data-x="attr-itemprop">itemprop</code> attributes must be specified.</p>

  <p>If either <code data-x="attr-meta-name">name</code>, <code
  data-x="attr-meta-http-equiv">http-equiv</code>, or <code data-x="attr-itemprop">itemprop</code> is
  specified, then the <code data-x="attr-meta-content">content</code> attribute must also be
  specified. Otherwise, it must be omitted.</p>

  <p>The <dfn element-attr for="meta"><code data-x="attr-meta-charset">charset</code></dfn>
  attribute specifies the <span data-x="encoding">character encoding</span> used by the document.
  This is a <span>character encoding declaration</span>. If the attribute is present, its value must
  be an <span>ASCII case-insensitive</span> match for the string "<code data-x="">utf-8</code>".</p>

  <p class="note">The <code data-x="attr-meta-charset">charset</code> attribute on the
  <code>meta</code> element has no effect in XML documents, but is allowed in XML documents in order
  to facilitate migration to and from XML.</p>

  <p>There must not be more than one <code>meta</code> element with a <code
  data-x="attr-meta-charset">charset</code> attribute per document.</p>

  <p>The <dfn for="meta" element-attr><code data-x="attr-meta-content">content</code></dfn>
  attribute gives the value of the document metadata or pragma directive when the element is used
  for those purposes. The allowed values depend on the exact context, as described in subsequent
  sections of this specification.</p>

  <p>If a <code>meta</code> element has a <dfn element-attr for="meta"><code
  data-x="attr-meta-name">name</code></dfn> attribute, it sets document metadata. Document metadata
  is expressed in terms of name-value pairs, the <code data-x="attr-meta-name">name</code> attribute
  on the <code>meta</code> element giving the name, and the <code
  data-x="attr-meta-content">content</code> attribute on the same element giving the value. The name
  specifies what aspect of metadata is being set; valid names and the meaning of their values are
  described in the following sections. If a <code>meta</code> element has no <code
  data-x="attr-meta-content">content</code> attribute, then the value part of the metadata
  name-value pair is the empty string.</p>

  <p>The <dfn element-attr for="meta"><code data-x="attr-meta-media">media</code></dfn> attribute
  says which media the metadata applies to. The value must be a <span>valid media query list</span>.
  Unless the <code data-x="attr-meta-name">name</code> is <code
  data-x="meta-theme-color">theme-color</code>, the <code data-x="attr-meta-media">media</code>
  attribute has no effect on the processing model and must not be used by authors.</p>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLMetaElement"><code data-x="dom-meta-name">name</code></dfn>, <dfn
  attribute for="HTMLMetaElement"><code data-x="dom-meta-content">content</code></dfn>, and <dfn
  attribute for="HTMLMetaElement"><code data-x="dom-meta-media">media</code></dfn> IDL attributes
  must <span>reflect</span> the respective content attributes of the same name. The IDL attribute
  <dfn attribute for="HTMLMetaElement"><code data-x="dom-meta-httpEquiv">httpEquiv</code></dfn> must
  <span>reflect</span> the content attribute <code
  data-x="attr-meta-http-equiv">http-equiv</code>.</p>

  </div>


  <h5>Standard metadata names</h5>

  <p>This specification defines a few names for the <code data-x="attr-meta-name">name</code>
  attribute of the <code>meta</code> element.</p>

  <p>Names are case-insensitive<span class=impl>, and must be compared in an <span>ASCII
  case-insensitive</span> manner</span>.</p>

  <dl>
   <dt><dfn attr-value for="meta/name"><code
   data-x="meta-application-name">application-name</code></dfn></dt>

   <dd>
    <p>The value must be a short free-form string giving the name of the web application that the
    page represents. If the page is not a web application, the <code
    data-x="meta-application-name">application-name</code> metadata name must not be used.
    Translations of the web application's name may be given, using the <code
    data-x="attr-lang">lang</code> attribute to specify the language of each name.</p>

    <p>There must not be more than one <code>meta</code> element with a given <span>language</span>
    and where the <code data-x="attr-meta-name">name</code> attribute value is an
    <span>ASCII case-insensitive</span> match for
    <code data-x="meta-application-name">application-name</code> per document.</p>

    <div w-nodev>

    <p>User agents may use the application name in UI in preference to the page's
    <code>title</code>, since the title might include status messages and the like relevant to the
    status of the page at a particular moment in time instead of just being the name of the
    application.</p>

    <p>To find the application name to use given an ordered list of languages (e.g. British English,
    American English, and English), user agents must run the following steps:</p>

    <ol>
     <li><p>Let <var>languages</var> be the list of languages.</p></li>

     <li><p>Let <var>default language</var> be the <span>language</span> of the
     <code>Document</code>'s <span>document element</span>, if any, and if that language is not
     unknown.</p>

     <li><p>If there is a <var>default language</var>, and if it is not the same language
     as any of the languages in <var>languages</var>, append it to <var>languages</var>.</p></li>

     <li>
      <p>Let <var>winning language</var> be the first language in <var>languages</var> for which
      there is a <code>meta</code> element in the <code>Document</code> where the
      <code data-x="attr-meta-name">name</code> attribute value is an
      <span>ASCII case-insensitive</span> match for
      <code data-x="meta-application-name">application-name</code> and whose
      <span>language</span> is the language in question.</p>

      <p>If none of the languages have such a <code>meta</code> element, then return;
      there's no given application name.</p>
     </li>

     <li>
      <p>Return the value of the <code data-x="attr-meta-content">content</code> attribute of the
      first <code>meta</code> element in the <code>Document</code> in <span>tree order</span> where the
      <code data-x="attr-meta-name">name</code> attribute value is an
      <span>ASCII case-insensitive</span> match for <code data-x="meta-application-name">application-name</code>
      and whose <span>language</span> is <var>winning language</var>.</p>
     </li>
    </ol>

    <p class="note">This algorithm would be used by a browser when it needs a name for the page, for
    instance, to label a bookmark. The languages it would provide to the algorithm would be the
    user's preferred languages.</p>

    </div>
   </dd>

   <dt><dfn attr-value for="meta/name"><code data-x="meta-author">author</code></dfn></dt>

   <dd><p>The value must be a free-form string giving the name of one of the page's
   authors.</p></dd>

   <dt><dfn attr-value for="meta/name"><code data-x="meta-description">description</code></dfn></dt>

   <dd><p>The value must be a free-form string that describes the page. The value must be
   appropriate for use in a directory of pages, e.g. in a search engine. There must not be more than
   one <code>meta</code> element where the <code data-x="attr-meta-name">name</code> attribute value
   is an <span>ASCII case-insensitive</span> match for
   <code data-x="meta-description">description</code> per document.</p></dd>

   <dt><dfn attr-value for="meta/name"><code data-x="meta-generator">generator</code></dfn></dt>

   <dd>
    <p>The value must be a free-form string that identifies one of the software packages used to
    generate the document. This value must not be used on pages whose markup is not generated by
    software, e.g. pages whose markup was written by a user in a text editor.</p>

    <div class="example">

     <p>Here is what a tool called "Frontweaver" could include in its output, in the page's
     <code>head</code> element, to identify itself as the tool used to generate the page:</p>

     <pre><code class="html">&lt;meta name=generator content="Frontweaver 8.2"></code></pre>

    </div>
   </dd>

   <dt><dfn attr-value for="meta/name"><code data-x="meta-keywords">keywords</code></dfn></dt>

   <dd>
    <p>The value must be a <span>set of comma-separated tokens</span>, each of which is a keyword
    relevant to the page.</p>

    <div class="example">

     <p>This page about typefaces on British motorways uses a <code>meta</code> element to specify
     some keywords that users might use to look for the page:</p>

     <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang="en-GB">
 &lt;head>
  &lt;title>Typefaces on UK motorways&lt;/title>
  &lt;meta name="keywords" content="british,type face,font,fonts,highway,highways">
 &lt;/head>
 &lt;body>
  ...</code></pre>

    </div>

    <p class="note">Many search engines do not consider such keywords, because this feature has
    historically been used unreliably and even misleadingly as a way to spam search engine results
    in a way that is not helpful for users.</p>

    <div w-nodev>

    <p>To obtain the list of keywords that the author has specified as applicable to the page, the
    user agent must run the following steps:</p>

    <ol>
     <li><p>Let <var>keywords</var> be an empty list.</p></li>

     <li>
      <p>For each <code>meta</code> element with a <code data-x="attr-meta-name">name</code>
      attribute and a <code data-x="attr-meta-content">content</code> attribute and where the <code
      data-x="attr-meta-name">name</code> attribute value is an <span>ASCII case-insensitive</span>
      match for <code data-x="meta-keywords">keywords</code>:</p>

      <ol>
       <li><p><span data-x="split a string on commas">Split the value of the element's <code
       data-x="attr-meta-content">content</code> attribute on commas</span>.</p></li>

       <li><p>Add the resulting tokens, if any, to <var>keywords</var>.</p></li>
      </ol>
     </li>

     <li><p>Remove any duplicates from <var>keywords</var>.</p></li>

     <li><p>Return <var>keywords</var>. This is the list of keywords that the author has
     specified as applicable to the page.</p></li>
    </ol>

    <p>User agents should not use this information when there is insufficient confidence in the
    reliability of the value.</p>

    <p class="example">For instance, it would be reasonable for a content management system to use
    the keyword information of pages within the system to populate the index of a site-specific
    search engine, but a large-scale content aggregator that used this information would likely find
    that certain users would try to game its ranking mechanism through the use of inappropriate
    keywords.</p>

    </div>
   </dd>

   <dt><dfn attr-value for="meta/name"><code data-x="meta-referrer">referrer</code></dfn></dt>

   <dd>
    <p>The value must be a <span>referrer policy</span>, which defines the default <span>referrer
    policy</span> for the <code>Document</code>. <ref>REFERRERPOLICY</ref></p>

    <div w-nodev>

    <p>If any <code>meta</code> element <var>element</var> is <span data-x="node is inserted into a
    document">inserted into the document</span>, or has its <code
    data-x="attr-meta-name">name</code> or <code data-x="attr-meta-content">content</code>
    attributes changed, user agents must run the following algorithm:</p>

    <ol>
     <li><p>If <var>element</var> is not <span>in a document tree</span>, then return.</p></li>

     <li><p>If <var>element</var> does not have a <code data-x="attr-meta-name">name</code>
     attribute whose value is an <span>ASCII case-insensitive</span> match for "<code
     data-x="meta-referrer">referrer</code>", then return.</p></li>

     <li><p>If <var>element</var> does not have a <code data-x="attr-meta-content">content</code>
     attribute, or that attribute's value is the empty string, then return.</p></li>

     <li><p>Let <var>value</var> be the value of <var>element</var>'s <code
     data-x="attr-meta-content">content</code> attribute, <span>converted to ASCII
     lowercase</span>.</p></li>

     <li>
      <p>If <var>value</var> is one of the values given in the first column of the following table,
      then set <var>value</var> to the value given in the second column:</p>

      <table>
       <thead>
        <tr>
         <th>Legacy value
         <th>Referrer policy
       <tbody>
        <tr>
         <td><code data-x="">never</code>
         <td><code data-x="referrer-policy-no-referrer">no-referrer</code>
        <tr>
         <td><code data-x="">default</code>
         <td>the <span>default referrer policy</span>
        <tr>
         <td><code data-x="">always</code>
         <td><code data-x="referrer-policy-unsafe-url">unsafe-url</code>
        <tr>
         <td><code data-x="">origin-when-crossorigin</code>
         <td><code data-x="referrer-policy-origin-when-cross-origin">origin-when-cross-origin</code>
      </table>
     </li>

     <li><p>If <var>value</var> is a <span>referrer policy</span>, then set <var>element</var>'s
     <span>node document</span>'s <span data-x="concept-document-policy-container">policy
     container</span>'s <span data-x="policy-container-referrer-policy">referrer policy</span> to
     <var>policy</var>.</p></li>
    </ol>

    <p class="note">For historical reasons, unlike other standard metadata names, the processing
    model for <code data-x="meta-referrer">referrer</code> is not responsive to element removals,
    and does not use <span>tree order</span>. Only the most-recently-inserted or
    most-recently-modified <code>meta</code> element in this state has an effect.</p>

    </div>
   </dd>

   <dt><dfn attr-value for="meta/name"><code data-x="meta-theme-color">theme-color</code></dfn></dt>

   <dd>
    <p>The value must be a string that matches the CSS <span>&lt;color></span> production, defining
    a suggested color that user agents should use to customize the display of the page or of the
    surrounding user interface. For example, a browser might color the page's title bar with the
    specified value, or use it as a color highlight in a tab bar or task switcher.</p>

    <p>Within an HTML document, the <code data-x="attr-meta-media">media</code> attribute value must
    be unique amongst all the <code>meta</code> elements with their <code
    data-x="attr-meta-name">name</code> attribute value set to an <span>ASCII
    case-insensitive</span> match for <code data-x="meta-theme-color">theme-color</code>.</p>

    <div class="example">

     <p>This standard itself uses "WHATWG green" as its theme color:</p>

     <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;title>HTML Standard&lt;/title>
&lt;meta name="theme-color" content="#3c790a">
...</code></pre>

    </div>

    <p>The <code data-x="attr-meta-media">media</code> attribute may be used to describe the context
    in which the provided color should be used.</p>

    <div class="example">
     <p>If we only wanted to use "WHATWG green" as this standard's theme color in dark mode,
     we could use the <code data-x="">prefers-color-scheme</code> media feature:</p>

     <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;title>HTML Standard&lt;/title>
&lt;meta name="theme-color" content="#3c790a" media="(prefers-color-scheme: dark)">
...</code></pre>
    </div>

    <div w-nodev>

     <p>To obtain a page's theme color, user agents must run the following steps:</p>

     <ol>
      <li>
       <p>Let <var>candidate elements</var> be the list of all <code>meta</code> elements that
       meet the following criteria, in <span>tree order</span>:</p>

       <ul>
        <li><p>the element is <span>in a document tree</span>;</p></li>

        <li><p>the element has a <code data-x="attr-meta-name">name</code> attribute, whose value
        is an <span>ASCII case-insensitive</span> match for <code
        data-x="meta-theme-color">theme-color</code>; and</p></li>

        <li><p>the element has a <code data-x="attr-meta-content">content</code> attribute.</p></li>
       </ul>
      </li>

      <li>
       <p>For each <var>element</var> in <var>candidate elements</var>:</p>

       <ol>
        <li><p>If <var>element</var> has a <code data-x="attr-link-media">media</code> attribute
        and the value of <var>element</var>'s <code data-x="attr-meta-media">media</code>
        attribute does not <span data-x="matches the environment">match the environment</span>, then
        <span>continue</span>.</p></li>

        <li><p>Let <var>value</var> be the result of <span data-x="strip leading and trailing ASCII
        whitespace">stripping leading and trailing ASCII whitespace</span> from the value of
        <var>element</var>'s <code data-x="attr-meta-content">content</code> attribute.</p></li>

        <li><p>Let <var>color</var> be the result of <span data-x="parse a CSS &lt;color>
        value">parsing</span> <var>value</var>.</p></li>

        <li><p>If <var>color</var> is not failure, then return <var>color</var>.</p></li>
       </ol>
      </li>

      <li><p>Return nothing (the page has no theme color).</p></li>
     </ol>

     <p>If any <code>meta</code> elements are <span data-x="node is inserted into a
     document">inserted into the document</span> or <span data-x="node is removed from a
     document">removed from the document</span>, or existing <code>meta</code> elements have their
     <code data-x="attr-meta-name">name</code>, <code data-x="attr-meta-content">content</code>, or
     <code data-x="attr-link-media">media</code> attributes changed, or if the environment changes
     such that any <code>meta</code> element's <code data-x="attr-link-media">media</code>
     attribute's value may now or may no longer <span data-x="matches the environment">match the
     environment</span>, user agents must re-run the above algorithm and apply the result to any
     affected UI.</p>

    </div>

    <p>When using the theme color in UI, user agents may adjust it in implementation-specific ways
    to make it more suitable for the UI in question. For example, if a user agent intends to use
    the theme color as a background and display white text over it, it might use a darker variant of
    the theme color in that part of the UI, to ensure adequate contrast.</p>
   </dd>

   <dt><dfn attr-value for="meta/name"><code
   data-x="meta-color-scheme">color-scheme</code></dfn></dt>

   <dd>
    <p>To aid user agents in rendering the page background with the desired color scheme immediately
    (rather than waiting for all CSS in the page to load), a <span>'color-scheme'</span> value can
    be provided in a <code>meta</code> element.</p>

    <p>The value must be a string that matches the syntax for the CSS <span>'color-scheme'</span>
    property value. It determines the <span>page's supported color-schemes</span>.</p>

    <p>There must not be more than one <code>meta</code> element with its <code
    data-x="attr-meta-name">name</code> attribute value set to an
    <span>ASCII case-insensitive</span> match for <code
    data-x="meta-color-scheme">color-scheme</code> per document.</p>

    <div class="example">
     <p>The following declaration indicates that the page is aware of and can handle a color scheme
     with dark background colors and light foreground colors:</p>

     <pre><code class="html">&lt;meta name="color-scheme" content="dark"></code></pre>
    </div>

    <div w-nodev>

     <p>To obtain a <span>page's supported color-schemes</span>, user agents must run the following steps:</p>

     <ol>
      <li>
       <p>Let <var>candidate elements</var> be the list of all <code>meta</code> elements that
       meet the following criteria, in <span>tree order</span>:</p>

       <ul>
        <li><p>the element is <span>in a document tree</span>;</p></li>

        <li><p>the element has a <code data-x="attr-meta-name">name</code> attribute, whose value
        is an <span>ASCII case-insensitive</span> match for <code
        data-x="meta-color-scheme">color-scheme</code>; and</p></li>

        <li><p>the element has a <code data-x="attr-meta-content">content</code> attribute.</p></li>
       </ul>
      </li>

      <li>
       <p>For each <var>element</var> in <var>candidate elements</var>:</p>

       <ol>
        <li>Let <var>parsed</var> be the result of <span data-x="parse a list of component
        values">parsing a list of component values</span> given the value of <var>element</var>'s
        <code data-x="attr-meta-content">content</code> attribute.</li>

        <li>If <var>parsed</var> is a valid CSS <span>'color-scheme'</span> property value, then
        return <var>parsed</var>.</li>
       </ol>
      </li>

      <li><p>Return null.</p></li>
     </ol>

     <p>If any <code>meta</code> elements are <span data-x="node is inserted into a
     document">inserted into the document</span> or <span data-x="node is removed from a
     document">removed from the document</span>, or existing <code>meta</code> elements have their
     <code data-x="attr-meta-name">name</code> or <code data-x="attr-meta-content">content</code>
     attributes changed, user agents must re-run the above algorithm.</p>

    </div>

    <p class="note">Because these rules check successive elements until they find a match, an
    author can provide multiple such values to handle fallback for legacy user agents. Opposite to
    how CSS fallback works for properties, the multiple meta elements needs to be arranged with the
    legacy values after the newer values.</p>
   </dd>
  </dl>



  <h5>Other metadata names</h5>

  <p>Anyone can create and use their own <dfn data-x="concept-meta-extensions">extensions to the
  predefined set of metadata names</dfn>. There is no requirement to register such extensions.</p>

  <p>However, a new metadata name should not be created in any of the following cases:</p>

  <ul>
   <li><p>If either the name is a <span>URL</span>, or the value of its accompanying <code
   data-x="attr-meta-content">content</code> attribute is a <span>URL</span>; in those cases,
   registering it as an <span data-x="concept-rel-extensions">extension to the predefined set of
   link types</span> is encouraged (rather than creating a new metadata name).</p></li>

   <li><p>If the name is for something expected to have processing requirements in user agents; in
   that case it ought to be standardized.</p></li>
  </ul>

  <p>Also, before creating and using a new metadata name, consulting the <a
  href="https://wiki.whatwg.org/wiki/MetaExtensions">WHATWG Wiki MetaExtensions page</a> is
  encouraged — to avoid choosing a metadata name that's already in use, and to avoid duplicating the
  purpose of any metadata names that are already in use, and to avoid new standardized names
  clashing with your chosen name. <ref>WHATWGWIKI</ref></p>

  <p>Anyone is free to edit the WHATWG Wiki MetaExtensions page at any time to add a metadata name.
  New metadata names can be specified with the following information:</p>

  <dl>
   <dt>Keyword</dt>

   <dd><p>The actual name being defined. The name should not be confusingly similar to any other
   defined name (e.g. differing only in case).</p></dd>


   <dt>Brief description</dt>

   <dd><p>A short non-normative description of what the metadata name's meaning is, including the
   format the value is required to be in.</p></dd>


   <dt>Specification</dt>

   <dd>A link to a more detailed description of the metadata name's semantics and requirements. It
   could be another page on the wiki, or a link to an external page.</dd>


   <dt>Synonyms</dt>

   <dd><p>A list of other names that have exactly the same processing requirements. Authors should
   not use the names defined to be synonyms (they are only intended to allow user agents to support
   legacy content). Anyone may remove synonyms that are not used in practice; only names that need to
   be processed as synonyms for compatibility with legacy content are to be registered in this
   way.</p></dd>


   <dt>Status</dt>

   <dd>
    <p>One of the following:</p>

    <dl>
     <dt>Proposed</dt>

     <dd>The name has not received wide peer review and approval. Someone has proposed it and is, or
     soon will be, using it.</dd>

     <dt>Ratified</dt>

     <dd>The name has received wide peer review and approval. It has a specification that
     unambiguously defines how to handle pages that use the name, including when they use it in
     incorrect ways.</dd>

     <dt>Discontinued</dt>

     <dd>The metadata name has received wide peer review and it has been found wanting. Existing
     pages are using this metadata name, but new pages should avoid it. The "brief description" and
     "specification" entries will give details of what authors should use instead, if anything.</dd>
    </dl>

    <p>If a metadata name is found to be redundant with existing values, it should be removed and
    listed as a synonym for the existing value.</p>

    <p>If a metadata name is added in the "proposed" state for a period of a month or more without
    being used or specified, then it may be removed from the WHATWG Wiki MetaExtensions page.</p>

    <p>If a metadata name is added with the "proposed" status and found to be redundant with
    existing values, it should be removed and listed as a synonym for the existing value. If a
    metadata name is added with the "proposed" status and found to be harmful, then it should be
    changed to "discontinued" status.</p>

    <p>Anyone can change the status at any time, but should only do so in accordance with the
    definitions above.</p>
   </dd>
  </dl>



  <h5>Pragma directives</h5>

  <p>When the <dfn for="meta" element-attr><code
  data-x="attr-meta-http-equiv">http-equiv</code></dfn> attribute is specified on a
  <code>meta</code> element, the element is a pragma directive.</p>

  <p>The <code data-x="attr-meta-http-equiv">http-equiv</code> attribute is an <span>enumerated
  attribute</span> with the following keywords and states:</p>

  <table class="yesno" id="table-http-equiv">
   <thead>
    <tr>
     <th>Keyword
     <th w-nodev>Conforming
     <th>State
     <th>Brief description
   <tbody>
    <tr w-nodev>
     <td><dfn attr-value for="meta/http-equiv"><code
     data-x="attr-meta-http-equiv-keyword-content-language">content-language</code></dfn>
     <td class="no">No
     <td><span data-x="attr-meta-http-equiv-content-language">Content language</span>
     <td>Sets the <span data-x="pragma-set default language">pragma-set default language</span>.
    <tr>
     <td><dfn attr-value for="meta/http-equiv"><code
     data-x="attr-meta-http-equiv-keyword-content-type">content-type</code></dfn>
     <td w-nodev>
     <td><span data-x="attr-meta-http-equiv-content-type">Encoding declaration</span>
     <td>An alternative form of setting the <code data-x="attr-meta-charset">charset</code>.
    <tr>
     <td><dfn attr-value for="meta/http-equiv"><code
     data-x="attr-meta-http-equiv-keyword-default-style">default-style</code></dfn>
     <td w-nodev>
     <td><span data-x="attr-meta-http-equiv-default-style">Default style</span>
     <td>Sets the <span data-x="CSS style sheet set name">name</span> of the default
     <span>CSS style sheet set</span>.
    <tr>
     <td><dfn attr-value for="meta/http-equiv"><code
     data-x="attr-meta-http-equiv-keyword-refresh">refresh</code></dfn>
     <td w-nodev>
     <td><span data-x="attr-meta-http-equiv-refresh">Refresh</span>
     <td>Acts as a timed redirect.
    <tr w-nodev>
     <td><dfn attr-value for="meta/http-equiv"><code
     data-x="attr-meta-http-equiv-keyword-set-cookie">set-cookie</code></dfn>
     <td class="no">No
     <td><span data-x="attr-meta-http-equiv-set-cookie">Set-Cookie</span>
     <td>Has no effect.
    <tr>
     <td><dfn attr-value for="meta/http-equiv"><code
     data-x="attr-meta-http-equiv-keyword-x-ua-compatible">x-ua-compatible</code></dfn>
     <td w-nodev>
     <td><span data-x="attr-meta-http-equiv-x-ua-compatible">X-UA-Compatible</span>
     <td>In practice, encourages Internet Explorer to more closely follow the specifications.
    <tr>
     <td><dfn attr-value for="meta/http-equiv"><code
     data-x="attr-meta-http-equiv-keyword-content-security-policy">content-security-policy</code></dfn>
     <td w-nodev>
     <td><span data-x="attr-meta-http-equiv-content-security-policy">Content security policy</span>
     <td><span data-x="enforce the policy">Enforces</span> a <span>Content Security
     Policy</span> on a <code>Document</code>.
  </table>

<!--
Content-Script-Type and Content-Style-Type don't do anything (and are non-conforming).

DATA: According to Henri's validator stats, here are common
non-conforming values used by people who validate (sample of 400
pages):

COUNT ERROR
   20 Bad value "Content-Language" for attribute "http-equiv" on element "meta".
   17 Bad value "Content-Style-Type" for attribute "http-equiv" on element "meta".
   12 Bad value "Expires" for attribute "http-equiv" on element "meta".
   11 Bad value "Pragma" for attribute "http-equiv" on element "meta".
   11 Bad value "imagetoolbar" for attribute "http-equiv" on element "meta".
   10 Bad value "Content-Script-Type" for attribute "http-equiv" on element "meta".
   10 Bad value "content-language" for attribute "http-equiv" on element "meta".
 [...]
https://hsivonen.com/test/moz/analysis.txt

Here's some more data, this time from Philip`, on a sample of 15k pages.

  COUNT VALUE
   1181 Content-Language
    430 Content-Style-Type
    342 imagetoolbar
    276 content-language
    269 Pragma
    260 expires
    227 Expires
    211 pragma
    146 Content-Script-Type
    132 keywords
    119 Page-Enter
    116 description
    106 reply-to
    102 Cache-Control
   [...]
https://philip.html5.org/data/meta-http-equiv.txt

These numbers are low; further study is required to establish what
people expect to have work and what is necessary.
-->

  <div w-nodev>

  <p>When a <code>meta</code> element is <span data-x="node is inserted into a document">inserted
  into the document</span>, if its <code data-x="attr-meta-http-equiv">http-equiv</code> attribute is
  present and represents one of the above states, then the user agent must run the algorithm
  appropriate for that state, as described in the following list:</p>

  </div>

  <dl>
   <dt w-nodev><dfn data-x="attr-meta-http-equiv-content-language">Content language state</dfn>
   (<code data-x="">http-equiv="<code data-x="attr-meta-http-equiv-keyword-content-language">content-language</code>"</code>)

   <dd w-nodev>

    <p class="note">This feature is non-conforming. Authors are encouraged to use the <code
    data-x="attr-lang">lang</code> attribute instead.</p>

    <p>This pragma sets the <dfn>pragma-set default language</dfn>. Until such a pragma is
    successfully processed, there is no <span>pragma-set default language</span>.</p>

    <!-- https://www.hixie.ch/tests/adhoc/html/meta/content-language/ -->

    <ol>
     <li><p>If the <code>meta</code> element has no <code data-x="attr-meta-content">content</code>
     attribute, then return.</p></li>

     <li><p>If the element's <code data-x="attr-meta-content">content</code> attribute contains a
     U+002C COMMA character (,) then return.</p></li><!-- if you remove this,
     un-comment-out the corresponding bits in the step below. -->

     <li><p>Let <var>input</var> be the value of the element's <code
     data-x="attr-meta-content">content</code> attribute.</p></li>

     <li><p>Let <var>position</var> point at the first character of <var>input</var>.</p></li>

     <li><p><span>Skip ASCII whitespace</span> within <var>input</var> given
     <var>position</var>.</p></li>

     <li><p><span>Collect a sequence of code points</span> that are not <span>ASCII
     whitespace</span> from <var>input</var> given <var>position</var>.</p></li>

     <li><p>Let <var>candidate</var> be the string that resulted from the previous
     step.</p></li>

     <li><p>If <var>candidate</var> is the empty string, return.</p></li>

     <li>
      <p>Set the <span>pragma-set default language</span> to <var>candidate</var>.</p>

      <p class="note">If the value consists of multiple space-separated tokens, tokens after the
      first are ignored.</p>
     </li>
    </ol>

    <p class="note">This pragma is almost, but not quite, entirely unlike the HTTP `<code
    data-x="http-content-language">Content-Language</code>` header of the same name.
    <ref>HTTP</ref></p>
   </dd>

   <dt><dfn data-x="attr-meta-http-equiv-content-type">Encoding declaration state</dfn> (<code data-x="">http-equiv="<code data-x="attr-meta-http-equiv-keyword-content-type">content-type</code>"</code>)

   <dd>
    <p>The <span data-x="attr-meta-http-equiv-content-type">Encoding declaration state</span> is
    just an alternative form of setting the <code data-x="attr-meta-charset">charset</code>
    attribute: it is a <span>character encoding declaration</span>. <span w-nodev>This state's user
    agent requirements are all handled by the parsing section of the specification.</span></p>

    <p>For <code>meta</code> elements with an <code data-x="attr-meta-http-equiv">http-equiv</code>
    attribute in the <span data-x="attr-meta-http-equiv-content-type">Encoding declaration
    state</span>, the <code data-x="attr-meta-content">content</code> attribute must have a value
    that is an <span>ASCII case-insensitive</span> match for a string that consists of:
    "<code data-x="">text/html;</code>", optionally followed by any number of <span>ASCII
    whitespace</span>, followed by "<code data-x="">charset=utf-8</code>".</p>

    <p>A document must not contain both a <code>meta</code> element with an <code
    data-x="attr-meta-http-equiv">http-equiv</code> attribute in the <span
    data-x="attr-meta-http-equiv-content-type">Encoding declaration state</span> and a
    <code>meta</code> element with the <code data-x="attr-meta-charset">charset</code> attribute
    present.</p>

    <p>The <span data-x="attr-meta-http-equiv-content-type">Encoding declaration state</span> may be
    used in <span>HTML documents</span>, but elements with an <code
    data-x="attr-meta-http-equiv">http-equiv</code> attribute in that state must not be used in
    <span>XML documents</span>.</p>
   </dd>

   <dt><dfn data-x="attr-meta-http-equiv-default-style">Default style state</dfn> (<code data-x="">http-equiv="<code data-x="attr-meta-http-equiv-keyword-default-style">default-style</code>"</code>)

   <dd>
    <p>This pragma sets the <span data-x="CSS style sheet set name">name</span> of the default
    <span>CSS style sheet set</span>.</p>

    <div w-nodev>

    <ol>
     <li><p>If the <code>meta</code> element has no <code data-x="attr-meta-content">content</code>
     attribute, or if that attribute's value is the empty string, then return.</p></li>

     <li><p><span>Change the preferred CSS style sheet set name</span> with the name being the value
     of the element's <code data-x="attr-meta-content">content</code> attribute.
     <ref>CSSOM</ref></p></li>
    </ol>

    </div>
   </dd>

   <dt><dfn data-x="attr-meta-http-equiv-refresh">Refresh state</dfn> (<code data-x="">http-equiv="<code data-x="attr-meta-http-equiv-keyword-refresh">refresh</code>"</code>)

   <dd>
    <p>This pragma acts as a timed redirect.</p>

    <p>A <code>Document</code> object has an associated <dfn for="Document">will declaratively
    refresh</dfn> (a boolean). It is initially false.</p>

    <div w-nodev>

    <ol>
     <li><p>If the <code>meta</code> element has no <code data-x="attr-meta-content">content</code>
     attribute, or if that attribute's value is the empty string, then return.</p></li>

     <li><p>Let <var>input</var> be the value of the element's <code
     data-x="attr-meta-content">content</code> attribute.</p></li>

     <li><p>Run the <span>shared declarative refresh steps</span> with the <code>meta</code>
     element's <span>node document</span>, <var>input</var>, and the <code>meta</code>
     element.</p></li>
    </ol>

    <p>The <dfn>shared declarative refresh steps</dfn>, given a <code>Document</code> object
    <var>document</var>, string <var>input</var>, and optionally a <code>meta</code> element
    <var>meta</var>, are as follows:</p>

    <ol>
     <li><p>If <var>document</var>'s <span>will declaratively refresh</span> is true, then
     return.</p></li>

     <li><p>Let <var>position</var> point at the first <span>code point</span> of
     <var>input</var>.</p></li>

     <li><p><span>Skip ASCII whitespace</span> within <var>input</var> given
     <var>position</var>.</p></li>

     <li><p>Let <var>time</var> be 0.</p></li>

     <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
     <var>input</var> given <var>position</var>, and let the result be
     <var>timeString</var>.</p></li>

     <li>
      <p>If <var>timeString</var> is the empty string, then:</p>

      <ol>
       <li><p>If the <span>code point</span> in <var>input</var> pointed to by <var>position</var>
       is not U+002E (.), then return.</p></li>
      </ol>
     </li>

     <li><p>Otherwise, set <var>time</var> to the result of parsing <var>timeString</var> using the
     <span>rules for parsing non-negative integers</span>.</p></li>

     <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> and
     U+002E FULL STOP characters (.) from <var>input</var> given <var>position</var>. Ignore any
     collected characters.</p></li>

     <li><p>Let <var>urlRecord</var> be <var>document</var>'s <span
     data-x="concept-document-URL">URL</span>.</p></li>

     <li>
      <p>If <var>position</var> is not past the end of <var>input</var>, then:</p>

      <ol>
       <li><p>If the <span>code point</span> in <var>input</var> pointed to by <var>position</var>
       is not U+003B (;), U+002C (,), or <span>ASCII whitespace</span>, then return.</p></li>

       <li><p><span>Skip ASCII whitespace</span> within <var>input</var> given
       <var>position</var>.</p></li>

       <li><p>If the <span>code point</span> in <var>input</var> pointed to by <var>position</var>
       is U+003B (;) or U+002C (,), then advance <var>position</var> to the next <span>code
       point</span>.</p></li>

       <li><p><span>Skip ASCII whitespace</span> within <var>input</var> given
       <var>position</var>.</p></li>
      </ol>
     </li>

     <li>
      <p>If <var>position</var> is not past the end of <var>input</var>, then:</p>

      <ol>
       <li><p>Let <var>urlString</var> be the substring of <var>input</var> from the
       <span>code point</span> at <var>position</var> to the end of the string.</p></li>

       <li><p>If the <span>code point</span> in <var>input</var> pointed to by <var>position</var>
       is U+0055 (U) or U+0075 (u), then advance <var>position</var> to the next <span>code
       point</span>. Otherwise, jump to the step labeled <i>skip quotes</i>.</p></li>

       <li><p>If the <span>code point</span> in <var>input</var> pointed to by <var>position</var>
       is U+0052 (R) or U+0072 (r), then advance <var>position</var> to the next <span>code
       point</span>. Otherwise, jump to the step labeled <i>parse</i>.</p></li>

       <li><p>If the <span>code point</span> in <var>input</var> pointed to by <var>position</var>
       is U+004C (L) or U+006C (l), then advance <var>position</var> to the next <span>code
       point</span>. Otherwise, jump to the step labeled <i>parse</i>.</p></li>

       <li><p><span>Skip ASCII whitespace</span> within <var>input</var> given
       <var>position</var>.</p></li>

       <li><p>If the <span>code point</span> in <var>input</var> pointed to by <var>position</var>
       is U+003D (=), then advance <var>position</var> to the next <span>code point</span>.
       Otherwise, jump to the step labeled <i>parse</i>.</p></li>

       <li><p><span>Skip ASCII whitespace</span> within <var>input</var> given
       <var>position</var>.</p></li>

       <li><p><i>Skip quotes</i>: If the <span>code point</span> in <var>input</var> pointed to by
       <var>position</var> is U+0027 (') or U+0022 ("), then let <var>quote</var> be that <span>code
       point</span>, and advance <var>position</var> to the next <span>code point</span>. Otherwise,
       let <var>quote</var> be the empty string.</p></li>

       <li><p>Set <var>urlString</var> to the substring of <var>input</var> from the <span>code
       point</span> at <var>position</var> to the end of the string.</p></li>

       <li><p>If <var>quote</var> is not the empty string, and there is a <span>code point</span> in
       <var>urlString</var> equal to <var>quote</var>, then truncate <var>urlString</var> at that
       <span>code point</span>, so that it and all subsequent <span data-x="code point">code
       points</span> are removed.</p>

       <li><p><i>Parse</i>: Set <var>urlRecord</var> to the result of <span>encoding-parsing a
       URL</span> given <var>urlString</var>, relative to <var>document</var>.</p></li>

       <li><p>If <var>urlRecord</var> is failure, then return.</p></li>
      </ol>
     </li>

     <li><p>Set <var>document</var>'s <span>will declaratively refresh</span> to true.</p></li>

     <li>
      <p>Perform one or more of the following steps:</p>

      <ul>
       <li>
        <p>After the refresh has come due (as defined below), if the user has not canceled the
        redirect and, if <var>meta</var> is given, <var>document</var>'s <span>active sandboxing
        flag set</span> does not have the <span>sandboxed automatic features browsing context
        flag</span> set, then <span data-x="navigate">navigate</span><!--DONAV meta refresh-->
        <var>document</var>'s <span>node navigable</span> to <var>urlRecord</var> using
        <var>document</var>, with <i data-x="navigation-hh">historyHandling</i> set to "<code
        data-x="NavigationHistoryBehavior-replace">replace</code>".</p>

        <p>For the purposes of the previous paragraph, a refresh is said to have come due as soon as
        the <em>later</em> of the following two conditions occurs:</p>

        <ul>
         <li>At least <var>time</var> seconds have elapsed since <var>document</var>'s
         <span>completely loaded time</span>, adjusted to take into account user or user agent
         preferences.</li>
         <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=774 -->

         <li>If <var>meta</var> is given, at least <var>time</var> seconds have elapsed since
         <var>meta</var> was <span data-x="node is inserted into a document">inserted into the
         document</span> <var>document</var>, adjusted to take into account user or user agent
         preferences.</li>
         <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=772 -->
        </ul>

        <p class="note">It is important to use <var>document</var> here, and not <var>meta</var>'s
        <span>node document</span>, as that might have changed between the initial set of steps and
        the refresh coming due and <var>meta</var> is not always given (in case of the HTTP
        `<code>Refresh</code>` header).</p>
       </li>

       <li><p>Provide the user with an interface that, when selected, <span
       data-x="navigate">navigates</span><!--DONAV meta refresh--> <var>document</var>'s <span>node
       navigable</span> to <var>urlRecord</var> using <var>document</var>.</p></li>

       <li><p>Do nothing.</p></li>
      </ul>

      <p>In addition, the user agent may, as with anything, inform the user of any and all aspects
      of its operation, including the state of any timers, the destinations of any timed redirects,
      and so forth.</p>
     </li>
    </ol>

    </div>

    <p id="conformance-attr-meta-http-equiv-refresh">For <code>meta</code> elements with an <code
    data-x="attr-meta-http-equiv">http-equiv</code> attribute in the <span
    data-x="attr-meta-http-equiv-refresh">Refresh state</span>, the <code
    data-x="attr-meta-content">content</code> attribute must have a value consisting either of:</p>

    <ul>
     <li>just a <span>valid non-negative integer</span>, or</li>

     <li>a <span>valid non-negative integer</span>, followed by a U+003B SEMICOLON character
     (;)<!--(add this at some point once the browsers all support it) or a U+002C COMMA character
     (,)-->, followed by one or more <span>ASCII whitespace</span>, followed by a substring that is
     an <span>ASCII case-insensitive</span> match for the string "<code data-x="">URL</code>",
     followed by a U+003D EQUALS SIGN character (=), followed by a <span>valid URL string</span>
     that does not start with a literal U+0027 APOSTROPHE (') or U+0022 QUOTATION MARK (")
     character.</li>
    </ul>

    <p>In the former case, the integer represents a number of seconds before the page is to be
    reloaded; in the latter case the integer represents a number of seconds before the page is to be
    replaced by the page at the given <span>URL</span>.</p>

    <div class="example">
     <p>A news organization's front page could include the following markup in the page's
     <code>head</code> element, to ensure that the page automatically reloads from the server every
     five minutes:</p>

     <pre><code class="html">&lt;meta http-equiv="Refresh" content="300"></code></pre>
    </div>

    <div class="example">
     <p>A sequence of pages could be used as an automated slide show by making each page refresh to
     the next page in the sequence, using markup such as the following:</p>

     <pre><code class="html">&lt;meta http-equiv="Refresh" content="20; URL=page4.html"></code></pre>
    </div>
   </dd>

   <dt w-nodev><dfn data-x="attr-meta-http-equiv-set-cookie">Set-Cookie state</dfn> (<code data-x="">http-equiv="<code data-x="attr-meta-http-equiv-keyword-set-cookie">set-cookie</code>"</code>)

   <dd w-nodev>
    <p>This pragma is non-conforming and has no effect.</p>

    <p>User agents are required to ignore this pragma.</p>
   </dd>


   <dt><dfn data-x="attr-meta-http-equiv-x-ua-compatible">X-UA-Compatible state</dfn> (<code data-x="">http-equiv="<code data-x="attr-meta-http-equiv-keyword-x-ua-compatible">x-ua-compatible</code>"</code>)

   <dd>
    <p>In practice, this pragma encourages Internet Explorer to more closely follow the
    specifications.</p>

    <p>For <code>meta</code> elements with an <code data-x="attr-meta-http-equiv">http-equiv</code>
    attribute in the <span data-x="attr-meta-http-equiv-x-ua-compatible">X-UA-Compatible state</span>, the
    <code data-x="attr-meta-content">content</code> attribute must have a value that is an
    <span>ASCII case-insensitive</span> match for the string "<code data-x="">IE=edge</code>".</p>

    <p>User agents are required to ignore this pragma.</p>
   </dd>


   <dt><dfn export data-x="attr-meta-http-equiv-content-security-policy">Content security policy state</dfn> (<code data-x="">http-equiv="<code data-x="attr-meta-http-equiv-keyword-content-security-policy">content-security-policy</code>"</code>)

   <dd>
    <p>This pragma <span data-x="enforce the policy">enforces</span> a <span>Content Security
    Policy</span> on a <code>Document</code>. <ref>CSP</ref></p>

    <div w-nodev>

    <ol>
     <li><p>If the <code>meta</code> element is not a child of a <code>head</code> element,
     return.</p></li>

     <li><p>If the <code>meta</code> element has no <code data-x="attr-meta-content">content</code>
     attribute, or if that attribute's value is the empty string, then return.</p></li>

     <li><p>Let <var>policy</var> be the result of executing Content Security Policy's <span>parse
     a serialized Content Security Policy</span> algorithm on the <code>meta</code> element's
     <code data-x="attr-meta-content">content</code> attribute's value, with a source of "meta",
     and a disposition of "enforce".</p></li>

     <li><p>Remove all occurrences of the <code
     data-x="report-uri directive">report-uri</code>, <code
     data-x="frame-ancestors directive">frame-ancestors</code>, and <code
     data-x="sandbox directive">sandbox</code> <span data-x="Content Security Policy
     directive">directives</span> from <var>policy</var>.</p></li>

     <li><p><span>Enforce the policy</span> <var>policy</var>.</p></li>
    </ol>

    </div>

    <p>For <code>meta</code> elements with an <code data-x="attr-meta-http-equiv">http-equiv</code>
    attribute in the <span data-x="attr-meta-http-equiv-content-security-policy">Content security
    policy state</span>, the <code data-x="attr-meta-content">content</code> attribute must have a
    value consisting of a <span data-x="Content Security Policy syntax">valid Content Security
    Policy</span>, but must not contain any <code data-x="report-uri directive">report-uri</code>,
    <code data-x="frame-ancestors directive">frame-ancestors</code>, or <code data-x="sandbox
    directive">sandbox</code> <span data-x="Content Security Policy directive">directives</span>.
    The <span>Content Security Policy</span> given in the <code
    data-x="attr-meta-content">content</code> attribute will be <span data-x="enforce the
    policy">enforced</span> upon the current document. <ref>CSP</ref></p>

    <p class="note">At the time of inserting the <code>meta</code> element to the document, it is
    possible that some resources have already been fetched. For example, images might be stored in
    the <span>list of available images</span> prior to dynamically inserting a <code>meta</code>
    element with an <code data-x="attr-meta-http-equiv">http-equiv</code> attribute in the <span
    data-x="attr-meta-http-equiv-content-security-policy">Content security policy state</span>.
    Resources that have already been fetched are not guaranteed to be blocked by a <span>Content
    Security Policy</span> that's <span data-x="enforce the policy">enforced</span> late.</p>

    <div class="example">

     <p>A page might choose to mitigate the risk of cross-site scripting attacks by preventing the
     execution of inline JavaScript, as well as blocking all plugin content, using a policy such
     as the following:

     <pre><code class="html">&lt;meta http-equiv="Content-Security-Policy" content="script-src 'self'; object-src 'none'"></code></pre>

    </div>
   </dd>
  </dl>

  <p>There must not be more than one <code>meta</code> element with any particular state in the
  document at a time.</p>


  <h5 id="charset">Specifying the document's character encoding</h5>

  <!-- READ ME WHEN EDITING: if we ever move this to the "writing HTML" section, then we have to
  duplicate the requirements in the parsing section for conformance checkers, and we have to make
  sure that the requirements for charset="" apply even in XML, for the <meta charset=""> polyglot
  hack. -->

  <p>A <dfn>character encoding declaration</dfn> is a mechanism by which the <span
  data-x="encoding">character encoding</span> used to store or transmit a document is specified.</p>

  <p>The Encoding standard requires use of the <span>UTF-8</span> <span data-x="encoding">character
  encoding</span> and requires use of the "<code data-x="">utf-8</code>" <span>encoding label</span>
  to identify it. Those requirements necessitate that the document's <span>character encoding
  declaration</span>, if it exists, specifies an <span>encoding label</span> using an <span>ASCII
  case-insensitive</span> match for "<code data-x="">utf-8</code>". Regardless of whether a
  <span>character encoding declaration</span> is present or not, the actual <span data-x="document's
  character encoding">character encoding</span> used to encode the document must be
  <span>UTF-8</span>. <ref>ENCODING</ref></p>

  <p w-nodev>To enforce the above rules, authoring tools must default to using <span>UTF-8</span>
  for newly-created documents.</p>

  <p>The following restrictions also apply:</p>

  <ul>
   <li>The character encoding declaration must be serialized without the use of <span
   data-x="syntax-charref">character references</span> or character escapes of any kind.</li>

   <li id="charset1024"><span data-x="" id="charset512">The element containing the character encoding
   declaration must be serialized completely within the first 1024 bytes of the
   document.</span></li> <!-- span is for historical reasons, to keep an old ID alive -->
  </ul>

  <p>In addition, due to a number of restrictions on <code>meta</code> elements, there can only be
  one <code>meta</code>-based character encoding declaration per document.</p> <!-- conformance
  criteria for this one are given in the XML spec, the <meta> section just after defining
  charset="", and the character encoding pragma section. Note that you _can_ have two character
  encoding declarations per document, if the document is an XML document and one is an XML
  declaration, the other is <meta charset>, and the encoding is UTF-8. -->

  <p>If an <span data-x="HTML documents">HTML document</span> does not start with a BOM, and its
  <span>encoding</span> is not explicitly given by <span data-x="Content-Type">Content-Type
  metadata</span>, and the document is not <span>an <code>iframe</code> <code
  data-x="attr-iframe-srcdoc">srcdoc</code> document</span>, then the encoding must be specified
  using a <code>meta</code> element with a <code data-x="attr-meta-charset">charset</code> attribute
  or a <code>meta</code> element with an <code data-x="attr-meta-http-equiv">http-equiv</code>
  attribute in the <span data-x="attr-meta-http-equiv-content-type">Encoding declaration
  state</span>.</p>

  <div class="note">
   <p>A character encoding declaration is required (either in the <span
   data-x="Content-Type">Content-Type metadata</span> or explicitly in the file) even when all
   characters are in the ASCII range, because a character encoding is needed to process non-ASCII
   characters entered by the user in forms, in URLs generated by scripts, and so forth.</p>

   <p>Using non-UTF-8 encodings can have unexpected results on form submission and URL encodings,
   which use the <span>document's character encoding</span> by default.</p>
  </div>

  <p>If the document is <span>an <code>iframe</code> <code data-x="attr-iframe-srcdoc">srcdoc</code>
  document</span>, the document must not have a <span>character encoding declaration</span>. (In
  this case, the source is already decoded, since it is part of the document that contained the
  <code>iframe</code>.)</p>

  <p>In XML, the XML declaration should be used for inline character encoding information, if
  necessary.</p>

  <div class="example">

   <p>In HTML, to declare that the character encoding is <span>UTF-8</span>, the author could
   include the following markup near the top of the document (in the <code>head</code> element):</p>

   <pre><code class="html">&lt;meta charset="utf-8"></code></pre>

   <p>In XML, the XML declaration would be used instead, at the very top of the markup:</p>

   <pre><code class="html">&lt;?xml version="1.0" encoding="utf-8"?></code></pre>

  </div>



  <h4>The <dfn element><code>style</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Metadata content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>metadata content</span> is expected.</dd>
   <dd>In a <code>noscript</code> element that is a child of a <code>head</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="text content">Text</span> that gives a <span>conformant style sheet</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-style-media">media</code></dd>
   <dd><code data-x="attr-style-blocking">blocking</code></dd>
   <dd>Also, the <code data-x="attr-style-title">title</code> attribute <span data-x="attr-style-title">has special semantics</span> on this element.</dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-style">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-style">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLStyleElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  attribute boolean <span data-x="dom-style-disabled">disabled</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-style-media">media</span>;
  [SameObject, PutForwards=<span data-x="dom-DOMTokenList-value">value</span>] readonly attribute <span>DOMTokenList</span> <span data-x="dom-style-blocking">blocking</span>;

  // <a href="#HTMLStyleElement-partial">also has obsolete members</a>
};
<span>HTMLStyleElement</span> includes <span>LinkStyle</span>;</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLStyleElement</code>.</dd>
  </dl>

  <p>The <code>style</code> element allows authors to embed CSS style sheets in their documents.
  The <code>style</code> element is one of several inputs to the styling processing
  model. The element does not <span data-x="represents">represent</span> content for the
  user.</p>

  <p>The <dfn attribute for="HTMLStyleElement"><code
  data-x="dom-style-disabled">disabled</code></dfn> getter steps are:</p>

  <ol>
   <li><p>If <span>this</span> does not have an <span>associated CSS style sheet</span>, return
   false.</p></li>

   <li><p>If <span>this</span>'s <span>associated CSS style sheet</span>'s <span
   data-x="concept-css-style-sheet-disabled-flag">disabled flag</span> is set, return true.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  <p>The <code data-x="dom-style-disabled">disabled</code> setter steps are:</p>

  <ol>
   <li><p>If <span>this</span> does not have an <span>associated CSS style sheet</span>,
   return.</p></li>

   <li><p>If the given value is true, set <span>this</span>'s <span>associated CSS style
   sheet</span>'s <span data-x="concept-css-style-sheet-disabled-flag">disabled flag</span>.
   Otherwise, unset <span>this</span>'s <span>associated CSS style sheet</span>'s <span
   data-x="concept-css-style-sheet-disabled-flag">disabled flag</span>.</p></li>
  </ol>

  <div class="example">
   <p>Importantly, <code data-x="dom-style-disabled">disabled</code> attribute assignments only take
   effect when the <code>style</code> element has an <span>associated CSS style sheet</span>:</p>

   <pre><code class="js">const style = document.createElement('style');
style.disabled = true;
style.textContent = 'body { background-color: red; }';
document.body.append(style);
console.log(style.disabled); // false</code></pre>
  </div>

  <p>The <dfn element-attr for="style"><code data-x="attr-style-media">media</code></dfn> attribute
  says which media the styles apply to. The value must be a <span>valid media query list</span>.
  <span w-nodev>The user agent must apply the styles when the <code
  data-x="attr-style-media">media</code> attribute's value <span>matches the environment</span> and
  the other relevant conditions apply, and must not apply them otherwise.</span></p><!-- note
  similar text in <link> element section -->

  <div w-nodev>

  <p class="note">The styles might be further limited in scope, e.g. in CSS with the use of <code
  data-x="">@media</code> blocks. This specification does not override such further restrictions or
  requirements.</p><!-- note similar text in <link> element section -->

  </div>

  <p id="style-default-media">The default, if the <code data-x="attr-style-media">media</code>
  attribute is omitted, is "<code data-x="">all</code>", meaning that by default styles apply to all
  media.</p>

  <p>The <dfn element-attr for="style"><code data-x="attr-style-blocking">blocking</code></dfn>
  attribute is a <span>blocking attribute</span>.</p>

  <p id="title-on-style">The <dfn element-attr for="style"><code
  data-x="attr-style-title">title</code></dfn> attribute on <code>style</code> elements defines
  <span data-x="CSS style sheet set">CSS style sheet sets</span>. If the <code>style</code> element
  has no <code data-x="attr-style-title">title</code> attribute, then it has no title; the <code
  data-x="attr-title">title</code> attribute of ancestors does not apply to the <code>style</code>
  element. If the <code>style</code> element is not <span>in a document tree</span>, then the <code
  data-x="attr-style-title">title</code> attribute is ignored. <ref>CSSOM</ref></p>

  <p class="note">The <code data-x="attr-style-title">title</code> attribute on <code>style</code>
  elements, like the <code data-x="attr-link-title">title</code> attribute on <code>link</code>
  elements, differs from the global <code data-x="attr-title">title</code> attribute in that a
  <code>style</code> block without a title does not inherit the title of the parent element: it
  merely has no title.</p>

  <p>The <span>child text content</span> of a <code>style</code> element must be that of a
  <span>conformant style sheet</span>.</p>

  <p>A <code>style</code> element is <span>implicitly potentially render-blocking</span> if the
  element was created by its <span>node document</span>'s parser.</p>

  <hr>

  <div w-nodev>

  <p>The user agent must run the <span>update a <code>style</code> block</span> algorithm whenever
  any of the following conditions occur:</p>
  <!-- note that a browsing context isn't needed:
       https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2739 -->

  <ul>
   <li><p>The element is popped off the <span>stack of open elements</span> of an <span>HTML
   parser</span> or <span>XML parser</span>.</p></li>

   <li><p>The element is not on the <span>stack of open elements</span> of an <span>HTML
   parser</span> or <span>XML parser</span>, and it <span>becomes connected</span> or <span
   data-x="becomes disconnected">disconnected</span>.</p></li>

   <li><p>The element's <span>children changed steps</span> run.</p></li>
  </ul>

  <p>The <dfn export>update a <code>style</code> block</dfn> algorithm is as follows:</p>

  <ol>
   <li><p>Let <var>element</var> be the <code>style</code> element.</p></li>

   <li><p>If <var>element</var> has an <span>associated CSS style sheet</span>, <span
   data-x="remove a CSS style sheet">remove the CSS style sheet</span> in question.</p></li>

   <li><p>If <var>element</var> is not <span>connected</span>, then return.</p></li>
   <!-- See:
          https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2740
          https://github.com/w3c/csswg-drafts/issues/3096 -->

   <li>
    <p>If <var>element</var>'s <code data-x="attr-style-type">type</code> attribute is present and
    its value is neither the empty string nor an <span>ASCII case-insensitive</span> match for
    "<code>text/css</code>", then return.</p>

    <p class="note">In particular, a <code data-x="attr-style-type">type</code> value with
    parameters, such as "<code data-x="">text/css; charset=utf-8</code>", will cause this algorithm
    to return early.</p>
   </li>

   <li><p>If the <span>Should element's inline behavior be blocked by Content Security
   Policy?</span> algorithm returns "<code data-x="">Blocked</code>" when executed upon the
   <code>style</code> element, "<code data-x="">style</code>", and the <code>style</code> element's
   <span>child text content</span>, then return. <ref>CSP</ref></p></li>

   <li>
    <p><span>Create a CSS style sheet</span> with the following properties:</p>

    <dl>
     <dt><span data-x="concept-css-style-sheet-type">type</span></dt>
     <dd><p><code>text/css</code></p></dd>

     <dt><span data-x="concept-css-style-sheet-owner-node">owner node</span></dt>
     <dd><p><var>element</var></p></dd>

     <dt><span data-x="concept-css-style-sheet-media">media</span></dt>

     <dd>
      <p>The <code data-x="attr-style-media">media</code> attribute of <var>element</var>.</p>

      <p class="note">This is a reference to the (possibly absent at this time) attribute, rather
      than a copy of the attribute's current value. <cite>CSSOM</cite> defines what happens
      when the attribute is dynamically set, changed, or removed.</p>
     </dd>

     <dt><span data-x="concept-css-style-sheet-title">title</span></dt>

     <dd>
      <p>The <code data-x="attr-style-title">title</code> attribute of <var>element</var>, if
      <var>element</var> is <span>in a document tree</span>, or the empty string otherwise.</p>

      <p class="note">Again, this is a <em>reference</em> to the attribute.</p>
     </dd>

     <dt><span data-x="concept-css-style-sheet-alternate-flag">alternate flag</span></dt>
     <dd><p>Unset.</p></dd>

     <dt><span data-x="concept-css-style-sheet-origin-clean-flag">origin-clean flag</span></dt>
     <dd><p>Set.</p></dd>

     <dt><span data-x="concept-css-style-sheet-location">location</span></dt>
     <dt><span data-x="concept-css-style-sheet-parent-CSS-style-sheet">parent CSS style sheet</span></dt>
     <dt><span data-x="concept-css-style-sheet-owner-CSS-rule">owner CSS rule</span></dt>
     <dd><p>null</p></dd>

     <dt><span data-x="concept-css-style-sheet-disabled-flag">disabled flag</span></dt>
     <dd><p>Left at its default value.</p></dd>

     <dt><span data-x="concept-css-style-sheet-CSS-rules">CSS rules</span></dt>
     <dd>
      <p>Left uninitialized.</p>

      <p class="XXX">This doesn't seem right. Presumably we should be using the element's
      <span>child text content</span>? Tracked as <a
      href="https://github.com/whatwg/html/issues/2997">issue #2997</a>.</p>
     </dd>
    </dl>
   </li>

   <li><p>If <var>element</var> <span>contributes a script-blocking style sheet</span>, <span
   data-x="set append">append</span> <var>element</var> to its <span>node document</span>'s
   <span>script-blocking style sheet set</span>.</p></li>

   <li><p>If <var>element</var>'s <code data-x="attr-style-media">media</code> attribute's value
   <span>matches the environment</span> and <var>element</var> is
   <span>potentially render-blocking</span>, then <span>block rendering</span> on
   <var>element</var>.</p></li>
  </ol>

  <!-- the following steps are similar to the text in the <link> element's default process the
  linked resource algorithm -->
  <p>Once the attempts to obtain the style sheet's <span>critical subresources</span>, if any, are
  complete, or, if the style sheet has no <span>critical subresources</span>, once the style sheet
  has been parsed and processed, the user agent must run these steps:</p>

  <p class="XXX">Fetching the <span>critical subresources</span> is not well-defined; probably <a
  href="https://github.com/whatwg/html/issues/968">issue #968</a> is the best resolution for that.
  In the meantime, any <span data-x="critical subresources">critical subresource</span> <span
  data-x="concept-request">request</span> should have its <span
  data-x="concept-request-render-blocking">render-blocking</span> set to whether or not the
  <code>style</code> element is currently <span>render-blocking</span>.</p>

  <ol>
   <li><p>Let <var>element</var> be the <code>style</code> element associated with the style sheet
   in question.</p></li>

   <li><p>Let <var>success</var> be true.</p></li>

   <li>
    <p>If the attempts to obtain any of the style sheet's <span>critical subresources</span> failed
    for any reason (e.g., DNS error, HTTP 404 response, a connection being prematurely closed,
    unsupported Content-Type), set <var>success</var> to false.</p>

    <p class="note">Note that content-specific errors, e.g., CSS parse errors or PNG decoding
    errors, do not affect <var>success</var>.</p>
   </li>

   <li>
    <p><span>Queue an element task</span> on the <span>networking task source</span> given
    <var>element</var> and the following steps:</p>

    <ol>
     <li><p>If <var>success</var> is true, <span data-x="concept-event-fire">fire an event</span>
     named <code data-x="event-load">load</code> at <var>element</var>.</p></li>

     <li><p>Otherwise, <span data-x="concept-event-fire">fire an event</span> named <code
     data-x="event-error">error</code> at <var>element</var>.</p></li>

     <li>
      <p>If <var>element</var> <span>contributes a script-blocking style sheet</span>:</p>

      <ol>
       <li><p><span>Assert</span>: <var>element</var>'s <span>node document</span>'s
       <span>script-blocking style sheet set</span> <span data-x="list contains">contains</span>
       <var>element</var>.</p></li>

       <li><p><span data-x="list remove">Remove</span> <var>element</var> from its <span>node
       document</span>'s <span>script-blocking style sheet set</span>.</p></li>
      </ol>
     </li>

     <li><p><span>Unblock rendering</span> on <var>element</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>The element must <span>delay the load event</span> of the element's <span>node document</span> until all the
  attempts to obtain the style sheet's <span>critical subresources</span>, if any, are complete.</p>

  </div>

  <p class="note">This specification does not specify a style system, but CSS is expected to be
  supported by most web browsers. <ref>CSS</ref></p>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLStyleElement"><code data-x="dom-style-media">media</code></dfn> and
  <dfn attribute for="HTMLStyleElement"><code data-x="dom-style-blocking">blocking</code></dfn> IDL
  attributes must each <span>reflect</span> the respective content attributes of the same name.</p>

  </div>

  <p>The <code>LinkStyle</code> interface is also implemented by this element. <ref>CSSOM</ref></p>

  <div class="example">

   <p>The following document has its stress emphasis styled as bright red text rather than italics
   text, while leaving titles of works and Latin words in their default italics. It shows how using
   appropriate elements enables easier restyling of documents.</p>

   <pre><code class="html">&lt;!DOCTYPE html>
&lt;html lang="en-US">
 &lt;head>
  &lt;title>My favorite book&lt;/title>
  &lt;style>
   body { color: black; background: white; }
   em { font-style: normal; color: red; }
  &lt;/style>
 &lt;/head>
 &lt;body>
  &lt;p>My &lt;em>favorite&lt;/em> book of all time has &lt;em>got&lt;/em> to be
  &lt;cite>A Cat's Life&lt;/cite>. It is a book by P. Rahmel that talks
  about the &lt;i lang="la">Felis catus&lt;/i> in modern human society.&lt;/p>
 &lt;/body>
&lt;/html></code></pre>

  </div>



  <div w-nodev>

  <h4>Interactions of styling and scripting</h4>

  <p>If the style sheet referenced no other resources (e.g., it was an internal style sheet given by
  a <code>style</code> element with no <code data-x="">@import</code> rules), then the style rules
  must be <span>immediately</span> made available to script; otherwise, the style rules must only be
  made available to script once the <span>event loop</span> reaches its <span>update the
  rendering</span> step.</p>

  <p id="a-style-sheet-that-is-blocking-scripts">An element <var>el</var> in the context of a
  <code>Document</code> of an <span>HTML parser</span> or <span>XML parser</span> <dfn>contributes a
  script-blocking style sheet</dfn> if all of the following are true:</p>

  <ul>
   <li><p><var>el</var> was created by that <code>Document</code>'s parser.</p></li>

   <li><p><var>el</var> is either a <code>style</code> element or a <code>link</code> element that
   was an <span data-x="rel-stylesheet">external resource link that contributes to the styling
   processing model</span> when the <var>el</var> was created by the parser.</p></li>

   <li><p><var>el</var>'s <code data-x="">media</code> attribute's value
   <span>matches the environment</span>.</p></li>

   <li><p><var>el</var>'s style sheet was enabled when the element was created by the
   parser.</p></li>

   <li><p>The last time the <span>event loop</span> reached <a href="#step1">step 1</a>,
   <var>el</var>'s <span>root</span> was that <code>Document</code>.</p></li>

   <li>
    <p>The user agent hasn't given up on loading that particular style sheet yet. A user agent
    may give up on loading a style sheet at any time.</p>

    <p class="note">Giving up on a style sheet before the style sheet loads, if the style sheet
    eventually does still load, means that the script might end up operating with incorrect
    information. For example, if a style sheet sets the color of an element to green, but a script
    that inspects the resulting style is executed before the sheet is loaded, the script will find
    that the element is black (or whatever the default color is), and might thus make poor choices
    (e.g., deciding to use black as the color elsewhere on the page, instead of green). Implementers
    have to balance the likelihood of a script using incorrect information with the performance impact
    of doing nothing while waiting for a slow network request to finish.</p>
   </li>
  </ul>

  <p class="XXX">It is expected that counterparts to the above rules also apply to
  <code>&lt;?xml-stylesheet?></code> PIs. However, this has not yet been thoroughly
  investigated.</p>

  <p>A <code>Document</code> has a <dfn>script-blocking style sheet set</dfn>, which is an <span
  data-x="set">ordered set</span>, initially empty.</p>

  <p>A <code>Document</code> <var>document</var> <dfn>has a style sheet that is blocking
  scripts</dfn> if the following steps return true:</p>

  <ol>
   <li><p>If <var>document</var>'s <span>script-blocking style sheet set</span> is not <span
   data-x="list is empty">empty</span>, then return true.</p></li>

   <li><p>If <var>document</var>'s <span>node navigable</span> is null, then return false.</p></li>

   <li><p>Let <var>containerDocument</var> be <var>document</var>'s <span>node navigable</span>'s
   <span data-x="nav-container">container document</span>.</p></li>

   <li><p>If <var>containerDocument</var> is non-null and <var>containerDocument</var>'s
   <span>script-blocking style sheet set</span> is not <span data-x="list is empty">empty</span>,
   then return true.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  <p>A <code>Document</code> <dfn>has no style sheet that is blocking scripts</dfn> if it does not
  <span data-x="has a style sheet that is blocking scripts">have a style sheet that is blocking
  scripts</span>.</p>

  </div>




  <h3 split-filename="sections">Sections</h3>

  <h4 id="the-body-element">The <dfn element><code>body</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As the second element in an <code>html</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="handler-window-onafterprint">onafterprint</code></dd>
   <dd><code data-x="handler-window-onbeforeprint">onbeforeprint</code></dd>
   <dd><code data-x="handler-window-onbeforeunload">onbeforeunload</code></dd>
   <dd><code data-x="handler-window-onhashchange">onhashchange</code></dd>
   <dd><code data-x="handler-window-onlanguagechange">onlanguagechange</code></dd>
   <dd><code data-x="handler-window-onmessage">onmessage</code></dd>
   <dd><code data-x="handler-window-onmessageerror">onmessageerror</code></dd>
   <dd><code data-x="handler-window-onoffline">onoffline</code></dd>
   <dd><code data-x="handler-window-ononline">ononline</code></dd>
   <dd><code data-x="handler-window-onpageswap">onpageswap</code></dd>
   <dd><code data-x="handler-window-onpagehide">onpagehide</code></dd>
   <dd><code data-x="handler-window-onpagereveal">onpagereveal</code></dd>
   <dd><code data-x="handler-window-onpageshow">onpageshow</code></dd>
   <dd><code data-x="handler-window-onpopstate">onpopstate</code></dd>
   <dd><code data-x="handler-window-onrejectionhandled">onrejectionhandled</code></dd>
   <dd><code data-x="handler-window-onstorage">onstorage</code></dd>
   <dd><code data-x="handler-window-onunhandledrejection">onunhandledrejection</code></dd>
   <dd><code data-x="handler-window-onunload">onunload</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-body">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-body">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLBodyElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  // <a href="#HTMLBodyElement-partial">also has obsolete members</a>
};

<span>HTMLBodyElement</span> includes <span>WindowEventHandlers</span>;</code></pre>
</dd>
   <dd w-dev>Uses <code>HTMLBodyElement</code>.</dd>
  </dl>

  <p>The <code>body</code> element <span>represents</span> the contents of the document.</p>

  <p>In conforming documents, there is only one <code>body</code> element. The <code
  data-x="dom-document-body">document.body</code> IDL attribute provides scripts with easy access to
  a document's <code>body</code> element.</p>

  <div w-nodev>

  <p class="note">Some DOM operations (for example, parts of the <span>drag and drop</span> model)
  are defined in terms of "<span>the body element</span>". This refers to a particular element in
  the DOM, as per the definition of the term, and not any arbitrary <code>body</code> element.</p>

  </div>

  <p>The <code>body</code> element exposes as <span>event handler content attributes</span> a number
  of the <span>event handlers</span> of the <code>Window</code> object. It also mirrors their
  <span>event handler IDL attributes</span>.</p>

  <p>The <span>event handlers</span> of the <code>Window</code> object named by the
  <span><code>Window</code>-reflecting body element event handler set</span>, exposed on the
  <code>body</code> element, replace the generic <span>event handlers</span> with the same names
  normally supported by <span>HTML elements</span>.</p>

  <p class="example">Thus, for example, a bubbling <code data-x="event-error">error</code> event
  dispatched on a child of <span>the body element</span> of a <code>Document</code> would first
  trigger the <code data-x="handler-onerror">onerror</code> <span>event handler content
  attributes</span> of that element, then that of the root <code>html</code> element, and only
  <em>then</em> would it trigger the <code data-x="handler-onerror">onerror</code> <span
  data-x="event handler content attributes">event handler content attribute</span> on the
  <code>body</code> element. This is because the event would bubble from the target, to the
  <code>body</code>, to the <code>html</code>, to the <code>Document</code>, to the
  <code>Window</code>, and the <span data-x="event handlers">event handler</span> on the
  <code>body</code> is watching the <code>Window</code> not the <code>body</code>. A regular event
  listener attached to the <code>body</code> using <code data-x="">addEventListener()</code>,
  however, would be run when the event bubbled through the <code>body</code> and not when it reaches
  the <code>Window</code> object.</p>

  <div class="example">

   <p>This page updates an indicator to show whether or not the user is online:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang="en">
 &lt;head>
  &lt;title>Online or offline?&lt;/title>
  &lt;script>
   function update(online) {
     document.getElementById('status').textContent =
       online ? 'Online' : 'Offline';
   }
  &lt;/script>
 &lt;/head>
 &lt;body ononline="update(true)"
       onoffline="update(false)"
       onload="update(navigator.onLine)">
  &lt;p>You are: &lt;span id="status">(Unknown)&lt;/span>&lt;/p>
 &lt;/body>
&lt;/html></code></pre>

  </div>



  <h4>The <dfn element><code>article</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Sectioning content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>sectioning content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-article">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-article">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>article</code> element <span>represents</span> a complete, or self-contained,
  composition in a document, page, application, or site and that is, in principle, independently
  distributable or reusable, e.g. in syndication. This could be a forum post, a magazine or
  newspaper article, a blog entry, a user-submitted comment, an interactive widget or gadget, or any
  other independent item of content.</p>

  <!-- for context regarding "site", see https://krijnhoetmer.nl/irc-logs/whatwg/20130306#l-152 -->

  <p>When <code>article</code> elements are nested, the inner <code>article</code> elements
  represent articles that are in principle related to the contents of the outer article. For
  instance, a blog entry on a site that accepts user-submitted comments could represent the comments
  as <code>article</code> elements nested within the <code>article</code> element for the blog
  entry.</p>

  <p>Author information associated with an <code>article</code> element (q.v. the
  <code>address</code> element) does not apply to nested <code>article</code> elements.</p>

  <p class="note">When used specifically with content to be redistributed in syndication, the
  <code>article</code> element is similar in purpose to the <code data-x="">entry</code> element in
  Atom. <ref>ATOM</ref>

  <p class="note">The schema.org microdata vocabulary can be used to provide the publication date
  for an <code>article</code> element, using one of the CreativeWork subtypes.</p>

  <p>When the main content of the page (i.e. excluding footers, headers, navigation blocks, and
  sidebars) is all one single self-contained composition, that content may be marked with an
  <code>article</code>, but it is technically redundant in that case (since it's self-evident that
  the page is a single composition, as it is a single document).</p>

  <div id="article-example" class="example">

   <p>This example shows a blog post using the <code>article</code> element, with some schema.org
   annotations:</p>

   <pre><code class="html">&lt;article itemscope itemtype="http://schema.org/BlogPosting">
 &lt;header>
  &lt;h2 itemprop="headline">The Very First Rule of Life&lt;/h2>
  &lt;p>&lt;time itemprop="datePublished" datetime="2009-10-09">3 days ago&lt;/time>&lt;/p>
  &lt;link itemprop="url" href="?comments=0">
 &lt;/header>
 &lt;p>If there's a microphone anywhere near you, assume it's hot and
 sending whatever you're saying to the world. Seriously.&lt;/p>
 &lt;p><em>...</em>&lt;/p>
 &lt;footer>
  &lt;a itemprop="discussionUrl" href="?comments=1">Show comments...&lt;/a>
 &lt;/footer>
&lt;/article></code></pre>

   <p>Here is that same blog post, but showing some of the comments:</p>

   <pre><code class="html">&lt;article itemscope itemtype="http://schema.org/BlogPosting">
 &lt;header>
  &lt;h2 itemprop="headline">The Very First Rule of Life&lt;/h2>
  &lt;p>&lt;time itemprop="datePublished" datetime="2009-10-09">3 days ago&lt;/time>&lt;/p>
  &lt;link itemprop="url" href="?comments=0">
 &lt;/header>
 &lt;p>If there's a microphone anywhere near you, assume it's hot and
 sending whatever you're saying to the world. Seriously.&lt;/p>
 &lt;p><em>...</em>&lt;/p>
 &lt;section>
  &lt;h1>Comments&lt;/h1>
  &lt;article itemprop="comment" itemscope itemtype="http://schema.org/Comment" id="c1">
   &lt;link itemprop="url" href="#c1">
   &lt;footer>
    &lt;p>Posted by: &lt;span itemprop="creator" itemscope itemtype="http://schema.org/Person">
     &lt;span itemprop="name">George Washington&lt;/span>
    &lt;/span>&lt;/p>
    &lt;p>&lt;time itemprop="dateCreated" datetime="2009-10-10">15 minutes ago&lt;/time>&lt;/p>
   &lt;/footer>
   &lt;p>Yeah! Especially when talking about your lobbyist friends!&lt;/p>
  &lt;/article>
  &lt;article itemprop="comment" itemscope itemtype="http://schema.org/Comment" id="c2">
   &lt;link itemprop="url" href="#c2">
   &lt;footer>
    &lt;p>Posted by: &lt;span itemprop="creator" itemscope itemtype="http://schema.org/Person">
     &lt;span itemprop="name">George Hammond&lt;/span>
    &lt;/span>&lt;/p>
    &lt;p>&lt;time itemprop="dateCreated" datetime="2009-10-10">5 minutes ago&lt;/time>&lt;/p>
   &lt;/footer>
   &lt;p>Hey, you have the same first name as me.&lt;/p>
  &lt;/article>
 &lt;/section>
&lt;/article></code></pre>

   <p>Notice the use of <code>footer</code> to give the information for each comment (such as who
   wrote it and when): the <code>footer</code> element <em>can</em> appear at the start of its
   section when appropriate, such as in this case. (Using <code>header</code> in this case wouldn't
   be wrong either; it's mostly a matter of authoring preference.)</p>

  </div>

  <div class="example">

   <p>In this example, <code>article</code> elements are used to host widgets on a portal page. The
   widgets are implemented as <span data-x="customized built-in element">customized built-in
   elements</span> in order to get specific styling and scripted behavior.</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang=en>
&lt;title>eHome Portal&lt;/title>
&lt;script src="/scripts/widgets.js">&lt;/script>
&lt;link rel=stylesheet href="/styles/main.css">
&lt;article is="stock-widget">
 &lt;h2>Stocks&lt;/h2>
 &lt;table>
  &lt;thead> &lt;tr> &lt;th> Stock &lt;th> Value &lt;th> Delta
  &lt;tbody> &lt;template> &lt;tr> &lt;td> &lt;td> &lt;td> &lt;/template>
 &lt;/table>
 &lt;p> &lt;input type=button value="Refresh" onclick="this.parentElement.refresh()">
&lt;/article>
&lt;article is="news-widget">
 &lt;h2>News&lt;/h2>
 &lt;ul>
  &lt;template>
   &lt;li>
    &lt;p>&lt;img> &lt;strong>&lt;/strong>
    &lt;p>
  &lt;/template>
 &lt;/ul>
 &lt;p> &lt;input type=button value="Refresh" onclick="this.parentElement.refresh()">
&lt;/article></code></pre>

  </div>



  <h4>The <dfn element><code>section</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Sectioning content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>sectioning content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-section">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-section">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>section</code> element <span>represents</span> a generic section of a document or
  application. A section, in this context, is a thematic grouping of content, typically with a
  heading.</p>

  <p class="example">Examples of sections would be chapters, the various tabbed pages in a tabbed
  dialog box, or the numbered sections of a thesis. A web site's home page could be split into
  sections for an introduction, news items, and contact information.</p>

  <p class="note">Authors are encouraged to use the <code>article</code> element instead of the
  <code>section</code> element when it would make sense to syndicate the contents of the
  element.</p>

  <p class="note" id="use-div-for-wrappers">The <code>section</code> element is not a generic
  container element. When an element is needed only for styling purposes or as a convenience for
  scripting, authors are encouraged to use the <code>div</code> element instead. A general rule is
  that the <code>section</code> element is appropriate only if the element's contents would be
  listed explicitly in the document's <span>outline</span>.</p>

  <div class="example">

   <p>In the following example, we see an article (part of a larger web page) about apples,
   containing two short sections.</p>

   <pre><code class="html">&lt;article>
 &lt;hgroup>
  &lt;h2>Apples&lt;/h2>
  &lt;p>Tasty, delicious fruit!&lt;/p>
 &lt;/hgroup>
 &lt;p>The apple is the pomaceous fruit of the apple tree.&lt;/p>
 &lt;section>
  &lt;h3>Red Delicious&lt;/h3>
  &lt;p>These bright red apples are the most common found in many
  supermarkets.&lt;/p>
 &lt;/section>
 &lt;section>
  &lt;h3>Granny Smith&lt;/h3>
  &lt;p>These juicy, green apples make a great filling for
  apple pies.&lt;/p>
 &lt;/section>
&lt;/article></code></pre>

  </div>

  <div class="example">

   <p>Here is a graduation programme with two sections, one for the list of people graduating, and
   one for the description of the ceremony. (The markup in this example features an uncommon style
   sometimes used to minimize the amount of <span>inter-element whitespace</span>.)</p>

   <pre><code class="html">&lt;!DOCTYPE Html>
&lt;Html Lang=En
 >&lt;Head
   >&lt;Title
     >Graduation Ceremony Summer 2022&lt;/Title
   >&lt;/Head
 >&lt;Body
   >&lt;H1
     >Graduation&lt;/H1
   >&lt;Section
     >&lt;H2
       >Ceremony&lt;/H2
     >&lt;P
       >Opening Procession&lt;/P
     >&lt;P
       >Speech by Valedictorian&lt;/P
     >&lt;P
       >Speech by Class President&lt;/P
     >&lt;P
       >Presentation of Diplomas&lt;/P
     >&lt;P
       >Closing Speech by Headmaster&lt;/P
   >&lt;/Section
   >&lt;Section
     >&lt;H2
       >Graduates&lt;/H2
     >&lt;Ul
       >&lt;Li
         >Molly Carpenter&lt;/Li
       >&lt;Li
         >Anastasia Luccio&lt;/Li
       >&lt;Li
         >Ebenezar McCoy&lt;/Li
       >&lt;Li
         >Karrin Murphy&lt;/Li
       >&lt;Li
         >Thomas Raith&lt;/Li
       >&lt;Li
         >Susan Rodriguez&lt;/Li
     >&lt;/Ul
   >&lt;/Section
 >&lt;/Body
>&lt;/Html></code></pre>

  </div>

  <div class="example">

   <p>In this example, a book author has marked up some sections as chapters and some as appendices,
   and uses CSS to style the headers in these two classes of section differently.</p>

   <pre><code class="html">&lt;style>
 section { border: double medium; margin: 2em; }
 section.chapter h2 { font: 2em Roboto, Helvetica Neue, sans-serif; }
 section.appendix h2 { font: small-caps 2em Roboto, Helvetica Neue, sans-serif; }
&lt;/style>
&lt;header>
 &lt;hgroup>
  &lt;h1>My Book&lt;/h1>
  &lt;p>A sample with not much content&lt;/p>
 &lt;/hgroup>
 &lt;p>&lt;small>Published by Dummy Publicorp Ltd.&lt;/small>&lt;/p>
&lt;/header>
&lt;section class="chapter">
 &lt;h2>My First Chapter&lt;/h2>
 &lt;p>This is the first of my chapters. It doesn't say much.&lt;/p>
 &lt;p>But it has two paragraphs!&lt;/p>
&lt;/section>
&lt;section class="chapter">
 &lt;h2>It Continues: The Second Chapter&lt;/h2>
 &lt;p>Bla dee bla, dee bla dee bla. Boom.&lt;/p>
&lt;/section>
&lt;section class="chapter">
 &lt;h2>Chapter Three: A Further Example&lt;/h2>
 &lt;p>It's not like a battle between brightness and earthtones would go
 unnoticed.&lt;/p>
 &lt;p>But it might ruin my story.&lt;/p>
&lt;/section>
&lt;section class="appendix">
 &lt;h2>Appendix A: Overview of Examples&lt;/h2>
 &lt;p>These are demonstrations.&lt;/p>
&lt;/section>
&lt;section class="appendix">
 &lt;h2>Appendix B: Some Closing Remarks&lt;/h2>
 &lt;p>Hopefully this long example shows that you &lt;em>can&lt;/em> style
 sections, so long as they are used to indicate actual sections.&lt;/p>
&lt;/section></code></pre>

  </div>



  <h4>The <dfn element><code>nav</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Sectioning content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>sectioning content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-nav">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-nav">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>nav</code> element <span>represents</span> a section of a page that links to other
  pages or to parts within the page: a section with navigation links.</p>

  <p class="note">Not all groups of links on a page need to be in a <code>nav</code> element &mdash;
  the element is primarily intended for sections that consist of major navigation blocks. In
  particular, it is common for footers to have a short list of links to various pages of a site,
  such as the terms of service, the home page, and a copyright page. The <code>footer</code> element
  alone is sufficient for such cases; while a <code>nav</code> element can be used in such cases, it
  is usually unnecessary.</p>

  <p class="note">User agents (such as screen readers) that are targeted at users who can benefit
  from navigation information being omitted in the initial rendering, or who can benefit from
  navigation information being immediately available, can use this element as a way to determine
  what content on the page to initially skip or provide on request (or both).</p>

  <div class="example">

   <p>In the following example, there are two <code>nav</code> elements, one for primary navigation
   around the site, and one for secondary navigation around the page itself.</p>

   <pre><code class="html">&lt;body>
 &lt;h1>The Wiki Center Of Exampland&lt;/h1>
 &lt;nav>
  &lt;ul>
   &lt;li>&lt;a href="/">Home&lt;/a>&lt;/li>
   &lt;li>&lt;a href="/events">Current Events&lt;/a>&lt;/li>
   <em>...more...</em>
  &lt;/ul>
 &lt;/nav>
 &lt;article>
  &lt;header>
   &lt;h2>Demos in Exampland&lt;/h2>
   &lt;p>Written by A. N. Other.&lt;/p>
  &lt;/header>
  &lt;nav>
   &lt;ul>
    &lt;li>&lt;a href="#public">Public demonstrations&lt;/a>&lt;/li>
    &lt;li>&lt;a href="#destroy">Demolitions&lt;/a>&lt;/li>
    <em>...more...</em>
   &lt;/ul>
  &lt;/nav>
  &lt;div>
   &lt;section id="public">
    &lt;h2>Public demonstrations&lt;/h2>
    &lt;p><em>...more...</em>&lt;/p>
   &lt;/section>
   &lt;section id="destroy">
    &lt;h2>Demolitions&lt;/h2>
    &lt;p><em>...more...</em>&lt;/p>
   &lt;/section>
   <em>...more...</em>
  &lt;/div>
  &lt;footer>
   &lt;p>&lt;a href="?edit">Edit&lt;/a> | &lt;a href="?delete">Delete&lt;/a> | &lt;a href="?Rename">Rename&lt;/a>&lt;/p>
  &lt;/footer>
 &lt;/article>
 &lt;footer>
  &lt;p>&lt;small>&copy; copyright 1998 Exampland Emperor&lt;/small>&lt;/p>
 &lt;/footer>
&lt;/body></code></pre>

  </div>

  <div class="example">

   <p>In the following example, the page has several places where links are present, but only one of
   those places is considered a navigation section.</p>

   <pre><code class="html">&lt;body itemscope itemtype="http://schema.org/Blog">
 &lt;header>
  &lt;h1>Wake up sheeple!&lt;/h1>
  &lt;p>&lt;a href="news.html">News&lt;/a> -
     &lt;a href="blog.html">Blog&lt;/a> -
     &lt;a href="forums.html">Forums&lt;/a>&lt;/p>
  &lt;p>Last Modified: &lt;span itemprop="dateModified">2009-04-01&lt;/span>&lt;/p>
  &lt;nav>
   &lt;h2>Navigation&lt;/h2>
   &lt;ul>
    &lt;li>&lt;a href="articles.html">Index of all articles&lt;/a>&lt;/li>
    &lt;li>&lt;a href="today.html">Things sheeple need to wake up for today&lt;/a>&lt;/li>
    &lt;li>&lt;a href="successes.html">Sheeple we have managed to wake&lt;/a>&lt;/li>
   &lt;/ul>
  &lt;/nav>
 &lt;/header>
 &lt;main>
  &lt;article itemprop="blogPosts" itemscope itemtype="http://schema.org/BlogPosting">
   &lt;header>
    &lt;h2 itemprop="headline">My Day at the Beach&lt;/h2>
   &lt;/header>
   &lt;div itemprop="articleBody">
    &lt;p>Today I went to the beach and had a lot of fun.&lt;/p>
    <em>...more content...</em>
   &lt;/div>
   &lt;footer>
    &lt;p>Posted &lt;time itemprop="datePublished" datetime="2009-10-10">Thursday&lt;/time>.&lt;/p>
   &lt;/footer>
  &lt;/article>
  <em>...more blog posts...</em>
 &lt;/main>
 &lt;footer>
  &lt;p>Copyright &copy;
   &lt;span itemprop="copyrightYear">2010&lt;/span>
   &lt;span itemprop="copyrightHolder">The Example Company&lt;/span>
  &lt;/p>
  &lt;p>&lt;a href="about.html">About&lt;/a> -
     &lt;a href="policy.html">Privacy Policy&lt;/a> -
     &lt;a href="contact.html">Contact Us&lt;/a>&lt;/p>
 &lt;/footer>
&lt;/body></code></pre>

   <p>You can also see microdata annotations in the above example that use the schema.org vocabulary
   to provide the publication date and other metadata about the blog post.</p>

  </div>

  <div class="example">

   <p>A <code>nav</code> element doesn't have to contain a list, it can contain other kinds of
   content as well. In this navigation block, links are provided in prose:</p>

   <pre><code class="html">&lt;nav>
 &lt;h1>Navigation&lt;/h1>
 &lt;p>You are on my home page. To the north lies &lt;a href="/blog">my
 blog&lt;/a>, from whence the sounds of battle can be heard. To the east
 you can see a large mountain, upon which many &lt;a
 href="/school">school papers&lt;/a> are littered. Far up thus mountain
 you can spy a little figure who appears to be me, desperately
 scribbling a &lt;a href="/school/thesis">thesis&lt;/a>.&lt;/p>
 &lt;p>To the west are several exits. One fun-looking exit is labeled &lt;a
 href="https://games.example.com/">"games"&lt;/a>. Another more
 boring-looking exit is labeled &lt;a
 href="https://isp.example.net/">ISP&trade;&lt;/a>.&lt;/p>
 &lt;p>To the south lies a dark and dank &lt;a href="/about">contacts
 page&lt;/a>. Cobwebs cover its disused entrance, and at one point you
 see a rat run quickly out of the page.&lt;/p>
&lt;/nav></code></pre>

  </div>

  <div class="example">

   <p>In this example, <code>nav</code> is used in an email application, to let the user switch
   folders:</p>

   <pre><code class="html">&lt;p>&lt;input type=button value="Compose" onclick="compose()">&lt;/p>
&lt;nav>
 &lt;h1>Folders&lt;/h1>
 &lt;ul>
  &lt;li> &lt;a href="/inbox" onclick="return openFolder(this.href)">Inbox&lt;/a> &lt;span class=count>&lt;/span>
  &lt;li> &lt;a href="/sent" onclick="return openFolder(this.href)">Sent&lt;/a>
  &lt;li> &lt;a href="/drafts" onclick="return openFolder(this.href)">Drafts&lt;/a>
  &lt;li> &lt;a href="/trash" onclick="return openFolder(this.href)">Trash&lt;/a>
  &lt;li> &lt;a href="/customers" onclick="return openFolder(this.href)">Customers&lt;/a>
 &lt;/ul>
&lt;/nav></code></pre>

  </div>



  <h4>The <dfn element><code>aside</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Sectioning content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>sectioning content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-aside">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-aside">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>aside</code> element <span>represents</span> a section of a page that consists of
  content that is tangentially related to the content around the <code>aside</code> element, and
  which could be considered separate from that content. Such sections are often represented as
  sidebars in printed typography.</p>

  <p>The element can be used for typographical effects like pull quotes or sidebars, for
  advertising, for groups of <code>nav</code> elements, and for other content that is considered
  separate from the main content of the page.</p>

  <p class="note">It's not appropriate to use the <code>aside</code> element just for
  parentheticals, since those are part of the main flow of the document.</p>

  <div class="example">

   <p>The following example shows how an aside is used to mark up background material on Switzerland
   in a much longer news story on Europe.</p>

   <pre><code class="html">&lt;aside>
 &lt;h2>Switzerland&lt;/h2>
 &lt;p>Switzerland, a land-locked country in the middle of geographic
 Europe, has not joined the geopolitical European Union, though it is
 a signatory to a number of European treaties.&lt;/p>
&lt;/aside></code></pre>

  </div>

  <div class="example">

   <p>The following example shows how an aside is used to mark up a pull quote in a longer
   article.</p>

   <pre><code class="html">...

&lt;p>He later joined a large company, continuing on the same work.
&lt;q>I love my job. People ask me what I do for fun when I'm not at
work. But I'm paid to do my hobby, so I never know what to
answer. Some people wonder what they would do if they didn't have to
work... but I know what I would do, because I was unemployed for a
year, and I filled that time doing exactly what I do now.&lt;/q>&lt;/p>

&lt;aside>
 &lt;q>People ask me what I do for fun when I'm not at work. But I'm
 paid to do my hobby, so I never know what to answer.&lt;/q>
&lt;/aside>

&lt;p>Of course his work &mdash; or should that be hobby? &mdash;
isn't his only passion. He also enjoys other pleasures.&lt;/p>

...</code></pre>

  </div>

  <div class="example">

   <p>The following extract shows how <code>aside</code> can be used for blogrolls and other side
   content on a blog:</p>

   <pre><code class="html">&lt;body>
 &lt;header>
  &lt;h1>My wonderful blog&lt;/h1>
  &lt;p>My tagline&lt;/p>
 &lt;/header>
 &lt;aside>
  &lt;!-- <em>this aside contains two sections that are tangentially related
  to the page, namely, links to other blogs, and links to blog posts
  from this blog</em> -->
  &lt;nav>
   &lt;h2>My blogroll&lt;/h2>
   &lt;ul>
    &lt;li>&lt;a href="https://blog.example.com/">Example Blog&lt;/a>
   &lt;/ul>
  &lt;/nav>
  &lt;nav>
   &lt;h2>Archives&lt;/h2>
   &lt;ol reversed>
    &lt;li>&lt;a href="/last-post">My last post&lt;/a>
    &lt;li>&lt;a href="/first-post">My first post&lt;/a>
   &lt;/ol>
  &lt;/nav>
 &lt;/aside>
 &lt;aside>
  &lt;!-- <em>this aside is tangentially related to the page also, it
  contains twitter messages from the blog author</em> -->
  &lt;h1>Twitter Feed&lt;/h1>
  &lt;blockquote cite="https://twitter.example.net/t31351234">
   I'm on vacation, writing my blog.
  &lt;/blockquote>
  &lt;blockquote cite="https://twitter.example.net/t31219752">
   I'm going to go on vacation soon.
  &lt;/blockquote>
 &lt;/aside>
 &lt;article>
  &lt;!-- <em>this is a blog post</em> -->
  &lt;h2>My last post&lt;/h2>
  &lt;p>This is my last post.&lt;/p>
  &lt;footer>
   &lt;p>&lt;a href="/last-post" rel=bookmark>Permalink&lt;/a>
  &lt;/footer>
 &lt;/article>
 &lt;article>
  &lt;!-- <em>this is also a blog post</em> -->
  &lt;h2>My first post&lt;/h2>
  &lt;p>This is my first post.&lt;/p>
  &lt;aside>
   &lt;!-- <em>this aside is about the blog post, since it's inside the
   &lt;article> element; it would be wrong, for instance, to put the
   blogroll here, since the blogroll isn't really related to this post
   specifically, only to the page as a whole</em> -->
   &lt;h2>Posting&lt;/h2>
   &lt;p>While I'm thinking about it, I wanted to say something about
   posting. Posting is fun!&lt;/p>
  &lt;/aside>
  &lt;footer>
   &lt;p>&lt;a href="/first-post" rel=bookmark>Permalink&lt;/a>
  &lt;/footer>
 &lt;/article>
 &lt;footer>
  &lt;p>&lt;a href="/archives">Archives&lt;/a> -
   &lt;a href="/about">About me&lt;/a> -
   &lt;a href="/copyright">Copyright&lt;/a>&lt;/p>
 &lt;/footer>
&lt;/body></code></pre>

  </div>



  <h4>The <dfn id="the-h1-element" element><code>h1</code></dfn>, <dfn id="the-h2-element"
  element><code>h2</code></dfn>, <dfn id="the-h3-element" element><code>h3</code></dfn>, <dfn
  id="the-h4-element" element><code>h4</code></dfn>, <dfn id="the-h5-element"
  element><code>h5</code></dfn>, and <dfn id="the-h6-element" element><code>h6</code></dfn>
  elements</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Heading content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As a child of an <code>hgroup</code> element.</dd>
   <dd>Where <span>heading content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-h1-h6">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-h1-h6">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLHeadingElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  // <a href="#HTMLHeadingElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLHeadingElement</code>.</dd>
  </dl>

  <p>These elements <span data-x="represents">represent</span> headings for their sections.</p>

  <p>The semantics and meaning of these elements are defined in the section on <a
  href="#headings-and-outlines">headings and outlines</a>.</p>

  <p>These elements have a <span>heading level</span> given by the number in their name. The
  <span>heading level</span> corresponds to the levels of nested sections. The <code>h1</code>
  element is for a top-level section, <code>h2</code> for a subsection, <code>h3</code> for a
  sub-subsection, and so on.</p>

  <div class="example">

   <p>As far as their respective document outlines (their heading and section structures) are
   concerned, these two snippets are semantically equivalent:</p>

   <pre><code class="html">&lt;body>
&lt;h1>Let's call it a draw(ing surface)&lt;/h1>
&lt;h2>Diving in&lt;/h2>
&lt;h2>Simple shapes&lt;/h2>
&lt;h2>Canvas coordinates&lt;/h2>
&lt;h3>Canvas coordinates diagram&lt;/h3>
&lt;h2>Paths&lt;/h2>
&lt;/body></code></pre>

   <pre><code class="html">&lt;body>
 &lt;h1>Let's call it a draw(ing surface)&lt;/h1>
 &lt;section>
  &lt;h2>Diving in&lt;/h2>
 &lt;/section>
 &lt;section>
  &lt;h2>Simple shapes&lt;/h2>
 &lt;/section>
 &lt;section>
  &lt;h2>Canvas coordinates&lt;/h2>
  &lt;section>
   &lt;h3>Canvas coordinates diagram&lt;/h3>
  &lt;/section>
 &lt;/section>
 &lt;section>
  &lt;h2>Paths&lt;/h2>
 &lt;/section>
&lt;/body></code></pre>

   <p>Authors might prefer the former style for its terseness, or the latter style for its
   additional styling hooks. Which is best is purely an issue of preferred authoring style.</p>
  </div>


  <h4>The <dfn element><code>hgroup</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Heading content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>heading content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>Zero or more <code>p</code> elements, followed by one <code>h1</code>, <code>h2</code>,
   <code>h3</code>, <code>h4</code>, <code>h5</code>, or <code>h6</code> element, followed by zero
   or more <code>p</code> elements, optionally intermixed with <span>script-supporting
   elements</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-hgroup">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-hgroup">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>hgroup</code> element <span>represents</span> a heading and related content. The
  element may be used to group an <code>h1</code>&ndash;<code>h6</code> element with one or more
  <code>p</code> elements containing content representing a subheading, alternative title, or
  tagline.</p>

  <div class="example">
   <p>Here are some examples of valid headings contained within an <code>hgroup</code> element.</p>

    <pre><code class="html">&lt;hgroup&gt;
 &lt;h1&gt;The reality dysfunction&lt;/h1&gt;
 &lt;p&gt;Space is not the only void&lt;/p&gt;
&lt;/hgroup&gt;</code></pre>

    <pre><code class="html">&lt;hgroup&gt;
 &lt;h1&gt;Dr. Strangelove&lt;/h1&gt;
 &lt;p&gt;Or: How I Learned to Stop Worrying and Love the Bomb&lt;/p&gt;
&lt;/hgroup&gt;</code></pre>
  </div>

  <h4>The <dfn element><code>header</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>, but with no <code>header</code> or <code>footer</code> element
   descendants.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd>If there is an ancestor <span>sectioning content</span> element: <a
   href="https://w3c.github.io/html-aria/#el-header">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-header">for implementers</a>.</dd>
   <dd>Otherwise: <a
   href="https://w3c.github.io/html-aria/#el-header">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-header-ancestorbody">for implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>header</code> element <span>represents</span> a group of introductory or navigational
  aids.</p>

  <p class="note">A <code>header</code> element is intended to usually contain a heading
  (an <code>h1</code>&ndash;<code>h6</code> element or an <code>hgroup</code> element), but this is
  not required. The <code>header</code> element can also be used to wrap a section's table of
  contents, a search form, or any relevant logos.</p>

  <div class="example">

  <p>Here are some sample headers. This first one is for a game:</p>

   <pre><code class="html">&lt;header&gt;
 &lt;p&gt;Welcome to...&lt;/p&gt;
 &lt;h1&gt;Voidwars!&lt;/h1&gt;
&lt;/header&gt;</code></pre>

   <p>The following snippet shows how the element can be used to mark
   up a specification's header:</p>

   <pre><code class="html">&lt;header&gt;
 &lt;hgroup&gt;
  &lt;h1&gt;Fullscreen API&lt;/h1&gt;
  &lt;p&gt;Living Standard &mdash; Last Updated 19 October 2015&lt;p&gt;
 &lt;/hgroup&gt;
 &lt;dl&gt;
  &lt;dt&gt;Participate:&lt;/dt&gt;
  &lt;dd&gt;&lt;a href="https://github.com/whatwg/fullscreen"&gt;GitHub whatwg/fullscreen&lt;/a&gt;&lt;/dd&gt;
  &lt;dt&gt;Commits:&lt;/dt&gt;
  &lt;dd&gt;&lt;a href="https://github.com/whatwg/fullscreen/commits"&gt;GitHub whatwg/fullscreen/commits&lt;/a&gt;&lt;/dd&gt;
 &lt;/dl&gt;
&lt;/header&gt;</code></pre>

  </div>

  <p class="note">The <code>header</code> element is not <span>sectioning content</span>; it doesn't
  introduce a new section.</p>

  <div class="example">

  <p>In this example, the page has a page heading given by the <code>h1</code> element, and two
  subsections whose headings are given by <code>h2</code> elements. The content after the
  <code>header</code> element is still part of the last subsection started in the
  <code>header</code> element, because the <code>header</code> element doesn't take part in the
  <span>outline</span> algorithm.</p>

   <pre><code class="html">&lt;body>
 &lt;header>
  &lt;h1>Little Green Guys With Guns&lt;/h1>
  &lt;nav>
   &lt;ul>
    &lt;li>&lt;a href="/games">Games&lt;/a>
    &lt;li>&lt;a href="/forum">Forum&lt;/a>
    &lt;li>&lt;a href="/download">Download&lt;/a>
   &lt;/ul>
  &lt;/nav>
  &lt;h2>Important News&lt;/h2> &lt;!-- this starts a second subsection -->
  &lt;!-- this is part of the subsection entitled "Important News" -->
  &lt;p>To play today's games you will need to update your client.&lt;/p>
  &lt;h2>Games&lt;/h2> &lt;!-- this starts a third subsection -->
 &lt;/header>
 &lt;p>You have three active games:&lt;/p>
 &lt;!-- this is still part of the subsection entitled "Games" -->
 ...</code></pre>

  </div>




  <h4>The <dfn element><code>footer</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>, but with no <code>header</code> or <code>footer</code> element
   descendants.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd>If there is an ancestor <span>sectioning content</span> element: <a
   href="https://w3c.github.io/html-aria/#el-footer">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-footer">for implementers</a>.</dd>
   <dd>Otherwise: <a
   href="https://w3c.github.io/html-aria/#el-footer">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-footer-ancestorbody">for implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>footer</code> element <span>represents</span> a footer for its nearest ancestor
  <span>sectioning content</span> element, or for <span>the body element</span> if there is no such
  ancestor. A footer typically contains information about its section such as who wrote it, links
  to related documents, copyright data, and the like.</p>

  <p>When the <code>footer</code> element contains entire sections, they <span
  data-x="represents">represent</span> appendices, indices, long colophons, verbose license
  agreements, and other such content.</p>

  <p class="note">Contact information for the author or editor of a section belongs in an
  <code>address</code> element, possibly itself inside a <code>footer</code>. Bylines and other
  information that could be suitable for both a <code>header</code> or a <code>footer</code> can be
  placed in either (or neither). The primary purpose of these elements is merely to help the author
  write self-explanatory markup that is easy to maintain and style; they are not intended to impose
  specific structures on authors.</p>

  <p>Footers don't necessarily have to appear at the <em>end</em> of a section, though they usually
  do.</p>

  <p>When there is no ancestor <span>sectioning content</span> element, then it applies to the whole
  page.</p>

  <p class="note">The <code>footer</code> element is not itself <span>sectioning content</span>; it
  doesn't introduce a new section.</p>

  <div class="example">

   <p>Here is a page with two footers, one at the top and one at the bottom, with the same
   content:</p>

   <pre><code class="html">&lt;body>
 &lt;footer>&lt;a href="../">Back to index...&lt;/a>&lt;/footer>
 &lt;hgroup>
  &lt;h1>Lorem ipsum&lt;/h1>
  &lt;p>The ipsum of all lorems&lt;/p>
 &lt;/hgroup>
 &lt;p>A dolor sit amet, consectetur adipisicing elit, sed do eiusmod
 tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim
 veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex
 ea commodo consequat. Duis aute irure dolor in reprehenderit in
 voluptate velit esse cillum dolore eu fugiat nulla
 pariatur. Excepteur sint occaecat cupidatat non proident, sunt in
 culpa qui officia deserunt mollit anim id est laborum.&lt;/p>
 &lt;footer>&lt;a href="../">Back to index...&lt;/a>&lt;/footer>
&lt;/body></code></pre>

  </div>

  <div class="example">

   <p>Here is an example which shows the <code>footer</code> element being used both for a site-wide
   footer and for a section footer.</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;HTML LANG="en">&lt;HEAD>
&lt;TITLE>The Ramblings of a Scientist&lt;/TITLE>
&lt;BODY>
&lt;H1>The Ramblings of a Scientist&lt;/H1>
&lt;ARTICLE>
 &lt;H1>Episode 15&lt;/H1>
 &lt;VIDEO SRC="/fm/015.ogv" CONTROLS PRELOAD>
  &lt;P>&lt;A HREF="/fm/015.ogv">Download video&lt;/A>.&lt;/P>
 &lt;/VIDEO>
 &lt;FOOTER> &lt;!-- footer for article -->
  &lt;P>Published &lt;TIME DATETIME="2009-10-21T18:26-07:00">on 2009/10/21 at 6:26pm&lt;/TIME>&lt;/P>
 &lt;/FOOTER>
&lt;/ARTICLE>
&lt;ARTICLE>
 &lt;H1>My Favorite Trains&lt;/H1>
 &lt;P>I love my trains. My favorite train of all time is a K&ouml;f.&lt;/P>
 &lt;P>It is fun to see them pull some coal cars because they look so
 dwarfed in comparison.&lt;/P>
 &lt;FOOTER> &lt;!-- footer for article -->
  &lt;P>Published &lt;TIME DATETIME="2009-09-15T14:54-07:00">on 2009/09/15 at 2:54pm&lt;/TIME>&lt;/P>
 &lt;/FOOTER>
&lt;/ARTICLE>
&lt;FOOTER> &lt;!-- site wide footer -->
 &lt;NAV>
  &lt;P>&lt;A HREF="/credits.html">Credits&lt;/A> &mdash;
     &lt;A HREF="/tos.html">Terms of Service&lt;/A> &mdash;
     &lt;A HREF="/index.html">Blog Index&lt;/A>&lt;/P>
 &lt;/NAV>
 &lt;P>Copyright &copy; 2009 Gordon Freeman&lt;/P>
&lt;/FOOTER>
&lt;/BODY>
&lt;/HTML></code></pre>

  </div>

  <div class="example">

   <p>Some site designs have what is sometimes referred to as "fat footers" &mdash; footers that
   contain a lot of material, including images, links to other articles, links to pages for sending
   feedback, special offers... in some ways, a whole "front page" in the footer.</p>

   <p>This fragment shows the bottom of a page on a site with a "fat footer":</p>

   <pre><code class="html">...
 &lt;footer>
  &lt;nav>
   &lt;section>
    &lt;h1>Articles&lt;/h1>
    &lt;p>&lt;img src="images/somersaults.jpeg" alt=""> Go to the gym with
    our somersaults class! Our teacher Jim takes you through the paces
    in this two-part article. &lt;a href="articles/somersaults/1">Part
    1&lt;/a> &middot; &lt;a href="articles/somersaults/2">Part 2&lt;/a>&lt;/p>
    &lt;p>&lt;img src="images/kindplus.jpeg"> Tired of walking on the edge of
    a clif&lt;!-- sic -->? Our guest writer Lara shows you how to bumble
    your way through the bars. &lt;a href="articles/kindplus/1">Read
    more...&lt;/a>&lt;/p>
    &lt;p>&lt;img src="images/crisps.jpeg"> The chips are down, now all
    that's left is a potato. What can you do with it? &lt;a
    href="articles/crisps/1">Read more...&lt;/a>&lt;/p>
   &lt;/section>
   &lt;ul>
    &lt;li> &lt;a href="/about">About us...&lt;/a>
    &lt;li> &lt;a href="/feedback">Send feedback!&lt;/a>
    &lt;li> &lt;a href="/sitemap">Sitemap&lt;/a>
   &lt;/ul>
  &lt;/nav>
  &lt;p>&lt;small>Copyright &copy; 2015 The Snacker &mdash;
  &lt;a href="/tos">Terms of Service&lt;/a>&lt;/small>&lt;/p>
 &lt;/footer>
&lt;/body></code></pre>

  </div>


  <h4>The <dfn element><code>address</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>, but with no <span>heading
   content</span> descendants, no <span>sectioning content</span>
   descendants, and no <code>header</code>, <code>footer</code>, or
   <code>address</code> element descendants.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-address">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-address">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>address</code> element <span>represents</span> the contact information for its
  nearest <code>article</code> or <code>body</code> element ancestor. If that is <span>the body
  element</span>, then the contact information applies to the document as a whole.</p>

  <div class="example">
   <p>For example, a page at the W3C web site related to HTML might
   include the following contact information:</p>
   <pre><code class="html">&lt;ADDRESS>
 &lt;A href="../People/Raggett/">Dave Raggett&lt;/A>,
 &lt;A href="../People/Arnaud/">Arnaud Le Hors&lt;/A>,
 contact persons for the &lt;A href="Activity">W3C HTML Activity&lt;/A>
&lt;/ADDRESS></code></pre>
  </div>

  <p>The <code>address</code> element must not be used to represent arbitrary addresses (e.g. postal
  addresses), unless those addresses are in fact the relevant contact information. (The
  <code>p</code> element is the appropriate element for marking up postal addresses in general.)</p>

  <p>The <code>address</code> element must not contain information other than contact
  information.</p>

  <div class="example">
   <p>For example, the following is non-conforming use of the
   <code>address</code> element:</p>
   <pre class="bad"><code class="html">&lt;ADDRESS>Last Modified: 1999/12/24 23:37:50&lt;/ADDRESS></code></pre>
  </div>

  <p>Typically, the <code>address</code> element would be included along with other information in a
  <code>footer</code> element.</p>

  <div w-nodev>

  <p>The contact information for a node <var>node</var> is a collection of
  <code>address</code> elements defined by the first applicable entry from the following list:</p>

  <dl class="switch">

   <dt>If <var>node</var> is an <code>article</code> element</dt>
   <dt>If <var>node</var> is a <code>body</code> element</dt>

   <dd>
    <p>The contact information consists of all the <code>address</code> elements that have <var>node</var> as an ancestor and do not have another <code>body</code> or
    <code>article</code> element ancestor that is a descendant of <var>node</var>.</p>
   </dd>

   <dt>If <var>node</var> has an ancestor element that is an <code>article</code> element</dt>
   <dt>If <var>node</var> has an ancestor element that is a <code>body</code> element</dt>

   <dd>
    <p>The contact information of <var>node</var> is the same as the contact information of
    the nearest <code>article</code> or <code>body</code> element ancestor, whichever is
    nearest.</p>
   </dd>

   <dt>If <var>node</var>'s <span>node document</span> has <span data-x="the body element">a body
   element</span></dt>

   <dd>
    <p>The contact information of <var>node</var> is the same as the contact information of
    <span>the body element</span> of the <code>Document</code>.</p>
   </dd>

   <dt>Otherwise</dt>

   <dd>
    <p>There is no contact information for <var>node</var>.</p>
   </dd>
  </dl>

  <p>User agents may expose the contact information of a node to the user, or use it for other
  purposes, such as indexing sections based on the sections' contact information.</p>

  </div>

  <div class="example">

   <p>In this example the footer contains contact information and a copyright notice.</p>

   <pre><code class="html">&lt;footer>
 &lt;address>
  For more details, contact
  &lt;a href="mailto:js@example.com">John Smith&lt;/a>.
 &lt;/address>
 &lt;p>&lt;small>&copy; copyright 2038 Example Corp.&lt;/small>&lt;/p>
&lt;/footer></code></pre>

  </div>



  <h4><span id="headings-and-outlines"></span><span id="outlines"></span>Headings and outlines</h4>

  <p><code>h1</code>&ndash;<code>h6</code> elements have a <dfn>heading level</dfn>, which is given
  by the number in the element's name.</p>

  <p>These elements <span data-x="represents">represent</span> <dfn
  data-x="concept-heading">headings</dfn>. The lower a <span
  data-x="concept-heading">heading</span>'s <span>heading level</span> is, the fewer ancestor
  sections the <span data-x="concept-heading">heading</span> has.</p>

  <p>The <dfn>outline</dfn> is all <span data-x="concept-heading">headings</span> in a
  document, in <span>tree order</span>.</p>

  <p>The <span>outline</span> should be used for generating document outlines, for example when
  generating tables of contents. When creating an interactive table of contents, entries should
  jump the user to the relevant <span data-x="concept-heading">heading</span>.</p>

  <p>If a document has one or more <span data-x="concept-heading">headings</span>, at least a
  single <span data-x="concept-heading">heading</span> within the <span>outline</span> should have
  a <span>heading level</span> of 1.</p>

  <p>Each <span data-x="concept-heading">heading</span> following another <span
  data-x="concept-heading">heading</span> <var>lead</var> in the <span>outline</span> must
  have a <span>heading level</span> that is less than, equal to, or 1 greater than <var>lead</var>'s
  <span>heading level</span>.</p>

  <div class="example">
   <p>The following example is non-conforming:</p>

   <pre class="bad"><code class="html">&lt;body&gt;
 &lt;h1&gt;Apples&lt;/h1&gt;
 &lt;p&gt;Apples are fruit.&lt;/p&gt;
 &lt;section&gt;
  &lt;h3&gt;Taste&lt;/h3&gt;
  &lt;p&gt;They taste lovely.&lt;/p&gt;
 &lt;/section&gt;
&lt;/body&gt;</code></pre>

   <p>It could be written as follows and then it would be conforming:</p>

   <pre><code class="html">&lt;body&gt;
 &lt;h1&gt;Apples&lt;/h1&gt;
 &lt;p&gt;Apples are fruit.&lt;/p&gt;
 &lt;section&gt;
  &lt;h2&gt;Taste&lt;/h2&gt;
  &lt;p&gt;They taste lovely.&lt;/p&gt;
 &lt;/section&gt;
&lt;/body&gt;</code></pre>

  </div>


  <h5>Sample outlines</h5>

  <div class="example">
   <p>The following markup fragment:</p>

   <pre><code class="html">&lt;body>
  &lt;hgroup id="document-title">
    &lt;h1>HTML: Living Standard&lt;/h1>
    &lt;p>Last Updated 12 August 2016&lt;/p>
  &lt;/hgroup>
  &lt;p>Some intro to the document.&lt;/p>
  &lt;h2>Table of contents&lt;/h2>
  &lt;ol id=toc>...&lt;/ol>
  &lt;h2>First section&lt;/h2>
  &lt;p>Some intro to the first section.&lt;/p>
&lt;/body></code></pre>

   <p>...results in 3 document headings:</p>

   <ol class="brief">
    <li><p><code data-x="">&lt;h1>HTML: Living Standard&lt;/h1></code></li>

    <li><p><code data-x="">&lt;h2>Table of contents&lt;/h2></code>.</p></li>

    <li><p><code data-x="">&lt;h2>First section&lt;/h2></code>.</p></li>
   </ol>

   <p>A rendered view of the <span>outline</span> might look like:</p>

   <p><img src="/images/outline.svg" width="465" height="120" alt="Top-level section with the heading &quot;HTML: Living Standard&quot; and two subsections; &quot;Table of contents&quot; and &quot;First section&quot;." class="darkmode-aware"></p>
  </div>

  <div class="example">
   <p>First, here is a document, which is a book with very short chapters and subsections:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang=en>
&lt;title>The Tax Book (all in one page)&lt;/title>
&lt;h1>The Tax Book&lt;/h1>
&lt;h2>Earning money&lt;/h2>
&lt;p>Earning money is good.&lt;/p>
&lt;h3>Getting a job&lt;/h3>
&lt;p>To earn money you typically need a job.&lt;/p>
&lt;h2>Spending money&lt;/h2>
&lt;p>Spending is what money is mainly used for.&lt;/p>
&lt;h3>Cheap things&lt;/h3>
&lt;p>Buying cheap things often not cost-effective.&lt;/p>
&lt;h3>Expensive things&lt;/h3>
&lt;p>The most expensive thing is often not the most cost-effective either.&lt;/p>
&lt;h2>Investing money&lt;/h2>
&lt;p>You can lend your money to other people.&lt;/p>
&lt;h2>Losing money&lt;/h2>
&lt;p>If you spend money or invest money, sooner or later you will lose money.
&lt;h3>Poor judgement&lt;/h3>
&lt;p>Usually if you lose money it's because you made a mistake.&lt;/p></code></pre>

   <p>Its <span>outline</span> could be presented as follows:</p>

   <ol class="brief">
    <li>
     The Tax Book
     <ol class="brief">
      <li>
       Earning money
       <ol class="brief">
        <li>Getting a job</li>
       </ol>
      </li>
      <li>
       Spending money
       <ol class="brief">
        <li>Cheap things</li>
        <li>Expensive things</li>
       </ol>
      </li>
      <li>Investing money</li>
      <li>
       Losing money
       <ol class="brief">
        <li>Poor judgement</li>
       </ol>
      </li>
     </ol>
   </ol>

   <p>Notice that the <code>title</code> element is not a <span
   data-x="concept-heading">heading</span>.</p>
  </div>

  <div class="example">
   <p>A document can contain multiple top-level headings:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang=en>
&lt;title>Alphabetic Fruit&lt;/title>
&lt;h1>Apples&lt;/h1>
&lt;p>Pomaceous.&lt;/p>
&lt;h1>Bananas&lt;/h1>
&lt;p>Edible.&lt;/p>
&lt;h1>Carambola&lt;/h1>
&lt;p>Star.&lt;/p></code></pre>

   <p>The document's <span>outline</span> could be presented as follows:</p>

   <ol class="brief">
    <li>Apples</li>
    <li>Bananas</li>
    <li>Carambola</li>
   </ol>
  </div>

  <div class="example">
   <p><code>header</code> elements do not influence the <span>outline</span> of a
   document:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang="en">
&lt;title>We're adopting a child! &mdash; Ray's blog&lt;/title>
&lt;h1>Ray's blog&lt;/h1>
&lt;article>
 &lt;header>
  &lt;nav>
   &lt;a href="?t=-1d">Yesterday&lt;/a>;
   &lt;a href="?t=-7d">Last week&lt;/a>;
   &lt;a href="?t=-1m">Last month&lt;/a>
  &lt;/nav>
  &lt;h2>We're adopting a child!&lt;/h2>
 &lt;/header>
 &lt;p>As of today, Janine and I have signed the papers to become
 the proud parents of baby Diane! We've been looking forward to
 this day for weeks.&lt;/p>
&lt;/article>
&lt;/html></code></pre>

   <p>The document's <span>outline</span> could be presented as follows:</p>

   <ol class="brief">
    <li> Ray's blog
     <ol class="brief">
      <li> We're adopting a child!
     </ol>
   </ol>
  </div>

  <hr>

  <div class="example">
   <p>The following example is conforming, but not encouraged as it has no <span
   data-x="concept-heading">heading</span> whose <span>heading level</span> is 1:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang=en>
&lt;title>Alphabetic Fruit&lt;/title>
&lt;section>
 &lt;h2>Apples&lt;/h2>
 &lt;p>Pomaceous.&lt;/p>
&lt;/section>
&lt;section>
 &lt;h2>Bananas&lt;/h2>
 &lt;p>Edible.&lt;/p>
&lt;/section>
&lt;section>
 &lt;h2>Carambola&lt;/h2>
 &lt;p>Star.&lt;/p>
&lt;/section></code></pre>

   <p>The document's <span>outline</span> could be presented as follows:</p>

   <ol class="brief">
    <li>
     <ol class="brief">
      <li>Apples</li>
      <li>Bananas</li>
      <li>Carambola</li>
     </ol>
    </li>
   </ol>
  </div>

  <div class="example">
   <p>The following example is conforming, but not encouraged as the first <span
   data-x="concept-heading">heading</span>'s <span>heading level</span> is not 1:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang=en>
&lt;title>Feathers on The Site of Encyclopedic Knowledge&lt;/title>
 &lt;h2>A plea from our caretakers&lt;/h2>
 &lt;p>Please, we beg of you, send help! We're stuck in the server room!&lt;/p>
&lt;h1>Feathers&lt;/h1>
&lt;p>Epidermal growths.&lt;/p></code></pre>

  <p>The document's <span>outline</span> could be presented as follows:</p>

  <ol class="brief">
    <li>
     <ol class="brief">
      <li>A plea from our caretakers</li>
     </ol>
    </li>
    <li>Feathers</li>
   </ol>
  </div>


  <h5>Exposing outlines to users</h5>

  <p>User agents are encouraged to expose page <span data-x="outline">outlines</span> to users to
  aid in navigation. This is especially true for non-visual media, e.g. screen readers.</p>

  <div class="example">
   <p>For instance, a user agent could map the arrow keys as follows:</p>

   <dl>
    <dt><kbd><kbd>Shift</kbd> + <kbd>&larr; Left</kbd></kbd></dt>
    <dd>Go to previous heading</dd>

    <dt><kbd><kbd>Shift</kbd> + <kbd>&rarr; Right</kbd></kbd></dt>
    <dd>Go to next heading</dd>

    <dt><kbd><kbd>Shift</kbd> + <kbd>&uarr; Up</kbd></kbd></dt>
    <dd>Go to next heading whose <span data-x="heading level">level</span> is one less than the
    current heading's level</dd>

    <dt><kbd><kbd>Shift</kbd> + <kbd>&darr; Down</kbd></kbd></dt>
    <dd>Go to next heading whose <span data-x="heading level">level</span> is the same as the
    current heading's level</dd>
   </dl>
  </div>


  <h4>Usage summary</h4>

  <!-- NON-NORMATIVE SECTION -->

  <table class="vertical-summary-table">
   <thead>
    <tr>
     <th rowspan=2>Element
     <th>Purpose
    <tr>
     <th>Example
   <tbody>
    <tr>
     <td rowspan=2><code>body</code>
     <td><!--REPRESENTS body-->
    <tr>
     <td><pre class="example"><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang="en">
 &lt;head> &lt;title>Steve Hill's Home Page&lt;/title> &lt;/head>
 <strong>&lt;body></strong> &lt;p>Hard Trance is My Life.&lt;/p> <strong>&lt;/body></strong>
&lt;/html></code></pre>

    <tr>
     <td rowspan=2><code>article</code>
     <td><!--REPRESENTS article-->
    <tr>
     <td><pre class="example"><code class="html"><strong>&lt;article></strong>
 &lt;img src="/tumblr_masqy2s5yn1rzfqbpo1_500.jpg" alt="Yellow smiley face with the caption 'masif'">
 &lt;p>My fave Masif tee so far!&lt;/p>
 &lt;footer>Posted 2 days ago&lt;/footer>
<strong>&lt;/article></strong>
<strong>&lt;article></strong>
 &lt;img src="/tumblr_m9tf6wSr6W1rzfqbpo1_500.jpg" alt="">
 &lt;p>Happy 2nd birthday Masif Saturdays!!!&lt;/p>
 &lt;footer>Posted 3 weeks ago&lt;/footer>
<strong>&lt;/article></strong></code></pre>

    <tr>
     <td rowspan=2><code>section</code>
     <td><!--REPRESENTS section-->
    <tr>
     <td><pre class="example"><code class="html">&lt;h1>Biography&lt;/h1>
<strong>&lt;section></strong>
 &lt;h1>The facts&lt;/h1>
 &lt;p>1500+ shows, 14+ countries&lt;/p>
<strong>&lt;/section></strong>
<strong>&lt;section></strong>
 &lt;h1>2010/2011 figures per year&lt;/h1>
 &lt;p>100+ shows, 8+ countries&lt;/p>
<strong>&lt;/section></strong></code></pre>

    <tr>
     <td rowspan=2><code>nav</code>
     <td><!--REPRESENTS nav-->
    <tr>
     <td><pre class="example"><code class="html"><strong>&lt;nav></strong>
 &lt;p>&lt;a href="/">Home&lt;/a>
 &lt;p>&lt;a href="/biog.html">Bio&lt;/a>
 &lt;p>&lt;a href="/discog.html">Discog&lt;/a>
<strong>&lt;/nav></strong></code></pre>

    <tr>
     <td rowspan=2><code>aside</code>
     <td><!--REPRESENTS aside-->
    <tr>
     <td><pre class="example"><code class="html">&lt;h1>Music&lt;/h1>
&lt;p>As any burner can tell you, the event has a lot of trance.&lt;/p>
<strong>&lt;aside></strong>You can buy the music we played at our &lt;a href="buy.html">playlist page&lt;/a>.<strong>&lt;/aside></strong>
&lt;p>This year we played a kind of trance that originated in Belgium, Germany, and the Netherlands in the mid-90s.&lt;/p></code></pre>

    <tr>
     <td rowspan=2><code>h1</code>&ndash;<code>h6</code>
     <td>A heading
    <tr>
     <td><pre class="example"><code class="html"><strong>&lt;h1></strong>The Guide To Music On The Playa<strong>&lt;/h1></strong>
<strong>&lt;h2></strong>The Main Stage<strong>&lt;/h2></strong>
&lt;p>If you want to play on a stage, you <!--non-normative-->should bring one.&lt;/p>
<strong>&lt;h2></strong>Amplified Music<strong>&lt;/h2></strong>
&lt;p>Amplifiers up to 300W or 90dB are welcome.&lt;/p></code></pre>

    <tr>
     <td rowspan=2><code>hgroup</code>
     <td><!--REPRESENTS hgroup-->
    <tr>
     <td><pre class="example"><code class="html"><strong>&lt;hgroup></strong>
 &lt;h1>Burning Music&lt;/h1>
 &lt;p>The Guide To Music On The Playa&lt;/p>
<strong>&lt;/hgroup></strong>
&lt;section>
 <strong>&lt;hgroup></strong>
  &lt;h1>Main Stage&lt;/h1>
  &lt;p>The Fiction Of A Music Festival&lt;/p>
 <strong>&lt;/hgroup></strong>
 &lt;p>If you want to play on a stage, you <!--non-normative-->should bring one.&lt;/p>
&lt;/section>
&lt;section>
 <strong>&lt;hgroup></strong>
  &lt;h1>Loudness!&lt;/h1>
  &lt;p>Questions About Amplified Music&lt;/p>
 <strong>&lt;/hgroup></strong>
 &lt;p>Amplifiers up to 300W or 90dB are welcome.&lt;/p>
&lt;/section></code></pre>

    <tr>
     <td rowspan=2><code>header</code>
     <td><!--REPRESENTS header-->
    <tr>
     <td><pre class="example"><code class="html">&lt;article>
 <strong>&lt;header></strong>
  &lt;h1>Hard Trance is My Life&lt;/h1>
  &lt;p>By DJ Steve Hill and Technikal&lt;/p>
 <strong>&lt;/header></strong>
 &lt;p>The album with the amusing punctuation has red artwork.&lt;/p>
&lt;/article></code></pre>

    <tr>
     <td rowspan=2><code>footer</code>
     <td><!--REPRESENTS footer-->
    <tr>
     <td><pre class="example"><code class="html">&lt;article>
 &lt;h1>Hard Trance is My Life&lt;/h1>
 &lt;p>The album with the amusing punctuation has red artwork.&lt;/p>
 <strong>&lt;footer></strong>
  &lt;p>Artists: DJ Steve Hill and Technikal&lt;/p>
 <strong>&lt;/footer></strong>
&lt;/article></code></pre>

  </table>


  <h5>Article or section?</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>A <code>section</code> forms part of something else. An <code>article</code> is its own thing.
  But how does one know which is which? Mostly the real answer is "it depends on author intent".</p>

  <p>For example, one could imagine a book with a "Granny Smith" chapter that just said "These
  juicy, green apples make a great filling for apple pies."; that would be a <code>section</code>
  because there'd be lots of other chapters on (maybe) other kinds of apples.</p>

  <p>On the other hand, one could imagine a tweet or reddit comment or tumblr post or newspaper
  classified ad that just said "Granny Smith. These juicy, green apples make a great filling for
  apple pies."; it would then be <code>article</code>s because that was the whole thing.</p>

  <p>A comment on an article is not part of the <code>article</code> on which it is commenting,
  therefore it is its own <code>article</code>.</p>



  <h3 split-filename="grouping-content">Grouping content</h3>

  <h4>The <dfn element><code>p</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-p">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-p">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLParagraphElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  // <a href="#HTMLParagraphElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLParagraphElement</code>.</dd>
  </dl>

  <p>The <code>p</code> element <span>represents</span> a <span>paragraph</span>.</p>

  <p class="note">While paragraphs are usually represented in visual media by blocks of text that
  are physically separated from adjacent blocks through blank lines, a style sheet or user agent
  would be equally justified in presenting paragraph breaks in a different manner, for instance
  using inline pilcrows (&#x00B6;).</p>

  <div class="example">
   <p>The following examples are conforming HTML fragments:</p>
   <pre><code class="html">&lt;p&gt;The little kitten gently seated herself on a piece of
carpet. Later in her life, this would be referred to as the time the
cat sat on the mat.&lt;/p&gt;</code></pre>
   <pre><code class="html">&lt;fieldset&gt;
 &lt;legend&gt;Personal information&lt;/legend&gt;
 &lt;p&gt;
   &lt;label&gt;Name: &lt;input name="n"&gt;&lt;/label&gt;
   &lt;label&gt;&lt;input name="anon" type="checkbox"&gt; Hide from other users&lt;/label&gt;
 &lt;/p&gt;
 &lt;p&gt;&lt;label&gt;Address: &lt;textarea name="a"&gt;&lt;/textarea&gt;&lt;/label&gt;&lt;/p&gt;
&lt;/fieldset&gt;</code></pre>
   <pre><code class="html">&lt;p&gt;There was once an example from Femley,&lt;br&gt;
Whose markup was of dubious quality.&lt;br&gt;
The validator complained,&lt;br&gt;
So the author was pained,&lt;br&gt;
To move the error from the markup to the rhyming.&lt;/p&gt;</code></pre>
  </div>

  <p>The <code>p</code> element should not be used when a more specific element is more
  appropriate.</p>

  <div class="example">

   <p>The following example is technically correct:</p>

<pre><code class="html">&lt;section&gt;
 &lt;!-- ... --&gt;
 &lt;p&gt;Last modified: 2001-04-23&lt;/p&gt;
 &lt;p&gt;Author: fred@example.com&lt;/p&gt;
&lt;/section&gt;</code></pre>

   <p>However, it would be better marked-up as:</p>

<pre><code class="html">&lt;section&gt;
 &lt;!-- ... --&gt;
 &lt;footer&gt;Last modified: 2001-04-23&lt;/footer&gt;
 &lt;address&gt;Author: fred@example.com&lt;/address&gt;
&lt;/section&gt;</code></pre>

   <p>Or:</p>

<pre><code class="html">&lt;section&gt;
 &lt;!-- ... --&gt;
 &lt;footer&gt;
  &lt;p&gt;Last modified: 2001-04-23&lt;/p&gt;
  &lt;address&gt;Author: fred@example.com&lt;/address&gt;
 &lt;/footer&gt;
&lt;/section&gt;</code></pre>

  </div>

  <div class="note">

   <p>List elements (in particular, <code>ol</code> and <code>ul</code> elements) cannot be children
   of <code>p</code> elements. When a sentence contains a bulleted list, therefore, one might wonder
   how it should be marked up.</p>

   <div class="example">
    <p>For instance, this fantastic sentence has bullets relating to</p>
    <ul>
     <li>wizards,
     <li>faster-than-light travel, and
     <li>telepathy,
    </ul>
    <p>and is further discussed below.</p>
   </div>

   <p>The solution is to realize that a <i data-x="paragraph">paragraph</i>, in HTML terms, is not a
   logical concept, but a structural one. In the fantastic example above, there are actually
   <em>five</em> <span data-x="paragraph">paragraphs</span> as defined by this specification: one
   before the list, one for each bullet, and one after the list.</p>

   <div class="example">

    <p>The markup for the above example could therefore be:</p>

    <pre><code class="html">&lt;p>For instance, this fantastic sentence has bullets relating to&lt;/p>
&lt;ul>
 &lt;li>wizards,
 &lt;li>faster-than-light travel, and
 &lt;li>telepathy,
&lt;/ul>
&lt;p>and is further discussed below.&lt;/p></code></pre>

   </div>

   <p>Authors wishing to conveniently style such "logical" paragraphs consisting of multiple
   "structural" paragraphs can use the <code>div</code> element instead of the <code>p</code>
   element.</p>

   <div class="example">

    <p>Thus for instance the above example could become the following:</p>

    <pre><code class="html">&lt;div>For instance, this fantastic sentence has bullets relating to
&lt;ul>
 &lt;li>wizards,
 &lt;li>faster-than-light travel, and
 &lt;li>telepathy,
&lt;/ul>
and is further discussed below.&lt;/div></code></pre>

    <p>This example still has five structural paragraphs, but now the author can style just the
    <code>div</code> instead of having to consider each part of the example separately.</p>

   </div>

  </div>



  <h4>The <dfn element><code>hr</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dd>As a child of a <code>select</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="concept-content-nothing">Nothing</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-hr">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-hr">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLHRElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  // <a href="#HTMLHRElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLHRElement</code>.</dd>
  </dl>

  <p>The <code>hr</code> element <span>represents</span> a <span>paragraph</span>-level thematic
  break, e.g., a scene change in a story, or a transition to another topic within a section of a
  reference book; alternatively, it represents a separator between a set of options of a
  <code>select</code> element.</p>

  <div class="example">

   <p>The following fictional extract from a project manual shows two sections that use the
   <code>hr</code> element to separate topics within the section.</p>

   <pre><code class="html">&lt;section>
 &lt;h1>Communication&lt;/h1>
 &lt;p>There are various methods of communication. This section
 covers a few of the important ones used by the project.&lt;/p>
 <strong>&lt;hr></strong>
 &lt;p>Communication stones seem to come in pairs and have mysterious
 properties:&lt;/p>
 &lt;ul>
  &lt;li>They can transfer thoughts in two directions once activated
  if used alone.&lt;/li>
  &lt;li>If used with another device, they can transfer one's
  consciousness to another body.&lt;/li>
  &lt;li>If both stones are used with another device, the
  consciousnesses switch bodies.&lt;/li>
 &lt;/ul>
 <strong>&lt;hr></strong>
 &lt;p>Radios use the electromagnetic spectrum in the meter range and
 longer.&lt;/p>
 <strong>&lt;hr></strong>
 &lt;p>Signal flares use the electromagnetic spectrum in the
 nanometer range.&lt;/p>
&lt;/section>
&lt;section>
 &lt;h1>Food&lt;/h1>
 &lt;p>All food at the project is rationed:&lt;/p>
 &lt;dl>
  &lt;dt>Potatoes&lt;/dt>
  &lt;dd>Two per day&lt;/dd>
  &lt;dt>Soup&lt;/dt>
  &lt;dd>One bowl per day&lt;/dd>
 &lt;/dl>
 <strong>&lt;hr></strong>
 &lt;p>Cooking is done by the chefs on a set rotation.&lt;/p>
&lt;/section></code></pre>

   <p>There is no need for an <code>hr</code> element between the sections themselves, since the
   <code>section</code> elements and the <code>h1</code> elements imply thematic changes
   themselves.</p>

  </div>

  <div class="example">

   <p>The following extract from <cite>Pandora's Star</cite> by Peter F. Hamilton shows two
   paragraphs that precede a scene change and the paragraph that follows it. The scene change,
   represented in the printed book by a gap containing a solitary centered star between the second
   and third paragraphs, is here represented using the <code>hr</code> element.</p>

   <!-- ISBN 1-4050-0020-1; bottom of page 14 -->

   <pre lang="en-GB"><code class="html">&lt;p>Dudley was ninety-two, in his second life, and fast approaching
time for another rejuvenation. Despite his body having the physical
age of a standard fifty-year-old, the prospect of a long degrading
campaign within academia was one he regarded with dread. For a
supposedly advanced civilization, the Intersolar Commonwealth could be
appallingly backward at times, not to mention cruel.&lt;/p>
&lt;p>&lt;i>Maybe it won't be that bad&lt;/i>, he told himself. The lie was
comforting enough to get him through the rest of the night's
shift.&lt;/p>
<strong>&lt;hr></strong>
&lt;p>The Carlton AllLander drove Dudley home just after dawn. Like the
astronomer, the vehicle was old and worn, but perfectly capable of
doing its job. It had a cheap diesel engine, common enough on a
semi-frontier world like Gralmond, although its drive array was a
thoroughly modern photoneural processor. With its high suspension and
deep-tread tyres it could plough along the dirt track to the
observatory in all weather and seasons, including the metre-deep snow
of Gralmond's winters.&lt;/p></code></pre>

  </div>

  <p class="note">The <code>hr</code> element does not affect the document's
  <span>outline</span>.</p>

  <h4>The <dfn element><code>pre</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-pre">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-pre">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLPreElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  // <a href="#HTMLPreElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLPreElement</code>.</dd>
  </dl>

  <p>The <code>pre</code> element <span>represents</span> a block of preformatted text, in which
  structure is represented by typographic conventions rather than by elements.</p>

  <p class="note">In <span>the HTML syntax</span>, a leading newline character immediately following
  the <code>pre</code> element start tag is stripped.</p>

  <p>Some examples of cases where the <code>pre</code> element could be used:</p>

  <ul>
   <li>Including an email, with paragraphs indicated by blank lines, lists indicated by lines
   prefixed with a bullet, and so on.</li>

   <li>Including fragments of computer code, with structure indicated according to the conventions
   of that language.</li>

   <li>Displaying ASCII art.</li>
  </ul>

  <p class="note">Authors are encouraged to consider how preformatted text will be experienced when
  the formatting is lost, as will be the case for users of speech synthesizers, braille displays,
  and the like. For cases like ASCII art, it is likely that an alternative presentation, such as a
  textual description, would be more universally accessible to the readers of the document.</p>

  <p>To represent a block of computer code, the <code>pre</code> element can be used with a
  <code>code</code> element; to represent a block of computer output the <code>pre</code> element
  can be used with a <code>samp</code> element. Similarly, the <code>kbd</code> element can be used
  within a <code>pre</code> element to indicate text that the user is to enter.</p>

  <div w-nodev>

  <p class="note">This element <a href="#bidireq">has rendering requirements involving the
  bidirectional algorithm</a>.</p>
  </div>

  <div class="example">

   <p>In the following snippet, a sample of computer code is presented.</p>

   <pre><code class="html">&lt;p>This is the &lt;code>Panel&lt;/code> constructor:&lt;/p>
&lt;pre>&lt;code>function Panel(element, canClose, closeHandler) {
  this.element = element;
  this.canClose = canClose;
  this.closeHandler = function () { if (closeHandler) closeHandler() };
}&lt;/code>&lt;/pre></code></pre>

  </div>

  <div class="example">

   <p>In the following snippet, <code>samp</code> and <code>kbd</code> elements are mixed in the
   contents of a <code>pre</code> element to show a session of Zork I.</p>

   <pre><code class="html">&lt;pre>&lt;samp>You are in an open field west of a big white house with a boarded
front door.
There is a small mailbox here.

>&lt;/samp> &lt;kbd>open mailbox&lt;/kbd>

&lt;samp>Opening the mailbox reveals:
A leaflet.

>&lt;/samp>&lt;/pre></code></pre>

  </div>

  <div class="example">

   <p>The following shows a contemporary poem that uses the <code>pre</code> element to preserve its
   unusual formatting, which forms an intrinsic part of the poem itself.</p>

   <pre><code class="html">&lt;pre>                maxling

it is with a          heart
               heavy

that i admit loss of a feline
        so           loved

a friend lost to the
        unknown
                                (night)

~cdr 11dec07&lt;/pre></code></pre>

  </div>



  <h4>The <dfn element><code>blockquote</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-blockquote-cite">cite</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-blockquote">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-blockquote">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLQuoteElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute USVString <span data-x="dom-quote-cite">cite</span>;
};</code></pre>
    <p class="note">The <code>HTMLQuoteElement</code> interface is
    also used by the <code>q</code> element.</p>
   </dd>
   <dd w-dev>Uses <code>HTMLQuoteElement</code>.</dd>
  </dl>

  <p>The <code>blockquote</code> element <span>represents</span> a section that is quoted from
  another source.</p>

  <p>Content inside a <code>blockquote</code> must be quoted from another source, whose address, if
  it has one, may be cited in the <dfn element-attr for="blockquote"><code
  data-x="attr-blockquote-cite">cite</code></dfn> attribute.</p>

  <p>If the <code data-x="attr-blockquote-cite">cite</code> attribute is present, it must be a
  <span>valid URL potentially surrounded by spaces</span>. <span w-nodev>To obtain the
  corresponding citation link, the value of the attribute must be <span data-x="encoding-parsing a
  URL">parsed</span> relative to the element's <span>node document</span>.</span> User agents may
  allow users to follow such citation links, but they are primarily intended for private use (e.g.,
  by server-side scripts collecting statistics about a site's use of quotations), not for
  readers.</p>

  <p>The content of a <code>blockquote</code> may be abbreviated or may have context added in the
  conventional manner for the text's language.</p>

  <div class="example">

   <p>For example, in English this is traditionally done using square brackets. Consider a page with
   the sentence "Jane ate the cracker. She then said she liked apples and fish."; it could be quoted
   as follows:</p>

   <pre><code class="html">&lt;blockquote>
 &lt;p>[Jane] then said she liked [...] fish.&lt;/p>
&lt;/blockquote></code></pre>

  </div>

  <p>Attribution for the quotation, if any, must be placed outside the <code>blockquote</code>
  element.</p> <!-- Eventually someone is going to ask about quotes that are self-attributing, e.g.
  someone quoting an email with full headers... -->

  <div class="example">

   <p>For example, here the attribution is given in a paragraph after the quote:</p>

   <pre><code class="html">&lt;blockquote>
 &lt;p>I contend that we are both atheists. I just believe in one fewer
 god than you do. When you understand why you dismiss all the other
 possible gods, you will understand why I dismiss yours.&lt;/p>
&lt;/blockquote>
&lt;p>&mdash; Stephen Roberts&lt;/p></code></pre>

   <p>The other examples below show other ways of showing attribution.</p>

  </div>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLQuoteElement"><code data-x="dom-quote-cite">cite</code></dfn> IDL
  attribute must <span>reflect</span> the element's <code data-x="">cite</code> content
  attribute.</p>

  </div>

  <div class="example">

   <p>Here a <code>blockquote</code> element is used in conjunction with a <code>figure</code>
   element and its <code>figcaption</code> to clearly relate a quote to its attribution (which is
   not part of the quote and therefore doesn't belong inside the <code>blockquote</code>
   itself):</p>

   <pre><code class="html">&lt;figure>
 &lt;blockquote>
  &lt;p>The truth may be puzzling. It may take some work to grapple with.
  It may be counterintuitive. It may contradict deeply held
  prejudices. It may not be consonant with what we desperately want to
  be true. But our preferences do not determine what's true. We have a
  method, and that method helps us to reach not absolute truth, only
  asymptotic approaches to the truth &mdash; never there, just closer
  and closer, always finding vast new oceans of undiscovered
  possibilities. Cleverly designed experiments are the key.&lt;/p>
 &lt;/blockquote>
 &lt;figcaption>Carl Sagan, in "&lt;cite>Wonder and Skepticism&lt;/cite>", from
 the &lt;cite>Skeptical Inquirer&lt;/cite> Volume 19, Issue 1 (January-February
 1995)&lt;/figcaption>
&lt;/figure></code></pre>

  </div>

  <div class="example">

   <p>This next example shows the use of <code>cite</code> alongside <code>blockquote</code>:</p>

   <pre><code class="html">&lt;p>His next piece was the aptly named &lt;cite>Sonnet 130&lt;/cite>:&lt;/p>
&lt;blockquote cite="https://quotes.example.org/s/sonnet130.html">
  &lt;p>My mistress' eyes are nothing like the sun,&lt;br>
  Coral is far more red, than her lips red,&lt;br>
  ...</code></pre>

  </div>

  <div class="example">

   <p>This example shows how a forum post could use <code>blockquote</code> to show what post a user
   is replying to. The <code>article</code> element is used for each post, to mark up the
   threading.</p>

   <pre><code class="html">&lt;article>
 &lt;h1>&lt;a href="https://bacon.example.com/?blog=109431">Bacon on a crowbar&lt;/a>&lt;/h1>
 &lt;article>
  &lt;header>&lt;strong>t3yw&lt;/strong> 12 points 1 hour ago&lt;/header>
  &lt;p>I bet a narwhal would love that.&lt;/p>
  &lt;footer>&lt;a href="?pid=29578">permalink&lt;/a>&lt;/footer>
  &lt;article>
   &lt;header>&lt;strong>greg&lt;/strong> 8 points 1 hour ago&lt;/header>
   &lt;blockquote>&lt;p>I bet a narwhal would love that.&lt;/p>&lt;/blockquote>
   &lt;p>Dude narwhals don't eat bacon.&lt;/p>
   &lt;footer>&lt;a href="?pid=29579">permalink&lt;/a>&lt;/footer>
   &lt;article>
    &lt;header>&lt;strong>t3yw&lt;/strong> 15 points 1 hour ago&lt;/header>
    &lt;blockquote>
     &lt;blockquote>&lt;p>I bet a narwhal would love that.&lt;/p>&lt;/blockquote>
     &lt;p>Dude narwhals don't eat bacon.&lt;/p>
    &lt;/blockquote>
    &lt;p>Next thing you'll be saying they don't get capes and wizard
    hats either!&lt;/p>
    &lt;footer>&lt;a href="?pid=29580">permalink&lt;/a>&lt;/footer>
    &lt;article>
     &lt;article>
      &lt;header>&lt;strong>boing&lt;/strong> -5 points 1 hour ago&lt;/header>
      &lt;p>narwhals are worse than ceiling cat&lt;/p>
      &lt;footer>&lt;a href="?pid=29581">permalink&lt;/a>&lt;/footer>
     &lt;/article>
    &lt;/article>
   &lt;/article>
  &lt;/article>
  &lt;article>
   &lt;header>&lt;strong>fred&lt;/strong> 1 points 23 minutes ago&lt;/header>
   &lt;blockquote>&lt;p>I bet a narwhal would love that.&lt;/p>&lt;/blockquote>
   &lt;p>I bet they'd love to peel a banana too.&lt;/p>
   &lt;footer>&lt;a href="?pid=29582">permalink&lt;/a>&lt;/footer>
  &lt;/article>
 &lt;/article>
&lt;/article></code></pre>

  </div>

  <div class="example">

   <p>This example shows the use of a <code>blockquote</code> for short snippets, demonstrating that
   one does not have to use <code>p</code> elements inside <code>blockquote</code> elements:</p>

   <pre><code class="html">&lt;p>He began his list of "lessons" with the following:&lt;/p>
&lt;blockquote>One should never assume that his side of
the issue will be recognized, let alone that it will
be conceded to have merits.&lt;/blockquote>
&lt;p>He continued with a number of similar points, ending with:&lt;/p>
&lt;blockquote>Finally, one should be prepared for the threat
of breakdown in negotiations at any given moment and not
be cowed by the possibility.&lt;/blockquote>
&lt;p>We shall now discuss these points...</code></pre>

  </div>

  <p class="note"><a href="#conversations">Examples of how to represent a conversation</a> are shown
  in a later section; it is not appropriate to use the <code>cite</code> and <code>blockquote</code>
  elements for this purpose.</p>



  <h4>The <dfn element><code>ol</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd>If the element's children include at least one <code>li</code> element: <span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>Zero or more <code>li</code> and <span data-x="script-supporting elements">script-supporting</span> elements.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-ol-reversed">reversed</code></dd>
   <dd><code data-x="attr-ol-start">start</code></dd>
   <dd><code data-x="attr-ol-type">type</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-ol">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-ol">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
<pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLOListElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute boolean <span data-x="dom-ol-reversed">reversed</span>;
  [<span>CEReactions</span>] attribute long <span data-x="dom-ol-start">start</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-ol-type">type</span>;

  // <a href="#HTMLOListElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLOListElement</code>.</dd>
  </dl>

  <p>The <code>ol</code> element <span>represents</span> a list of items, where the items have been
  intentionally ordered, such that changing the order would change the meaning of the document.</p>

  <p>The items of the list are the <code>li</code> element child nodes of the <code>ol</code>
  element, in <span>tree order</span>.</p>

  <p>The <dfn element-attr for="ol"><code data-x="attr-ol-reversed">reversed</code></dfn> attribute
  is a <span>boolean attribute</span>. If present, it indicates that the list is a descending list
  (..., 3, 2, 1). If the attribute is omitted, the list is an ascending list (1, 2, 3, ...).</p>

  <p>The <dfn element-attr for="ol"><code data-x="attr-ol-start">start</code></dfn> attribute, if
  present, must be a <span>valid integer</span>. It is used to determine the <span
  data-x="concept-ol-start">starting value</span> of the list.</p>

  <div w-nodev>

  <!-- https://github.com/whatwg/html/issues/1617
       https://github.com/whatwg/html/issues/1911 -->

  <p>An <code>ol</code> element has a <dfn data-x="concept-ol-start">starting value</dfn>, which is
  an integer determined as follows:</p>

  <ol>
   <li>
    <p>If the <code>ol</code> element has a <code data-x="attr-ol-start">start</code> attribute,
    then:</p>

    <ol>
     <li><p>Let <var>parsed</var> be the result of <span data-x="rules for parsing integers">parsing
     the value of the attribute as an integer</span>.</p></li>

     <li><p>If <var>parsed</var> is not an error, then return <var>parsed</var>.</p></li>
    </ol>
   </li>

   <li><p>If the <code>ol</code> element has a <code data-x="attr-ol-reversed">reversed</code>
   attribute, then return the number of <span data-x="list owner">owned <code>li</code>
   elements</span>.</p></li>

   <li><p>Return 1.</p></li>
  </ol>

  </div>

  <p>The <dfn element-attr for="ol"><code data-x="attr-ol-type">type</code></dfn> attribute can be
  used to specify the kind of marker to use in the list, in the cases where that matters (e.g.
  because items are to be <span>referenced</span> by their number/letter). The attribute, if
  specified, must have a value that is <span>identical to</span> one of the characters given in the
  first cell of one of the rows of the following table. <span w-nodev>The <code
  data-x="attr-ol-type">type</code> attribute represents the state given in the cell in the second
  column of the row whose first cell matches the attribute's value; if none of the cells match, or
  if the attribute is omitted, then the attribute represents the <span
  data-x="attr-ol-type-state-decimal">decimal</span> state.</span></p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Description
     <th colspan=8>Examples for values 1-3 and 3999-4001
   <tbody>
    <tr>
     <td><dfn attr-value for="ol/type"><code data-x="attr-ol-type-keyword-decimal">1</code></dfn>
     (U+0031)
     <td><dfn data-x="attr-ol-type-state-decimal">decimal</dfn>
     <td>Decimal numbers
     <td class="eg"><samp>1.</samp> <td class="eg"><samp>2.</samp> <td class="eg"><samp>3.</samp> <td class="eg">... <td class="eg"><samp>3999.</samp> <td class="eg"><samp>4000.</samp> <td class="eg"><samp>4001.</samp> <td class="eg">...
    <tr>
     <td><dfn attr-value for="ol/type"><code
     data-x="attr-ol-type-keyword-lower-alpha">a</code></dfn> (U+0061)
     <td><dfn data-x="attr-ol-type-state-lower-alpha">lower-alpha</dfn>
     <td>Lowercase latin alphabet
     <td class="eg"><samp>a.</samp> <td class="eg"><samp>b.</samp> <td class="eg"><samp>c.</samp> <td class="eg">... <td class="eg"><samp>ewu.</samp> <td class="eg"><samp>ewv.</samp> <td class="eg"><samp>eww.</samp> <td class="eg">...
    <tr>
     <td><dfn attr-value for="ol/type"><code
     data-x="attr-ol-type-keyword-upper-alpha">A</code></dfn> (U+0041)
     <td><dfn data-x="attr-ol-type-state-upper-alpha">upper-alpha</dfn>
     <td>Uppercase latin alphabet
     <td class="eg"><samp>A.</samp> <td class="eg"><samp>B.</samp> <td class="eg"><samp>C.</samp> <td class="eg">... <td class="eg"><samp>EWU.</samp> <td class="eg"><samp>EWV.</samp> <td class="eg"><samp>EWW.</samp> <td class="eg">...
    <tr>

     <td><dfn attr-value for="ol/type"><code data-x="attr-ol-type-keyword-lower-roman">i</code></dfn> (U+0069)

     <td><dfn data-x="attr-ol-type-state-lower-roman">lower-roman</dfn>
     <td>Lowercase roman numerals
     <td class="eg"><samp>i.</samp> <td class="eg"><samp>ii.</samp> <td class="eg"><samp>iii.</samp> <td class="eg">... <td class="eg"><samp>mmmcmxcix.</samp> <td class="eg"><samp>i&#x0305;v&#x0305;.</samp> <td class="eg"><samp>i&#x0305;v&#x0305;i.</samp> <td class="eg">...
    <tr>
     <td><dfn attr-value for="ol/type"><code
     data-x="attr-ol-type-keyword-upper-roman">I</code></dfn> (U+0049)
     <td><dfn data-x="attr-ol-type-state-upper-roman">upper-roman</dfn>
     <td>Uppercase roman numerals
     <td class="eg"><samp>I.</samp> <td class="eg"><samp>II.</samp> <td class="eg"><samp>III.</samp> <td class="eg">... <td class="eg"><samp>MMMCMXCIX.</samp> <td class="eg"><samp>I&#x0305;V&#x0305;.</samp> <td class="eg"><samp>I&#x0305;V&#x0305;I.</samp> <td class="eg">...
  </table>

  <div w-nodev>

  <p>User agents should render the items of the list in a manner consistent with the state of the
  <code data-x="attr-ol-type">type</code> attribute of the <code>ol</code> element. Numbers less than
  or equal to zero should always use the decimal system regardless of the <code
  data-x="attr-ol-type">type</code> attribute.</p>

  <p class="note">For CSS user agents, a mapping for this attribute to the
  <span>'list-style-type'</span> CSS property is given <a href="#decohints">in the Rendering
  section</a> (the mapping is straightforward: the states above have the same names as their
  corresponding CSS values).</p>

  </div>

  <p class="note">It is possible to redefine the default CSS list styles used to implement this
  attribute in CSS user agents; doing so will affect how list items are rendered.</p>

  <!-- v2: resuming numbering of lists from previous lists? -->

  <div w-nodev>

  <p>The <dfn attribute for="HTMLOListElement"><code data-x="dom-ol-reversed">reversed</code></dfn>
  and <dfn attribute for="HTMLOListElement"><code data-x="dom-ol-type">type</code></dfn> IDL
  attributes must <span>reflect</span> the respective content attributes of the same name.</p>

  <p>The <dfn attribute for="HTMLOListElement"><code data-x="dom-ol-start">start</code></dfn> IDL
  attribute must <span>reflect</span> the content attribute of the same name, with a <span>default
  value</span> of 1.</p>

  <p class="note">This means that the <code data-x="dom-ol-start">start</code> IDL attribute does
  not necessarily match the list's <span data-x="concept-ol-start">starting value</span>, in cases
  where the <code data-x="attr-ol-start">start</code> content attribute is omitted and the <code
  data-x="attr-ol-reversed">reversed</code> content attribute is specified.</p>

  </div>

  <div class="example">

   <p>The following markup shows a list where the order matters, and where the <code>ol</code>
   element is therefore appropriate. Compare this list to the equivalent list in the <code>ul</code>
   section to see an example of the same items using the <code>ul</code> element.</p>

   <pre><code class="html">&lt;p>I have lived in the following countries (given in the order of when
I first lived there):&lt;/p>
&lt;ol>
 &lt;li>Switzerland
 &lt;li>United Kingdom
 &lt;li>United States
 &lt;li>Norway
&lt;/ol></code></pre>

   <p>Note how changing the order of the list changes the meaning of the document. In the following
   example, changing the relative order of the first two items has changed the birthplace of the
   author:</p>

   <pre><code class="html">&lt;p>I have lived in the following countries (given in the order of when
I first lived there):&lt;/p>
&lt;ol>
 &lt;li>United Kingdom
 &lt;li>Switzerland
 &lt;li>United States
 &lt;li>Norway
&lt;/ol></code></pre>

  </div>


  <h4>The <dfn element><code>ul</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd>If the element's children include at least one <code>li</code> element: <span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>Zero or more <code>li</code> and <span data-x="script-supporting elements">script-supporting</span> elements.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-ul">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-ul">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLUListElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  // <a href="#HTMLUListElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLUListElement</code>.</dd>
  </dl>

  <p>The <code>ul</code> element <span>represents</span> a list of items, where the order of the
  items is not important &mdash; that is, where changing the order would not materially change the
  meaning of the document.</p>

  <p>The items of the list are the <code>li</code> element child nodes of the <code>ul</code>
  element.</p>

  <div class="example">

   <p>The following markup shows a list where the order does not matter, and where the
   <code>ul</code> element is therefore appropriate. Compare this list to the equivalent list in the
   <code>ol</code> section to see an example of the same items using the <code>ol</code>
   element.</p>

   <pre><code class="html">&lt;p>I have lived in the following countries:&lt;/p>
&lt;ul>
 &lt;li>Norway
 &lt;li>Switzerland
 &lt;li>United Kingdom
 &lt;li>United States
&lt;/ul></code></pre>

   <p>Note that changing the order of the list does not change the meaning of the document. The
   items in the snippet above are given in alphabetical order, but in the snippet below they are
   given in order of the size of their current account balance in 2007, without changing the meaning
   of the document whatsoever:</p>

   <pre><code class="html">&lt;p>I have lived in the following countries:&lt;/p>
&lt;ul>
 &lt;li>Switzerland
 &lt;li>Norway
 &lt;li>United Kingdom
 &lt;li>United States
&lt;/ul></code></pre>

  </div>


  <h4 id="the-menu-element">The <dfn element id="menus"><code>menu</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd>If the element's children include at least one <code>li</code> element: <span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>Zero or more <code>li</code> and <span data-x="script-supporting elements">script-supporting</span> elements.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-menu">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-menu">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
<pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLMenuElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  // <a href="#HTMLMenuElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLMenuElement</code>.</dd>
  </dl>

  <p>The <code>menu</code> element <span>represents</span> a toolbar consisting of its contents, in
  the form of an unordered list of items (represented by <code>li</code> elements), each of
  which represents a command that the user can perform or activate.</p>

  <p class="note">The <code>menu</code> element is simply a semantic alternative to <code>ul</code>
  to express an unordered list of commands (a "toolbar").</p>

  <div class="example">
   <p>In this example, a text-editing application uses a <code>menu</code> element to provide a
   series of editing commands:</p>

   <pre><code class="html">&lt;menu>
 &lt;li>&lt;button onclick="copy()">&lt;img src="copy.svg" alt="Copy">&lt;/button>&lt;/li>
 &lt;li>&lt;button onclick="cut()">&lt;img src="cut.svg" alt="Cut">&lt;/button>&lt;/li>
 &lt;li>&lt;button onclick="paste()">&lt;img src="paste.svg" alt="Paste">&lt;/button>&lt;/li>
&lt;/menu></code></pre>

   <p>Note that the styling to make this look like a conventional toolbar menu is up to the
   application.</p>
  </div>


  <h4>The <dfn element><code>li</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Inside <code>ol</code> elements.</dd>
   <dd>Inside <code>ul</code> elements.</dd>
   <dd>Inside <code>menu</code> elements.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd>If the element is not a child of an <code>ul</code> or <code>menu</code> element: <code data-x="attr-li-value">value</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-li">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-li">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
<pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLLIElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute long <span data-x="dom-li-value">value</span>;

  // <a href="#HTMLLIElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLLIElement</code>.</dd>
  </dl>

  <p>The <code>li</code> element <span>represents</span> a list item. If its parent element is an
  <code>ol</code>, <code>ul</code>, or <code>menu</code> element, then the element is an item of the
  parent element's list, as defined for those elements. Otherwise, the list item has no defined
  list-related relationship to any other <code>li</code> element.</p>

  <p>The <dfn element-attr for="li"><code data-x="attr-li-value">value</code></dfn> attribute, if
  present, must be a <span>valid integer</span>. It is used to determine the <span>ordinal
  value</span> of the list item, when the <code>li</code>'s <span>list owner</span> is an
  <code>ol</code> element.</p>

  <div w-nodev>

  <hr>

  <p>Any element whose <span>computed value</span> of <span>'display'</span> is 'list-item' has a
  <dfn>list owner</dfn>, which is determined as follows:</p>

  <ol>
   <li><p>If the element is not <span>being rendered</span>, return null; the element has no
   <span>list owner</span>.</p>

   <li><p>Let <var>ancestor</var> be the element's parent.</p></li>

   <li><p>If the element has an <code>ol</code>, <code>ul</code>, or <code>menu</code> ancestor, set
   <var>ancestor</var> to the closest such ancestor element.</p></li>

   <li>
    <p>Return the closest inclusive ancestor of <var>ancestor</var> that produces a <span>CSS
    box</span>.</p>

    <p class="note">Such an element will always exist, as at the very least the <span>document
    element</span> will always produce a <span>CSS box</span>.</p>
   </li>
  </ol>

  <p>To determine the <dfn>ordinal value</dfn> of each element owned by a given <span>list
  owner</span> <var>owner</var>, perform the following steps:</p>

  <ol>
   <li><p>Let <var>i</var> be 1.</p></li>

   <li><p>If <var>owner</var> is an <code>ol</code> element, let <var>numbering</var> be
   <var>owner</var>'s <span data-x="concept-ol-start">starting value</span>. Otherwise, let
   <var>numbering</var> be 1.</p></li>

   <li><p><i>Loop</i>: If <var>i</var> is greater than the number of <span
   data-x="list owner">list items that <var>owner</var> owns</span>, then return; all
   of <var>owner</var>'s <span data-x="list owner">owned list items</span> have been assigned <span
   data-x="ordinal value">ordinal values</span>.</p></li>

   <li><p>Let <var>item</var> be the <var>i</var>th of <var>owner</var>'s <span data-x="list
   owner">owned list items</span>, in <span>tree order</span>.</p>

   <li>
    <p>If <var>item</var> is an <code>li</code> element that has a <code
    data-x="attr-li-value">value</code> attribute, then:</p>

    <ol>
     <li><p>Let <var>parsed</var> be the result of <span data-x="rules for parsing integers">parsing
     the value of the attribute as an integer</span>.</p></li>

     <li><p>If <var>parsed</var> is not an error, then set <var>numbering</var> to
     <var>parsed</var>.</p></li>
    </ol>
   </li>

   <li><p>The <span>ordinal value</span> of <var>item</var> is <var>numbering</var>.</p></li>

   <li><p>If <var>owner</var> is an <code>ol</code> element, and <var>owner</var> has a <code
   data-x="attr-ol-reversed">reversed</code> attribute, decrement <var>numbering</var> by 1;
   otherwise, increment <var>numbering</var> by 1.</p></li>

   <li><p>Increment <var>i</var> by 1.</p></li>

   <li><p>Go to the step labeled <i>loop</i>.</p>
  </ol>

  <hr>

  <p>The <dfn attribute for="HTMLLIElement"><code data-x="dom-li-value">value</code></dfn> IDL
  attribute must <span>reflect</span> the value of the <code data-x="attr-li-value">value</code>
  content attribute.</p>

  <div class="example">
   <p>The element's <code data-x="dom-li-value">value</code> IDL attribute does not directly
   correspond to its <span>ordinal value</span>; it simply <span data-x="reflect">reflects</span>
   the content attribute. For example, given this list:

   <pre><code class="html">&lt;ol>
 &lt;li>Item 1
 &lt;li value="3">Item 3
 &lt;li>Item 4
&lt;/ol></code></pre>

   <p>The <span data-x="ordinal value">ordinal values</span> are 1, 3, and 4, whereas the <code
   data-x="dom-li-value">value</code> IDL attributes return 0, 3, 0 on getting.</p>
  </div>

  </div>

  <div class="example">

   <p>The following example, the top ten movies are listed (in reverse order). Note the way the list
   is given a title by using a <code>figure</code> element and its <code>figcaption</code>
   element.</p>

   <pre><code class="html">&lt;figure>
 &lt;figcaption>The top 10 movies of all time&lt;/figcaption>
 &lt;ol>
  &lt;li value="10">&lt;cite>Josie and the Pussycats&lt;/cite>, 2001&lt;/li>
  &lt;li value="9">&lt;cite lang="sh">Црна мачка, бели мачор&lt;/cite>, 1998&lt;/li>
  &lt;li value="8">&lt;cite>A Bug's Life&lt;/cite>, 1998&lt;/li>
  &lt;li value="7">&lt;cite>Toy Story&lt;/cite>, 1995&lt;/li>
  &lt;li value="6">&lt;cite>Monsters, Inc&lt;/cite>, 2001&lt;/li>
  &lt;li value="5">&lt;cite>Cars&lt;/cite>, 2006&lt;/li>
  &lt;li value="4">&lt;cite>Toy Story 2&lt;/cite>, 1999&lt;/li>
  &lt;li value="3">&lt;cite>Finding Nemo&lt;/cite>, 2003&lt;/li>
  &lt;li value="2">&lt;cite>The Incredibles&lt;/cite>, 2004&lt;/li>
  &lt;li value="1">&lt;cite>Ratatouille&lt;/cite>, 2007&lt;/li>
 &lt;/ol>
&lt;/figure></code></pre>

   <p>The markup could also be written as follows, using the <code
   data-x="attr-ol-reversed">reversed</code> attribute on the <code>ol</code> element:</p>

   <pre><code class="html">&lt;figure>
 &lt;figcaption>The top 10 movies of all time&lt;/figcaption>
 &lt;ol reversed>
  &lt;li>&lt;cite>Josie and the Pussycats&lt;/cite>, 2001&lt;/li>
  &lt;li>&lt;cite lang="sh">Црна мачка, бели мачор&lt;/cite>, 1998&lt;/li>
  &lt;li>&lt;cite>A Bug's Life&lt;/cite>, 1998&lt;/li>
  &lt;li>&lt;cite>Toy Story&lt;/cite>, 1995&lt;/li>
  &lt;li>&lt;cite>Monsters, Inc&lt;/cite>, 2001&lt;/li>
  &lt;li>&lt;cite>Cars&lt;/cite>, 2006&lt;/li>
  &lt;li>&lt;cite>Toy Story 2&lt;/cite>, 1999&lt;/li>
  &lt;li>&lt;cite>Finding Nemo&lt;/cite>, 2003&lt;/li>
  &lt;li>&lt;cite>The Incredibles&lt;/cite>, 2004&lt;/li>
  &lt;li>&lt;cite>Ratatouille&lt;/cite>, 2007&lt;/li>
 &lt;/ol>
&lt;/figure></code></pre>
  </div>

  <p class="note">While it is conforming to include heading elements (e.g. <code>h1</code>) inside
  <code>li</code> elements, it likely does not convey the semantics that the author intended. A
  heading starts a new section, so a heading in a list implicitly splits the list into spanning
  multiple sections.</p>



  <h4>The <dfn element><code>dl</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd>If the element's children include at least one name-value group: <span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>Either: Zero or more groups each consisting of one or more <code>dt</code> elements followed by one or more <code>dd</code> elements, optionally intermixed with <span>script-supporting elements</span>.</dd>
   <dd>Or: One or more <code>div</code> elements, optionally intermixed with <span>script-supporting elements</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-dl">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-dl">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLDListElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  // <a href="#HTMLDListElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLDListElement</code>.</dd>
  </dl>

  <p>The <code>dl</code> element <span>represents</span> an association list consisting of zero or
  more name-value groups (a description list). A name-value group consists of one or more names
  (<code>dt</code> elements, possibly as children of a <code>div</code> element child) followed by
  one or more values (<code>dd</code> elements, possibly as children of a <code>div</code> element
  child), ignoring any nodes other than <code>dt</code> and <code>dd</code> element children, and
  <code>dt</code> and <code>dd</code> elements that are children of <code>div</code> element
  children. Within a single <code>dl</code> element, there should not be more than one
  <code>dt</code> element for each name.</p>

  <p>Name-value groups may be terms and definitions, metadata topics and values, questions and
  answers, or any other groups of name-value data.</p>

  <p>The values within a group are alternatives; multiple paragraphs forming part of the same value
  must all be given within the same <code>dd</code> element.</p>

  <p>The order of the list of groups, and of the names and values within each group, may be
  significant.</p>

  <p>In order to annotate groups with <span>microdata</span> attributes, or other <span>global
  attributes</span> that apply to whole groups, or just for styling purposes, each group in a
  <code>dl</code> element can be wrapped in a <code>div</code> element. This does not change the
  semantics of the <code>dl</code> element.</p>

  <div w-nodev>

  <p>The name-value groups of a <code>dl</code> element <var>dl</var> are determined using the
  following algorithm. A name-value group has a name (a list of <code>dt</code> elements, initially
  empty) and a value (a list of <code>dd</code> elements, initially empty).</p>

  <ol>
   <li><p>Let <var>groups</var> be an empty list of name-value groups.</p></li>

   <li><p>Let <var>current</var> be a new name-value group.</p></li>

   <li><p>Let <var>seenDd</var> be false.</p></li>

   <li><p>Let <var>child</var> be <var>dl</var>'s <span>first child</span>.</p></li>

   <li><p>Let <var>grandchild</var> be null.</p></li>

   <li>
    <p>While <var>child</var> is not null:</p>

    <ol>
     <li>
      <p>If <var>child</var> is a <code>div</code> element, then:</p>

      <ol>
       <li><p>Let <var>grandchild</var> be <var>child</var>'s <span>first child</span>.</p></li>

       <li>
        <p>While <var>grandchild</var> is not null:</p>

        <ol>
         <li><p><span>Process <code>dt</code> or <code>dd</code></span> for
         <var>grandchild</var>.</p></li>

         <li><p>Set <var>grandchild</var> to <var>grandchild</var>'s <span>next
         sibling</span>.</p></li>
        </ol>
       </li>
      </ol>
     </li>

     <li><p>Otherwise, <span>process <code>dt</code> or <code>dd</code></span> for
     <var>child</var>.</p></li>

     <li><p>Set <var>child</var> to <var>child</var>'s <span>next sibling</span>.</p></li>
    </ol>
   </li>

   <li><p>If <var>current</var> is not empty, then append <var>current</var> to
   <var>groups</var>.</p></li>

   <li><p>Return <var>groups</var>.</p></li>
  </ol>

  <p>To <dfn>process <code>dt</code> or <code>dd</code></dfn> for a node <var>node</var> means to
  follow these steps:</p>

  <ol>
   <li><p>Let <var>groups</var>, <var>current</var>, and <var>seenDd</var> be the same variables as
   those of the same name in the algorithm that invoked these steps.</p></li>

   <li>
    <p>If <var>node</var> is a <code>dt</code> element, then:</p>

    <ol>
     <li><p>If <var>seenDd</var> is true, then append <var>current</var> to <var>groups</var>, set
     <var>current</var> to a new name-value group, and set <var>seenDd</var> to false.</p></li>

     <li><p>Append <var>node</var> to <var>current</var>'s name.</p></li>
    </ol>
   </li>

   <li><p>Otherwise, if <var>node</var> is a <code>dd</code> element, then append <var>node</var> to
   <var>current</var>'s value and set <var>seenDd</var> to true.</p></li>
  </ol>

  <p class="note">When a name-value group has an empty list as name or value, it is often due to
  accidentally using <code>dd</code> elements in the place of <code>dt</code> elements and vice
  versa. Conformance checkers can spot such mistakes and might be able to advise authors how to
  correctly use the markup.</p>

  </div>

  <div class="example">

   <p>In the following example, one entry ("Authors") is linked to two values ("John" and
   "Luke").</p>

   <pre><code class="html">&lt;dl&gt;
 &lt;dt&gt; Authors
 &lt;dd&gt; John
 &lt;dd&gt; Luke
 &lt;dt&gt; Editor
 &lt;dd&gt; Frank
&lt;/dl&gt;</code></pre>
  </div>

  <div class="example">

   <p>In the following example, one definition is linked to two terms.</p>

   <pre><code class="html">&lt;dl&gt;
 &lt;dt lang="en-US"&gt; &lt;dfn>color&lt;/dfn> &lt;/dt&gt;
 <span lang="en-GB">&lt;dt lang="en-GB"&gt; &lt;dfn>colour&lt;/dfn> &lt;/dt&gt;</span>
 &lt;dd&gt; A sensation which (in humans) derives from the ability of
 the fine structure of the eye to distinguish three differently
 filtered analyses of a view. &lt;/dd&gt;
&lt;/dl&gt;</code></pre>
  </div>

  <div class="example">

   <p>The following example illustrates the use of the <code>dl</code> element to mark up metadata
   of sorts. At the end of the example, one group has two metadata labels ("Authors" and "Editors")
   and two values ("Robert Rothman" and "Daniel Jackson"). This example also uses the
   <code>div</code> element around the groups of <code>dt</code> and <code>dd</code> element, to aid
   with styling.</p>

   <pre><code class="html">&lt;dl>
 &lt;div>
  &lt;dt> Last modified time &lt;/dt>
  &lt;dd> 2004-12-23T23:33Z &lt;/dd>
 &lt;/div>
 &lt;div>
  &lt;dt> Recommended update interval &lt;/dt>
  &lt;dd> 60s &lt;/dd>
 &lt;/div>
 &lt;div>
  &lt;dt> Authors &lt;/dt>
  &lt;dt> Editors &lt;/dt>
  &lt;dd> Robert Rothman &lt;/dd>
  &lt;dd> Daniel Jackson &lt;/dd>
 &lt;/div>
&lt;/dl></code></pre>
  </div>

  <div class="example">

   <p>The following example shows the <code>dl</code> element used to give a set of instructions.
   The order of the instructions here is important (in the other examples, the order of the blocks
   was not important).</p>

   <pre><code class="html">&lt;p&gt;Determine the victory points as follows (use the
first matching case):&lt;/p&gt;
&lt;dl&gt;
 &lt;dt&gt; If you have exactly five gold coins &lt;/dt&gt;
 &lt;dd&gt; You get five victory points &lt;/dd&gt;
 &lt;dt&gt; If you have one or more gold coins, and you have one or more silver coins &lt;/dt&gt;
 &lt;dd&gt; You get two victory points &lt;/dd&gt;
 &lt;dt&gt; If you have one or more silver coins &lt;/dt&gt;
 &lt;dd&gt; You get one victory point &lt;/dd&gt;
 &lt;dt&gt; Otherwise &lt;/dt&gt;
 &lt;dd&gt; You get no victory points &lt;/dd&gt;
&lt;/dl&gt;</code></pre>
  </div>

  <div class="example">

   <p>The following snippet shows a <code>dl</code> element being used as a glossary. Note the use
   of <code>dfn</code> to indicate the word being defined.</p>

   <pre><code class="html">&lt;dl>
 &lt;dt>&lt;dfn>Apartment&lt;/dfn>, n.&lt;/dt>
 &lt;dd>An execution context grouping one or more threads with one or
 more COM objects.&lt;/dd>
 &lt;dt>&lt;dfn>Flat&lt;/dfn>, n.&lt;/dt>
 &lt;dd>A deflated tire.&lt;/dd>
 &lt;dt>&lt;dfn>Home&lt;/dfn>, n.&lt;/dt>
 &lt;dd>The user's login directory.&lt;/dd>
&lt;/dl></code></pre>
  </div>

  <div class="example">

   <p>This example uses <span>microdata</span> attributes in a <code>dl</code> element, together
   with the <code>div</code> element, to annotate the ice cream desserts at a French restaurant.</p>
   <!-- https://commons.wikimedia.org/wiki/File:Carte_des_desserts_2012_des_cafétérias_Toquenelle_(1).jpg -->

   <pre lang="fr"><code class="html">&lt;dl>
 &lt;div itemscope itemtype="http://schema.org/Product">
  &lt;dt itemprop="name">Café ou Chocolat Liégeois
  &lt;dd itemprop="offers" itemscope itemtype="http://schema.org/Offer">
   &lt;span itemprop="price">3.50&lt;/span>
   &lt;data itemprop="priceCurrency" value="EUR">€&lt;/data>
  &lt;dd itemprop="description">
   2 boules Café ou Chocolat, 1 boule Vanille, sauce café ou chocolat, chantilly
 &lt;/div>

 &lt;div itemscope itemtype="http://schema.org/Product">
  &lt;dt itemprop="name">Américaine
  &lt;dd itemprop="offers" itemscope itemtype="http://schema.org/Offer">
   &lt;span itemprop="price">3.50&lt;/span>
   &lt;data itemprop="priceCurrency" value="EUR">€&lt;/data>
  &lt;dd itemprop="description">
   1 boule Crème brûlée, 1 boule Vanille, 1 boule Caramel, chantilly
 &lt;/div>
&lt;/dl></code></pre>

   <p>Without the <code>div</code> element the markup would need to use the <code
   data-x="attr-itemref">itemref</code> attribute to link the data in the <code>dd</code> elements
   with the item, as follows.</p>

   <pre lang="fr"><code class="html">&lt;dl>
 &lt;dt itemscope itemtype="http://schema.org/Product" itemref="1-offer 1-description">
  &lt;span itemprop="name">Café ou Chocolat Liégeois&lt;/span>
 &lt;dd id="1-offer" itemprop="offers" itemscope itemtype="http://schema.org/Offer">
  &lt;span itemprop="price">3.50&lt;/span>
  &lt;data itemprop="priceCurrency" value="EUR">€&lt;/data>
 &lt;dd id="1-description" itemprop="description">
  2 boules Café ou Chocolat, 1 boule Vanille, sauce café ou chocolat, chantilly

 &lt;dt itemscope itemtype="http://schema.org/Product" itemref="2-offer 2-description">
  &lt;span itemprop="name">Américaine&lt;/span>
 &lt;dd id="2-offer" itemprop="offers" itemscope itemtype="http://schema.org/Offer">
  &lt;span itemprop="price">3.50&lt;/span>
  &lt;data itemprop="priceCurrency" value="EUR">€&lt;/data>
 &lt;dd id="2-description" itemprop="description">
  1 boule Crème brûlée, 1 boule Vanille, 1 boule Caramel, chantilly
&lt;/dl></code></pre>

  </div>

  <p class="note">The <code>dl</code> element is inappropriate for marking up dialogue. See some <a
  href="#conversations">examples of how to mark up dialogue</a>.</p>



  <h4>The <dfn element><code>dt</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Before <code>dd</code> or <code>dt</code> elements inside <code>dl</code> elements.</dd>
   <dd>Before <code>dd</code> or <code>dt</code> elements inside <code>div</code> elements that are children of a <code>dl</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>, but with no <code>header</code>, <code>footer</code>, <span>sectioning content</span>, or <span>heading content</span> descendants.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-dt">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-dt">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>dt</code> element <span>represents</span> the term, or name, part of a
  term-description group in a description list (<code>dl</code> element).</p>

  <p class="note">The <code>dt</code> element itself, when used in a <code>dl</code> element, does
  not indicate that its contents are a term being defined, but this can be indicated using the
  <code>dfn</code> element.</p>

  <div class="example">

   <p>This example shows a list of frequently asked questions (a FAQ) marked up using the
   <code>dt</code> element for questions and the <code>dd</code> element for answers.</p>

   <pre><code class="html">&lt;article>
 &lt;h1>FAQ&lt;/h1>
 &lt;dl>
  &lt;dt>What do we want?&lt;/dt>
  &lt;dd>Our data.&lt;/dd>
  &lt;dt>When do we want it?&lt;/dt>
  &lt;dd>Now.&lt;/dd>
  &lt;dt>Where is it?&lt;/dt>
  &lt;dd>We are not sure.&lt;/dd>
 &lt;/dl>
&lt;/article></code></pre>

  </div>



  <h4>The <dfn element><code>dd</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>After <code>dt</code> or <code>dd</code> elements inside <code>dl</code> elements.</dd>
   <dd>After <code>dt</code> or <code>dd</code> elements inside <code>div</code> elements that are children of a <code>dl</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-dd">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-dd">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>dd</code> element <span>represents</span> the description, definition, or value, part
  of a term-description group in a description list (<code>dl</code> element).</p>

  <div class="example">

   <p>A <code>dl</code> can be used to define a vocabulary list, like in a dictionary. In the
   following example, each entry, given by a <code>dt</code> with a <code>dfn</code>, has several
   <code>dd</code>s, showing the various parts of the definition.</p>

   <pre><code class="html">&lt;dl>
 &lt;dt>&lt;dfn>happiness&lt;/dfn>&lt;/dt>
 &lt;dd class="pronunciation">/&#x02C8;h&aelig;pin&#x0259;s/&lt;/dd>
 &lt;dd class="part-of-speech">&lt;i>&lt;abbr>n.&lt;/abbr>&lt;/i>&lt;/dd>
 &lt;dd>The state of being happy.&lt;/dd>
 &lt;dd>Good fortune; success. &lt;q>Oh &lt;b>happiness&lt;/b>! It worked!&lt;/q>&lt;/dd>
 &lt;dt>&lt;dfn>rejoice&lt;/dfn>&lt;/dt>
 &lt;dd class="pronunciation">/r&#x026A;&#x02C8;d&#x0292;&#x0254;&#x026A;s/&lt;/dd>
 &lt;dd>&lt;i class="part-of-speech">&lt;abbr>v.intr.&lt;/abbr>&lt;/i> To be delighted oneself.&lt;/dd>
 &lt;dd>&lt;i class="part-of-speech">&lt;abbr>v.tr.&lt;/abbr>&lt;/i> To cause one to be delighted.&lt;/dd>
&lt;/dl></code></pre>
  </div>


  <h4>The <dfn element><code>figure</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>Either: one <code>figcaption</code> element followed by <span>flow content</span>.</dd>
   <dd>Or: <span>flow content</span> followed by one <code>figcaption</code> element.</dd>
   <dd>Or: <span>flow content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-figure">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-figure">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>figure</code> element <span>represents</span> some <span>flow content</span>,
  optionally with a caption, that is self-contained (like a complete sentence) and is typically
  <span>referenced</span> as a single unit from the main flow of the document.</p>

  <p class="note">"Self-contained" in this context does not necessarily mean independent. For
  example, each sentence in a paragraph is self-contained; an image that is part of a sentence would
  be inappropriate for <code>figure</code>, but an entire sentence made of images would be
  fitting.</p>

  <p>The element can thus be used to annotate illustrations, diagrams, photos, code listings,
  etc.</p>

  <div class="note" id="figure-note-about-references">
   <p>When a <code>figure</code> is referred to from the main content of the document by identifying
   it by its caption (e.g., by figure number), it enables such content to be easily moved away from
   that primary content, e.g., to the side of the page, to dedicated pages, or to an appendix,
   without affecting the flow of the document.</p>

   <p>If a <code>figure</code> element is <span>referenced</span> by its relative position, e.g.,
   "in the photograph above" or "as the next figure shows", then moving the figure would disrupt the
   page's meaning. Authors are encouraged to consider using labels to refer to figures, rather than
   using such relative references, so that the page can easily be restyled without affecting the
   page's meaning.</p>
  </div>

  <p>The <span w-nodev>first</span> <code>figcaption</code> element child of the element, if any,
  represents the caption of the <code>figure</code> element's contents. If there is no child
  <code>figcaption</code> element, then there is no caption.</p>

  <p>A <code>figure</code> element's contents are part of the surrounding flow. If the purpose of
  the page is to display the figure, for example a photograph on an image sharing site, the
  <code>figure</code> and <code>figcaption</code> elements can be used to explicitly provide a
  caption for that figure. For content that is only tangentially related, or that serves a separate
  purpose than the surrounding flow, the <code>aside</code> element should be used (and can itself
  wrap a <code>figure</code>). For example, a pull quote that repeats content from an
  <code>article</code> would be more appropriate in an <code>aside</code> than in a
  <code>figure</code>, because it isn't part of the content, it's a repetition of the content for
  the purposes of enticing readers or highlighting key topics.</p>

  <div class="example">
   <p>This example shows the <code>figure</code> element to mark up a code listing.</p>

   <pre><code class="html">&lt;p>In &lt;a href="#l4">listing 4&lt;/a> we see the primary core interface
API declaration.&lt;/p>
&lt;figure id="l4">
 &lt;figcaption>Listing 4. The primary core interface API declaration.&lt;/figcaption>
 &lt;pre>&lt;code>interface PrimaryCore {
 boolean verifyDataLine();
 undefined sendData(sequence&amp;lt;byte> data);
 undefined initSelfDestruct();
}&lt;/code>&lt;/pre>
&lt;/figure>
&lt;p>The API is designed to use UTF-8.&lt;/p></code></pre>
  </div>

  <div class="example">
   <p>Here we see a <code>figure</code> element to mark up a photo that is the main content of the
   page (as in a gallery).</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang="en">
&lt;title>Bubbles at work &mdash; My Gallery&trade;&lt;/title>
&lt;figure>
 &lt;img src="bubbles-work.jpeg"
      alt="Bubbles, sitting in his office chair, works on his
           latest project intently.">
 &lt;figcaption>Bubbles at work&lt;/figcaption>
&lt;/figure>
&lt;nav>&lt;a href="19414.html">Prev&lt;/a> &mdash; &lt;a href="19416.html">Next&lt;/a>&lt;/nav></code></pre>
  </div>

  <div class="example">
   <p>In this example, we see an image that is <em>not</em> a figure, as well as an image and a
   video that are. The first image is literally part of the example's second sentence, so it's not a
   self-contained unit, and thus <code>figure</code> would be inappropriate.</p>

   <pre><code class="html">&lt;h2>Malinko's comics&lt;/h2>

&lt;p>This case centered on some sort of "intellectual property"
infringement related to a comic (see Exhibit A). The suit started
after a trailer ending with these words:

&lt;blockquote>
 &lt;img src="promblem-packed-action.png" alt="ROUGH COPY! Promblem-Packed Action!">
&lt;/blockquote>

&lt;p>...was aired. A lawyer, armed with a Bigger Notebook, launched a
preemptive strike using snowballs. A complete copy of the trailer is
included with Exhibit B.

&lt;figure>
 &lt;img src="ex-a.png" alt="Two squiggles on a dirty piece of paper.">
 &lt;figcaption>Exhibit A. The alleged &lt;cite>rough copy&lt;/cite> comic.&lt;/figcaption>
&lt;/figure>

&lt;figure>
 &lt;video src="ex-b.mov">&lt;/video>
 &lt;figcaption>Exhibit B. The &lt;cite>Rough Copy&lt;/cite> trailer.&lt;/figcaption>
&lt;/figure>

&lt;p>The case was resolved out of court.</code></pre>
  </div>

  <div class="example">
   <p>Here, a part of a poem is marked up using <code>figure</code>.</p>

   <pre><code class="html">&lt;figure>
 &lt;p>'Twas brillig, and the slithy toves&lt;br>
 Did gyre and gimble in the wabe;&lt;br>
 All mimsy were the borogoves,&lt;br>
 And the mome raths outgrabe.&lt;/p>
 &lt;figcaption>&lt;cite>Jabberwocky&lt;/cite> (first verse). Lewis Carroll, 1832-98&lt;/figcaption>
&lt;/figure></code></pre>
  </div>

  <div class="example">
   <p>In this example, which could be part of a much larger work discussing a castle, nested
   <code>figure</code> elements are used to provide both a group caption and individual captions for
   each figure in the group:</p>

   <pre><code class="html">&lt;figure>
 &lt;figcaption>The castle through the ages: 1423, 1858, and 1999 respectively.&lt;/figcaption>
 &lt;figure>
  &lt;figcaption>Etching. Anonymous, ca. 1423.&lt;/figcaption>
  &lt;img src="castle1423.jpeg" alt="The castle has one tower, and a tall wall around it.">
 &lt;/figure>
 &lt;figure>
  &lt;figcaption>Oil-based paint on canvas. Maria Towle, 1858.&lt;/figcaption>
  &lt;img src="castle1858.jpeg" alt="The castle now has two towers and two walls.">
 &lt;/figure>
 &lt;figure>
  &lt;figcaption>Film photograph. Peter Jankle, 1999.&lt;/figcaption>
  &lt;img src="castle1999.jpeg" alt="The castle lies in ruins, the original tower all that remains in one piece.">
 &lt;/figure>
&lt;/figure></code></pre>
  </div>

  <div class="example">
   <p>The previous example could also be more succinctly written as follows (using <code
   data-x="attr-title">title</code> attributes in place of the nested
   <code>figure</code>/<code>figcaption</code> pairs):</p>

   <pre><code class="html">&lt;figure>
 &lt;img src="castle1423.jpeg" title="Etching. Anonymous, ca. 1423."
      alt="The castle has one tower, and a tall wall around it.">
 &lt;img src="castle1858.jpeg" title="Oil-based paint on canvas. Maria Towle, 1858."
      alt="The castle now has two towers and two walls.">
 &lt;img src="castle1999.jpeg" title="Film photograph. Peter Jankle, 1999."
      alt="The castle lies in ruins, the original tower all that remains in one piece.">
 &lt;figcaption>The castle through the ages: 1423, 1858, and 1999 respectively.&lt;/figcaption>
&lt;/figure></code></pre>
  </div>

  <div class="example">
   <p>The figure is sometimes <span>referenced</span> only implicitly from the content:</p>

   <!-- from http://www.reuters.com/article/us-usa-fiscal-idUSBRE98N11220131017 -->

   <pre><code class="html">&lt;article>
 &lt;h1>Fiscal negotiations stumble in Congress as deadline nears&lt;/h1>
 &lt;figure>
  &lt;img src="obama-reid.jpeg" alt="Obama and Reid sit together smiling in the Oval Office.">
  &lt;figcaption>Barack Obama and Harry Reid. White House press photograph.&lt;/figcaption>
 &lt;/figure>
 &lt;p>Negotiations in Congress to end the fiscal impasse sputtered on Tuesday, leaving both chambers
 grasping for a way to reopen the government and raise the country's borrowing authority with a
 Thursday deadline drawing near.&lt;/p>
 ...
&lt;/article></code></pre>
  </div>


  <h4>The <dfn element><code>figcaption</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As the first or last child of a <code>figure</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-figcaption">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-figcaption">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>figcaption</code> element <span>represents</span> a caption or legend for the rest of
  the contents of the <code>figcaption</code> element's parent <code>figure</code> element<span
  w-nodev>, if any</span>.</p>

  <div class="example">
   <p>The element can contain additional information about the source:</p>

   <pre><code class="html">&lt;figcaption>
 &lt;p>A duck.&lt;/p>
 &lt;p>&lt;small>Photograph courtesy of 🌟 News.&lt;/small>&lt;/p>
&lt;/figcaption></code></pre>

   <pre><code class="html">&lt;figcaption>
 &lt;p>Average rent for 3-room apartments, excluding non-profit apartments&lt;/p>
 &lt;p>Zürich’s Statistics Office — &lt;time datetime=2017-11-14>14 November 2017&lt;/time>&lt;/p>
&lt;/figcaption></code></pre>
  </div>


  <h4>The <dfn element><code>main</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected, but only if it is a <span>hierarchically correct
   <code>main</code> element</span>.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-main">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-main">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>main</code> element <span>represents</span> the dominant contents of the
  document.</p>

  <p>A document must not have more than one <code>main</code> element that does not have the <code
  data-x="attr-hidden">hidden</code> attribute specified.</p>

  <p>A <dfn>hierarchically correct <code>main</code> element</dfn> is one whose ancestor elements
  are limited to <code>html</code>, <code>body</code>, <code>div</code>, <code>form</code> without
  an <span data-x="concept-accessible-name">accessible name</span>, and <span data-x="autonomous
  custom element">autonomous custom elements</span>. Each <code>main</code> element must be a
  <span>hierarchically correct <code>main</code> element</span>.</p>

  <div class="example">

   <p>In this example, the author has used a presentation where each component of the page is
   rendered in a box. To wrap the main content of the page (as opposed to the header, the footer,
   the navigation bar, and a sidebar), the <code>main</code> element is used.</p>

   <pre><code class="html">&lt;!DOCTYPE html>
&lt;html lang="en">
&lt;title>RPG System 17&lt;/title>
&lt;style>
 header, nav, aside, main, footer {
   margin: 0.5em; border: thin solid; padding: 0.5em;
   background: #EFF; color: black; box-shadow: 0 0 0.25em #033;
 }
 h1, h2, p { margin: 0; }
 nav, main { float: left; }
 aside { float: right; }
 footer { clear: both; }
&lt;/style>
&lt;header>
 &lt;h1>System Eighteen&lt;/h1>
&lt;/header>
&lt;nav>
 &lt;a href="../16/">&larr; System 17&lt;/a>
 &lt;a href="../18/">RPXIX &rarr;&lt;/a>
&lt;/nav>
&lt;aside>
 &lt;p>This system has no HP mechanic, so there's no healing.
&lt;/aside>
&lt;main>
 &lt;h2>Character creation&lt;/h2>
 &lt;p>Attributes (magic, strength, agility) are purchased at the cost of one point per level.&lt;/p>
 &lt;h2>Rolls&lt;/h2>
 &lt;p>Each encounter, roll the dice for all your skills. If you roll more than the opponent, you win.&lt;/p>
&lt;/main>
&lt;footer>
 &lt;p>Copyright &copy; 2013
&lt;/footer>
&lt;/html></code></pre>

   <p>In the following example, multiple <code>main</code> elements are used and script is used to
   make navigation work without a server roundtrip and to set the <code
   data-x="attr-hidden">hidden</code> attribute on those that are not current:

   <pre><code class="html">&lt;!doctype html>
&lt;html lang=en-CA>
&lt;meta charset=utf-8>
&lt;title> &hellip; &lt;/title>
&lt;link rel=stylesheet href=spa.css>
&lt;script src=spa.js async>&lt;/script>
&lt;nav>
 &lt;a href=/>Home&lt;/a>
 &lt;a href=/about>About&lt;/a>
 &lt;a href=/contact>Contact&lt;/a>
&lt;/nav>
&lt;main>
 &lt;h1>Home&lt;/h1>
 &hellip;
&lt;/main>
&lt;main hidden>
 &lt;h1>About&lt;/h1>
 &hellip;
&lt;/main>
&lt;main hidden>
 &lt;h1>Contact&lt;/h1>
 &hellip;
&lt;/main>
&lt;footer>Made with ❤️ by &lt;a href=https://example.com/>Example 👻&lt;/a>.&lt;/footer></code></pre>

  </div>


  <h4>The <dfn element><code>search</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-search">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-search">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>search</code> element <span>represents</span> a part of a document or application
  that contains a set of form controls or other content related to performing a search or filtering
  operation. This could be a search of the web site or application; a way of searching or filtering
  search results on the current web page; or a global or Internet-wide search function.</p>

  <p class="note">It's not appropriate to use the <code>search</code> element just for presenting
  search results, though suggestions and links as part of "quick search" results can be
  included as part of a search feature. Rather, a returned web page of search results would instead
  be expected to be presented as part of the main content of that web page.</p>

  <div class="example">
   <p>In the following example, the author is including a search form within the
   <code>header</code> of the web page:</p>

   <pre><code class="html">&lt;header>
  &lt;h1>&lt;a href="/">My fancy blog&lt;/a>&lt;/h1>
  ...
  &lt;search>
    &lt;form action="search.php">
      &lt;label for="query">Find an article&lt;/label>
      &lt;input id="query" name="q" type="search">
      &lt;button type="submit">Go!&lt;/button>
    &lt;/form>
  &lt;/search>
&lt;/header></code></pre>
  </div>

  <div class="example">
   <p>In this example, the author has implemented their web application's search functionality
   entirely with JavaScript. There is no use of the <code>form</code> element to perform
   server-side submission, but the containing <code>search</code> element semantically identifies
   the purpose of the descendant content as representing search capabilities.</p>

   <pre><code class="html">&lt;search>
  &lt;label>
    Find and filter your query
    &lt;input type="search" id="query">
  &lt;/label>
  &lt;label>
    &lt;input type="checkbox" id="exact-only">
    Exact matches only
  &lt;/label>

  &lt;section>
    &lt;h3>Results found:&lt;/h3>
    &lt;ul id="results">
      &lt;li>
        &lt;p>&lt;a href="services/consulting">Consulting services&lt;/a>&lt;/p>
        &lt;p>
          Find out how can we help you improve your business with our integrated consultants, Bob and Bob.
        &lt;/p>
      &lt;/li>
      ...
    &lt;/ul>
    &lt;!--
      when a query returns or filters out all results
      render the no results message here
    -->
    &lt;output id="no-results">&lt;/output>
  &lt;/section>
&lt;/search></code></pre>
  </div>

  <div class="example">
    <p>In the following example, the page has two search features. The first is located in the web page's
    <code>header</code> and serves as a global mechanism to search the web site's content. Its purpose is
    indicated by its specified <code>title</code> attribute. The second is included as part of the main content
    of the page, as it represents a mechanism to search and filter the content of the current page. It contains
    a heading to indicate its purpose.</p>

    <pre><code class="html">&lt;body>
  &lt;header>
    ...
    &lt;search title="Website">
      ...
    &lt;/search>
  &lt;/header>
  &lt;main>
    &lt;h1>Hotels near your location&lt;/h1>
     &lt;search>
       &lt;h2>Filter results&lt;/h2>
       ...
     &lt;/search>
     &lt;article>
      &lt;!-- search result content -->
    &lt;/article>
  &lt;/main>
&lt;/body></code></pre>
  </div>


  <h4>The <dfn element><code>div</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dd>As a child of a <code>dl</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>If the element is a child of a <code>dl</code> element: one or more <code>dt</code> elements followed by one or more <code>dd</code> elements, optionally intermixed with <span>script-supporting elements</span>.</dd>
   <dd>If the element is not a child of a <code>dl</code> element: <span>flow content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-div">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-div">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLDivElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  // <a href="#HTMLDivElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLDivElement</code>.</dd>
  </dl>

  <p>The <code>div</code> element has no special meaning at all. It <span>represents</span> its
  children. It can be used with the <code data-x="attr-class">class</code>, <code
  data-x="attr-lang">lang</code>, and <code data-x="attr-title">title</code> attributes to mark up
  semantics common to a group of consecutive elements. It can also be used in a <code>dl</code>
  element, wrapping groups of <code>dt</code> and <code>dd</code> elements.</p>

  <p class="note">Authors are strongly encouraged to view the <code>div</code> element as an element
  of last resort, for when no other element is suitable. Use of more appropriate elements instead of
  the <code>div</code> element leads to better accessibility for readers and easier maintainability
  for authors.</p>

  <div class="example">

   <p>For example, a blog post would be marked up using <code>article</code>, a chapter using
   <code>section</code>, a page's navigation aids using <code>nav</code>, and a group of form
   controls using <code>fieldset</code>.</p>

   <p>On the other hand, <code>div</code> elements can be useful for stylistic purposes or to wrap
   multiple paragraphs within a section that are all to be annotated in a similar way. In the
   following example, we see <code>div</code> elements used as a way to set the language of two
   paragraphs at once, instead of setting the language on the two paragraph elements separately:</p>

   <pre><code class="html">&lt;article lang="en-US">
 &lt;h1>My use of language and my cats&lt;/h1>
 &lt;p>My cat's behavior hasn't changed much since her absence, except
 that she plays her new physique to the neighbors regularly, in an
 attempt to get pets.&lt;/p>
 &lt;div lang="en-GB">
  &lt;p>My other cat, <!--en-GB-->coloured black and white, is a sweetie. He followed
  us to the pool today, walking down the pavement with us. Yesterday
  he apparently visited our neighbours. I wonder if he <!--en-GB-->recognises that
  their flat is a mirror image of ours.&lt;/p>
  &lt;p>Hm, I just noticed that in the last paragraph I used British
  English. But I'm supposed to write in American English. So I
  shouldn't say "pavement" or "flat" or <!--en-GB-->"colour"...&lt;/p>
 &lt;/div>
 &lt;p>I should say "sidewalk" and "apartment" and "color"!&lt;/p>
&lt;/article></code></pre>

  </div>


  <h3 split-filename="text-level-semantics">Text-level semantics</h3>

  <h4>The <dfn element><code>a</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd>If the element has an <code data-x="attr-hyperlink-href">href</code> attribute: <span>Interactive content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Transparent</span>, but there must be no <span>interactive content</span> descendant,
   <code>a</code> element descendant, or descendant with the <code
   data-x="attr-tabindex">tabindex</code> attribute specified.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-hyperlink-href">href</code></dd>
   <dd><code data-x="attr-hyperlink-target">target</code></dd>
   <dd><code data-x="attr-hyperlink-download">download</code></dd>
   <dd><code data-x="attr-hyperlink-ping">ping</code></dd>
   <dd><code data-x="attr-hyperlink-rel">rel</code></dd>
   <dd><code data-x="attr-hyperlink-hreflang">hreflang</code></dd>
   <dd><code data-x="attr-hyperlink-type">type</code></dd>
   <dd><code data-x="attr-hyperlink-referrerpolicy">referrerpolicy</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd>If the element has an <code data-x="attr-hyperlink-href">href</code> attribute: <a
   href="https://w3c.github.io/html-aria/#el-a">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-a">for implementers</a>.</dd>
   <dd>Otherwise: <a
   href="https://w3c.github.io/html-aria/#el-a-no-href">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-a-no-href">for implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLAnchorElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-a-target">target</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-a-download">download</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-a-ping">ping</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-a-rel">rel</span>;
  [SameObject, PutForwards=<span data-x="dom-DOMTokenList-value">value</span>] readonly attribute <span>DOMTokenList</span> <span data-x="dom-a-relList">relList</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-a-hreflang">hreflang</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-a-type">type</span>;

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-a-text">text</span>;

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-a-referrerPolicy">referrerPolicy</span>;

  // <a href="#HTMLAnchorElement-partial">also has obsolete members</a>
};
<span>HTMLAnchorElement</span> includes <span>HTMLHyperlinkElementUtils</span>;</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLAnchorElement</code>.</dd>
  </dl>

  <p>If the <code>a</code> element has an <code data-x="attr-hyperlink-href">href</code> attribute,
  then it <span>represents</span> a <span>hyperlink</span> (a hypertext anchor) labeled by its
  contents.</p>

  <!-- v2: Eric Meyer requests the ability to nest links so that a big hyperlink, e.g. representing
  a calendar event, can be clickable, while within it there are subareas that represent links to
  distinct areas, e.g. a link to see photos of the event, or to edit the event, or some such. -->

  <p>If the <code>a</code> element has no <code data-x="attr-hyperlink-href">href</code> attribute,
  then the element <span>represents</span> a placeholder for where a link might otherwise have been
  placed, if it had been relevant, consisting of just the element's contents.</p>

  <p>The <code data-x="attr-hyperlink-target">target</code>, <code
  data-x="attr-hyperlink-download">download</code>, <code data-x="attr-hyperlink-ping">ping</code>,
  <code data-x="attr-hyperlink-rel">rel</code>, <code
  data-x="attr-hyperlink-hreflang">hreflang</code>, <code data-x="attr-hyperlink-type">type</code>,
  and <code data-x="attr-hyperlink-referrerpolicy">referrerpolicy</code> attributes must be omitted
  if the <code data-x="attr-hyperlink-href">href</code> attribute is not present.</p>

  <p>If the <code data-x="attr-itemprop">itemprop</code> attribute is specified on an <code>a</code> element,
  then the <code data-x="attr-hyperlink-href">href</code> attribute must also be specified.</p>

  <div class="example">

   <p>If a site uses a consistent navigation toolbar on every page, then the link that would
   normally link to the page itself could be marked up using an <code>a</code> element:</p>

   <pre><code class="html">&lt;nav>
 &lt;ul>
  &lt;li> &lt;a href="/">Home&lt;/a> &lt;/li>
  &lt;li> &lt;a href="/news">News&lt;/a> &lt;/li>
  &lt;li> &lt;a>Examples&lt;/a> &lt;/li>
  &lt;li> &lt;a href="/legal">Legal&lt;/a> &lt;/li>
 &lt;/ul>
&lt;/nav></code></pre>

  </div>

  <div w-nodev>

  <p>The <code data-x="attr-hyperlink-href">href</code>, <code
  data-x="attr-hyperlink-target">target</code>, <code
  data-x="attr-hyperlink-download">download</code>, <code data-x="attr-hyperlink-ping">ping</code>,
  and <code data-x="attr-hyperlink-referrerpolicy">referrerpolicy</code> attributes affect what
  happens when users <span data-x="following hyperlinks">follow hyperlinks</span> or <span
  data-x="downloading hyperlinks">download hyperlinks</span> created using the <code>a</code>
  element. The <code data-x="attr-hyperlink-rel">rel</code>, <code
  data-x="attr-hyperlink-hreflang">hreflang</code>, and <code data-x="attr-hyperlink-type">type</code>
  attributes may be used to indicate to the user the likely nature of the target resource before the
  user follows the link.</p>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>a</var>.<span subdfn data-x="dom-a-text">text</span></code></dt>

   <dd><p>Same as <code>textContent</code>.</p></dd>
  </dl>

  <div w-nodev>

  <p>The IDL attributes <dfn attribute for="HTMLAnchorElement"><code
  data-x="dom-a-download">download</code></dfn>, <dfn for="HTMLAnchorElement" attribute><code
  data-x="dom-a-ping">ping</code></dfn>, <dfn attribute for="HTMLAnchorElement"><code
  data-x="dom-a-target">target</code></dfn>, <dfn attribute for="HTMLAnchorElement"><code
  data-x="dom-a-rel">rel</code></dfn>, <dfn attribute for="HTMLAnchorElement"><code
  data-x="dom-a-hreflang">hreflang</code></dfn>, and <dfn attribute for="HTMLAnchorElement"><code
  data-x="dom-a-type">type</code></dfn>, must <span>reflect</span> the respective content attributes
  of the same name.</p>

  <p>The IDL attribute <dfn attribute for="HTMLAnchorElement"><code
  data-x="dom-a-rellist">relList</code></dfn> must <span>reflect</span> the <code
  data-x="attr-hyperlink-rel">rel</code> content attribute.</p>

  <p>The IDL attribute <dfn attribute for="HTMLAnchorElement"><code
  data-x="dom-a-referrerPolicy">referrerPolicy</code></dfn> must <span>reflect</span> the <code
  data-x="attr-hyperlink-referrerpolicy">referrerpolicy</code> content attribute, <span>limited to
  only known values</span>.</p>

  <p>The <dfn attribute for="HTMLAnchorElement"><code data-x="dom-a-text">text</code></dfn>
  attribute's getter must return this element's <span>descendant text content</span>.</p>

  <p>The <code data-x="dom-a-text">text</code> attribute's setter must <span>string replace
  all</span> with the given value within this element.</p>

  </div>

  <div class="example">
   <p>The <code>a</code> element can be wrapped around entire paragraphs, lists, tables, and so
   forth, even entire sections, so long as there is no interactive content within (e.g., buttons or
   other links). This example shows how this can be used to make an entire advertising block into a
   link:</p>

   <pre><code class="html">&lt;aside class="advertising">
 &lt;h1>Advertising&lt;/h1>
 &lt;a href="https://ad.example.com/?adid=1929&amp;amp;pubid=1422">
  &lt;section>
   &lt;h1>Mellblomatic 9000!&lt;/h1>
   &lt;p>Turn all your widgets into mellbloms!&lt;/p>
   &lt;p>Only $9.99 plus shipping and handling.&lt;/p>
  &lt;/section>
 &lt;/a>
 &lt;a href="https://ad.example.com/?adid=375&amp;amp;pubid=1422">
  &lt;section>
   &lt;h1>The Mellblom Browser&lt;/h1>
   &lt;p>Web browsing at the speed of light.&lt;/p>
   &lt;p>No other browser goes faster!&lt;/p>
  &lt;/section>
 &lt;/a>
&lt;/aside></code></pre>
  </div>

  <div class="example">
   <p>The following example shows how a bit of script can be used to effectively make an entire row
   in a job listing table a hyperlink:</p>

   <pre><code class="html">&lt;table>
 &lt;tr>
  &lt;th>Position
  &lt;th>Team
  &lt;th>Location
 &lt;tr>
  &lt;td>&lt;a href="/jobs/manager">Manager&lt;/a>
  &lt;td>Remotees
  &lt;td>Remote
 &lt;tr>
  &lt;td>&lt;a href="/jobs/director">Director&lt;/a>
  &lt;td>Remotees
  &lt;td>Remote
 &lt;tr>
  &lt;td>&lt;a href="/jobs/astronaut">Astronaut&lt;/a>
  &lt;td>Architecture
  &lt;td>Remote
&lt;/table>
&lt;script>
document.querySelector("table").onclick = ({ target }) => {
  if (target.parentElement.localName === "tr") {
    const link = target.parentElement.querySelector("a");
    if (link) {
      link.click();
    }
  }
}
&lt;/script></code></pre>
  </div>


  <h4>The <dfn element><code>em</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-em">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-em">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>em</code> element <span>represents</span> stress emphasis of its contents.</p>

  <p>The level of stress that a particular piece of content has is given by its number of ancestor
  <code>em</code> elements.</p>

  <p>The placement of stress emphasis changes the meaning of the sentence. The element thus forms an
  integral part of the content. The precise way in which stress is used in this way depends on the
  language.</p>

  <div class="example">

   <p>These examples show how changing the stress emphasis changes the meaning. First, a general
   statement of fact, with no stress:</p>

   <pre><code class="html">&lt;p>Cats are cute animals.&lt;/p></code></pre>

   <p>By emphasizing the first word, the statement implies that the kind of animal under discussion
   is in question (maybe someone is asserting that dogs are cute):</p>

   <pre><code class="html">&lt;p>&lt;em>Cats&lt;/em> are cute animals.&lt;/p></code></pre>

   <p>Moving the stress to the verb, one highlights that the truth of the entire sentence is in
   question (maybe someone is saying cats are not cute):</p>

   <pre><code class="html">&lt;p>Cats &lt;em>are&lt;/em> cute animals.&lt;/p></code></pre>

   <p>By moving it to the adjective, the exact nature of the cats is reasserted (maybe someone
   suggested cats were <em>mean</em> animals):</p>

   <pre><code class="html">&lt;p>Cats are &lt;em>cute&lt;/em> animals.&lt;/p></code></pre>

   <p>Similarly, if someone asserted that cats were vegetables, someone correcting this might
   emphasize the last word:</p>

   <pre><code class="html">&lt;p>Cats are cute &lt;em>animals&lt;/em>.&lt;/p></code></pre>

   <p>By emphasizing the entire sentence, it becomes clear that the speaker is fighting hard to get
   the point across. This kind of stress emphasis also typically affects the punctuation, hence the
   exclamation mark here.</p>

   <pre><code class="html">&lt;p>&lt;em>Cats are cute animals!&lt;/em>&lt;/p></code></pre>

   <p>Anger mixed with emphasizing the cuteness could lead to markup such as:</p>

   <pre><code class="html">&lt;p>&lt;em>Cats are &lt;em>cute&lt;/em> animals!&lt;/em>&lt;/p></code></pre>

  </div>

  <div class="note">

   <p>The <code>em</code> element isn't a generic "italics" element. Sometimes, text is intended to
   stand out from the rest of the paragraph, as if it was in a different mood or voice. For this,
   the <code>i</code> element is more appropriate.</p>

   <p>The <code>em</code> element also isn't intended to convey importance; for that purpose, the
   <code>strong</code> element is more appropriate.</p>

   <!--

    Thus the following is a bad use of <em>:

      <p><em>Note</em>: ...</p>

    You should use <strong> or <i> for this instead (depending on
    exactly what you're doing).

   -->

  </div>



  <h4>The <dfn element><code>strong</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-strong">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-strong">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>strong</code> element <span>represents</span> strong importance, seriousness, or
  urgency for its contents.</p>

  <p><strong>Importance</strong>: the <code>strong</code> element can be used in a heading, caption,
  or paragraph to distinguish the part that really matters from other parts that might be more
  detailed, more jovial, or merely boilerplate. (This is distinct from marking up subheadings, for
  which the <code>hgroup</code> element is appropriate.)</p>

  <p class="example">For example, the first word of the previous paragraph is marked up with
  <code>strong</code> to distinguish it from the more detailed text in the rest of the
  paragraph.</p>

  <p><strong>Seriousness</strong>: the <code>strong</code> element can be used to mark up a warning
  or caution notice.</p>

  <p><strong>Urgency</strong>: the <code>strong</code> element can be used to denote contents that
  the user needs to see sooner than other parts of the document.</p>

  <p>The relative level of importance of a piece of content is given by its number of ancestor
  <code>strong</code> elements; each <code>strong</code> element increases the importance of its
  contents.</p>

  <p>Changing the importance of a piece of text with the <code>strong</code> element does not change
  the meaning of the sentence.</p>

  <div class="example">

   <p>Here, the word "chapter" and the actual chapter number are mere boilerplate, and the actual
   name of the chapter is marked up with <code>strong</code>:</p>

   <pre><code class="html">&lt;h1>Chapter 1: &lt;strong>The Praxis&lt;/strong>&lt;/h1></code></pre>

   <p>In the following example, the name of the diagram in the caption is marked up with
   <code>strong</code>, to distinguish it from boilerplate text (before) and the description
   (after):</p>

   <pre><code class="html">&lt;figcaption>Figure 1. &lt;strong>Ant colony dynamics&lt;/strong>. The ants in this colony are
affected by the heat source (upper left) and the food source (lower right).&lt;/figcaption></code></pre>

   <p>In this example, the heading is really "Flowers, Bees, and Honey", but the author has added a
   light-hearted addition to the heading. The <code>strong</code> element is thus used to mark up
   the first part to distinguish it from the latter part.</p>

   <pre><code class="html">&lt;h1>&lt;strong>Flowers, Bees, and Honey&lt;/strong> and other things I don't understand&lt;/h1></code></pre>

  </div>

  <div class="example">
   <p>Here is an example of a warning notice in a game, with the
   various parts marked up according to how important they are:</p>
   <!-- DO NOT REFLOW THIS EXAMPLE it has been carefully balanced -->
   <pre><code class="html">&lt;p>&lt;strong>Warning.&lt;/strong> This dungeon is dangerous.
&lt;strong>Avoid the ducks.&lt;/strong> Take any gold you find.
&lt;strong>&lt;strong>Do not take any of the diamonds&lt;/strong>,
they are explosive and &lt;strong>will destroy anything within
ten meters.&lt;/strong>&lt;/strong> You have been warned.&lt;/p></code></pre>
  </div>

  <div class="example">

   <p>In this example, the <code>strong</code> element is used to denote the part of the text that
   the user is intended to read first.</p>

   <pre><code class="html">&lt;p>Welcome to Remy, the reminder system.&lt;/p>
&lt;p>Your tasks for today:&lt;/p>
&lt;ul>
 &lt;li>&lt;p>&lt;strong>Turn off the oven.&lt;/strong>&lt;/p>&lt;/li>
 &lt;li>&lt;p>Put out the trash.&lt;/p>&lt;/li>
 &lt;li>&lt;p>Do the laundry.&lt;/p>&lt;/li>
&lt;/ul></code></pre>

  </div>


  <h4>The <dfn element><code>small</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-small">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-small">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>small</code> element <span>represents</span> side comments such as small print.</p>

  <p class="note">Small print typically features disclaimers, caveats, legal restrictions, or
  copyrights. Small print is also sometimes used for attribution, or for satisfying licensing
  requirements.</p>

  <p class="note">The <code>small</code> element does not "de-emphasize" or lower the importance of
  text emphasized by the <code>em</code> element or marked as important with the <code>strong</code>
  element. To mark text as not emphasized or important, simply do not mark it up with the
  <code>em</code> or <code>strong</code> elements respectively.</p>

  <p>The <code>small</code> element should not be used for extended spans of text, such as multiple
  paragraphs, lists, or sections of text. It is only intended for short runs of text. The text of a
  page listing terms of use, for instance, would not be a suitable candidate for the
  <code>small</code> element: in such a case, the text is not a side comment, it is the main content
  of the page.</p>

  <p>The <code>small</code> element must not be used for subheadings; for that purpose, use the
  <code>hgroup</code> element.</p>

  <div class="example">

   <p>In this example, the <code>small</code> element is used to indicate that value-added tax is
   not included in a price of a hotel room:</p>

   <pre class="example"><code class="html">&lt;dl>
 &lt;dt>Single room
 &lt;dd>199 &euro; &lt;small>breakfast included, VAT not included&lt;/small>
 &lt;dt>Double room
 &lt;dd>239 &euro; &lt;small>breakfast included, VAT not included&lt;/small>
&lt;/dl></code></pre>

  </div>

  <div class="example">

   <p>In this second example, the <code>small</code> element is used for a side comment in an
   article.</p>

   <pre><code class="html">&lt;p>Example Corp today announced record profits for the
second quarter &lt;small>(Full Disclosure: Foo News is a subsidiary of
Example Corp)&lt;/small>, leading to speculation about a third quarter
merger with Demo Group.&lt;/p></code></pre>

   <p>This is distinct from a sidebar, which might be multiple paragraphs long and is removed from
   the main flow of text. In the following example, we see a sidebar from the same article. This
   sidebar also has small print, indicating the source of the information in the sidebar.</p>

   <pre><code class="html">&lt;aside>
 &lt;h1>Example Corp&lt;/h1>
 &lt;p>This company mostly creates small software and Web
 sites.&lt;/p>
 &lt;p>The Example Corp company mission is "To provide entertainment
 and news on a sample basis".&lt;/p>
 &lt;p>&lt;small>Information obtained from &lt;a
 href="https://example.com/about.html">example.com&lt;/a> home
 page.&lt;/small>&lt;/p>
&lt;/aside></code></pre>

  </div>

  <div class="example">

   <p>In this last example, the <code>small</code> element is marked as being <em>important</em>
   small print.</p>

   <pre><code class="html">&lt;p>&lt;strong>&lt;small>Continued use of this service will result in a kiss.&lt;/small>&lt;/strong>&lt;/p></code></pre>

  </div>


  <h4>The <dfn element><code>s</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-s">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-s">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>s</code> element <span>represents</span> contents that are no longer accurate or no
  longer relevant.</p>

  <p class="note">The <code>s</code> element is not appropriate when indicating document edits; to
  mark a span of text as having been removed from a document, use the <code>del</code> element.</p>

  <div class="example">

   <p>In this example a recommended retail price has been marked as no longer relevant as the
   product in question has a new sale price.</p>

   <pre><code class="html">&lt;p>Buy our Iced Tea and Lemonade!&lt;/p>
&lt;p>&lt;s>Recommended retail price: $3.99 per bottle&lt;/s>&lt;/p>
&lt;p>&lt;strong>Now selling for just $2.99 a bottle!&lt;/strong>&lt;/p></code></pre>

  </div>


  <h4>The <dfn element><code>cite</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-cite">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-cite">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>cite</code> element <span>represents</span> the title of a work (e.g.
    a book,
    a paper,
    an essay,
    a poem,
    a score,
    a song,
    a script,
    a film,
    a TV show,
    a game,
    a sculpture,
    a painting,
    a theatre production,
    a play,
    an opera,
    a musical,
    an exhibition,
    a legal case report,
    a computer program, <!-- "a software package" ? -->
  etc.). This can be a work that is being quoted or <span>referenced</span> in detail (i.e., a
  citation), or it can just be a work that is mentioned in passing.</p>

  <p>A person's name is not the title of a work &mdash; even if people call that person a piece of
  work &mdash; and the element must therefore not be used to mark up people's names. (In some cases,
  the <code>b</code> element might be appropriate for names; e.g. in a gossip article where the
  names of famous people are keywords rendered with a different style to draw attention to them. In
  other cases, if an element is <em>really</em> needed, the <code>span</code> element can be
  used.)</p>

<!--(some people argue a ship is a work)
  <p>A ship is similarly not a work, and the element must not be used to mark up ship names (the
  <code>i</code> element can be used for that purpose).</p>
-->

  <div class="example">

   <p>This next example shows a typical use of the <code>cite</code> element:</p>

   <pre><code class="html">&lt;p>My favorite book is &lt;cite>The Reality Dysfunction&lt;/cite> by
Peter F. Hamilton. My favorite comic is &lt;cite>Pearls Before
Swine&lt;/cite> by Stephan Pastis. My favorite track is &lt;cite>Jive
Samba&lt;/cite> by the Cannonball Adderley Sextet.&lt;/p></code></pre>

  </div>

  <div class="example">

   <p>This is correct usage:</p>

   <pre><code class="html">&lt;p>According to the Wikipedia article &lt;cite>HTML&lt;/cite>, as it
stood in mid-February 2008, leaving attribute values unquoted is
unsafe. This is obviously an over-simplification.&lt;/p></code></pre>

   <p>The following, however, is incorrect usage, as the <code>cite</code> element here is
   containing far more than the title of the work:</p>

   <pre class="bad"><code class="html">&lt;!-- do not copy this example, it is an example of bad usage! -->
&lt;p>According to &lt;cite>the Wikipedia article on HTML&lt;/cite>, as it
stood in mid-February 2008, leaving attribute values unquoted is
unsafe. This is obviously an over-simplification.&lt;/p></code></pre>

  </div>

  <div class="example">

   <p>The <code>cite</code> element is a key part of any citation in a bibliography, but it is only
   used to mark the title:</p>

   <pre><code class="html">&lt;p>&lt;cite>Universal Declaration of Human Rights&lt;/cite>, United Nations,
December 1948. Adopted by General Assembly resolution 217 A (III).&lt;/p></code></pre>

  </div>

  <p class="note">A <em>citation</em> is not a <em>quote</em> (for which the <code>q</code> element
  is appropriate).</p>

  <div class="example">

   <p>This is incorrect usage, because <code>cite</code> is not for quotes:</p>

   <pre class="bad"><code class="html">&lt;p>&lt;cite>This is wrong!&lt;/cite>, said Ian.&lt;/p></code></pre>

   <p>This is also incorrect usage, because a person is not a work:</p>

   <pre class="bad"><code class="html">&lt;p>&lt;q>This is still wrong!&lt;/q>, said &lt;cite>Ian&lt;/cite>.&lt;/p></code></pre>

   <p>The correct usage does not use a <code>cite</code> element:</p>

   <pre><code class="html">&lt;p>&lt;q>This is correct&lt;/q>, said Ian.&lt;/p></code></pre>

   <p>As mentioned above, the <code>b</code> element might be relevant for marking names as being
   keywords in certain kinds of documents:</p>

   <pre><code class="html">&lt;p>And then &lt;b>Ian&lt;/b> said &lt;q>this might be right, in a
gossip column, maybe!&lt;/q>.&lt;/p></code></pre>

  </div>



  <h4>The <dfn element><code>q</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-q-cite">cite</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-q">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-q">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLQuoteElement</code>.</dd>
  </dl>

  <p>The <code>q</code> element <span>represents</span> some <span data-x="phrasing content">phrasing
  content</span> quoted from another source.</p>

  <p>Quotation punctuation (such as quotation marks) that is quoting the contents of the element
  must not appear immediately before, after, or inside <code>q</code> elements; they will be
  inserted into the rendering by the user agent.</p>

  <p>Content inside a <code>q</code> element must be quoted from another source, whose address, if
  it has one, may be cited in the <dfn element-attr for="q"><code
  data-x="attr-q-cite">cite</code></dfn> attribute. The source may be fictional, as when quoting
  characters in a novel or screenplay.</p>

  <p>If the <code data-x="attr-q-cite">cite</code> attribute is present, it must be a <span>valid
  URL potentially surrounded by spaces</span>. <span w-nodev>To obtain the corresponding citation
  link, the value of the attribute must be <span data-x="encoding-parsing a URL">parsed</span>
  relative to the element's <span>node document</span>.</span> User agents may allow users to follow
  such citation links, but they are primarily intended for private use (e.g., by server-side scripts
  collecting statistics about a site's use of quotations), not for readers.</p>

  <p>The <code>q</code> element must not be used in place of quotation marks that do not represent
  quotes; for example, it is inappropriate to use the <code>q</code> element for marking up
  sarcastic statements.</p>

  <p>The use of <code>q</code> elements to mark up quotations is entirely optional; using explicit
  quotation punctuation without <code>q</code> elements is just as correct.</p>

  <div class="example">

   <p>Here is a simple example of the use of the <code>q</code> element:</p>

   <pre><code class="html">&lt;p>The man said &lt;q>Things that are impossible just take
longer&lt;/q>. I disagreed with him.&lt;/p></code></pre>

  </div>

  <div class="example">

   <p>Here is an example with both an explicit citation link in the <code>q</code> element, and an
   explicit citation outside:</p>

   <pre><code class="html">&lt;p>The W3C page &lt;cite>About W3C&lt;/cite> says the W3C's
mission is &lt;q cite="https://www.w3.org/Consortium/">To lead the
World Wide Web to its full potential by developing protocols and
guidelines that ensure long-term growth for the Web&lt;/q>. I
disagree with this mission.&lt;/p></code></pre>

  </div>

  <div class="example">

   <p>In the following example, the quotation itself contains a quotation:</p>

   <pre><code class="html">&lt;p>In &lt;cite>Example One&lt;/cite>, he writes &lt;q>The man
said &lt;q>Things that are impossible just take longer&lt;/q>. I
disagreed with him&lt;/q>. Well, I disagree even more!&lt;/p></code></pre>

  </div>

  <div class="example">

   <p>In the following example, quotation marks are used instead of the <code>q</code> element:</p>

   <pre><code class="html">&lt;p>His best argument was &#x275D;I disagree&#x275E;, which
I thought was laughable.&lt;/p></code></pre>

  </div>

  <div class="example">

   <p>In the following example, there is no quote &mdash; the quotation marks are used to name a
   word. Use of the <code>q</code> element in this case would be inappropriate.</p>

   <pre><code class="html">&lt;p>The word "ineffable" could have been used to describe the disaster
resulting from the campaign's mismanagement.&lt;/p></code></pre>

  </div>


  <h4>The <dfn element><code>dfn</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>, but there must be no <code>dfn</code> element descendants.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd>Also, the <code data-x="attr-dfn-title">title</code> attribute <span data-x="attr-dfn-title">has special semantics</span> on this element.</dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-dfn">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-dfn">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>dfn</code> element <span>represents</span> the defining instance of a term. The <span
  data-x="paragraph">paragraph</span>, <span data-x="dl">description list group</span>, or <span
  data-x="sectioning content">section</span> that is the nearest ancestor of the <code>dfn</code>
  element must also contain the definition(s) for the <span data-x="defining term">term</span> given
  by the <code>dfn</code> element.</p>

  <p><dfn>Defining term</dfn>: if the <code>dfn</code> element has a <dfn element-attr
  for="dfn"><code data-x="attr-dfn-title">title</code></dfn> attribute, then the exact value of that
  attribute is the term being defined. Otherwise, if it contains exactly one element child node and
  no child <code>Text</code> nodes, and that child element is an <code>abbr</code> element with a
  <code data-x="attr-abbr-title">title</code> attribute, then the exact value of <em>that</em>
  attribute is the term being defined. Otherwise, it is the <span>descendant text content</span> of
  the <code>dfn</code> element that gives the term being defined.</p>

  <!-- note that this means <dfn>x \n x</dfn> won't match <span>x x</span> -->

  <p>If the <code data-x="attr-dfn-title">title</code> attribute of the <code>dfn</code> element is
  present, then it must contain only the term being defined.</p>

  <p class="note">The <code data-x="attr-title">title</code> attribute of ancestor elements does not
  affect <code>dfn</code> elements.</p>

  <p>An <code>a</code> element that links to a <code>dfn</code> element represents an instance of
  the term defined by the <code>dfn</code> element.</p>

  <div class="example">

   <p>In the following fragment, the term "Garage Door Opener" is first defined in the first
   paragraph, then used in the second. In both cases, its abbreviation is what is actually
   displayed.</p>

   <pre><code class="html">&lt;p>The <strong>&lt;dfn>&lt;abbr title="Garage Door Opener">GDO&lt;/abbr>&lt;/dfn></strong>
is a device that allows off-world teams to open the iris.&lt;/p>
&lt;!-- ... later in the document: -->
&lt;p>Teal'c activated his <strong>&lt;abbr title="Garage Door Opener">GDO&lt;/abbr></strong>
and so Hammond ordered the iris to be opened.&lt;/p></code></pre>

   <p>With the addition of an <code>a</code> element, the <span data-x="referenced">reference</span>
   can be made explicit:</p>

   <pre><code class="html">&lt;p>The &lt;dfn <strong>id=gdo</strong>>&lt;abbr title="Garage Door Opener">GDO&lt;/abbr>&lt;/dfn>
is a device that allows off-world teams to open the iris.&lt;/p>
&lt;!-- ... later in the document: -->
&lt;p>Teal'c activated his <strong>&lt;a href=#gdo></strong>&lt;abbr title="Garage Door Opener">GDO&lt;/abbr><strong>&lt;/a></strong>
and so Hammond ordered the iris to be opened.&lt;/p></code></pre>

  </div>



  <h4>The <dfn element><code>abbr</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd>Also, the <code data-x="attr-abbr-title">title</code> attribute <span data-x="attr-abbr-title">has special semantics</span> on this element.</dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-abbr">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-abbr">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>abbr</code> element <span>represents</span> an abbreviation or acronym, optionally
  with its expansion. The <dfn element-attr for="abbr"><code
  data-x="attr-abbr-title">title</code></dfn> attribute may be used to provide an expansion of the
  abbreviation. The attribute, if specified, must contain an expansion of the abbreviation, and
  nothing else.</p>

  <div class="example">

   <p>The paragraph below contains an abbreviation marked up with the <code>abbr</code> element.
   This paragraph <span data-x="defining term">defines the term</span> "Web Hypertext Application
   Technology Working Group".</p>

   <pre><code class="html">&lt;p>The &lt;dfn id=whatwg>&lt;abbr
title="Web Hypertext Application Technology Working Group">WHATWG&lt;/abbr>&lt;/dfn>
is a loose unofficial collaboration of web browser manufacturers and
interested parties who wish to develop new technologies designed to
allow authors to write and deploy Applications over the World Wide
Web.&lt;/p></code></pre>

   <p>An alternative way to write this would be:</p>

   <pre><code class="html">&lt;p>The &lt;dfn id=whatwg>Web Hypertext Application Technology
Working Group&lt;/dfn> (&lt;abbr
title="Web Hypertext Application Technology Working Group">WHATWG&lt;/abbr>)
is a loose unofficial collaboration of web browser manufacturers and
interested parties who wish to develop new technologies designed to
allow authors to write and deploy Applications over the World Wide
Web.&lt;/p></code></pre>

  </div>

  <div class="example">

   <p>This paragraph has two abbreviations. Notice how only one is defined; the other, with no
   expansion associated with it, does not use the <code>abbr</code> element.</p>

   <pre><code class="html">&lt;p>The
&lt;abbr title="Web Hypertext Application Technology Working Group">WHATWG&lt;/abbr>
started working on HTML5 in 2004.&lt;/p></code></pre>

  </div>

  <div class="example">

   <p>This paragraph links an abbreviation to its definition.</p>

   <pre><code class="html">&lt;p>The &lt;a href="#whatwg">&lt;abbr
title="Web Hypertext Application Technology Working Group">WHATWG&lt;/abbr>&lt;/a>
community does not have much representation from Asia.&lt;/p></code></pre>

  </div>

  <div class="example">

   <p>This paragraph marks up an abbreviation without giving an expansion, possibly as a hook to
   apply styles for abbreviations (e.g. smallcaps).</p>

   <pre><code class="html">&lt;p>Philip` and Dashiva both denied that they were going to
get the issue counts from past revisions of the specification to
backfill the &lt;abbr>WHATWG&lt;/abbr> issue graph.&lt;/p></code></pre>

  </div>

  <p>If an abbreviation is pluralized, the expansion's grammatical number (plural vs singular) must
  match the grammatical number of the contents of the element.</p>

  <div class="example">

   <p>Here the plural is outside the element, so the expansion is in the singular:</p>

   <pre><code class="html">&lt;p>Two &lt;abbr title="Working Group">WG&lt;/abbr>s worked on
this specification: the &lt;abbr>WHATWG&lt;/abbr> and the
&lt;abbr>HTMLWG&lt;/abbr>.&lt;/p></code></pre>

   <p>Here the plural is inside the element, so the expansion is in the plural:</p>

   <pre><code class="html">&lt;p>Two &lt;abbr title="Working Groups">WGs&lt;/abbr> worked on
this specification: the &lt;abbr>WHATWG&lt;/abbr> and the
&lt;abbr>HTMLWG&lt;/abbr>.&lt;/p></code></pre>

  </div>

  <p>Abbreviations do not have to be marked up using this element. It is expected to be useful in
  the following cases:</p>

  <ul>
   <li>Abbreviations for which the author wants to give expansions, where using the
   <code>abbr</code> element with a <code data-x="attr-title">title</code> attribute is an
   alternative to including the expansion inline (e.g. in parentheses).</li>

   <li>Abbreviations that are likely to be unfamiliar to the document's readers, for which authors
   are encouraged to either mark up the abbreviation using an <code>abbr</code> element with a <code
   data-x="attr-title">title</code> attribute or include the expansion inline in the text the first
   time the abbreviation is used.</li>

   <li>Abbreviations whose presence needs to be semantically annotated, e.g. so that they can be
   identified from a style sheet and given specific styles, for which the <code>abbr</code> element
   can be used without a <code data-x="attr-title">title</code> attribute.</li>
  </ul>

  <p class="note">Providing an expansion in a <code data-x="attr-title">title</code> attribute once
  will not necessarily cause other <code>abbr</code> elements in the same document with the same
  contents but without a <code data-x="attr-title">title</code> attribute to behave as if they had
  the same expansion. Every <code>abbr</code> element is independent.</p>


  <h4>The <dfn element><code>ruby</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>See prose.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-ruby">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-ruby">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>ruby</code> element allows one or more spans of phrasing content to be marked with
  ruby annotations. Ruby annotations are short runs of text presented alongside base text, primarily
  used in East Asian typography as a guide for pronunciation or to include other annotations. In
  Japanese, this form of typography is also known as <i>furigana</i>.</p>

  <p>The content model of <code>ruby</code> elements consists of one or more of the following
  sequences:</p>

  <ol>
   <li>
    <p>One or the other of the following:</p>

    <ul>
     <li><p><span>Phrasing content</span>, but with no <code>ruby</code> elements and with no
     <code>ruby</code> element descendants</p></li>

     <li><p>A single <code>ruby</code> element that itself has no <code>ruby</code> element
     descendants</p></li>
    </ul>
   <li>
    <p>One or the other of the following:</p>

    <ul>
     <li><p>One or more <code>rt</code> elements</p>

     <li><p>An <code>rp</code> element followed by one or more <code>rt</code> elements, each of
     which is itself followed by an <code>rp</code> element</p></li>
    </ul>
  </ol>

  <!-- ( ( phrasing-content | <ruby> ) ( <rt>+ | <rp> ( <rt> <rp> )+ ) )+ // with further ruby descendant restrictions -->

  <p>The <code>ruby</code> and <code>rt</code> elements can be used for a variety of kinds of
  annotations, including in particular (though by no means limited to) those described below. For
  more details on Japanese Ruby in particular, and how to render Ruby for Japanese, see
  <cite>Requirements for Japanese Text Layout</cite>. <ref>JLREQ</ref></p>

  <p class="note">At the time of writing, CSS does not yet provide a way to fully control the
  rendering of the HTML <code>ruby</code> element. It is hoped that CSS will be extended to support
  the styles described below in due course.</p>

  <!-- examples in the following list are mostly based on those in https://www.w3.org/International/datespace/2010/02/jlreq-examples/ -->

  <dl>
   <dt>Mono-ruby for individual base characters in Japanese

   <dd>
    <p>One or more hiragana or katakana characters (the ruby annotation) are placed with each
    ideographic character (the base text). This is used to provide readings of kanji characters.

    <div class="example"> <!-- B as in BASE -->
     <pre><code class="html">&lt;ruby>B&lt;rt>annotation&lt;/ruby></code></pre>
    </div>

    <div class="example">
     <p>In this example, notice how each annotation corresponds to a single base character.
     <pre><code class="html">&lt;ruby>&#21531;&lt;rt>&#12367;&#12435;&lt;/ruby>&lt;ruby>&#23376;&lt;rt>&#12375;&lt;/ruby>&#12399;&lt;ruby>&#21644;&lt;rt>&#12431;&lt;/ruby>&#12375;&#12390;&lt;ruby>&#21516;&lt;rt>&#12393;&#12358;&lt;/ruby>&#12380;&#12378;&#12290;</code></pre>
     <p lang=ja><ruby>&#21531;<rt>&#12367;&#12435;</ruby><ruby>&#23376;<rt>&#12375;</ruby>&#12399;<ruby>&#21644;<rt>&#12431;</ruby>&#12375;&#12390;<ruby>&#21516;<rt>&#12393;&#12358;</ruby>&#12380;&#12378;&#12290;
     <p>This example can also be written as follows, using one <code>ruby</code> element with two
     segments of base text and two annotations (one for each) rather than two back-to-back
     <code>ruby</code> elements each with one base text segment and annotation (as in the markup
     above):
     <pre><code class="html">&lt;ruby>&#21531;&lt;rt>&#12367;&#12435;&lt;/rt>&#23376;&lt;rt>&#12375;&lt;/ruby>&#12399;&lt;ruby>&#21644;&lt;rt>&#12431;&lt;/ruby>&#12375;&#12390;&lt;ruby>&#21516;&lt;rt>&#12393;&#12358;&lt;/ruby>&#12380;&#12378;&#12290;</code></pre>
    </div>
   </dd>


   <dt>Mono-ruby for compound words (jukugo)

   <dd>
    <p>This is similar to the previous case: each ideographic character in the compound word (the
    base text) has its reading given in hiragana or katakana characters (the ruby annotation). The
    difference is that the base text segments form a compound word rather than being separate from
    each other.

    <div class="example"> <!-- B as in BASE -->
     <pre><code class="html">&lt;ruby>B&lt;rt>annotation&lt;/rt>B&lt;rt>annotation&lt;/ruby></code></pre>
    </div>

    <div class="example">
     <p>In this example, notice again how each annotation corresponds to a single base character. In this example, each compound word (jukugo) corresponds to a single <code>ruby</code> element.</p>
     <p>The rendering here is expected to be that each annotation be placed over (or next to, in vertical text) the corresponding base character, with the annotations not overhanging any of the adjacent characters.</p>
     <pre><code class="html">&lt;ruby>&#39740;&lt;rt>&#12365;&lt;/rt>&#38272;&lt;rt>&#12418;&#12435;&lt;/rt>&lt;/ruby>&#12398;&lt;ruby>&#26041;&lt;rt>&#12411;&#12358;&lt;/rt>&#35282;&lt;rt>&#12364;&#12367;&lt;/rt>&lt;/ruby>&#12434;&lt;ruby>&#20957;&lt;rt>&#12366;&#12423;&#12358;&lt;/rt>&#35222;&lt;rt>&#12375;&lt;/rt>&lt;/ruby>&#12377;&#12427;</code></pre>
     <p lang=ja><ruby>&#39740;<rt>&#12365;</rt>&#38272;<rt>&#12418;&#12435;</rt></ruby>&#12398;<ruby>&#26041;<rt>&#12411;&#12358;</rt>&#35282;<rt>&#12364;&#12367;</rt></ruby>&#12434;<ruby>&#20957;<rt>&#12366;&#12423;&#12358;</rt>&#35222;<rt>&#12375;</rt></ruby>&#12377;&#12427;

    </div>
   </dd>


   <dt>Jukugo-ruby

   <dd>
    <p>This is semantically identical to the previous case (each individual ideographic character in
    the base compound word has its reading given in an annotation in hiragana or katakana
    characters), but the rendering is the more complicated Jukugo Ruby rendering.

    <div class="example">
     <p>This is the same example as above for mono-ruby for compound words. The different rendering is expected to be achieved using different styling (e.g. in CSS), and is not shown here.</p>
     <pre><code class="html">&lt;ruby>&#39740;&lt;rt>&#12365;&lt;/rt>&#38272;&lt;rt>&#12418;&#12435;&lt;/rt>&lt;/ruby>&#12398;&lt;ruby>&#26041;&lt;rt>&#12411;&#12358;&lt;/rt>&#35282;&lt;rt>&#12364;&#12367;&lt;/rt>&lt;/ruby>&#12434;&lt;ruby>&#20957;&lt;rt>&#12366;&#12423;&#12358;&lt;/rt>&#35222;&lt;rt>&#12375;&lt;/rt>&lt;/ruby>&#12377;&#12427;</code></pre>
     <!-- Once CSS is updated to describe this, invoke the CSS and unhide this -->
     <!--
     <p lang=ja><ruby>&#39740;<rt>&#12365;</rt>&#38272;<rt>&#12418;&#12435;</rt></ruby>&#12398;<ruby>&#26041;<rt>&#12411;&#12358;</rt>&#35282;<rt>&#12364;&#12367;</rt></ruby>&#12434;<ruby>&#20957;<rt>&#12366;&#12423;&#12358;</rt>&#35222;<rt>&#12375;</rt></ruby>&#12377;&#12427;
     -->
    </div>

    <p class="note">For more details on <a
    href="https://www.w3.org/TR/jlreq/#positioning_of_jukugoruby">Jukugo Ruby rendering</a>, see
    Appendix F in the <cite>Requirements for Japanese Text Layout</cite>. <ref>JLREQ</ref></p>
   </dd>


   <dt>Group ruby for describing meanings

   <dd>
    <p>The annotation describes the meaning of the base text, rather than (or in addition to) the
    pronunciation. As such, both the base text and the annotation can be multiple characters long.

    <div class="example"> <pre><code class="html">&lt;ruby>BASE&lt;rt>annotation&lt;/ruby></code></pre> </div>

    <div class="example">
     <p>Here a compound ideographic word has its corresponding katakana given as an annotation.
     <pre><code class="html">&lt;ruby>&#22659;&#30028;&#38754;&lt;rt>&#12452;&#12531;&#12479;&#12540;&#12501;&#12455;&#12540;&#12473;&lt;/ruby></code></pre>
     <p lang=ja><ruby>&#22659;&#30028;&#38754;<rt>&#12452;&#12531;&#12479;&#12540;&#12501;&#12455;&#12540;&#12473;</ruby>
    </div>

    <div class="example">
     <p>Here a compound ideographic word has its translation in English provided as an annotation.
     <pre><code class="html">&lt;ruby lang="ja">&#32232;&#38598;&#32773;&lt;rt lang="en">editor&lt;/ruby></code></pre>
     <p><ruby lang="ja">&#32232;&#38598;&#32773;<rt lang="en">editor</ruby>
    </div>
   </dd>


   <dt>Group ruby for Jukuji readings

   <dd>
    <p>A phonetic reading that corresponds to multiple base characters, because a one-to-one mapping
    would be difficult. (In English, the words "Colonel" and "Lieutenant" are examples of words
    where a direct mapping of pronunciation to individual letters is, in some dialects, rather
    unclear.)

    <div class="example">
     <p>In this example, the name of a species of flowers has a phonetic reading provided using group ruby:
     <pre><code class="html">&lt;ruby>&#32043;&#38525;&#33457;&lt;rt>&#12354;&#12376;&#12373;&#12356;&lt;/ruby></code></pre>
     <p lang=ja><ruby>&#32043;&#38525;&#33457;<rt>&#12354;&#12376;&#12373;&#12356;</ruby>
    </div>
   </dd>


   <dt>Text with both phonetic and semantic annotations (double-sided ruby)

   <dd>
    <p>Sometimes, ruby styles described above are combined.

    <p>If this results in two annotations covering the same single base segment, then the
    annotations can just be placed back to back.

    <div class="example">
     <pre><code class="html">&lt;ruby>BASE&lt;rt>annotation 1&lt;rt>annotation 2&lt;/ruby></code></pre>
    </div>

    <div class="example">
     <pre><code class="html">&lt;ruby>B&lt;rt>a&lt;rt>a&lt;/ruby>&lt;ruby>A&lt;rt>a&lt;rt>a&lt;/ruby>&lt;ruby>S&lt;rt>a&lt;rt>a&lt;/ruby>&lt;ruby>E&lt;rt>a&lt;rt>a&lt;/ruby></code></pre>
    </div>

    <div class="example">

     <p>In this contrived example, some symbols are given names in English and French.

     <pre><code class="html">&lt;ruby>
 &#x2665; &lt;rt> Heart &lt;rt lang=fr> C&oelig;ur &lt;/rt>
 &#x2618; &lt;rt> Shamrock &lt;rt lang=fr> Tr&egrave;fle &lt;/rt>
 &#x2736; &lt;rt> Star &lt;rt lang=fr> &Eacute;toile &lt;/rt>
&lt;/ruby></code></pre>

    </div>

    <p>In more complicated situations such as the following examples, a nested <code>ruby</code>
    element is used to give the inner annotations, and then that whole <code>ruby</code> is then
    given an annotation at the "outer" level.

    <div class="example">
     <pre><code class="html">&lt;ruby>&lt;ruby>B&lt;rt>a&lt;/rt>A&lt;rt>n&lt;/rt>S&lt;rt>t&lt;/rt>E&lt;rt>n&lt;/rt>&lt;/ruby>&lt;rt>annotation&lt;/ruby></code></pre>
    </div>

    <div class="example">
     <p>Here both a phonetic reading and the meaning are given in ruby annotations. The annotation on the nested <code>ruby</code> element gives a mono-ruby phonetic annotation for each base character, while the annotation in the <code>rt</code> element that is a child of the outer <code>ruby</code> element gives the meaning using hiragana.
     <pre><code class="html">&lt;ruby>&lt;ruby>&#26481;&lt;rt>&#12392;&#12358;&lt;/rt>&#21335;&lt;rt>&#12394;&#12435;&lt;/rt>&lt;/ruby>&lt;rt>&#12383;&#12388;&#12415;&lt;/rt>&lt;/ruby>&#12398;&#26041;&#35282;</code></pre>
     <p lang=ja><ruby><ruby>&#26481;<rt>&#12392;&#12358;</rt>&#21335;<rt>&#12394;&#12435;</rt></ruby><rt>&#12383;&#12388;&#12415;</rt></ruby>&#12398;&#26041;&#35282;
    </div>

    <div class="example">
     <p>This is the same example, but the meaning is given in English instead of Japanese:
     <pre><code class="html">&lt;ruby>&lt;ruby>&#26481;&lt;rt>&#12392;&#12358;&lt;/rt>&#21335;&lt;rt>&#12394;&#12435;&lt;/rt>&lt;/ruby>&lt;rt lang=en>Southeast&lt;/rt>&lt;/ruby>&#12398;&#26041;&#35282;</code></pre>
     <p lang=ja><ruby><ruby>&#26481;<rt>&#12392;&#12358;</rt>&#21335;<rt>&#12394;&#12435;</rt></ruby><rt lang=en>Southeast</rt></ruby>&#12398;&#26041;&#35282;
    </div>
   </dd>
  </dl>

  <hr>

  <p>Within a <code>ruby</code> element that does not have a <code>ruby</code> element ancestor,
  content is segmented and segments are placed into three categories: base text segments, annotation
  segments, and ignored segments. Ignored segments do not form part of the document's semantics
  (they consist of some <span>inter-element whitespace</span> and <code>rp</code> elements, the
  latter of which are used for legacy user agents that do not support ruby at all). Base text
  segments can overlap (with a limit of two segments overlapping any one position in the DOM, and
  with any segment having an earlier start point than an overlapping segment also having an equal or
  later end point, and any segment have a later end point than an overlapping segment also having an
  equal or earlier start point<!-- if anyone can find a better way of phrasing this parenthetical,
  do let me know! -->). Annotation segments correspond to <code>rt</code> elements. Each annotation
  segment can be associated with a base text segment, and each base text segment can have annotation
  segments associated with it. (In a conforming document, each base text segment is associated with
  at least one annotation segment, and each annotation segment is associated with one base text
  segment.) A <code>ruby</code> element <span>represents</span> the union of the segments of base
  text it contains, along with the mapping from those base text segments to annotation segments.
  <span w-nodev>Segments are described in terms of DOM ranges; annotation segment ranges always
  consist of exactly one element. <ref>DOM</ref></span></p>

  <div w-nodev>

  <p>At any particular time, the segmentation and categorization of content of a <code>ruby</code>
  element is the result that would be obtained from running the following algorithm:</p>

  <ol>
   <li><p>Let <var>base text segments</var> be an empty list of base text segments, each
   potentially with a list of base text subsegments.</p></li>

   <li><p>Let <var>annotation segments</var> be an empty list of annotation segments, each
   potentially being associated with a base text segment or subsegment.</p></li>

   <li><p>Let <var>root</var> be the <code>ruby</code> element for which the algorithm is
   being run.</p></li>

   <li><p>If <var>root</var> has a <code>ruby</code> element ancestor, then jump to the
   step labeled <i>end</i>.</p></li>

   <li><p>Let <var>current parent</var> be <var>root</var>.</p></li>

   <li><p>Let <var>index</var> be 0.</p></li>

   <li><p>Let <var>start index</var> be null.</p></li>

   <li><p>Let <var>saved start index</var> be null.</p></li>

   <li><p>Let <var>current base text</var> be null.</p></li>


   <!-- This is spaghetti code. If someone can work out a way to phrase this that is as unambiguous
   but maps more cleanly to structured code, please let me know. -->


   <li><p><i>Start mode</i>: If <var>index</var> is greater than or equal to the number of child
   nodes in <var>current parent</var>, then jump to the step labeled <i>end mode</i>.</p>

   <li><p>If the <var>index</var>th node in <var>current parent</var> is an
   <code>rt</code> or <code>rp</code> element, jump to the step labeled <i>annotation
   mode</i>.</p></li>

   <!-- if we get here then the first node in a ruby element or after an <rt> or <rp> is something
   we want in a base text -->

   <li><p>Set <var>start index</var> to the value of <var>index</var>.</p></li>

   <li><p><i>Base mode</i>: If the <var>index</var>th node in <var>current
   parent</var> is a <code>ruby</code> element, and if <var>current parent</var> is the
   same element as <var>root</var>, then <span>push a ruby level</span> and then jump to
   the step labeled <i>start mode</i>.</p></li>

   <li><p>If the <var>index</var>th node in <var>current parent</var> is an
   <code>rt</code> or <code>rp</code> element, then <span>set the current base text</span> and then
   jump to the step labeled <i>annotation mode</i>.</p></li>

   <li><p>Increment <var>index</var> by one.</p></li>

   <li><p><i>Base mode post-increment</i>: If <var>index</var> is greater than or equal to the
   number of child nodes in <var>current parent</var>, then jump to the step labeled <i>end
   mode</i>.</p></li>

   <li><p>Jump back to the step labeled <i>base mode</i>.</p></li>


   <li><p><i>Annotation mode</i>: If the <var>index</var>th node in <var>current
   parent</var> is an <code>rt</code> element, then <span>push a ruby annotation</span> and jump to
   the step labeled <i>annotation mode increment</i>.</p></li>

   <li><p>If the <var>index</var>th node in <var>current parent</var> is an
   <code>rp</code> element, jump to the step labeled <i>annotation mode increment</i>.</p></li>

   <li><p>If the <var>index</var>th node in <var>current parent</var> is not a
   <code>Text</code> node, or is a <code>Text</code> node that is not <span>inter-element
   whitespace</span>, then jump to the step labeled <i>base mode</i>.</p></li>

   <!-- index points at whitespace -->

   <li><p><i>Annotation mode increment</i>: Let <var>lookahead index</var> be <var>index</var> plus one.</p></li>

   <li><p><i>Annotation mode white-space skipper</i>: If <var>lookahead index</var> is
   equal to the number of child nodes in <var>current parent</var> then jump to the step
   labeled <i>end mode</i>.</p></li>

   <li><p>If the <var>lookahead index</var>th node in <var>current parent</var> is
   an <code>rt</code> element or an <code>rp</code> element, then set <var>index</var> to
   <var>lookahead index</var> and jump to the step labeled <i>annotation mode</i>.</p></li>

   <li><p>If the <var>lookahead index</var>th node in <var>current parent</var> is
   not a <code>Text</code> node, or is a <code>Text</code> node that is not <span>inter-element
   whitespace</span>, then jump to the step labeled <i>base mode</i> (without further incrementing
   <var>index</var>, so the <span>inter-element whitespace</span> seen so far becomes part
   of the next base text segment).</p></li>

   <li><p>Increment <var>lookahead index</var> by one.</p></li>

   <li><p>Jump to the step labeled <i>annotation mode white-space skipper</i>.</p></li>


   <li><p><i>End mode</i>: If <var>current parent</var> is not the same element as <var>root</var>, then <span>pop a ruby level</span> and jump to the step labeled <i>base mode
   post-increment</i>.</p></li>

   <li><p><i>End</i>: Return <var>base text segments</var> and <var>annotation
   segments</var>. Any content of the <code>ruby</code> element not described by segments in either
   of those lists is implicitly in an <i>ignored segment</i>.</p></li>
  </ol>


  <p>When the steps above say to <dfn>set the current base text</dfn>, it means to run the following
  steps at that point in the algorithm:</p>

  <ol>
   <li><p>Let <var>text range</var> be a DOM range whose <span
   data-x="concept-range-start">start</span> is the <span data-x="concept-range-bp">boundary
   point</span> (<var>current parent</var>, <var>start index</var>) and whose
   <span data-x="concept-range-end">end</span> is the <span data-x="concept-range-bp">boundary
   point</span> (<var>current parent</var>, <var>index</var>).</p></li>

   <li><p>Let <var>new text segment</var> be a base text segment described by the range
   <var>text range</var>.</p>

   <li><p>Add <var>new text segment</var> to <var>base text
   segments</var>.</p></li>

   <li><p>Let <var>current base text</var> be <var>new text
   segment</var>.</p></li>

   <li><p>Let <var>start index</var> be null.</p></li>
  </ol>


  <p>When the steps above say to <dfn>push a ruby level</dfn>, it means to run the following steps
  at that point in the algorithm:</p>

  <ol>
   <li><p>Let <var>current parent</var> be the <var>index</var>th node in <var>current parent</var>.</p></li>

   <li><p>Let <var>index</var> be 0.</p></li>

   <li><p>Set <var>saved start index</var> to the value of <var>start
   index</var>.</p></li>

   <li><p>Let <var>start index</var> be null.</p></li>
  </ol>


  <p>When the steps above say to <dfn>pop a ruby level</dfn>, it means to run the following steps at
  that point in the algorithm:</p>

  <ol>
   <li><p>Let <var>index</var> be the position of <var>current parent</var> in
   <var>root</var>.</p></li>

   <li><p>Let <var>current parent</var> be <var>root</var>.</p></li>

   <li><p>Increment <var>index</var> by one.</p></li>

   <li><p>Set <var>start index</var> to the value of <var>saved start
   index</var>.</p></li>

   <li><p>Let <var>saved start index</var> be null.</p></li>
  </ol>


  <p>When the steps above say to <dfn>push a ruby annotation</dfn>, it means to run the following
  steps at that point in the algorithm:</p>

  <ol>
   <li><p>Let <var>rt</var> be the <code>rt</code> element that is the <var>index</var>th node of <var>current parent</var>.</p></li>

   <li><p>Let <var>annotation range</var> be a DOM range whose <span
   data-x="concept-range-start">start</span> is the <span data-x="concept-range-bp">boundary
   point</span> (<var>current parent</var>, <var>index</var>) and whose <span
   data-x="concept-range-end">end</span> is the <span data-x="concept-range-bp">boundary point</span>
   (<var>current parent</var>, <var>index</var> plus one) (i.e. that contains only
   <var>rt</var>).</p></li>

   <li><p>Let <var>new annotation segment</var> be an annotation segment described by the
   range <var>annotation range</var>.</p></li>

   <li><p>If <var>current base text</var> is not null, associate <var>new
   annotation segment</var> with <var>current base text</var>.</p></li>

   <li><p>Add <var>new annotation segment</var> to <var>annotation
   segments</var>.</p></li>
  </ol>

  </div>

  <div class="example">

   <!-- this is the hiragana for the word "kanji" ("Chinese character") in Japanese -->
   <!-- in Japanese, ruby-like typography is called "furigana" -->

   <p>In this example, each ideograph in the Japanese text <span data-x=""
   lang="ja">&#28450;&#23383;</span> is annotated with its reading in hiragana.</p>

   <pre lang="ja"><code class="html">...
&lt;ruby>&#28450;&lt;rt>&#12363;&#12435;&lt;/rt>&#23383;&lt;rt>&#12376;&lt;/rt>&lt;/ruby>
...</code></pre>

   <p>This might be rendered as:</p>

   <p><img src="/images/sample-ruby-ja.png" width="171" height="78"
           alt="The two main ideographs, each with its annotation in hiragana rendered in a smaller font above it."></p>

  </div>

  <div class="example">

   <!-- this is the zhuyin fuhao (aka bopomofo) for the word "hanzi" ("Chinese character") in traditional Chinese, as used in Taiwan -->

   <p>In this example, each ideograph in the traditional Chinese text <span data-x=""
   lang="zh-TW">&#28450;&#23383;</span> is annotated with its bopomofo reading.</p>

   <pre lang="zh-TW"><code class="html">&lt;ruby>&#28450;&lt;rt>&#12559;&#12578;&#715;&lt;/rt>&#23383;&lt;rt>&#12567;&#715;&lt;/rt>&lt;/ruby></code></pre>

   <p>This might be rendered as:</p>

   <p><img src="/images/sample-ruby-bopomofo.png" width="78" height="100"
           alt="The two main ideographs, each with its bopomofo annotation rendered in a smaller font next to it."></p>

  </div>

  <div class="example">

   <!-- this is the pinyin for the word "hanzi" ("Chinese character") in simplified Chinese, as used in mainland China -->

   <p>In this example, each ideograph in the simplified Chinese text <span data-x=""
   lang="zh-CN">&#27721;&#23383;</span> is annotated with its pinyin reading.</p>

   <pre lang="zh-CN"><code class="html">...&lt;ruby>&#27721;&lt;rt>h&#224;n&lt;/rt>&#23383;&lt;rt>z&#236;&lt;/rt>&lt;/ruby>...</code></pre>

   <p>This might be rendered as:</p>

   <p><img src="/images/sample-ruby-pinyin.png" width="173" height="79"
           alt="The two main ideographs, each with its pinyin annotation rendered in a smaller font above it."></p>

  </div>

  <!-- Note: Examples are 32px/16px Stone Sans Sem ITC TT -->

  <div class="example">

   <p>In this more contrived example, the acronym "HTML" has four annotations: one for the whole
   acronym, briefly describing what it is, one for the letters "HT" expanding them to "Hypertext",
   one for the letter "M" expanding it to "Markup", and one for the letter "L" expanding it to
   "Language".</p>

   <pre><code class="html">&lt;ruby>
 &lt;ruby>HT&lt;rt>Hypertext&lt;/rt>M&lt;rt>Markup&lt;/rt>L&lt;rt>Language&lt;/rt>&lt;/ruby>
 &lt;rt>An abstract language for describing documents and applications
&lt;/ruby></code></pre>

  </div>


  <h4>The <dfn element><code>rt</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As a child of a <code>ruby</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-rt">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-rt">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>rt</code> element marks the ruby text component of a ruby annotation. When it is the
  child of a <code>ruby</code> element, it doesn't <span data-x="represents">represent</span>
  anything itself, but the <code>ruby</code> element uses it as part of determining what <em>it</em>
  <span>represents</span>.</p>

  <div w-nodev>

  <p>An <code>rt</code> element that is not a child of a <code>ruby</code> element
  <span>represents</span> the same thing as its children.</p>

  </div>


  <h4>The <dfn element><code>rp</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As a child of a <code>ruby</code> element, either immediately before or immediately after an <code>rt</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="text content">Text</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-rp">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-rp">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>rp</code> element can be used to provide parentheses or other content around a ruby
  text component of a ruby annotation, to be shown by user agents that don't support ruby
  annotations.</p>

  <p>An <code>rp</code> element <span w-nodev>that is a child of a <code>ruby</code>
  element</span> <span>represents</span> nothing. <span w-nodev>An <code>rp</code> element
  whose parent element is not a <code>ruby</code> element <span>represents</span> its
  children.</span></p>

  <div class="example">

   <p>The example above, in which each ideograph in the text <span data-x=""
   lang="ja">&#28450;&#23383;</span> is annotated with its phonetic reading, could be expanded to
   use <code>rp</code> so that in legacy user agents the readings are in parentheses:</p>

   <pre lang="ja"><code class="html">...
&lt;ruby>漢&lt;rp>(&lt;/rp>&lt;rt>かん&lt;/rt>&lt;rp>)&lt;/rp>字&lt;rp>(&lt;/rp>&lt;rt>じ&lt;/rt>&lt;rp>)&lt;/rp>&lt;/ruby>
...</code></pre>

   <p>In conforming user agents the rendering would be as above, but in user agents that do not
   support ruby, the rendering would be:</p>

   <pre lang="ja">... 漢(かん)字(じ)...</pre>

  </div>

  <div class="example">

   <p>When there are multiple annotations for a segment, <code>rp</code> elements can also be placed
   between the annotations. Here is another copy of an earlier contrived example showing some
   symbols with names given in English and French, but this time with <code>rp</code> elements as
   well:

   <pre><code class="html">&lt;ruby>
&#x2665;&lt;rp>: &lt;/rp>&lt;rt>Heart&lt;/rt>&lt;rp>, &lt;/rp>&lt;rt lang=fr>C&oelig;ur&lt;/rt>&lt;rp>.&lt;/rp>
&#x2618;&lt;rp>: &lt;/rp>&lt;rt>Shamrock&lt;/rt>&lt;rp>, &lt;/rp>&lt;rt lang=fr>Tr&egrave;fle&lt;/rt>&lt;rp>.&lt;/rp>
&#x2736;&lt;rp>: &lt;/rp>&lt;rt>Star&lt;/rt>&lt;rp>, &lt;/rp>&lt;rt lang=fr>&Eacute;toile&lt;/rt>&lt;rp>.&lt;/rp>
&lt;/ruby></code></pre>

   <p>This would make the example render as follows in non-ruby-capable user agents:

   <pre>&#x2665;: Heart, <span lang=fr>C&oelig;ur</span>. &#x2618;: Shamrock, <span lang=fr>Tr&egrave;fle</span>. &#x2736;: Star, <span lang=fr>&Eacute;toile</span>.</pre>

  </div>


  <h4>The <dfn element><code>data</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-data-value">value</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-data">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-data">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLDataElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-data-value">value</span>;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLDataElement</code>.</dd>
  </dl>

  <p>The <code>data</code> element <span>represents</span> its contents, along with a
  machine-readable form of those contents in the <code data-x="attr-data-value">value</code>
  attribute.</p>

  <p>The <dfn element-attr for="data"><code data-x="attr-data-value">value</code></dfn> attribute
  must be present. Its value must be a representation of the element's contents in a
  machine-readable format.</p>

  <p class="note">When the value is date- or time-related, the more specific <code>time</code>
  element can be used instead.</p>

  <p>The element can be used for several purposes.</p>

  <p>When combined with microformats or the <a href="#microdata">microdata attributes</a> defined in
  this specification, the element serves to provide both a machine-readable value for the purposes
  of data processors, and a human-readable value for the purposes of rendering in a web browser. In
  this case, the format to be used in the <code data-x="attr-data-value">value</code> attribute is
  determined by the microformats or microdata vocabulary in use.</p>

  <p>The element can also, however, be used in conjunction with scripts in the page, for when a
  script has a literal value to store alongside a human-readable value. In such cases, the format to
  be used depends only on the needs of the script. (The <code data-x="attr-data-*">data-*</code>
  attributes can also be useful in such situations.)</p>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLDataElement"><code data-x="dom-data-value">value</code></dfn> IDL
  attribute must <span>reflect</span> the content attribute of the same name.</p>

  </div>

  <div class="example">

   <p>Here, a short table has its numeric values encoded using the <code>data</code> element so
   that the table sorting JavaScript library can provide a sorting mechanism on each column
   despite the numbers being presented in textual form in one column and in a decomposed form in
   another.</p>

   <pre><code class="html">&lt;script src="sortable.js">&lt;/script>
&lt;table class="sortable">
 &lt;thead> &lt;tr> &lt;th> Game &lt;th> Corporations &lt;th> Map Size
 &lt;tbody>
  &lt;tr> &lt;td> 1830 &lt;td> &lt;data value="8">Eight&lt;/data> &lt;td> &lt;data value="93">19+74 hexes (93 total)&lt;/data>
  &lt;tr> &lt;td> 1856 &lt;td> &lt;data value="11">Eleven&lt;/data> &lt;td> &lt;data value="99">12+87 hexes (99 total)&lt;/data>
  &lt;tr> &lt;td> 1870 &lt;td> &lt;data value="10">Ten&lt;/data> &lt;td> &lt;data value="149">4+145 hexes (149 total)&lt;/data>
&lt;/table></code></pre>

  </div>




  <h4>The <dfn element><code>time</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>If the element has a <code data-x="attr-time-datetime">datetime</code> attribute: <span>Phrasing content</span>.</dd>
   <dd>Otherwise: <span data-x="text content">Text</span>, but must match requirements described in prose below.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-time-datetime">datetime</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-time">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-time">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
<pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLTimeElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-time-dateTime">dateTime</span>;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLTimeElement</code>.</dd>
  </dl>

  <p>The <code>time</code> element <span>represents</span> its contents, along with a
  machine-readable form of those contents in the <code data-x="attr-time-datetime">datetime</code>
  attribute. The kind of content is limited to various kinds of dates, times, time-zone offsets, and
  durations, as described below.</p>

  <p>The <dfn element-attr for="time"><code data-x="attr-time-datetime">datetime</code></dfn>
  attribute may be present. If present, its value must be a representation of the element's contents
  in a machine-readable format.</p>

  <p>A <code>time</code> element that does not have a <code
  data-x="attr-time-datetime">datetime</code> content attribute must not have any element
  descendants.</p>

  <p>The <dfn>datetime value</dfn> of a <code>time</code> element is the value of the element's
  <code data-x="attr-time-datetime">datetime</code> content attribute, if it has one, otherwise the
  <span>child text content</span> of the <code>time</code> element.</p>

  <p>The <span>datetime value</span> of a <code>time</code> element must match one of the following
  syntaxes.</p>

  <dl>
   <dt>A <span>valid month string</span></dt>

   <dd>
    <pre class="example"><code class="html">&lt;time>2011-11&lt;/time></code></pre>
   </dd>


   <dt>A <span>valid date string</span></dt>

   <dd>
    <pre class="example"><code class="html">&lt;time>2011-11-18&lt;/time></code></pre>
   </dd>


   <dt>A <span>valid yearless date string</span></dt>

   <dd>
    <pre class="example"><code class="html">&lt;time>11-18&lt;/time></code></pre>
   </dd>


   <dt>A <span>valid time string</span></dt>

   <dd>
    <pre class="example"><code class="html">&lt;time>14:54&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>14:54:39&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>14:54:39.929&lt;/time></code></pre>
   </dd>


   <dt>A <span>valid local date and time string</span></dt>

   <dd>
    <pre class="example"><code class="html">&lt;time>2011-11-18T14:54&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18T14:54:39&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18T14:54:39.929&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18 14:54&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18 14:54:39&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18 14:54:39.929&lt;/time></code></pre>

    <p class="note">Times with dates but without a time zone offset are useful for specifying events
    that are observed at the same specific time in each time zone, throughout a day. For example,
    the 2020 new year is celebrated at 2020-01-01 00:00 in each time zone, not at the same precise
    moment across all time zones. For events that occur at the same time across all time zones, for
    example a videoconference meeting, a <span>valid global date and time string</span> is likely
    more useful.</p>
   </dd>


   <dt>A <span>valid time-zone offset string</span></dt>

   <dd>
    <pre class="example"><code class="html">&lt;time>Z&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>+0000&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>+00:00&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>-0800&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>-08:00&lt;/time></code></pre>

    <p class="note">For times without dates (or times referring to events that recur on multiple
    dates), specifying the geographic location that controls the time is usually more useful than
    specifying a time zone offset, because geographic locations change time zone offsets with
    daylight saving time. In some cases, geographic locations even change time zone, e.g. when the
    boundaries of those time zones are redrawn, as happened with Samoa at the end of 2011. There
    exists a time zone database that describes the boundaries of time zones and what rules apply
    within each such zone, known as the <i>time zone database</i>. <ref>TZDATABASE</ref></p>
   </dd>


   <dt>A <span>valid global date and time string</span></dt>

   <dd>
    <pre class="example"><code class="html">&lt;time>2011-11-18T14:54Z&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18T14:54:39Z&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18T14:54:39.929Z&lt;/time></code></pre>

    <pre class="example"><code class="html">&lt;time>2011-11-18T14:54+0000&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18T14:54:39+0000&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18T14:54:39.929+0000&lt;/time></code></pre>

    <pre class="example"><code class="html">&lt;time>2011-11-18T14:54+00:00&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18T14:54:39+00:00&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18T14:54:39.929+00:00&lt;/time></code></pre>

    <pre class="example"><code class="html">&lt;time>2011-11-18T06:54-0800&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18T06:54:39-0800&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18T06:54:39.929-0800&lt;/time></code></pre>

    <pre class="example"><code class="html">&lt;time>2011-11-18T06:54-08:00&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18T06:54:39-08:00&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18T06:54:39.929-08:00&lt;/time></code></pre>

    <pre class="example"><code class="html">&lt;time>2011-11-18 14:54Z&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18 14:54:39Z&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18 14:54:39.929Z&lt;/time></code></pre>

    <pre class="example"><code class="html">&lt;time>2011-11-18 14:54+0000&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18 14:54:39+0000&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18 14:54:39.929+0000&lt;/time></code></pre>

    <pre class="example"><code class="html">&lt;time>2011-11-18 14:54+00:00&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18 14:54:39+00:00&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18 14:54:39.929+00:00&lt;/time></code></pre>

    <pre class="example"><code class="html">&lt;time>2011-11-18 06:54-0800&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18 06:54:39-0800&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18 06:54:39.929-0800&lt;/time></code></pre>

    <pre class="example"><code class="html">&lt;time>2011-11-18 06:54-08:00&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18 06:54:39-08:00&lt;/time></code></pre>
    <pre class="example"><code class="html">&lt;time>2011-11-18 06:54:39.929-08:00&lt;/time></code></pre>

    <p class="note">Times with dates and a time zone offset are useful for specifying specific
    events, or recurring virtual events where the time is not anchored to a specific geographic
    location. For example, the precise time of an asteroid impact, or a particular meeting in a
    series of meetings held at 1400 UTC every day, regardless of whether any particular part of the
    world is observing daylight saving time or not. For events where the precise time varies by the
    local time zone offset of a specific geographic location, a <span>valid local date and time
    string</span> combined with that geographic location is likely more useful.</p>
   </dd>


   <dt>A <span>valid week string</span></dt>

   <dd>
    <pre class="example"><code class="html">&lt;time>2011-W47&lt;/time></code></pre>
   </dd>


   <dt>Four or more <span>ASCII digits</span>, at least one of which is not U+0030 DIGIT ZERO (0)</dt>

   <dd>
    <pre class="example"><code class="html">&lt;time>2011&lt;/time></code></pre>

    <pre class="example"><code class="html">&lt;time>0001&lt;/time></code></pre>
   </dd>


   <dt>A <span>valid duration string</span></dt>

   <dd>
    <pre class="example"><code class="html">&lt;time>PT4H18M3S&lt;/time></code></pre>

    <pre class="example"><code class="html">&lt;time>4h 18m 3s&lt;/time></code></pre>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn>machine-readable equivalent of the element's contents</dfn> must be obtained from the
  element's <span>datetime value</span> by using the following algorithm:</p>

  <ol>
   <li><p>If <span data-x="parse a month string">parsing a month string</span> from the element's
   <span>datetime value</span> returns a <span data-x="concept-month">month</span>, that is the
   machine-readable equivalent; return.</p></li>

   <li><p>If <span data-x="parse a date string">parsing a date string</span> from the element's
   <span>datetime value</span> returns a <span data-x="concept-date">date</span>, that is the
   machine-readable equivalent; return.</p></li>

   <li><p>If <span data-x="parse a yearless date string">parsing a yearless date string</span> from
   the element's <span>datetime value</span> returns a <span data-x="concept-yearless-date">yearless
   date</span>, that is the machine-readable equivalent; return.</p></li>

   <li><p>If <span data-x="parse a time string">parsing a time string</span> from the element's
   <span>datetime value</span> returns a <span data-x="concept-time">time</span>, that is the
   machine-readable equivalent; return.</p></li>

   <li><p>If <span data-x="parse a local date and time string">parsing a local date and time
   string</span> from the element's <span>datetime value</span> returns a <span
   data-x="concept-datetime-local">local date and time</span>, that is the machine-readable
   equivalent; return.</p></li>

   <li><p>If <span data-x="parse a time-zone offset string">parsing a time-zone offset string</span>
   from the element's <span>datetime value</span> returns a <span data-x="concept-timezone">time-zone
   offset</span>, that is the machine-readable equivalent; return.</p></li>

   <li><p>If <span data-x="parse a global date and time string">parsing a global date and time
   string</span> from the element's <span>datetime value</span> returns a <span
   data-x="concept-datetime">global date and time</span>, that is the machine-readable equivalent;
   return.</p></li>

   <li><p>If <span data-x="parse a week string">parsing a week string</span> from the element's
   <span>datetime value</span> returns a <span data-x="concept-week">week</span>, that is the
   machine-readable equivalent; return.</p></li>

   <li><p>If the element's <span>datetime value</span> consists of only <span>ASCII digits</span>,
   at least one of which is not U+0030 DIGIT ZERO (0), then the machine-readable equivalent is the
   base-ten interpretation of those digits, representing a year; return.</p></li>

   <li><p>If <span data-x="parse a duration string">parsing a duration string</span> from the
   element's <span>datetime value</span> returns a <span data-x="concept-duration">duration</span>,
   that is the machine-readable equivalent; return.</p></li>

   <li><p>There is no machine-readable equivalent.</p></li>
  </ol>

  <p class="note">The algorithms referenced above are intended to be designed such that for any
  arbitrary string <var>s</var>, only one of the algorithms returns a value. A more
  efficient approach might be to create a single algorithm that parses all these data types in one
  pass; developing such an algorithm is left as an exercise to the reader.</p>

  <p>The <dfn attribute for="HTMLTimeElement"><code data-x="dom-time-dateTime">dateTime</code></dfn>
  IDL attribute must <span>reflect</span> the element's <code
  data-x="attr-time-datetime">datetime</code> content attribute.</p>

  </div>

  <div class="example">

   <p>The <code>time</code> element can be used to encode dates, for example in microformats. The
   following shows a hypothetical way of encoding an event using a variant on hCalendar that uses
   the <code>time</code> element:</p>

   <pre><code class="html">&lt;div class="vevent">
 &lt;a class="url" href="http://www.web2con.com/">http://www.web2con.com/&lt;/a>
 &lt;span class="summary">Web 2.0 Conference&lt;/span>:
 &lt;time class="dtstart" datetime="2005-10-05">October 5&lt;/time> -
 &lt;time class="dtend" datetime="2005-10-07">7&lt;/time>,
 at the &lt;span class="location">Argent Hotel, San Francisco, CA&lt;/span>
&lt;/div></code></pre>

  </div>

  <div class="example">

   <p>Here, a fictional microdata vocabulary based on the Atom vocabulary is used with the
   <code>time</code> element to mark up a blog post's publication date.</p>

   <pre><code class="html">&lt;article itemscope itemtype="https://n.example.org/rfc4287">
 &lt;h1 itemprop="title">Big tasks&lt;/h1>
 &lt;footer>Published &lt;time itemprop="published" datetime="2009-08-29">two days ago&lt;/time>.&lt;/footer>
 &lt;p itemprop="content">Today, I went out and bought a bike for my kid.&lt;/p>
&lt;/article></code></pre>

  </div>

  <div class="example">

   <p>In this example, another article's publication date is marked up using <code>time</code>, this
   time using the schema.org microdata vocabulary:</p>

   <pre><code class="html">&lt;article itemscope itemtype="http://schema.org/BlogPosting">
 &lt;h1 itemprop="headline">Small tasks&lt;/h1>
 &lt;footer>Published &lt;time itemprop="datePublished" datetime="2009-08-30">yesterday&lt;/time>.&lt;/footer>
 &lt;p itemprop="articleBody">I put a bike bell on her bike.&lt;/p>
&lt;/article></code></pre>

  </div>

  <div class="example">

   <p>In the following snippet, the <code>time</code> element is used to encode a date in the
   ISO8601 format, for later processing by a script:</p>

   <pre><code class="html">&lt;p>Our first date was &lt;time datetime="2006-09-23">a Saturday&lt;/time>.&lt;/p></code></pre>

   <p>In this second snippet, the value includes a time:</p>

   <pre><code class="html">&lt;p>We stopped talking at &lt;time datetime="2006-09-24T05:00-07:00">5am the next morning&lt;/time>.&lt;/p></code></pre>

   <p>A script loaded by the page (and thus privy to the page's internal convention of marking up
   dates and times using the <code>time</code> element) could scan through the page and look at all
   the <code>time</code> elements therein to create an index of dates and times.</p>

  </div>

  <div class="example">

   <p>For example, this element conveys the string "Friday" with the additional semantic that the
   18th of November 2011 is the meaning that corresponds to "Friday":</p>

   <pre><code class="html">Today is &lt;time datetime="2011-11-18">Friday&lt;/time>.</code></pre>

  </div>

  <div class="example">

   <p>In this example, a specific time in the Pacific Standard Time timezone is specified:</p>

   <pre><code class="html">Your next meeting is at &lt;time datetime="2011-11-18T15:00-08:00">3pm&lt;/time>.</code></pre>

  </div>




  <h4>The <dfn element><code>code</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-code">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-code">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>code</code> element <span>represents</span> a fragment of computer code. This could
  be an XML element name, a filename, a computer program, or any other string that a computer would
  recognize.</p>

  <p>There is no formal way to indicate the language of computer code being marked up. Authors who
  wish to mark <code>code</code> elements with the language used, e.g. so that syntax highlighting
  scripts can use the right rules, can use the <code data-x="attr-class">class</code> attribute, e.g.
  by adding a class prefixed with "<code data-x="">language-</code>" to the element.</p>

  <div class="example">

   <p>The following example shows how the element can be used in a paragraph to mark up element
   names and computer code, including punctuation.</p>

   <pre><code class="html">&lt;p>The &lt;code>code&lt;/code> element represents a fragment of computer
code.&lt;/p>

&lt;p>When you call the &lt;code>activate()&lt;/code> method on the
&lt;code>robotSnowman&lt;/code> object, the eyes glow.&lt;/p>

&lt;p>The example below uses the &lt;code>begin&lt;/code> keyword to indicate
the start of a statement block. It is paired with an &lt;code>end&lt;/code>
keyword, which is followed by the &lt;code>.&lt;/code> punctuation character
(full stop) to indicate the end of the program.&lt;/p></code></pre>

  </div>

  <div class="example">

   <p>The following example shows how a block of code could be marked up using the <code>pre</code>
   and <code>code</code> elements.</p>

   <pre><code class="html">&lt;pre>&lt;code class="language-pascal">var i: Integer;
begin
   i := 1;
end.&lt;/code>&lt;/pre></code></pre>

   <p>A class is used in that example to indicate the language used.</p>

  </div>

  <p class="note">See the <code>pre</code> element for more details.</p>


  <h4>The <dfn element><code>var</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-var">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-var">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>var</code> element <span>represents</span> a variable. This could be an actual
  variable in a mathematical expression or programming context, an identifier representing a
  constant, a symbol identifying a physical quantity, a function parameter, or just be a term used
  as a placeholder in prose.</p>

  <div class="example">
   <p>In the paragraph below, the letter "n" is being used as a
   variable in prose:</p>
   <pre><code class="html">&lt;p>If there are &lt;var>n&lt;/var> pipes leading to the ice
cream factory then I expect at &lt;em>least&lt;/em> &lt;var>n&lt;/var>
flavors of ice cream to be available for purchase!&lt;/p></code></pre>
  </div>

  <p>For mathematics, in particular for anything beyond the simplest of expressions, MathML is more
  appropriate. However, the <code>var</code> element can still be used to refer to specific
  variables that are then mentioned in MathML expressions.</p>

  <div class="example">

   <p>In this example, an equation is shown, with a legend that references the variables in the
   equation. The expression itself is marked up with MathML, but the variables are mentioned in the
   figure's legend using <code>var</code>.</p>

   <pre><code class="html">&lt;figure>
 &lt;math>
  &lt;mi>a&lt;/mi>
  &lt;mo>=&lt;/mo>
  &lt;msqrt>
   &lt;msup>&lt;mi>b&lt;/mi>&lt;mn>2&lt;/mn>&lt;/msup>
   &lt;mi>+&lt;/mi>
   &lt;msup>&lt;mi>c&lt;/mi>&lt;mn>2&lt;/mn>&lt;/msup>
  &lt;/msqrt>
 &lt;/math>
 &lt;figcaption>
  Using Pythagoras' theorem to solve for the hypotenuse &lt;var>a&lt;/var> of
  a triangle with sides &lt;var>b&lt;/var> and &lt;var>c&lt;/var>
 &lt;/figcaption>
&lt;/figure></code></pre>

  </div>

  <div class="example">

   <p>Here, the equation describing mass-energy equivalence is used in a sentence, and the
   <code>var</code> element is used to mark the variables and constants in that equation:</p>

   <pre><code class="html">&lt;p>Then she turned to the blackboard and picked up the chalk. After a few moment's
thought, she wrote &lt;var>E&lt;/var> = &lt;var>m&lt;/var> &lt;var>c&lt;/var>&lt;sup>2&lt;/sup>. The teacher
looked pleased.&lt;/p></code></pre>

  </div>


  <h4>The <dfn element><code>samp</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-samp">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-samp">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>samp</code> element <span>represents</span> sample or quoted output from another
  program or computing system.</p>

  <p class="note">See the <code>pre</code> and <code>kbd</code> elements for more details.</p>

  <p class="note">This element can be contrasted with the <code>output</code> element, which can be
  used to provide immediate output in a web application.</p>

  <div class="example">
   <p>This example shows the <code>samp</code> element being used
   inline:</p>
   <pre><code class="html">&lt;p>The computer said &lt;samp>Too much cheese in tray
two&lt;/samp> but I didn't know what that meant.&lt;/p></code></pre>
  </div>

  <div class="example">

   <p>This second example shows a block of sample output from a console program. Nested
   <code>samp</code> and <code>kbd</code> elements allow for the styling of specific elements
   of the sample output using a style sheet. There's also a few parts of the <code>samp</code> that
   are annotated with even more detailed markup, to enable very precise styling. To achieve this,
   <code>span</code> elements are used.</p>

   <pre><code class="html">&lt;pre>&lt;samp>&lt;span class="prompt">jdoe@mowmow:~$&lt;/span> &lt;kbd>ssh demo.example.com&lt;/kbd>
Last login: Tue Apr 12 09:10:17 2005 from mowmow.example.com on pts/1
Linux demo 2.6.10-grsec+gg3+e+fhs6b+nfs+gr0501+++p3+c4a+gr2b-reslog-v6.189 #1 SMP Tue Feb 1 11:22:36 PST 2005 i686 unknown

&lt;span class="prompt">jdoe@demo:~$&lt;/span> &lt;span class="cursor">_&lt;/span>&lt;/samp>&lt;/pre></code></pre>

  </div>

  <div class="example">

   <p>This third example shows a block of input and its respective output. The example uses
   both <code>code</code> and <code>samp</code> elements.</p>

   <pre><code class="html">&lt;pre>
&lt;code class=&quot;language-javascript&quot;>console.log(2.3 + 2.4)&lt;/code>
&lt;samp>4.699999999999999&lt;/samp>
&lt;/pre></code></pre>

  </div>


  <h4>The <dfn element><code>kbd</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-kbd">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-kbd">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>kbd</code> element <span>represents</span> user input (typically keyboard input,
  although it may also be used to represent other input, such as voice commands).</p>

  <p>When the <code>kbd</code> element is nested inside a <code>samp</code> element, it represents
  the input as it was echoed by the system.</p>

  <p>When the <code>kbd</code> element <em>contains</em> a <code>samp</code> element, it represents
  input based on system output, for example invoking a menu item.</p>

  <p>When the <code>kbd</code> element is nested inside another <code>kbd</code> element, it
  represents an actual key or other single unit of input as appropriate for the input mechanism.</p>

  <div class="example">

   <p>Here the <code>kbd</code> element is used to indicate keys to press:</p>

   <pre><code class="html">&lt;p>To make George eat an apple, press &lt;kbd>&lt;kbd>Shift&lt;/kbd> + &lt;kbd>F3&lt;/kbd>&lt;/kbd>&lt;/p></code></pre>

   <p>In this second example, the user is told to pick a particular menu item. The outer
   <code>kbd</code> element marks up a block of input, with the inner <code>kbd</code> elements
   representing each individual step of the input, and the <code>samp</code> elements inside them
   indicating that the steps are input based on something being displayed by the system, in this
   case menu labels:</p>

   <pre><code class="html">&lt;p>To make George eat an apple, select
    &lt;kbd>&lt;kbd>&lt;samp>File&lt;/samp>&lt;/kbd>|&lt;kbd>&lt;samp>Eat Apple...&lt;/samp>&lt;/kbd>&lt;/kbd>
&lt;/p></code></pre>

   <p>Such precision isn't necessary; the following is equally fine:</p>

   <pre><code class="html">&lt;p>To make George eat an apple, select &lt;kbd>File | Eat Apple...&lt;/kbd>&lt;/p></code></pre>

  </div>


  <h4>The <dfn element id="the-sub-element"><code>sub</code></dfn> and <dfn element
  id="the-sup-element"><code>sup</code></dfn> elements</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd>The <code>sub</code> element: <a
   href="https://w3c.github.io/html-aria/#el-sub">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-sub">for implementers</a>.</dd>
   <dd>The <code>sup</code> element: <a
   href="https://w3c.github.io/html-aria/#el-sup">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-sup">for implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Use <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>sup</code> element <span>represents</span> a superscript and the <code>sub</code>
  element <span>represents</span> a subscript.</p>

  <p>These elements must be used only to mark up typographical conventions with specific meanings,
  not for typographical presentation for presentation's sake. For example, it would be inappropriate
  for the <code>sub</code> and <code>sup</code> elements to be used in the name of the LaTeX
  document preparation system. In general, authors should use these elements only if the
  <em>absence</em> of those elements would change the meaning of the content.</p>

  <p>In certain languages, superscripts are part of the typographical conventions for some
  abbreviations.</p>

  <div class="example">
   <pre><code class="html">&lt;p>Their names are
&lt;span lang="fr">&lt;abbr>M&lt;sup>lle&lt;/sup>&lt;/abbr> Gwendoline&lt;/span> and
&lt;span lang="fr">&lt;abbr>M&lt;sup>me&lt;/sup>&lt;/abbr> Denise&lt;/span>.&lt;/p></code></pre>
  </div>

  <p>The <code>sub</code> element can be used inside a <code>var</code> element, for variables that
  have subscripts.</p>

  <div class="example">

   <p>Here, the <code>sub</code> element is used to represent the subscript that identifies the
   variable in a family of variables:</p>

   <pre><code class="html">&lt;p>The coordinate of the &lt;var>i&lt;/var>th point is
(&lt;var>x&lt;sub>&lt;var>i&lt;/var>&lt;/sub>&lt;/var>, &lt;var>y&lt;sub>&lt;var>i&lt;/var>&lt;/sub>&lt;/var>).
For example, the 10th point has coordinate
(&lt;var>x&lt;sub>10&lt;/sub>&lt;/var>, &lt;var>y&lt;sub>10&lt;/sub>&lt;/var>).&lt;/p></code></pre>
  </div>

  <p>Mathematical expressions often use subscripts and superscripts. Authors are encouraged to use
  MathML for marking up mathematics, but authors may opt to use <code>sub</code> and
  <code>sup</code> if detailed mathematical markup is not desired. <ref>MATHML</ref></p>

  <div class="example">
   <pre><code class="html">&lt;var>E&lt;/var>=&lt;var>m&lt;/var>&lt;var>c&lt;/var>&lt;sup>2&lt;/sup></code></pre>
   <pre><code class="html">f(&lt;var>x&lt;/var>, &lt;var>n&lt;/var>) = log&lt;sub>4&lt;/sub>&lt;var>x&lt;/var>&lt;sup>&lt;var>n&lt;/var>&lt;/sup></code></pre>
  </div>



  <h4>The <dfn element><code>i</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-i">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-i">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>i</code> element <span>represents</span> a span of text in an alternate voice or
  mood, or otherwise offset from the normal prose in a manner indicating a different quality of
  text, such as a taxonomic designation, a technical term, an idiomatic phrase from another
  language, transliteration, a thought, or a ship name in Western texts.</p>

  <p>Terms in languages different from the main text should be annotated with <code
  data-x="attr-lang">lang</code> attributes (or, in XML, <span data-x="attr-xml-lang"><code
  data-x="">lang</code> attributes in the <span>XML namespace</span></span>).</p>

  <div class="example">
   <p>The examples below show uses of the <code>i</code> element:</p>
   <pre><code class="html">&lt;p>The &lt;i class="taxonomy">Felis silvestris catus&lt;/i> is cute.&lt;/p>
&lt;p>The term &lt;i>prose content&lt;/i> is defined above.&lt;/p>
&lt;p>There is a certain &lt;i lang="fr">je ne sais quoi&lt;/i> in the air.&lt;/p></code></pre>
   <p>In the following example, a dream sequence is marked up using
   <code>i</code> elements.</p>
   <pre><code class="html">&lt;p>Raymond tried to sleep.&lt;/p>
&lt;p>&lt;i>The ship sailed away on Thursday&lt;/i>, he
dreamt. &lt;i>The ship had many people aboard, including a beautiful
princess called Carey. He watched her, day-in, day-out, hoping she
would notice him, but she never did.&lt;/i>&lt;/p>
&lt;p>&lt;i>Finally one night he picked up the courage to speak with
her&mdash;&lt;/i>&lt;/p>
&lt;p>Raymond woke with a start as the fire alarm rang out.&lt;/p></code></pre>
  </div>

  <p>Authors can use the <code data-x="attr-class">class</code> attribute on the <code>i</code>
  element to identify why the element is being used, so that if the style of a particular use (e.g.
  dream sequences as opposed to taxonomic terms) is to be changed at a later date, the author
  doesn't have to go through the entire document (or series of related documents) annotating each
  use.</p>

  <p>Authors are encouraged to consider whether other elements might be more applicable than the
  <code>i</code> element, for instance the <code>em</code> element for marking up stress emphasis,
  or the <code>dfn</code> element to mark up the defining instance of a term.</p>

  <p class="note">Style sheets can be used to format <code>i</code> elements, just like any other
  element can be restyled. Thus, it is not the case that content in <code>i</code> elements will
  necessarily be italicized.</p>


  <h4>The <dfn element><code>b</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-b">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-b">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>b</code> element <span>represents</span> a span of text to which attention is being
  drawn for utilitarian purposes without conveying any extra importance and with no implication of
  an alternate voice or mood, such as key words in a document abstract, product names in a review,
  actionable words in interactive text-driven software, or an article lede.</p>

  <div class="example">

   <p>The following example shows a use of the <code>b</code> element to highlight key words without
   marking them up as important:</p>

   <pre><code class="html">&lt;p>The &lt;b>frobonitor&lt;/b> and &lt;b>barbinator&lt;/b> components are fried.&lt;/p></code></pre>

  </div>

  <div class="example">

   <p>In the following example, objects in a text adventure are highlighted as being special by use
   of the <code>b</code> element.</p>

   <pre><code class="html">&lt;p>You enter a small room. Your &lt;b>sword&lt;/b> glows
brighter. A &lt;b>rat&lt;/b> scurries past the corner wall.&lt;/p></code></pre>

  </div>

  <div class="example">

   <p>Another case where the <code>b</code> element is appropriate is in marking up the lede (or
   lead) sentence or paragraph. The following example shows how a <a
   href="http://news.bbc.co.uk/2/hi/uk_news/scotland/north_east/7101506.stm">BBC article about
   kittens adopting a rabbit as their own</a> could be marked up:</p>

   <pre><code class="html">&lt;article>
 &lt;h2>Kittens 'adopted' by pet rabbit&lt;/h2>
 &lt;p>&lt;b class="lede">Six abandoned kittens have found an
 unexpected new mother figure &mdash; a pet rabbit.&lt;/b>&lt;/p>
 &lt;p>Veterinary nurse Melanie Humble took the three-week-old
 kittens to her Aberdeen home.&lt;/p>
<i>[...]</i></code></pre>

  </div>

  <p>As with the <code>i</code> element, authors can use the <code data-x="attr-class">class</code>
  attribute on the <code>b</code> element to identify why the element is being used, so that if the
  style of a particular use is to be changed at a later date, the author doesn't have to go through
  annotating each use.</p>

  <p>The <code>b</code> element should be used as a last resort when no other element is more
  appropriate. In particular, headings should use the <code>h1</code> to <code>h6</code> elements,
  stress emphasis should use the <code>em</code> element, importance should be denoted with the
  <code>strong</code> element, and text marked or highlighted should use the <code>mark</code>
  element.</p>

  <div class="example">

   <p>The following would be <em>incorrect</em> usage:</p>

   <pre class="bad"><code class="html">&lt;p>&lt;b>WARNING!&lt;/b> Do not frob the barbinator!&lt;/p></code></pre>

   <p>In the previous example, the correct element to use would have been <code>strong</code>, not
   <code>b</code>.</p>

  </div>

  <p class="note">Style sheets can be used to format <code>b</code> elements, just like any other
  element can be restyled. Thus, it is not the case that content in <code>b</code> elements will
  necessarily be boldened.</p>



  <h4>The <dfn element><code>u</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-u">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-u">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>u</code> element <span>represents</span> a span of text with an unarticulated, though
  explicitly rendered, non-textual annotation, such as labeling the text as being a proper name in
  Chinese text (a Chinese proper name mark), or labeling the text as being misspelt.</p>

  <p>In most cases, another element is likely to be more appropriate: for marking stress emphasis,
  the <code>em</code> element should be used; for marking key words or phrases either the
  <code>b</code> element or the <code>mark</code> element should be used, depending on the context;
  for marking book titles, the <code>cite</code> element should be used<!-- even for the Chinese
  wavy underline 'book title mark' -->; for labeling text with explicit textual annotations, the
  <code>ruby</code> element should be used; for technical terms, taxonomic designation,
  transliteration, a thought, or for labeling ship names in Western texts, the <code>i</code>
  element should be used.</p>

  <p class="note">The default rendering of the <code>u</code> element in visual presentations
  clashes with the conventional rendering of hyperlinks (underlining). Authors are encouraged to
  avoid using the <code>u</code> element where it could be confused for a hyperlink.</p>

  <div class="example">

   <p>In this example, a <code>u</code> element is used to mark a word as misspelt:</p>

   <pre><code class="html">&lt;p>The &lt;u>see&lt;/u> is full of fish.&lt;/p></code></pre>

  </div>



  <h4>The <dfn element><code>mark</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-mark">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-mark">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <!-- v2: attribute that means "highlight this on the scrollbar" -->

  <p>The <code>mark</code> element <span>represents</span> a run of text in one document marked or
  highlighted for <span data-x="referenced">reference</span> purposes, due to its relevance in
  another context. When used in a quotation or other block of text referred to from the prose, it
  indicates a highlight that was not originally present but which has been added to bring the
  reader's attention to a part of the text that might not have been considered important by the
  original author when the block was originally written, but which is now under previously
  unexpected scrutiny. When used in the main prose of a document, it indicates a part of the
  document that has been highlighted due to its likely relevance to the user's current activity.</p>

  <div class="example">

   <p>This example shows how the <code>mark</code> element can be used to bring attention to a
   particular part of a quotation:</p>

   <pre><code class="html">&lt;p lang="en-US">Consider the following quote:&lt;/p>
<span lang="en-GB">&lt;blockquote lang="en-GB">
 &lt;p>Look around and you will find, no-one's really
 <!--en-GB-->&lt;mark>colour&lt;/mark> blind.&lt;/p>
&lt;/blockquote></span>
&lt;p lang="en-US">As we can tell from the &lt;em>spelling&lt;/em> of the word,
the person writing this quote is clearly not American.&lt;/p></code></pre>

   <p>(If the goal was to mark the element as misspelt, however, the <code>u</code> element,
   possibly with a class, would be more appropriate.)</p>

  </div>

  <div class="example">

   <p>Another example of the <code>mark</code> element is highlighting parts of a document that are
   matching some search string. If someone looked at a document, and the server knew that the user
   was searching for the word "kitten", then the server might return the document with one paragraph
   modified as follows:</p>

   <pre><code class="html">&lt;p>I also have some &lt;mark>kitten&lt;/mark>s who are visiting me
these days. They're really cute. I think they like my garden! Maybe I
should adopt a &lt;mark>kitten&lt;/mark>.&lt;/p></code></pre>

  </div>

  <div class="example">

   <p>In the following snippet, a paragraph of text refers to a specific part of a code
   fragment.</p>

   <pre><code class="html">&lt;p>The highlighted part below is where the error lies:&lt;/p>
&lt;pre>&lt;code>var i: Integer;
begin
   i := &lt;mark>1.1&lt;/mark>;
end.&lt;/code>&lt;/pre></code></pre>

   <p>This is separate from <em>syntax highlighting</em>, for which <code>span</code> is more
   appropriate. Combining both, one would get:</p>

   <pre><code class="html">&lt;p>The highlighted part below is where the error lies:&lt;/p>
&lt;pre>&lt;code>&lt;span class=keyword>var&lt;/span> &lt;span class=ident>i&lt;/span>: &lt;span class=type>Integer&lt;/span>;
&lt;span class=keyword>begin&lt;/span>
   &lt;span class=ident>i&lt;/span> := &lt;span class=literal>&lt;mark>1.1&lt;/mark>&lt;/span>;
&lt;span class=keyword>end&lt;/span>.&lt;/code>&lt;/pre></code></pre>

  </div>

  <div class="example">

   <p>This is another example showing the use of <code>mark</code> to highlight a part of quoted
   text that was originally not emphasized. In this example, common typographic conventions have led
   the author to explicitly style <code>mark</code> elements in quotes to render in italics.</p>

   <pre><code class="html">&lt;style>
 blockquote mark, q mark {
   font: inherit; font-style: italic;
   text-decoration: none;
   background: transparent; color: inherit;
 }
 .bubble em {
   font: inherit; font-size: larger;
   text-decoration: underline;
 }
&lt;/style>
&lt;article>
 &lt;h1>She knew&lt;/h1>
 &lt;p>Did you notice the subtle joke in the joke on panel 4?&lt;/p>
 &lt;blockquote>
  &lt;p class="bubble">I didn't &lt;em>want&lt;/em> to believe. &lt;mark>Of course
  on some level I realized it was a known-plaintext attack.&lt;/mark> But I
  couldn't admit it until I saw for myself.&lt;/p>
 &lt;/blockquote>
 &lt;p>(Emphasis mine.) I thought that was great. It's so pedantic, yet it
 explains everything neatly.&lt;/p>
&lt;/article></code></pre>

   <p>Note, incidentally, the distinction between the <code>em</code> element in this example, which
   is part of the original text being quoted, and the <code>mark</code> element, which is
   highlighting a part for comment.</p>

  </div>

  <div class="example">

   <p>The following example shows the difference between denoting the <em>importance</em> of a span
   of text (<code>strong</code>) as opposed to denoting the <em>relevance</em> of a span of text
   (<code>mark</code>). It is an extract from a textbook, where the extract has had the parts
   relevant to the exam highlighted. The safety warnings, important though they may be, are
   apparently not relevant to the exam.</p>

   <pre><code class="html">&lt;h3>Wormhole Physics Introduction&lt;/h3>

&lt;p>&lt;mark>A wormhole in normal conditions can be held open for a
maximum of just under 39 minutes.&lt;/mark> Conditions that can increase
the time include a powerful energy source coupled to one or both of
the gates connecting the wormhole, and a large gravity well (such as a
black hole).&lt;/p>

&lt;p>&lt;mark>Momentum is preserved across the wormhole. Electromagnetic
radiation can travel in both directions through a wormhole,
but matter cannot.&lt;/mark>&lt;/p>

&lt;p>When a wormhole is created, a vortex normally forms.
&lt;strong>Warning: The vortex caused by the wormhole opening will
annihilate anything in its path.&lt;/strong> Vortexes can be avoided when
using sufficiently advanced dialing technology.&lt;/p>

&lt;p>&lt;mark>An obstruction in a gate will prevent it from accepting a
wormhole connection.&lt;/mark>&lt;/p></code></pre>

  </div>


  <h4>The <dfn element><code>bdi</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd>Also, the <code data-x="attr-dir">dir</code><!-- no-annotate --> global attribute has special semantics on this element.</dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-bdi">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-bdi">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>bdi</code> element <span>represents</span> a span of text that is to be isolated from
  its surroundings for the purposes of bidirectional text formatting. <ref>BIDI</ref></p>

  <p class="note">The <code data-x="attr-dir">dir</code> global attribute defaults to <code
  data-x="attr-dir-auto">auto</code> on this element (it never inherits from the parent element like
  with other elements).</p>

  <div w-nodev>

  <p class="note">This element <a href="#bidireq">has rendering requirements involving the
  bidirectional algorithm</a>.</p>

  </div>

  <div class="example">

   <p>This element is especially useful when embedding user-generated content with an unknown
   directionality.</p>

   <p>In this example, usernames are shown along with the number of posts that the user has
   submitted. If the <code>bdi</code> element were not used, the username of the Arabic user would
   end up confusing the text (the bidirectional algorithm would put the colon and the number "3"
   next to the word "User" rather than next to the word "posts").</p>

   <pre><code class="html">&lt;ul>
 &lt;li>User &lt;bdi>jcranmer&lt;/bdi>: 12 posts.
 &lt;li>User &lt;bdi>hober&lt;/bdi>: 5 posts.
 &lt;li>User &lt;bdi><bdo dir=rtl>&#x625;&#x64a;&#x627;&#x646;</bdo>&lt;/bdi>: 3 posts.
&lt;/ul></code></pre>

   <figure>
    <img src="/images/sample-bdi.png" alt="">
    <figcaption>When using the <code>bdi</code> element, the username acts as expected.</figcaption>
   </figure>

   <figure>
    <img src="/images/sample-not-bdi.png" alt="">
    <figcaption>If the <code>bdi</code> element were to be replaced by a <code>b</code> element, the username would confuse the bidirectional algorithm and the third bullet would end up saying "User 3 :", followed by the Arabic name (right-to-left), followed by "posts" and a period.</figcaption>
   </figure>

  </div>


  <h4>The <dfn element><code>bdo</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd>Also, the <code data-x="attr-dir">dir</code><!-- no-annotate --> global attribute has special semantics on this element.</dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-bdo">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-bdo">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>bdo</code> element <span>represents</span> explicit text directionality formatting
  control for its children. It allows authors to override the Unicode bidirectional algorithm by
  explicitly specifying a direction override. <ref>BIDI</ref></p>

  <p>Authors must specify the <code data-x="attr-dir">dir</code> attribute on this element, with the
  value <code data-x="attr-dir-ltr">ltr</code> to specify a left-to-right override and with the value <code data-x="attr-dir-rtl">rtl</code> to
  specify a right-to-left override. The <code data-x="attr-dir-auto">auto</code> value must not be specified.</p>

  <div w-nodev>

  <p class="note">This element <a href="#bidireq">has rendering requirements involving the
  bidirectional algorithm</a>.</p>

  </div>



  <h4>The <dfn element><code>span</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-span">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-span">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLSpanElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLSpanElement</code>.</dd>
  </dl>

  <p>The <code>span</code> element doesn't mean anything on its own, but can be useful when used
  together with the <span>global attributes</span>, e.g. <code data-x="attr-class">class</code>,
  <code data-x="attr-lang">lang</code>, or <code data-x="attr-dir">dir</code>. It
  <span>represents</span> its children.</p>

  <div class="example">

   <p>In this example, a code fragment is marked up using <code>span</code> elements and <code
   data-x="attr-class">class</code> attributes so that its keywords and identifiers can be
   color-coded from CSS:</p>

   <!-- extract from https://www.cs.cmu.edu/~dst/DeCSS/Gallery/vlc-dvd_css-c.txt -->

   <pre><code class="html">&lt;pre>&lt;code class="lang-c">&lt;span class="keyword">for&lt;/span> (&lt;span class="ident">j&lt;/span> = 0; &lt;span class="ident">j&lt;/span> &amp;lt; 256; &lt;span class="ident">j&lt;/span>++) {
  &lt;span class="ident">i_t3&lt;/span> = (&lt;span class="ident">i_t3&lt;/span> & 0x1ffff) | (&lt;span class="ident">j&lt;/span> &amp;lt;&amp;lt; 17);
  &lt;span class="ident">i_t6&lt;/span> = (((((((&lt;span class="ident">i_t3&lt;/span> >> 3) ^ &lt;span class="ident">i_t3&lt;/span>) >> 1) ^ &lt;span class="ident">i_t3&lt;/span>) >> 8) ^ &lt;span class="ident">i_t3&lt;/span>) >> 5) & 0xff;
  &lt;span class="keyword">if&lt;/span> (&lt;span class="ident">i_t6&lt;/span> == &lt;span class="ident">i_t1&lt;/span>)
    &lt;span class="keyword">break&lt;/span>;
}&lt;/code>&lt;/pre></code></pre>

  </div>



  <h4>The <dfn element><code>br</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="concept-content-nothing">Nothing</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-br">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-br">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLBRElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  // <a href="#HTMLBRElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLBRElement</code>.</dd>
  </dl>

  <p>The <code>br</code> element <span>represents</span> a line break.</p>

  <p class="note">While line breaks are usually represented in visual media by physically moving
  subsequent text to a new line, a style sheet or user agent would be equally justified in causing
  line breaks to be rendered in a different manner, for instance as green dots, or as extra
  spacing.</p>

  <p><code>br</code> elements must be used only for line breaks that are actually part of the
  content, as in poems or addresses.</p>

  <div class="example">

  <p>The following example is correct usage of the <code>br</code> element:</p>

   <pre><code class="html">&lt;p&gt;P. Sherman&lt;br&gt;
42 Wallaby Way&lt;br&gt;
Sydney&lt;/p&gt;</code></pre>

  </div>

  <p><code>br</code> elements must not be used for separating thematic groups in a paragraph.</p>

  <div class="example">

   <p>The following examples are non-conforming, as they abuse the <code>br</code> element:</p>

   <pre><code class="html">&lt;p&gt;&lt;a ...&gt;34 comments.&lt;/a&gt;&lt;br&gt;
&lt;a ...&gt;Add a comment.&lt;/a&gt;&lt;/p&gt;</code></pre>

   <pre><code class="html">&lt;p&gt;&lt;label&gt;Name: &lt;input name="name"&gt;&lt;/label&gt;&lt;br&gt;
&lt;label&gt;Address: &lt;input name="address"&gt;&lt;/label&gt;&lt;/p&gt;</code></pre>

   <p>Here are alternatives to the above, which are correct:</p>

   <pre><code class="html">&lt;p&gt;&lt;a ...&gt;34 comments.&lt;/a&gt;&lt;/p&gt;
&lt;p&gt;&lt;a ...&gt;Add a comment.&lt;/a&gt;&lt;/p&gt;</code></pre>

   <pre><code class="html">&lt;p&gt;&lt;label&gt;Name: &lt;input name="name"&gt;&lt;/label&gt;&lt;/p&gt;
&lt;p&gt;&lt;label&gt;Address: &lt;input name="address"&gt;&lt;/label&gt;&lt;/p&gt;</code></pre>

  </div>

  <p>If a <span>paragraph</span> consists of nothing but a single <code>br</code> element, it
  represents a placeholder blank line (e.g. as in a template). Such blank lines must not be used for
  presentation purposes.</p>

  <div w-nodev>

  <p>Any content inside <code>br</code> elements must not be considered part of the surrounding
  text.</p>

  <p class="note">This element <a href="#bidireq">has rendering requirements involving the
  bidirectional algorithm</a>.</p>

  </div>


  <h4>The <dfn element><code>wbr</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="concept-content-nothing">Nothing</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-wbr">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-wbr">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>wbr</code> element <span>represents</span> a line break opportunity.</p>

  <div class="example">

   <p>In the following example, someone is quoted as saying something which, for effect, is written
   as one long word. However, to ensure that the text can be wrapped in a readable fashion, the
   individual words in the quote are separated using a <code>wbr</code> element.</p>

   <pre><code class="html">&lt;p&gt;So then she pointed at the tiger and screamed
"there&lt;wbr>is&lt;wbr>no&lt;wbr>way&lt;wbr>you&lt;wbr>are&lt;wbr>ever&lt;wbr>going&lt;wbr>to&lt;wbr>catch&lt;wbr>me"!&lt;/p&gt;</code></pre>

  </div>

  <div w-nodev>

   <p>Any content inside <code>wbr</code> elements must not be considered part of the surrounding
   text.</p>

   <pre class="example"><code class="js">var wbr = document.createElement("wbr");
wbr.textContent = "This is wrong";
document.body.appendChild(wbr);</code></pre>

   <p class="note">This element <a href="#bidireq">has rendering requirements involving the
   bidirectional algorithm</a>.</p>

  </div>



  <h4 id="usage-summary">Usage summary</h4>

  <!-- NON-NORMATIVE SECTION -->

  <table>
   <thead>
    <tr>
     <th>Element
     <th>Purpose
     <th>Example
   <tbody>
    <tr>
     <td><code>a</code>
     <td>Hyperlinks
     <td><pre class="example"><code class="html">Visit my <strong>&lt;a href="drinks.html">drinks&lt;/a></strong> page.</code></pre>

    <tr>
     <td><code>em</code>
     <td>Stress emphasis
     <td><pre class="example"><code class="html">I <!--non-normative-->must say I <strong>&lt;em>adore&lt;/em></strong> lemonade.</code></pre>

    <tr>
     <td><code>strong</code>
     <td>Importance
     <td><pre class="example"><code class="html">This tea is <strong>&lt;strong>very hot&lt;/strong></strong>.</code></pre>

    <tr>
     <td><code>small</code>
     <td>Side comments
     <td><pre class="example"><code class="html">These grapes are made into wine. <strong>&lt;small>Alcohol is addictive.&lt;/small></strong></code></pre>

    <tr>
     <td><code>s</code>
     <td>Inaccurate text
     <td><pre class="example"><code class="html">Price: <strong>&lt;s>&pound;4.50&lt;/s></strong> &pound;2.00!</code></pre>

    <tr>
     <td><code>cite</code>
     <td>Titles of works
     <td><pre class="example"><code class="html">The case <strong>&lt;cite>Hugo v. Danielle&lt;/cite></strong> is relevant here.</code></pre>

    <tr>
     <td><code>q</code>
     <td>Quotations
     <td><pre class="example"><code class="html">The judge said <strong>&lt;q>You can drink water from the fish tank&lt;/q></strong> but advised against it.</code></pre>

    <tr>
     <td><code>dfn</code>
     <td>Defining instance
     <td><pre class="example"><code class="html">The term <strong>&lt;dfn>organic food&lt;/dfn></strong> refers to food produced without synthetic chemicals.</code></pre>

    <tr>
     <td><code>abbr</code>
     <td>Abbreviations
     <td><pre class="example"><code class="html">Organic food in Ireland is certified by the <strong>&lt;abbr title="Irish Organic Farmers and Growers Association">IOFGA&lt;/abbr></strong>.</code></pre>

    <tr>
     <td><code>ruby</code>, <code>rt</code>, <code>rp</code>
     <td>Ruby annotations
     <td><pre class="example"><code class="html"><strong>&lt;ruby> OJ &lt;rp>(&lt;rt>Orange Juice&lt;rp>)&lt;/ruby></strong></code></pre>

    <tr>
     <td><code>data</code>
     <td>Machine-readable equivalent
     <td><pre class="example"><code class="html">Available starting today! <strong>&lt;data value="UPC:022014640201">North Coast Organic Apple Cider&lt;/data></strong></code></pre>

    <tr>
     <td><code>time</code>
     <td>Machine-readable equivalent of date- or time-related data
     <td><pre class="example"><code class="html">Available starting on <strong>&lt;time datetime="2011-11-18">November 18th&lt;/time></strong>!</code></pre>

    <tr>
     <td><code>code</code>
     <td>Computer code
     <td><pre class="example"><code class="html">The <strong>&lt;code>fruitdb&lt;/code></strong> program can be used for tracking fruit production.</code></pre>

    <tr>
     <td><code>var</code>
     <td>Variables
     <td><pre class="example"><code class="html">If there are <strong>&lt;var>n&lt;/var></strong> fruit in the bowl, at least <strong>&lt;var>n&lt;/var></strong>&#xf7;2 will be ripe.</code></pre>

    <tr>
     <td><code>samp</code>
     <td>Computer output
     <td><pre class="example"><code class="html">The computer said <strong>&lt;samp>Unknown error -3&lt;/samp></strong>.</code></pre>

    <tr>
     <td><code>kbd</code>
     <td>User input
     <td><pre class="example"><code class="html">Hit <strong>&lt;kbd>F1&lt;/kbd></strong> to continue.</code></pre>

    <tr>
     <td><code>sub</code>
     <td>Subscripts
     <td><pre class="example"><code class="html">Water is H<strong>&lt;sub>2&lt;/sub></strong>O.</code></pre>

    <tr>
     <td><code>sup</code>
     <td>Superscripts
     <td><pre class="example"><code class="html">The Hydrogen in heavy water is usually <strong>&lt;sup>2&lt;/sup></strong>H.</code></pre>

    <tr>
     <td><code>i</code>
     <td>Alternative voice
     <td><pre class="example"><code class="html">Lemonade consists primarily of <strong>&lt;i>Citrus limon&lt;/i></strong>.</code></pre>

    <tr>
     <td><code>b</code>
     <td>Keywords
     <td><pre class="example"><code class="html">Take a <strong>&lt;b>lemon&lt;/b></strong> and squeeze it with a <strong>&lt;b>juicer&lt;/b></strong>.</code></pre>

    <tr>
     <td><code>u</code>
     <td>Annotations
     <td><pre class="example"><code class="html">The mixture of apple juice and <strong>&lt;u class="spelling">eldeflower&lt;/u&gt;</strong> juice is very pleasant.</code></pre>

    <tr>
     <td><code>mark</code>
     <td>Highlight
     <td><pre class="example"><code class="html">Elderflower cordial, with one <strong>&lt;mark>part&lt;/mark></strong> cordial to ten <strong>&lt;mark>part&lt;/mark></strong>s water, stands a<strong>&lt;mark>part&lt;/mark></strong> from the rest.</code></pre>

    <tr>
     <td><code>bdi</code>
     <td>Text directionality isolation
     <td><pre class="example"><code class="html">The <!--non-normative-->recommended restaurant is <strong>&lt;bdi lang="">My Juice Caf&eacute; (At The Beach)&lt;/bdi></strong>.</code></pre>

    <tr>
     <td><code>bdo</code>
     <td>Text directionality formatting
     <td><pre class="example"><code class="html">The proposal is to write English, but in reverse order. "Juice" would become "<strong>&lt;bdo dir=rtl>Juice&lt;/bdo></strong>"></code></pre>

    <tr>
     <td><code>span</code>
     <td>Other
     <td><pre class="example"><code class="html">In French we call it <strong>&lt;span lang="fr">sirop de sureau&lt;/span></strong>.</code></pre>

    <tr>
     <td><code>br</code>
     <td>Line break
     <td><pre class="example"><code class="html">Simply Orange Juice Company<strong>&lt;br></strong>Apopka, FL 32703<strong>&lt;br></strong>U.S.A.</code></pre>

    <tr>
     <td><code>wbr</code>
     <td>Line breaking opportunity
     <td><pre class="example"><code class="html">www.simply<strong>&lt;wbr></strong>orange<strong>&lt;wbr></strong>juice.com</code></pre>

  </table>



  <h3 split-filename="links" id="links">Links</h3>

  <h4>Introduction</h4>

  <p>Links are a conceptual construct, created by <code>a</code>, <code>area</code>,
  <code>form</code>, and <code>link</code> elements, that <span data-x="represents">represent</span>
  a connection between two resources, one of which is the current <code>Document</code>. There are
  three kinds of links in HTML:</p>

  <dl>
   <dt><dfn data-x="external resource link" data-lt="external resource link" export>Links to
   external resources</dfn></dt>

   <dd><p>These are links to resources that are to be used to augment the current document,
   generally automatically processed by the user agent. All <span data-x="external resource
   link">external resource links</span> have a <span>fetch and process the linked resource</span>
   algorithm which describes how the resource is obtained.</p></dd>

   <dt><dfn data-x="hyperlink" data-lt="hyperlink" export>Hyperlinks</dfn></dt>

   <dd><p>These are links to other resources that are generally exposed to the user by the user
   agent so that the user can cause the user agent to <span>navigate</span> to those resources, e.g.
   to visit them in a browser or download them.</p></dd>

   <dt><dfn data-x="Internal resource link" export>Internal resource links</dfn></dt>

   <dd><p>These are links to resources within the current document, used to give those resources
   special meaning or behavior.</p></dd>
  </dl>

  <p>For <code>link</code> elements with an <code data-x="attr-link-href">href</code> attribute and a
  <code data-x="attr-link-rel">rel</code> attribute, links must be created for the keywords of the
  <code data-x="attr-link-rel">rel</code> attribute, as defined for those keywords in the <a
  href="#linkTypes">link types</a> section.</p>

  <p>Similarly, for <code>a</code> and <code>area</code> elements with an <code
  data-x="attr-hyperlink-href">href</code> attribute and a <code
  data-x="attr-hyperlink-rel">rel</code> attribute, links must be created for the keywords of the
  <code data-x="attr-hyperlink-rel">rel</code> attribute as defined for those keywords in the <a
  href="#linkTypes">link types</a> section. Unlike <code>link</code> elements, however,
  <code>a</code> and <code>area</code> elements with an <code data-x="attr-hyperlink-href">href</code>
  attribute that either do not have a <code data-x="attr-hyperlink-rel">rel</code> attribute, or
  whose <code data-x="attr-hyperlink-rel">rel</code> attribute has no keywords that are defined as
  specifying <span data-x="hyperlink">hyperlinks</span>, must also create a <span>hyperlink</span>.
  This implied hyperlink has no special meaning (it has no <a href="#linkTypes">link type</a>)
  beyond linking the element's <span>node document</span> to the resource given by the element's <code
  data-x="attr-hyperlink-href">href</code> attribute.</p>

  <p>Similarly, for <code>form</code> elements with a <code data-x="attr-form-rel">rel</code>
  attribute, links must be created for the keywords of the <code data-x="attr-form-rel">rel</code>
  attribute as defined for those keywords in the <a href="#linkTypes">link types</a> section.
  <code>form</code> elements that do not have a <code data-x="attr-form-rel">rel</code> attribute,
  or whose <code data-x="attr-form-rel">rel</code> attribute has no keywords that are defined as
  specifying <span data-x="hyperlink">hyperlinks</span>, must also create a <span>hyperlink</span>.

  <p>A <span>hyperlink</span> can have one or more <dfn data-x="hyperlink annotation">hyperlink
  annotations</dfn> that modify the processing semantics of that hyperlink.</p>


  <h4>Links created by <code>a</code> and <code>area</code> elements</h4>

  <p>The <dfn element-attr for="a,area"><code data-x="attr-hyperlink-href">href</code></dfn>
  attribute on <code>a</code> and <code>area</code> elements must have a value that is a <span>valid
  URL potentially surrounded by spaces</span>.</p>

  <p class="note">The <code data-x="attr-hyperlink-href">href</code> attribute on <code>a</code> and
  <code>area</code> elements is not required; when those elements do not have <code
  data-x="attr-hyperlink-href">href</code> attributes they do not create hyperlinks.</p>

  <p>The <dfn element-attr for="a,area"><code data-x="attr-hyperlink-target">target</code></dfn>
  attribute, if present, must be a <span>valid navigable target name or keyword</span>. It gives the
  name of the <span>navigable</span> that will be used. <span w-nodev>User agents use this
  name when <span>following hyperlinks</span>.</span></p>

  <p>The <dfn element-attr for="a,area"><code data-x="attr-hyperlink-download">download</code></dfn>
  attribute, if present, indicates that the author intends the hyperlink to be used for <span
  data-x="downloading hyperlinks">downloading a resource</span>. The attribute may have a value; the
  value, if any, specifies the default filename that the author recommends for use in labeling the
  resource in a local file system. There are no restrictions on allowed values, but authors are
  cautioned that most file systems have limitations with regard to what punctuation is supported in
  filenames, and user agents are likely to adjust filenames accordingly.</p>

  <p>The <dfn id="ping" element-attr for="a,area"
  data-x="attr-hyperlink-ping"><code>ping</code></dfn> attribute, if present, gives the URLs of the
  resources that are interested in being notified if the user follows the hyperlink. The value must
  be a <span>set of space-separated tokens</span>, each of which must be a <span>valid non-empty
  URL</span> whose <span data-x="concept-url-scheme">scheme</span> is an <span>HTTP(S)
  scheme</span>. <span w-nodev>The value is used by the user agent for <span>hyperlink
  auditing</span>.</span></p>

  <p><span w-dev subdfn data-x="dom-a-rellist"></span>The <dfn element-attr for="a,area"><code
  data-x="attr-hyperlink-rel">rel</code></dfn> attribute on <code>a</code> and <code>area</code>
  elements controls what kinds of links the elements create. The attribute's value must be an
  <span>unordered set of unique space-separated tokens</span>. The <a href="#linkTypes">allowed
  keywords and their meanings</a> are defined below.</p>

  <p><code data-x="attr-hyperlink-rel">rel</code>'s <span
  data-x="concept-supported-tokens">supported tokens</span> are the keywords defined in <a
  href="#linkTypes">HTML link types</a> which are allowed on <code>a</code> and <code>area</code>
  elements, impact the processing model, and are supported by the user agent. The possible <span
  data-x="concept-supported-tokens">supported tokens</span> are <code
  data-x="rel-noreferrer">noreferrer</code>, <code data-x="rel-noopener">noopener</code>, and <code
  data-x="rel-opener">opener</code>. <code data-x="attr-hyperlink-rel">rel</code>'s <span
  data-x="concept-supported-tokens">supported tokens</span> must only include the tokens from this
  list that the user agent implements the processing model for.</p>

  <p>The <code data-x="attr-hyperlink-rel">rel</code> attribute has no default value. If the
  attribute is omitted or if none of the values in the attribute are recognized by the user agent,
  then the document has no particular relationship with the destination resource other than there
  being a hyperlink between the two.</p>

  <p>The <dfn element-attr for="a,area"><code data-x="attr-hyperlink-hreflang">hreflang</code></dfn>
  attribute on <code>a</code> elements that create <span data-x="hyperlink">hyperlinks</span>, if
  present, gives the language of the linked resource. It is purely advisory. The value must be a
  valid BCP 47 language tag. <ref>BCP47</ref> <span w-nodev>User agents must not consider this
  attribute authoritative &mdash; upon fetching the resource, user agents must use only language
  information associated with the resource to determine its language, not metadata included in the
  link to the resource.</span></p>

  <p>The <dfn element-attr for="a,area"><code data-x="attr-hyperlink-type">type</code></dfn>
  attribute, if present, gives the <span>MIME type</span> of the linked resource. It is purely
  advisory. The value must be a <span>valid MIME type string</span>. <span w-nodev>User agents must
  not consider the <code data-x="attr-hyperlink-type">type</code> attribute authoritative &mdash;
  upon fetching the resource, user agents must not use metadata included in the link to the resource
  to determine its type.</span></p>

  <p>The <dfn element-attr for="a,area"><code
  data-x="attr-hyperlink-referrerpolicy">referrerpolicy</code></dfn> attribute is a <span>referrer
  policy attribute</span>. Its purpose is to set the <span>referrer policy</span> used when
  <span>following hyperlinks</span>. <ref>REFERRERPOLICY</ref></p>

  <hr>

  <p>When an <code>a</code> or <code>area</code> element's <span>activation behavior</span> is
  invoked, the user agent may allow the user to indicate a preference regarding whether the
  hyperlink is to be used for <span data-x="navigate">navigation</span> or whether the resource it
  specifies is to be downloaded.</p>

  <p>In the absence of a user preference, the default should be navigation if the element has no
  <code data-x="attr-hyperlink-download">download</code> attribute, and should be to download the
  specified resource if it does.</p>

  <p>The <span>activation behavior</span> of an <code>a</code> or <code>area</code> element
  <var>element</var> given an event <var>event</var> is:</p>

  <ol>
   <li><p>If <var>element</var> has no <code data-x="attr-hyperlink-href">href</code> attribute,
   then return.</p></li>

   <li><p>Let <var>hyperlinkSuffix</var> be null.</p></li>

   <li><p>If <var>element</var> is an <code>a</code> element, and <var>event</var>'s <span
   data-x="concept-event-target">target</span> is an <code>img</code> with an <code
   data-x="attr-img-ismap">ismap</code> attribute specified, then:</p>

    <ol>
     <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?%3C%21DOCTYPE%20html%3E%0A...%3Ca%20href%3D%22%23%22%3E%3Cimg%20ismap%20usemap%3D%22%23a%22%20src%3D/resources/images/smallcats%3E%3C/a%3E%0A%3Cmap%20name%3Da%3E%3Carea%20shape%3Drect%20coords%3D0%2C0%2C50%2C50%20href%3Db%3E%3C/map%3E -->

     <li><p>Let <var>x</var> and <var>y</var> be 0.</p></li>

     <li><p>If <var>event</var>'s <code data-x="dom-Event-isTrusted">isTrusted</code> attribute is
     initialized to true, then set <var>x</var> to the distance in <span data-x="'px'">CSS
     pixels</span> from the left edge of the image to the location of the click, and set
     <var>y</var> to the distance in <span data-x="'px'">CSS pixels</span> from the top edge of the
     image to the location of the click.</p></li>

     <li><p>If <var>x</var> is negative, set <var>x</var> to 0.</p></li>

     <li><p>If <var>y</var> is negative, set <var>y</var> to 0.</p></li>

     <li><p>Set <var>hyperlinkSuffix</var> to the concatenation of U+003F (?), the value of
     <var>x</var> expressed as a base-ten integer using <span>ASCII digits</span>, U+002C (,), and
     the value of <var>y</var> expressed as a base-ten integer using <span>ASCII
     digits</span>.</p></li>
    </ol>
   </li>

   <li><p>Let <var>userInvolvement</var> be <var>event</var>'s <span data-x="event-uni">user
   navigation involvement</span>.</p></li>

   <li>
    <p>If the user has expressed a preference to download the hyperlink, then set
    <var>userInvolvement</var> to "<code data-x="uni-browser-ui">browser UI</code>".</p>

    <p class="note">That is, if the user has expressed a specific preference for downloading, this
    no longer counts as merely "<code data-x="uni-activation">activation</code>".</p>
   </li>

   <li><p>If <var>element</var> has a <code data-x="attr-hyperlink-download">download</code>
   attribute, or if the user has expressed a preference to download the hyperlink, then <span
   data-x="downloading hyperlinks">download the hyperlink</span> created by <var>element</var> with
   <i data-x="downloading-hyperlinkSuffix">hyperlinkSuffix</i> set to <var>hyperlinkSuffix</var> and
   <i data-x="downloading-userInvolvement">userInvolvement</i> set to
   <var>userInvolvement</var>.</p></li>

   <li><p>Otherwise, <span data-x="following hyperlinks">follow the hyperlink</span> created by
   <var>element</var> with <i data-x="following-hyperlinkSuffix">hyperlinkSuffix</i> set to
   <var>hyperlinkSuffix</var> and <i data-x="following-userInvolvement">userInvolvement</i> set to
   <var>userInvolvement</var>.</p></li>
  </ol>


  <h4>API for <code>a</code> and <code>area</code> elements</h4>

  <pre><code class="idl">interface mixin <dfn interface>HTMLHyperlinkElementUtils</dfn> {
  [<span>CEReactions</span>] stringifier attribute USVString <span data-x="dom-hyperlink-href">href</span>;
  readonly attribute USVString <span data-x="dom-hyperlink-origin">origin</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-hyperlink-protocol">protocol</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-hyperlink-username">username</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-hyperlink-password">password</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-hyperlink-host">host</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-hyperlink-hostname">hostname</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-hyperlink-port">port</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-hyperlink-pathname">pathname</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-hyperlink-search">search</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-hyperlink-hash">hash</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>hyperlink</var>.<span data-x="">toString()</span></code></dt>
   <dt><code data-x=""><var>hyperlink</var>.<span subdfn data-x="dom-hyperlink-href">href</span></code></dt>
   <dd>
    <p>Returns the hyperlink's URL.</p>

    <p>Can be set, to change the URL.</p>
   </dd>

   <dt><code data-x=""><var>hyperlink</var>.<span subdfn data-x="dom-hyperlink-origin">origin</span></code></dt>
   <dd><p>Returns the hyperlink's URL's origin.</p></dd>

   <dt><code data-x=""><var>hyperlink</var>.<span subdfn data-x="dom-hyperlink-protocol">protocol</span></code></dt>
   <dd>
    <p>Returns the hyperlink's URL's scheme.</p>

    <p>Can be set, to change the URL's scheme.</p>
   </dd>

   <dt><code data-x=""><var>hyperlink</var>.<span subdfn data-x="dom-hyperlink-username">username</span></code></dt>
   <dd>
    <p>Returns the hyperlink's URL's username.</p>

    <p>Can be set, to change the URL's username.</p>
   </dd>
   <!-- This feature is technically non-conforming -->

   <dt><code data-x=""><var>hyperlink</var>.<span subdfn data-x="dom-hyperlink-password">password</span></code></dt>
   <dd>
    <p>Returns the hyperlink's URL's password.</p>

    <p>Can be set, to change the URL's password.</p>
   </dd>
   <!-- This feature is technically non-conforming -->

   <dt><code data-x=""><var>hyperlink</var>.<span subdfn data-x="dom-hyperlink-host">host</span></code></dt>
   <dd>
    <p>Returns the hyperlink's URL's host and port (if different from the default port for the
    scheme).</p>

    <p>Can be set, to change the URL's host and port.</p>
   </dd>

   <dt><code data-x=""><var>hyperlink</var>.<span subdfn data-x="dom-hyperlink-hostname">hostname</span></code></dt>
   <dd>
    <p>Returns the hyperlink's URL's host.</p>

    <p>Can be set, to change the URL's host.</p>
   </dd>

   <dt><code data-x=""><var>hyperlink</var>.<span subdfn data-x="dom-hyperlink-port">port</span></code></dt>
   <dd>
    <p>Returns the hyperlink's URL's port.</p>

    <p>Can be set, to change the URL's port.</p>
   </dd>

   <dt><code data-x=""><var>hyperlink</var>.<span subdfn data-x="dom-hyperlink-pathname">pathname</span></code></dt>
   <dd>
    <p>Returns the hyperlink's URL's path.</p>

    <p>Can be set, to change the URL's path.</p>
   </dd>

   <dt><code data-x=""><var>hyperlink</var>.<span subdfn data-x="dom-hyperlink-search">search</span></code></dt>
   <dd>
    <p>Returns the hyperlink's URL's query (includes leading "<code data-x="">?</code>" if
    non-empty).</p>

    <p>Can be set, to change the URL's query (ignores leading "<code data-x="">?</code>").</p>
   </dd>

   <dt><code data-x=""><var>hyperlink</var>.<span subdfn data-x="dom-hyperlink-hash">hash</span></code></dt>
   <dd>
    <p>Returns the hyperlink's URL's fragment (includes leading "<code data-x="">#</code>" if
    non-empty).</p>

    <p>Can be set, to change the URL's fragment (ignores leading "<code data-x="">#</code>").</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>An element implementing the <code>HTMLHyperlinkElementUtils</code> mixin has an associated <dfn
  data-x="concept-hyperlink-url">url</dfn> (null or a <span>URL</span>). It is initially null.

  <p>An element implementing the <code>HTMLHyperlinkElementUtils</code> mixin has an associated <dfn
  data-x="concept-hyperlink-url-set">set the url</dfn> algorithm, which runs these steps:</p>

  <ol>
   <li><p>Set this element's <span data-x="concept-hyperlink-url">url</span> to null.</p></li>

   <li><p>If this element's <code data-x="attr-hyperlink-href">href</code> content attribute is
   absent, then return.</p></li>

   <li><p>Let <var>url</var> be the result of <span>encoding-parsing a URL</span> given this
   element's <code data-x="attr-hyperlink-href">href</code> content attribute's value, relative to
   this element's <span>node document</span>.</p></li>

   <li><p>If <var>url</var> is not failure, then set this element's <span
   data-x="concept-hyperlink-url">url</span> to <var>url</var>.</p></li>
  </ol>

  <p>When elements implementing the <code>HTMLHyperlinkElementUtils</code> mixin are created, and
  whenever those elements have their <code data-x="attr-hyperlink-href">href</code> content
  attribute set, changed, or removed, the user agent must <span
  data-x="concept-hyperlink-url-set">set the url</span>.</p>

  <p class="note">This is only observable for <code data-x="blob protocol">blob:</code> URLs as
  <span data-x="url parser">parsing</span> them involves a <span>Blob URL Store</span> lookup.</p>

  <p>An element implementing the <code>HTMLHyperlinkElementUtils</code> mixin has an associated
  <!--en-GB--><dfn id="reinitialise-url">reinitialize url</dfn> algorithm, which runs these
  steps:</p>

  <ol>
   <li><p>If the element's <span data-x="concept-hyperlink-url">url</span> is non-null, its <span
   data-x="concept-url-scheme">scheme</span> is "<code data-x="">blob</code>", and it has an
   <span>opaque path</span>, then terminate these steps.</p></li>

   <li><p><span data-x="concept-hyperlink-url-set">Set the url</span>.</p></li>
  </ol>

  <p>To <dfn>update <code>href</code></dfn>, set the element's <code
  data-x="attr-hyperlink-href">href</code> content attribute's value to the element's <span
  data-x="concept-hyperlink-url">url</span>, <span
  data-x="concept-url-serializer">serialized</span>.</p>

  <hr>

  <p>The <dfn attribute for="HTMLHyperlinkElementUtils"><code
  data-x="dom-hyperlink-href">href</code></dfn> getter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-hyperlink-url">url</span>.</p></li>

   <li><p>If <var>url</var> is null and <span>this</span> has no <code
   data-x="attr-hyperlink-href">href</code> content attribute, return the empty string.

   <li><p>Otherwise, if <var>url</var> is null, return <span>this</span>'s <code
   data-x="attr-hyperlink-href">href</code> content attribute's value.</p></li>

   <li><p>Return <var>url</var>, <span data-x="concept-url-serializer">serialized</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-hyperlink-href">href</code> setter steps are to set <span>this</span>'s
  <code data-x="attr-hyperlink-href">href</code> content attribute's value to the given value.

  <p>The <dfn attribute for="HTMLHyperlinkElementUtils"><code
  data-x="dom-hyperlink-origin">origin</code></dfn> getter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>If <span>this</span>'s <span data-x="concept-hyperlink-url">url</span> is null, return the
   empty string.</p></li>

   <li><p>Return the <span data-x="serialization of an origin">serialization</span> of
   <span>this</span>'s <span data-x="concept-hyperlink-url">url</span>'s <span
   data-x="concept-url-origin">origin</span>.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLHyperlinkElementUtils"><code
  data-x="dom-hyperlink-protocol">protocol</code></dfn> getter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>If <span>this</span>'s <span data-x="concept-hyperlink-url">url</span> is null, return
   "<code data-x="">:</code>".</p></li>

   <li><p>Return <span>this</span>'s <span data-x="concept-hyperlink-url">url</span>'s <span
   data-x="concept-url-scheme">scheme</span>, followed by "<code data-x="">:</code>".</p></li>
  </ol>

  <p>The <code data-x="dom-hyperlink-protocol">protocol</code> setter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>If <span>this</span>'s <span data-x="concept-hyperlink-url">url</span> is null, then
   return.</p></li>

   <li>
    <p><span data-x="basic url parser">Basic URL parse</span> the given value, followed by "<code
    data-x="">:</code>", with <span>this</span>'s <span data-x="concept-hyperlink-url">url</span> as
    <span data-x="basic url parser url"><i>url</i></span> and <span>scheme start state</span> as
    <span data-x="basic url parser state override"><i>state override</i></span>.</p>

    <p class="note">Because the URL parser ignores multiple consecutive colons, providing a value
    of "<code data-x="">https:</code>" (or even "<code data-x="">https::::</code>") is the same as
    providing a value of "<code data-x="">https</code>".</p>
   </li>

   <li><p><span>Update <code>href</code></span>.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLHyperlinkElementUtils"><code
  data-x="dom-hyperlink-username">username</code></dfn> getter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>If <span>this</span>'s <span data-x="concept-hyperlink-url">url</span> is null, return the
   empty string.</p></li>

   <li><p>Return <span>this</span>'s <span data-x="concept-hyperlink-url">url</span>'s <span
   data-x="concept-url-username">username</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-hyperlink-username">username</code> setter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-hyperlink-url">url</span>.</p></li>

   <li><p>If <var>url</var> is null or <var>url</var>
   <span>cannot have a username/password/port</span>, then return.</p></li>

   <li><p><span>Set the username</span>, given <var>url</var> and the given value.</p></li>

   <li><p><span>Update <code>href</code></span>.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLHyperlinkElementUtils"><code
  data-x="dom-hyperlink-password">password</code></dfn> getter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-hyperlink-url">url</span>.</p></li>

   <li><p>If <var>url</var> is null, then return the empty string.</p></li>

   <li><p>Return <var>url</var>'s <span data-x="concept-url-password">password</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-hyperlink-password">password</code> setter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-hyperlink-url">url</span>.</p></li>

   <li><p>If <var>url</var> is null or <var>url</var>
   <span>cannot have a username/password/port</span>, then return.</p></li>

   <li><p><span>Set the password</span>, given <var>url</var> and the given value.</p></li>

   <li><p><span>Update <code>href</code></span>.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLHyperlinkElementUtils"><code
  data-x="dom-hyperlink-host">host</code></dfn> getter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-hyperlink-url">url</span>.</p></li>

   <li><p>If <var>url</var> or <var>url</var>'s <span data-x="concept-url-host">host</span> is null,
   return the empty string.</p></li>

   <li><p>If <var>url</var>'s <span data-x="concept-url-port">port</span> is null, return
   <var>url</var>'s <span data-x="concept-url-host">host</span>, <span data-x="host
   serializer">serialized</span>.</p></li>

   <li><p>Return <var>url</var>'s <span data-x="concept-url-host">host</span>, <span data-x="host
   serializer">serialized</span>, followed by "<code data-x="">:</code>" and <var>url</var>'s <span
   data-x="concept-url-port">port</span>, <span data-x="serialize an
   integer">serialized</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-hyperlink-host">host</code> setter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-hyperlink-url">url</span>.</p></li>

   <li><p>If <var>url</var> is null or <var>url</var> has an <span>opaque path</span>, then
   return.</p></li>

   <li><p><span data-x="basic url parser">Basic URL parse</span> the given value, with
   <var>url</var> as <span data-x="basic url parser url"><i>url</i></span> and <span>host
   state</span> as <span data-x="basic url parser state override"><i>state
   override</i></span>.</p></li>

   <li><p><span>Update <code>href</code></span>.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLHyperlinkElementUtils"><code
  data-x="dom-hyperlink-hostname">hostname</code></dfn> getter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-hyperlink-url">url</span>.</p></li>

   <li><p>If <var>url</var> or <var>url</var>'s <span data-x="concept-url-host">host</span> is null,
   return the empty string.</p></li>

   <li><p>Return <var>url</var>'s <span data-x="concept-url-host">host</span>, <span data-x="host
   serializer">serialized</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-hyperlink-hostname">hostname</code> setter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-hyperlink-url">url</span>.</p></li>

   <li><p>If <var>url</var> is null or <var>url</var> has an <span>opaque path</span>, then
   return.</p></li>

   <li><p><span data-x="basic url parser">Basic URL parse</span> the given value, with
   <var>url</var> as <span data-x="basic url parser url"><i>url</i></span> and <span>hostname
   state</span> as <span data-x="basic url parser state override"><i>state
   override</i></span>.</p></li>

   <li><p><span>Update <code>href</code></span>.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLHyperlinkElementUtils"><code
  data-x="dom-hyperlink-port">port</code></dfn> getter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-hyperlink-url">url</span>.</p></li>

   <li><p>If <var>url</var> or <var>url</var>'s <span data-x="concept-url-port">port</span> is null,
   return the empty string.</p></li>

   <li><p>Return <var>url</var>'s <span data-x="concept-url-port">port</span>, <span
   data-x="serialize an integer">serialized</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-hyperlink-port">port</code> setter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-hyperlink-url">url</span>.</p></li>

   <li><p>If <var>url</var> is null or <var>url</var>
   <span>cannot have a username/password/port</span>, then return.</p></li>

   <li><p>If the given value is the empty string, then set <var>url</var>'s <span
   data-x="concept-url-port">port</span> to null.</p></li>

   <li><p>Otherwise, <span data-x="basic url parser">basic URL parse</span> the given value, with
   <var>url</var> as <span data-x="basic url parser url"><i>url</i></span> and <span>port
   state</span> as <span data-x="basic url parser state override"><i>state
   override</i></span>.</p></li>

   <li><p><span>Update <code>href</code></span>.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLHyperlinkElementUtils"><code
  data-x="dom-hyperlink-pathname">pathname</code></dfn> getter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-hyperlink-url">url</span>.</p></li>

   <li><p>If <var>url</var> is null, then return the empty string.</p></li>

   <li><p>Return the result of <span data-x="URL path serializer">URL path serializing</span>
   <var>url</var>.</p></li>
  </ol>

  <p>The <code data-x="dom-hyperlink-pathname">pathname</code> setter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-hyperlink-url">url</span>.</p></li>

   <li><p>If <var>url</var> is null or <var>url</var> has an <span>opaque path</span>, then
   return.</p></li>

   <li><p>Set <var>url</var>'s <span data-x="concept-url-path">path</span> to the empty
   list.</p></li>

   <li><p><span data-x="basic url parser">Basic URL parse</span> the given value, with
   <var>url</var> as <span data-x="basic url parser url"><i>url</i></span> and <span>path start
   state</span> as <span data-x="basic url parser state override"><i>state
   override</i></span>.</p></li>

   <li><p><span>Update <code>href</code></span>.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLHyperlinkElementUtils"><code
  data-x="dom-hyperlink-search">search</code></dfn> getter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-hyperlink-url">url</span>.</p></li>

   <li><p>If <var>url</var> is null, or <var>url</var>'s <span
   data-x="concept-url-query">query</span> is either null or the empty string, return the empty
   string.</p></li>

   <li><p>Return "<code data-x="">?</code>", followed by <var>url</var>'s <span
   data-x="concept-url-query">query</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-hyperlink-search">search</code> setter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-hyperlink-url">url</span>.</p></li>

   <li><p>If <var>url</var> is null, terminate these steps.</p></li>

   <li><p>If the given value is the empty string, set <var>url</var>'s <span
   data-x="concept-url-query">query</span> to null.

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p>Let <var>input</var> be the given value with a single leading "<code data-x="">?</code>"
     removed, if any.</p></li>

     <li><p>Set <var>url</var>'s <span data-x="concept-url-query">query</span> to the empty
     string.</p></li>

     <li><p><span data-x="basic url parser">Basic URL parse</span> <var>input</var>, with
     <var>url</var> as <span data-x="basic url parser url"><i>url</i></span> and <span>query
     state</span> as <span data-x="basic url parser state override"><i>state
     override</i></span>.</p></li>
    </ol>
   </li>

   <li><p><span>Update <code>href</code></span>.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLHyperlinkElementUtils"><code
  data-x="dom-hyperlink-hash">hash</code></dfn> getter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-hyperlink-url">url</span>.</p></li>

   <li><p>If <var>url</var> is null, or <var>url</var>'s <span
   data-x="concept-url-fragment">fragment</span> is either null or the empty string, return the
   empty string.</p></li>

   <li><p>Return "<code data-x="">#</code>", followed by <var>url</var>'s <span
   data-x="concept-url-fragment">fragment</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-hyperlink-hash">hash</code> setter steps are:</p>

  <ol>
   <li><p><span>Reinitialize url</span>.</p></li>

   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-hyperlink-url">url</span>.</p></li>

   <li><p>If <var>url</var> is null, then return.</p></li>

   <li><p>If the given value is the empty string, set <var>url</var>'s <span
   data-x="concept-url-fragment">fragment</span> to null.

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p>Let <var>input</var> be the given value with a single leading "<code data-x="">#</code>"
     removed, if any.</p></li>

     <li><p>Set <var>url</var>'s <span data-x="concept-url-fragment">fragment</span> to the empty
     string.</p></li>

     <li><p><span data-x="basic url parser">Basic URL parse</span> <var>input</var>, with
     <var>url</var> as <span data-x="basic url parser url"><i>url</i></span> and <span>fragment
     state</span> as <span data-x="basic url parser state override"><i>state
     override</i></span>.</p></li>
    </ol>
   </li>

   <li><p><span>Update <code>href</code></span>.</p></li>
  </ol>


  <h4>Following hyperlinks</h4>

  <p>An element <var>element</var> <dfn>cannot navigate</dfn> if any of the following are true:</p>

  <ul>
   <li><p><var>element</var>'s <span>node document</span> is not <span>fully active</span>;
   or</p></li>

   <li><p><var>element</var> is not an <code>a</code> element and is not
   <span>connected</span>.</p></li>
  </ul>

  <p class="note">This is also used by <span data-x="concept-form-submit">form submission</span> for
  the <code>form</code> element. The exception for <code>a</code> elements is for compatibility with
  web content.</p>

  <p>To <dfn>get an element's noopener</dfn>, given an <code>a</code>, <code>area</code>, or
  <code>form</code> element <var>element</var>, a <span>URL record</span> <var>url</var>, and a
  string <var>target</var>, perform the following steps. They return a boolean.</p>

  <ol>
   <li><p>If <var>element</var>'s <a href="#linkTypes">link types</a> include the <code
   data-x="rel-noopener">noopener</code> or <code data-x="rel-noreferrer">noreferrer</code>
   keyword, then return true.</li>

   <li id="opener-processing-model"><p>If <var>element</var>'s <a href="#linkTypes">link types</a>
   do not include the <code data-x="rel-opener">opener</code> keyword and <var>target</var> is an
   <span>ASCII case-insensitive</span> match for "<code data-x="">_blank</code>", then return
   true.</p></li>

   <li>
    <p>If <var>url</var>'s <span data-x="concept-url-blob-entry">blob URL entry</span> is not
    null:</p>

    <ol>
     <li><p>Let <var>blobOrigin</var> be <var>url</var>'s <span
     data-x="concept-url-blob-entry">blob URL entry</span>'s <span
     data-x="blob-url-entry-environment">environment</span>'s <span
     data-x="concept-settings-object-origin">origin</span>.</p></li>

     <li><p>Let <var>topLevelOrigin</var> be <var>element</var>'s <span>relevant settings
     object</span>'s <span>top-level origin</span>.</p></li>

     <li><p>If <var>blobOrigin</var> is not <span>same site</span> with <var>topLevelOrigin</var>,
     then return true.</p></li>
    </ol>
   </li>

   <li><p>Return false.</p></li>
  </ol>

  <p>To <dfn data-x="following hyperlinks">follow the hyperlink</dfn> created by an element
  <var>subject</var>, given an optional <dfn
  data-x="following-hyperlinkSuffix"><var>hyperlinkSuffix</var></dfn> (default null) and an
  optional <dfn data-x="following-userInvolvement"><var>userInvolvement</var></dfn> (default "<code
  data-x="uni-none">none</code>"):</p>

  <ol>
   <li><p>If <var>subject</var> <span>cannot navigate</span>, then return.</p></li>

   <li><p>Let <var>targetAttributeValue</var> be the empty string.</p></li>

   <li><p>If <var>subject</var> is an <code>a</code> or <code>area</code> element, then set
   <var>targetAttributeValue</var> to the result of <span data-x="get an element's target">getting
   an element's target</span> given <var>subject</var>.</p></li>

   <li><p>Let <var>urlRecord</var> be the result of <span>encoding-parsing a URL</span> given
   <var>subject</var>'s <code data-x="attr-hyperlink-href">href</code> attribute value, relative to
   <var>subject</var>'s <span>node document</span>.</p></li>

   <li><p>If <var>urlRecord</var> is failure, then return.</p></li>

   <li><p>Let <var>noopener</var> be the result of <span data-x="get an element's noopener">getting
   an element's noopener</span> with <var>subject</var>, <var>urlRecord</var>, and
   <var>targetAttributeValue</var>.</p></li>

   <li><p>Let <var>targetNavigable</var> be the first return value of applying <span>the rules for
   choosing a navigable</span> given <var>targetAttributeValue</var>, <var>subject</var>'s
   <span>node navigable</span>, and <var>noopener</var>.</p></li>

   <li><p>If <var>targetNavigable</var> is null, then return.</p></li>

   <li><p>Let <var>urlString</var> be the result of applying the <span
   data-x="concept-url-serializer">URL serializer</span> to <var>urlRecord</var>.</p></li>

   <li><p>If <var>hyperlinkSuffix</var> is non-null, then append it to
   <var>urlString</var>.</p></li>

   <li><p>Let <var>referrerPolicy</var> be the current state of <var>subject</var>'s <code
   data-x="">referrerpolicy</code> content attribute.</p></li>

   <li id="noreferrer-a-area-processing-model"><p>If <var>subject</var>'s <a href="#linkTypes">link
   types</a> includes the <code data-x="rel-noreferrer">noreferrer</code> keyword, then set
   <var>referrerPolicy</var> to "<code data-x="">no-referrer</code>".</p></li>

   <li>
    <p><span>Navigate</span><!--DONAV hyperlink--> <var>targetNavigable</var> to
    <var>urlString</var> using <var>subject</var>'s <span>node document</span>, with <i
    data-x="navigation-referrer-policy">referrerPolicy</i> set to <var>referrerPolicy</var>, <i
    data-x="navigation-user-involvement">userInvolvement</i> set to <var>userInvolvement</var>,
    and <i data-x="navigation-source-element">sourceElement</i> set to <var>subject</var>.</p>

    <p class="note">Unlike many other types of navigations, following hyperlinks does not have
    special "<code data-x="NavigationHistoryBehavior-replace">replace</code>" behavior for when
    documents are not <span>completely loaded</span>. This is true for both user-initiated instances
    of following hyperlinks, as well as script-triggered ones via, e.g., <code
    data-x="">aElement.click()</code>.</p>
   </li>
  </ol>

  </div>


<!--DOWNLOAD-->

  <h4>Downloading resources</h4>

  <p>In some cases, resources are intended for later use rather than immediate viewing. To indicate
  that a resource is intended to be downloaded for use later, rather than immediately used, the
  <code data-x="attr-hyperlink-download">download</code> attribute can be specified on the
  <code>a</code> or <code>area</code> element that creates the <span>hyperlink</span> to that
  resource.</p>

  <p>The attribute can furthermore be given a value, to specify the filename that user agents are
  to use when storing the resource in a file system. This value can be overridden by the `<code
  data-x="http-content-disposition">Content-Disposition</code>` HTTP header's filename parameters.
  <ref>RFC6266</ref></p>

  <p>In cross-origin situations, the <code data-x="attr-hyperlink-download">download</code>
  attribute has to be combined with the `<code
  data-x="http-content-disposition">Content-Disposition</code>` HTTP header, specifically with the
  <code data-x="">attachment</code> disposition type, to avoid the user being warned of possibly
  nefarious activity. (This is to protect users from being made to download sensitive personal or
  confidential information without their full understanding.)</p>

  <div w-nodev>

  <hr>

  <p>To <dfn data-x="downloading hyperlinks" export>download the hyperlink</dfn> created by an
  element <var>subject</var>, given an optional <dfn
  data-x="downloading-hyperlinkSuffix"><var>hyperlinkSuffix</var></dfn> (default null) and an
  optional <dfn data-x="downloading-userInvolvement"><var>userInvolvement</var></dfn> (default
  "<code data-x="uni-none">none</code>"):</p>

  <ol>
   <li><p>If <var>subject</var> <span>cannot navigate</span>, then return.</p></li>

   <li id="allowed-to-download"><p>If <var>subject</var>'s <span>node document</span>'s <span>active
   sandboxing flag set</span> has the <span>sandboxed downloads browsing context flag</span> set,
   then return.</p></li>

   <li><p>Let <var>urlString</var> be the result of <span>encoding-parsing-and-serializing a
   URL</span> given <var>subject</var>'s <code data-x="attr-hyperlink-href">href</code> attribute
   value, relative to <var>subject</var>'s <span>node document</span>.</p></li>

   <li><p>If <var>urlString</var> is failure, then return.</p></li>

   <li><p>If <var>hyperlinkSuffix</var> is non-null, then append it to
   <var>urlString</var>.</p></li>

   <li>
    <p>If <var>userInvolvement</var> is not "<code data-x="uni-browser-ui">browser UI</code>",
    then:</p>

    <ol>
     <li><p><span>Assert</span>: <var>subject</var> has a <code
     data-x="attr-hyperlink-download">download</code> attribute.</p></li>

     <li><p>Let <var>navigation</var> be <var>subject</var>'s <span>relevant global object</span>'s
     <span data-x="window-navigation-api">navigation API</span>.</p></li>

     <li><p>Let <var>filename</var> be the value of <var>subject</var>'s <code
     data-x="attr-hyperlink-download">download</code> attribute.</p></li>

     <li><p>Let <var>continue</var> be the result of <span data-x="fire a download request navigate
     event">firing a download request <code data-x="event-navigate">navigate</code> event</span> at
     <var>navigation</var> with <i data-x="fire-navigate-download-destinationURL">destinationURL</i>
     set to <var>urlString</var>, <i
     data-x="fire-navigate-download-userInvolvement">userInvolvement</i> set to
     <var>userInvolvement</var>, <i data-x="fire-navigate-download-sourceElement">sourceElement</i>
     set to <var>subject</var>, and <i data-x="fire-navigate-download-filename">filename</i> set to
     <var>filename</var>.</p></li>

     <li><p>If <var>continue</var> is false, then return.</p></li>
    </ol>
   </li>

   <li>
    <p>Run these steps <span>in parallel</span>:</p>

    <ol>
     <li><p>Optionally, the user agent may abort these steps, if it believes doing so would
     safeguard the user from a potentially hostile download.</p></li>

     <li><p>Let <var>request</var> be a new <span data-x="concept-request">request</span> whose
     <span data-x="concept-request-url">URL</span> is <var>urlString</var>, <span
     data-x="concept-request-client">client</span> is <span>entry settings object</span>, <span
     data-x="concept-request-initiator">initiator</span> is "<code data-x="">download</code>", <span
     data-x="concept-request-destination">destination</span> is the empty string, and whose
     <span>synchronous flag</span> and <span>use-URL-credentials flag</span> are set.</p></li>

     <!--FETCH--><li><p><span>Handle as a download</span> the result of <span
     data-x="concept-fetch">fetching</span> <var>request</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>To <dfn>handle as a download</dfn> a <span data-x="concept-response">response</span>
  <var>response</var>:</p>

  <ol>
   <li><p>Let <var>suggestedFilename</var> be the result of <span>getting the suggested
   filename</span> for <var>response</var>.</p></li>

   <li><p>Provide the user with a way to save <var>response</var> for later use. If the user agent
   needs a filename, it should use <var>suggestedFilename</var>. Report any problems downloading the
   file to the user.</p></li>

   <li><p>Return <var>suggestedFilename</var>.</p></li>
  </ol>

  <p>To <dfn data-x="getting the suggested filename">get the suggested filename</dfn> for a <span
  data-x="concept-response">response</span> <var>response</var>:</p>

  <p class="warning">This algorithm is intended to mitigate security dangers involved in downloading
  files from untrusted sites, and user agents are strongly urged to follow it.</p> <!-- but it's
  optional, since it's not really an interoperability issue -->

  <ol>
   <li><p>Let <var>filename</var> be the undefined value.</p></li>

   <!-- Content-Disposition: attachment; filename="" is always honoured, even cross-origin -->
   <li><p>If <var>response</var> has a `<code
   data-x="http-content-disposition">Content-Disposition</code>` header, that header specifies the
   <code data-x="">attachment</code> disposition type, and the header includes filename information,
   then let <var>filename</var> have the value specified by the header, and jump to the step labeled
   <i>sanitize</i> below. <ref>RFC6266</ref></p></li>

   <li><p>Let <var>interface origin</var> be the <span
   data-x="concept-document-origin">origin</span> of the <code>Document</code> in which the <span
   data-x="downloading hyperlinks">download</span> or <span>navigate</span> action resulting in the
   download was initiated, if any.</p></li>

   <li><p>Let <var>response origin</var> be the <span>origin</span> of the URL of
   <var>response</var>, unless that URL's <span data-x="concept-url-scheme">scheme</span> component
   is <code data-x="">data</code>, in which case let <var>response origin</var> be the same as the
   <var>interface origin</var>, if any.</p></li>

   <li><p>If there is no <var>interface origin</var>, then let <var>trusted operation</var> be true.
   Otherwise, let <var>trusted operation</var> be true if <var>response origin</var> is the
   <span>same origin</span> as <var>interface origin</var>, and false otherwise.</p></li>

   <!-- Content-Disposition: *; filename="" overrides download="" for same-origin -->
   <li><p>If <var>trusted operation</var> is true and <var>response</var> has a `<code
   data-x="http-content-disposition">Content-Disposition</code>` header and that header includes
   filename information, then let <var>filename</var> have the value specified by the header, and
   jump to the step labeled <i>sanitize</i> below. <ref>RFC6266</ref></p></li>

   <li><p>If the download was not initiated from a <span>hyperlink</span> created by an
   <code>a</code> or <code>area</code> element, or if the element of the <span>hyperlink</span> from
   which it was initiated did not have a <code data-x="attr-hyperlink-download">download</code>
   attribute when the download was initiated, or if there was such an attribute but its value when
   the download was initiated was the empty string, then jump to the step labeled <i>no proposed
   filename</i>.</p></li>

   <li><p>Let <var>proposed filename</var> have the value of the <code
   data-x="attr-hyperlink-download">download</code> attribute of the element of the
   <span>hyperlink</span> that initiated the download at the time the download was
   initiated.</p></li>

   <!-- download="" works for all links when same-origin -->
   <!-- it is not used cross-origin, though, even for explicit user gestures -->
   <!-- (except... see next step) -->
   <li><p>If <var>trusted operation</var> is true, let <var>filename</var> have
   the value of <var>proposed filename</var>, and jump to the step labeled <i>sanitize</i>
   below.</p></li>

   <!-- Content-Disposition: attachment with no filename="" allows download="" cross-origin -->
   <li><p>If <var>response</var> has a `<code
   data-x="http-content-disposition">Content-Disposition</code>` header and that header specifies
   the <code data-x="">attachment</code> disposition type, let <var>filename</var> have the value of
   <var>proposed filename</var>, and jump to the step labeled <i>sanitize</i> below.
   <ref>RFC6266</ref></p></li>

   <!-- fallback for same-origin resources or explicit downloads: use the response's filename -->
   <li><p><i>No proposed filename</i>: If <var>trusted operation</var> is true, or if the user
   indicated a preference for having the response in question downloaded, let <var>filename</var>
   have a value derived from the <span>URL</span> of <var>response</var> in an
   <span>implementation-defined</span> manner, and jump to the step labeled <i>sanitize</i>
   below.</p></li>

   <!-- no C-D: a header on cross-origin raises red flags -->
   <li>
    <p>Let <var>filename</var> be set to the user's preferred filename or to a filename selected by
    the user agent, and jump to the step labeled <i>sanitize</i> below.</p>

    <div class="warning">

     <p>If the algorithm reaches this step, then a download was begun from a different origin than
     <var>response</var>, and the origin did not mark the file as suitable for downloading, and the
     download was not initiated by the user. This could be because a <code
     data-x="attr-hyperlink-download">download</code> attribute was used to trigger the download, or
     because <var>response</var> is not of a type that the user agent supports.</p>

     <p>This could be dangerous, because, for instance, a hostile server could be trying to get a
     user to unknowingly download private information and then re-upload it to the hostile server,
     by tricking the user into thinking the data is from the hostile server.</p>

     <p>Thus, it is in the user's interests that the user be somehow notified that
     <var>response</var> comes from quite a different source, and to prevent confusion, any
     suggested filename from the potentially hostile <var>interface origin</var> should be
     ignored.</p>

    </div>
   </li>

   <li><p><i>Sanitize</i>: Optionally, allow the user to influence <var>filename</var>. For
   example, a user agent could prompt the user for a filename, potentially providing the value of
   <var>filename</var> as determined above as a default value.</p></li>

   <li>
    <p>Adjust <var>filename</var> to be suitable for the local file system.</p>

    <p class="example">For example, this could involve removing characters that are not legal in
    filenames, or trimming leading and trailing whitespace.</p>
   </li>

   <li><p>If the platform conventions do not in any way use <span
   data-x="concept-extension">extensions</span> to determine the types of file on the file system,
   then return <var>filename</var> as the filename.</p></li>

   <li><p>Let <var>claimed type</var> be the type given by <var>response</var>'s <span
   data-x="Content-Type">Content-Type metadata</span>, if any is known. Let <var>named type</var> be
   the type given by <var>filename</var>'s <span data-x="concept-extension">extension</span>, if any
   is known. For the purposes of this step, a <i>type</i> is a mapping of a <span>MIME type</span>
   to an <span data-x="concept-extension">extension</span>.</p></li>

   <li><p>If <var>named type</var> is consistent with the user's preferences (e.g., because the
   value of <var>filename</var> was determined by prompting the user), then return
   <var>filename</var> as the filename.</p></li>

   <li><p>If <var>claimed type</var> and <var>named type</var> are the same type (i.e., the type
   given by <var>response</var>'s <span data-x="Content-Type">Content-Type metadata</span> is
   consistent with the type given by <var>filename</var>'s <span
   data-x="concept-extension">extension</span>), then return <var>filename</var> as the
   filename.</p></li>

   <li>
    <p>If the <var>claimed type</var> is known, then alter <var>filename</var> to
    add an <span data-x="concept-extension">extension</span> corresponding to <var>claimed
    type</var>.</p>

    <p>Otherwise, if <var>named type</var> is known to be potentially dangerous (e.g. it
    will be treated by the platform conventions as a native executable, shell script, HTML
    application, or executable-macro-capable document) then optionally alter <var>filename</var> to add a known-safe <span data-x="concept-extension">extension</span>
    (e.g. "<code data-x="">.txt</code>").</p>

    <p class="note">This last step would make it impossible to download executables, which might not
    be desirable. As always, implementers are forced to balance security and usability in this
    matter.</p>
   </li>

   <li><p>Return <var>filename</var> as the filename.</p></li>
  </ol>

  <p>For the purposes of this algorithm, a file <dfn data-x="concept-extension">extension</dfn>
  consists of any part of the filename that platform conventions dictate will be used for
  identifying the type of the file. For example, many operating systems use the part of the filename
  following the last dot ("<code data-x="">.</code>") in the filename to determine the type of the
  file, and from that the manner in which the file is to be opened or executed.</p>

  <p>User agents should ignore any directory or path information provided by the response itself,
  its <span>URL</span>, and any <code data-x="attr-hyperlink-download">download</code> attribute, in
  deciding where to store the resulting file in the user's file system.</p>

  </div>


  <h4><dfn>Hyperlink auditing</dfn></h4>

  <div w-nodev>

  <p>If a <span>hyperlink</span> created by an <code>a</code> or <code>area</code> element has a
  <code data-x="attr-hyperlink-ping">ping</code> attribute, and the user follows the hyperlink, and
  the value of the element's <code data-x="attr-hyperlink-href">href</code> attribute can be <span
  data-x="encoding-parsing a URL">parsed</span>, relative to the element's <span>node
  document</span>, without failure, then the user agent must take the <code
  data-x="attr-hyperlink-ping">ping</code> attribute's value, <span data-x="split a string on ASCII
  whitespace">split that string on ASCII whitespace</span>, <span data-x="encoding-parsing a
  URL">parse</span> each resulting token, relative to the element's <span>node document</span>, and
  then run these steps for each resulting <span>URL</span> <var>ping URL</var>, ignoring when
  parsing returns failure:</p>

  <ol>
   <li><p>If <var>ping URL</var>'s <span data-x="concept-url-scheme">scheme</span> is not an
   <span>HTTP(S) scheme</span>, then return.</p></li>

   <li><p>Optionally, return. (For example, the user agent might wish to ignore any or
   all ping URLs in accordance with the user's expressed preferences.)</p></li>

   <li><p>Let <var>settingsObject</var> be the element's <span>node document</span>'s
   <span>relevant settings object</span>.</p></li>

   <li><p>Let <var>request</var> be a new <span data-x="concept-request">request</span> whose <span
   data-x="concept-request-url">URL</span> is <var>ping URL</var>, <span
   data-x="concept-request-method">method</span> is `<code data-x="">POST</code>`, <span
   data-x="concept-request-header-list">header list</span> is « (`<code>Content-Type</code>`,
   `<code>text/ping</code>`) », <span data-x="concept-request-body">body</span> is `<code
   data-x="">PING</code>`, <span data-x="concept-request-client">client</span> is
   <var>settingsObject</var>, <span data-x="concept-request-destination">destination</span> is the
   empty string, <span data-x="concept-request-credentials-mode">credentials mode</span> is "<code
   data-x="">include</code>", <span data-x="concept-request-referrer">referrer</span> is "<code
   data-x="">no-referrer</code>", and whose <span>use-URL-credentials flag</span> is set, and whose
   <span data-x="concept-request-initiator-type">initiator type</span> is "<code
   data-x="">ping</code>".</p></li>

   <li>
    <p>Let <var>target URL</var> be the result of <span>encoding-parsing-and-serializing a
    URL</span> given the element's <code data-x="attr-hyperlink-href">href</code> attribute's value,
    relative to the element's <span>node document</span>, and then:</p>

    <dl class="switch">
     <dt>If the <span data-x="concept-document-url">URL</span> of the <code>Document</code> object
     containing the hyperlink being audited and <var>ping URL</var> have the <span>same
     origin</span></dt>

     <dt>If the origins are different, but the <span data-x="concept-url-scheme">scheme</span> of
     the <span data-x="concept-document-url">URL</span> of the <code>Document</code> containing the
     hyperlink being audited is not "<code data-x="">https</code>"</dt>

     <dd><var>request</var> must include a `<code>Ping-From</code>` header with, as its value, the
     <span data-x="concept-document-url">URL</span> of the document containing the hyperlink, and a
     `<code>Ping-To</code>` HTTP header with, as its value, the <var>target URL</var>.</dd>

     <dt>Otherwise</dt>

     <dd><var>request</var> must include a `<code>Ping-To</code>` HTTP header with, as its value,
     <var>target URL</var>. <span class="note"><var>request</var> does not include a
     `<code>Ping-From</code>` header.</span></dd>
    </dl>
   </li>

   <li><p><span data-x="concept-fetch">Fetch</span> <var>request</var>.</p></li>
  </ol>

  <p>This may be done <span>in parallel</span> with the primary fetch, and is independent of the
  result of that fetch.</p>

  <p>User agents should allow the user to adjust this behavior, for example in conjunction with a
  setting that disables the sending of HTTP `<code data-x="http-referer">Referer</code>` (sic)
  headers. Based on the user's preferences, UAs may either <span>ignore</span> the <code
  data-x="attr-hyperlink-ping">ping</code> attribute altogether, or selectively ignore URLs in the
  list (e.g. ignoring any third-party URLs); this is explicitly accounted for in the steps
  above.</p>

  <p>User agents must ignore any entity bodies returned in the responses. User agents may close the
  connection prematurely once they start receiving a response body.</p>

  <p>When the <code data-x="attr-hyperlink-ping">ping</code> attribute is present, user agents
  should clearly indicate to the user that following the hyperlink will also cause secondary
  requests to be sent in the background, possibly including listing the actual target URLs.</p>

  <p class="example">For example, a visual user agent could include the hostnames of the target ping
  URLs along with the hyperlink's actual URL in a status bar or tooltip.</p>

  </div>

  <div class="note">

   <p>The <code data-x="attr-hyperlink-ping">ping</code> attribute is redundant with pre-existing
   technologies like HTTP redirects and JavaScript in allowing web pages to track which off-site
   links are most popular or allowing advertisers to track click-through rates.</p>

   <p>However, the <code data-x="attr-hyperlink-ping">ping</code> attribute provides these advantages
   to the user over those alternatives:</p>

   <ul>
    <li>It allows the user to see the final target URL unobscured.</li>

    <li>It allows the UA to inform the user about the out-of-band notifications.</li>

    <li>It allows the user to disable the notifications without losing the underlying link
    functionality.</li>

    <li>It allows the UA to optimize the use of available network bandwidth so that the target page
    loads faster.</li>
   </ul>

   <p>Thus, while it is possible to track users without this feature, authors are encouraged to use
   the <code data-x="attr-hyperlink-ping">ping</code> attribute so that the user agent can make the
   user experience more transparent.</p>

  </div>

  <!-- resolving ping urls happens at audit time, so base URL changes affect the values of ping
  attributes -->

  <h5 id="the-ping-headers">The `<code>Ping-From</code>` and `<code>Ping-To</code>` headers</h5>

  <p>The `<dfn http-header><code>Ping-From</code></dfn>` and `<dfn
  http-header><code>Ping-To</code></dfn>` HTTP request headers are included in <span>hyperlink
  auditing</span> requests. Their value is a <span>URL</span>, <span
  data-x="concept-url-serializer">serialized</span>.</p>


  <h4 id="linkTypes">Link types</h4>

  <p>The following table summarizes the link types that are defined by this specification, by their
  corresponding keywords. This table is non-normative; the actual definitions for the link types are
  given in the next few sections.</p>

  <p>In this section, the term <i>referenced document</i> refers to the resource identified by the
  element representing the link, and the term <i>current document</i> refers to the resource within
  which the element representing the link finds itself.</p>

  <div w-nodev>

  <p>To determine which link types apply to a <code>link</code>, <code>a</code>, <code>area</code>,
  or <code>form</code> element, the element's <code data-x="">rel</code> attribute must be <span
  data-x="split a string on ASCII whitespace">split on ASCII whitespace</span>. The resulting tokens
  are the keywords for the link types that apply to that element.</p>

  </div>

  <p>Except where otherwise specified, a keyword must not be specified more than once per <code
  data-x="attr-hyperlink-rel">rel</code> attribute.</p>

  <p>Some of the sections that follow the table below list synonyms for certain keywords. The
  indicated synonyms <span w-nodev>are to be handled as specified by user agents, but</span> must
  not be used in documents (for example, the keyword "<code data-x="">copyright</code>").</p>

  <p>Keywords are always <span>ASCII case-insensitive</span><span w-nodev>, and must be
  compared as such</span>.</p>

  <p class="example">Thus, <code data-x="">rel="next"</code> is the same as <code
  data-x="">rel="NEXT"</code>.</p>

  <p>Keywords that are <dfn>body-ok</dfn> affect whether <code>link</code> elements are
  <span>allowed in the body</span>. The <span>body-ok</span> keywords are
  <code data-x="rel-dns-prefetch">dns-prefetch</code>,
  <code data-x="rel-modulepreload">modulepreload</code>,
  <code data-x="rel-pingback">pingback</code>,
  <code data-x="rel-preconnect">preconnect</code>,
  <code data-x="rel-prefetch">prefetch</code>,
  <code data-x="rel-preload">preload</code>, and
  <code data-x="rel-stylesheet">stylesheet</code>.</p>

  <p>New link types that are to be implemented by web browsers are to be added to this standard. The
  remainder can be <span data-x="concept-rel-extensions">registered as extensions</span>.</p>

  <table class="yesno" id="table-link-relations">
   <thead>
    <tr>
     <th rowspan="2">Link type</th>
     <th colspan="3">Effect on...</th>
     <th rowspan="2"><span>body-ok</span></th>
     <th rowspan="2">Has `<code data-x="http-link">Link</code>` processing</th>
     <th rowspan="2">Brief description</th>
    </tr>
    <tr>
     <th><code>link</code></th>
     <th><code>a</code> and <code>area</code></th>
     <th><code>form</code></th>
    </tr>
   </thead>
   <tbody>

    <tr>
     <td><code data-x="rel-alternate">alternate</code></td>
     <td colspan="2"><span>Hyperlink</span></td>
     <td><em>not allowed</em></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Gives alternate representations of the current document.</td>
    </tr>

    <tr>
     <td><code data-x="rel-canonical">canonical</code></td>
     <td><span>Hyperlink</span></td>
     <td colspan="2"><em>not allowed</em></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Gives the preferred URL for the current document.</td>
    </tr>

    <tr>
     <td><code data-x="rel-author">author</code></td>
     <td colspan="2"><span>Hyperlink</span></td>
     <td><em>not allowed</em></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Gives a link to the author of the current document or article.</td>
    </tr>

    <tr>
     <td><code data-x="rel-bookmark">bookmark</code></td>
     <td><em>not allowed</em></td>
     <td><span>Hyperlink</span></td>
     <td><em>not allowed</em></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Gives the permalink for the nearest ancestor section.</td>
    </tr>

    <tr>
     <td><code data-x="rel-dns-prefetch">dns-prefetch</code></td>
     <td><span data-x="external resource link">External Resource</span></td>
     <td colspan="2"><em>not allowed</em></td>
     <td class="yes"> Yes </td>
     <td class="no"> &middot; </td>
     <td>Specifies that the user agent should preemptively perform DNS resolution for the target resource's <span>origin</span>.</td>
    </tr>

    <tr>
     <td><code data-x="rel-expect">expect</code></td>
     <td><span data-x="internal resource link">Internal Resource</span></td>
     <td colspan="2"><em>not allowed</em></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Expect an element with the target ID to appear in the current document.</td>
    </tr>

    <tr>
     <td><code data-x="rel-external">external</code></td>
     <td><em>not allowed</em></td>
     <td colspan="2"><span data-x="hyperlink annotation">Annotation</span></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Indicates that the referenced document is not part of the same site as the current document.</td>
    </tr>

    <tr>
     <td><code data-x="rel-help">help</code></td>
     <td colspan="3"><span>Hyperlink</span></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Provides a link to context-sensitive help.</td>
    </tr>

    <tr>
     <td><code data-x="rel-icon">icon</code></td>
     <td><span data-x="external resource link">External Resource</span></td>
     <td colspan="2"><em>not allowed</em></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Imports an icon to represent the current document.</td>
    </tr>

    <tr>
     <td><code data-x="rel-manifest">manifest</code></td>
     <td><span data-x="external resource link">External Resource</span></td>
     <td colspan="2"><em>not allowed</em></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Imports or links to an <span>application manifest</span>. <ref>MANIFEST</ref></td>
    </tr>

    <tr>
     <td><code data-x="rel-modulepreload">modulepreload</code></td>
     <td><span data-x="external resource link">External Resource</span></td>
     <td colspan="2"><em>not allowed</em></td>
     <td class="yes"> Yes </td>
     <td class="no"> &middot; </td>
     <td>Specifies that the user agent must preemptively <span data-x="fetch a single module script">fetch the module
     script</span> and store it in the document's <span data-x="concept-document-module-map">module map</span> for later
     evaluation. Optionally, the module's dependencies can be fetched as well.</td>
    </tr>

    <tr>
     <td><code data-x="rel-license">license</code></td>
     <td colspan="3"><span>Hyperlink</span></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Indicates that the main content of the current document is covered by the copyright license described by the referenced document.</td>
    </tr>

    <tr>
     <td><code data-x="rel-next">next</code></td>
     <td colspan="3"><span>Hyperlink</span></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Indicates that the current document is a part of a series, and that the next document in the series is the referenced document.</td>
    </tr>

    <tr>
     <td><code data-x="rel-nofollow">nofollow</code></td>
     <td><em>not allowed</em></td>
     <td colspan="2"><span data-x="hyperlink annotation">Annotation</span></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Indicates that the current document's original author or publisher does not endorse the referenced document.</td>
    </tr>

    <tr>
     <td><code data-x="rel-noopener">noopener</code></td>
     <td><em>not allowed</em></td>
     <td colspan="2"><span data-x="hyperlink annotation">Annotation</span></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Creates a <span>top-level traversable</span> with a non-<span>auxiliary browsing
     context</span> if the hyperlink would otherwise create one that was auxiliary (i.e., has an
     appropriate <code data-x="attr-hyperlink-target">target</code> attribute value).</td>
    </tr>

    <tr>
     <td><code data-x="rel-noreferrer">noreferrer</code></td>
     <td><em>not allowed</em></td>
     <td colspan="2"><span data-x="hyperlink annotation">Annotation</span></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>No `<code data-x="http-referer">Referer</code>` (sic) header will be included.
     Additionally, has the same effect as <code data-x="rel-noopener">noopener</code>.</td>
    </tr>

    <tr>
     <td><code data-x="rel-opener">opener</code></td>
     <td><em>not allowed</em></td>
     <td colspan="2"><span data-x="hyperlink annotation">Annotation</span></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Creates an <span>auxiliary browsing context</span> if the hyperlink would otherwise create
     a <span>top-level traversable</span> with a non-<span>auxiliary browsing context</span> (i.e.,
     has "<code data-x="">_blank</code>" as <code data-x="attr-hyperlink-target">target</code>
     attribute value).</td>
    </tr>

    <tr>
     <td><code data-x="rel-pingback">pingback</code></td>
     <td><span data-x="external resource link">External Resource</span></td>
     <td colspan="2"><em>not allowed</em></td>
     <td class="yes"> Yes </td>
     <td class="no"> &middot; </td>
     <td>Gives the address of the pingback server that handles pingbacks to the current document.</td>
    </tr>

    <tr>
     <td><code data-x="rel-preconnect">preconnect</code></td>
     <td><span data-x="external resource link">External Resource</span></td>
     <td colspan="2"><em>not allowed</em></td>
     <td class="yes"> Yes </td>
     <td class="yes"> Yes </td>
     <td>Specifies that the user agent should preemptively connect to the target resource's <span>origin</span>.</td>
    </tr>

    <tr>
     <td><code data-x="rel-prefetch">prefetch</code></td>
     <td><span data-x="external resource link">External Resource</span></td>
     <td colspan="2"><em>not allowed</em></td>
     <td class="yes"> Yes </td>
     <td class="no"> &middot; </td>
     <td>Specifies that the user agent should preemptively <span data-x="concept-fetch">fetch</span> and cache the target resource as it is likely to be required for a followup <span data-x="navigate">navigation</span>.</td>
    </tr>

    <tr>
     <td><code data-x="rel-preload">preload</code></td>
     <td><span data-x="external resource link">External Resource</span></td>
     <td colspan="2"><em>not allowed</em></td>
     <td class="yes"> Yes </td>
     <td class="yes"> Yes </td>
     <td>Specifies that the user agent must preemptively <span data-x="concept-fetch">fetch</span> and cache the target resource for current <span data-x="navigate">navigation</span> according to the <span data-x="concept-potential-destination">potential destination</span> given by the <code data-x="attr-link-as">as</code> attribute (and the <span data-x="concept-request-priority">priority</span> associated with the <span data-x="concept-potential-destination-translate">corresponding</span> <span data-x="concept-request-destination">destination</span>).</td>
    </tr>

    <tr>
     <td><code data-x="rel-prev">prev</code></td>
     <td colspan="3"><span>Hyperlink</span></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Indicates that the current document is a part of a series, and that the previous document in the series is the referenced document.</td>
    </tr>

    <tr>
     <td><code data-x="rel-privacy-policy">privacy-policy</code></td>
     <td colspan="2"><span>Hyperlink</span></td>
     <td><em>not allowed</em></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Gives a link to information about the data collection and usage practices that apply to the current document.</td>
    </tr>

    <tr>
     <td><code data-x="rel-search">search</code></td>
     <td colspan="3"><span>Hyperlink</span></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Gives a link to a resource that can be used to search through the current document and its related pages.</td>
    </tr>

    <tr>
     <td><code data-x="rel-stylesheet">stylesheet</code></td>
     <td><span data-x="external resource link">External Resource</span></td>
     <td colspan="2"><em>not allowed</em></td>
     <td class="yes"> Yes </td>
     <td class="no"> &middot; </td>
     <td>Imports a style sheet.</td>
    </tr>

    <tr>
     <td><code data-x="rel-tag">tag</code></td>
     <td><em>not allowed</em></td>
     <td><span>Hyperlink</span></td>
     <td><em>not allowed</em></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Gives a tag (identified by the given address) that applies to the current document.</td>
    </tr>

    <tr>
     <td><code data-x="rel-terms-of-service">terms-of-service</code></td>
     <td colspan="2"><span>Hyperlink</span></td>
     <td><em>not allowed</em></td>
     <td class="no"> &middot; </td>
     <td class="no"> &middot; </td>
     <td>Gives a link to information about the agreements between the current document's provider and users who wish to use the current document.</td>
    </tr>
   </tbody>
  </table>


  <h5 id="rel-alternate"><span id="link-type-alternate"></span>Link type "<dfn
  for="link/rel,a/rel,area/rel" attr-value><code data-x="rel-alternate">alternate</code></dfn>"</h5>

  <p>The <code data-x="rel-alternate">alternate</code> keyword may be used with <code>link</code>,
  <code>a</code>, and <code>area</code> elements.</p>

  <p>The meaning of this keyword depends on the values of the other attributes.</p>

  <dl class="switch">

   <dt>If the element is a <code>link</code> element and the <code data-x="attr-link-rel">rel</code>
   attribute also contains the keyword <code data-x="rel-stylesheet">stylesheet</code></dt>

   <dd>
    <p>The <code data-x="rel-alternate">alternate</code> keyword modifies the meaning of the <code
    data-x="rel-stylesheet">stylesheet</code> keyword in the way described for that keyword. The
    <code data-x="rel-alternate">alternate</code> keyword does not create a link of its own.</p>

    <div class="example">

     <p>Here, a set of <code>link</code> elements provide some style sheets:</p>

     <pre><code class="html">&lt;!-- a persistent style sheet -->
&lt;link rel="stylesheet" href="default.css">

&lt;!-- the preferred alternate style sheet -->
&lt;link rel="stylesheet" href="green.css" title="Green styles">

&lt;!-- some alternate style sheets -->
&lt;link rel="alternate stylesheet" href="contrast.css" title="High contrast">
&lt;link rel="alternate stylesheet" href="big.css" title="Big fonts">
&lt;link rel="alternate stylesheet" href="wide.css" title="Wide screen"></code></pre>

    </div>
   </dd>


   <dt>If the <code data-x="rel-alternate">alternate</code> keyword is used with the <code
   data-x="attr-hyperlink-type">type</code> attribute set to the value <code
   data-x="">application/rss+xml</code> or the value <code data-x="">application/atom+xml</code></dt>

   <dd>
    <p>The keyword creates a <span>hyperlink</span> referencing a syndication feed (though not
    necessarily syndicating exactly the same content as the current page).</p>

    <div w-nodev>

    <p>For the purposes of feed autodiscovery, user agents should consider all <code>link</code>
    elements in the document with the <code data-x="rel-alternate">alternate</code> keyword used and
    with their <code data-x="attr-hyperlink-type">type</code> attribute set to the value <code
    data-x="">application/rss+xml</code> or the value <code
    data-x="">application/atom+xml</code>. If the user agent has the concept of a default
    syndication feed, the first such element (in <span>tree order</span>) should be used as the
    default.</p>

    </div>

    <div class="example">
     <p>The following <code>link</code> elements give syndication feeds for a blog:</p>

     <pre><code class="html">&lt;link rel="alternate" type="application/atom+xml" href="posts.xml" title="Cool Stuff Blog">
&lt;link rel="alternate" type="application/atom+xml" href="posts.xml?category=robots" title="Cool Stuff Blog: robots category">
&lt;link rel="alternate" type="application/atom+xml" href="comments.xml" title="Cool Stuff Blog: Comments"></code></pre>

     <p>Such <code>link</code> elements would be used by user agents engaged in feed autodiscovery,
     with the first being the default (where applicable).</p>

     <p>The following example offers various different syndication feeds to the user, using
     <code>a</code> elements:</p>

     <pre><code class="html">&lt;p>You can access the planets database using Atom feeds:&lt;/p>
&lt;ul>
 &lt;li>&lt;a href="recently-visited-planets.xml" rel="alternate" type="application/atom+xml">Recently Visited Planets&lt;/a>&lt;/li>
 &lt;li>&lt;a href="known-bad-planets.xml" rel="alternate" type="application/atom+xml">Known Bad Planets&lt;/a>&lt;/li>
 &lt;li>&lt;a href="unexplored-planets.xml" rel="alternate" type="application/atom+xml">Unexplored Planets&lt;/a>&lt;/li>
&lt;/ul></code></pre>

     <p>These links would not be used in feed autodiscovery.</p>
    </div>
   </dd>


   <dt>Otherwise</dt>

   <dd>
    <p>The keyword creates a <span>hyperlink</span> referencing an alternate representation of the
    current document.</p>

    <p>The nature of the referenced document is given by the <code
    data-x="attr-hyperlink-hreflang">hreflang</code>, and <code
    data-x="attr-hyperlink-type">type</code> attributes.</p>

    <p>If the <code data-x="rel-alternate">alternate</code> keyword is used with the <code
    data-x="attr-hyperlink-hreflang">hreflang</code> attribute, and that attribute's value differs
    from the <span>document element</span>'s <span>language</span>, it indicates that the referenced
    document is a translation.</p>

    <p>If the <code data-x="rel-alternate">alternate</code> keyword is used with the <code
    data-x="attr-hyperlink-type">type</code> attribute, it indicates that the referenced document is
    a reformulation of the current document in the specified format.</p>

    <p>The <code data-x="attr-hyperlink-hreflang">hreflang</code> and <code
    data-x="attr-hyperlink-type">type</code> attributes can be combined when specified with the <code
    data-x="rel-alternate">alternate</code> keyword.</p>

    <div class="example">

     <p>The following example shows how you can specify versions of the page that use alternative
     formats, are aimed at other languages, and that are intended for other media:</p>

     <pre><code class="html">&lt;link rel=alternate href="/en/html" hreflang=en type=text/html title="English HTML">
&lt;link rel=alternate href="/fr/html" hreflang=fr type=text/html title="French HTML">
&lt;link rel=alternate href="/en/html/print" hreflang=en type=text/html media=print title="English HTML (for printing)">
&lt;link rel=alternate href="/fr/html/print" hreflang=fr type=text/html media=print title="French HTML (for printing)">
&lt;link rel=alternate href="/en/pdf" hreflang=en type=application/pdf title="English PDF">
&lt;link rel=alternate href="/fr/pdf" hreflang=fr type=application/pdf title="French PDF"></code></pre>

    </div>

    <p>This relationship is transitive &mdash; that is, if a document links to two other documents
    with the link type "<code data-x="rel-alternate">alternate</code>", then, in addition to implying
    that those documents are alternative representations of the first document, it is also implying
    that those two documents are alternative representations of each other.</p>
   </dd>
  </dl>


  <h5>Link type "<dfn attr-value for="link/rel,a/rel,area/rel"><code
  data-x="rel-author">author</code></dfn>"</h5>

  <p>The <code data-x="rel-author">author</code> keyword may be used with <code>link</code>,
  <code>a</code>, and <code>area</code> elements. This keyword creates a <span>hyperlink</span>.</p>

  <p>For <code>a</code> and <code>area</code> elements, the <code data-x="rel-author">author</code>
  keyword indicates that the referenced document provides further information about the author of
  the nearest <code>article</code> element ancestor of the element defining the hyperlink, if there
  is one, or of the page as a whole, otherwise.</p>

  <p>For <code>link</code> elements, the <code data-x="rel-author">author</code> keyword indicates
  that the referenced document provides further information about the author for the page as a
  whole.</p>

  <p class="note">The "referenced document" can be, and often is, a <code data-x="mailto
  protocol">mailto:</code> URL giving the email address of the author. <ref>MAILTO</ref></p>

  <div w-nodev>

  <p><strong>Synonyms</strong>: For historical reasons, user agents must also treat
  <code>link</code>, <code>a</code>, and <code>area</code> elements that have a <code
  data-x="">rev</code> attribute with the value "<code data-x="">made</code>" as having the <code
  data-x="rel-author">author</code> keyword specified as a link relationship.</p>

  </div>


  <h5>Link type "<dfn attr-value for="a/rel,area/rel"><code
  data-x="rel-bookmark">bookmark</code></dfn>"</h5>

  <p>The <code data-x="rel-bookmark">bookmark</code> keyword may be used with <code>a</code> and
  <code>area</code> elements. This keyword creates a <span>hyperlink</span>.</p>

  <p>The <code data-x="rel-bookmark">bookmark</code> keyword gives a permalink for the nearest
  ancestor <code>article</code> element of the linking element in question, or of
  the section the linking element is most closely associated with, if
  there are no ancestor <code>article</code> elements.</p>

  <div class="example">

   <p>The following snippet has three permalinks. A user agent could determine which permalink
   applies to which part of the spec by looking at where the permalinks are given.</p>

   <pre><code class="html"> ...
 &lt;body>
  &lt;h1>Example of permalinks&lt;/h1>
  &lt;div id="a">
   &lt;h2>First example&lt;/h2>
   &lt;p>&lt;a href="a.html" rel="bookmark">This permalink applies to
   only the content from the first H2 to the second H2&lt;/a>. The DIV isn't
   exactly that section, but it roughly corresponds to it.&lt;/p>
  &lt;/div>
  &lt;h2>Second example&lt;/h2>
  &lt;article id="b">
   &lt;p>&lt;a href="b.html" rel="bookmark">This permalink applies to
   the outer ARTICLE element&lt;/a> (which could be, e.g., a blog post).&lt;/p>
   &lt;article id="c">
    &lt;p>&lt;a href="c.html" rel="bookmark">This permalink applies to
    the inner ARTICLE element&lt;/a> (which could be, e.g., a blog comment).&lt;/p>
   &lt;/article>
  &lt;/article>
 &lt;/body>
 ...</code></pre>

  </div>


  <h5>Link type "<dfn attr-value for="link/rel"><code
  data-x="rel-canonical">canonical</code></dfn>"</h5>

  <p>The <code data-x="rel-canonical">canonical</code> keyword may be used with <code>link</code>
  element. This keyword creates a <span>hyperlink</span>.</p>

  <p>The <code data-x="rel-canonical">canonical</code> keyword indicates that URL given by the <code
  data-x="attr-link-href">href</code> attribute is the preferred URL for the current document. That
  helps search engines reduce duplicate content, as described in more detail in <cite>The Canonical
  Link Relation</cite>. <ref>RFC6596</ref></p>


  <h5>Link type "<dfn attr-value for="link/rel"><code
  data-x="rel-dns-prefetch">dns-prefetch</code></dfn>"</h5>

  <p>The <code data-x="rel-dns-prefetch">dns-prefetch</code> keyword may be used with
  <code>link</code> elements. This keyword creates an <span data-x="external resource link">external
  resource link</span>. This keyword is <span>body-ok</span>.</p>

  <p>The <code data-x="rel-dns-prefetch">dns-prefetch</code> keyword indicates that preemptively
  performing DNS resolution for the <span>origin</span> of the specified resource is likely to be
  beneficial, as it is highly likely that the user will require resources located at that
  <span>origin</span>, and the user experience would be improved by preempting the latency costs
  associated with DNS resolution.</p>

  <p>There is no default type for resources given by the <code
  data-x="rel-dns-prefetch">dns-prefetch</code> keyword.</p>

  <p>The appropriate times to <span data-x="fetch and process the linked resource">fetch and
  process</span> this type of link are:</p>

  <ul>
   <li><p>When the <span>external resource link</span> is created on a <code>link</code> element
   that is already <span>browsing-context connected</span>.</p></li>
   <!-- e.g. rel="" changed, href="" set... -->

   <li><p>When the <span>external resource link</span>'s <code>link</code> element <span>becomes
   browsing-context connected</span>.</p></li>

   <li><p>When the <code data-x="attr-link-href">href</code> attribute of the <code>link</code>
   element of an <span>external resource link</span> that is already <span>browsing-context
   connected</span> is changed.</p></li>
  </ul>

  <p>The <span>fetch and process the linked resource</span> steps for this type of linked resource,
  given a <code>link</code> element <var>el</var>, are:

  <ol>
   <li><p>Let <var>url</var> be the result of <span>encoding-parsing a URL</span> given
   <var>el</var>'s <code data-x="attr-link-href">href</code> attribute's value, relative to
   <var>el</var>'s <span>node document</span>.</p></li>

   <li><p>If <var>url</var> is failure, then return.</p></li>

   <li><p>Let <var>partitionKey</var> be the result of <span data-x="determine the network
   partition key">determining the network partition key</span> given <var>el</var>'s
   <span>node document</span>'s <span>relevant settings object</span>.</p></li>

   <li>
    <p>The user agent should <span>resolve an origin</span> given <var>partitionKey</var> and
    <var>url</var>'s <span data-x="concept-url-origin">origin</span>.</p>

    <p class="note">As the results of this algorithm can be cached, future fetches could be
    faster.</p>
   </li>
  </ol>


  <h5>Link type "<dfn attr-value for="link/rel"><code data-x="rel-expect">expect</code></dfn>"</h5>

  <p>The <code data-x="rel-expect">expect</code> keyword may be used with <code>link</code>
  elements. This keyword creates an <span data-x="internal resource link">internal resource
  link</span>.</p>

  <p>An <span>internal resource link</span> created by the <code data-x="rel-expect">expect</code>
  keyword can be used to <span data-x="render-blocked">block rendering</span> until the element that
  it <span data-x="indicated part">indicates</span> is connected to the document and fully
  parsed.</p>

  <p>There is no default type for resources given by the <code data-x="rel-expect">expect</code>
  keyword.</p>

  <p>Whenever any of the following conditions occur for a <code>link</code> element
  <var>el</var>:</p>

  <ul>
   <li><p>the <code data-x="rel-expect">expect</code> <span>internal resource link</span> is created
   on <var>el</var> that is already <span>browsing-context connected</span>;</p></li>

   <li><p>an <code data-x="rel-expect">expect</code> <span>internal resource link</span> has been
   created on <var>el</var> and <var>el</var> becomes
   <span>browsing-context connected</span>;</p></li>

   <li><p>an <code data-x="rel-expect">expect</code><span>internal resource link</span> has been
   created on <var>el</var>, <var>el</var> is already <span>browsing-context connected</span>, and
   <var>el</var>'s <code data-x="attr-link-href">href</code> attribute is set, changed, or removed;
   or</p></li>

   <li><p>an <code data-x="rel-expect">expect</code> <span>internal resource link</span> has been
   created on <var>el</var>, <var>el</var> is already <span>browsing-context connected</span>, and
   <var>el</var>'s <code data-x="attr-link-media">media</code> attribute is set, changed, or
   removed,</p></li>
  </ul>

  <p>then <span data-x="process internal resource link">process</span> <var>el</var>.</p>

  <p>To <dfn>process internal resource link</dfn> given a <code>link</code> element <var>el</var>,
  run these steps:</p>

  <ol>
   <li><p>Let <var>doc</var> be <var>el</var>'s <span>node document</span>.</p></li>

   <li><p>Let <var>url</var> be the result of <span>encoding-parsing a URL</span> given
   <var>el</var>'s <code data-x="attr-link-href">href</code> attribute's value, relative to
   <var>doc</var>.</p></li>

   <li><p>If this fails, or if <var>url</var> does not <span
   data-x="concept-url-equals">equal</span> <var>doc</var>'s
   <span data-x="concept-document-url">URL</span> with <i data-x="url equals exclude
   fragments">exclude fragments</i> set to false, then <span>unblock rendering</span> on
   <var>el</var> and return.</p></li>

   <li><p>Let <var>indicatedElement</var> be the result of <span
   data-x="select the indicated part">selecting the indicated part</span> given <var>doc</var>
   and <var>url</var>.</p></li>

   <li>
    <p>If all of the following are true:</p>

    <ul>
     <li><p><var>doc</var>'s <span>current document readiness</span> is "<code
     data-x="">loading</code>";</p></li>

     <li><p><var>el</var> creates an <span>internal resource link</span>;</p></li>

     <li><p><var>el</var> is <span>browsing-context connected</span>;</p></li>

     <li><p><var>el</var>'s <code data-x="attr-link-rel">rel</code> attribute
     contains <code data-x="rel-expect">expect</code>;</p></li>

     <li><p><var>el</var> is <span>potentially render-blocking</span>;</p></li>

     <li><p><var>el</var>'s <code data-x="attr-link-media">media</code> attribute
     <span>matches the environment</span>; and</p></li>

     <li><p><var>indicatedElement</var> is not an element, or is on a
     <span>stack of open elements</span> of an <span>HTML parser</span> whose associated
     <code>Document</code> is <var>doc</var>,</p></li>
     </ul>

     <p>then <span>block rendering</span> on <var>el</var>.</p>
   </li>

   <li><p>Otherwise, <span>unblock rendering</span> on <var>el</var>.</p></li>
  </ol>

  <p>To <dfn>process internal resource links</dfn> given a <code>Document</code> <var>doc</var>:</p>

  <ol>
   <li><p><span data-x="list iterate">For each</span> <code
   data-x="rel-expect">expect</code> <code>link</code> element <var>link</var> in
   <var>doc</var>'s <span>render-blocking element set</span>, <span
   data-x="process internal resource link">process</span> <var>link</var>.</p></li>
  </ol>

  <p>The following <span
  data-x="concept-element-attributes-change-ext">attribute change steps</span>, given
  <var>element</var>, <var>localName</var>, <var>value</var>, and <var>namespace</var>, are used to
  ensure <code data-x="rel-expect">expect</code> <code>link</code> elements respond to dynamic <code
  data-x="attr-id">id</code> and <code data-x="attr-a-name">name</code> changes:</p>

  <ol>
   <li><p>If <var>namespace</var> is not null, then return.</p></li>

   <li><p>If <var>element</var> is in a <span>stack of open elements</span> of an
   <span>HTML parser</span>, then return.</p></li>

   <li>
    <p>If any of the following is true:</p>

    <ul>
     <li><p><var>localName</var> is <code data-x="attr-id">id</code>; or</p></li>

     <li><p><var>localName</var> is <code
     data-x="attr-a-name">name</code> and <var>element</var> is an <code>a</code> element,</p></li>
    </ul>

    <p>then <span>process internal resource links</span> given <var>element</var>'s
    <span>node document</span>.</p>
   </li>
  </ol>

  <h5>Link type "<dfn attr-value for="a/rel,area/rel,form/rel"><code
  data-x="rel-external">external</code></dfn>"</h5>
  <!-- fifth and sixth most used <a rel> value (sixth is "external nofollow") -->

  <p>The <code data-x="rel-external">external</code> keyword may be used with <code>a</code>,
  <code>area</code>, and <code>form</code> elements. This keyword does not create a
  <span>hyperlink</span>, but <span data-x="hyperlink annotation">annotates</span> any other
  hyperlinks created by the element (the implied hyperlink, if no other keywords create one).</p>

  <p>The <code data-x="rel-external">external</code> keyword indicates that the link is leading to a
  document that is not part of the site that the current document forms a part of.</p>


  <h5>Link type "<dfn attr-value for="link/rel,a/rel,area/rel,form/rel"><code
  data-x="rel-help">help</code></dfn>"</h5>

  <p>The <code data-x="rel-help">help</code> keyword may be used with <code>link</code>,
  <code>a</code>, <code>area</code>, and <code>form</code> elements. This keyword creates a
  <span>hyperlink</span>.</p>

  <p>For <code>a</code>, <code>area</code>, and <code>form</code> elements, the <code
  data-x="rel-help">help</code> keyword indicates that the referenced document provides further help
  information for the parent of the element defining the hyperlink, and its children.</p>

  <div class="example">

   <p>In the following example, the form control has associated context-sensitive help. The user
   agent could use this information, for example, displaying the referenced document if the user
   presses the "Help" or "F1" key.</p>

   <pre><code class="html"> &lt;p>&lt;label> Topic: &lt;input name=topic> &lt;a href="help/topic.html" rel="help">(Help)&lt;/a>&lt;/label>&lt;/p></code></pre>

  </div>

  <p>For <code>link</code> elements, the <code data-x="rel-help">help</code> keyword indicates that
  the referenced document provides help for the page as a whole.</p>

  <p>For <code>a</code> and <code>area</code> elements, on some browsers, the <code
  data-x="rel-help">help</code> keyword causes the link to use a different cursor.</p>


  <h5 id=rel-icon>Link type "<dfn attr-value for="link/rel"><code
  data-x="rel-icon">icon</code></dfn>"</h5>

  <p>The <code data-x="rel-icon">icon</code> keyword may be used with <code>link</code> elements.
  This keyword creates an <span>external resource link</span>.</p>

  <div w-nodev>

  <p>The specified resource is an icon representing the page or site, and should be used by the user
  agent when representing the page in the user interface.</p>

  </div>

  <p>Icons could be auditory icons, visual icons, or other kinds of icons. <span w-nodev>If
  multiple icons are provided, the user agent must select the most appropriate icon according to the
  <code data-x="attr-link-type">type</code>, <code data-x="attr-link-media">media</code>, and <code
  data-x="attr-link-sizes">sizes</code> attributes. If there are multiple equally appropriate icons,
  user agents must use the last one declared in <span>tree order</span> at the time that the user
  agent collected the list of icons. If the user agent tries to use an icon but that icon is
  determined, upon closer examination, to in fact be inappropriate (e.g. because it uses an
  unsupported format), then the user agent must try the next-most-appropriate icon as determined by
  the attributes.</span></p>

  <div w-nodev>

  <p class="note">User agents are not required to update icons when the list of icons changes, but
  are encouraged to do so.</p>

  <p>There is no default type for resources given by the <code data-x="rel-icon">icon</code> keyword.
  However, for the purposes of <a href="#concept-link-type-sniffing">determining the type of the
  resource</a>, user agents must expect the resource to be an image.</p>

  </div>

  <p>The <code data-x="attr-link-sizes">sizes</code> keywords represent icon sizes in raw pixels (as
  opposed to <span data-x="'px'">CSS pixels</span>).</p>

  <p class="note">An icon that is 50 <span data-x="'px'">CSS pixels</span> wide intended for
  displays with a device pixel density of two device pixels per <span data-x="'px'">CSS pixel</span>
  (2x, 192dpi) would have a width of 100 raw pixels. This feature does not support indicating that a
  different resource is to be used for small high-resolution icons vs large low-resolution icons
  (e.g. 50&times;50 2x vs 100&times;100 1x).</p>

  <div w-nodev>

  <p>To parse and process the attribute's value, the user agent must first <span data-x="split a
  string on ASCII whitespace">split the attribute's value on ASCII whitespace</span>, and must then
  parse each resulting keyword to determine what it represents.</p>

  </div>

  <p>The <dfn><code data-x="attr-link-sizes-any">any</code></dfn> keyword represents that the
  resource contains a scalable icon, e.g. as provided by an SVG image.</p>

  <div w-nodev>

  <p>Other keywords must be further parsed as follows to determine what they represent:</p>

  <ol>
   <li><p>If the keyword doesn't contain exactly one U+0078 LATIN SMALL LETTER X or U+0058 LATIN
   CAPITAL LETTER X character, then this keyword doesn't represent anything. Return for
   that keyword.</p></li>

   <li><p>Let <var>width string</var> be the string before the "<code data-x="">x</code>" or
   "<code data-x="">X</code>".</p></li>

   <li><p>Let <var>height string</var> be the string after the "<code data-x="">x</code>" or
   "<code data-x="">X</code>".</p></li>

   <li><p>If either <var>width string</var> or <var>height string</var> start with
   a U+0030 DIGIT ZERO (0) character or contain any characters other than <span>ASCII digits</span>,
   then this keyword doesn't represent anything. Return for that keyword.</p></li>

   <li><p>Apply the <span>rules for parsing non-negative integers</span> to <var>width
   string</var> to obtain <var>width</var>.</p></li>

   <li><p>Apply the <span>rules for parsing non-negative integers</span> to <var>height
   string</var> to obtain <var>height</var>.</p></li>

   <li><p>The keyword represents that the resource contains a bitmap icon with a width of <var>width</var> device pixels and a height of <var>height</var> device
   pixels.</p></li>
  </ol>

  </div>

  <p>The keywords specified on the <code data-x="attr-link-sizes">sizes</code> attribute must not
  represent icon sizes that are not actually available in the linked resource.</p>

  <div w-nodev>

  <p>The <span>linked resource fetch setup steps</span> for this type of linked resource, given a
  <code>link</code> element <var>el</var> and <span data-x="concept-request">request</span>
  <var>request</var>, are:</p>

  <ol>
   <li><p>Set <var>request</var>'s <span data-x="concept-request-destination">destination</span> to
   "<code data-x="">image</code>".</p></li>

   <li><p>Return true.</p></li>
  </ol>

  <p>The <span>process a link header</span> steps for this type of linked resource are to do
  nothing.</p>

  <p>In the absence of a <code>link</code> with the <code data-x="rel-icon">icon</code> keyword, for
  <code>Document</code> objects whose <span data-x="concept-document-url">URL</span>'s
  <span data-x="concept-url-scheme">scheme</span> is an <span>HTTP(S) scheme</span>, user agents may
  instead run these steps <span>in parallel</span>:</p>

  <ol>
   <li><p>Let <var>request</var> be a new <span data-x="concept-request">request</span> whose
   <span data-x="concept-request-url">URL</span> is the <span>URL record</span> obtained by
   resolving the <span>URL</span> "<code data-x="">/favicon.ico</code>" against the
   <code>Document</code> object's <span data-x="concept-document-url">URL</span>, <span
   data-x="concept-request-client">client</span> is the <code>Document</code> object's
   <span>relevant settings object</span>, <span
   data-x="concept-request-destination">destination</span> is "<code data-x="">image</code>",
   <span>synchronous flag</span> is set, <span data-x="concept-request-credentials-mode">credentials
   mode</span> is "<code data-x="">include</code>", and whose <span>use-URL-credentials flag</span>
   is set.</p></li>

   <!--FETCH--><li><p>Let <var>response</var> be the result of <span
   data-x="concept-fetch">fetching</span> <var>request</var>.</p></li>

   <li><p>Use <var>response</var>'s <span>unsafe response</span> as an icon as if it had been
   declared using the <code data-x="rel-icon">icon</code> keyword.</p></li>
  </ol>

  </div>

  <div class="example">

   <p>The following snippet shows the top part of an application with several icons.</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang="en">
 &lt;head>
  &lt;title>lsForums &mdash; Inbox&lt;/title>
  &lt;link rel=icon href=favicon.png sizes="16x16" type="image/png">
  &lt;link rel=icon href=windows.ico sizes="32x32 48x48" type="image/vnd.microsoft.icon">
  &lt;link rel=icon href=mac.icns sizes="128x128 512x512 8192x8192 32768x32768">
  &lt;link rel=icon href=iphone.png sizes="57x57" type="image/png">
  &lt;link rel=icon href=gnome.svg sizes="any" type="image/svg+xml">
  &lt;link rel=stylesheet href=lsforums.css>
  &lt;script src=lsforums.js>&lt;/script>
  &lt;meta name=application-name content="lsForums">
 &lt;/head>
 &lt;body>
  ...</code></pre>

  </div>

  <p>For historical reasons, the <code data-x="rel-icon">icon</code> keyword may be preceded by the
  keyword "<code data-x="">shortcut</code>". If the "<code data-x="">shortcut</code>" keyword is
  present, the <code data-x="attr-hyperlink-rel">rel</code> attribute's entire value must be an
  <span>ASCII case-insensitive</span> match for the string "<code
  data-x="">shortcut&nbsp;icon</code>" (with a single U+0020 SPACE character between the tokens and
  no other <span>ASCII whitespace</span>).</p>


  <h5>Link type "<dfn attr-value for="link/rel,a/rel,area/rel,form/rel"><code
  data-x="rel-license">license</code></dfn>"</h5>

  <p>The <code data-x="rel-license">license</code> keyword may be used with <code>link</code>,
  <code>a</code>, <code>area</code>, and <code>form</code> elements. This keyword creates a
  <span>hyperlink</span>.</p>

  <p>The <code data-x="rel-license">license</code> keyword indicates that the referenced document
  provides the copyright license terms under which the main content of the current document is
  provided.</p>

  <p>This specification does not specify how to distinguish between the main content of a document
  and content that is not deemed to be part of that main content. The distinction should be made
  clear to the user.</p>

  <div class="example">

   <p>Consider a photo sharing site. A page on that site might describe and show a photograph, and
   the page might be marked up as follows:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang="en">
 &lt;head>
  &lt;title>Exampl Pictures: Kissat&lt;/title>
  &lt;link rel="stylesheet" href="/style/default">
 &lt;/head>
 &lt;body>
  &lt;h1>Kissat&lt;/h1>
  &lt;nav>
   &lt;a href="../">Return to photo index&lt;/a>
  &lt;/nav>
  &lt;figure>
   &lt;img src="/pix/39627052_fd8dcd98b5.jpg">
   &lt;figcaption>Kissat&lt;/figcaption>
  &lt;/figure>
  &lt;p>One of them has six toes!&lt;/p>
  &lt;p>&lt;small>&lt;a rel="license" href="http://www.opensource.org/licenses/mit-license.php">MIT Licensed&lt;/a>&lt;/small>&lt;/p>
  &lt;footer>
   &lt;a href="/">Home&lt;/a> | &lt;a href="../">Photo index&lt;/a>
   &lt;p>&lt;small>&copy; copyright 2009 Exampl Pictures. All Rights Reserved.&lt;/small>&lt;/p>
  &lt;/footer>
 &lt;/body>
&lt;/html></code></pre>

   <p>In this case the <code data-x="rel-license">license</code> applies to just the photo (the main
   content of the document), not the whole document. In particular not the design of the page
   itself, which is covered by the copyright given at the bottom of the document. This could be made
   clearer in the styling (e.g. making the license link prominently positioned near the photograph,
   while having the page copyright in light small text at the foot of the page).</p>

  </div>

  <div w-nodev>

  <p><strong>Synonyms</strong>: For historical reasons, user agents must also treat the keyword
  "<code data-x="">copyright</code>" like the <code data-x="rel-license">license</code> keyword.</p>

  </div>


  <h5>Link type "<dfn attr-value for="link/rel"><code
  data-x="rel-manifest">manifest</code></dfn>"</h5>

  <p>The <code data-x="rel-manifest">manifest</code> keyword may be used with <code>link</code> elements.
  This keyword creates an <span data-x="external resource link">external resource link</span>.</p>

  <p>The <code data-x="rel-manifest">manifest</code> keyword indicates the manifest file that provides
  metadata associated with the current document.</p>

  <div w-nodev>

  <p>There is no default type for resources given by the <code data-x="rel-manifest">manifest</code>
  keyword.</p>

  <p>When a web application is not <span data-x="installed web application">installed</span>, the
  appropriate time to <span>fetch and process the linked resource</span> for this link type is when
  the user agent deems it necessary. For example, when the user chooses to <span data-x="installed
  web application">install the web application</span>.</p>

  <p>For an <span data-x="installed web application">installed web application</span>, the
  appropriate times to <span>fetch and process the linked resource</span> for this link type
  are:</p>

  <ul>
   <li><p>When the <span>external resource link</span> is created on a <code>link</code> element
   that is already <span>browsing-context connected</span>.</p></li>

   <li><p>When the <span>external resource link</span>'s <code>link</code> element <span>becomes
   browsing-context connected</span>.</p></li>

   <li><p>When the <code data-x="attr-link-href">href</code> attribute of the <code>link</code>
   element of an <span>external resource link</span> that is already <span>browsing-context
   connected</span> is changed.</p></li>
  </ul>

  <p>In any case, only the first <code>link</code> element in <span>tree order</span> whose <code
  data-x="attr-link-rel">rel</code> attribute contains the token <code
  data-x="rel-manifest">manifest</code> may be used.</p>

  <p>A user agent must not <span>delay the load event</span> for this link type.</p>

  <p>The <span>linked resource fetch setup steps</span> for this type of linked resource, given a
  <code>link</code> element <var>el</var> and <span data-x="concept-request">request</span>
  <var>request</var>, are:</p>

  <ol>
   <li><p>Let <var>navigable</var> be <var>el</var>'s <span>node document</span>'s <span>node
   navigable</span>.</p></li>

   <li><p>If <var>navigable</var> is null, then return false.</p></li>

   <li><p>If <var>navigable</var> is not a <span>top-level traversable</span>, then return
   false.</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-initiator">initiator</span> to
   "<code data-x="">manifest</code>".</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-destination">destination</span> to
   "<code data-x="">manifest</code>".</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-mode">mode</span> to "<code
   data-x="">cors</code>".</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-credentials-mode">credentials
   mode</span> to the <span>CORS settings attribute credentials mode</span> for <var>el</var>'s
   <code data-x="attr-link-crossorigin">crossorigin</code> content attribute.</p></li>

   <li><p>Return true.</p></li>
  </ol>

  <p>To <span data-x="process the linked resource">process this type of linked resource</span> given
  a <code>link</code> element <var>el</var>, boolean <var>success</var>, <span
  data-x="concept-response">response</span> <var>response</var>, and <span>byte sequence</span>
  <var>bodyBytes</var>:</p>

  <ol>
   <li><p>If <var>response</var>'s <span data-x="Content-Type">Content-Type metadata</span> is not a
   <span>JSON MIME type</span>, then set <var>success</var> to false.</p></li>

   <li>
    <p>If <var>success</var> is true:</p>

    <ol>
     <li><p>Let <var>document URL</var> be <var>el</var>'s <span>node document</span>'s <span
     data-x="concept-document-url">URL</span>.</p></li>

     <li><p>Let <var>manifest URL</var> be <var>response</var>'s <span
     data-x="concept-response-url">URL</span>.</p></li>

     <li><p><span>Process the manifest</span> given <var>document URL</var>, <var>manifest
     URL</var>, and <var>bodyBytes</var>. <ref>MANIFEST</ref></p></li>
    </ol>
   </li>
  </ol>

  <p>The <span>process a link header</span> steps for this type of linked resource are to do
  nothing.</p>

  </div>


  <h5>Link type "<dfn attr-value for="link/rel"><code
  data-x="rel-modulepreload">modulepreload</code></dfn>"</h5>

  <p>The <code data-x="rel-modulepreload">modulepreload</code> keyword may be used with
  <code>link</code> elements. This keyword creates an <span>external resource link</span>. This
  keyword is <span>body-ok</span>.</p>

  <p>The <code data-x="rel-modulepreload">modulepreload</code> keyword is a specialized alternative
  to the <code data-x="rel-preload">preload</code> keyword, with a processing model geared toward
  preloading <span data-x="module script">module scripts</span>. In particular, it uses the specific
  fetch behavior for module scripts (including, e.g., a different interpretation of the <code
  data-x="attr-link-crossorigin">crossorigin</code> attribute), and places the result into the
  appropriate <span data-x="concept-document-module-map">module map</span> for later evaluation. In
  contrast, a similar <span>external resource link</span> using the <code
  data-x="rel-preload">preload</code> keyword would place the result in the preload cache, without
  affecting the document's <span data-x="concept-document-module-map">module map</span>.</p>

  <p>Additionally, implementations can take advantage of the fact that <span data-x="module
  script">module scripts</span> declare their dependencies in order to fetch the specified module's
  dependency as well. This is intended as an optimization opportunity, since the user agent knows
  that, in all likelihood, those dependencies will also be needed later. It will not generally be
  observable without using technology such as service workers, or monitoring on the server side.
  Notably, the appropriate <code data-x="event-load">load</code> or <code
  data-x="event-error">error</code> events will occur after the specified module is fetched, and
  will not wait for any dependencies.</p>

  <div w-nodev>

  <p>A user agent must not <span>delay the load event</span> for this link type.</p>

  <p>The appropriate times to <span>fetch and process the linked resource</span> for such a link
  are:</p>

  <ul>
   <li><p>When the <span>external resource link</span> is created on a <code>link</code> element
   that is already <span>browsing-context connected</span>.</p></li>

   <li><p>When the <span>external resource link</span>'s <code>link</code> element <span>becomes
   browsing-context connected</span>.</p></li>

   <li><p>When the <code data-x="attr-link-href">href</code> attribute of the <code>link</code>
   element of an <span>external resource link</span> that is already <span>browsing-context
   connected</span> is changed.</p></li>
  </ul>

  </div>

  <p class="note">Unlike some other link relations, changing the relevant attributes (such as <code
  data-x="attr-link-as">as</code>, <code data-x="attr-link-crossorigin">crossorigin</code>, and
  <code data-x="attr-link-referrerpolicy">referrerpolicy</code>) of such a <code>link</code>
  does not trigger a new fetch. This is because the document's <span
  data-x="concept-document-module-map">module map</span> has already been populated by a previous
  fetch, and so re-fetching would be pointless.</p>

  <div w-nodev>

  <p>The <span>fetch and process the linked resource</span> algorithm for <code
  data-x="rel-modulepreload">modulepreload</code> links, given a <code>link</code> element
  <var>el</var>, is as follows:</p>

  <ol>
   <li><p>If <var>el</var>'s <code data-x="attr-link-href">href</code> attribute's value is the
   empty string, then return.</p></li>

   <li><p>Let <var>destination</var> be the current state of <var>el</var>'s <code
   data-x="attr-link-as">as</code> attribute (a <span
   data-x="concept-request-destination">destination</span>), or "<code data-x="">script</code>" if
   it is in no state.</p></li>

   <li><p>If <var>destination</var> is not <span
   data-x="concept-script-like-destination">script-like</span>, then <span>queue an element
   task</span> on the <span>networking task source</span> given <var>el</var> to <span
   data-x="concept-event-fire">fire an event</span> named <code data-x="event-error">error</code>
   at <var>el</var>, and return.</p></li>

   <li><p>Let <var>url</var> be the result of <span>encoding-parsing a URL</span> given
   <var>el</var>'s <code data-x="attr-link-href">href</code> attribute's value, relative to
   <var>el</var>'s <span>node document</span>.</p></li>

   <li><p>If <var>url</var> is failure, then return.</p></li>

   <li><p>Let <var>settings object</var> be <var>el</var>'s <span>node document</span>'s
   <span>relevant settings object</span>.</p></li>

   <li><p>Let <var>credentials mode</var> be the <span>CORS settings attribute credentials
   mode</span> for <var>el</var>'s <code data-x="attr-link-crossorigin">crossorigin</code>
   attribute.</p></li>

   <li><p>Let <var>cryptographic nonce</var> be
   <var>el</var>.<span>[[CryptographicNonce]]</span>.</p></li>

   <li><p>Let <var>integrity metadata</var> be the value of <var>el</var>'s <code
   data-x="attr-link-integrity">integrity</code> attribute, if it is specified, or the empty string
   otherwise.</p></li>

   <li><p>If <var>el</var> does not have an <code data-x="attr-link-integrity">integrity</code>
   attribute, then set <var>integrity metadata</var> to the result of <span>resolving a module
   integrity metadata</span> with <var>url</var> and <var>settings object</var>.</p></li>

   <li><p>Let <var>referrer policy</var> be the current state of <var>el</var>'s <code
   data-x="attr-link-referrerpolicy">referrerpolicy</code> attribute.</p></li>

   <li><p>Let <var>fetch priority</var> be the current state of <var>el</var>'s <code
   data-x="attr-link-fetchpriority">fetchpriority</code> attribute.</p></li>

   <li><p>Let <var>options</var> be a <span>script fetch options</span> whose <span
   data-x="concept-script-fetch-options-nonce">cryptographic nonce</span> is <var>cryptographic
   nonce</var>, <span data-x="concept-script-fetch-options-integrity">integrity metadata</span> is
   <var>integrity metadata</var>, <span data-x="concept-script-fetch-options-parser">parser
   metadata</span> is "<code data-x="">not-parser-inserted</code>", <span
   data-x="concept-script-fetch-options-credentials">credentials mode</span> is <var>credentials
   mode</var>, <span data-x="concept-script-fetch-options-referrer-policy">referrer
   policy</span> is <var>referrer policy</var>, and <span
   data-x="concept-script-fetch-options-fetch-priority">fetch priority</span> is
   <var>fetch priority</var>.</p></li>

   <li>
    <p><span>Fetch a modulepreload module script graph</span> given <var>url</var>,
    <var>destination</var>, <var>settings object</var>, <var>options</var>, and with the following
    steps given <var>result</var>:</p>

    <ol>
     <li><p>If <var>result</var> is null, then <span data-x="concept-event-fire">fire an
     event</span> named <code data-x="event-error">error</code> at <var>el</var>,
     and return.</p></li>

     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-load">load</code> at <var>el</var>.</p></li>
    </ol>
   </li>
  </ol>

   <p>The <span>process a link header</span> steps for this type of linked resource are to do
   nothing.</p>

  </div>

  <div class="example" id="example-modulepreload-manifest">
   <p>The following snippet shows the top part of an application with several modules preloaded:</p>

   <pre><code class="html">&lt;!DOCTYPE html>
&lt;html lang="en">
&lt;title>IRCFog&lt;/title>

&lt;link rel="modulepreload" href="app.mjs">
&lt;link rel="modulepreload" href="helpers.mjs">
&lt;link rel="modulepreload" href="irc.mjs">
&lt;link rel="modulepreload" href="fog-machine.mjs">

&lt;script type="module" src="app.mjs">
...</code></pre>

   <p>Assume that the module graph for the application is as follows:</p>

   <img src="/images/ircfog-modules.svg" width="301" height="151" alt="The module graph is rooted at app.mjs, which depends on irc.mjs and fog-machine.mjs. In turn, irc.mjs depends on helpers.mjs." class="darkmode-aware">

   <p>Here we see the application developer has used <code
   data-x="rel-modulepreload">modulepreload</code> to declare all of the modules in their module graph,
   ensuring that the user agent initiates fetches for them all. Without such preloading, the user
   agent might need to go through multiple network roundtrips before discovering <code
   data-x="">helpers.mjs</code>, if technologies such as HTTP/2 Server Push are not in play. In
   this way, <code data-x="rel-modulepreload">modulepreload</code> <code>link</code> elements can be
   used as a sort of "manifest" of the application's modules.</p>
  </div>

  <div class="example" id="example-modulepreload-dynamic-import">
   <p>The following code shows how <code data-x="rel-modulepreload">modulepreload</code> links can
   be used in conjunction with <code>import()</code> to ensure network fetching is done ahead of
   time, so that when <code>import()</code> is called, the module is already ready (but not
   evaluated) in the <span>module map</span>:</p>

   <pre><code class="html">&lt;link rel="modulepreload" href="awesome-viewer.mjs">

&lt;button onclick="import('./awesome-viewer.mjs').then(m => m.view())">
  View awesome thing
&lt;/button></code></pre>
  </div>

  <h5>Link type "<dfn attr-value for="a/rel,area/rel,form/rel"><code
  data-x="rel-nofollow">nofollow</code></dfn>"</h5>

  <p>The <code data-x="rel-nofollow">nofollow</code> keyword may be used with <code>a</code>,
  <code>area</code>, and <code>form</code> elements. This keyword does not create a
  <span>hyperlink</span>, but <span data-x="hyperlink annotation">annotates</span> any other
  hyperlinks created by the element (the implied hyperlink, if no other keywords create one).</p>

  <p>The <code data-x="rel-nofollow">nofollow</code> keyword indicates that the link is not endorsed
  by the original author or publisher of the page, or that the link to the referenced document was
  included primarily because of a commercial relationship between people affiliated with the two
  pages.</p>


  <h5>Link type "<dfn attr-value for="a/rel,area/rel,form/rel"><code
  data-x="rel-noopener">noopener</code></dfn>"</h5>

  <p>The <code data-x="rel-noopener">noopener</code> keyword may be used with <code>a</code>,
  <code>area</code>, and <code>form</code> elements. This keyword does not create a
  <span>hyperlink</span>, but <span data-x="hyperlink annotation">annotates</span> any other
  hyperlinks created by the element (the implied hyperlink, if no other keywords create one).</p>

  <p>The keyword indicates that any newly created <span>top-level traversable</span> which results
  from following the <span>hyperlink</span> will not contain an <span>auxiliary browsing
  context</span>. E.g., the resulting <code>Window</code>'s <code data-x="dom-opener">opener</code>
  getter will return null.</p>

  <p w-nodev class="note">See also the <a href="#noopener">processing model</a>.</p>

  <div class="example">
   <p>This typically creates a <span>top-level traversable</span> with an <span>auxiliary browsing
   context</span> (assuming there is no existing <span>navigable</span> whose <span
   data-x="nav-target">target name</span> is "<code data-x="">example</code>"):</p>

   <pre><code class="html">&lt;a href=help.html target=example>Help!&lt;/a></code></pre>

   <p>This creates a <span>top-level traversable</span> with a non-<span>auxiliary browsing
   context</span> (assuming the same thing):</p>

   <pre><code class="html">&lt;a href=help.html target=example rel=noopener>Help!&lt;/a></code></pre>

   <p>These are equivalent and only navigate the <span data-x="nav-parent">parent
   navigable</span>:</p>

   <pre><code class="html">&lt;a href=index.html target=_parent>Home&lt;/a></code></pre>
   <pre><code class="html">&lt;a href=index.html target=_parent rel=noopener>Home&lt;/a></code></pre>
  </div>


  <h5>Link type "<dfn attr-value for="a/rel,area/rel,form/rel"><code
  data-x="rel-noreferrer">noreferrer</code></dfn>"</h5>

  <p>The <code data-x="rel-noreferrer">noreferrer</code> keyword may be used with <code>a</code>,
  <code>area</code>, and <code>form</code> elements. This keyword does not create a
  <span>hyperlink</span>, but <span data-x="hyperlink annotation">annotates</span> any other
  hyperlinks created by the element (the implied hyperlink, if no other keywords create one).</p>

  <p>It indicates that no referrer information is to be leaked when following the link and also
  implies the <code data-x="rel-noopener">noopener</code> keyword behavior under the same
  conditions.</p>

  <p w-nodev class="note">See also the <a href="#noreferrer-a-area-processing-model">processing
  model</a> where referrer is directly manipulated.</p>

  <p class="example"><code data-x="">&lt;a href="..." rel="noreferrer" target="_blank"&gt;</code>
  has the same behavior as <code
  data-x="">&lt;a href="..." rel="noreferrer noopener" target="_blank"&gt;</code>.</p>


  <h5>Link type "<dfn attr-value for="a/rel,area/rel,form/rel"><code
  data-x="rel-opener">opener</code></dfn>"</h5>

  <p>The <code data-x="rel-opener">opener</code> keyword may be used with <code>a</code>,
  <code>area</code>, and <code>form</code> elements. This keyword does not create a
  <span>hyperlink</span>, but <span data-x="hyperlink annotation">annotates</span> any other
  hyperlinks created by the element (the implied hyperlink, if no other keywords create one).</p>

  <p>The keyword indicates that any newly created <span>top-level traversable</span> which results
  from following the <span>hyperlink</span> will contain an <span>auxiliary browsing
  context</span>.</p>

  <p w-nodev class="note">See also the <a href="#opener-processing-model">processing model</a>.</p>

  <div class="example">
   <p>In the following example the <code data-x="rel-opener">opener</code> is used to allow the help
   page popup to navigate its opener, e.g., in case what the user is looking for can be found
   elsewhere. An alternative might be to use a named target, rather than <code
   data-x="">_blank</code>, but this has the potential to clash with existing names.</p>

   <pre><code class="html">&lt;a href="..." <mark>rel=opener</mark> target=_blank>Help!&lt;/a></code></pre>
  </div>


  <h5>Link type "<dfn attr-value for="link/rel"><code
  data-x="rel-pingback">pingback</code></dfn>"</h5>

  <p>The <code data-x="rel-pingback">pingback</code> keyword may be used with <code>link</code>
  elements. This keyword creates an <span data-x="external resource link">external resource
  link</span>. This keyword is <span>body-ok</span>.</p>

  <p>For the semantics of the <code data-x="rel-pingback">pingback</code> keyword, see
  <cite>Pingback 1.0</cite>. <ref>PINGBACK</ref></p>


  <h5>Link type "<dfn attr-value for="link/rel"><code
  data-x="rel-preconnect">preconnect</code></dfn>"</h5>

  <p>The <code data-x="rel-preconnect">preconnect</code> keyword may be used with <code>link</code>
  elements. This keyword creates an <span data-x="external resource link">external resource
  link</span>. This keyword is <span>body-ok</span>.</p>

  <p>The <code data-x="rel-preconnect">preconnect</code> keyword indicates that preemptively
  initiating a connection to the <span>origin</span> of the specified resource is likely to be
  beneficial, as it is highly likely that the user will require resources located at that
  <span>origin</span>, and the user experience would be improved by preempting the latency costs
  associated with establishing the connection.</p>

  <p>There is no default type for resources given by the <code
  data-x="rel-preconnect">preconnect</code> keyword.</p>

  <p>A user agent must not <span>delay the load event</span> for this link type.</p>

  <p>The appropriate times to <span data-x="fetch and process the linked resource">fetch and
  process</span> this type of link are:</p>

  <ul>
   <li><p>When the <span>external resource link</span> is created on a <code>link</code> element
   that is already <span>browsing-context connected</span>.</p></li>
   <!-- e.g. rel="" changed, href="" set... -->

   <li><p>When the <span>external resource link</span>'s <code>link</code> element <span>becomes
   browsing-context connected</span>.</p></li>

   <li><p>When the <code data-x="attr-link-href">href</code> attribute of the <code>link</code>
   element of an <span>external resource link</span> that is already <span>browsing-context
   connected</span> is changed.</p></li>

   <li><p>When the <code data-x="attr-link-crossorigin">crossorigin</code> attribute of the
   <code>link</code> element of an <span data-x="external resource link">external resource
   link</span> that is already <span>browsing-context connected</span> is set, changed, or
   removed.</p></li>
  </ul>

  <p>The <span>fetch and process the linked resource</span> steps for this type of linked resource,
  given a <code>link</code> element <var>el</var>, are to <span
  data-x="create link options from element">create link options</span> from <var>el</var> and
  to <span>preconnect</span> given the result.</p>

  <p>The <span>process a link header</span> step for this type of linked resource given a <span
  data-x="link processing options">link processing options</span> <var>options</var> are to
  <span>preconnect</span> given <var>options</var>.</p>

  <p>To <dfn>preconnect</dfn> given a <span>link processing options</span> <var>options</var>:</p>

  <ol>
   <li><p>If <var>options</var>'s
   <span data-x="link options href">href</span> is an empty string, return.</p></li>

   <li>
    <p>Let <var>url</var> be the result of <span>encoding-parsing a URL</span> given
    <var>options</var>'s <span data-x="link options href">href</span>, relative to
    <var>options</var>'s <span data-x="link options base URL">base URL</span>.</p>

    <p class="XXX">Passing the base URL instead of a document or environment is tracked by <a
    href="https://github.com/whatwg/html/issues/9715">issue #9715</a>.</p>
   </li>

   <li><p>If <var>url</var> is failure, then return.</p></li>

   <li><p>If <var>url</var>'s <span data-x="concept-url-scheme">scheme</span> is not an
   <span>HTTP(S) scheme</span>, then return.</p></li>

   <li><p>Let <var>partitionKey</var> be the result of <span data-x="determine the network
   partition key">determining the network partition key</span> given <var>options</var>'s
   <span data-x="link options environment">environment</span>.</p></li>

   <li><p>Let <var>useCredentials</var> be true.</p></li>

   <li><p>If <var>options</var>'s <span data-x="link options crossorigin">crossorigin</span> is
   <span data-x="attr-crossorigin-anonymous">Anonymous</span> and <var>options</var>'s <span
   data-x="link options origin">origin</span> does not
   have the <span>same origin</span> as <var>url</var>'s
   <span data-x="concept-url-origin">origin</span>, then set <var>useCredentials</var> to
   false.</p></li>

   <li>
    <p>The user agent should <span>obtain a connection</span> given <var>partitionKey</var>,
    <var>url</var>'s <span data-x="concept-url-origin">origin</span>, and
    <var>useCredentials</var>.</p>

    <p class="note">This connection is obtained but not used directly. It will remain in the
    <span>connection pool</span> for subsequent use.</p>

    <p>The user agent should attempt to initiate a preconnect and perform the full connection
    handshake (DNS+TCP for HTTP, and DNS+TCP+TLS for HTTPS origins) whenever possible, but is
    allowed to elect to perform a partial handshake (DNS only for HTTP, and DNS or DNS+TCP for
    HTTPS origins), or skip it entirely, due to resource constraints or other reasons.</p>

    <p>The optimal number of connections per origin is dependent on the negotiated protocol, users
    current connectivity profile, available device resources, global connection limits, and other
    context specific variables. As a result, the decision for how many connections should be opened
    is deferred to the user agent.</p>
   </li>
  </ol>

  <h5>Link type "<dfn attr-value for="link/rel"><code
  data-x="rel-prefetch">prefetch</code></dfn>"</h5>

  <p>The <code data-x="rel-prefetch">prefetch</code> keyword may be used with <code>link</code>
  elements. This keyword creates an <span data-x="external resource link">external resource
  link</span>. This keyword is <span>body-ok</span>.</p>

  <p>The <code data-x="rel-prefetch">prefetch</code> keyword indicates that preemptively <span
  data-x="concept-fetch">fetching</span> and caching the specified resource or same-site document is
  likely to be beneficial, as it is highly likely that the user will require this resource for
  future navigations.</p>

  <p>There is no default type for resources given by the <code data-x="rel-prefetch">prefetch</code>
  keyword.</p>

  <p>The appropriate times to <span data-x="fetch and process the linked resource">fetch and
  process</span> this type of link are:</p>

  <ul>
   <li><p>When the <span>external resource link</span> is created on a <code>link</code> element
   that is already <span>browsing-context connected</span>.</p></li>

   <li><p>When the <span>external resource link</span>'s <code>link</code> element <span>becomes
   browsing-context connected</span>.</p></li>

   <li><p>When the <code data-x="attr-link-href">href</code> attribute of the <code>link</code>
   element of an <span>external resource link</span> that is already <span>browsing-context
   connected</span> is changed.</p></li>

   <li><p>When the <code data-x="attr-link-crossorigin">crossorigin</code> attribute of the
   <code>link</code> element of an <span data-x="external resource link">external resource
   link</span> that is already <span>browsing-context connected</span> is set, changed, or
   removed.</p></li>
  </ul>

  <p>The <span>fetch and process the linked resource</span> algorithm for <code
  data-x="rel-prefetch">prefetch</code> links, given a <code>link</code> element
  <var>el</var>, is as follows:</p>

  <ol>
   <li><p>If <var>el</var>'s <code data-x="attr-link-href">href</code> attribute's value is the
   empty string, then return.</p></li>

   <li><p>Let <var>options</var> be the result of <span data-x="create link options from
   element">creating link options</span> from <var>el</var>.</p></li>

   <li><p>Set <var>options</var>'s <span data-x="link options destination">destination</span> to
   the empty string.</p></li>

   <li><p>Let <var>request</var> be the result of <span data-x="create a link request">creating a
   link request</span> given <var>options</var>.</p></li>

   <li><p>If <var>request</var> is null, then return.</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-initiator">initiator</span> to
   "<code data-x="">prefetch</code>".</p></li>

   <li>
    <p>Let <var>processPrefetchResponse</var> be the following steps given a <span
    data-x="concept-response">response</span> <var>response</var> and null, failure, or a
    <span>byte sequence</span> <var>bytesOrNull</var>:</p>

    <ol>
     <li><p>If <var>response</var> is a <span>network error</span>, <span
     data-x="concept-event-fire">fire an event</span> named <code
     data-x="event-error">error</code> at <var>el</var>.</p></li>

     <li><p>Otherwise, <span data-x="concept-event-fire">fire an event</span> named <code
     data-x="event-load">load</code> at <var>el</var>.</p></li>
    </ol>
   </li>

   <li><p>The user agent should <span data-x="concept-fetch">fetch</span> <var>request</var>, with
   <i data-x="processResponseConsumeBody">processResponseConsumeBody</i> set to
   <var>processPrefetchResponse</var>. User agents may delay the fetching of <var>request</var> to
   prioritize other requests that are necessary for the current document.</p></li>
  </ol>

  <p>The <span>process a link header</span> steps for this type of linked resource are to do
  nothing.</p>

  <h5>Link type "<dfn for="link/rel" attr-value><code
  data-x="rel-preload">preload</code></dfn>"</h5>

  <p>The <code data-x="rel-preload">preload</code> keyword may be used with <code>link</code>
  elements. This keyword creates an <span data-x="external resource link">external resource
  link</span>. This keyword is <span>body-ok</span>.</p>

  <p>The <code data-x="rel-preload">preload</code> keyword indicates that the user agent will
  preemptively <span data-x="concept-fetch">fetch</span> and cache the specified resource according
  to the <span data-x="concept-potential-destination">potential destination</span> given by the
  <code data-x="attr-link-as">as</code> attribute, and the <span
  data-x="concept-request-priority">priority</span> given by the <code
  data-x="attr-link-fetchpriority">fetchpriority</code> attribute, as it is highly likely that the
  user will require this resource for the current navigation.

  <p class="note">User-agents might perform additional operations when a resource is loaded, such as
  preemptively <span data-x="dom-img-decode">decoding images</span> or <span
  data-x="Create a CSS style sheet">creating stylesheets</span>. However, these additional
  operations cannot have observable effects.</p>

  <p>There is no default type for resources given by the <code data-x="rel-preload">preload</code>
  keyword.</p>

  <p>A user agent must not <span>delay the load event</span> for this link type.</p>

  <p>The appropriate times to <span>fetch and process the linked resource</span> for such a link
   are:</p>

  <ul>
   <li><p>When the <span>external resource link</span> is created on a <code>link</code> element
   that is already <span>browsing-context connected</span>.</p></li>

   <li><p>When the <span>external resource link</span>'s <code>link</code> element <span>becomes
   browsing-context connected</span>.</p></li>

   <li><p>When the <code data-x="attr-link-href">href</code> attribute of the <code>link</code>
   element of an <span>external resource link</span> that is already <span>browsing-context
   connected</span> is changed.</p></li>

   <li><p>When the <code data-x="attr-link-as">as</code> attribute of the <code>link</code>
   element of an <span>external resource link</span> that is already <span>browsing-context
   connected</span> is changed.</p></li>

   <li><p>When the <code data-x="attr-link-type">type</code> attribute of the <code>link</code>
   element of an <span>external resource link</span> that is already <span>browsing-context
   connected</span>, but was previously not obtained due to the <code
   data-x="attr-link-type">type</code> attribute specifying an unsupported type for the request
   <span data-x="concept-request-destination">destination</span>, is set, removed, or
   changed.</p></li>

   <li><p>When the <code data-x="attr-link-media">media</code> attribute of the <code>link</code>
   element of an <span>external resource link</span> that is already <span>browsing-context
   connected</span>, but was previously not obtained due to the <code
   data-x="attr-link-media">media</code> attribute not
   <span data-x="matches the environment">matching the environment</span>, is changed or
   removed.</p></li>
  </ul>

  <p>A <code>Document</code> has a <dfn>map of preloaded resources</dfn>, which is an
  <span>ordered map</span>, initially empty.</p>

  <p>A <dfn>preload key</dfn> is a <span>struct</span>. It has the following <span data-x="struct
  item">items</span>:</p>

  <dl>
   <dt><dfn data-x="preload URL">URL</dfn>
   <dd>A <span>URL</span>

   <dt><dfn data-x="preload destination">destination</dfn>
   <dd>A string

   <dt><dfn data-x="preload mode">mode</dfn>
   <dd>A <span data-x="concept-request-mode">request mode</span>, either
   "<code data-x="">same-origin</code>", "<code data-x="">cors</code>", or
   "<code data-x="">no-cors</code>"

   <dt><dfn data-x="preload credentials mode">credentials mode</dfn>
   <dd>A <span data-x="concept-request-credentials-mode">credentials mode</span>
  </dl>

  <p>A <dfn>preload entry</dfn> is a <span>struct</span>. It has the following <span data-x="struct
  item">items</span>:</p>

  <dl>
   <dt><dfn data-x="preload integrity metadata">integrity metadata</dfn>
   <dd>A string

   <dt><dfn data-x="preload response">response</dfn>
   <dd>Null or a <span data-x="concept-response">response</span>

   <dt><dfn data-x="preload on response available">on response available</dfn>
   <dd>Null, or an algorithm accepting a <span data-x="concept-response">response</span> or null
  </dl>

  <p>To <dfn export>consume a preloaded resource</dfn> for <code>Window</code> <var>window</var>,
  given a <span>URL</span> <var>url</var>, a string <var>destination</var>, a string
  <var>mode</var>, a string <var>credentialsMode</var>, a string <var>integrityMetadata</var>, and
  <var>onResponseAvailable</var>, which is an algorithm accepting a <span
  data-x="concept-response">response</span>:</p>

  <ol>
   <li><p>Let <var>key</var> be a <span>preload key</span> whose <span data-x="preload
   URL">URL</span> is <var>url</var>, <span data-x="preload destination">destination</span> is
   <var>destination</var>, <span data-x="preload mode">mode</span> is <var>mode</var>, and
   <span data-x="preload credentials mode">credentials mode</span> is
   <var>credentialsMode</var>.</p></li>

   <li><p>Let <var>preloads</var> be <var>window</var>'s <span
   data-x="concept-document-window">associated <code>Document</code></span>'s <span>map of
   preloaded resources</span>.</p></li>

   <li><p>If <var>key</var> does not <span data-x="map exists">exist</span> in <var>preloads</var>,
   then return false.</p></li>

   <li><p>Let <var>entry</var> be <var>preloads</var>[<var>key</var>].</p></li>

   <li><p>Let <var>consumerIntegrityMetadata</var> be the result of
   <span data-x="parse integrity metadata">parsing</span> <var>integrityMetadata</var>.</p></li>

   <li><p>Let <var>preloadIntegrityMetadata</var> be the result of
   <span data-x="parse integrity metadata">parsing</span> <var>entry</var>'s
   <span data-x="preload integrity metadata">integrity metadata</span>.</p></li>

   <li>
    <p>If none of the following conditions apply:</p>

    <ul>
     <li><p><var>consumerIntegrityMetadata</var> is <code data-x="">no metadata</code>;</p></li>

     <li>
      <p><var>consumerIntegrityMetadata</var> is equal to <var>preloadIntegrityMetadata</var>;
      or</p>

      <p class="XXX">This comparison would ignore unknown integrity options. See <a
      href="https://github.com/w3c/webappsec-subresource-integrity/issues/116">issue #116.</a></p>
     </li>
    </ul>

    <p>then return false.</p>

    <p class="note">A mismatch in integrity metadata between the preload and the consumer, even if
    both match the data, would lead to an additional fetch from the network.</p>

    <p class="note">It is important that <span data-x="network error">network errors</span> are
    added to the preload cache so that if a preload request results in an error, the erroneous
    response isn't re-requested from the network later. This also has security implications;
    consider the case where a developer specifies subresource integrity metadata on a preload
    request, but not the following resource request. If the preload request fails subresource
    integrity verification and is discarded, the resource request will fetch and consume a
    potentially-malicious response from the network without verifying its integrity.
    <ref>SRI</ref></p>
   </li>

   <li><p><span data-x="map remove">Remove</span> <var>preloads</var>[<var>key</var>].</p></li>

   <li><p>If <var>entry</var>'s <span data-x="preload response">response</span> is null, then set
   <var>entry</var>'s <span data-x="preload on response available">on response available</span> to
   <var>onResponseAvailable</var>.</p></li>

   <li><p>Otherwise, call <var>onResponseAvailable</var> with <var>entry</var>'s <span
   data-x="preload response">response</span>.</p></li>

   <li><p>Return true.</p></li>
  </ol>

  <p>For the purposes of this section, a string <var>type</var>
  <dfn data-x="match preload type">matches</dfn> a string <var>destination</var> if the following
  algorithm returns true:</p>
  <ol>
   <li><p>If <var>type</var> is an empty string, then return true.</p></li>

   <li><p>If <var>destination</var> is "<code data-x="">fetch</code>", then return true.</p></li>

   <li><p>Let <var>mimeTypeRecord</var> be the result of
   <span data-x="Parse a mime type">parsing</span> <var>type</var>.</p></li>

   <li><p>If <var>mimeTypeRecord</var> is failure, then return false.</p></li>

   <li><p>If <var>mimeTypeRecord</var> is not <span
   data-x="Is MIME type supported by the user agent?">supported by the user agent</span>, then
   return false.</p>

   <li>
    <p>If any of the following are true:</p>

    <ul>
     <li><p><var>destination</var> is "<code data-x="">audio</code>" or "<code
     data-x="">video</code>", and <var>mimeTypeRecord</var> is an
     <span>audio or video MIME type</span>;</p></li>

     <li><p><var>destination</var> is a
     <span data-x="concept-script-like-destination">script-like destination</span> and
     <var>mimeTypeRecord</var> is a <span>JavaScript MIME type</span>;</p></li>

     <li><p><var>destination</var> is "<code data-x="">image</code>" and
     <var>mimeTypeRecord</var> is an <span>image MIME type</span>;</p></li>

     <li><p><var>destination</var> is "<code data-x="">font</code>" and
     <var>mimeTypeRecord</var> is a <span>font MIME type</span>;</p></li>

     <li><p><var>destination</var> is "<code data-x="">json</code>" and
     <var>mimeTypeRecord</var> is a <span>JSON MIME type</span>;</p></li>

     <li><p><var>destination</var> is "<code data-x="">style</code>" and
     <var>mimeTypeRecord</var>'s <span data-x="MIME type essence">essence</span> is
     <code>text/css</code>; or</p></li>

     <li><p><var>destination</var> is "<code data-x="">track</code>" and
     <var>mimeTypeRecord</var>'s <span data-x="MIME type essence">essence</span> is
     <code>text/vtt</code>,</p></li>
    </ul>

    <p>then return true.</p>
   </li>

   <li><p>Return false.</p></li>
  </ol>

  <p>To <dfn>create a preload key</dfn> for a <span data-x="concept-request">request</span>
  <var>request</var>, return a new <span>preload key</span> whose <span data-x="preload
  URL">URL</span> is <var>request</var>'s <span data-x="concept-request-url">URL</span>, <span
  data-x="preload destination">destination</span> is <var>request</var>'s <span
  data-x="concept-request-destination">destination</span>, <span data-x="preload
  mode">mode</span> is <var>request</var>'s <span data-x="concept-request-mode">mode</span>, and
  <span data-x="preload credentials mode">credentials mode</span> is <var>request</var>'s <span
  data-x="concept-request-credentials-mode">credentials mode</span>.</p>

  <p>To <dfn>translate a preload destination</dfn> given a string <var>destination</var>:</p>

  <ol>
   <li><p>If <var>destination</var> is not "<code data-x="">fetch</code>", "<code data-x="">font</code>",
   "<code data-x="">image</code>", "<code data-x="">script</code>", "<code data-x="">style</code>",
   or "<code data-x="">track</code>", then return null.</p></li>

   <li><p>Return the result of <span
   data-x="concept-potential-destination-translate">translating</span>
   <var>destination</var>.</p></li>
  </ol>

  <p>To <dfn>preload</dfn> given a <span
  data-x="link processing options">link processing options</span> <var>options</var> and
  an optional <var>processResponse</var>, which is an algorithm accepting a <span
  data-x="concept-response">response</span>:</p>

  <ol>
   <li><p>If <var>options</var>'s <span data-x="link options type">type</span> doesn't <span
   data-x="match preload type">match</span> <var>options</var>'s <span
   data-x="link options destination">destination</span>, then return.</p></li>

   <li><p>If <var>options</var>'s <span data-x="link options destination">destination</span> is
   "<code data-x="">image</code>" and <var>options</var>'s <span data-x="link options source
   set">source set</span> is not null, then set <var>options</var>'s <span data-x="link options
   href">href</span> to the result of <span data-x="select an image source from a source
   set">selecting an image source</span> from <var>options</var>'s <span data-x="link options
   source set">source set</span>.</p></li>

   <li><p>Let <var>request</var> be the result of <span data-x="create a link request">creating a
   link request</span> given <var>options</var>.</p></li>

   <li><p>If <var>request</var> is null, then return.</p></li>

   <li><p>Let <var>unsafeEndTime</var> be 0.</p></li>

   <li><p>Let <var>entry</var> be a new <span>preload entry</span> whose
   <span data-x="preload integrity metadata">integrity metadata</span> is <var>options</var>'s
   <span data-x="link options integrity">integrity</span>.</p></li>

   <li><p>Let <var>key</var> be the result of <span data-x="create a preload key">creating a
   preload key</span> given <var>request</var>.</p></li>

   <li><p>If <var>options</var>'s <span data-x="link options document">document</span> is "<code
   data-x="">pending</code>", then set <var>request</var>'s <span
   data-x="concept-request-initiator-type">initiator type</span> to "<code data-x="">early
   hint</code>".</p></li>

   <li><p>Let <var>controller</var> be null.</p></li>

   <li><p>Let <var>reportTiming</var> given a <code>Document</code> <var>document</var> be to
   <span>report timing</span> for <var>controller</var> given <var>document</var>'s <span>relevant
   global object</span>.</p></li>

   <li>
    <p><!--FETCH-->Set <var>controller</var> to the result of <span
    data-x="concept-fetch">fetching</span> <var>request</var>, with <i
    data-x="processResponseConsumeBody">processResponseConsumeBody</i> set to the following steps
    given a <span data-x="concept-response">response</span> <var>response</var> and null, failure,
    or a <span>byte sequence</span> <var>bodyBytes</var>:</p>

    <ol>
     <li>
      <p>If <var>bodyBytes</var> is a <span>byte sequence</span>, then set <var>response</var>'s
      <span data-x="concept-response-body">body</span> to <var>bodyBytes</var> <span>as a
      body</span>.</p>

      <p class=note>By using <i data-x="processResponseConsumeBody">processResponseConsumeBody</i>,
      we have <span data-x="body safely extract">extracted</span> the entire <span
      data-x="concept-response-body">body</span>. This is necessary to ensure the preloader loads the
      entire body from the network, regardless of whether the preload will be consumed (which is
      uncertain at this point). This step then resets the request's body to a new body containing the
      same bytes, so that other specifications can read from it at the time of actual consumption,
      despite us having already done so once.</p>
     </li>

     <li><p>Otherwise, set <var>response</var> to a <span>network error</span>.</p></li>

     <li><p>Set <var>unsafeEndTime</var> to the <span>unsafe shared current time</span>.</p></li>

     <li><p>If <var>options</var>'s <span data-x="link options document">document</span> is not
     null, then call <var>reportTiming</var> given <var>options</var>'s <span
     data-x="link options document">document</span>.</p></li>

     <li><p>If <var>entry</var>'s <span data-x="preload on response available">on response
     available</span> is null, then set <var>entry</var>'s <span data-x="preload
     response">response</span> to <var>response</var>; otherwise call <var>entry</var>'s <span
     data-x="preload on response available">on response available</span> given
     <var>response</var>.</p></li>

     <li><p>If <var>processResponse</var> is given, then call <var>processResponse</var> with
     <var>response</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>Let <var>commit</var> be the following steps given a <code>Document</code>
    <var>document</var>:</p>

    <ol>
     <li><p>If <var>entry</var>'s <span data-x="preload response">response</span> is not null, then
     call <var>reportTiming</var> given <var>document</var>.</p></li>

     <li><p>Set <var>document</var>'s <span>map of preloaded resources</span>[<var>key</var>] to
     <var>entry</var>.</p></li>
    </ol>
   </li>

   <li><p>If <var>options</var>'s <span
   data-x="link options document">document</span> is null, then set <var>options</var>'s <span
   data-x="link options on document ready">on document ready</span> to <var>commit</var>.
   Otherwise, call <var>commit</var> with <var>options</var>'s <span
   data-x="link options document">document</span>.</p></li>
  </ol>

  <p>The <span>fetch and process the linked resource</span> steps for this type of linked resource,
  given a <code>link</code> element <var>el</var>, are:</p>

  <ol>
   <li><p><span>Update the source set</span> for <var>el</var>.</p></li>

   <li><p>Let <var>options</var> be the result of <span
   data-x="create link options from element">creating link options</span> from
   <var>el</var>.</p></li>

   <li>
    <p><span>Preload</span> <var>options</var>, with the following steps given a <span
    data-x="concept-response">response</span> <var>response</var>:

    <ol>
     <li>
      <p>If <var>response</var> is a <span>network error</span>, <span
      data-x="concept-event-fire">fire an event</span> named <code
      data-x="event-error">error</code> at <var>el</var>. Otherwise, <span
      data-x="concept-event-fire">fire an event</span> named
      <code data-x="event-load">load</code> at <var>el</var>.</p>

      <p class="XXX">The actual browsers' behavior is different from the spec here, and the
      feasibility of changing the behavior has not yet been investigated. See <a
      href="https://github.com/whatwg/html/issues/1142">issue #1142</a>.</p>
     </li>
    </ol>
   </li>
  </ol>

  <p>The <span>process a link header</span> step for this type of link
  given a <span data-x="link processing options">link processing options</span> <var>options</var>
  is to <span>preload</span> <var>options</var>.</p>


  <h5>Link type "<dfn attr-value for="link/rel,a/rel,area/rel"><code
  data-x="rel-privacy-policy">privacy-policy</code></dfn>"</h5>

  <p>The <code data-x="rel-privacy-policy">privacy-policy</code> keyword may be used with
  <code>link</code>, <code>a</code>, and <code>area</code> elements. This keyword creates a
  <span>hyperlink</span>.</p>

  <p>The <code data-x="rel-privacy-policy">privacy-policy</code> keyword indicates that the
  referenced document contains information about the data collection and usage practices that apply
  to the current document, as described in more detail in <cite>Additional Link Relation
  Types</cite>. The referenced document may be a standalone privacy policy, or a specific section of
  some more general document. <ref>RFC6903</ref></p>


  <h5>Link type "<dfn attr-value for="link/rel,a/rel,area/rel,form/rel"><code
  data-x="rel-search">search</code></dfn>"</h5>

  <p>The <code data-x="rel-search">search</code> keyword may be used with <code>link</code>,
  <code>a</code>, <code>area</code>, and <code>form</code> elements. This keyword creates a
  <span>hyperlink</span>.</p>

  <p>The <code data-x="rel-search">search</code> keyword indicates that the referenced document
  provides an interface specifically for searching the document and its related resources.</p>

  <p class="note">OpenSearch description documents can be used with <code>link</code> elements and
  the <code data-x="rel-search">search</code> link type to enable user agents to autodiscover search
  interfaces. <ref>OPENSEARCH</ref></p>


  <h5>Link type "<dfn attr-value for="link/rel"><code
  data-x="rel-stylesheet">stylesheet</code></dfn>"</h5>

  <p>The <code data-x="rel-stylesheet">stylesheet</code> keyword may be used with <code>link</code>
  elements. This keyword creates an <span data-x="external resource link">external resource
  link</span> that contributes to the styling processing model. This keyword is
  <span>body-ok</span>.</p>

  <p>The specified resource is a <span>CSS style sheet</span> that describes how to present the
  document.</p>

  <p>If the <code data-x="rel-alternate">alternate</code> keyword is also specified on the
  <code>link</code> element, then <dfn id="the-link-is-an-alternative-stylesheet">the link is an
  alternative style sheet</dfn>; in this case, the <code data-x="attr-title">title</code> attribute
  must be specified on the <code>link</code> element, with a non-empty value.</p>

  <p>The default type for resources given by the <code data-x="rel-stylesheet">stylesheet</code>
  keyword is <code>text/css</code>.</p>

  <p>A <code>link</code> element of this type is <span>implicitly potentially render-blocking</span>
  if the element was created by its <span>node document</span>'s parser.</p>

  <div w-nodev>

  <p>When the <code data-x="attr-link-disabled">disabled</code> attribute of a <code>link</code>
  element with a <code data-x="rel-stylesheet">stylesheet</code> keyword is set, <span
  data-x="disable a CSS style sheet">disable</span> the <span>associated CSS style sheet</span>.</p>

  <p>The appropriate times to <span data-x="fetch and process the linked resource">fetch and
  process</span> this type of link are:

  <!-- has a browsing context: https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=4299 -->
  <ul>
   <li><p>When the <span>external resource link</span> is created on a <code>link</code> element
   that is already <span>browsing-context connected</span>.</p></li>
   <!-- e.g. rel="" changed, href="" set... -->

   <li><p>When the <span>external resource link</span>'s <code>link</code> element <span>becomes
   browsing-context connected</span>.</p></li>

   <li><p>When the <code data-x="attr-link-href">href</code> attribute of the <code>link</code>
   element of an <span>external resource link</span> that is already <span>browsing-context
   connected</span> is changed.</p></li>
   <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2588 -->

   <li><p>When the <code data-x="attr-link-disabled">disabled</code> attribute of the
   <code>link</code> element of an <span>external resource link</span> that is already
   <span>browsing-context connected</span> is set, changed, or removed.</p></li>

   <li><p>When the <code data-x="attr-link-crossorigin">crossorigin</code> attribute of the
   <code>link</code> element of an <span data-x="external resource link">external resource
   link</span> that is already <span>browsing-context connected</span> is set, changed, or
   removed.</p></li>

   <li><p>When the <code data-x="attr-link-type">type</code> attribute of the <code>link</code>
   element of an <span>external resource link</span> that is already <span>browsing-context
   connected</span> is set or changed to a value that does not or no longer matches the <span
   data-x="Content-Type">Content-Type metadata</span> of the previous obtained external resource, if
   any.</p></li>

   <li><p>When the <code data-x="attr-link-type">type</code> attribute of the <code>link</code>
   element of an <span>external resource link</span> that is already <span>browsing-context
   connected</span>, but was previously not obtained due to the <code
   data-x="attr-link-type">type</code> attribute specifying an unsupported type, is removed or
   changed.</p></li>

   <li><p>When the <span>external resource link</span> that is already <span>browsing-context
   connected</span> changes from being <span data-x="the link is an alternative style sheet">an
   alternative style sheet</span> to not being one, or vice versa.</p></li>
  </ul>

  <p><strong>Quirk</strong>: If the document has been set to <span>quirks mode</span>, has the
  <span>same origin</span> as the <span>URL</span> of the external resource<!-- CVE-2010-0654 -->,
  and the <span data-x="Content-Type">Content-Type metadata</span> of the external resource is not a
  supported style sheet type, the user agent must instead assume it to be <code>text/css</code>.</p>

  <p>The <span>linked resource fetch setup steps</span> for this type of linked resource, given a
  <code>link</code> element <var>el</var> and <span data-x="concept-request">request</span>
  <var>request</var>, are:</p>

  <ol>
   <li><p>If <var>el</var>'s <code data-x="attr-link-disabled">disabled</code> attribute is set,
   then return false.</p></li>

   <li><p>If <var>el</var> <span>contributes a script-blocking style sheet</span>, <span data-x="set
   append">append</span> <var>el</var> to its <span>node document</span>'s <span>script-blocking
   style sheet set</span>.</p></li>

   <li><p>If <var>el</var>'s <code data-x="attr-link-media">media</code> attribute's value
   <span>matches the environment</span> and <var>el</var> is
   <span>potentially render-blocking</span>, then <span>block rendering</span> on
   <var>el</var>.</p></li>

   <li><p>If <var>el</var> is currently <span>render-blocking</span>, then set <var>request</var>'s
   <span data-x="concept-request-render-blocking">render-blocking</span> to true.</p></li>

   <li><p>Return true.</p></li>
  </ol>

  <p class="XXX">See <a href="https://github.com/whatwg/html/issues/968">issue #968</a> for plans
  to use the CSSOM <a href="https://drafts.csswg.org/cssom/#fetching-css-style-sheets">fetch a CSS
  style sheet</a> algorithm instead of the <span>default fetch and process the linked
  resource</span> algorithm. In the meantime, any <span data-x="critical subresources">critical
  subresource</span> <span data-x="concept-request">request</span> should have its <span
  data-x="concept-request-render-blocking">render-blocking</span> set to whether or not the
  <code>link</code> element is currently <span>render-blocking</span>.</p>

  <p>To <span data-x="process the linked resource">process this type of linked resource</span>
  given a <code>link</code> element <var>el</var>, boolean <var>success</var>, <span
  data-x="concept-response">response</span> <var>response</var>, and <span>byte sequence</span>
  <var>bodyBytes</var>:</p>

  <!-- note that we can only get this far if we have a browsing context, since if you don't have a
  browsing context (and are active) you can't ever get a result back from "fetch", and thus you
  never obtain the sheet -->
  <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2741 -->

  <ol>
   <li><p>If the resource's <span data-x="Content-Type">Content-Type metadata</span> is not
   <code>text/css</code>, then set <var>success</var> to false.</p></li>

   <li>
    <p>If <var>el</var> no longer creates an <span>external resource link</span> that contributes to
    the styling processing model, or if, since the resource in question was <span data-x="fetch and
    process the linked resource">fetched</span>, it has become appropriate to <span data-x="fetch
    and process the linked resource">fetch</span> it again, then:</p>

    <ol>
     <li><p><span data-x="list remove">Remove</span> <var>el</var> from <var>el</var>'s <span>node
     document</span>'s <span>script-blocking style sheet set</span>.</p></li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li><p>If <var>el</var> has an <span>associated CSS style sheet</span>, <span data-x="remove a
   CSS style sheet">remove the CSS style sheet</span>.</p></li>

   <li>
    <p>If <var>success</var> is true, then:</p>

    <ol>
     <li>
      <p><span>Create a CSS style sheet</span> with the following properties:</p>

      <dl>
       <dt><span data-x="concept-css-style-sheet-type">type</span></dt>
       <dd><p><code>text/css</code></p></dd>

       <dt><span data-x="concept-css-style-sheet-location">location</span></dt>
       <dd>
        <p><var>response</var>'s <span data-x="concept-response-url-list">URL list</span>[0]</p>

        <p class="XXX">We provide a URL here on the assumption that <a
        href="https://github.com/w3c/csswg-drafts/issues/9316">w3c/csswg-drafts issue #9316</a> will
        be fixed.</p>
       </dd>

       <dt><span data-x="concept-css-style-sheet-owner-node">owner node</span></dt>
       <dd><p><var>el</var></p></dd>

       <dt><span data-x="concept-css-style-sheet-media">media</span></dt>
       <dd>
        <p>The <code data-x="attr-link-media">media</code> attribute of <var>el</var>.</p>

        <p class="note">This is a reference to the (possibly absent at this time) attribute, rather
        than a copy of the attribute's current value. <cite>CSSOM</cite> defines what happens
        when the attribute is dynamically set, changed, or removed.</p>
       </dd>

       <dt><span data-x="concept-css-style-sheet-title">title</span></dt>
       <dd>
        <p>The <code data-x="attr-link-title">title</code> attribute of <var>el</var>, if
        <var>el</var> is <span>in a document tree</span>, or the empty string otherwise.</p>

        <p class="note">This is similarly a reference to the attribute, rather than a copy of the
        attribute's current value.</p>
       </dd>

       <dt><span data-x="concept-css-style-sheet-alternate-flag">alternate flag</span></dt>
       <dd><p>Set if <span>the link is an alternative style sheet</span> and <var>el</var>'s
       <span>explicitly enabled</span> is false; unset otherwise.</p></dd>

       <dt><span data-x="concept-css-style-sheet-origin-clean-flag">origin-clean flag</span></dt>
       <dd><p>Set if the resource is <span>CORS-same-origin</span>; unset otherwise.</p></dd>

       <dt>
        <span data-x="concept-css-style-sheet-parent-CSS-style-sheet">parent CSS style
        sheet</span>
       </dt>
       <dt><span data-x="concept-css-style-sheet-owner-CSS-rule">owner CSS rule</span></dt>
       <dd><p>null</p></dd>

       <dt><span data-x="concept-css-style-sheet-disabled-flag">disabled flag</span></dt>
       <dd><p>Left at its default value.</p></dd>

       <dt><span data-x="concept-css-style-sheet-CSS-rules">CSS rules</span></dt>
        <dd>
         <p>Left uninitialized.</p>

         <p class="XXX">This doesn't seem right. Presumably we should be using <var>bodyBytes</var>?
         Tracked as <a href="https://github.com/whatwg/html/issues/2997">issue #2997</a>.</p>
        </dd>
      </dl>

      <p>The CSS <span>environment encoding</span> is the result of running the following steps:
      <ref>CSSSYNTAX</ref></p>

      <ol>
       <li><p>If <var>el</var> has a <code data-x="attr-link-charset">charset</code> attribute,
       <span data-x="getting an encoding">get an encoding</span> from that attribute's value. If
       that succeeds, return the resulting encoding. <ref>ENCODING</ref></p></li>

       <li><p>Otherwise, return the <span>document's character encoding</span>.
       <ref>DOM</ref></p></li>
      </ol>
     </li>

     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-load">load</code> at <var>el</var>.</p></li>
    </ol>
   </li>

   <li><p>Otherwise, <span data-x="concept-event-fire">fire an event</span> named <code
   data-x="event-error">error</code> at <var>el</var>.</p></li>

   <li>
    <p>If <var>el</var> <span>contributes a script-blocking style sheet</span>, then:</p>

    <ol>
     <li><p><span>Assert</span>: <var>el</var>'s <span>node document</span>'s <span>script-blocking
     style sheet set</span> <span data-x="list contains">contains</span> <var>el</var>.</p></li>

     <li><p><span data-x="list remove">Remove</span> <var>el</var> from its <span>node
     document</span>'s <span>script-blocking style sheet set</span>.</p></li>
    </ol>
   </li>

   <li><p><span>Unblock rendering</span> on <var>el</var>.</p></li>
  </ol>

  <p>The <span>process a link header</span> steps for this type of linked resource are to do
  nothing.</p>

  </div>


  <h5>Link type "<dfn attr-value for="a/rel,area/rel"><code
  data-x="rel-tag">tag</code></dfn>"</h5>

  <p>The <code data-x="rel-tag">tag</code> keyword may be used with <code>a</code> and
  <code>area</code> elements. This keyword creates a <span>hyperlink</span>.</p>

  <p>The <code data-x="rel-tag">tag</code> keyword indicates that the <em>tag</em> that the
  referenced document represents applies to the current document.</p>

  <p class="note">Since it indicates that the tag <em>applies to the current document</em>, it would
  be inappropriate to use this keyword in the markup of a <a href="#tag-cloud">tag cloud</a>, which
  lists the popular tags across a set of pages.</p>

  <div class="example">

   <p>This document is about some gems, and so it is <i>tagged</i> with "<code
   data-x="">https://en.wikipedia.org/wiki/Gemstone</code>" to unambiguously categorize it as applying
   to the "jewel" kind of gems, and not to, say, the towns in the US, the Ruby package format, or
   the Swiss locomotive class:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang="en">
 &lt;head>
  &lt;title>My Precious&lt;/title>
 &lt;/head>
 &lt;body>
  &lt;header>&lt;h1>My precious&lt;/h1> &lt;p>Summer 2012&lt;/p>&lt;/header>
  &lt;p>Recently I managed to dispose of a red gem that had been
  bothering me. I now have a much nicer blue sapphire.&lt;/p>
  &lt;p>The red gem had been found in a bauxite stone while I was digging
  out the office level, but nobody was willing to haul it away. The
  same red gem stayed there for literally years.&lt;/p>
  &lt;footer>
   Tags: &lt;a rel=tag href="https://en.wikipedia.org/wiki/Gemstone">Gemstone&lt;/a>
  &lt;/footer>
 &lt;/body>
&lt;/html></code></pre>

  </div>

  <div class="example">

   <p>In <em>this</em> document, there are two articles. The "<code data-x="rel-tag">tag</code>"
   link, however, applies to the whole page (and would do so wherever it was placed, including if it
   was within the <code>article</code> elements).</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang="en">
 &lt;head>
  &lt;title>Gem 4/4&lt;/title>
 &lt;/head>
 &lt;body>
  &lt;article>
   &lt;h1>801: Steinbock&lt;/h1>
   &lt;p>The number 801 Gem 4/4 electro-diesel has an ibex and was rebuilt in 2002.&lt;/p>
  &lt;/article>
  &lt;article>
   &lt;h1>802: Murmeltier&lt;/h1>
   &lt;figure>
    &lt;img src="https://upload.wikimedia.org/wikipedia/commons/b/b0/Trains_de_la_Bernina_en_hiver_2.jpg"
         alt="The 802 was red with pantographs and tall vents on the side.">
    &lt;figcaption>The 802 in the 1980s, above Lago Bianco.&lt;/figcaption>
   &lt;/figure>
   &lt;p>The number 802 Gem 4/4 electro-diesel has a marmot and was rebuilt in 2003.&lt;/p>
  &lt;/article>
  &lt;p class="topic">&lt;a rel=tag href="https://en.wikipedia.org/wiki/Rhaetian_Railway_Gem_4/4">Gem 4/4&lt;/a>&lt;/p>
 &lt;/body>
&lt;/html></code></pre>

  </div>


  <h5>Link Type "<dfn attr-value for="link/rel,a/rel,area/rel"><code
  data-x="rel-terms-of-service">terms-of-service</code></dfn>"</h5>

  <p>The <code data-x="rel-terms-of-service">terms-of-service</code> keyword may be used with
  <code>link</code>, <code>a</code>, and <code>area</code> elements. This keyword creates a
  <span>hyperlink</span>.</p>

  <p>The <code data-x="rel-terms-of-service">terms-of-service</code> keyword indicates that the
  referenced document contains information about the agreements between the current document's
  provider and users who wish to use the current document, as described in more detail in
  <cite>Additional Link Relation Types</cite>. <ref>RFC6903</ref></p>


  <h5>Sequential link types</h5>

  <p>Some documents form part of a sequence of documents.</p>

  <p>A sequence of documents is one where each document can have a <em>previous sibling</em> and a
  <em>next sibling</em>. A document with no previous sibling is the start of its sequence, a
  document with no next sibling is the end of its sequence.</p>

  <p>A document may be part of multiple sequences.</p>


  <h6>Link type "<dfn attr-value for="link/rel,a/rel,area/rel,form/rel"><code
  data-x="rel-next">next</code></dfn>"</h6>

  <p>The <code data-x="rel-next">next</code> keyword may be used with <code>link</code>,
  <code>a</code>, <code>area</code>, and <code>form</code> elements. This keyword creates a
  <span>hyperlink</span>.</p>

  <p>The <code data-x="rel-next">next</code> keyword indicates that the document is part of a
  sequence, and that the link is leading to the document that is the next logical document in the
  sequence.</p>

  <p w-nodev>When the <code data-x="rel-next">next</code> keyword is used with a <code>link</code>
  element, user agents should process such links as if they were using one of the <code
  data-x="rel-dns-prefetch">dns-prefetch</code>, <code data-x="rel-preconnect">preconnect</code>, or
  <code data-x="rel-prefetch">prefetch</code> keywords. Which keyword the user agent wishes to use
  is implementation-dependent; for example, a user agent may wish to use the less-costly <code
  data-x="rel-preconnect">preconnect</code> processing model when trying to conserve data, battery
  power, or processing power, or may wish to pick a keyword depending on heuristic analysis of past
  user behavior in similar scenarios.</p>

  <h6>Link type "<dfn attr-value for="link/rel,a/rel,area/rel,form/rel"><code
  data-x="rel-prev">prev</code></dfn>"</h6>

  <p>The <code data-x="rel-prev">prev</code> keyword may be used with <code>link</code>,
  <code>a</code>, <code>area</code>, and <code>form</code> elements. This keyword creates a
  <span>hyperlink</span>.</p>

  <p>The <code data-x="rel-prev">prev</code> keyword indicates that the document is part of a
  sequence, and that the link is leading to the document that is the previous logical document in
  the sequence.</p>

  <div w-nodev>

  <p><strong>Synonyms</strong>: For historical reasons, user agents must also treat the keyword
  "<code data-x="">previous</code>" like the <code data-x="rel-prev">prev</code> keyword.</p>

  </div>


  <h5>Other link types</h5>

  <p><dfn data-x="concept-rel-extensions">Extensions to the predefined set of link types</dfn> may
  be registered on the <a
  href="https://microformats.org/wiki/existing-rel-values#HTML5_link_type_extensions">microformats
  page for existing rel values</a>. <ref>MFREL</ref></p>

  <p>Anyone is free to edit the <span data-x="">microformats page for existing rel values</span> at
  any time to add a type. Extension types must be specified with the following information:</p>

  <dl>
   <dt>Keyword</dt>

   <dd>
    <p>The actual value being defined. The value should not be confusingly similar to any other
    defined value (e.g. differing only in case).</p>

    <p>If the value contains a U+003A COLON character (:), it must also be an <span>absolute
    URL</span>.</p>
   </dd>


   <dt>Effect on... <code>link</code></dt>

   <dd>
    <p>One of the following:</p>

    <dl>
     <dt>Not allowed</dt>

     <dd>The keyword must not be specified on <code>link</code> elements.</dd>

     <dt>Hyperlink</dt>

     <dd>The keyword may be specified on a <code>link</code> element; it creates a
     <span>hyperlink</span>.</dd>

     <dt>External Resource</dt>

     <dd>The keyword may be specified on a <code>link</code> element; it creates an <span>external
     resource link</span>.</dd>
    </dl>
   </dd>


   <dt>Effect on... <code>a</code> and <code>area</code></dt>

   <dd>
    <p>One of the following:</p>

    <dl>
     <dt>Not allowed</dt>

     <dd>The keyword must not be specified on <code>a</code> and <code>area</code> elements.</dd>

     <dt>Hyperlink</dt>

     <dd>The keyword may be specified on <code>a</code> and <code>area</code> elements; it creates a
     <span>hyperlink</span>.</dd>

     <dt>External Resource</dt>

     <dd>The keyword may be specified on <code>a</code> and <code>area</code> elements; it creates
     an <span>external resource link</span>.</dd>

     <dt>Hyperlink Annotation</dt>

     <dd>The keyword may be specified on <code>a</code> and <code>area</code> elements; it <span
     data-x="hyperlink annotation">annotates</span> other <span data-x="hyperlink">hyperlinks</span>
     created by the element.</dd>
    </dl>
   </dd>


   <dt>Effect on... <code>form</code></dt>
   <dd>
    <p>One of the following:</p>

    <dl>
     <dt>Not allowed</dt>
     <dd>The keyword must not be specified on <code>form</code> elements.</dd>

     <dt>Hyperlink</dt>
     <dd>The keyword may be specified on <code>form</code> elements; it creates a
     <span>hyperlink</span>.</dd>

     <dt>External Resource</dt>
     <dd>The keyword may be specified on <code>form</code> elements; it creates an <span>external
     resource link</span>.</dd>

     <dt>Hyperlink Annotation</dt>
     <dd>The keyword may be specified on <code>form</code> elements; it <span data-x="hyperlink
     annotation">annotates</span> other <span data-x="hyperlink">hyperlinks</span> created by the
     element.</dd>
    </dl>
   </dd>


   <dt>Brief description</dt>

   <dd><p>A short non-normative description of what the keyword's meaning is.</p></dd>


   <dt>Specification</dt>

   <dd><p>A link to a more detailed description of the keyword's semantics and requirements. It
   could be another page on the wiki, or a link to an external page.</p></dd>


   <dt>Synonyms</dt>

   <dd><p>A list of other keyword values that have exactly the same processing requirements. Authors
   should not use the values defined to be synonyms, they are only intended to allow user agents to
   support legacy content. Anyone may remove synonyms that are not used in practice; only names that
   need to be processed as synonyms for compatibility with legacy content are to be registered in
   this way.</p></dd>


   <dt>Status</dt>

   <dd>
    <p>One of the following:</p>

    <dl>
     <dt>Proposed</dt>

     <dd>The keyword has not received wide peer review and approval. Someone has proposed it and is,
     or soon will be, using it.</dd>

     <dt>Ratified</dt>

     <dd>The keyword has received wide peer review and approval. It has a specification that
     unambiguously defines how to handle pages that use the keyword, including when they use it in
     incorrect ways.</dd>

     <dt>Discontinued</dt>

     <dd>The keyword has received wide peer review and it has been found wanting. Existing pages are
     using this keyword, but new pages should avoid it. The "brief description" and "specification"
     entries will give details of what authors should use instead, if anything.</dd>
    </dl>

    <p>If a keyword is found to be redundant with existing values, it should be removed and listed
    as a synonym for the existing value.</p>

    <p>If a keyword is registered in the "proposed" state for a period of a month or more without
    being used or specified, then it may be removed from the registry.</p>

    <p>If a keyword is added with the "proposed" status and found to be redundant with existing
    values, it should be removed and listed as a synonym for the existing value. If a keyword is
    added with the "proposed" status and found to be harmful, then it should be changed to
    "discontinued" status.</p>

    <p>Anyone can change the status at any time, but should only do so in accordance with the
    definitions above.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>Conformance checkers must use the information given on the <span data-x="">microformats page
  for existing rel values</span> to establish if a value is allowed or not: values defined in this
  specification or marked as "proposed" or "ratified" must be accepted when used on the elements for
  which they apply as described in the "Effect on..." field, whereas values marked as "discontinued"
  or not listed in either this specification or on the aforementioned page must be rejected as
  invalid. Conformance checkers may cache this information (e.g. for performance reasons or to avoid
  the use of unreliable network connectivity).</p>

  <p>When an author uses a new type not defined by either this specification or the wiki page,
  conformance checkers should offer to add the value to the wiki, with the details described above,
  with the "proposed" status.</p>

  </div>

  <p>Types defined as extensions in the <a
  href="https://microformats.org/wiki/existing-rel-values#HTML5_link_type_extensions">microformats
  page for existing rel values</a> with the status "proposed" or "ratified" may be used with the
  <code data-x="">rel</code> attribute on <code>link</code>, <code>a</code>, and <code>area</code>
  elements in accordance to the "Effect on..." field. <ref>MFREL</ref></p>



  <h3 split-filename="edits">Edits</h3>

  <p>The <code>ins</code> and <code>del</code> elements represent edits to the document.</p>


  <h4>The <dfn element><code>ins</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Transparent</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-mod-cite">cite</code></dd>
   <dd><code data-x="attr-mod-datetime">datetime</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-ins">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-ins">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLModElement</code>.</dd>
  </dl>

  <p>The <code>ins</code> element <span>represents</span> an addition to the document.</p>

  <div class="example">

   <p>The following represents the addition of a single paragraph:</p>

   <pre><code class="html">&lt;aside>
 &lt;ins>
  &lt;p> I like fruit. &lt;/p>
 &lt;/ins>
&lt;/aside></code></pre>

   <p>As does the following, because everything in the <code>aside</code> element here counts as
   <span>phrasing content</span> and therefore there is just one <span>paragraph</span>:</p>

   <pre><code class="html">&lt;aside>
 &lt;ins>
  Apples are &lt;em>tasty&lt;/em>.
 &lt;/ins>
 &lt;ins>
  So are pears.
 &lt;/ins>
&lt;/aside></code></pre>

  </div>

  <p><code>ins</code> elements should not cross <span data-x="paragraph">implied paragraph</span>
  boundaries.</p>

  <div class="example">

   <p>The following example represents the addition of two paragraphs, the second of which was
   inserted in two parts. The first <code>ins</code> element in this example thus crosses a
   paragraph boundary, which is considered poor form.</p>

   <pre class="bad"><code class="html">&lt;aside>
 &lt;!-- don't do this -->
 &lt;ins datetime="2005-03-16 00:00Z">
  &lt;p> I like fruit. &lt;/p>
  Apples are &lt;em>tasty&lt;/em>.
 &lt;/ins>
 &lt;ins datetime="2007-12-19 00:00Z">
  So are pears.
 &lt;/ins>
&lt;/aside></code></pre>

   <p>Here is a better way of marking this up. It uses more elements, but none of the elements cross
   implied paragraph boundaries.</p>

   <pre><code class="html">&lt;aside>
 &lt;ins datetime="2005-03-16 00:00Z">
  &lt;p> I like fruit. &lt;/p>
 &lt;/ins>
 &lt;ins datetime="2005-03-16 00:00Z">
  Apples are &lt;em>tasty&lt;/em>.
 &lt;/ins>
 &lt;ins datetime="2007-12-19 00:00Z">
  So are pears.
 &lt;/ins>
&lt;/aside></code></pre>

   <!-- Those dates aren't random. They're the start and end of something. Can you guess what? -->

  </div>


  <h4>The <dfn element><code>del</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Transparent</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-mod-cite">cite</code></dd>
   <dd><code data-x="attr-mod-datetime">datetime</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-del">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-del">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLModElement</code>.</dd>
  </dl>

  <p>The <code>del</code> element <span>represents</span> a removal from the document.</p>

  <p><code>del</code> elements should not cross <span data-x="paragraph">implied paragraph</span>
  boundaries.</p>

  <div class="example">

   <p>The following shows a "to do" list where items that have been done are crossed-off with the
   date and time of their completion.</p>

   <pre><code class="html">&lt;h1>To Do&lt;/h1>
&lt;ul>
 &lt;li>Empty the dishwasher&lt;/li>
 &lt;li>&lt;del datetime="2009-10-11T01:25-07:00">Watch Walter Lewin's lectures&lt;/del>&lt;/li>
 &lt;li>&lt;del datetime="2009-10-10T23:38-07:00">Download more tracks&lt;/del>&lt;/li>
 &lt;li>Buy a printer&lt;/li>
&lt;/ul></code></pre>

  </div>



  <h4>Attributes common to <code>ins</code> and <code>del</code> elements</h4>

  <p>The <dfn element-attr for="ins,del"><code data-x="attr-mod-cite">cite</code></dfn> attribute
  may be used to specify the <span data-x="concept-document-url">URL</span> of a document that
  explains the change. When that document is long, for instance the minutes of a meeting, authors
  are encouraged to include a <span data-x="concept-url-fragment">fragment</span> pointing to the
  specific part of that document that discusses the change.</p>

  <p>If the <code data-x="attr-mod-cite">cite</code> attribute is present, it must be a <span>valid
  URL potentially surrounded by spaces</span> that explains the change. <span w-nodev>To obtain
  the corresponding citation link, the value of the attribute must be <span data-x="encoding-parsing
  a URL">parsed</span> relative to the element's <span>node document</span>.</span> User agents may
  allow users to follow such citation links, but they are primarily intended for private use (e.g.,
  by server-side scripts collecting statistics about a site's edits), not for readers.</p>


  <p>The <dfn element-attr for="ins,del"><code data-x="attr-mod-datetime">datetime</code></dfn>
  attribute may be used to specify the time and date of the change.</p>

  <p>If present, the <code data-x="attr-mod-datetime">datetime</code> attribute's value must be a
  <span>valid date string with optional time</span>.</p>

  <div w-nodev>

  <p>User agents must parse the <code data-x="attr-mod-datetime">datetime</code> attribute according
  to the <span>parse a date or time string</span> algorithm. If that doesn't return a <span
  data-x="concept-date">date</span> or a <span data-x="concept-datetime">global date and time</span>,
  then the modification has no associated timestamp (the value is non-conforming; it is not a
  <span>valid date string with optional time</span>). Otherwise, the modification is marked as
  having been made at the given <span data-x="concept-date">date</span> or <span
  data-x="concept-datetime">global date and time</span>. If the given value is a <span
  data-x="concept-datetime">global date and time</span> then user agents should use the associated
  time-zone offset information to determine which time zone to present the given datetime in.</p>

  </div>

  <p>This value <span w-nodev>may be shown to the user, but it</span> is primarily intended for
  private use.</p>

  <div w-nodev>

  <p>The <code>ins</code> and <code>del</code> elements must implement the
  <code>HTMLModElement</code> interface:</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLModElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute USVString <span data-x="dom-mod-cite">cite</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-mod-dateTime">dateTime</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLModElement"><code data-x="dom-mod-cite">cite</code></dfn> IDL
  attribute must <span>reflect</span> the element's <code data-x="attr-mod-cite">cite</code> content
  attribute. The <dfn attribute for="HTMLModElement"><code
  data-x="dom-mod-dateTime">dateTime</code></dfn> IDL attribute must <span>reflect</span> the
  element's <code data-x="attr-mod-datetime">datetime</code> content attribute.</p>

  </div>



  <h4>Edits and paragraphs</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>Since the <code>ins</code> and <code>del</code> elements do not affect <span
  data-x="paragraph">paragraphing</span>, it is possible, in some cases where paragraphs are <span
  data-x="paragraph">implied</span> (without explicit <code>p</code> elements), for an
  <code>ins</code> or <code>del</code> element to span both an entire paragraph or other
  non-<span>phrasing content</span> elements and part of another paragraph. For example:</p>

  <pre><code class="html">&lt;section>
 &lt;ins>
  &lt;p>
   This is a paragraph that was inserted.
  &lt;/p>
  This is another paragraph whose first sentence was inserted
  at the same time as the paragraph above.
 &lt;/ins>
 This is a second sentence, which was there all along.
&lt;/section></code></pre>

  <p>By only wrapping some paragraphs in <code>p</code> elements, one can even get the end of one
  paragraph, a whole second paragraph, and the start of a third paragraph to be covered by the same
  <code>ins</code> or <code>del</code> element (though this is very confusing, and not considered
  good practice):</p>

  <pre class="bad"><code class="html">&lt;section>
 This is the first paragraph. &lt;ins>This sentence was
 inserted.
 &lt;p>This second paragraph was inserted.&lt;/p>
 This sentence was inserted too.&lt;/ins> This is the
 third paragraph in this example.
 &lt;!-- (don't do this) -->
&lt;/section></code></pre>

  <p>However, due to the way <span data-x="paragraph">implied paragraphs</span> are defined, it is
  not possible to mark up the end of one paragraph and the start of the very next one using the same
  <code>ins</code> or <code>del</code> element. You instead have to use one (or two) <code>p</code>
  element(s) and two <code>ins</code> or <code>del</code> elements, as for example:</p>

  <pre><code class="html">&lt;section>
 &lt;p>This is the first paragraph. &lt;del>This sentence was
 deleted.&lt;/del>&lt;/p>
 &lt;p>&lt;del>This sentence was deleted too.&lt;/del> That
 sentence needed a separate &amp;lt;del&amp;gt; element.&lt;/p>
&lt;/section></code></pre>

  <p>Partly because of the confusion described above, authors are strongly encouraged to always mark
  up all paragraphs with the <code>p</code> element, instead of having <code>ins</code> or
  <code>del</code> elements that cross <span data-x="paragraph">implied paragraphs</span>
  boundaries.</p>


  <h4>Edits and lists</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>The content models of the <code>ol</code> and <code>ul</code> elements do not allow
  <code>ins</code> and <code>del</code> elements as children. Lists always represent all their
  items, including items that would otherwise have been marked as deleted.</p>

  <p>To indicate that an item is inserted or deleted, an <code>ins</code> or <code>del</code>
  element can be wrapped around the contents of the <code>li</code> element. To indicate that an
  item has been replaced by another, a single <code>li</code> element can have one or more
  <code>del</code> elements followed by one or more <code>ins</code> elements.</p>

  <div class="example">

   <p>In the following example, a list that started empty had items added and removed from it over
   time. The bits in the example that have been emphasized show the parts that are the "current"
   state of the list. The list item numbers don't take into account the edits, though.</p>

   <pre><code class="html">&lt;h1>Stop-ship bugs&lt;/h1>
&lt;ol>
 &lt;li>&lt;ins datetime="2008-02-12T15:20Z"><em>Bug 225:
 Rain detector doesn't work in snow</em>&lt;/ins>&lt;/li>
 &lt;li>&lt;del datetime="2008-03-01T20:22Z">&lt;ins datetime="2008-02-14T12:02Z">Bug 228:
 Water buffer overflows in April&lt;/ins>&lt;/del>&lt;/li>
 &lt;li>&lt;ins datetime="2008-02-16T13:50Z"><em>Bug 230:
 Water heater doesn't use renewable fuels</em>&lt;/ins>&lt;/li>
 &lt;li>&lt;del datetime="2008-02-20T21:15Z">&lt;ins datetime="2008-02-16T14:25Z">Bug 232:
 Carbon dioxide emissions detected after startup&lt;/ins>&lt;/del>&lt;/li>
&lt;/ol></code></pre>

  </div>

  <div class="example">

   <p>In the following example, a list that started with just fruit was replaced by a list with just
   colors.</p>

   <pre><code class="html">&lt;h1>List of &lt;del>fruits&lt;/del>&lt;ins>colors&lt;/ins>&lt;/h1>
&lt;ul>
 &lt;li>&lt;del>Lime&lt;/del>&lt;ins>Green&lt;/ins>&lt;/li>
 &lt;li>&lt;del>Apple&lt;/del>&lt;/li>
 &lt;li>Orange&lt;/li>
 &lt;li>&lt;del>Pear&lt;/del>&lt;/li>
 &lt;li>&lt;ins>Teal&lt;/ins>&lt;/li>
 &lt;li>&lt;del>Lemon&lt;/del>&lt;ins>Yellow&lt;/ins>&lt;/li>
 &lt;li>Olive&lt;/li>
 &lt;li>&lt;ins>Purple&lt;/ins>&lt;/li>
&lt;/ul></code></pre>

  </div>


  <h4>Edits and tables</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>The elements that form part of the table model have complicated content model requirements that
  do not allow for the <code>ins</code> and <code>del</code> elements, so indicating edits to a
  table can be difficult.</p>

  <p>To indicate that an entire row or an entire column has been added or removed, the entire
  contents of each cell in that row or column can be wrapped in <code>ins</code> or <code>del</code>
  elements (respectively).</p>

  <div class="example">

   <p>Here, a table's row has been added:</p>

   <pre><code class="html">&lt;table>
 &lt;thead>
  &lt;tr> &lt;th> Game name           &lt;th> Game publisher   &lt;th> Verdict
 &lt;tbody>
  &lt;tr> &lt;td> Diablo 2            &lt;td> Blizzard         &lt;td> 8/10
  &lt;tr> &lt;td> Portal              &lt;td> Valve            &lt;td> 10/10
<strong>  &lt;tr> &lt;td> &lt;ins>Portal 2&lt;/ins> &lt;td> &lt;ins>Valve&lt;/ins> &lt;td> &lt;ins>10/10&lt;/ins></strong>
&lt;/table></code></pre>

   <p>Here, a column has been removed (the time at which it was removed is given also, as is a link
   to the page explaining why):</p>

   <pre><code class="html">&lt;table>
 &lt;thead>
  &lt;tr> &lt;th> Game name           &lt;th> Game publisher   &lt;th> <strong>&lt;del cite="/edits/r192" datetime="2011-05-02 14:23Z">Verdict&lt;/del></strong>
 &lt;tbody>
  &lt;tr> &lt;td> Diablo 2            &lt;td> Blizzard         &lt;td> <strong>&lt;del cite="/edits/r192" datetime="2011-05-02 14:23Z">8/10&lt;/del></strong>
  &lt;tr> &lt;td> Portal              &lt;td> Valve            &lt;td> <strong>&lt;del cite="/edits/r192" datetime="2011-05-02 14:23Z">10/10&lt;/del></strong>
  &lt;tr> &lt;td> Portal 2            &lt;td> Valve            &lt;td> <strong>&lt;del cite="/edits/r192" datetime="2011-05-02 14:23Z">10/10&lt;/del></strong>
&lt;/table></code></pre>

  </div>

  <p>Generally speaking, there is no good way to indicate more complicated edits (e.g. that a cell
  was removed, moving all subsequent cells up or to the left).</p>




  <h3 split-filename="embedded-content" id="embedded-content">Embedded content</h3>

  <h4>The <dfn element><code>picture</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Embedded content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>embedded content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>Zero or more <code>source</code> elements, followed by one <code>img</code> element,
   optionally intermixed with <span>script-supporting elements</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-picture">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-picture">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLPictureElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLPictureElement</code>.</dd>
  </dl>

  <p>The <code>picture</code> element is a container
  which provides multiple sources to its contained <code>img</code> element
  to allow authors to declaratively control or give hints to the user agent about which image resource to use,
  based on the screen pixel density, <span>viewport</span> size, image format, and other factors.
  It <span>represents</span> its children.</p>

  <p class="note">The <code>picture</code> element is somewhat different from the similar-looking
  <code>video</code> and <code>audio</code> elements. While all of them contain <code>source</code>
  elements, the <code>source</code> element's <code data-x="attr-source-src">src</code> attribute
  has no meaning when the element is nested within a <code>picture</code> element, and the resource
  selection algorithm is different. Also, the <code>picture</code> element itself does not display
  anything; it merely provides a context for its contained <code>img</code> element that enables it
  to choose from multiple <span data-x="URL">URLs</span>.</p>


  <h4>The <dfn element><code>source</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As a child of a <code>picture</code> element, before the <code>img</code> element.</dd>
   <dd>As a child of a <span>media element</span>, before any <span>flow content</span> or
   <code>track</code> elements.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="concept-content-nothing">Nothing</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-source-type">type</code></dd>
   <dd><code data-x="attr-source-media">media</code></dd>
   <dd><code data-x="attr-source-src">src</code> (in <code>audio</code> or <code>video</code>)</dd>
   <dd><code data-x="attr-source-srcset">srcset</code> (in <code>picture</code>)</dd>
   <dd><code data-x="attr-source-sizes">sizes</code> (in <code>picture</code>)</dd>
   <dd><code data-x="attr-dim-width">width</code> (in <code>picture</code>)</dd>
   <dd><code data-x="attr-dim-height">height</code> (in <code>picture</code>)</dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-source">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-source">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLSourceElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute USVString <span data-x="dom-source-src">src</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-source-type">type</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-source-srcset">srcset</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-source-sizes">sizes</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-source-media">media</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-dim-width">width</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-dim-height">height</span>;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLSourceElement</code>.</dd>
  </dl>

  <p>The <code>source</code> element allows authors to specify multiple alternative
  <span data-x="source set">source sets</span> for <code>img</code> elements or multiple alternative
  <span data-x="media resource">media resources</span> for <span data-x="media element">media
  elements</span>. It does not <span data-x="represents">represent</span> anything on its own.</p>

  <p>The <dfn element-attr for="source" data-x="attr-source-type"><code>type</code></dfn> attribute
  may be present. If present, the value must be a <span>valid MIME type string</span>.</p>

  <p>The <dfn element-attr for="source" data-x="attr-source-media"><code>media</code></dfn>
  attribute may also be present. If present, the value must contain a <span>valid media query
  list</span>. The user agent will skip to the next <code>source</code> element if the value does
  not <span data-x="matches the environment">match the environment</span>.</p>

  <p class="note">The <code data-x="attr-source-media">media</code> attribute is only evaluated
  once during the <span data-x="concept-media-load-algorithm">resource selection algorithm</span>
  for <span data-x="media element">media elements</span>. In contrast, when using the
  <code>picture</code> element, the user agent will <a href="#img-environment-changes">react to
  changes in the environment</a>.</p>

  <p>The remainder of the requirements depend on whether the parent is a <code>picture</code>
  element or a <span>media element</span>:</p>

  <dl class="switch">
   <dt>The <code>source</code> element's parent is a <code>picture</code> element</dt>

   <dd>
    <p>The <dfn element-attr for="source" data-x="attr-source-srcset"><code>srcset</code></dfn>
    attribute must be present, and is a <span>srcset attribute</span>.</p>

    <p>The <code data-x="attr-source-srcset">srcset</code> attribute contributes the <span
    data-x="image source">image sources</span> to the <span>source set</span>, if the
    <code>source</code> element is selected.</p>

    <p>If the <code data-x="attr-source-srcset">srcset</code> attribute has any <span data-x="image
    candidate string">image candidate strings</span> using a <span>width descriptor</span>, the
    <code data-x="attr-source-sizes">sizes</code> attribute may also be present. If, additionally,
    the following sibling <code>img</code> element does not <span data-x="allows auto-sizes">allow
    auto-sizes</span>, the <code data-x="attr-source-sizes">sizes</code> attribute must be present.
    The <dfn element-attr for="source" data-x="attr-source-sizes"><code>sizes</code></dfn>
    attribute is a <span>sizes attribute</span>, which contributes the <span>source size</span> to
    the <span>source set</span>, if the <code>source</code> element is selected.</p>

    <p class="note">If the <code>img</code> element <span>allows auto-sizes</span>, then the <code
    data-x="attr-source-sizes">sizes</code> attribute can be omitted on previous sibling
    <code>source</code> elements. In such cases, it is equivalent to specifying <code
    data-x="valdef-sizes-auto">auto</code>.</p>

    <p>The <code>source</code> element supports <span>dimension attributes</span>. The
    <code>img</code> element can use the <code data-x="attr-dim-width">width</code> and <code
    data-x="attr-dim-height">height</code> attributes of a <code>source</code> element, instead of
    those on the <code>img</code> element itself, to determine its rendered dimensions and
    aspect-ratio, <a href="#dimRendering">as defined in the Rendering section</a>.</p>

    <p>The <code data-x="attr-source-type">type</code> attribute gives the type of the images in the
    <span>source set</span>, to allow the user agent to skip to the next <code>source</code> element
    if it does not support the given type.</p>

    <p class="note">If the <code data-x="attr-source-type">type</code> attribute is <em>not</em>
    specified, the user agent will not select a different <code>source</code> element if it finds
    that it does not support the image format after fetching it.</p>

    <p>When a <code>source</code> element has a following sibling <code>source</code> element or
    <code>img</code> element with a <code data-x="attr-img-srcset">srcset</code> attribute
    specified, it must have at least one of the following:</p>

    <ul>
     <li><p>A <code data-x="attr-source-media">media</code> attribute specified with a value that,
     after <span data-x="strip leading and trailing ASCII whitespace">stripping leading and trailing
     ASCII whitespace</span>, is not the empty string and is not an <span>ASCII
     case-insensitive</span> match for the string "<code data-x="">all</code>".</p></li>

     <li><p>A <code data-x="attr-source-type">type</code> attribute specified.</p></li>
    </ul>

    <p>The <code data-x="attr-source-src">src</code> attribute must not be present.</p>
   </dd>

   <dt>The <code>source</code> element's parent is a <span>media element</span></dt>

   <dd>
    <p>The <dfn element-attr for="source"><code data-x="attr-source-src">src</code></dfn> attribute
    gives the <span>URL</span> of the <span>media resource</span>. The value must be a <span>valid
    non-empty URL potentially surrounded by spaces</span>. This attribute must be present.</p>

    <p>The <code data-x="attr-source-type">type</code> attribute gives the type of the <span>media
    resource</span>, to help the user agent determine if it can play this <span>media
    resource</span> before fetching it. The <code data-x="">codecs</code> parameter, which certain
    MIME types define, might be necessary to specify exactly how the resource is encoded.
    <ref>RFC6381</ref></p>

    <p class="note">Dynamically modifying a <code>source</code> element's <code
    data-x="attr-source-src">src</code> or <code data-x="attr-source-type">type</code> attribute
    when the element is already inserted in a <code>video</code> or <code>audio</code> element will
    have no effect. To change what is playing, just use the <code data-x="attr-media-src">src</code>
    attribute on the <span>media element</span> directly, possibly making use of the <code
    data-x="dom-navigator-canPlayType">canPlayType()</code> method to pick from amongst available
    resources. Generally, manipulating <code>source</code> elements manually after the document has
    been parsed is an unnecessarily complicated approach.</p>

    <div class="example">

     <p>The following list shows some examples of how to use the <code data-x="">codecs=</code> MIME
     parameter in the <code data-x="attr-source-type">type</code> attribute.</p>

     <dl>
      <dt>H.264 Constrained baseline profile video (main and extended video compatible) level 3 and Low-Complexity AAC audio in MP4 container</dt>
      <dd><pre><code class="html">&lt;source src='video.mp4' type='video/mp4; codecs="avc1.42E01E, mp4a.40.2"'></code></pre></dd>

      <dt>H.264 Extended profile video (baseline-compatible) level 3 and Low-Complexity AAC audio in MP4 container</dt>
      <dd><pre><code class="html">&lt;source src='video.mp4' type='video/mp4; codecs="avc1.58A01E, mp4a.40.2"'></code></pre></dd>

      <dt>H.264 Main profile video level 3 and Low-Complexity AAC audio in MP4 container</dt>
      <dd><pre><code class="html">&lt;source src='video.mp4' type='video/mp4; codecs="avc1.4D401E, mp4a.40.2"'></code></pre></dd>

      <dt>H.264 'High' profile video (incompatible with main, baseline, or extended profiles) level 3 and Low-Complexity AAC audio in MP4 container</dt>
      <dd><pre><code class="html">&lt;source src='video.mp4' type='video/mp4; codecs="avc1.64001E, mp4a.40.2"'></code></pre></dd>


      <dt>MPEG-4 Visual Simple Profile Level 0 video and Low-Complexity AAC audio in MP4 container</dt>
      <dd><pre><code class="html">&lt;source src='video.mp4' type='video/mp4; codecs="mp4v.20.8, mp4a.40.2"'></code></pre></dd>

      <dt>MPEG-4 Advanced Simple Profile Level 0 video and Low-Complexity AAC audio in MP4 container</dt>
      <dd><pre><code class="html">&lt;source src='video.mp4' type='video/mp4; codecs="mp4v.20.240, mp4a.40.2"'></code></pre></dd>

      <dt>MPEG-4 Visual Simple Profile Level 0 video and AMR audio in 3GPP container</dt>
      <dd><pre><code class="html">&lt;source src='video.3gp' type='video/3gpp; codecs="mp4v.20.8, samr"'></code></pre></dd>


      <dt>Theora video and Vorbis audio in Ogg container</dt>
      <dd><pre><code class="html">&lt;source src='video.ogv' type='video/ogg; codecs="theora, vorbis"'></code></pre></dd>

      <dt>Theora video and Speex audio in Ogg container</dt>
      <dd><pre><code class="html">&lt;source src='video.ogv' type='video/ogg; codecs="theora, speex"'></code></pre></dd>

      <dt>Vorbis audio alone in Ogg container</dt>
      <dd><pre><code class="html">&lt;source src='audio.ogg' type='audio/ogg; codecs=vorbis'></code></pre></dd>

      <dt>Speex audio alone in Ogg container</dt>
      <dd><pre><code class="html">&lt;source src='audio.spx' type='audio/ogg; codecs=speex'></code></pre></dd>

      <dt>FLAC audio alone in Ogg container</dt>
      <dd><pre><code class="html">&lt;source src='audio.oga' type='audio/ogg; codecs=flac'></code></pre></dd>

      <dt>Dirac video and Vorbis audio in Ogg container</dt>
      <dd><pre><code class="html">&lt;source src='video.ogv' type='video/ogg; codecs="dirac, vorbis"'></code></pre></dd>

<!-- awaiting definition by the Matroska guys:
      <dt>Theora video and Vorbis audio in Matroska container</dt>
      <dd><pre><code class="html">&lt;source src='video.mkv' type='video/x-matroska; codecs="theora, vorbis"'></code></pre></dd>
-->

<!-- awaiting definition by the Ogg or BBC guys:
      <dt>Dirac video and Vorbis audio in Matroska container</dt>
      <dd><pre><code class="html">&lt;source src='video.mkv' type='video/x-matroska; codecs='></code></pre></dd>
-->

<!-- awaiting definition by the Microsoft guys:
      <dt>WMV9 video and WMA 2 audio in ASF container</dt>
      <dd><pre><code class="html">&lt;source src='video.wmv' type='video/x-ms-wmv; codecs='></code></pre></dd>

      <dt>WMV8 video and WMA 2 audio in ASF container</dt>
      <dd><pre><code class="html">&lt;source src='video.wmv' type='video/x-ms-wmv; codecs='></code></pre></dd>

      <dt>VC-1 video and WMA 10 Pro audio in ASF container</dt>
      <dd><pre><code class="html">&lt;source src='video.wmv' type='video/x-ms-wmv; codecs='></code></pre></dd>

      <dt>XviD video and MP3 audio in AVI container</dt>
      <dd><pre><code class="html">&lt;source src='video.avi' type='video/x-msvideo; codecs='></code></pre></dd>

      <dt>Motion-JPEG video and uncompressed PCM audio in AVI container</dt>
      <dd><pre><code class="html">&lt;source src='video.avi' type='video/x-msvideo; codecs='></code></pre></dd>
-->

<!-- awaiting definition by Real:
      <dt>Real Video 10 video and High-Efficiency AAC audio in Real Media container</dt>
      <dd><pre><code class="html">&lt;source src='video.rm' type='application/vnd.rn-realmedia; codecs='></code></pre></dd>
-->

<!--  undefined:
      <dt>MPEG-1 video and MPEG-1 Audio Layer II audio in MPEG-1 program stream</dt>
      <dd><pre><code class="html">&lt;source src='video.mpg' type='video/mpeg; codecs='></code></pre></dd>
-->
     </dl>
    </div>

    <p>The <code data-x="attr-source-srcset">srcset</code> and <code
    data-x="attr-source-sizes">sizes</code> attributes must not be present.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <code>source</code> <span>HTML element insertion steps</span>, given
  <var>insertedNode</var>, are:</p>

  <ol>
   <li><p>Let <var>parent</var> be <var>insertedNode</var>'s <span>parent</span>.</p></li>

   <li><p>If <var>parent</var> is a <span>media element</span> that has no <code
   data-x="attr-media-src">src</code> attribute and whose <code
   data-x="dom-media-networkState">networkState</code> has the value <code
   data-x="dom-media-NETWORK_EMPTY">NETWORK_EMPTY</code>, then invoke that <span>media
   element</span>'s <span data-x="concept-media-load-algorithm">resource selection
   algorithm</span>.</p></li>

   <li><p>If <var>parent</var> is a <code>picture</code> element, then <span data-x="list
   iterate">for each</span> <var>child</var> of <var>parent</var>'s <span
   data-x="concept-tree-child">children</span>, if <var>child</var> is an <code>img</code> element,
   then count this as a <span data-x="relevant mutations">relevant mutation</span> for
   <var>child</var>.</p></li>
  </ol>

  <p>The <code>source</code> <span>HTML element moving steps</span>, given <var>movedNode</var> and
  <var>oldParent</var>, are:</p>

  <ol>
   <li><p>If <var>oldParent</var> is a <code>picture</code> element, then <span data-x="list
   iterate">for each</span> <var>child</var> of <var>oldParent</var>'s <span
   data-x="concept-tree-child">children</span>, if <var>child</var> is an <code>img</code> element,
   then count this as a <span data-x="relevant mutations">relevant mutation</span> for
   <var>child</var>.</p></li>
  </ol>

  <p>The <code>source</code> <span>HTML element removing steps</span>, given <var>removedNode</var>
  and <var>oldParent</var>, are:</p>

  <ol>
   <li><p>If <var>oldParent</var> is a <code>picture</code> element, then <span data-x="list
   iterate">for each</span> <var>child</var> of <var>oldParent</var>'s <span
   data-x="concept-tree-child">children</span>, if <var>child</var> is an <code>img</code> element,
   then count this as a <span data-x="relevant mutations">relevant mutation</span> for
   <var>child</var>.</p></li>
  </ol>

  <p>The IDL attributes <dfn attribute for="HTMLSourceElement"><code
  data-x="dom-source-src">src</code></dfn>, <dfn attribute for="HTMLSourceElement"><code
  data-x="dom-source-type">type</code></dfn>, <dfn attribute for="HTMLSourceElement"
  data-x="dom-source-srcset"><code>srcset</code></dfn>, <dfn attribute for="HTMLSourceElement"
  data-x="dom-source-sizes"><code>sizes</code></dfn>, and <dfn attribute for="HTMLSourceElement"
  data-x="dom-source-media"><code>media</code></dfn> must <span>reflect</span> the respective
  content attributes of the same name.</p>

  </div>

  <div class="example">
   <p>If the author isn't sure if user agents will all be able to render the media resources
   provided, the author can listen to the <code data-x="event-error">error</code> event on the last
   <code>source</code> element and trigger fallback behavior:</p>

   <pre><code class="html">&lt;script>
 function fallback(video) {
   // replace &lt;video> with its contents
   while (video.hasChildNodes()) {
     if (video.firstChild instanceof HTMLSourceElement)
       video.removeChild(video.firstChild);
     else
       video.parentNode.insertBefore(video.firstChild, video);
   }
   video.parentNode.removeChild(video);
 }
&lt;/script>
&lt;video controls autoplay>
 &lt;source src='video.mp4' type='video/mp4; codecs="avc1.42E01E, mp4a.40.2"'>
 &lt;source src='video.ogv' type='video/ogg; codecs="theora, vorbis"'
         onerror="fallback(parentNode)">
 ...
&lt;/video></code></pre>
  </div>



  <h4>The <dfn element><code>img</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Embedded content</span>.</dd>
   <dd><span>Form-associated element</span>.</dd>
   <dd>If the element has a <code data-x="attr-hyperlink-usemap">usemap</code> attribute: <span>Interactive content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>embedded content</span> is expected.</dd>
   <dd>As a child of a <code>picture</code> element, after all <code>source</code> elements.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="concept-content-nothing">Nothing</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-img-alt">alt</code></dd>
   <dd><code data-x="attr-img-src">src</code></dd>
   <dd><code data-x="attr-img-srcset">srcset</code></dd>
   <dd><code data-x="attr-img-sizes">sizes</code></dd>
   <dd><code data-x="attr-img-crossorigin">crossorigin</code></dd>
   <dd><code data-x="attr-hyperlink-usemap">usemap</code></dd>
   <dd><code data-x="attr-img-ismap">ismap</code></dd>
   <dd><code data-x="attr-dim-width">width</code></dd>
   <dd><code data-x="attr-dim-height">height</code></dd>
   <dd><code data-x="attr-img-referrerpolicy">referrerpolicy</code></dd>
   <dd><code data-x="attr-img-decoding">decoding</code></dd>
   <dd><code data-x="attr-img-loading">loading</code></dd>
   <dd><code data-x="attr-img-fetchpriority">fetchpriority</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd>If the element has a non-empty <code data-x="attr-img-alt">alt</code> attribute: <a
   href="https://w3c.github.io/html-aria/#el-img">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-img">for implementers</a>.</dd>
   <dd>Otherwise: <a
   href="https://w3c.github.io/html-aria/#el-img-empty-alt">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-img-empty-alt">for implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window,
 <span>LegacyFactoryFunction</span>=<span data-x="dom-image">Image</span>(optional unsigned long width, optional unsigned long height)]
interface <dfn interface>HTMLImageElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-img-alt">alt</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-img-src">src</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-img-srcset">srcset</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-img-sizes">sizes</span>;
  [<span>CEReactions</span>] attribute DOMString? <span data-x="dom-img-crossOrigin">crossOrigin</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-img-useMap">useMap</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-img-isMap">isMap</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-img-width">width</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-img-height">height</span>;
  readonly attribute unsigned long <span data-x="dom-img-naturalWidth">naturalWidth</span>;
  readonly attribute unsigned long <span data-x="dom-img-naturalHeight">naturalHeight</span>;
  readonly attribute boolean <span data-x="dom-img-complete">complete</span>;
  readonly attribute USVString <span data-x="dom-img-currentSrc">currentSrc</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-img-referrerPolicy">referrerPolicy</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-img-decoding">decoding</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-img-loading">loading</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-img-fetchPriority">fetchPriority</span>;

  <span data-x="idl-Promise">Promise</span>&lt;undefined&gt; <span data-x="dom-img-decode">decode</span>();

  // <a href="#HTMLImageElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLImageElement</code>.</dd>
  </dl>

  <p>An <code>img</code> element represents an image.</p>

  <p>An <code>img</code> element has a <dfn data-x="concept-img-dimension-attribute-source">dimension
  attribute source</dfn>, initially set to the element itself.</p>

  <!-- v2 ideas for <img>:

     * Maps sites would like to know which images are already cached, so that they can use images
       that are vaguely suitable while they wait for the most appropriate image to download.

       Almost like lowsrc="", except that many images might be appropriate.

       Slight hitch: their images are at a different origin, and we don't want to allow arbitrary
       cross-origin inspection (privacy leak risk). So it will require them to do CORS opt-in.

     * See note at rel=noreferrer.

  -->

  <p>The image given by the <dfn element-attr for="img" data-x="attr-img-src"><code>src</code></dfn>
  and <dfn data-x="attr-img-srcset" element-attr for="img"><code>srcset</code></dfn> attributes, and
  any previous sibling <code>source</code> elements' <code data-x="attr-source-srcset">srcset</code>
  attributes if the parent is a <code>picture</code> element, is the embedded content; the value of
  the <dfn element-attr for="img" data-x="attr-img-alt"><code>alt</code></dfn> attribute provides
  equivalent content for those who cannot process images or who have image loading disabled (i.e. it
  is the <code>img</code> element's <span>fallback content</span>).</p>

  <p>The requirements on the <code data-x="attr-img-alt">alt</code> attribute's value are described
  <a href="#alt">in a separate section</a>.</p>

  <p>The <code data-x="attr-img-src">src</code> attribute must be present, and must contain a
  <span>valid non-empty URL potentially surrounded by spaces</span> referencing a non-interactive,
  optionally animated, image resource that is neither paged nor scripted.</p>

  <p class="note">The requirements above imply that images can be static bitmaps (e.g. PNGs, GIFs,
  JPEGs), single-page vector documents (single-page PDFs, XML files with an SVG document element),
  animated bitmaps (APNGs, animated GIFs), animated vector graphics (XML files with an SVG
  <span>document element</span> that use declarative SMIL animation), and so forth. However, these
  definitions preclude SVG files with script, multipage PDF files, interactive MNG files, HTML
  documents, plain text documents, and the like. <ref>PNG</ref> <ref>GIF</ref> <ref>JPEG</ref>
  <ref>PDF</ref> <ref>XML</ref> <ref>APNG</ref> <ref>SVG</ref> <ref>MNG</ref> </p>

  <p>The <code data-x="attr-img-srcset">srcset</code> attribute may also be present, and is a
  <span>srcset attribute</span>.</p>

  <p>The <code data-x="attr-img-srcset">srcset</code> attribute and the <code
  data-x="attr-img-src">src</code> attribute (if <span data-x="width descriptor">width
  descriptors</span> are not used) contribute the <span data-x="image source">image sources</span>
  to the <span>source set</span> (if no <code>source</code> element was selected).</p>

  <p>If the <code data-x="attr-img-srcset">srcset</code> attribute is present and has any <span
  data-x="image candidate string">image candidate strings</span> using a <span>width
  descriptor</span>, the <code data-x="attr-img-sizes">sizes</code> attribute must also be present.
  If the <code data-x="attr-img-srcset">srcset</code> attribute is <em>not</em> specified, and the
  <code data-x="attr-img-loading">loading</code> attribute is in the <span
  data-x="attr-loading-lazy-state">Lazy</span> state, the <code
  data-x="attr-img-sizes">sizes</code> attribute may be specified with the value "<code
  data-x="">auto</code>" (<span>ASCII case-insensitive</span>). The <dfn element-attr for="img"
  data-x="attr-img-sizes"><code>sizes</code></dfn> attribute is a <span>sizes attribute</span>,
  which contributes the <span>source size</span> to the <span>source set</span> (if no
  <code>source</code> element was selected).</p>

  <p>An <code>img</code> element <dfn>allows auto-sizes</dfn> if:</p>

  <ul>
   <li>its <code data-x="attr-img-loading">loading</code> attribute is in the <span
  data-x="attr-loading-lazy-state">Lazy</span> state, and</li>

   <li>its <code data-x="attr-img-sizes">sizes</code> attribute's value is "<code
   data-x="">auto</code>" (<span>ASCII case-insensitive</span>), or starts with "<code
   data-x="">auto,</code>" (<span>ASCII case-insensitive</span>).</li>
  </ul>

  <p>The <dfn element-attr for="img" data-x="attr-img-crossorigin"><code>crossorigin</code></dfn>
  attribute is a <span>CORS settings attribute</span>. Its purpose is to allow images from
  third-party sites that allow cross-origin access to be used with <code>canvas</code>.</p>

  <p>The <dfn element-attr for="img"
  data-x="attr-img-referrerpolicy"><code>referrerpolicy</code></dfn> attribute is a <span>referrer
  policy attribute</span>. Its purpose is to set the <span>referrer policy</span> used when <span
  data-x="concept-fetch">fetching</span> the image. <ref>REFERRERPOLICY</ref></p>

  <p>The <dfn element-attr for="img" data-x="attr-img-decoding"><code>decoding</code></dfn>
  attribute indicates the preferred method to <span data-x="img-decoding-process">decode</span> this
  image. The attribute, if present, must be an <span>image decoding hint</span>. This attribute's <i
  data-x="missing value default">missing value default</i> and <i data-x="invalid value
  default">invalid value default</i> are both the <span
  data-x="attr-img-decoding-auto-state">auto</span> state.</p>

  <p>The <dfn element-attr for="img"
  data-x="attr-img-fetchpriority"><code>fetchpriority</code></dfn> attribute is a <span>fetch
  priority attribute</span>. Its purpose is to set the <span
  data-x="concept-request-priority">priority</span> used when <span
  data-x="concept-fetch">fetching</span> the image.</p>

  <p>The <dfn element-attr for="img" data-x="attr-img-loading"><code>loading</code></dfn> attribute is a <span>lazy
  loading attribute</span>. Its purpose is to indicate the policy for loading images that are
  outside the viewport.</p>

  <p>When the <code data-x="attr-img-loading">loading</code> attribute's state is changed to the
  <span data-x="attr-loading-eager-state">Eager</span> state, the user agent must run these
  steps:</p>

  <ol>
   <li><p>Let <var>resumptionSteps</var> be the <code>img</code> element's <span>lazy load
   resumption steps</span>.</p></li>

   <li><p>If <var>resumptionSteps</var> is null, then return.</p></li>

   <li><p>Set the <code>img</code>'s <span>lazy load resumption steps</span> to null.</p></li>

   <li><p>Invoke <var>resumptionSteps</var>.</p></li>
  </ol>

  <div class="example">
<pre><code class="html">&lt;img src="1.jpeg" alt="1">
&lt;img src="2.jpeg" loading=eager alt="2">
&lt;img src="3.jpeg" loading=lazy alt="3">
&lt;div id=very-large>&lt;/div> &lt;!-- Everything after this div is below the viewport -->
&lt;img src="4.jpeg" alt="4">
&lt;img src="5.jpeg" loading=lazy alt="5"></code></pre>

   <p>In the example above, the images load as follows:</p>

   <dl class="switch">
    <dt><code data-x="">1.jpeg</code>, <code data-x="">2.jpeg</code>,
    <code data-x="">4.jpeg</code></dt>
    <dd><p>The images load eagerly and delay the window's load event.</p></dd>

    <dt><code data-x="">3.jpeg</code></dt>
    <dd><p>The image loads when layout is known, due to being in the viewport, however it does not
    delay the window's load event.</p></dd>

    <dt><code data-x="">5.jpeg</code></dt>
    <dd><p>The image loads only once scrolled into the viewport, and does not delay the window's
    load event.</p></dd>
   </dl>

   <p class="note">Developers are encouraged to specify a preferred aspect ratio via <code
   data-x="attr-dim-width">width</code> and <code data-x="attr-dim-height">height</code> attributes
   on lazy loaded images, even if CSS sets the image's width and height properties, to prevent the
   page layout from shifting around after the image loads.</p>
  </div>

  <p>The <code>img</code> <span>HTML element insertion steps</span>, given <var>insertedNode</var>,
  are:</p>

  <ol>
   <li><p>If <var>insertedNode</var>'s parent is a <code>picture</code> element, then, count this as
   a <span data-x="relevant mutations">relevant mutation</span> for
   <var>insertedNode</var>.</p></li>
  </ol>

  <p>The <code>img</code> <span>HTML element moving steps</span>, given <var>movedNode</var> and
  <var>oldParent</var>, are:</p>

  <ol>
   <li><p>If <var>oldParent</var> is a <code>picture</code> element, then, count this as a <span
   data-x="relevant mutations">relevant mutation</span> for <var>movedNode</var>.</p></li>
  </ol>

  <p>The <code>img</code> <span>HTML element removing steps</span>, given <var>removedNode</var> and
  <var>oldParent</var>, are:</p>

  <ol>
   <li><p>If <var>oldParent</var> is a <code>picture</code> element, then, count this as a
   <span data-x="relevant mutations">relevant mutation</span> for <var>removedNode</var>.</p></li>
  </ol>

  <hr>

  <p>The <code>img</code> element must not be used as a layout tool. In particular, <code>img</code>
  elements should not be used to display transparent images, as such images rarely convey meaning and
  rarely add anything useful to the document.</p>

  <div w-nodev>

  <hr>

  <p>What an <code>img</code> element represents depends on the <code
  data-x="attr-img-src">src</code> attribute and the <code data-x="attr-img-alt">alt</code>
  attribute.</p>

  <dl class="switch">

   <dt>If the <code data-x="attr-img-src">src</code> attribute is set and the <code
   data-x="attr-img-alt">alt</code> attribute is set to the empty string</dt>

   <dd>
    <p>The image is either decorative or supplemental to the rest of the content, redundant with
    some other information in the document.</p>

    <p>If the image is <span data-x="img-available">available</span> and the user agent is configured
    to display that image, then the element <span>represents</span> the element's image data.</p>

    <p>Otherwise, the element <span>represents</span> nothing, and may be omitted completely from
    the rendering. User agents may provide the user with a notification that an image is present but
    has been omitted from the rendering.</p>
   </dd>

   <dt>If the <code data-x="attr-img-src">src</code> attribute is set and the <code
   data-x="attr-img-alt">alt</code> attribute is set to a value that isn't empty</dt>

   <dd>
    <p>The image is a key part of the content; the <code data-x="attr-img-alt">alt</code> attribute
    gives a textual equivalent or replacement for the image.</p>

    <p>If the image is <span data-x="img-available">available</span> and the user agent is configured
    to display that image, then the element <span>represents</span> the element's image data.</p>

    <p>Otherwise, the element <span>represents</span> the text given by the <code
    data-x="attr-img-alt">alt</code> attribute. User agents may provide the user with a notification
    that an image is present but has been omitted from the rendering.</p>
   </dd>

   <dt>If the <code data-x="attr-img-src">src</code> attribute is set and the <code
   data-x="attr-img-alt">alt</code> attribute is not</dt>

   <dd>
    <p>The image might be a key part of the content, and there is no textual equivalent of the image
    available.</p>

    <p class="note">In a conforming document, the absence of the <code
    data-x="attr-img-alt">alt</code> attribute indicates that the image is a key part of the content
    but that a textual replacement for the image was not available when the image was generated.</p>

    <p>If the image is <span data-x="img-available">available</span> and the user agent is configured
    to display that image, then the element <span>represents</span> the element's image data.</p>

    <p>If the image has a <code data-x="attr-img-src">src</code> attribute whose value is
    the empty string, then the element <span>represents</span> nothing.</p>

    <p>Otherwise, the user agent should display some sort of indicator that there is an image that
    is not being rendered, and may, if requested by the user, or if so configured, or when required
    to provide contextual information in response to navigation, provide caption information for the
    image, derived as follows:</p>

    <ol>
     <!-- when editing this list, search for the two other occurrences of 'critical-no-alt' -->

     <!-- NOTE: the order of these steps is important; it's intended to make the innermost caption
     override the outer ones -->

     <li><p>If the image has a <code data-x="attr-title">title</code> attribute whose value is not
     the empty string, then return the value of that attribute.</p></li>

     <li><p>If the image is a descendant of a <code>figure</code> element that has a child
     <code>figcaption</code> element, and, ignoring the <code>figcaption</code> element and its
     descendants, the <code>figure</code> element has no <span>flow content</span> descendants other
     than <span>inter-element whitespace</span> and the <code>img</code> element, then return the
     contents of the first such <code>figcaption</code> element.</p></li>

     <li><p>Return nothing. (There is no caption information.)</p></li>
    </ol>
   </dd>

   <dt>If the <code data-x="attr-img-src">src</code> attribute is not set and either the <code
   data-x="attr-img-alt">alt</code> attribute is set to the empty string or the <code
   data-x="attr-img-alt">alt</code> attribute is not set at all</dt>

   <dd>
    <p>The element <span>represents</span> nothing.</p>
   </dd>

   <dt>Otherwise</dt>

   <dd>
    <p>The element <span>represents</span> the text given by the <code
    data-x="attr-img-alt">alt</code> attribute.</p>
   </dd>
  </dl>

  <p>The <code data-x="attr-img-alt">alt</code> attribute does not represent advisory information.
  User agents must not present the contents of the <code data-x="attr-img-alt">alt</code> attribute
  in the same way as content of the <code data-x="attr-title">title</code> attribute.</p>

  <p>User agents may always provide the user with the option to display any image, or to prevent any
  image from being displayed. User agents may also apply heuristics to help the user make use of the
  image when the user is unable to see it, e.g. due to a visual disability or because they are using
  a text terminal with no graphics capabilities. Such heuristics could include, for instance,
  optical character recognition (OCR) of text found within the image.</p>

  <p class="warning">While user agents are encouraged to repair cases of missing <code
  data-x="attr-img-alt">alt</code> attributes, authors must not rely on such behavior. <a
  href="#alt">Requirements for providing text to act as an alternative for images</a> are described
  in detail below.</p>

  <p>The <em>contents</em> of <code>img</code> elements, if any, are ignored for the purposes of
  rendering.</p>

  </div>

  <hr>

  <p>The <code data-x="attr-hyperlink-usemap">usemap</code> attribute,
  if present, can indicate that the image has an associated
  <span>image map</span>.</p>

  <p>The <dfn element-attr for="img" data-x="attr-img-ismap"><code>ismap</code></dfn> attribute,
  when used on an element that is a descendant of an <code>a</code> element with an <code
  data-x="attr-hyperlink-href">href</code> attribute, indicates by its presence that the element
  provides access to a server-side image map. This affects how events are handled on the
  corresponding <code>a</code> element.</p>

  <p>The <code data-x="attr-img-ismap">ismap</code> attribute is a
  <span>boolean attribute</span>. The attribute must not be specified
  on an element that does not have an ancestor <code>a</code> element
  with an <code data-x="attr-hyperlink-href">href</code> attribute.</p>

  <p class="note">The <code data-x="attr-hyperlink-usemap">usemap</code> and <code
  data-x="attr-img-ismap">ismap</code> attributes can result in confusing behavior when used
  together with <code>source</code> elements with the <code data-x="attr-source-media">media</code>
  attribute specified in a <code>picture</code> element.</p>

  <p>The <code>img</code> element supports <span>dimension
  attributes</span>.</p>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLImageElement" data-x="dom-img-alt"><code>alt</code></dfn>, <dfn
  attribute for="HTMLImageElement" data-x="dom-img-src"><code>src</code></dfn>, <dfn attribute
  for="HTMLImageElement" data-x="dom-img-srcset"><code>srcset</code></dfn>, and <dfn attribute
  for="HTMLImageElement" data-x="dom-img-sizes"><code>sizes</code></dfn> IDL attributes must
  <span>reflect</span> the respective content attributes of the same name.</p>

  <p>The <dfn attribute for="HTMLImageElement"
  data-x="dom-img-crossOrigin"><code>crossOrigin</code></dfn> IDL attribute must
  <span>reflect</span> the <code data-x="attr-img-crossorigin">crossorigin</code> content attribute,
  <span>limited to only known values</span>.</p>

  <p>The <dfn attribute for="HTMLImageElement" data-x="dom-img-useMap"><code>useMap</code></dfn> IDL
  attribute must <span>reflect</span> the <code data-x="attr-hyperlink-usemap">usemap</code> content
  attribute.</p>

  <p>The <dfn attribute for="HTMLImageElement" data-x="dom-img-isMap"><code>isMap</code></dfn> IDL
  attribute must <span>reflect</span> the <code data-x="attr-img-ismap">ismap</code> content
  attribute.</p>

  <p>The <dfn attribute for="HTMLImageElement"><code
  data-x="dom-img-referrerPolicy">referrerPolicy</code></dfn> IDL attribute must
  <span>reflect</span> the <code data-x="attr-img-referrerpolicy">referrerpolicy</code> content
  attribute, <span>limited to only known values</span>.</p>

  <p>The <dfn attribute for="HTMLImageElement"><code data-x="dom-img-decoding">decoding</code></dfn>
  IDL attribute must <span>reflect</span> the <code data-x="attr-img-decoding">decoding</code>
  content attribute, <span>limited to only known values</span>.</p>

  <p>The <dfn attribute for="HTMLImageElement"><code data-x="dom-img-loading">loading</code></dfn>
  IDL attribute must <span>reflect</span> the <code data-x="attr-img-loading">loading</code> content
  attribute, <span>limited to only known values</span>.</p>

  <p>The <dfn attribute for="HTMLImageElement"><code
  data-x="dom-img-fetchPriority">fetchPriority</code></dfn> IDL attribute
  must <span>reflect</span> the <code data-x="attr-img-fetchpriority">fetchpriority</code> content
  attribute, <span>limited to only known values</span>.</p>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>image</var>.<span subdfn data-x="dom-img-width">width</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>image</var>.<span subdfn data-x="dom-img-height">height</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>These attributes return the actual rendered dimensions of the image, or 0 if the dimensions
    are not known.</p>

    <p>They can be set, to change the corresponding content attributes.</p>
   </dd>

   <dt><code data-x=""><var>image</var>.<span subdfn data-x="dom-img-naturalWidth">naturalWidth</span></code></dt>
   <dt><code data-x=""><var>image</var>.<span subdfn data-x="dom-img-naturalHeight">naturalHeight</span></code></dt>

   <dd>
    <p>These attributes return the natural dimensions of the image, or 0 if the dimensions are not
    known.</p>
   </dd>

   <dt><code data-x=""><var>image</var>.<span subdfn data-x="dom-img-complete">complete</span></code></dt>

   <dd>
    <p>Returns true if the image has been completely downloaded or if no image is specified;
    otherwise, returns false.</p>
   </dd>

   <dt><code data-x=""><var>image</var>.<span subdfn data-x="dom-img-currentSrc">currentSrc</span></code></dt>

   <dd><p>Returns the image's <span>absolute URL</span>.</p></dd>

   <dt><code data-x=""><var>image</var>.<span subdfn data-x="dom-img-decode">decode</span>()</code></dt>

   <dd>
    <p>This method causes the user agent to <span data-x="img-decoding-process">decode</span> the
    image <span>in parallel</span>, returning a promise that fulfills when decoding is complete.</p>

    <p>The promise will be rejected with an <span>"<code>EncodingError</code>"</span>
    <code>DOMException</code> if the image cannot be decoded.</p>
   </dd>

   <dt><code data-x=""><var>image</var> = new <span subdfn data-x="dom-image">Image</span>([ <var>width</var> [, <var>height</var> ] ])</code></dt>

   <dd>
    <p>Returns a new <code>img</code> element, with the <code data-x="attr-dim-width">width</code>
    and <code data-x="attr-dim-height">height</code> attributes set to the values passed in the
    relevant arguments, if applicable.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The IDL attributes <dfn attribute for="HTMLImageElement"
  data-x="dom-img-width"><code>width</code></dfn> and <dfn attribute for="HTMLImageElement"
  data-x="dom-img-height"><code>height</code></dfn> must return the rendered width and height of the
  image, in <span data-x="'px'">CSS pixels</span>, if the image is <span>being rendered</span>; or
  else the <span>density-corrected natural width and height</span> of the image, in <span
  data-x="'px'">CSS pixels</span>, if the image has <span>density-corrected natural width and
  height</span> and is <i data-x="img-available">available</i> but is not <span>being
  rendered</span>; or else 0, if the image is not <i data-x="img-available">available</i> or does
  not have <span>density-corrected natural width and height</span>. <ref>CSS</ref></p>

  <p>On setting, they must act as if they <span data-x="reflect">reflected</span> the respective
  content attributes of the same name.</p>

  <p>The IDL attributes <dfn attribute for="HTMLImageElement"
  data-x="dom-img-naturalWidth"><code>naturalWidth</code></dfn> and <dfn attribute
  for="HTMLImageElement" data-x="dom-img-naturalHeight"><code>naturalHeight</code></dfn> must return
  the <span>density-corrected natural width and height</span> of the image, in <span
  data-x="'px'">CSS pixels</span>, if the image has <span>density-corrected natural width and
  height</span> and is <i data-x="img-available">available</i>, or else 0. <ref>CSS</ref></p>

  <p class="note">Since the <span>density-corrected natural width and height</span> of an image
  take into account any orientation specified in its metadata, <code
  data-x="dom-img-naturalWidth">naturalWidth</code> and <code
  data-x="dom-img-naturalHeight">naturalHeight</code> reflect the dimensions after applying any
  rotation needed to correctly orient the image, regardless of the value of the
  <span>'image-orientation'</span> property.</p>

  <p>The <dfn attribute for="HTMLImageElement"
  data-x="dom-img-complete"><code>complete</code></dfn> getter steps are:</p>

  <ol>
   <li>
    <p>If any of the following are true:</p>

    <ul>
     <!--TODO: check picture parent-->

     <li><p>both the <code data-x="attr-img-src">src</code> attribute and the <code
     data-x="attr-img-srcset">srcset</code> attribute are omitted;</p></li>

     <li><p>the <code data-x="attr-img-srcset">srcset</code> attribute is omitted and the <code
     data-x="attr-img-src">src</code> attribute's value is the empty string;</p></li>
     <!-- we only have this hack for src="", not srcset=""; present but empty or bogus srcset=""
     still means complete=false if it's not in the img-error state -->

     <li><p>the <code>img</code> element's <span>current request</span>'s <span
     data-x="img-req-state">state</span> is <span data-x="img-all">completely available</span> and
     its <span>pending request</span> is null; or</p></li>

     <li><p>the <code>img</code> element's <span>current request</span>'s <span
     data-x="img-req-state">state</span> is <span data-x="img-error">broken</span> and its
     <span>pending request</span> is null,</p></li>
    </ul>

    <p>then return true.</p>
   </li>

   <li><p>Return false.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLImageElement"
  data-x="dom-img-currentSrc"><code>currentSrc</code></dfn> IDL attribute must return the
  <code>img</code> element's <span>current request</span>'s <span data-x="img-req-url">current
  URL</span>.</p>

  <p>The <dfn method for="HTMLImageElement" data-x="dom-img-decode"><code>decode()</code></dfn>
  method, when invoked, must perform the following steps:</p>

  <ol>
   <li><p>Let <var>promise</var> be a new promise.</p></li>

   <li>
    <p><span>Queue a microtask</span> to perform the following steps:</p>

    <div class="note">
     <p>This is done because <span data-x="update the image data">updating the image data</span>
     takes place in a microtask as well. Thus, to make code such as</p>

     <pre><code class="js">img.src = "stars.jpg";
img.decode();</code></pre>

     <p>properly decode <code data-x="">stars.jpg</code>, we need to delay any processing by one
     microtask.</p>
    </div>

    <ol>
     <li><p>Let <var>global</var> be <span>this</span>'s <span>relevant global
     object</span>.</p></li>

     <li>
      <p>If any of the following are true:</p>

      <ul>
       <li><p><span>this</span>'s <span>node document</span> is not <span>fully active</span>;
       or</p></li>

       <li><p><span>this</span>'s <span>current request</span>'s <span
       data-x="img-req-state">state</span> is <span data-x="img-error">broken</span>,</p></li>
      </ul>

      <p>then reject <var>promise</var> with an <span>"<code>EncodingError</code>"</span>
      <code>DOMException</code>.</p>
     </li>

     <li>
      <p>Otherwise, <span>in parallel</span>, wait for one of the following cases to occur, and
      perform the corresponding actions:</p>

      <dl class="switch">
       <dt>This <code>img</code> element's <span>node document</span> stops being <span>fully
       active</span></dt>
       <dt>This <code>img</code> element's <span>current request</span> changes or is mutated</dt>
       <dt>This <code>img</code> element's <span>current request</span>'s <span
       data-x="img-req-state">state</span> becomes <span data-x="img-error">broken</span></dt>
       <dd><p><span>Queue a global task</span> on the <span>DOM manipulation task source</span> with
       <var>global</var> to reject <var>promise</var> with an
       <span>"<code>EncodingError</code>"</span> <code>DOMException</code>.</p></dd>

       <dt>This <code>img</code> element's <span>current request</span>'s <span
       data-x="img-req-state">state</span> becomes <span data-x="img-all">completely
       available</span></dt>
       <dd>
        <p><span data-x="img-decoding-process">Decode</span> the image.</p>

        <p>If decoding does not need to be performed for this image (for example because it is a
        vector graphic) or the decoding process completes successfully, then <span>queue a global
        task</span> on the <span>DOM manipulation task source</span> with <var>global</var> to
        resolve <var>promise</var> with undefined.</p>

        <p>If decoding fails (for example due to invalid image data), then <span>queue a global
        task</span> on the <span>DOM manipulation task source</span> with <var>global</var> to
        reject <var>promise</var> with an <span>"<code>EncodingError</code>"</span>
        <code>DOMException</code>.</p>

        <p>User agents should ensure that the decoded media data stays readily available until at
        least the end of the next successful <span>update the rendering</span> step in the
        <span>event loop</span>. This is an important part of the API contract, and should not be
        broken if at all possible. (Typically, this would only be violated in low-memory situations
        that require evicting decoded image data, or when the image is too large to keep in decoded
        form for this period of time.)</p>
       </dd>
      </dl>

      <p class="note">Animated images will become <span data-x="img-all">completely available</span>
      only after all their frames are loaded. Thus, even though an implementation could decode the
      first frame before that point, the above steps will not do so, instead waiting until all
      frames are available.</p>
     </li>
    </ol>
   </li>

   <li><p>Return <var>promise</var>.</p></li>
  </ol>

  </div>

  <div class="example">
   <p>Without the <code data-x="dom-img-decode">decode()</code> method, the process of loading an
   <code>img</code> element and then displaying it might look like the following:</p>

   <pre><code class="js">const img = new Image();
img.src = "nebula.jpg";
img.onload = () => {
    document.body.appendChild(img);
};
img.onerror = () => {
    document.body.appendChild(new Text("Could not load the nebula :("));
};</code></pre>

   <p>However, this can cause notable dropped frames, as the paint that occurs after inserting the
   image into the DOM causes a synchronous decode on the main thread.</p>

   <p>This can instead be rewritten using the <code data-x="dom-img-decode">decode()</code>
   method:</p>

   <pre><code class="js">const img = new Image();
img.src = "nebula.jpg";
img.decode().then(() => {
    document.body.appendChild(img);
}).catch(() => {
    document.body.appendChild(new Text("Could not load the nebula :("));
});</code></pre>

   <p>This latter form avoids the dropped frames of the original, by allowing the user agent to
   decode the image <span>in parallel</span>, and only inserting it into the DOM (and thus causing
   it to be painted) once the decoding process is complete.</p>
  </div>

  <div class="example">
   <p>Because the <code data-x="dom-img-decode">decode()</code> method attempts to ensure that the
   decoded image data is available for at least one frame, it can be combined with the <code
   data-x="dom-AnimationFrameProvider-requestAnimationFrame">requestAnimationFrame()</code> API.
   This means it can be used with coding styles or frameworks that ensure that all DOM modifications
   are batched together as <span data-x="map of animation frame callbacks">animation frame
   callbacks</span>:</p>

   <pre><code class="js">const container = document.querySelector("#container");

const { containerWidth, containerHeight } = computeDesiredSize();
requestAnimationFrame(() => {
 container.style.width = containerWidth;
 container.style.height = containerHeight;
});

// ...

const img = new Image();
img.src = "supernova.jpg";
img.decode().then(() => {
    requestAnimationFrame(() => container.appendChild(img));
});</code></pre>
  </div>

  <div w-nodev>

  <p>A legacy factory function is provided for creating <code>HTMLImageElement</code> objects (in
  addition to the factory methods from DOM such as <code
  data-x="dom-Document-createElement">createElement()</code>): <dfn
  data-x="dom-image"><code>Image(<var>width</var>, <var>height</var>)</code></dfn>. When invoked,
  the legacy factory function must perform the following steps:</p>

  <ol>
   <li><p>Let <var>document</var> be the <span>current global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>Let <var>img</var> be the result of <span data-x="create an element">creating an
   element</span> given <var>document</var>, "<code data-x="">img</code>", and the <span>HTML
   namespace</span>.</p></li>

   <li><p>If <var>width</var> is given, then <span data-x="concept-element-attributes-set-value">set
   an attribute value</span> for <var>img</var> using "<code data-x="attr-dim-width">width</code>"
   and <var>width</var>.</p></li>

   <li><p>If <var>height</var> is given, then <span
   data-x="concept-element-attributes-set-value">set an attribute value</span> for <var>img</var>
   using "<code data-x="attr-dim-height">height</code>" and <var>height</var>.</p></li>

   <li><p>Return <var>img</var>.</p></li>
  </ol>

  </div>

  <div class="example">

   <p>A single image can have different appropriate alternative text depending on the context.</p>

   <p>In each of the following cases, the same image is used, yet the <code
   data-x="attr-img-alt">alt</code> text is different each time. The image is the coat of arms of the
   Carouge municipality in the canton Geneva in Switzerland.</p>

   <p>Here it is used as a supplementary icon:</p>
   <pre><code class="html">&lt;p>I lived in &lt;img src="carouge.svg" alt=""> Carouge.&lt;/p></code></pre>

   <p>Here it is used as an icon representing the town:</p>
   <pre><code class="html">&lt;p>Home town: &lt;img src="carouge.svg" alt="Carouge">&lt;/p></code></pre>

   <p>Here it is used as part of a text on the town:</p>

   <pre><code class="html">&lt;p>Carouge has a coat of arms.&lt;/p>
&lt;p>&lt;img src="carouge.svg" alt="The coat of arms depicts a lion, sitting in front of a tree.">&lt;/p>
&lt;p>It is used as decoration all over the town.&lt;/p></code></pre>

   <p>Here it is used as a way to support a similar text where the description is given as well as,
   instead of as an alternative to, the image:</p>

   <pre><code class="html">&lt;p>Carouge has a coat of arms.&lt;/p>
&lt;p>&lt;img src="carouge.svg" alt="">&lt;/p>
&lt;p>The coat of arms depicts a lion, sitting in front of a tree.
It is used as decoration all over the town.&lt;/p></code></pre>

   <p>Here it is used as part of a story:</p>

   <pre><code class="html">&lt;p>She picked up the folder and a piece of paper fell out.&lt;/p>
&lt;p>&lt;img src="carouge.svg" alt="Shaped like a shield, the paper had a
red background, a green tree, and a yellow lion with its tongue
hanging out and whose tail was shaped like an S.">&lt;/p>
&lt;p>She stared at the folder. S! The answer she had been looking for all
this time was simply the letter S! How had she not seen that before? It all
came together now. The phone call where Hector had referred to a lion's tail,
the time Maria had stuck her tongue out...&lt;/p></code></pre>

   <p>Here it is not known at the time of publication what the image will be, only that it will be a
   coat of arms of some kind, and thus no replacement text can be provided, and instead only a brief
   caption for the image is provided, in the <code data-x="attr-title">title</code> attribute:</p>

   <pre><code class="html">&lt;p>The last user to have uploaded a coat of arms uploaded this one:&lt;/p>
&lt;p>&lt;img src="last-uploaded-coat-of-arms.cgi" title="User-uploaded coat of arms.">&lt;/p></code></pre>

   <p>Ideally, the author would find a way to provide real replacement text even in this case, e.g.
   by asking the previous user. Not providing replacement text makes the document more difficult to
   use for people who are unable to view images, e.g. blind users, or users or very low-bandwidth
   connections or who pay by the byte, or users who are forced to use a text-only web browser.</p>

  </div>

  <div class="example">

   <p>Here are some more examples showing the same picture used in different contexts, with
   different appropriate alternate texts each time.</p>

   <pre><code class="html">&lt;article>
 &lt;h1>My cats&lt;/h1>
 &lt;h2>Fluffy&lt;/h2>
 &lt;p>Fluffy is my favorite.&lt;/p>
 &lt;img src="fluffy.jpg" alt="She likes playing with a ball of yarn.">
 &lt;p>She's just too cute.&lt;/p>
 &lt;h2>Miles&lt;/h2>
 &lt;p>My other cat, Miles just eats and sleeps.&lt;/p>
&lt;/article></code></pre>

   <pre><code class="html">&lt;article>
 &lt;h1>Photography&lt;/h1>
 &lt;h2>Shooting moving targets indoors&lt;/h2>
 &lt;p>The trick here is to know how to anticipate; to know at what speed and
 what distance the subject will pass by.&lt;/p>
 &lt;img src="fluffy.jpg" alt="A cat flying by, chasing a ball of yarn, can be
 photographed quite nicely using this technique.">
 &lt;h2>Nature by night&lt;/h2>
 &lt;p>To achieve this, you'll need either an extremely sensitive film, or
 immense flash lights.&lt;/p>
&lt;/article></code></pre>

   <pre><code class="html">&lt;article>
 &lt;h1>About me&lt;/h1>
 &lt;h2>My pets&lt;/h2>
 &lt;p>I've got a cat named Fluffy and a dog named Miles.&lt;/p>
 &lt;img src="fluffy.jpg" alt="Fluffy, my cat, tends to keep itself busy.">
 &lt;p>My dog Miles and I like go on long walks together.&lt;/p>
 &lt;h2>music&lt;/h2>
 &lt;p>After our walks, having emptied my mind, I like listening to Bach.&lt;/p>
&lt;/article></code></pre>

   <pre><code class="html">&lt;article>
 &lt;h1>Fluffy and the Yarn&lt;/h1>
 &lt;p>Fluffy was a cat who liked to play with yarn. She also liked to jump.&lt;/p>
 &lt;aside>&lt;img src="fluffy.jpg" alt="" title="Fluffy">&lt;/aside>
 &lt;p>She would play in the morning, she would play in the evening.&lt;/p>
&lt;/article></code></pre>

  </div>


  <h4 split-filename="images">Images</h4>

  <h5>Introduction</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>To embed an image in HTML, when there is only a single image resource, use the <code>img</code>
  element and its <code data-x="attr-img-src">src</code> attribute.</p>

  <div class="example">

<pre><code class="html">&lt;h2>From today's featured article&lt;/h2>
<strong>&lt;img src="/uploads/100-marie-lloyd.jpg" alt="" width="100" height="150"></strong>
&lt;p>&lt;b>&lt;a href="/wiki/Marie_Lloyd">Marie Lloyd&lt;/a>&lt;/b> (1870&ndash;1922)
was an English &lt;a href="/wiki/Music_hall">music hall&lt;/a> singer, ...</code></pre>

  </div>

  <p>However, there are a number of situations for which the author might wish to use multiple image
  resources that the user agent can choose from:</p>

  <ul>
   <li>
    <p>Different users might have different environmental characteristics:</p>

    <ul>
     <li>
      <p>The users' physical screen size might be different from one another.</p>

      <div class="example">
       <p>A mobile phone's screen might be 4 inches diagonally, while a laptop's screen might be 14
       inches diagonally.</p>

       <svg viewBox="0 0 80 32" height="200" font-size="2.5" font-family="sans-serif" role="img"
       aria-label="The phone's screen is much smaller compared to the laptop's screen.">
        <!-- smartphone -->
        <rect width="6" fill="white" stroke="black" x="6" y="18" height="12" rx="1"/>
        <text text-anchor="middle" dominant-baseline="middle" transform="translate(9 24)
        rotate(-60)">4&Prime;</text>
        <!-- laptop -->
        <rect width="40" fill="white" stroke="black" x="30" y="2" height="26" rx="2"
        stroke-width="2"/>
        <line y2="30" stroke-linecap="round" stroke="black" x1="26" x2="74" y1="30"
        stroke-width="2"/>
        <text text-anchor="middle" dominant-baseline="middle" transform="translate(50 15)
        rotate(-30)">14&Prime;</text>
       </svg>
      </div>

      <p class="note">This is only relevant when an image's rendered size depends on the
      <span>viewport</span> size.</p>
     </li>

     <li>
      <p>The users' screen pixel density might be different from one another.</p>

      <div class="example">
       <p>A mobile phone's screen might have three times as many physical pixels per inch
      compared to another mobile phone's screen, regardless of their physical screen size.</p>

       <svg viewBox="0 0 56 27" height="170" font-size="2.5" font-family="sans-serif" role="img"
       aria-label="One phone has big pixels, the other has small pixels.">
        <defs>
         <pattern id="img-intro-pixel" x="0" y="0" width="3" height="3" stroke-linecap="round"
         patternUnits="userSpaceOnUse">
          <rect x="0" y="0" width="3" height="3" fill="black"/>
          <line x1="0.5" y1="0.5" x2="0.5" y2="2.5" stroke="red"/>
          <line x1="1.5" y1="0.5" x2="1.5" y2="2.5" stroke="lime"/>
          <line x1="2.5" y1="0.5" x2="2.5" y2="2.5" stroke="blue"/>
         </pattern>
        </defs>
        <!-- smartphones -->
        <rect width="6" fill="white" stroke="black" x="6" y="8" height="12" rx="1"/>
        <rect width="6" fill="white" stroke="black" x="36" y="8" height="12" rx="1"/>
        <!-- magnifiers -->
        <line x1="15" y1="16" x2="22" y2="24" stroke="brown" stroke-width="3"/>
        <circle r="8" fill="url(#img-intro-pixel)" stroke="black" cx="10" cy="10"/>
        <line x1="45" y1="16" x2="52" y2="24" stroke="brown" stroke-width="3"/>
        <circle r="24" fill="url(#img-intro-pixel)" stroke="black" stroke-width="3" cx="120" cy="30"
        transform="scale(0.333333)"/>
        <!-- text -->
        <text dominant-baseline="middle" x="20" y="10">1x</text>
        <text dominant-baseline="middle" x="50" y="10">3x</text>
       </svg>
      </div>
     </li>

     <li>
      <p>The users' zoom level might be different from one another, or might change for a single
      user over time.</p>

      <p class="example">A user might zoom in to a particular image to be able to get a more
      detailed look.</p>

      <p>The zoom level and the screen pixel density (the previous point) can both affect the number
      of physical screen pixels per <span data-x="'px'">CSS pixel</span>. This ratio is usually
      referred to as <dfn>device-pixel-ratio</dfn>.</p>
     </li>

     <li>
      <p>The users' screen orientation might be different from one another, or might change for a
      single user over time.</p>

      <div class="example">
       <p>A tablet can be held upright or rotated 90 degrees, so that the screen is either
       "portrait" or "landscape".</p>

       <svg viewBox="0 0 60 32" height="200" font-size="2.5" font-family="sans-serif" role="img"
       aria-label="The tablet has two orientations.">
        <!-- tablets -->
        <rect width="14" fill="white" stroke="black" x="6" y="5" height="20" rx="1"/>
        <line stroke="black" stroke-width="1.1" x1="6" y1="24"  x2="20" y2="24"/>
        <text text-anchor="middle" dominant-baseline="middle" x="13" y="14.5">Portrait</text>
        <rect width="20" fill="white" stroke="black" x="30" y="11" height="14" rx="1"/>
        <line stroke="black" stroke-width="1.1" x1="31" y1="25"  x2="31" y2="11"/>
        <text text-anchor="middle" dominant-baseline="middle" x="40.5" y="18">Landscape</text>
       </svg>
      </div>
     </li>

     <li>
      <p>The users' network speed, network latency and bandwidth cost might be different from one
      another, or might change for a single user over time.</p>

      <p class="example">A user might be on a fast, low-latency and constant-cost connection while
      at work, on a slow, low-latency and constant-cost connection while at home, and on a
      variable-speed, high-latency and variable-cost connection anywhere else.</p>
     </li>
    </ul>
   </li>

   <li>
    <p>Authors might want to show the same image content but with different rendered size depending
    on, usually, the width of the <span>viewport</span>. This is usually referred to as
    <dfn>viewport-based selection</dfn>.</p>

    <div class="example">
     <p>A web page might have a banner at the top that always spans the entire <span>viewport</span>
     width. In this case, the rendered size of the image depends on the physical size of the screen
     (assuming a maximised browser window).</p>

     <svg viewBox="0 0 52 25.6" height="160" role="img" aria-label="The upright-held phone shows a
     small wolf at the top, and the tablet shows the same image but it is bigger.">
      <!-- wolf.jpg from https://commons.wikimedia.org/wiki/File:WolfRunningInSnow.jpg -->

      <!-- smartphone -->
      <rect width="6" fill="white" stroke="black" x="6" y="8" height="12" rx="1"/>
      <rect width="4" fill="#716966" x="7" y="9" height="2"/>
      <image width="4" x="7" y="9" height="2" xlink:href="/images/wolf.jpg"/>

      <!-- tablet -->
      <rect width="20" fill="white" stroke="black" x="25" y="6" height="14" rx="1"/>
      <line stroke="black" stroke-width="1.1" x1="26" y1="20"  x2="26" y2="6"/>
      <rect width="17" fill="#716966" x="27" y="7" height="8.5"/>
      <image width="17" x="27" y="7" height="8.5" xlink:href="/images/wolf.jpg"/>

     </svg>
    </div>

    <div class="example">
     <p>Another web page might have images in columns, with a single column for screens with a small
     physical size, two columns for screens with medium physical size, and three columns for screens
     with big physical size, with the images varying in rendered size in each case to fill up the
     <span>viewport</span>. In this case, the rendered size of an image might be <em>bigger</em> in
     the one-column layout compared to the two-column layout, despite the screen being smaller.</p>

     <svg viewBox="0 0 115 32" height="200" font-size="2.5" font-family="sans-serif" role="img"
     aria-label="The rotated phone shows a top part of an image of a kettlebell swing; the
     upright-held tablet shows a bit smaller images in two columns; the laptop shows images in three
     columns.">
      <!-- kettlebell.jpg from https://pixabay.com/en/functional-mobility-articular-606568/ -->

      <!-- narrow -->
      <rect width="12" fill="white" stroke="black" x="6" y="24" height="6" rx="1"/>
      <rect x="7" y="25" width="10" height="4.5" fill="#987b5a"/>
      <image x="7" y="25" width="10" height="4.5" xlink:href="/images/kettlebell.jpg"
      preserveAspectRatio="xMinYMin slice"/>
      <text text-anchor="middle" x="12" y="21">Narrow, 1 column</text>

      <!-- medium -->
      <rect width="14" fill="white" stroke="black" x="32" y="10" height="20" rx="1"/>
      <line stroke="black" stroke-width="1.1" x1="32" y1="29"  x2="46" y2="29"/>
      <rect x="33" y="11" width="5.5" height="5.5" fill="#987b5a"/>
      <image x="33" y="11" width="5.5" height="5.5" xlink:href="/images/kettlebell.jpg"/>
      <rect x="39.5" y="11" width="5.5" height="5.5" fill="burlywood"/>
      <rect x="33" y="17.5" width="5.5" height="5.5" fill="silver"/>
      <text text-anchor="middle" x="39" y="7">Medium, 2 columns</text>

      <!-- wide -->
      <rect width="40" fill="white" stroke="black" x="65" y="2" height="26" rx="2"
      stroke-width="2"/>
      <line y2="30" stroke-linecap="round" stroke="black" x1="61" x2="109" y1="30"
      stroke-width="2"/>
      <rect x="67" y="4" width="10" height="10" fill="#987b5a"/>
      <image x="67" y="4" width="10" height="10" xlink:href="/images/kettlebell.jpg"/>
      <rect x="80" y="4" width="10" height="10" fill="burlywood"/>
      <rect x="93" y="4" width="10" height="10" fill="silver"/>
      <text text-anchor="middle" x="85" y="25">Wide, 3 columns</text>
     </svg>
    </div>
   </li>

   <li>
    <p>Authors might want to show different image content depending on the rendered size of the
    image. This is usually referred to as <dfn>art direction</dfn>.</p>

    <div class="example">
     <p>When a web page is viewed on a screen with a large physical size (assuming a maximised
     browser window), the author might wish to include some less relevant parts surrounding the
     critical part of the image. When the same web page is viewed on a screen with a small physical
     size, the author might wish to show only the critical part of the image.</p>

     <svg viewBox="0 0 52 25.6" height="160" role="img" aria-label="The upright-held phone shows a
     cropped image of a wolf; the rotated tablet shows the uncropped image.">
      <!-- wolf.jpg from https://commons.wikimedia.org/wiki/File:WolfRunningInSnow.jpg -->

      <!-- smartphone -->
      <rect width="6" fill="white" stroke="black" x="6" y="8" height="12" rx="1"/>
      <rect width="4" fill="#716966" x="7" y="9" height="6"/>
      <image width="4" x="7" y="9" height="6" xlink:href="/images/wolf.jpg"
      preserveAspectRatio="xMidYMid slice"/>

      <!-- tablet -->
      <rect width="20" fill="white" stroke="black" x="25" y="6" height="14" rx="1"/>
      <line stroke="black" stroke-width="1.1" x1="26" y1="20"  x2="26" y2="6"/>
      <rect width="17" fill="#716966" x="27" y="7" height="8.5"/>
      <image width="17" x="27" y="7" height="8.5" xlink:href="/images/wolf.jpg"/>

     </svg>
    </div>
   </li>

   <li>
    <p>Authors might want to show the same image content but using different image formats,
    depending on which image formats the user agent supports. This is usually referred to as
    <dfn>image format-based selection</dfn>.</p>

    <p class="example">A web page might have some images in the JPEG, WebP and JPEG XR image
    formats, with the latter two having better compression abilities compared to JPEG. Since
    different user agents can support different image formats, with some formats offering better
    compression ratios, the author would like to serve the better formats to user agents that
    support them, while providing JPEG fallback for user agents that don't.<!--TODO: refs?--></p>
   </li>
  </ul>

  <p>The above situations are not mutually exclusive. For example, it is reasonable to combine
  different resources for different <span>device-pixel-ratio</span> with different resources for
  <span>art direction</span>.</p>

  <p>While it is possible to solve these problems using scripting, doing so introduces some other
  problems:</p>

  <ul>
   <li><p>Some user agents aggressively download images specified in the HTML markup, before scripts
   have had a chance to run, so that web pages complete loading sooner. If a script changes which
   image to download, the user agent will potentially start two separate downloads, which can
   instead cause worse page loading performance.</p></li>

   <li><p>If the author avoids specifying any image in the HTML markup and instead instantiates a
   single download from script, that avoids the double download problem above but then no image will
   be downloaded at all for users with scripting disabled and the aggressive image downloading
   optimization will also be disabled.</p></li>
  </ul>

  <p>With this in mind, this specification introduces a number of features to address the above
  problems in a declarative manner.</p>

  <dl>
   <dt><span>Device-pixel-ratio</span>-based selection when the rendered size of the image is
   fixed</dt>

   <dd>
    <p>The <code data-x="attr-img-src">src</code> and <code data-x="attr-img-srcset">srcset</code>
    attributes on the <code>img</code> element can be used, using the <code data-x="">x</code>
    descriptor, to provide multiple images that only vary in their size (the smaller image is a
    scaled-down version of the bigger image).</p>

    <p class="note">The <code data-x="">x</code> descriptor is not appropriate when the rendered
    size of the image depends on the <span>viewport</span> width
    (<span>viewport-based selection</span>), but can be used together with
    <span>art direction</span>.</p>

    <div class="example">

<pre><code class="html">&lt;h2>From today's featured article&lt;/h2>
&lt;img <strong>src="/uploads/100-marie-lloyd.jpg"</strong>
     <strong>srcset="/uploads/150-marie-lloyd.jpg 1.5x, /uploads/200-marie-lloyd.jpg 2x"</strong>
     alt="" width="100" height="150">
&lt;p>&lt;b>&lt;a href="/wiki/Marie_Lloyd">Marie Lloyd&lt;/a>&lt;/b> (1870&ndash;1922)
was an English &lt;a href="/wiki/Music_hall">music hall&lt;/a> singer, ...</code></pre>

     <p>The user agent can choose any of the given resources depending on the user's screen's pixel
     density, zoom level, and possibly other factors such as the user's network conditions.</p>

     <p>For backwards compatibility with older user agents that don't yet understand the <code
     data-x="attr-img-srcset">srcset</code> attribute, one of the URLs is specified in the
     <code>img</code> element's <code data-x="attr-img-src">src</code> attribute. This will result
     in something useful (though perhaps lower-resolution than the user would like) being displayed
     even in older user agents. For new user agents, the <code data-x="attr-img-src">src</code>
     attribute participates in the resource selection, as if it was specified in <code
     data-x="attr-img-srcset">srcset</code> with a <code data-x="">1x</code> descriptor.</p>

     <p>The image's rendered size is given in the <code data-x="attr-dim-width">width</code> and
     <code data-x="attr-dim-height">height</code> attributes, which allows the user agent to
     allocate space for the image before it is downloaded.</p>

    </div>
   </dd>

   <dt><span>Viewport-based selection</span></dt>

   <dd>
    <p>The <code data-x="attr-img-srcset">srcset</code> and <code
    data-x="attr-img-sizes">sizes</code> attributes can be used, using the <code data-x="">w</code>
    descriptor, to provide multiple images that only vary in their size (the smaller image is a
    scaled-down version of the bigger image).</p>

    <div class="example">

     <p>In this example, a banner image takes up the entire <span>viewport</span> width (using
     appropriate CSS).</p>

<pre><code class="html">&lt;h1>&lt;img <strong>sizes="100vw" srcset="wolf-400.jpg 400w, wolf-800.jpg 800w, wolf-1600.jpg 1600w"</strong>
     src="wolf-400.jpg" alt="The rad wolf">&lt;/h1></code></pre>

     <p>The user agent will calculate the effective pixel density of each image from the specified
     <code data-x="">w</code> descriptors and the specified rendered size in the <code
     data-x="attr-img-sizes">sizes</code> attribute. It can then choose any of the given resources
     depending on the user's screen's pixel density, zoom level, and possibly other factors such as
     the user's network conditions.</p>

     <p>If the user's screen is 320 <span data-x="'px'">CSS pixels</span> wide, this is equivalent
     to specifying <code data-x="">wolf-400.jpg 1.25x, wolf-800.jpg 2.5x, wolf-1600.jpg 5x</code>.
     On the other hand, if the user's screen is 1200 <span data-x="'px'">CSS pixels</span> wide,
     this is equivalent to specifying <code
     data-x="">wolf-400.jpg 0.33x, wolf-800.jpg 0.67x, wolf-1600.jpg 1.33x</code>. By using the
     <code data-x="">w</code> descriptors and the <code data-x="attr-img-sizes">sizes</code>
     attribute, the user agent can choose the correct image source to download regardless of how
     large the user's device is.</p>

     <p>For backwards compatibility, one of the URLs is specified in the <code>img</code> element's
     <code data-x="attr-img-src">src</code> attribute. In new user agents, the <code
     data-x="attr-img-src">src</code> attribute is ignored when the <code
     data-x="attr-img-srcset">srcset</code> attribute uses <code data-x="">w</code> descriptors.</p>

    </div>

    <div class="example">

     <p>In this example, the web page has three layouts depending on the width of the
     <span>viewport</span>. The narrow layout has one column of images (the width of each image is
     about 100%), the middle layout has two columns of images (the width of each image is about
     50%), and the widest layout has three columns of images, and some page margin (the width of
     each image is about 33%). It breaks between these layouts when the <span>viewport</span> is
     <code data-x="">30em</code> wide and <code data-x="">50em</code> wide, respectively.</p>

<pre><code class="html">&lt;img <strong>sizes="(max-width: 30em) 100vw, (max-width: 50em) 50vw, calc(33vw - 100px)"</strong>
     <strong>srcset="swing-200.jpg 200w, swing-400.jpg 400w, swing-800.jpg 800w, swing-1600.jpg 1600w"</strong>
     src="swing-400.jpg" alt="Kettlebell Swing"></code></pre>

     <p>The <code data-x="attr-img-sizes">sizes</code> attribute sets up the layout breakpoints at
     <code data-x="">30em</code> and <code data-x="">50em</code>, and declares the image sizes
     between these breakpoints to be <code data-x="">100vw</code>, <code data-x="">50vw</code>, or
     <code data-x="">calc(33vw - 100px)</code>. These sizes do not necessarily have to match up
     exactly with the actual image width as specified in the CSS.</p>

     <p>The user agent will pick a width from the <code data-x="attr-img-sizes">sizes</code>
     attribute, using the first item with a <span>&lt;media-condition></span> (the part in
     parentheses) that evaluates to true, or using the last item (<code data-x="">calc(33vw -
     100px)</code>) if they all evaluate to false.</p>

     <p>For example, if the <span>viewport</span> width is <code data-x="">29em</code>, then <code
     data-x="">(max-width: 30em)</code> evaluates to true and <code data-x="">100vw</code> is used,
     so the image size, for the purpose of resource selection, is <code data-x="">29em</code>. If
     the <span>viewport</span> width is instead <code data-x="">32em</code>, then
     <code data-x="">(max-width: 30em)</code> evaluates to false, but
     <code data-x="">(max-width: 50em)</code> evaluates to true
     and <code data-x="">50vw</code> is used, so the image size, for the purpose of resource
     selection, is <code data-x="">16em</code> (half the <span>viewport</span> width). Notice that
     the slightly wider <span>viewport</span> results in a smaller image because of the different
     layout.</p>

     <p>The user agent can then calculate the effective pixel density and choose an appropriate
     resource similarly to the previous example.</p>

    </div>

    <div class="example">

     <p>This example is the same as the previous example, but the image is <span
     data-x="attr-loading-lazy-state">lazy-loaded</span>. In this case, the <code
     data-x="attr-img-sizes">sizes</code> attribute can use the <code
     data-x="valdef-sizes-auto">auto</code> keyword, and the user agent will use the <code
     data-x="attr-dim-width">width</code> attribute (or the width specified in CSS) for the
     <span>source size</span>.</p>

<pre><code class="html">&lt;img <strong>loading="lazy" width="200" height="200" sizes="auto"</strong>
     srcset="swing-200.jpg 200w, swing-400.jpg 400w, swing-800.jpg 800w, swing-1600.jpg 1600w"
     src="swing-400.jpg" alt="Kettlebell Swing"></code></pre>

     <p>For better backwards-compatibility with legacy user agents that don't support the <code
     data-x="valdef-sizes-auto">auto</code> keyword, fallback sizes can be specified if desired.</p>

<pre><code class="html">&lt;img loading="lazy" width="200" height="200"
     <strong>sizes="auto, (max-width: 30em) 100vw, (max-width: 50em) 50vw, calc(33vw - 100px)"</strong>
     srcset="swing-200.jpg 200w, swing-400.jpg 400w, swing-800.jpg 800w, swing-1600.jpg 1600w"
     src="swing-400.jpg" alt="Kettlebell Swing"></code></pre>

    </div>
   </dd>

   <dt><span>Art direction</span>-based selection</dt>

   <dd>
    <p>The <code>picture</code> element and the <code>source</code> element, together with the <code
    data-x="attr-source-media">media</code> attribute, can be used to provide multiple images that
    vary the image content (for instance the smaller image might be a cropped version of the bigger
    image).</p>

    <div class="example">

<pre><code class="html">&lt;picture>
  &lt;source <strong>media="(min-width: 45em)"</strong> srcset="large.jpg">
  &lt;source <strong>media="(min-width: 32em)"</strong> srcset="med.jpg">
  &lt;img src="small.jpg" alt="The wolf runs through the snow.">
&lt;/picture></code></pre>

     <p>The user agent will choose the first <code>source</code> element for which the media query
     in the <code data-x="attr-source-media">media</code> attribute matches, and then choose an
     appropriate URL from its <code data-x="attr-source-srcset">srcset</code> attribute.</p>

     <p>The rendered size of the image varies depending on which resource is chosen. To specify
     dimensions that the user agent can use before having downloaded the image, CSS can be used.</p>

<pre><code class="css">img { width: 300px; height: 300px }
@media (min-width: 32em) { img { width: 500px; height:300px } }
@media (min-width: 45em) { img { width: 700px; height:400px } }</code></pre>

    </div>

    <div class="example">

     <p>This example combines <span>art direction</span>- and <span>device-pixel-ratio</span>-based
     selection. A banner that takes half the <span>viewport</span> is provided in two versions,
     one for wide screens and one for narrow screens.</p>

<pre><code class="html">&lt;h1>
 &lt;picture>
  &lt;source media="(max-width: 500px)" srcset="banner-phone.jpeg, banner-phone-HD.jpeg 2x">
  &lt;img src="banner.jpeg" srcset="banner-HD.jpeg 2x" alt="The Breakfast Combo">
 &lt;/picture>
&lt;/h1></code></pre>

    </div>
   </dd>

   <dt><span>Image format-based selection</span></dt>

   <dd>
    <p>The <code data-x="attr-source-type">type</code> attribute on the <code>source</code> element
    can be used to provide multiple images in different formats.</p>

    <div class="example">

<pre><code class="html">&lt;h2>From today's featured article&lt;/h2>
&lt;picture>
 &lt;source srcset="/uploads/100-marie-lloyd.webp" <strong>type="image/webp"</strong>>
 &lt;source srcset="/uploads/100-marie-lloyd.jxr" <strong>type="image/vnd.ms-photo"</strong>>
 &lt;img src="/uploads/100-marie-lloyd.jpg" alt="" width="100" height="150">
&lt;/picture>
&lt;p>&lt;b>&lt;a href="/wiki/Marie_Lloyd">Marie Lloyd&lt;/a>&lt;/b> (1870&ndash;1922)
was an English &lt;a href="/wiki/Music_hall">music hall&lt;/a> singer, ...</code></pre>

     <p>In this example, the user agent will choose the first source that has a <code
     data-x="attr-source-type">type</code> attribute with a supported MIME type. If the user agent
     supports WebP images, the first <code>source</code> element will be chosen. If not, but the
     user agent does support JPEG XR images, the second <code>source</code> element will be chosen.
     If neither of those formats are supported, the <code>img</code> element will be chosen.</p>

    </div>
   </dd>
  </dl>


  <h6>Adaptive images</h6>

  <!-- NON-NORMATIVE SECTION -->

  <p>CSS and media queries can be used to construct graphical page layouts that adapt dynamically to
  the user's environment, in particular to different <span>viewport</span> dimensions and pixel
  densities. For content, however, CSS does not help; instead, we have the <code>img</code> element's
  <code data-x="attr-img-srcset">srcset</code> attribute and the <code>picture</code> element.
  This section walks through a sample case showing how to use these features.</p>

  <p>Consider a situation where on wide screens (wider than 600 <span data-x="'px'">CSS
  pixels</span>) a 300&times;150 image named <code data-x="">a-rectangle.png</code> is to be used,
  but on smaller screens (600 <span data-x="'px'">CSS pixels</span> and less), a smaller
  100&times;100 image called <code data-x="">a-square.png</code> is to be used. The markup for this
  would look like this:</p>

  <pre><code class="html">&lt;figure>
 &lt;picture>
  &lt;source srcset="a-square.png" media="(max-width: 600px)">
  &lt;img src="a-rectangle.png" alt="Barney Frank wears a suit and glasses.">
 &lt;/picture>
 &lt;figcaption>Barney Frank, 2011&lt;/figcaption>
&lt;/figure></code></pre>

  <p class="note">For details on what to put in the <code data-x="attr-img-alt">alt</code>
  attribute, see the <a href="#alt">Requirements for providing text to act as an alternative for
  images</a> section.</p>

  <p>The problem with this is that the user agent does not necessarily know what dimensions to use
  for the image when the image is loading. To avoid the layout having to be reflowed multiple times
  as the page is loading, CSS and CSS media queries can be used to provide the dimensions:</p>

  <pre><code class="html">&lt;style>
 #a { width: 300px; height: 150px; }
 @media (max-width: 600px) { #a { width: 100px; height: 100px; } }
&lt;/style>
&lt;figure>
 &lt;picture>
  &lt;source srcset="a-square.png" media="(max-width: 600px)">
  &lt;img src="a-rectangle.png" alt="Barney Frank wears a suit and glasses." id="a">
 &lt;/picture>
 &lt;figcaption>Barney Frank, 2011&lt;/figcaption>
&lt;/figure></code></pre>

  <p>Alternatively, the <code data-x="attr-dim-width">width</code> and <code
  data-x="attr-dim-height">height</code> attributes can be used to provide the width and height for
  legacy user agents, using CSS just for the user agents that support <code>picture</code>:</p>

  <pre><code class="html">&lt;style media="(max-width: 600px)">
 #a { width: 100px; height: 100px; }
&lt;/style>
&lt;figure>
 &lt;picture>
  &lt;source srcset="a-square.png" media="(max-width: 600px)">
  &lt;img src="a-rectangle.png" width="300" height="150"
  alt="Barney Frank wears a suit and glasses." id="a">
 &lt;/picture>
 &lt;figcaption>Barney Frank, 2011&lt;/figcaption>
&lt;/figure></code></pre>

  <hr>

  <p>The <code>img</code> element is used with the <code data-x="attr-img-src">src</code> attribute,
  which gives the URL of the image to use for legacy user agents that do not support the
  <code>picture</code> element. This leads to a question of which image to provide in the <code
  data-x="attr-img-src">src</code> attribute.</p>

  <p>If the author wants the biggest image in legacy user agents, the markup could be as
  follows:</p>

  <pre><code class="html">&lt;picture>
 &lt;source srcset="pear-mobile.jpeg" media="(max-width: 720px)">
 &lt;source srcset="pear-tablet.jpeg" media="(max-width: 1280px)">
 &lt;img src="pear-desktop.jpeg" alt="The pear is juicy.">
&lt;/picture></code></pre>

  <p>However, if legacy mobile user agents are more important, one can list all three images in the
  <code>source</code> elements, overriding the <code data-x="attr-img-src">src</code> attribute
  entirely.</p>

  <pre><code class="html">&lt;picture>
 &lt;source srcset="pear-mobile.jpeg" media="(max-width: 720px)">
 &lt;source srcset="pear-tablet.jpeg" media="(max-width: 1280px)">
 &lt;source srcset="pear-desktop.jpeg">
 &lt;img src="pear-mobile.jpeg" alt="The pear is juicy.">
&lt;/picture></code></pre>

  <p>Since at this point the <code data-x="attr-img-src">src</code> attribute is actually being
  ignored entirely by <code>picture</code>-supporting user agents, the <code
  data-x="attr-img-src">src</code> attribute can default to any image, including one that is neither
  the smallest nor biggest:</p>

  <pre><code class="html">&lt;picture>
 &lt;source srcset="pear-mobile.jpeg" media="(max-width: 720px)">
 &lt;source srcset="pear-tablet.jpeg" media="(max-width: 1280px)">
 &lt;source srcset="pear-desktop.jpeg">
 &lt;img src="pear-tablet.jpeg" alt="The pear is juicy.">
&lt;/picture></code></pre>

  <hr>

  <p>Above the <code data-x="">max-width</code> media feature is used, giving the maximum
  (<span>viewport</span>) dimensions that an image is intended for. It is also possible to use <code
  data-x="">min-width</code> instead.</p>

  <pre><code class="html">&lt;picture>
 &lt;source srcset="pear-desktop.jpeg" media="(min-width: 1281px)">
 &lt;source srcset="pear-tablet.jpeg" media="(min-width: 721px)">
 &lt;img src="pear-mobile.jpeg" alt="The pear is juicy.">
&lt;/picture></code></pre>


  <h5 id="attributes-common-to-source-and-img-elements">Attributes common to <code>source</code>,
  <code>img</code>, and <code>link</code> elements</h5>

  <h6>Srcset attributes</h6>

  <p>A <dfn>srcset attribute</dfn> is an attribute with requirements defined in this section.</p>

  <p>If present, its value must consist of one or more <span data-x="image candidate string">image
  candidate strings</span>, each separated from the next by a U+002C COMMA character (,). If an
  <span>image candidate string</span> contains no descriptors and no <span>ASCII whitespace</span>
  after the URL, the following <span>image candidate string</span>, if there is one, must begin with
  one or more <span>ASCII whitespace</span>.</p>

  <p>An <dfn>image candidate string</dfn> consists of the following components, in order, with the
  further restrictions described below this list:</p>

  <ol>
   <li><p>Zero or more <span>ASCII whitespace</span>.</p></li>

   <li><p>A <span>valid non-empty URL</span> that does not start or end with a U+002C COMMA
   character (,), referencing a non-interactive, optionally animated, image resource that is neither
   paged nor scripted.</p></li>

   <li><p>Zero or more <span>ASCII whitespace</span>.</p></li>

   <li>
    <p>Zero or one of the following:</p>

    <ul>
     <li><p>A <dfn>width descriptor</dfn>, consisting of: <span>ASCII whitespace</span>, a
     <span>valid non-negative integer</span> giving a number greater than zero representing the
     <dfn>width descriptor value</dfn>, and a U+0077 LATIN SMALL LETTER W character.</p></li>

     <li><p>A <dfn>pixel density descriptor</dfn>, consisting of: <span>ASCII whitespace</span>, a
     <span>valid floating-point number</span> giving a number greater than zero representing the
     <dfn>pixel density descriptor value</dfn>, and a U+0078 LATIN SMALL LETTER X
     character.</p></li>
    </ul>
   </li>

   <li><p>Zero or more <span>ASCII whitespace</span>.</p></li>
  </ol>

  <p>There must not be an <span>image candidate string</span> for an element that has the same
  <span>width descriptor value</span> as another <span>image candidate string</span>'s <span>width
  descriptor value</span> for the same element.</p>

  <p>There must not be an <span>image candidate string</span> for an element that has the same
  <span>pixel density descriptor value</span> as another <span>image candidate string</span>'s
  <span>pixel density descriptor value</span> for the same element. For the purpose of this
  requirement, an <span>image candidate string</span> with no descriptors is equivalent to an
  <span>image candidate string</span> with a <code data-x="">1x</code> descriptor.</p>

  <p>If an <span>image candidate string</span> for an element has the <span>width descriptor</span>
  specified, all other <span data-x="image candidate string">image candidate strings</span> for that
  element must also have the <span>width descriptor</span> specified.</p>

  <p>The specified width in an <span>image candidate string</span>'s <span>width descriptor</span>
  must match the <span>natural width</span> in the resource given by the
  <span>image candidate string</span>'s URL, if it has a <span>natural width</span>.</p>

  <p>If an element has a <span>sizes attribute</span> present, all <span data-x="image
  candidate string">image candidate strings</span> for that element must have the <span>width
  descriptor</span> specified.</p>


  <h6>Sizes attributes</h6>

  <p>A <dfn>sizes attribute</dfn> is an attribute with requirements defined in this section.</p>

  <p>If present, the value must be a <span>valid source size list</span>.</p>

  <p>A <dfn>valid source size list</dfn> is a string that matches the following grammar:
  <ref>CSSVALUES</ref> <ref>MQ</ref></p>

<pre><code class="html"><dfn type>&lt;source-size-list></dfn> = <span>&lt;source-size></span>#? , <span>&lt;source-size-value></span>
<dfn type>&lt;source-size></dfn> = <span>&lt;media-condition></span> <span>&lt;source-size-value></span> | <span data-x="valdef-sizes-auto">auto</span>
<dfn type>&lt;source-size-value></dfn> = <span>&lt;length></span> | <span data-x="valdef-sizes-auto">auto</span></code></pre>

  <p>A <span>&lt;source-size-value></span> that is a <span>&lt;length></span> must not be negative,
  and must not use CSS functions other than the <span>math functions</span>.</p>

  <p>The keyword <dfn><code data-x="valdef-sizes-auto">auto</code></dfn> is a width that is
  computed in <span>parse a sizes attribute</span>. If present, it must be the first entry and the
  entire <span>&lt;source-size-list></span> value must either be the string "<code
  data-x="">auto</code>" (<span>ASCII case-insensitive</span>) or start with the string "<code
  data-x="">auto,</code>" (<span>ASCII case-insensitive</span>).</p>

  <p class="note">If the <code>img</code> element that initiated the image loading (with the
  <span>update the image data</span> or <span data-x="img-environment-changes">react to environment
  changes</span> algorithms) <span>allows auto-sizes</span> and is <span>being rendered</span>,
  then <code data-x="valdef-sizes-auto">auto</code> is the <span>concrete object size</span> width.
  Otherwise, the <code data-x="valdef-sizes-auto">auto</code> value is ignored and the next
  <span>source size</span> is used instead, if any.</p>

  <p>The <code data-x="valdef-sizes-auto">auto</code> keyword may be specified in the <code
  data-x="attr-source-sizes">sizes</code> attribute of <code>source</code> elements and <code
  data-x="attr-img-sizes">sizes</code> attribute of <code>img</code> elements, if the following
  conditions are met. Otherwise, <code data-x="valdef-sizes-auto">auto</code> must not be
  specified.</p>

  <ul>
   <li><p>The element is a <code>source</code> element with a following sibling
   <code>img</code> element.</p></li>

   <li><p>The element is an <code>img</code> element.</p></li>

   <li><p>The <code>img</code> element referenced in either condition above <span>allows
   auto-sizes</span>.</p></li>
  </ul>

  <p class="note">In addition, it is strongly encouraged to specify dimensions using the <code
  data-x="attr-dim-width">width</code> and <code data-x="attr-dim-height">height</code> attributes
  or with CSS. Without specified dimensions, the image will likely render with 300x150 dimensions
  because <code data-x="">sizes="auto"</code> implies <code data-x="">contain-intrinsic-size: 300px
  150px</code> in <a href="#img-contain-size">the Rendering section</a>.</p>

  <p>The <span>&lt;source-size-value></span> gives the intended layout width of the image. The
  author can specify different widths for different environments with
  <span>&lt;media-condition></span>s.</p>

  <p class="note">Percentages are not allowed in a <span>&lt;source-size-value></span>,
  to avoid confusion about what it would be relative to.
  The <span>'vw'</span> unit can be used for sizes relative to the <span>viewport</span>
  width.</p>

  <div w-nodev>

  <h5 id="images-processing-model">Processing model</h5>

  <hr>

  <p>An <code>img</code> element has a <dfn>current request</dfn> and a <dfn>pending request</dfn>.
  The <span>current request</span> is initially set to a new <span>image request</span>.
  The <span>pending request</span> is initially set to null.</p>

  <p>An <dfn export>image request</dfn> has a <dfn export for="image request"
  data-x="img-req-state">state</dfn>, <dfn export for="image request"
  data-x="img-req-url">current URL</dfn>, and <dfn export for="image request"
  data-x="img-req-data">image data</dfn>.</p>

  <p>An <span>image request</span>'s <span data-x="img-req-state">state</span> is one of the
  following:</p>

  <dl>
   <dt><dfn data-x="img-none">Unavailable</dfn></dt>
   <dd>The user agent hasn't obtained any image data,
   or has obtained some or all of the image data but
   hasn't yet decoded enough of the image to get the image dimensions.</dd>

   <dt><dfn data-x="img-inc">Partially available</dfn></dt>
   <dd>The user agent has obtained some of the image data and at least the image dimensions are
   available.</dd>

   <dt><dfn data-x="img-all">Completely available</dfn></dt>
   <dd>The user agent has obtained all of the image data and at least the image dimensions are
   available.</dd>

   <dt><dfn data-x="img-error">Broken</dfn></dt>
   <dd>The user agent has obtained all of the image data that it can, but it cannot even decode the
   image enough to get the image dimensions (e.g. the image is corrupted, or the format is not
   supported, or no data could be obtained).</dd>
  </dl>

  <p>An <span>image request</span>'s <span data-x="img-req-url">current URL</span> is initially the
  empty string.</p>

  <p>An <span>image request</span>'s <span data-x="img-req-data">image data</span> is the decoded
  image data.</p>

  <p>When an <span>image request</span>'s <span data-x="img-req-state">state</span> is either <span
  data-x="img-inc">partially available</span> or <span data-x="img-all">completely available</span>,
  the <span>image request</span> is said to be <dfn data-x="img-available">available</dfn>.</p>

  <p>When an <code>img</code> element's <span>current request</span>'s <span
  data-x="img-req-state">state</span> is <span data-x="img-all">completely available</span> and the
  user agent can decode the media data without errors, then the <code>img</code> element is said to
  be <dfn data-x="img-good">fully decodable</dfn>.</p>

  <p>An <span>image request</span>'s <span data-x="img-req-state">state</span> is initially <span
  data-x="img-none">unavailable</span>.</p>

  <p>When an <code>img</code> element's <span>current request</span> is <span
  data-x="img-available">available</span>, the <code>img</code> element provides a <span>paint
  source</span> whose width is the image's <span data-x="density-corrected natural width and
  height">density-corrected natural width</span> (if any), whose height is the image's <span
  data-x="density-corrected natural width and height">density-corrected natural height</span>
  (if any), and whose appearance is the natural appearance of the image.</p>

  <hr>

  <p>An <code>img</code> element is said to <dfn>use <code>srcset</code> or
  <code>picture</code></dfn> if it has a <code data-x="attr-img-srcset">srcset</code> attribute
  specified or if it has a parent that is a <code>picture</code> element.</p>

  <hr>

  <p>Each <code>img</code> element has a <dfn>last selected source</dfn>, which must initially be
  null.</p>

  <p>Each <span>image request</span> has a <dfn>current pixel density</dfn>, which must initially be
  1.</p>

  <p>Each <span>image request</span> has <dfn>preferred density-corrected dimensions</dfn>, which is
  either a struct consisting of a width and a height or is null. It must initially be null.</p>

  <p>To determine the <dfn id="density-corrected-intrinsic-width-and-height">density-corrected
  natural width and height</dfn> of an <code>img</code> element <var>img</var>:</p>

  <ol>
   <li>
    <p>Let <var>dim</var> be <var>img</var>'s <span>current request</span>'s <span>preferred
    density-corrected dimensions</span>.</p>

    <p class="note">The <span>preferred density-corrected dimensions</span> are set in the
    <span>prepare an image for presentation</span> algorithm based on meta information in the
    image.</p>
   </li>

   <li><p>If <var>dim</var> is null, set <var>dim</var> to <var>img</var>'s <span>natural
   dimensions</span>.</p></li>

   <li><p>Set <var>dim</var>'s width to <var>dim</var>'s width divided by <var>img</var>'s
   <span>current request</span>'s <span>current pixel density</span>.</p></li>

   <li><p>Set <var>dim</var>'s height to <var>dim</var>'s height divided by <var>img</var>'s
   <span>current request</span>'s <span>current pixel density</span>.</p></li>

   <li><p>Return <var>dim</var>.</p></li>
  </ol>

  <p class="example">For example, if the <span>current pixel density</span> is 3.125, that means
  that there are 300 device pixels per <span data-x="'in'">CSS inch</span>, and thus if the image
  data is 300x600, it has <span>density-corrected natural width and height</span> of 96 <span
  data-x="'px'">CSS pixels</span> by 192 <span data-x="'px'">CSS pixels</span>.</p>

  <p>All <code>img</code> and <code>link</code> elements are associated with a <span>source set</span>.</p>

  <p>A <dfn>source set</dfn> is an ordered set of zero or more <span data-x="image source">image sources</span>
  and a <span>source size</span>.</p>

  <p>An <dfn>image source</dfn> is a <span>URL</span>, and optionally either a <span>pixel density
  descriptor</span>, or a <span>width descriptor</span>.</p>

  <p>A <dfn>source size</dfn> is a <span>&lt;source-size-value></span>.
  When a <span>source size</span> has a unit relative to the <span>viewport</span>,
  it must be interpreted relative to the <code>img</code> element's <span>node document</span>'s
  <span>viewport</span>. Other units must be interpreted the same as in Media Queries.
  <ref>MQ</ref></p>

  <hr>

  <p>A <dfn data-x="concept-microsyntax-parse-error">parse error</dfn> for algorithms in this
  section indicates a non-fatal mismatch between input and requirements. User agents are encouraged
  to expose <span data-x="concept-microsyntax-parse-error">parse error</span>s somehow.</p>

  <hr>

  <p>Whether the image is fetched successfully or not (e.g. whether the response status was an
  <span>ok status</span>) must be ignored when determining the image's type and whether it is a
  valid image.</p>

  <p class="note">This allows servers to return images with error responses, and have them
  displayed.</p>

  <p id="img-determine-type">The user agent should apply the <span data-x="Content-Type sniffing:
  image">image sniffing rules</span> to determine the type of the image, with the image's <span
  data-x="Content-Type">associated Content-Type headers</span> giving the <var>official
  type</var>. If these rules are not applied, then the type of the image must be the type given by
  the image's <span data-x="Content-Type">associated Content-Type headers</span>.</p>

  <p>User agents must not support non-image resources with the <code>img</code> element (e.g. XML
  files whose <span>document element</span> is an HTML element). User agents must not run executable
  code (e.g. scripts) embedded in the image resource. User agents must only display the first page
  of a multipage resource (e.g. a PDF file). User agents must not allow the resource to act in an
  interactive fashion, but should honour any animation in the resource.</p>

  <p>This specification does not specify which image types are to be supported.</p>


  <h6>When to obtain images</h6>

  <p>By default, images are obtained immediately. User agents may provide users with the option to
  instead obtain them on-demand. (The on-demand option might be used by bandwidth-constrained users,
  for example.)</p>

  <p>When obtaining images immediately, the user agent must synchronously <span>update the image
  data</span> of the <code>img</code> element, with the <i>restart animation</i> flag set if so
  stated, whenever that element is created or has experienced <span>relevant mutations</span>.</p>

  <p>When obtaining images on demand, the user agent must <span>update the image data</span> of an
  <code>img</code> element whenever it needs the image data (i.e., on demand), but only if the
  <code>img</code> element's <span>current request</span>'s <span
  data-x="img-req-state">state</span> is <span data-x="img-none">unavailable</span>. When an
  <code>img</code> element has experienced <span>relevant mutations</span>, if the user agent only
  obtains images on demand, the <code>img</code> element's <span>current request</span>'s <span
  data-x="img-req-state">state</span> must return to <span data-x="img-none">unavailable</span>.</p>


  <h6>Reacting to DOM mutations</h6>

  <p>The <dfn>relevant mutations</dfn> for an <code>img</code> element are as follows:</p>

  <ul>
   <li><p>The element's <code data-x="attr-img-src">src</code>, <code
   data-x="attr-img-srcset">srcset</code>, <code data-x="attr-dim-width">width</code>, or <code
   data-x="attr-img-sizes">sizes</code> attributes are set, changed, or removed.</p></li>

   <li><p>The element's <code data-x="attr-img-src">src</code> attribute is set to the same value as the previous value.
   This must set the <i>restart animation</i> flag for the <span>update the image data</span> algorithm.</p></li>

   <li><p>The element's <code data-x="attr-img-crossorigin">crossorigin</code> attribute's state is changed.</p></li>

   <li><p>The element's <code data-x="attr-img-referrerpolicy">referrerpolicy</code> attribute's
   state is changed.</p></li>

   <li><p>The <code>img</code> or <code>source</code> <span>HTML element insertion steps</span>,
   <span>HTML element removing steps</span>, and <span>HTML element moving steps</span> count the
   mutation as a <span data-x="relevant mutations">relevant mutation</span>.</p></li>

   <li><p>The element's parent is a <code>picture</code> element and a <code>source</code> element
   that is a previous sibling has its <code data-x="attr-source-srcset">srcset</code>, <code
   data-x="attr-source-sizes">sizes</code>, <code data-x="attr-source-media">media</code>, <code
   data-x="attr-source-type">type</code>, <code data-x="attr-dim-width">width</code> or <code
   data-x="attr-dim-height">height</code> attributes set, changed, or removed.</p></li>

   <li><p>The element's <span data-x="concept-node-adopt-ext">adopting steps</span> are run.</p></li>

   <li><p>If the element <span>allows auto-sizes</span>: the element starts or stops <span>being
   rendered</span>, or its <span>concrete object size</span> width changes. This must set the
   <i>maybe omit events</i> flag for the <span>update the image data</span> algorithm.</p></li>

   <!-- NOT when the base URL changes (except for when adopted into a new doc) -->
  </ul>


  <h6>The list of available images</h6>

  <p>Each <code>Document</code> object must have a <dfn>list of available images</dfn>. Each image
  in this list is identified by a tuple consisting of an <span>absolute URL</span>, a <span>CORS
  settings attribute</span> mode, and, if the mode is not <span data-x="attr-crossorigin-none">No
  CORS</span>, an <span>origin</span>.
  Each image furthermore has an <dfn>ignore higher-layer caching</dfn> flag.
  User agents may copy entries from one <code>Document</code>
  object's <span>list of available images</span> to another at any time (e.g. when the
  <code>Document</code> is created, user agents can add to it all the images that are loaded in
  other <code>Document</code>s), but must not change the keys of entries copied in this way when
  doing so, and must unset the <span>ignore higher-layer caching</span> flag for the copied entry.
  User agents may also remove images from such lists at any time (e.g. to save
  memory).
  User agents must remove entries in the <span>list of available images</span> as appropriate
  given higher-layer caching semantics for the resource (e.g. the HTTP `<code
  data-x="http-cache-control">Cache-Control</code>` response header) when the <span>ignore
  higher-layer caching</span> flag is unset.</p>

  <p class="note">The <span>list of available images</span> is intended to enable synchronous
  switching when changing the <code data-x="attr-img-src">src</code> attribute to a URL that has
  previously been loaded, and to avoid re-downloading images in the same document even when they
  don't allow caching per HTTP. It is not used to avoid re-downloading the same image while the
  previous image is still loading.</p>

  <p class="note">The user agent can also store the image data separately from the <span>list of available images</span>.</p>

  <p class="example">For example, if a resource has the HTTP response header
  `<code data-x="">Cache-Control: must-revalidate</code>`, and its <span>ignore higher-layer
  caching</span> flag is unset, the user agent would remove it from the <span>list of available
  images</span> but could keep the image data separately, and use that if the server responds with a
  <code data-x="">304 Not Modified</code> status.</p>

  <h6>Decoding images</h6>

  <p>Image data is usually encoded in order to reduce file size. This means that in order for the
  user agent to present the image to the screen, the data needs to be decoded.
  <dfn data-x="img-decoding-process">Decoding</dfn> is the process which converts an image's media
  data into a bitmap form, suitable for presentation to the screen. Note that this process can be
  slow relative to other processes involved in presenting content. Thus, the user agent can choose
  when to perform decoding, in order to create the best user experience.</p>

  <p>Image decoding is said to be synchronous if it prevents presentation of other content until it
  is finished. Typically, this has an effect of atomically presenting the image and any other
  content at the same time. However, this presentation is delayed by the amount of time it takes to
  perform the decode.</p>

  <p>Image decoding is said to be asynchronous if it does not prevent presentation of other content.
  This has an effect of presenting non-image content faster. However, the image content is missing
  on screen until the decode finishes. Once the decode is finished, the screen is updated with the
  image.</p>

  <p>In both synchronous and asynchronous decoding modes, the final content is presented to screen
  after the same amount of time has elapsed. The main difference is whether the user agent presents
  non-image content ahead of presenting the final content.</p>

  <p>In order to aid the user agent in deciding whether to perform synchronous or asynchronous
  decode, the <code data-x="attr-img-decoding">decoding</code> attribute can be set on
  <code>img</code> elements. The possible values of the <code
  data-x="attr-img-decoding">decoding</code> attribute are the following <dfn>image decoding
  hint</dfn> keywords:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Description
   <tbody>
    <tr>
     <td><dfn attr-value for="img/decoding"><code data-x="attr-img-decoding-sync">sync</code></dfn>
     <td><dfn data-x="attr-img-decoding-sync-state">Sync</dfn>
     <td>Indicates a preference to <span data-x="img-decoding-process">decode</span> this image
     synchronously for atomic presentation with other content.
    <tr>
     <td><dfn attr-value for="img/decoding"><code
     data-x="attr-img-decoding-async">async</code></dfn>
     <td><dfn data-x="attr-img-decoding-async-state">Async</dfn>
     <td>Indicates a preference to <span data-x="img-decoding-process">decode</span> this image
     asynchronously to avoid delaying presentation of other content.
    <tr>
     <td><dfn attr-value for="img/decoding"><code data-x="attr-img-decoding-auto">auto</code></dfn>
     <td><dfn data-x="attr-img-decoding-auto-state">Auto</dfn>
     <td>Indicates no preference in decoding mode (the default).
  </table>

  <p>When <span data-x="img-decoding-process">decoding</span> an image, the user agent should
  respect the preference indicated by the <code data-x="attr-img-decoding">decoding</code>
  attribute's state. If the state indicated is <span
  data-x="attr-img-decoding-auto-state">auto</span>, then the user agent is free to choose any
  decoding behavior.</p>

  <p class="note">It is also possible to control the decoding behavior using the <code
  data-x="dom-img-decode">decode()</code> method. Since the <code
  data-x="dom-img-decode">decode()</code> method performs <span
  data-x="img-decoding-process">decoding</span> independently from the process responsible for
  presenting content to screen, it is unaffected by the <code
  data-x="attr-img-decoding">decoding</code> attribute.</p>

  <h6>Updating the image data</h6>

  <p class="note">This algorithm cannot be called from steps running <span>in parallel</span>. If
  a user agent needs to call this algorithm from steps running <span>in parallel</span>, it needs to
  <span data-x="queue a task">queue</span> a task to do so.</p>

  <p>When the user agent is to <dfn>update the image data</dfn> of an <code>img</code> element,
  optionally with the <i>restart animations</i> flag set, optionally with the <i>maybe omit
  events</i> flag set, it must run the following steps:</p>

  <ol>
   <li>
    <p>If the element's <span>node document</span> is not <span>fully active</span>, then:</p>

    <ol>
     <li><p>Continue running this algorithm <span>in parallel</span>.</p></li>

     <li><p>Wait until the element's <span>node document</span> is <span>fully
     active</span>.</p></li>

     <li><p>If another instance of this algorithm for this <code>img</code> element was started after this instance
     (even if it aborted and is no longer running), then return.</p></li>

     <li><p><span>Queue a microtask</span> to continue this algorithm.</p></li>
    </ol>
   </li>

   <li><p>If the user agent cannot support images, or its support for images has been disabled, then
   <span>abort the image request</span> for the <span>current request</span> and the <span>pending
   request</span>, set <span>current request</span>'s <span data-x="img-req-state">state</span> to
   <span data-x="img-none">unavailable</span>, set <span>pending request</span> to null, and
   return.</p></li>

   <li><p>Let <var>previousURL</var> be the <span>current request</span>'s <span
   data-x="img-req-url">current URL</span>.</p></li>

   <li><p>Let <var>selected source</var> be null and <var>selected pixel density</var> be undefined.

   <li><p>If the element does not <span>use <code>srcset</code> or <code>picture</code></span> and
   it has a <code data-x="attr-img-src">src</code> attribute specified whose value is not the empty
   string, then set <var>selected source</var> to the value of the element's <code
   data-x="attr-img-src">src</code> attribute and set <var>selected pixel density</var> to
   1.0.</p></li>

   <li><p>Set the element's <span>last selected source</span> to <var>selected
   source</var>.</p></li>

   <li>
    <p>If <var>selected source</var> is not null, then:</p>

    <ol>
     <li><p>Let <var>urlString</var> be the result of <span>encoding-parsing-and-serializing a
     URL</span> given <var>selected source</var>, relative to the element's <span>node
     document</span>.</p></li>
     <!-- This does not change currentSrc -->

     <li><p>If <var>urlString</var> is failure, then abort this inner set of steps.</p></li>

     <li><p>Let <var>key</var> be a tuple consisting of <var>urlString</var>, the <code>img</code>
     element's <code data-x="attr-img-crossorigin">crossorigin</code> attribute's mode, and, if that
     mode is not <span data-x="attr-crossorigin-none">No CORS</span>, the <span>node
     document</span>'s <span data-x="concept-document-origin">origin</span>.</p></li>

     <li>
      <p>If the <span>list of available images</span> contains an entry for <var>key</var>,
      then:</p>

      <ol>
       <li><p>Set the <span>ignore higher-layer caching</span> flag for that entry.</p></li>

       <li><p><span>Abort the image request</span> for the <span>current request</span> and the
       <span>pending request</span>.</p></li>

       <li><p>Set <span>pending request</span> to null.</p></li>

       <li><p>Let <span>current request</span> be a new <span>image request</span> whose <span
       data-x="img-req-data">image data</span> is that of the entry and whose <span
       data-x="img-req-state">state</span> is <span data-x="img-all">completely
       available</span>.</p></li>

       <li><p><span data-x="prepare an image for presentation">Prepare <span>current request</span>
       for presentation</span> given the <code>img</code> element.</p></li>

       <li><p>Set <span>current request</span>'s <span>current pixel density</span> to <var>selected
       pixel density</var>.</p></li>

       <li>
        <p><span>Queue an element task</span> on the <span>DOM manipulation task source</span>
        given the <code>img</code> element and the following steps:</p>

        <ol>
         <li><p>If <i>restart animation</i> is set, then <span>restart the
         animation</span>.</p></li>

         <li><p>Set <span>current request</span>'s <span data-x="img-req-url">current URL</span> to
         <var>urlString</var>.</p></li>

         <li><p>If <i>maybe omit events</i> is not set or <var>previousURL</var> is not equal to
         <var>urlString</var>, then <span data-x="concept-event-fire">fire an event</span> named
         <code data-x="event-load">load</code> at the <code>img</code> element.</p></li>
        </ol>
       </li>

       <li><p>Abort the <span>update the image data</span> algorithm.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p><span>Queue a microtask</span> to perform the rest of this algorithm, allowing the <span
   data-x="concept-task">task</span> that invoked this algorithm to continue.</p></li>

   <li>
    <p>If another instance of this algorithm for this <code>img</code> element was started after
    this instance (even if it aborted and is no longer running), then return.</p>

    <p class="note">Only the last instance takes effect, to avoid multiple requests when, for
    example, the <code data-x="attr-img-src">src</code>, <code data-x="attr-img-srcset">srcset</code>,
    and <code data-x="attr-img-crossorigin">crossorigin</code> attributes are all set in
    succession.</p>
   </li>

   <li><p>Let <var>selected source</var> and <var>selected pixel density</var> be the URL and pixel
   density that results from <span data-x="select an image source">selecting an image source</span>,
   respectively.</p></li>

   <li>
    <p>If <var>selected source</var> is null, then:</p>

    <ol>
     <li><p>Set the <span>current request</span>'s <span data-x="img-req-state">state</span> to
     <span data-x="img-error">broken</span>, <span>abort the image request</span> for the
     <span>current request</span> and the <span>pending request</span>, and set <span>pending
     request</span> to null.</p></li>

     <li>
      <p><span>Queue an element task</span> on the <span>DOM manipulation task source</span> given
      the <code>img</code> element and the following steps:</p>

      <ol>
       <li><p>Change the <span>current request</span>'s <span data-x="img-req-url">current
       URL</span> to the empty string.</p></li>

       <li>
        <p>If all of the following are true:</p>

        <ul>
         <li><p>the element has a <code data-x="attr-img-src">src</code> attribute or it <span
         data-x="use srcset or picture">uses <code>srcset</code> or
         <code>picture</code></span>; and</p></li>

         <li><p><i>maybe omit events</i> is not set or <var>previousURL</var> is not the empty
         string,</p></li>
        </ul>

        <p>then <span data-x="concept-event-fire">fire an event</span> named <code
        data-x="event-error">error</code> at the <code>img</code> element.</p>
       </li>
      </ol>
     </li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li><p>Let <var>urlString</var> be the result of <span>encoding-parsing-and-serializing a
   URL</span> given <var>selected source</var>, relative to the element's <span>node
   document</span>.</p></li>

   <li>
    <p>If <var>urlString</var> is failure, then:</p>

    <ol>
     <li><p><span>Abort the image request</span> for the <span>current request</span> and the
     <span>pending request</span>.</p></li>

     <li><p>Set the <span>current request</span>'s <span data-x="img-req-state">state</span> to
     <span data-x="img-error">broken</span>.</p></li>

     <li><p>Set <span>pending request</span> to null.</p></li>

     <li>
      <p><span>Queue an element task</span> on the <span>DOM manipulation task
      source</span> given the <code>img</code> element and the following steps:</p>

      <ol>
       <li><p>Change the <span>current request</span>'s <span data-x="img-req-url">current
       URL</span> to <var>selected source</var>.</p></li>

       <li><p>If <i>maybe omit events</i> is not set or <var>previousURL</var> is not equal to
       <var>selected source</var>, then <span data-x="concept-event-fire">fire an event</span>
       named <code data-x="event-error">error</code> at the <code>img</code> element.</p></li>
      </ol>
     </li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li><p>If the <span>pending request</span> is not null and <var>urlString</var> is the same as
   the <span>pending request</span>'s <span data-x="img-req-url">current URL</span>, then
   return.</p></li>

   <li><p>If <var>urlString</var> is the same as the <span>current request</span>'s <span
   data-x="img-req-url">current URL</span> and <span>current request</span>'s <span
   data-x="img-req-state">state</span> is <span data-x="img-inc">partially available</span>, then
   <span>abort the image request</span> for the <span>pending request</span>, <span>queue an element
   task</span> on the <span>DOM manipulation task source</span> given the <code>img</code> element
   to <span>restart the animation</span> if <i>restart animation</i> is set, and return.</p></li>

   <li><p><span>Abort the image request</span> for the <span>pending request</span>.</p></li>

   <li><p>Set <var>image request</var> to a new <span>image request</span> whose <span
   data-x="img-req-url">current URL</span> is <var>urlString</var>.</p></li>

   <li><p>If <span>current request</span>'s <span data-x="img-req-state">state</span> is <span
   data-x="img-none">unavailable</span> or <span data-x="img-error">broken</span>, then set the
   <span>current request</span> to <var>image request</var>. Otherwise, set the <span>pending
   request</span> to <var>image request</var>.</p></li>

   <li><p>Let <var>request</var> be the result of <span data-x="create a potential-CORS
   request">creating a potential-CORS request</span> given <var>urlString</var>, "<code
   data-x="">image</code>", and the current state of the element's <code
   data-x="attr-img-crossorigin">crossorigin</code> content attribute.</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-client">client</span> to the
   element's <span>node document</span>'s <span>relevant settings object</span>.</p></li>

   <li><p>If the element <span data-x="use srcset or picture">uses <code>srcset</code> or
   <code>picture</code></span>, set <var>request</var>'s <span
   data-x="concept-request-initiator">initiator</span> to "<code
   data-x="">imageset</code>".</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-referrer-policy">referrer
   policy</span> to the current state of the element's <code
   data-x="attr-img-referrerpolicy">referrerpolicy</code> attribute.</p></li>

   <li><p>Set <var>request</var>'s <span
   data-x="concept-request-priority">priority</span> to the current state of the
   element's <code data-x="attr-img-fetchpriority">fetchpriority</code> attribute.</p></li>

   <li><p>Let <var>delay load event</var> be true if the <code>img</code>'s <span>lazy loading
   attribute</span> is in the <span data-x="attr-loading-eager-state">Eager</span> state, or if
   <span data-x="concept-n-noscript">scripting is disabled</span> for the <code>img</code>, and
   false otherwise.</p></li>

   <li>
    <p>If the <span>will lazy load element steps</span> given the <code>img</code> return true,
    then:</p>

    <ol>
     <li><p>Set the <code>img</code>'s <span>lazy load resumption steps</span> to the rest of this
     algorithm starting with the step labeled <i>fetch the image</i>.</p></li>

     <li><p><span>Start intersection-observing a lazy loading element</span> for the
     <code>img</code> element.</p></li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li>
    <!--FETCH-->
    <p><i>Fetch the image</i>: <span data-x="concept-fetch">Fetch</span> <var>request</var>. Return
    from this algorithm, and run the remaining steps as part of the fetch's <i
    data-x="processResponse">processResponse</i> for the <span
    data-x="concept-response">response</span> <var>response</var>.</p>

    <!-- TODO "Interaction with the Preload Scanner" from the picture spec -->

    <p>The resource obtained in this fashion, if any, is <var>image request</var>'s <span
    data-x="img-req-data">image data</span>. It can be either <span>CORS-same-origin</span> or
    <span>CORS-cross-origin</span>; this affects the image's interaction with other APIs (e.g.,
    when used on a <code>canvas</code>).</p>

    <!-- same text in <input type=image> section and similar text elsewhere -->
    <p>When <var>delay load event</var> is true, fetching the image must <span>delay the load
    event</span> of the element's <span>node document</span> until the <span
    data-x="concept-task">task</span> that is <span data-x="queue a task">queued</span> by the
    <span>networking task source</span> once the resource has been fetched (<a
    href="#img-load">defined below</a>) has been run.
    <!--TODO or the fetch was aborted, presumably --></p>

    <p class="warning">This, unfortunately, can be used to perform a rudimentary port scan of the
    user's local network (especially in conjunction with scripting, though scripting isn't actually
    necessary to carry out such an attack). User agents may implement <span
    data-x="origin">cross-origin</span> access control policies that are stricter than those
    described above to mitigate this attack, but unfortunately such policies are typically not
    compatible with existing web content.</p>
   </li>

   <li>
    <p>As soon as possible, jump to the first applicable entry from the following list:</p>

    <dl class="switch">

     <dt>If the resource type is <code>multipart/x-mixed-replace</code></dt>

     <dd>
      <p>The next <span data-x="concept-task">task</span> that is <span data-x="queue a
      task">queued</span> by the <span>networking task source</span> while the image is being
      fetched must run the following steps:</p>

      <ol>
       <li>
        <p>If <var>image request</var> is the <span>pending request</span>
        and at least one body part has been completely decoded,
        <span>abort the image request</span> for the <span>current request</span>,
        and <span>upgrade the pending request to the current request</span>.</p>
       </li>

       <li><p>Otherwise, if <var>image request</var> is the <span>pending request</span> and the
       user agent is able to determine that <var>image request</var>'s image is corrupted in some
       fatal way such that the image dimensions cannot be obtained, <span>abort the image
       request</span> for the <span>current request</span>, <span>upgrade the pending request to the
       current request</span>, and set the <span>current request</span>'s <span
       data-x="img-req-state">state</span> to <span data-x="img-error">broken</span>.</p></li>

       <li><p>Otherwise, if <var>image request</var> is the <span>current request</span>, its <span
       data-x="img-req-state">state</span> is <span data-x="img-none">unavailable</span>, and the
       user agent is able to determine <var>image request</var>'s image's width and height, set the
       <span>current request</span>'s <span data-x="img-req-state">state</span> to <span
       data-x="img-inc">partially available</span>.</p></li>

       <li><p>Otherwise, if <var>image request</var> is the <span>current request</span>, its <span
       data-x="img-req-state">state</span> is <span data-x="img-none">unavailable</span>, and the
       user agent is able to determine that <var>image request</var>'s image is corrupted in some
       fatal way such that the image dimensions cannot be obtained, set the <span>current
       request</span>'s <span data-x="img-req-state">state</span> to <span
       data-x="img-error">broken</span>.
       <!--TODO: abort the fetch? fire error event?--></p>
       </li>
      </ol>

      <p>Each <span data-x="concept-task">task</span> that is <span data-x="queue a
      task">queued</span> by the <span>networking task source</span> while the image is being
      fetched must update the presentation of the image, but as each new body part comes in, if the
      user agent is able to determine the image's width and height, it must <span data-x="prepare an
      image for presentation">prepare the <code>img</code> element's <span>current request</span>
      for presentation</span> given the <code>img</code> element and replace the previous image.
      Once one body part has been completely decoded, perform the following steps:</p>

      <ol>
       <li><p>Set the <code>img</code> element's <span>current request</span>'s <span
       data-x="img-req-state">state</span> to <span data-x="img-all">completely
       available</span>.</p></li>

       <li><p>If <i>maybe omit events</i> is not set or <var>previousURL</var> is not equal to
       <var>urlString</var>, then <span>queue an element task</span> on the <span>DOM manipulation task source</span>
       given the <code>img</code> element to <span data-x="concept-event-fire">fire an event</span>
       named <code data-x="event-load">load</code> at the <code>img</code> element.</p></li>
      </ol>
      <!--TODO what if the image is broken?
          TODO change state and fire in the same task? -->
     </dd>

     <dt>If the resource type and data corresponds to a supported image format, <a
     href="#img-determine-type">as described below</a></dt>

     <dd>
      <p>The next <span data-x="concept-task">task</span> that is <span data-x="queue a
      task">queued</span> by the <span>networking task source</span> while the image is being
      fetched must run the following steps:</p>

      <ol>
       <li><p>If the user agent is able to determine <var>image request</var>'s image's width and
       height, and <var>image request</var> is <span>pending request</span>, set <var>image
       request</var>'s <span data-x="img-req-state">state</span> to <span data-x="img-inc">partially
       available</span>.</p></li>

       <li><p>Otherwise, if the user agent is able to determine <var>image request</var>'s image's
       width and height, and <var>image request</var> is <span>current request</span>, <span
       data-x="prepare an image for presentation">prepare <var>image request</var> for
       presentation</span> given the <code>img</code> element and set <var>image request</var>'s
       <span data-x="img-req-state">state</span> to <span data-x="img-inc">partially
       available</span>.</p></li>

       <li>
        <p>Otherwise, if the user agent is able to determine that <var>image request</var>'s image
        is corrupted in some fatal way such that the image dimensions cannot be obtained, and
        <var>image request</var> is <span>pending request</span>:</p>

        <ol>
         <li><p><span>Abort the image request</span> for the <span>current request</span> and the
         <span>pending request</span>.</p></li>

         <li><p><span>Upgrade the pending request to the current request</span>.</p></li>

         <li><p>Set <span>current request</span>'s <span data-x="img-req-state">state</span> to
         <span data-x="img-error">broken</span>.</p></li>

         <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
         data-x="event-error">error</code> at the <code>img</code> element.</p></li>
        </ol>
       </li>

       <li>
        <p>Otherwise, if the user agent is able to determine that <var>image request</var>'s image
        is corrupted in some fatal way such that the image dimensions cannot be obtained, and
        <var>image request</var> is <span>current request</span>:</p>

        <ol>
         <li><p><span>Abort the image request</span> for <var>image request</var>.</p></li>

         <li><p>If <i>maybe omit events</i> is not set or <var>previousURL</var> is not equal to
         <var>urlString</var>, then <span data-x="concept-event-fire">fire an event</span> named
         <code data-x="event-error">error</code> at the <code>img</code> element.</p></li>
        </ol>
       </li>
      </ol>

      <p id="img-load">That <span data-x="concept-task">task</span>, and each subsequent <span
      data-x="concept-task">task</span>, that is <span data-x="queue a task">queued</span> by the
      <span>networking task source</span> while the image is being fetched, if <var>image
      request</var> is the <span>current request</span>, must update the presentation of the image
      appropriately (e.g., if the image is a progressive JPEG, each packet can improve the
      resolution of the image).</p>

      <p>Furthermore, the last <span data-x="concept-task">task</span> that is <span data-x="queue a
      task">queued</span> by the <span>networking task source</span> once the resource has been
      fetched must additionally run these steps:</p>

      <ol>
       <!--
       the download was successful
       the user agent was able to determine the image's width and height
       image request is current request or pending request
       pending request might be null or non-null
       -->

       <li><p>If <var>image request</var> is the <span>pending request</span>,
       <span>abort the image request</span> for the <span>current request</span>,
       <span>upgrade the pending request to the current request</span> and
       <span data-x="prepare an image for presentation">prepare <var>image request</var> for
       presentation</span> given the <code>img</code> element.</p></li>

       <li><p>Set <var>image request</var> to the <span data-x="img-all">completely
       available</span> state.</p></li>

       <li><p>Add the image to the <span>list of available images</span> using the key
       <var>key</var>, with the <span>ignore higher-layer caching</span> flag set.</p></li>

       <li><p>If <i>maybe omit events</i> is not set or <var>previousURL</var> is not equal to
       <var>urlString</var>, then <span data-x="concept-event-fire">fire an event</span> named
       <code data-x="event-load">load</code> at the <code>img</code> element.</p></li>
      </ol>
     </dd>

     <dt>Otherwise</dt>

     <dd><p>The image data is not in a supported file format; the user agent must set <var>image
     request</var>'s <span data-x="img-req-state">state</span> to <span
     data-x="img-error">broken</span>, <span>abort the image request</span> for the <span>current
     request</span> and the <span>pending request</span>, <span>upgrade the pending request to the
     current request</span> if <var>image request</var> is the <span>pending request</span>, and
     then, if <i>maybe omit events</i> is not set or <var>previousURL</var> is not equal to
     <var>urlString</var>, <span>queue an element task</span> on the <span>DOM manipulation task
     source</span> given the <code>img</code> element to <span data-x="concept-event-fire">fire an
     event</span> named <code data-x="event-error">error</code> at the <code>img</code>
     element.</p></dd>
    </dl>
   </li>
  </ol>

  <p>While a user agent is running the above algorithm for an element <var>x</var>, there must be a
  strong reference from the element's <span>node document</span> to the element <var>x</var>, even
  if that element is not <span>connected</span>.</p>

  <p>To <dfn>abort the image request</dfn> for an <span>image request</span> or null <var>image
  request</var> means to run the following steps:</p>

  <ol>
   <li><p>If <var>image request</var> is null, then return.</p></li>
   <li><p>Forget <var>image request</var>'s <span data-x="img-req-data">image data</span>, if any.</p></li>
   <li><p>Abort any instance of the <span data-x="concept-fetch">fetching</span> algorithm for
   <var>image request</var>, discarding any pending tasks generated by that algorithm.</p></li>
  </ol>

  <p>To <dfn>upgrade the pending request to the current request</dfn> for an <code>img</code> element means to run the following steps:</p>

  <ol>
   <li><p>Let the <code>img</code> element's <span>current request</span> be the <span>pending request</span>.</p></li>

   <li><p>Let the <code>img</code> element's <span>pending request</span> be null.</p></li>
  </ol>


  <h6>Preparing an image for presentation</h6>

  <p>To <dfn>prepare an image for presentation</dfn> for an <span>image request</span>
  <var>req</var> given image element <var>img</var>:</p>

  <ol>
   <li><p>Let <var>exifTagMap</var> be the EXIF tags obtained from <var>req</var>'s <span
   data-x="img-req-data">image data</span>, as defined by the relevant codec.
   <ref>EXIF</ref></p></li>

   <li><p>Let <var>physicalWidth</var> and <var>physicalHeight</var> be the width and height
   obtained from <var>req</var>'s <span data-x="img-req-data">image data</span>, as defined by the
   relevant codec.</p></li>

   <li><p>Let <var>dimX</var> be the value of <var>exifTagMap</var>'s tag <code
   data-x="">0xA002</code> (<code data-x="">PixelXDimension</code>).</p></li>

   <li><p>Let <var>dimY</var> be the value of <var>exifTagMap</var>'s tag <code
   data-x="">0xA003</code> (<code data-x="">PixelYDimension</code>).</p></li>

   <li><p>Let <var>resX</var> be the value of <var>exifTagMap</var>'s tag <code
   data-x="">0x011A</code> (<code data-x="">XResolution</code>).</p></li>

   <li><p>Let <var>resY</var> be the value of <var>exifTagMap</var>'s tag <code
   data-x="">0x011B</code> (<code data-x="">YResolution</code>).</p></li>

   <li><p>Let <var>resUnit</var> be the value of <var>exifTagMap</var>'s tag <code
   data-x="">0x0128</code> (<code data-x="">ResolutionUnit</code>).</p></li>

   <li><p>If either <var>dimX</var> or <var>dimY</var> is not a positive integer, then
   return.</p></li>

   <li><p>If either <var>resX</var> or <var>resY</var> is not a positive floating-point number, then
   return.</p></li>

   <li><p>If <var>resUnit</var> is not equal to <code data-x="">2</code> (<code
   data-x="">Inch</code>), then return.</p></li>

   <li><p>Let <var>widthFromDensity</var> be the value of <var>physicalWidth</var>, multiplied by 72
   and divided by <var>resX</var>.</p></li>

   <li><p>Let <var>heightFromDensity</var> be the value of <var>physicalHeight</var>, multiplied by
   72 and divided by <var>resY</var>.</p></li>

   <li><p>If <var>widthFromDensity</var> is not equal to <var>dimX</var> or
   <var>heightFromDensity</var> is not equal to <var>dimY</var>, then return.</p></li>

   <li>
    <p>If <var>req</var>'s <span data-x="img-req-data">image data</span> is
    <span>CORS-cross-origin</span>, then set <var>img</var>'s <span>natural dimensions</span> to
    <var>dimX</var> and <var>dimY</var>, scale <var>img</var>'s pixel data accordingly, and return.</p>
   </li>

   <li><p>Set <var>req</var>'s <span>preferred density-corrected dimensions</span> to a struct with
   its width set to <var>dimX</var> and its height set to <var>dimY</var>.</p></li>

   <li><p>Update <var>req</var>'s <code>img</code> element's presentation appropriately.</p></li>
  </ol>

  <p class="note">Resolution in EXIF is equivalent to <span data-x="'pt'">CSS points per
  inch</span>, therefore 72 is the base for computing size from resolution.</p>

  <p class="XXX">It is not yet specified what would be the case if EXIF arrives after the image is
  already presented. See <a href="https://github.com/w3c/csswg-drafts/issues/4929">issue
  #4929</a>.</p>



  <h6>Selecting an image source</h6>

  <p>To <dfn>select an image source</dfn> given an <code>img</code> element <var>el</var>:</p>

  <ol>
   <li><p><span>Update the source set</span> for <var>el</var>.</p></li>

   <li><p>If <var>el</var>'s <span>source set</span> is empty, return null as the URL and undefined
   as the pixel density.</p></li>

   <li><p>Return the result of <span data-x="select an image source from a source set">selecting an
   image</span> from <var>el</var>'s <span>source set</span>.</p></li>
  </ol>

  <p>To <dfn>select an image source from a source set</dfn> given a <span>source set</span>
  <var>sourceSet</var>:</p>

  <ol>
   <li><p>If an entry <var>b</var> in <var>sourceSet</var> has the same associated <span>pixel
   density descriptor</span> as an earlier entry <var>a</var> in <var>sourceSet</var>, then remove
   entry <var>b</var>. Repeat this step until none of the entries in <var>sourceSet</var> have the
   same associated <span>pixel density descriptor</span> as an earlier entry.</p></li>

   <li><p>In an <span>implementation-defined</span> manner, choose one <span>image source</span>
   from <var>sourceSet</var>. Let this be <var>selectedSource</var>.</p></li>

   <li><p>Return <var>selectedSource</var> and its associated pixel density.</p></li>
  </ol>

  <h6>Creating a source set from attributes</h6>

  <p>When asked to <dfn>create a source set</dfn> given a string <var>default source</var>, a string
  <var>srcset</var>, a string <var>sizes</var>, and an element or null <var>img</var>:</p>

  <ol>
   <li><p>Let <var>source set</var> be an empty <span>source set</span>.</p></li>

   <li><p>If <var>srcset</var> is not an empty string, then set
   <var>source set</var> to the result of <span data-x="parse a srcset attribute">parsing</span>
   <var>srcset</var>.</p></li>

   <li><p>Let <span>source size</span> be the result of <span
   data-x="parse a sizes attribute">parsing</span> <var>sizes</var> with <var>img</var>.</p></li>

   <li><p>If <var>default source</var> is not the empty string and
   <var>source set</var> does not contain an <span>image source</span> with a <span>pixel
   density descriptor</span> value of 1, and no <span>image source</span> with a <span>width
   descriptor</span>, append <var>default source</var> to <var>source set</var>.</p></li>

   <li><p><span>Normalize the source densities</span> of <var>source set</var>.</p></li>

   <li><p>Return <var>source set</var>.</p></li>
 </ol>


  <h6>Updating the source set</h6>

  <p>When asked to <dfn>update the source set</dfn> for a given <code>img</code> or
  <code>link</code> element <var>el</var>, user agents must do the following:</p>

  <ol>
   <li><p>Set <var>el</var>'s <span>source set</span> to an empty <span>source set</span>.</p></li>

   <li><p>Let <var>elements</var> be « <var>el</var> ».</p></li>

   <li><p>If <var>el</var> is an <code>img</code> element whose parent node is a
   <code>picture</code> element, then <span data-x="list replace">replace</span> the contents of
   <var>elements</var> with <var>el</var>'s parent node's child elements, retaining relative
   order.</p></li>

   <li><p>Let <var>img</var> be <var>el</var> if <var>el</var> is an <code>img</code> element,
   otherwise null.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>child</var> in <var>elements</var>:</p>

    <ol>
     <li>
      <p>If <var>child</var> is <var>el</var>:</p>

      <ol>
       <li><p>Let <var>default source</var> be the empty string.</p></li>

       <li><p>Let <var>srcset</var> be the empty string.</p></li>

       <li><p>Let <var>sizes</var> be the empty string.</p></li>

       <li><p>If <var>el</var> is an <code>img</code> element that has a <code
       data-x="attr-img-srcset">srcset</code> attribute, then set <var>srcset</var> to that
       attribute's value.</p></li>

       <li><p>Otherwise, if <var>el</var> is a <code>link</code> element that has an <code
       data-x="attr-link-imagesrcset">imagesrcset</code> attribute, then set <var>srcset</var> to
       that attribute's value.</p></li>

       <li><p>If <var>el</var> is an <code>img</code> element that has a <code
       data-x="attr-img-sizes">sizes</code> attribute, then set <var>sizes</var> to that attribute's
       value.</p></li>

       <li><p>Otherwise, if <var>el</var> is a <code>link</code> element that has an <code
       data-x="attr-link-imagesizes">imagesizes</code> attribute, then set <var>sizes</var> to that
       attribute's value.</p></li>

       <li><p>If <var>el</var> is an <code>img</code> element that has a <code
       data-x="attr-img-src">src</code> attribute, then set <var>default source</var> to that
       attribute's value.</p></li>

       <li><p>Otherwise, if <var>el</var> is a <code>link</code> element that has an <code
       data-x="attr-link-href">href</code> attribute, then set <var>default source</var> to that
       attribute's value.</p></li>

       <li><p>Let <var>el</var>'s <span>source set</span> be the result of <span data-x="create a
       source set">creating a source set</span> given <var>default source</var>, <var>srcset</var>,
       <var>sizes</var>, and <var>img</var>.</p></li>

       <li>
        <p>Return.</p>

        <p class="note">If <var>el</var> is a <code>link</code> element, then <var>elements</var>
        contains only <var>el</var>, so this step will be reached immediately and the rest of the
        algorithm will not run.</p>
       </li>
      </ol>
     </li>

     <li><p>If <var>child</var> is not a <code>source</code> element, then
     <span>continue</span>.</p></li>

     <li><p>If <var>child</var> does not have a <code data-x="attr-source-srcset">srcset</code>
     attribute, <span>continue</span> to the next child.</p></li>

     <li><p><span data-x="parse a srcset attribute">Parse <var>child</var>'s srcset attribute</span> and
     let the returned <span>source set</span> be <var>source set</var>.</p></li>

     <li><p>If <var>source set</var> has zero <span data-x="image source">image sources</span>,
     <span>continue</span> to the next child.</p></li>

     <li><p>If <var>child</var> has a <code data-x="attr-source-media">media</code> attribute, and
     its value does not <span data-x="matches the environment">match the environment</span>,
     <span>continue</span> to the next child.</p></li>

     <li><p><span data-x="parse a sizes attribute">Parse <var>child</var>'s sizes attribute</span>
     with <var>img</var>, and let <var>source set</var>'s <span>source size</span> be the returned
     value.</p></li>

     <li><p>If <var>child</var> has a <code data-x="attr-source-type">type</code> attribute, and its
     value is an unknown or unsupported <span>MIME type</span>, <span>continue</span> to the next child.</p></li>

     <li><p>If <var>child</var> has <code data-x="attr-dim-width">width</code> or <code
     data-x="attr-dim-height">height</code> attributes, set <var>el</var>'s <span
     data-x="concept-img-dimension-attribute-source">dimension attribute source</span> to
     <var>child</var>. Otherwise, set <var>el</var>'s <span
     data-x="concept-img-dimension-attribute-source">dimension attribute source</span> to
     <var>el</var>.</p></li>

     <li><p><span>Normalize the source densities</span> of <var>source set</var>.</p></li>

     <li><p>Let <var>el</var>'s <span>source set</span> be <var>source set</var>.</p></li>

     <li><p>Return.</p></li>
    </ol>
   </li>
  </ol>

  <p class="note">Each <code>img</code> element independently considers its previous sibling
  <code>source</code> elements plus the <code>img</code> element itself for selecting an <span>image
  source</span>, ignoring any other (invalid) elements, including other <code>img</code> elements in
  the same <code>picture</code> element, or <code>source</code> elements that are following siblings
  of the relevant <code>img</code> element.</p>


  <h6>Parsing a srcset attribute</h6>

  <p>When asked to <dfn>parse a srcset attribute</dfn> from an element, parse the value of the
  element's <span>srcset attribute</span> as follows:</p>

  <ol>
   <li><p>Let <var>input</var> be the value passed to this algorithm.</p></li>

   <li><p>Let <var>position</var> be a pointer into <var>input</var>,
   initially pointing at the start of the string.</p></li>

   <li><p>Let <var>candidates</var> be an initially empty <span>source set</span>.</p></li>

   <li><p><i>Splitting loop</i>: <span>Collect a sequence of code points</span> that are <span>ASCII
   whitespace</span> or U+002C COMMA characters from <var>input</var> given <var>position</var>. If
   any U+002C COMMA characters were collected, that is a <span
   data-x="concept-microsyntax-parse-error">parse error</span>.</p></li>

   <li><p>If <var>position</var> is past the end of <var>input</var>,
   return <var>candidates</var>.</p></li>

   <li><p><span>Collect a sequence of code points</span> that are not <span>ASCII whitespace</span>
   from <var>input</var> given <var>position</var>, and let that be <var>url</var>.</p></li>

   <li><p>Let <var>descriptors</var> be a new empty list.</p></li>

   <li>
    <p>If <var>url</var> ends with U+002C (,), then:</p>

    <ol>
     <li><p>Remove all trailing U+002C COMMA characters from <var>url</var>.
     If this removed more than one character,
     that is a <span data-x="concept-microsyntax-parse-error">parse error</span>.</p></li>
    </ol>

    <p>Otherwise:</p>

    <ol>
     <li><p><i>Descriptor tokenizer</i>: <span>Skip ASCII whitespace</span> within <var>input</var>
     given <var>position</var>.</p></li>

     <li><p>Let <var>current descriptor</var> be the empty string.</p></li>

     <li><p>Let <var>state</var> be <i>in descriptor</i>.</p></li>

     <li>
      <p>Let <var>c</var> be the character at <var>position</var>.
      Do the following depending on the value of <var>state</var>.
      For the purpose of this step, "EOF" is a special character representing
      that <var>position</var> is past the end of <var>input</var>.</p>

      <dl class="switch">

       <dt><i>In descriptor</i></dt>

       <dd>
        <p>Do the following, depending on the value of <var>c</var>:</p>

        <dl class="switch">

         <dt><span>ASCII whitespace</span></dt>

         <dd><p>If <var>current descriptor</var> is not empty,
         append <var>current descriptor</var> to <var>descriptors</var>
         and let <var>current descriptor</var> be the empty string.
         Set <var>state</var> to <i>after descriptor</i>.</p></dd>

         <dt>U+002C COMMA (,)</dt>

         <dd><p>Advance <var>position</var> to the next character in <var>input</var>.
         If <var>current descriptor</var> is not empty,
         append <var>current descriptor</var> to <var>descriptors</var>.
         Jump to the step labeled <i>descriptor parser</i>.</p></dd>

         <dt>U+0028 LEFT PARENTHESIS (()</dt>

         <dd><p>Append <var>c</var> to <var>current descriptor</var>.
         Set <var>state</var> to <i>in parens</i>.</p></dd>

         <dt>EOF</dt>

         <dd><p>If <var>current descriptor</var> is not empty,
         append <var>current descriptor</var> to <var>descriptors</var>.
         Jump to the step labeled <i>descriptor parser</i>.</p></dd>

         <dt>Anything else</dt>

         <dd><p>Append <var>c</var> to <var>current descriptor</var>.</p></dd>
        </dl>
       </dd>

       <dt><i>In parens</i></dt>

       <dd>
        <p>Do the following, depending on the value of <var>c</var>:</p>

        <dl class="switch">

         <dt>U+0029 RIGHT PARENTHESIS ())</dt>

         <dd><p>Append <var>c</var> to <var>current descriptor</var>.
         Set <var>state</var> to <i>in descriptor</i>.</p></dd>

         <dt>EOF</dt>

         <dd><p>Append <var>current descriptor</var> to <var>descriptors</var>.
         Jump to the step labeled <i>descriptor parser</i>.</p></dd>

         <dt>Anything else</dt>

         <dd><p>Append <var>c</var> to <var>current descriptor</var>.</p></dd>
        </dl>
       </dd>

       <dt><i>After descriptor</i></dt>

       <dd>
        <p>Do the following, depending on the value of <var>c</var>:</p>

        <dl class="switch">

         <dt><span>ASCII whitespace</span></dt>

         <dd><p>Stay in this state.</p></dd>

         <dt>EOF</dt>

         <dd><p>Jump to the step labeled <i>descriptor parser</i>.</p></dd>

         <dt>Anything else</dt>

         <dd><p>Set <var>state</var> to <i>in descriptor</i>.
         Set <var>position</var> to the <em>previous</em> character in <var>input</var>.</p></dd>
        </dl>
       </dd>
      </dl>

      <p>Advance <var>position</var> to the next character in <var>input</var>. Repeat this
      step.</p>

      <p class="note">In order to be compatible with future additions,
      this algorithm supports multiple descriptors and descriptors with parens.</p>
     </li>
    </ol>
   </li>

   <li><p><i>Descriptor parser</i>: Let <var>error</var> be <i>no</i>.</p></li>

   <li><p>Let <var>width</var> be <i>absent</i>.</p></li>

   <li><p>Let <var>density</var> be <i>absent</i>.</p></li>

   <li><p>Let <var>future-compat-h</var> be <i>absent</i>.</p></li>

   <li>
    <p>For each descriptor in <var>descriptors</var>,
    run the appropriate set of steps from the following list:</p>

    <dl class=switch>

     <dt>If the descriptor consists of a <span>valid non-negative integer</span>
     followed by a U+0077 LATIN SMALL LETTER W character</dt>

     <dd>
      <ol>
       <li>
        <p>If the user agent does not support the <code data-x="attr-img-sizes">sizes</code> attribute,
        let <var>error</var> be <i>yes</i>.</p>

        <p class="note">A conforming user agent will support the <code data-x="attr-img-sizes">sizes</code> attribute.
        However, user agents typically implement and ship features in an incremental manner in practice.</p>
       </li>

       <li><p>If <var>width</var> and <var>density</var>
       are not both <i>absent</i>,
       then let <var>error</var> be <i>yes</i>.</p></li>

       <li><p>Apply the <span>rules for parsing non-negative integers</span> to the descriptor. If
       the result is 0, let <var>error</var> be <i>yes</i>. Otherwise, let <var>width</var> be the
       result.</p></li>
      </ol>
     </dd>

     <dt>If the descriptor consists of a <span>valid floating-point number</span>
     followed by a U+0078 LATIN SMALL LETTER X character</dt>

     <dd>
      <ol>
       <li><p>If <var>width</var>, <var>density</var> and <var>future-compat-h</var>
       are not all <i>absent</i>,
       then let <var>error</var> be <i>yes</i>.</p></li>

       <li>
        <p>Apply the <span>rules for parsing floating-point number values</span> to the descriptor.
        If the result is less than 0, let <var>error</var> be <i>yes</i>. Otherwise, let
        <var>density</var> be the result.</p>

        <p class="note">If <var>density</var> is 0, the <span>natural dimensions</span> will be
        infinite. User agents are <a href="https://infra.spec.whatwg.org/#algorithm-limits">expected
        to have limits</a> in how big images can be rendered.</p>
       </li>
      </ol>
     </dd>

     <dt>If the descriptor consists of a <span>valid non-negative integer</span>
     followed by a U+0068 LATIN SMALL LETTER H character</dt>

     <dd>
      <p>This is a <span data-x="concept-microsyntax-parse-error">parse error</span>.</p>

      <ol>
       <li><p>If <var>future-compat-h</var> and <var>density</var>
       are not both <i>absent</i>,
       then let <var>error</var> be <i>yes</i>.</p></li>

       <li><p>Apply the <span>rules for parsing non-negative integers</span> to the descriptor. If
       the result is 0, let <var>error</var> be <i>yes</i>. Otherwise, let
       <var>future-compat-h</var> be the result.</p></li>
      </ol>
     </dd>

     <dt>Anything else</dt>

     <dd><p>Let <var>error</var> be <i>yes</i>.</p></dd>
    </dl>
   </li>

   <li><p>If <var>future-compat-h</var> is not <i>absent</i> and <var>width</var> is <i>absent</i>,
   let <var>error</var> be <i>yes</i>.</p></li>

   <li><p>If <var>error</var> is still <i>no</i>,
   then append a new <span>image source</span> to <var>candidates</var>
   whose URL is <var>url</var>,
   associated with a width <var>width</var> if not <i>absent</i>
   and a pixel density <var>density</var> if not <i>absent</i>.
   Otherwise, there is a <span data-x="concept-microsyntax-parse-error">parse error</span>.</p></li>

   <li><p>Return to the step labeled <i>splitting loop</i>.</p></li>
  </ol>


  <h6>Parsing a sizes attribute</h6>

  <p>When asked to <dfn>parse a sizes attribute</dfn> from an element <var>element</var>, with an
  <code>img</code> element or null <var>img</var>:</p>

  <ol>
   <li><p>Let <var>unparsed sizes list</var> be the result of <span data-x="parse a comma-separated
   list of component values">parsing a comma-separated list of component values</span> from the
   value of <var>element</var>'s <span>sizes attribute</span> (or the empty string, if the attribute
   is absent). <ref>CSSSYNTAX</ref></p></li>

   <li><p>Let <var>size</var> be null.</p></li>

   <li>
    <p>For each <var>unparsed size</var> in <var>unparsed sizes list</var>:</p>

    <ol>
     <li><p>Remove all consecutive <span>&lt;whitespace-token></span>s
     from the end of <var>unparsed size</var>.
     If <var>unparsed size</var> is now empty,
     then that is a <span data-x="concept-microsyntax-parse-error">parse error</span>;
     <span>continue</span>.</p></li>

     <li><p>If the last <span>component value</span> in <var>unparsed size</var>
     is a valid non-negative <span>&lt;source-size-value></span>,
     then set <var>size</var> to its value
     and remove the <span>component value</span> from <var>unparsed size</var>.
     Any CSS function other than the <span>math functions</span> is invalid.
     Otherwise, there is a <span data-x="concept-microsyntax-parse-error">parse error</span>;
     <span>continue</span>.</p></li>

     <li>
      <p>If <var>size</var> is <code data-x="valdef-sizes-auto">auto</code>, and <var>img</var>
      is not null, and <var>img</var> is <span>being rendered</span>, and <var>img</var>
      <span>allows auto-sizes</span>, then set <var>size</var> to the <span>concrete object
      size</span> width of <var>img</var>, in <span data-x="'px'">CSS pixels</span>.</p>

      <p class="note">If <var>size</var> is still <code data-x="valdef-sizes-auto">auto</code>,
      then it will be ignored.</p>
     </li>

     <li>
      <p>Remove all consecutive <span>&lt;whitespace-token></span>s from the end of <var>unparsed
      size</var>. If <var>unparsed size</var> is now empty:</p>

      <ol>
       <li><p>If this was not the last item in <var>unparsed sizes list</var>, that is a <span
       data-x="concept-microsyntax-parse-error">parse error</span>.</p></li>

       <li><p>If <var>size</var> is not <code data-x="valdef-sizes-auto">auto</code>, then return
       <var>size</var>. Otherwise, continue.</p></li>
      </ol>
     </li>

     <li><p>Parse the remaining <span data-x="component value">component values</span> in <var>unparsed size</var>
     as a <span>&lt;media-condition></span>.
     If it does not parse correctly,
     or it does parse correctly but the <span>&lt;media-condition></span> evaluates to false,
     <span>continue</span>. <ref>MQ</ref></p></li>

     <li><p>If <var>size</var> is not <code data-x="valdef-sizes-auto">auto</code>, then return
     <var>size</var>. Otherwise, continue.</p></li>
    </ol>
   </li>

   <li><p>Return <code data-x="">100vw</code>.</p></li>
  </ol>

  <p class="note">It is invalid to use a bare <span>&lt;source-size-value></span> that is a <span>&lt;length></span>
  (without an accompanying <span>&lt;media-condition></span>)
  as an entry in the <span>&lt;source-size-list></span> that is not the last entry.
  However, the parsing algorithm allows it at any point in the <span>&lt;source-size-list></span>,
  and will accept it immediately as the size
  if the preceding entries in the list weren't used.
  This is to enable future extensions,
  and protect against simple author errors such as a final trailing comma.
  A bare <code data-x="valdef-sizes-auto">auto</code> keyword is allowed to have other entries
  following it to provide a fallback for legacy user agents.</p>


  <h6>Normalizing the source densities</h6>

  <p>An <span>image source</span> can have a <span>pixel density descriptor</span>, a <span>width
  descriptor</span>, or no descriptor at all accompanying its URL. Normalizing a <span>source
  set</span> gives every <span>image source</span> a <span>pixel density descriptor</span>.</p>

  <p>When asked to <!--en-GB--><dfn id="normalise-the-source-densities">normalize the source
  densities</dfn> of a <span>source set</span> <var>source set</var>, the user agent must do the
  following:</p>

  <ol>
   <li><p>Let <var>source size</var> be <var>source set</var>'s <span>source size</span>.</p></li>

   <li>
    <p>For each <span>image source</span> in <var>source set</var>:</p>

    <ol>
     <li><p>If the <span>image source</span> has a <span>pixel density descriptor</span>, <span>continue</span> to the next
     <span>image source</span>.</p></li>

     <li>
      <p>Otherwise, if the <span>image source</span> has a <span>width descriptor</span>, replace the <span>width
      descriptor</span> with a <span>pixel density descriptor</span> with a <span data-x="pixel density descriptor
      value">value</span> of the <span>width descriptor value</span> divided by <var>source size</var> and a unit
      of <code data-x="">x</code>.</p>

      <p class="note">If the <span>source size</span> is 0, then the density would be infinity,
      which results in the <span>natural dimensions</span> being 0 by 0.</p>
     </li>

     <li><p>Otherwise, give the <span>image source</span> a <span>pixel density descriptor</span> of <code
     data-x="">1x</code>.</p></li>
    </ol>
   </li>
  </ol>


  <h6>Reacting to environment changes</h6>

  <p>The user agent may at any time run the following algorithm to update an <code>img</code>
  element's image in order to <dfn data-x="img-environment-changes">react to changes in the
  environment</dfn>. (User agents are <em>not required</em> to ever run this algorithm; for example,
  if the user is not looking at the page any more, the user agent might want to wait until the user
  has returned to the page before determining which image to use, in case the environment changes
  again in the meantime.)</p>

  <p class=note>User agents are encouraged to run this algorithm in particular when the user changes
  the <span>viewport</span>'s size (e.g. by resizing the window or changing the page zoom), and when
  an <code>img</code> element is <span data-x="node is inserted into a document">inserted into a
  document</span>, so that the <span>density-corrected natural width and height</span> match the
  new <span>viewport</span>, and so that the correct image is chosen when <span>art direction</span>
  is involved.</p>

  <ol>
   <li><p><span>Await a stable state</span>. The <span>synchronous section</span>
   consists of all the remaining steps of this algorithm until the algorithm says the
   <span>synchronous section</span> has ended. (Steps in <span data-x="synchronous
   section">synchronous sections</span> are marked with &#x231B;.)</p></li>

   <li><p>&#x231B; If the <code>img</code> element does not <span>use <code>srcset</code> or
   <code>picture</code></span>, its <span>node document</span> is not <span>fully active</span>, has
   image data whose resource type is <code>multipart/x-mixed-replace</code>, or the <span>pending
   request</span> is not null, then return.</p></li>

   <!-- we don't support replacing push-JPEG images because defining what happens with
   the tasks and events and so on in that case would become implausibly complicated. -->

   <li><p>&#x231B; Let <var>selected source</var> and <var>selected pixel
   density</var> be the URL and pixel density that results from <span data-x="select an image
   source">selecting an image source</span>, respectively.</p></li>

   <li><p>&#x231B; If <var>selected source</var> is null, then return.</p></li>
   <!-- not sure this can ever actually happen -->

   <li><p>&#x231B; If <var>selected source</var> and <var>selected pixel
   density</var> are the same as the element's <span>last selected source</span> and <span>current
   pixel density</span>, then return.</p></li> <!-- note that this check happens before
   base URL resolution, so changing the base URL doesn't trigger an update if nothing else changed
   -->

   <li><p>&#x231B; Let <var>urlString</var> be the result of <span>encoding-parsing-and-serializing
   a URL</span> given <var>selected source</var>, relative to the element's <span>node
   document</span>.</p></li>

   <li><p>&#x231B; If <var>urlString</var> is failure, then return.</p></li>

   <li><p>&#x231B; Let <var>corsAttributeState</var> be the state of the element's <code
   data-x="attr-img-crossorigin">crossorigin</code> content attribute.</p></li>

   <li><p>&#x231B; Let <var>origin</var> be the <code>img</code> element's <span>node
   document</span>'s <span data-x="concept-document-origin">origin</span>.</p></li>

   <li><p>&#x231B; Let <var>client</var> be the <code>img</code> element's <span>node
   document</span>'s <span>relevant settings object</span>.</p></li>

   <li><p>&#x231B; Let <var>key</var> be a tuple consisting of <var>urlString</var>,
   <var>corsAttributeState</var>, and, if <var>corsAttributeState</var> is not <span
   data-x="attr-crossorigin-none">No CORS</span>, <var>origin</var>.</p></li>

   <li><p>&#x231B; Let <var>image request</var> be a new <span>image request</span>
   whose <span data-x="img-req-url">current URL</span> is <var>urlString</var>.</p></li>

   <li><p>&#x231B; Let the element's <span>pending request</span> be <var>image request</var>.</p></li>

   <li><p>End the <span>synchronous section</span>, continuing the remaining steps
   <span>in parallel</span>.</p></li>

   <li>
    <p>If the <span>list of available images</span> contains an entry for <var>key</var>,
    then set <var>image request</var>'s <span data-x="img-req-data">image data</span> to that of the entry.
    Continue to the next step.</p>

    <p>Otherwise:</p>

    <ol>
     <li><p>Let <var>request</var> be the result of <span
     data-x="create a potential-CORS request">creating a potential-CORS request</span> given
     <var>urlString</var>, "<code data-x="">image</code>", and
     <var>corsAttributeState</var>.</p></li>

     <li><p>Set <var>request</var>'s <span data-x="concept-request-client">client</span> to
     <var>client</var>, <span data-x="concept-request-initiator">initiator</span> to "<code
     data-x="">imageset</code>", and set <var>request</var>'s <span>synchronous
     flag</span>.</p></li>

     <li><p>Set <var>request</var>'s
     <span data-x="concept-request-referrer-policy">referrer policy</span> to the current state of
     the element's <code data-x="attr-img-referrerpolicy">referrerpolicy</code> attribute.</p></li>

     <li><p>Set <var>request</var>'s <span
     data-x="concept-request-priority">priority</span> to the current state of the element's <code
     data-x="attr-img-fetchpriority">fetchpriority</code> attribute.</p></li>

     <!--FETCH--><li><p>Let <var>response</var> be the result of <span
     data-x="concept-fetch">fetching</span> <var>request</var>.</p></li>

     <li><p>If <var>response</var>'s <span>unsafe response</span> is a <span>network error</span> or
     if the image format is unsupported (as determined by applying the <span
     data-x="Content-Type sniffing: image">image sniffing rules</span>, again as mentioned earlier),
     or if the user agent is able to determine that <var>image request</var>'s image is corrupted in
     some fatal way such that the image dimensions cannot be obtained, or if the resource type is
     <code>multipart/x-mixed-replace</code>, then let <span>pending request</span> be null and abort
     these steps.</p></li>

     <li><p>Otherwise, <var>response</var>'s <span>unsafe response</span> is <var>image
     request</var>'s <span data-x="img-req-data">image data</span>. It can be either
     <span>CORS-same-origin</span> or <span>CORS-cross-origin</span>; this affects the image's
     interaction with other APIs (e.g., when used on a <code>canvas</code>).</p></li>
    </ol>
   </li>

   <li>
    <p><span>Queue an element task</span> on the <span>DOM manipulation task source</span> given
    the <code>img</code> element and the following steps:</p>

    <ol>
     <li><p>If the <code>img</code> element has experienced <span>relevant mutations</span>
     since this algorithm started, then let <span>pending request</span> be null and abort these
     steps.</p>

     <li><p>Let the <code>img</code> element's <span>last selected source</span> be <var>selected
     source</var> and the <code>img</code> element's <span>current pixel density</span> be
     <var>selected pixel density</var>.</p></li>

     <li><p>Set the <var>image request</var>'s <span data-x="img-req-state">state</span> to <span
     data-x="img-all">completely available</span>.</p></li>

     <li><p>Add the image to the <span>list of available images</span> using the key <var>key</var>,
     with the <span>ignore higher-layer caching</span> flag set.</p></li>

     <li><p><span>Upgrade the pending request to the current request</span>.</p></li>

     <li><p><span data-x="prepare an image for presentation">Prepare <var>image request</var> for
     presentation</span> given the <code>img</code> element.</p></li>

     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-load">load</code> at the <code>img</code> element.</p></li>
    </ol>
   </li>
  </ol>

  </div>


  <h5 id="alt">Requirements for providing text to act as an alternative for images</h5>


  <h6>General guidelines</h6>

  <p>Except where otherwise specified, the <code data-x="attr-img-alt">alt</code> attribute must be
  specified and its value must not be empty; the value must be an appropriate replacement for the
  image. The specific requirements for the <code data-x="attr-img-alt">alt</code> attribute depend on
  what the image is intended to represent, as described in the following sections.</p>

  <p>The most general rule to consider when writing alternative text is the following: <strong>the
  intent is that replacing every image with the text of its <code data-x="attr-img-alt">alt</code>
  attribute does not change the meaning of the page</strong>.</p>

  <p>So, in general, alternative text can be written by considering what one would have written had
  one not been able to include the image.</p>

  <p>A corollary to this is that the <code data-x="attr-img-alt">alt</code> attribute's value should
  never contain text that could be considered the image's <em>caption</em>, <em>title</em>, or
  <em>legend</em>. It is supposed to contain replacement text that could be used by users
  <em>instead</em> of the image; it is not meant to supplement the image. The <code
  data-x="attr-title">title</code> attribute can be used for supplemental information.</p>

  <p>Another corollary is that the <code data-x="attr-img-alt">alt</code> attribute's value should
  not repeat information that is already provided in the prose next to the image.</p>

  <p class="note">One way to think of alternative text is to think about how you would read the page
  containing the image to someone over the phone, without mentioning that there is an image present.
  Whatever you say instead of the image is typically a good start for writing the alternative
  text.</p>


  <h6>A link or button containing nothing but the image</h6>

  <p>When an <code>a</code> element that creates a <span>hyperlink</span>, or a <code>button</code>
  element, has no textual content but contains one or more images, the <code
  data-x="attr-img-alt">alt</code> attributes must contain text that together convey the purpose of
  the link or button.</p>

  <div class="example">

   <p>In this example, a user is asked to pick their preferred color from a list of three. Each color
   is given by an image, but for users who have configured their user agent not to display images,
   the color names are used instead:</p>

   <pre><code class="html">&lt;h1>Pick your color&lt;/h1>
&lt;ul>
 &lt;li>&lt;a href="green.html"><strong>&lt;img src="green.jpeg" alt="Green"></strong>&lt;/a>&lt;/li>
 &lt;li>&lt;a href="blue.html"><strong>&lt;img src="blue.jpeg" alt="Blue"></strong>&lt;/a>&lt;/li>
 &lt;li>&lt;a href="red.html"><strong>&lt;img src="red.jpeg" alt="Red"></strong>&lt;/a>&lt;/li>
&lt;/ul></code></pre>

  </div>

  <div class="example">

   <p>In this example, each button has a set of images to indicate the kind of color output desired
   by the user. The first image is used in each case to give the alternative text.</p>

   <pre><code class="html">&lt;button name="rgb"><strong>&lt;img src="red" alt="RGB">&lt;img src="green" alt="">&lt;img src="blue" alt=""></strong>&lt;/button>
&lt;button name="cmyk"><strong>&lt;img src="cyan" alt="CMYK">&lt;img src="magenta" alt="">&lt;img src="yellow" alt="">&lt;img src="black" alt=""></strong>&lt;/button></code></pre>

   <p>Since each image represents one part of the text, it could also be written like this:</p>

   <pre><code class="html">&lt;button name="rgb"><strong>&lt;img src="red" alt="R">&lt;img src="green" alt="G">&lt;img src="blue" alt="B"></strong>&lt;/button>
&lt;button name="cmyk"><strong>&lt;img src="cyan" alt="C">&lt;img src="magenta" alt="M">&lt;img src="yellow" alt="Y">&lt;img src="black" alt="K"></strong>&lt;/button></code></pre>

   <p>However, with other alternative text, this might not work, and putting all the alternative
   text into one image in each case might make more sense:</p>

   <pre><code class="html">&lt;button name="rgb"><strong>&lt;img src="red" alt="sRGB profile">&lt;img src="green" alt="">&lt;img src="blue" alt=""></strong>&lt;/button>
&lt;button name="cmyk"><strong>&lt;img src="cyan" alt="CMYK profile">&lt;img src="magenta" alt="">&lt;img src="yellow" alt="">&lt;img src="black" alt=""></strong>&lt;/button></code></pre>

  </div>



  <h6>A phrase or paragraph with an alternative graphical representation: charts, diagrams, graphs, maps, illustrations</h6>

  <p>Sometimes something can be more clearly stated in graphical form, for example as a flowchart, a
  diagram, a graph, or a simple map showing directions. In such cases, an image can be given using
  the <code>img</code> element, but the lesser textual version must still be given, so that users
  who are unable to view the image (e.g. because they have a very slow connection, or because they
  are using a text-only browser, or because they are listening to the page being read out by a
  hands-free automobile voice web browser, or simply because they are blind) are still able to
  understand the message being conveyed.</p>

  <p>The text must be given in the <code data-x="attr-img-alt">alt</code> attribute, and must convey
  the same message as the image specified in the <code data-x="attr-img-src">src</code>
  attribute.</p>

  <p>It is important to realize that the alternative text is a <em>replacement</em> for the image,
  not a description of the image.</p>

  <div class="example">

   <p>In the following example we have <a href="/images/parsing-model-overview.svg">a flowchart</a>
   in image form, with text in the <code data-x="attr-img-alt">alt</code> attribute rephrasing the
   flowchart in prose form:</p>

   <pre><code class="html">&lt;p>In the common case, the data handled by the tokenization stage
comes from the network, but it can also come from script.&lt;/p>
&lt;p><strong>&lt;img src="images/parsing-model-overview.svg" alt="The Network
passes data to the Input Stream Preprocessor, which passes it to the
Tokenizer, which passes it to the Tree Construction stage. From there,
data goes to both the DOM and to Script Execution. Script Execution is
linked to the DOM, and, using document.write(), passes data to the
Tokenizer."></strong>&lt;/p></code></pre>

  </div>

  <div class="example">

   <p>Here's another example, showing a good solution and a bad solution to the problem of including
   an image in a description.</p>

   <p>First, here's the good solution. This sample shows how the alternative text should just be
   what you would have put in the prose if the image had never existed.</p>

   <pre><code class="html">&lt;!-- This is the correct way to do things. -->
&lt;p>
 You are standing in an open field west of a house.
 <strong>&lt;img src="house.jpeg" alt="The house is white, with a boarded front door."></strong>
 There is a small mailbox here.
&lt;/p></code></pre>

   <p>Second, here's the bad solution. In this incorrect way of doing things, the alternative text
   is simply a description of the image, instead of a textual replacement for the image. It's bad
   because when the image isn't shown, the text doesn't flow as well as in the first example.</p>

   <pre class="bad"><code class="html">&lt;!-- <em>This is the wrong way to do things.</em> -->
&lt;p>
 You are standing in an open field west of a house.
 &lt;img src="house.jpeg" alt="A white house, with a boarded front door.">
 There is a small mailbox here.
&lt;/p></code></pre>

   <p>Text such as "Photo of white house with boarded door" would be equally bad alternative text
   (though it could be suitable for the <code data-x="attr-title">title</code> attribute or in the
   <code>figcaption</code> element of a <code>figure</code> with this image).</p>

  </div>


  <h6>A short phrase or label with an alternative graphical representation: icons, logos</h6>

  <p>A document can contain information in iconic form. The icon is intended to help users of visual
  browsers to recognize features at a glance.</p>

  <p>In some cases, the icon is supplemental to a text label conveying the same meaning. In those
  cases, the <code data-x="attr-img-alt">alt</code> attribute must be present but must be empty.</p>

  <div class="example">

   <p>Here the icons are next to text that conveys the same meaning, so they have an empty <code
   data-x="attr-img-alt">alt</code> attribute:</p>

   <pre><code class="html">&lt;nav>
 &lt;p>&lt;a href="/help/"><strong>&lt;img src="/icons/help.png" alt=""></strong> Help&lt;/a>&lt;/p>
 &lt;p>&lt;a href="/configure/"><strong>&lt;img src="/icons/configuration.png" alt=""></strong>
 Configuration Tools&lt;/a>&lt;/p>
&lt;/nav></code></pre>

  </div>

  <p>In other cases, the icon has no text next to it describing what it means; the icon is supposed
  to be self-explanatory. In those cases, an equivalent textual label must be given in the <code
  data-x="attr-img-alt">alt</code> attribute.</p>

  <div class="example">

   <p>Here, posts on a news site are labeled with an icon indicating their topic.</p>

   <pre><code class="html">&lt;body>
 &lt;article>
  &lt;header>
   &lt;h1>Ratatouille wins &lt;i>Best Movie of the Year&lt;/i> award&lt;/h1>
   &lt;p><strong>&lt;img src="movies.png" alt="Movies"></strong>&lt;/p>
  &lt;/header>
  &lt;p>Pixar has won yet another &lt;i>Best Movie of the Year&lt;/i> award,
  making this its 8th win in the last 12 years.&lt;/p>
 &lt;/article>
 &lt;article>
  &lt;header>
   &lt;h1>Latest TWiT episode is online&lt;/h1>
   &lt;p><strong>&lt;img src="podcasts.png" alt="Podcasts"></strong>&lt;/p>
  &lt;/header>
  &lt;p>The latest TWiT episode has been posted, in which we hear
  several tech news stories as well as learning much more about the
  iPhone. This week, the panelists compare how reflective their
  iPhones' Apple logos are.&lt;/p>
 &lt;/article>
&lt;/body></code></pre>

  </div>

  <p>Many pages include logos, insignia, flags, or emblems, which stand for a particular entity such
  as a company, organization, project, band, software package, country, or some such.</p>

  <p>If the logo is being used to represent the entity, e.g. as a page heading, the <code
  data-x="attr-img-alt">alt</code> attribute must contain the name of the entity being represented by
  the logo. The <code data-x="attr-img-alt">alt</code> attribute must <em>not</em> contain text like
  the word "logo", as it is not the fact that it is a logo that is being conveyed, it's the entity
  itself.</p>

  <p>If the logo is being used next to the name of the entity that it represents, then the logo is
  supplemental, and its <code data-x="attr-img-alt">alt</code> attribute must instead be empty.</p>

  <p>If the logo is merely used as decorative material (as branding, or, for example, as a side
  image in an article that mentions the entity to which the logo belongs), then the entry below on
  purely decorative images applies. If the logo is actually being discussed, then it is being used
  as a phrase or paragraph (the description of the logo) with an alternative graphical
  representation (the logo itself), and the first entry above applies.</p>

  <div class="example">

   <p>In the following snippets, all four of the above cases are present. First, we see a logo used
   to represent a company:</p>

   <pre><code class="html">&lt;h1><strong>&lt;img src="XYZ.gif" alt="The XYZ company"></strong>&lt;/h1></code></pre>

   <p>Next, we see a paragraph which uses a logo right next to the company name, and so doesn't have
   any alternative text:

   <pre><code class="html">&lt;article>
 &lt;h2>News&lt;/h2>
 &lt;p>We have recently been looking at buying the <strong>&lt;img src="alpha.gif"
 alt=""> &Alpha;&Beta;&Gamma; company</strong>, a small Greek company
 specializing in our type of product.&lt;/p></code></pre>

   <p>In this third snippet, we have a logo being used in an aside, as part of the larger article
   discussing the acquisition:</p>

<pre><code class="html"><strong>&lt;aside>&lt;p>&lt;img src="alpha-large.gif" alt="">&lt;/p>&lt;/aside></strong>
 &lt;p>The &Alpha;&Beta;&Gamma; company has had a good quarter, and our
 pie chart studies of their accounts suggest a much bigger blue slice
 than its green and orange slices, which is always a good sign.&lt;/p>
&lt;/article></code></pre>

   <p>Finally, we have an opinion piece talking about a logo, and the logo is therefore described in
   detail in the alternative text.</p>

   <pre><code class="html">&lt;p>Consider for a moment their logo:&lt;/p>

<strong>&lt;p>&lt;img src="/images/logo" alt="It consists of a green circle with a
green question mark centered inside it.">&lt;/p></strong>

&lt;p>How unoriginal can you get? I mean, oooooh, a question mark, how
&lt;em>revolutionary&lt;/em>, how utterly &lt;em>ground-breaking&lt;/em>, I'm
sure everyone will rush to adopt those specifications now! They could
at least have tried for some sort of, I don't know, sequence of
rounded squares with varying shades of green and bold white outlines,
at least that would look good on the cover of a blue book.&lt;/p></code></pre>

   <p>This example shows how the alternative text should be written such that if the image isn't <i
   data-x="img-available">available</i>, and the text is used instead, the text flows seamlessly into
   the surrounding text, as if the image had never been there in the first place.</p>

  </div>


  <h6>Text that has been rendered to a graphic for typographical effect</h6>

  <p>Sometimes, an image just consists of text, and the purpose of the image is not to highlight the
  actual typographic effects used to render the text, but just to convey the text itself.</p>

  <p>In such cases, the <code data-x="attr-img-alt">alt</code> attribute must be present but must
  consist of the same text as written in the image itself.</p>

  <div class="example">

   <p>Consider a graphic containing the text "Earth Day", but with the letters all decorated with
   flowers and plants. If the text is merely being used as a heading, to spice up the page for
   graphical users, then the correct alternative text is just the same text "Earth Day", and no
   mention need be made of the decorations:</p>

   <pre><code class="html">&lt;h1><strong>&lt;img src="earthdayheading.png" alt="Earth Day"></strong>&lt;/h1></code></pre>

  </div>

  <div class="example">

   <p>An illuminated manuscript might use graphics for some of its images. The alternative text in
   such a situation is just the character that the image represents.</p>

   <pre><code class="html">&lt;p>&lt;img src="initials/o.svg" alt="O">nce upon a time and a long long time ago, late at
night, when it was dark, over the hills, through the woods, across a great ocean, in a land far
away, in a small house, on a hill, under a full moon...</code></pre> <!-- The End. -->

  </div>

  <p>When an image is used to represent a character that cannot otherwise be represented in Unicode,
  for example gaiji, itaiji, or new characters such as novel currency symbols, the alternative text
  should be a more conventional way of writing the same thing, e.g. using the phonetic hiragana or
  katakana to give the character's pronunciation.</p>

  <div class="example">

   <p>In this example from 1997, a new-fangled currency symbol that looks like a curly E with two
   bars in the middle instead of one is represented using an image. The alternative text gives the
   character's pronunciation.</p>

   <pre><code class="html">&lt;p>Only &lt;img src="euro.png" alt="euro ">5.99!</code></pre>

  </div>

  <p>An image should not be used if characters would serve an identical purpose. Only when the text
  cannot be directly represented using text, e.g., because of decorations or because there is no
  appropriate character (as in the case of gaiji), would an image be appropriate.</p>

  <p class="note">If an author is tempted to use an image because their default system font does not
  support a given character, then web fonts are a better solution than images.</p>



  <h6>A graphical representation of some of the surrounding text</h6>

  <p>In many cases, the image is actually just supplementary, and its presence merely reinforces the
  surrounding text. In these cases, the <code data-x="attr-img-alt">alt</code> attribute must be
  present but its value must be the empty string.</p>

  <p>In general, an image falls into this category if removing the image doesn't make the page any
  less useful, but including the image makes it a lot easier for users of visual browsers to
  understand the concept.</p>

  <div class="example">

   <p>A flowchart that repeats the previous paragraph in graphical form:</p>

   <pre><code class="html">&lt;p>The Network passes data to the Input Stream Preprocessor, which
passes it to the Tokenizer, which passes it to the Tree Construction
stage. From there, data goes to both the DOM and to Script Execution.
Script Execution is linked to the DOM, and, using document.write(),
passes data to the Tokenizer.&lt;/p>
<strong>&lt;p>&lt;img src="images/parsing-model-overview.svg" alt="">&lt;/p></strong></code></pre>

   <p>In these cases, it would be wrong to include alternative text that consists of just a caption.
   If a caption is to be included, then either the <code data-x="attr-title">title</code> attribute
   can be used, or the <code>figure</code> and <code>figcaption</code> elements can be used. In the
   latter case, the image would in fact be a phrase or paragraph with an alternative graphical
   representation, and would thus require alternative text.</p>

   <pre><code class="html">&lt;!-- Using the title="" attribute -->
&lt;p>The Network passes data to the Input Stream Preprocessor, which
passes it to the Tokenizer, which passes it to the Tree Construction
stage. From there, data goes to both the DOM and to Script Execution.
Script Execution is linked to the DOM, and, using document.write(),
passes data to the Tokenizer.&lt;/p>
&lt;p><strong>&lt;img src="images/parsing-model-overview.svg" alt=""
        title="Flowchart representation of the parsing model."></strong>&lt;/p></code></pre>

   <pre><code class="html">&lt;!-- Using &lt;figure> and &lt;figcaption> -->
&lt;p>The Network passes data to the Input Stream Preprocessor, which
passes it to the Tokenizer, which passes it to the Tree Construction
stage. From there, data goes to both the DOM and to Script Execution.
Script Execution is linked to the DOM, and, using document.write(),
passes data to the Tokenizer.&lt;/p>
&lt;figure>
 <strong>&lt;img src="images/parsing-model-overview.svg" alt="The Network leads to
 the Input Stream Preprocessor, which leads to the Tokenizer, which
 leads to the Tree Construction stage. The Tree Construction stage
 leads to two items. The first is Script Execution, which leads via
 document.write() back to the Tokenizer. The second item from which
 Tree Construction leads is the DOM. The DOM is related to the Script
 Execution."></strong>
 &lt;figcaption>Flowchart representation of the parsing model.&lt;/figcaption>
&lt;/figure></code></pre>

   <pre class="bad"><code class="html">&lt;!-- This is WRONG. Do not do this. Instead, do what the above examples do. -->
&lt;p>The Network passes data to the Input Stream Preprocessor, which
passes it to the Tokenizer, which passes it to the Tree Construction
stage. From there, data goes to both the DOM and to Script Execution.
Script Execution is linked to the DOM, and, using document.write(),
passes data to the Tokenizer.&lt;/p>
&lt;p>&lt;img src="images/parsing-model-overview.svg"
        alt="Flowchart representation of the parsing model.">&lt;/p>
&lt;!-- Never put the image's caption in the alt="" attribute! --></code></pre>

  </div>

  <div class="example">

   <p>A graph that repeats the previous paragraph in graphical form:</p>

   <pre><code class="html">&lt;p>According to a study covering several billion pages,
about 62% of documents on the web in 2007 triggered the Quirks
rendering mode of web browsers, about 30% triggered the Almost
Standards mode, and about 9% triggered the Standards mode.&lt;/p>
<strong>&lt;p>&lt;img src="rendering-mode-pie-chart.png" alt="">&lt;/p></strong></code></pre>

  </div>



  <h6>Ancillary images</h6>

  <p>Sometimes, an image is not critical to the content, but is nonetheless neither purely
  decorative nor entirely redundant with the text. In these cases, the <code
  data-x="attr-img-alt">alt</code> attribute must be present, and its value should either be the
  empty string, or a textual representation of the information that the image conveys. If the image
  has a caption giving the image's title, then the <code data-x="attr-img-alt">alt</code>
  attribute's value must not be empty (as that would be quite confusing for non-visual readers).</p>

  <div class="example">

   <p>Consider a news article about a political figure, in which the individual's face was shown in
   an image. The image is not purely decorative, as it is relevant to the story. The image is not
   entirely redundant with the story either, as it shows what the politician looks like. Whether
   any alternative text need be provided is an authoring decision, decided by whether the image
   influences the interpretation of the prose.</p>

   <p>In this first variant, the image is shown without context, and no alternative text is
   provided:</p>

   <pre><code class="html">&lt;p><strong>&lt;img src="president.jpeg" alt=""></strong> Ahead of today's referendum,
the President wrote an open letter to all registered voters. In it, she admitted that the country was
divided.&lt;/p></code></pre>

   <p>If the picture is just a face, there might be no value in describing it. It's of no interest
   to the reader whether the individual has red hair or blond hair, whether the individual has white
   skin or black skin, whether the individual has one eye or two eyes.</p>

   <p>However, if the picture is more dynamic, for instance showing the politician as angry, or
   particularly happy, or devastated, some alternative text would be useful in setting the tone of
   the article, a tone that might otherwise be missed:</p>

   <pre><code class="html">&lt;p><strong>&lt;img src="president.jpeg" alt="The President is sad."></strong>
Ahead of today's referendum, the President wrote an open letter to all
registered voters. In it, she admitted that the country was divided.
&lt;/p></code></pre>

   <pre><code class="html">&lt;p><strong>&lt;img src="president.jpeg" alt="The President is happy!"></strong>
Ahead of today's referendum, the President wrote an open letter to all
registered voters. In it, she admitted that the country was divided.
&lt;/p></code></pre>

   <p>Whether the individual was "sad" or "happy" makes a difference to how the rest of the
   paragraph is to be interpreted: is she likely saying that she is unhappy with the country
   being divided, or is she saying that the prospect of a divided country is good for her
   political career? The interpretation varies based on the image.</p>

  </div>

  <div class="example">

   <p>If the image has a caption, then including alternative text avoids leaving the non-visual user
   confused as to what the caption refers to.</p>

   <pre><code class="html">&lt;p>Ahead of today's referendum, the President wrote an open letter to
all registered voters. In it, she admitted that the country was divided.&lt;/p>
<strong>&lt;figure>
 &lt;img src="president.jpeg"
      alt="A high forehead, cheerful disposition, and dark hair round out the President's face.">
 &lt;figcaption> The President of Ruritania. Photo &copy; 2014 PolitiPhoto. &lt;/figcaption>
&lt;/figure></strong></code></pre>

  </div>



  <h6>A purely decorative image that doesn't add any information</h6>

  <p>If an image is decorative but isn't especially page-specific &mdash; for example an image that
  forms part of a site-wide design scheme &mdash; the image should be specified in the site's CSS,
  not in the markup of the document.</p>

  <p>However, a decorative image that isn't discussed by the surrounding text but still has some
  relevance can be included in a page using the <code>img</code> element. Such images are
  decorative, but still form part of the content. In these cases, the <code
  data-x="attr-img-alt">alt</code> attribute must be present but its value must be the empty
  string.</p>

  <div class="example">

   <p>Examples where the image is purely decorative despite being relevant would include things like
   a photo of the Black Rock City landscape in a blog post about an event at Burning Man, or an
   image of a painting inspired by a poem, on a page reciting that poem. The following snippet shows
   an example of the latter case (only the first verse is included in this snippet):</p>

   <pre><code class="html">&lt;h1>The Lady of Shalott&lt;/h1>
<strong>&lt;p>&lt;img src="shalott.jpeg" alt="">&lt;/p></strong>
&lt;p>On either side the river lie&lt;br>
Long fields of barley and of rye,&lt;br>
That clothe the wold and meet the sky;&lt;br>
And through the field the road run by&lt;br>
To many-tower'd Camelot;&lt;br>
And up and down the people go,&lt;br>
Gazing where the lilies blow&lt;br>
Round an island there below,&lt;br>
The island of Shalott.&lt;/p></code></pre>

  </div>


  <h6>A group of images that form a single larger picture with no links</h6>

  <p>When a picture has been sliced into smaller image files that are then displayed together to
  form the complete picture again, one of the images must have its <code
  data-x="attr-img-alt">alt</code> attribute set as per the relevant rules that would be appropriate
  for the picture as a whole, and then all the remaining images must have their <code
  data-x="attr-img-alt">alt</code> attribute set to the empty string.</p>

  <div class="example">

   <p>In the following example, a picture representing a company logo for <span data-x="">XYZ
   Corp</span> has been split into two pieces, the first containing the letters "XYZ" and the second
   with the word "Corp". The alternative text ("XYZ Corp") is all in the first image.</p>

   <pre><code class="html">&lt;h1><strong>&lt;img src="logo1.png" alt="XYZ Corp">&lt;img src="logo2.png" alt=""></strong>&lt;/h1></code></pre>

  </div>

  <div class="example">

   <p>In the following example, a rating is shown as three filled stars and two empty stars. While
   the alternative text could have been "&#x2605;&#x2605;&#x2605;&#x2606;&#x2606;", the author has
   instead decided to more helpfully give the rating in the form "3 out of 5". That is the
   alternative text of the first image, and the rest have blank alternative text.</p>

   <pre><code class="html">&lt;p>Rating: &lt;meter max=5 value=3><strong>&lt;img src="1" alt="3 out of 5"
  >&lt;img src="1" alt="">&lt;img src="1" alt="">&lt;img src="0" alt=""
  >&lt;img src="0" alt=""></strong>&lt;/meter>&lt;/p></code></pre>

  </div>



  <h6>A group of images that form a single larger picture with links</h6>

  <p>Generally, <span data-x="image map">image maps</span> should be used instead of slicing an image
  for links.</p>

  <p>However, if an image is indeed sliced and any of the components of the sliced picture are the
  sole contents of links, then one image per link must have alternative text in its <code
  data-x="attr-img-alt">alt</code> attribute representing the purpose of the link.</p>

  <div class="example">

   <p>In the following example, a picture representing the flying spaghetti monster emblem, with
   each of the left noodly appendages and the right noodly appendages in different images, so that
   the user can pick the left side or the right side in an adventure.</p>

   <pre><code class="html">&lt;h1>The Church&lt;/h1>
&lt;p>You come across a flying spaghetti monster. Which side of His
Noodliness do you wish to reach out for?&lt;/p>
<strong>&lt;p>&lt;a href="?go=left" >&lt;img src="fsm-left.png"  alt="Left side. ">&lt;/a
  >&lt;img src="fsm-middle.png" alt=""
  >&lt;a href="?go=right">&lt;img src="fsm-right.png" alt="Right side.">&lt;/a>&lt;/p></strong></code></pre>

  </div>



  <h6>A key part of the content</h6>

  <p>In some cases, the image is a critical part of the content. This could be the case, for
  instance, on a page that is part of a photo gallery. The image is the whole <em>point</em> of the
  page containing it.</p>

  <p>How to provide alternative text for an image that is a key part of the content depends on the
  image's provenance.</p>

  <dl>
   <dt>The general case</dt>

   <dd>
    <p>When it is possible for detailed alternative text to be provided, for example if the image is
    part of a series of screenshots in a magazine review, or part of a comic strip, or is a
    photograph in a blog entry about that photograph, text that can serve as a substitute for the
    image must be given as the contents of the <code data-x="attr-img-alt">alt</code> attribute.</p>

    <div class="example">

     <p>A screenshot in a gallery of screenshots for a new OS, with some alternative text:</p>

     <pre><code class="html">&lt;figure>
 <strong>&lt;img src="KDE%20Light%20desktop.png"
      alt="The desktop is blue, with icons along the left hand side in
           two columns, reading System, Home, K-Mail, etc. A window is
           open showing that menus wrap to a second line if they
           cannot fit in the window. The window has a list of icons
           along the top, with an address bar below it, a list of
           icons for tabs along the left edge, a status bar on the
           bottom, and two panes in the middle. The desktop has a bar
           at the bottom of the screen with a few buttons, a pager, a
           list of open applications, and a clock."></strong>
 &lt;figcaption>Screenshot of a KDE desktop.&lt;/figcaption>
&lt;/figure></code></pre>

    </div>

    <div class="example">

     <p>A graph in a financial report:</p>

     <pre><code class="html"><strong>&lt;img src="sales.gif"
     title="Sales graph"
     alt="From 1998 to 2005, sales increased by the following percentages
     with each year: 624%, 75%, 138%, 40%, 35%, 9%, 21%"></strong></code></pre>

     <p>Note that "sales graph" would be inadequate alternative text for a sales graph. Text that
     would be a good <em>caption</em> is not generally suitable as replacement text.</p>

    </div>
   </dd>


   <dt>Images that defy a complete description</dt>

   <dd>
    <p>In certain cases, the nature of the image might be such that providing thorough alternative
    text is impractical. For example, the image could be indistinct, or could be a complex fractal,
    or could be a detailed topographical map.</p>

    <p>In these cases, the <code data-x="attr-img-alt">alt</code> attribute must contain some
    suitable alternative text, but it may be somewhat brief.</p>

    <div class="example">

     <p>Sometimes there simply is no text that can do justice to an image. For example, there is
     little that can be said to usefully describe a Rorschach inkblot test. However, a description,
     even if brief, is still better than nothing:</p>

     <pre><code class="html">&lt;figure>
 <strong>&lt;img src="/commons/a/a7/Rorschach1.jpg" alt="A shape with left-right
 symmetry with indistinct edges, with a small gap in the center, two
 larger gaps offset slightly from the center, with two similar gaps
 under them. The outline is wider in the top half than the bottom
 half, with the sides extending upwards higher than the center, and
 the center extending below the sides."></strong>
 &lt;figcaption>A black outline of the first of the ten cards
 in the Rorschach inkblot test.&lt;/figcaption>
&lt;/figure></code></pre>

     <p>Note that the following would be a very bad use of alternative text:</p>

     <pre class="bad"><code class="html">&lt;!-- This example is wrong. Do not copy it. -->
&lt;figure>
 &lt;img src="/commons/a/a7/Rorschach1.jpg" alt="A black outline
 of the first of the ten cards in the Rorschach inkblot test.">
 &lt;figcaption>A black outline of the first of the ten cards
 in the Rorschach inkblot test.&lt;/figcaption>
&lt;/figure></code></pre>

     <p>Including the caption in the alternative text like this isn't useful because it effectively
     duplicates the caption for users who don't have images, taunting them twice yet not helping
     them any more than if they had only read or heard the caption once.</p>

    </div>

    <div class="example">

     <p>Another example of an image that defies full description is a fractal, which, by definition,
     is infinite in detail.</p>

     <p>The following example shows one possible way of providing alternative text for the full view
     of an image of the Mandelbrot set.</p>

     <pre><code class="html"><strong>&lt;img src="ms1.jpeg" alt="The Mandelbrot set appears as a cardioid with
its cusp on the real axis in the positive direction, with a smaller
bulb aligned along the same center line, touching it in the negative
direction, and with these two shapes being surrounded by smaller bulbs
of various sizes."></strong></code></pre>

    </div>

    <div class="example">

     <p>Similarly, a photograph of a person's face, for example in a biography, can be considered
     quite relevant and key to the content, but it can be hard to fully substitute text for:</p>

     <pre><code class="html">&lt;section class="bio">
 &lt;h1>A Biography of Isaac Asimov&lt;/h1>
 &lt;p>Born &lt;b>Isaak Yudovich Ozimov&lt;/b> in 1920, Isaac was a prolific author.&lt;/p>
 &lt;p>&lt;img src="headpics/asimov.jpeg" alt="Isaac Asimov had dark hair, a tall forehead, and wore glasses.
 Later in life, he wore long white sideburns.">&lt;/p>
 &lt;p>Asimov was born in Russia, and moved to the US when he was three years old.&lt;/p>
 &lt;p>...&lt;/p>
&lt;/section></code></pre>

     <p>In such cases it is unnecessary (and indeed discouraged) to include a reference to the
     presence of the image itself in the alternative text, since such text would be redundant with
     the browser itself reporting the presence of the image. For example, if the alternative text
     was "A photo of Isaac Asimov", then a conforming user agent might read that out as "(Image) A
     photo of Isaac Asimov" rather than the more useful "(Image) Isaac Asimov had dark hair, a tall
     forehead, and wore glasses...".</p>

    </div>
   </dd>


   <dt id="unknown-images">Images whose contents are not known</dt>

   <dd>
    <p>In some unfortunate cases, there might be no alternative text available at all, either
    because the image is obtained in some automated fashion without any associated alternative text
    (e.g., a webcam), or because the page is being generated by a script using user-provided images
    where the user did not provide suitable or usable alternative text (e.g. photograph sharing
    sites), or because the author does not themself know what the images represent (e.g. a blind
    photographer sharing an image on their blog).</p>

    <p>In such cases, the <code data-x="attr-img-alt">alt</code> attribute may be omitted, but one of
    the following conditions must be met as well:</p>

    <ul>
     <!-- when editing this list, search for the two other occurrences of 'critical-no-alt' -->

     <!-- NOTE: the order of these steps is important; it's intended to encourage using <figcaption>
     rather than data-x="" -->

     <li id="figcaption-as-alt-condition"><p>The <code>img</code> element is in a
     <code>figure</code> element that contains a <code>figcaption</code> element that contains
     content other than <span>inter-element whitespace</span>, and, ignoring the
     <code>figcaption</code> element and its descendants, the <code>figure</code> element has no
     <span>flow content</span> descendants other than <span>inter-element whitespace</span> and the
     <code>img</code> element.</p></li>

     <li>
      <p>The <code data-x="attr-title">title</code> attribute is present and has a non-empty
      value.</p>

      <!-- search for title-warning if modifying this paragraph -->
      <p class="note">Relying on the <code data-x="attr-title">title</code> attribute is currently
      discouraged as many user agents do not expose the attribute in an accessible manner as
      required by this specification (e.g. requiring a pointing device such as a mouse to cause a
      tooltip to appear, which excludes keyboard-only users and touch-only users, such as anyone
      with a modern phone or tablet).</p>
     </li>
    </ul>

    <p class="note">Such cases are to be kept to an absolute minimum. If there is even the slightest
    possibility of the author having the ability to provide real alternative text, then it would not
    be acceptable to omit the <code data-x="attr-img-alt">alt</code> attribute.</p>

    <div class="example">

     <p>A photo on a photo-sharing site, if the site received the image with no metadata other than
     the caption, could be marked up as follows:</p>

     <pre><code class="html">&lt;figure>
 <strong>&lt;img src="1100670787_6a7c664aef.jpg"></strong>
 &lt;figcaption>Bubbles traveled everywhere with us.&lt;/figcaption>
&lt;/figure></code></pre>

     <p>It would be better, however, if a detailed description of the important parts of the image
     obtained from the user and included on the page.</p>

    </div>

    <div class="example">

     <p>A blind user's blog in which a photo taken by the user is shown. Initially, the user might
     not have any idea what the photo they took shows:</p>

     <pre><code class="html">&lt;article>
 &lt;h1>I took a photo&lt;/h1>
 &lt;p>I went out today and took a photo!&lt;/p>
 &lt;figure>
  <strong>&lt;img src="photo2.jpeg"></strong>
  &lt;figcaption>A photograph taken blindly from my front porch.&lt;/figcaption>
 &lt;/figure>
&lt;/article></code></pre>

     <p>Eventually though, the user might obtain a description of the image from their friends and
     could then include alternative text:</p>

     <pre><code class="html">&lt;article>
 &lt;h1>I took a photo&lt;/h1>
 &lt;p>I went out today and took a photo!&lt;/p>
 &lt;figure>
  <strong>&lt;img src="photo2.jpeg" alt="The photograph shows my squirrel
  feeder hanging from the edge of my roof. It is half full, but there
  are no squirrels around. In the background, out-of-focus trees fill the
  shot. The feeder is made of wood with a metal grate, and it contains
  peanuts. The edge of the roof is wooden too, and is painted white
  with light blue streaks."></strong>
  &lt;figcaption>A photograph taken blindly from my front porch.&lt;/figcaption>
 &lt;/figure>
&lt;/article></code></pre>

    </div>

    <div class="example">

     <p>Sometimes the entire point of the image is that a textual description is not available, and
     the user is to provide the description. For instance, the point of a CAPTCHA image is to see if
     the user can literally read the graphic. Here is one way to mark up a CAPTCHA (note the <code
     data-x="attr-title">title</code> attribute):</p>

     <pre><code class="html">&lt;p>&lt;label>What does this image say?
<strong>&lt;img src="captcha.cgi?id=8934" title="CAPTCHA"></strong>
&lt;input type=text name=captcha>&lt;/label>
(If you cannot see the image, you can use an &lt;a
href="?audio">audio&lt;/a> test instead.)&lt;/p></code></pre>

     <p>Another example would be software that displays images and asks for alternative text
     precisely for the purpose of then writing a page with correct alternative text. Such a page
     could have a table of images, like this:</p>

     <pre><code class="html">&lt;table>
 &lt;thead>
  &lt;tr> &lt;th> Image &lt;th> Description
 &lt;tbody>
  &lt;tr>
   &lt;td> <strong>&lt;img src="2421.png" title="Image 640 by 100, filename 'banner.gif'"></strong>
   &lt;td> &lt;input name="alt2421">
  &lt;tr>
   &lt;td> <strong>&lt;img src="2422.png" title="Image 200 by 480, filename 'ad3.gif'"></strong>
   &lt;td> &lt;input name="alt2422">
&lt;/table></code></pre>

     <p>Notice that even in this example, as much useful information as possible is still included
     in the <code data-x="attr-title">title</code> attribute.</p>

    </div>

    <p class="note">Since some users cannot use images at all (e.g. because they have a very slow
    connection, or because they are using a text-only browser, or because they are listening to the
    page being read out by a hands-free automobile voice web browser, or simply because they are
    blind), the <code data-x="attr-img-alt">alt</code> attribute is only allowed to be omitted rather
    than being provided with replacement text when no alternative text is available and none can be
    made available, as in the above examples. Lack of effort from the part of the author is not an
    acceptable reason for omitting the <code data-x="attr-img-alt">alt</code> attribute.</p>
   </dd>
  </dl>



  <h6>An image not intended for the user</h6>

  <p>Generally authors should avoid using <code>img</code> elements for purposes other than showing
  images.</p>

  <p>If an <code>img</code> element is being used for purposes other than showing an image, e.g. as
  part of a service to count page views, then the <code data-x="attr-img-alt">alt</code> attribute
  must be the empty string.</p>

  <p>In such cases, the <code data-x="attr-dim-width">width</code> and <code
  data-x="attr-dim-height">height</code> attributes should both be set to zero.</p>


  <h6 id="an-image-in-an-e-mail-or-private-document-intended-for-a-specific-person-who-is-known-to-be-able-to-view-images">An image in an email or private document intended for a specific person who is known to be able to view images</h6>

  <p><i>This section does not apply to documents that are publicly accessible, or whose target
  audience is not necessarily personally known to the author, such as documents on a web site,
  emails sent to public mailing lists, or software documentation.</i></p>

  <p>When an image is included in a private communication (such as an HTML email) aimed at a
  specific person who is known to be able to view images, the <code data-x="attr-img-alt">alt</code>
  attribute may be omitted. However, even in such cases authors are strongly urged to include
  alternative text (as appropriate according to the kind of image involved, as described in the
  above entries), so that the email is still usable should the user use a mail client that does not
  support images, or should the document be forwarded on to other users whose abilities might not
  include easily seeing images.</p>




  <div w-nodev>

  <h6>Guidance for markup generators</h6>

  <p>Markup generators (such as WYSIWYG authoring tools) should, wherever possible, obtain
  alternative text from their users. However, it is recognized that in many cases, this will not be
  possible.</p>

  <p>For images that are the sole contents of links, markup generators should examine the link
  target to determine the title of the target, or the URL of the target, and use information
  obtained in this manner as the alternative text.</p>

  <p>For images that have captions, markup generators should use the <code>figure</code> and
  <code>figcaption</code> elements, or the <code data-x="attr-title">title</code> attribute, to
  provide the image's caption.</p>

  <p>As a last resort, implementers should either set the <code data-x="attr-img-alt">alt</code>
  attribute to the empty string, under the assumption that the image is a purely decorative image
  that doesn't add any information but is still specific to the surrounding content, or omit the
  <code data-x="attr-img-alt">alt</code> attribute altogether, under the assumption that the image is
  a key part of the content.</p>

  <p>Markup generators may specify a <dfn><code data-x="attr-img-generator-unable-to-provide-required-alt">generator-unable-to-provide-required-alt</code></dfn>
  attribute on <code>img</code> elements for which they have been unable to obtain alternative text
  and for which they have therefore omitted the <code data-x="attr-img-alt">alt</code> attribute. The
  value of this attribute must be the empty string. Documents containing such attributes are not
  conforming, but conformance checkers will <a href="#guidance-for-conformance-checkers">silently
  ignore</a> this error.</p>

  <p class="note">This is intended to avoid markup generators from being pressured into replacing
  the error of omitting the <code data-x="attr-img-alt">alt</code> attribute with the even more
  egregious error of providing phony alternative text, because state-of-the-art automated
  conformance checkers cannot distinguish phony alternative text from correct alternative text.</p>

  <p>Markup generators should generally avoid using the image's own filename as the alternative
  text. Similarly, markup generators should avoid generating alternative text from any content that
  will be equally available to presentation user agents (e.g., web browsers).</p>

  <p class="note">This is because once a page is generated, it will typically not be updated,
  whereas the browsers that later read the page can be updated by the user, therefore the browser is
  likely to have more up-to-date and finely-tuned heuristics than the markup generator did when
  generating the page.</p>

  </div>

  <div w-nodev>

  <h6 id="guidance-for-conformance-checkers">Guidance for conformance checkers</h6>

  <p>A conformance checker must report the lack of an <code data-x="attr-img-alt">alt</code>
  attribute as an error unless one of the conditions listed below applies:</p>

  <ul>
   <!-- when editing this list, search for the two other occurrences of 'critical-no-alt' -->

   <li><p>The <code>img</code> element is in a <code>figure</code> element that satisfies <a
   href="#figcaption-as-alt-condition">the conditions described above</a>.</p></li>

   <li><p>The <code>img</code> element has a <code data-x="attr-title">title</code> attribute with a
   value that is not the empty string (also as <a href="#unknown-images">described
   above</a>).</p></li>

   <!-- the following are additional entries not included in the aforementioned list, as they apply
   only to conformance checkers -->

   <li><p>The conformance checker has been configured to assume that the document is an email or
   document intended for a specific person who is known to be able to view images.</p></li>

   <li><p>The <code>img</code> element has a (non-conforming) <code
   data-x="attr-img-generator-unable-to-provide-required-alt">generator-unable-to-provide-required-alt</code>
   attribute whose value is the empty string. A conformance checker that is not reporting the lack
   of an <code data-x="attr-img-alt">alt</code> attribute as an error must also not report the
   presence of the empty <code
   data-x="attr-img-generator-unable-to-provide-required-alt">generator-unable-to-provide-required-alt</code>
   attribute as an error. (This case does not represent a case where the document is conforming,
   only that the generator could not determine appropriate alternative text &mdash; validators are
   not required to show an error in this case, because such an error might encourage markup
   generators to include bogus alternative text purely in an attempt to silence validators.
   Naturally, conformance checkers <em>may</em> report the lack of an <code
   data-x="attr-img-alt">alt</code> attribute as an error even in the presence of the <code
   data-x="attr-img-generator-unable-to-provide-required-alt">generator-unable-to-provide-required-alt</code>
   attribute; for example, there could be a user option to report <em>all</em> conformance errors
   even those that might be the more or less inevitable result of using a markup
   generator.)</p></li>
  </ul>

  </div>


  <h4 split-filename="iframe-embed-object">The
  <dfn element><code>iframe</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Embedded content</span>.</dd>
   <dd><span>Interactive content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>embedded content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="concept-content-nothing">Nothing</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-iframe-src">src</code></dd>
   <dd><code data-x="attr-iframe-srcdoc">srcdoc</code></dd>
   <dd><code data-x="attr-iframe-name">name</code></dd>
   <dd><code data-x="attr-iframe-sandbox">sandbox</code></dd>
   <dd><code data-x="attr-iframe-allow">allow</code></dd>
   <dd><code data-x="attr-iframe-allowfullscreen">allowfullscreen</code></dd>
   <dd><code data-x="attr-dim-width">width</code></dd>
   <dd><code data-x="attr-dim-height">height</code></dd>
   <dd><code data-x="attr-iframe-referrerpolicy">referrerpolicy</code></dd>
   <dd><code data-x="attr-iframe-loading">loading</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-iframe">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-iframe">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLIFrameElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute USVString <span data-x="dom-iframe-src">src</span>;
  [<span>CEReactions</span>] attribute (<code data-x="tt-trustedhtml">TrustedHTML</code> or DOMString) <span data-x="dom-iframe-srcdoc">srcdoc</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-iframe-name">name</span>;
  [SameObject, PutForwards=<span data-x="dom-DOMTokenList-value">value</span>] readonly attribute <span>DOMTokenList</span> <span data-x="dom-iframe-sandbox">sandbox</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-iframe-allow">allow</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-iframe-allowFullscreen">allowFullscreen</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-dim-width">width</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-dim-height">height</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-iframe-referrerPolicy">referrerPolicy</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-iframe-loading">loading</span>;
  readonly attribute <span>Document</span>? <span data-x="dom-iframe-contentDocument">contentDocument</span>;
  readonly attribute <span>WindowProxy</span>? <span data-x="dom-iframe-contentWindow">contentWindow</span>;
  <span>Document</span>? <span data-x="dom-media-getSVGDocument">getSVGDocument</span>();

  // <a href="#HTMLIFrameElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLIFrameElement</code>.</dd>
  </dl>

  <!-- INTRO -->

  <p>The <code>iframe</code> element <span>represents</span> its <span>content navigable</span>.</p>


  <!-- SRC/SRCDOC -->

  <p>The <dfn element-attr for="iframe"><code data-x="attr-iframe-src">src</code></dfn> attribute
  gives the <span>URL</span> of a page that the element's <span>content navigable</span> is to
  contain. The attribute, if present, must be a <span>valid non-empty URL potentially surrounded by
  spaces</span>. If the <code data-x="attr-itemprop">itemprop</code> attribute is specified on an
  <code>iframe</code> element, then the <code data-x="attr-iframe-src">src</code> attribute must
  also be specified.</p>

  <p>The <dfn for="iframe" element-attr><code data-x="attr-iframe-srcdoc">srcdoc</code></dfn>
  attribute gives the content of the page that the element's <span>content navigable</span> is to
  contain. The value of the attribute is used to <span data-x="create navigation params from a
  srcdoc resource">construct</span> <dfn export>an <code>iframe</code> <code
  data-x="attr-iframe-srcdoc">srcdoc</code> document</dfn>, which is a <code>Document</code> whose
  <span data-x="concept-document-url">URL</span> <span>matches <code>about:srcdoc</code></span>.</p>

  <p>The <code data-x="attr-iframe-srcdoc">srcdoc</code> attribute, if present, must have a value
  using <span>the HTML syntax</span> that consists of the following syntactic components, in the
  given order:</p>

  <ol>
   <li>Any number of <span data-x="syntax-comments">comments</span> and <span>ASCII
   whitespace</span>.</li>

   <li>Optionally, a <span data-x="syntax-doctype">DOCTYPE</span>.

   <li>Any number of <span data-x="syntax-comments">comments</span> and <span>ASCII
   whitespace</span>.</li>

   <li>The <span>document element</span>, in the form of an <code>html</code> <span
   data-x="syntax-elements">element</span>.</li>

   <li>Any number of <span data-x="syntax-comments">comments</span> and <span>ASCII
   whitespace</span>.</li>
  </ol>

  <p class="note">The above requirements apply in <span>XML documents</span> as well.

  <div class="example">

   <p>Here a blog uses the <code data-x="attr-iframe-srcdoc">srcdoc</code> attribute in conjunction
   with the <code data-x="attr-iframe-sandbox">sandbox</code> attribute described below to provide
   users of user agents that support this feature with an extra layer of protection from script
   injection in the blog post comments:</p>

   <pre><code class="html">&lt;article>
 &lt;h1>I got my own magazine!&lt;/h1>
 &lt;p>After much effort, I've finally found a publisher, and so now I
 have my own magazine! Isn't that awesome?! The first issue will come
 out in September, and we have articles about getting food, and about
 getting in boxes, it's going to be great!&lt;/p>
 &lt;footer>
  &lt;p>Written by &lt;a href="/users/cap">cap&lt;/a>, 1 hour ago.
 &lt;/footer>
 &lt;article>
  &lt;footer> Thirteen minutes ago, &lt;a href="/users/ch">ch&lt;/a> wrote: &lt;/footer>
  &lt;iframe sandbox srcdoc="&lt;p>did you get a cover picture yet?">&lt;/iframe>
 &lt;/article>
 &lt;article>
  &lt;footer> Nine minutes ago, &lt;a href="/users/cap">cap&lt;/a> wrote: &lt;/footer>
  &lt;iframe sandbox srcdoc="&lt;p>Yeah, you can see it &lt;a href=&amp;quot;/gallery?mode=cover&amp;amp;amp;page=1&amp;quot;>in my gallery&lt;/a>.">&lt;/iframe>
 &lt;/article>
 &lt;article>
  &lt;footer> Five minutes ago, &lt;a href="/users/ch">ch&lt;/a> wrote: &lt;/footer>
  &lt;iframe sandbox srcdoc="&lt;p>hey that's earl's table.
&lt;p>you should get earl&amp;amp;amp;me on the next cover.">&lt;/iframe>
 &lt;/article></code></pre>

   <p>Notice the way that quotes have to be escaped (otherwise the <code
   data-x="attr-iframe-srcdoc">srcdoc</code> attribute would end prematurely), and the way raw
   ampersands (e.g. in URLs or in prose) mentioned in the sandboxed content have to be
   <em>doubly</em> escaped &mdash; once so that the ampersand is preserved when originally parsing
   the <code data-x="attr-iframe-srcdoc">srcdoc</code> attribute, and once more to prevent the
   ampersand from being misinterpreted when parsing the sandboxed content.</p>

   <p>Furthermore, notice that since the <span data-x="syntax-doctype">DOCTYPE</span> is optional in
   <span data-x="an iframe srcdoc document"><code>iframe</code> <code
   data-x="attr-iframe-srcdoc">srcdoc</code> documents</span>, and the <code>html</code>,
   <code>head</code>, and <code>body</code> elements have <a href="#syntax-tag-omission">optional
   start and end tags</a>, and the <code>title</code> element is also optional in <span data-x="an iframe
   srcdoc document"><code>iframe</code> <code data-x="attr-iframe-srcdoc">srcdoc</code>
   documents</span>, the markup in a <code data-x="attr-iframe-srcdoc">srcdoc</code> attribute can be
   relatively succinct despite representing an entire document, since only the contents of the
   <code>body</code> element need appear literally in the syntax. The other elements are still
   present, but only by implication.</p>

  </div>

  <p class="note">In <span>the HTML syntax</span>, authors need only remember to use U+0022
  QUOTATION MARK characters (") to wrap the attribute contents and then to escape all U+0026
  AMPERSAND (&amp;) and U+0022 QUOTATION MARK (") characters, and to specify the <code
  data-x="attr-iframe-sandbox">sandbox</code> attribute, to ensure safe embedding of content. (And
  remember to escape ampersands before quotation marks, to ensure quotation marks become &amp;quot;
  and not &amp;amp;quot;.)</p>

  <p class="note">In XML the U+003C LESS-THAN SIGN character (&lt;) needs to be escaped as well. In
  order to prevent <a href="https://www.w3.org/TR/xml/#AVNormalize">attribute-value
  normalization</a>, some of XML's whitespace characters &mdash; specifically U+0009 CHARACTER
  TABULATION (tab), U+000A LINE FEED (LF), and U+000D CARRIAGE RETURN (CR) &mdash; also need to be
  escaped. <ref>XML</ref></p>

  <p class="note">If the <code data-x="attr-iframe-src">src</code> attribute and the <code
  data-x="attr-iframe-srcdoc">srcdoc</code> attribute are both specified together, the <code
  data-x="attr-iframe-srcdoc">srcdoc</code> attribute takes priority. This allows authors to provide
  a fallback <span>URL</span> for legacy user agents that do not support the <code
  data-x="attr-iframe-srcdoc">srcdoc</code> attribute.</p>


  <div w-nodev>

  <hr> <!-- SRC/SRCDOC PROCESSING MODEL -->

  <p>The <code>iframe</code> <span>HTML element post-connection steps</span>, given
  <var>insertedNode</var>, are:</p>

  <ol>
   <li><p><span>Create a new child navigable</span> for <var>insertedNode</var>.</p></li>

   <li><p>If <var>insertedNode</var> has a <code data-x="attr-iframe-sandbox">sandbox</code>
   attribute, then <span data-x="parse a sandboxing directive">parse the sandboxing
   directive</span> given the attribute's value and <var>insertedNode</var>'s
   <span><code>iframe</code> sandboxing flag set</span>.</p></li>

   <li><p><span>Process the <code>iframe</code> attributes</span> for <var>insertedNode</var>, with
   <var data-x="process-iframe-initial-insertion">initialInsertion</var> set to true.</p></li>
  </ol>

  <p>The <code>iframe</code> <span>HTML element removing steps</span>, given
  <var>removedNode</var>, are to <span>destroy a child navigable</span> given
  <var>removedNode</var>.</p>

  <p class="note">This happens without any <code data-x="event-unload">unload</code> events firing
  (the element's <span>content document</span> is <em><span data-x="destroy a child
  navigable">destroyed</span></em>, not <em><span data-x="unload a
  document">unloaded</span></em>).</p>

  <p class="XXX">Although <code>iframe</code>s are processed while in a <span>shadow tree</span>,
  per the above, several other aspects of their behavior are not well-defined with regards to
  shadow trees. See <a href="https://github.com/whatwg/html/issues/763">issue #763</a> for more
  detail.</p>

  <!-- START of section that's very similar to <frame> -->

  <p>Whenever an <code>iframe</code> element with a non-null <span>content navigable</span> has its
  <code data-x="attr-iframe-srcdoc">srcdoc</code> attribute set, changed, or removed, the user
  agent must <span>process the <code>iframe</code> attributes</span>.</p>

  <p>Similarly, whenever an <code>iframe</code> element with a non-null <span>content
  navigable</span> but with no <code data-x="attr-iframe-srcdoc">srcdoc</code> attribute specified
  has its <code data-x="attr-iframe-src">src</code> attribute set, changed, or removed, the user
  agent must <span>process the <code>iframe</code> attributes</span>.</p> <!-- It doesn't happen
  when the base URL is changed, though. -->

  <p>To <dfn>process the <code>iframe</code> attributes</dfn> for an element <var>element</var>,
  with an optional boolean <dfn
  data-x="process-iframe-initial-insertion"><var>initialInsertion</var></dfn> (default false):</p>

  <ol>
   <li>
    <p>If <var>element</var>'s <code data-x="attr-iframe-srcdoc">srcdoc</code> attribute is
    specified, then:</p>

    <ol>
     <li><p>Set <var>element</var>'s <span>current navigation was lazy loaded</span> boolean to
     false.</p></li>

     <li>
      <p>If the <span>will lazy load element steps</span> given <var>element</var> return true,
      then:</p>

      <ol>
       <li><p>Set <var>element</var>'s <span>lazy load resumption steps</span> to the rest of this
       algorithm starting with the step labeled <i>navigate to the srcdoc resource</i>.</p></li>

       <li><p>Set <var>element</var>'s <span>current navigation was lazy loaded</span> boolean to
       true.</p></li>

       <li><p><span>Start intersection-observing a lazy loading element</span> for
       <var>element</var>.</p></li>

       <li><p>Return.</p></li>
      </ol>
     </li>

     <li>
      <p><i>Navigate to the srcdoc resource</i>: <span>Navigate an <code>iframe</code> or
      <code>frame</code></span> given <var>element</var>, <code>about:srcdoc</code>, the empty
      string, and the value of <var>element</var>'s <code data-x="attr-iframe-srcdoc">srcdoc</code>
      attribute.</p>

      <p>The resulting <code>Document</code> must be considered <span>an <code>iframe</code> <code
      data-x="attr-iframe-srcdoc">srcdoc</code> document</span>.</p>
     </li>
    </ol>
   </li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p>Let <var>url</var> be the result of running the <span>shared attribute processing steps
     for <code>iframe</code> and <code>frame</code> elements</span> given <var>element</var> and
     <var>initialInsertion</var>.</p></li>

     <li><p>If <var>url</var> is null, then return.</p></li>

     <li>
      <p>If <var>url</var> <span>matches <code>about:blank</code></span> and
      <var>initialInsertion</var> is true, then:</p>

      <ol>
       <li><p>Run the <span>iframe load event steps</span> given <var>element</var>.</p></li>

       <li><p>Return.</p></li>
      </ol>
     </li>

     <li><p>Let <var>referrerPolicy</var> be the current state of <var>element</var>'s <code
     data-x="attr-iframe-referrerpolicy">referrerpolicy</code> content attribute.</p></li>

     <li><p>Set <var>element</var>'s <span>current navigation was lazy loaded</span> boolean to
     false.</p></li>

     <li>
      <p>If the <span>will lazy load element steps</span> given <var>element</var> return true,
      then:</p>

      <ol>
       <li><p>Set <var>element</var>'s <span>lazy load resumption steps</span> to the rest of this
       algorithm starting with the step labeled <i>navigate</i>.</p></li>

       <li><p>Set <var>element</var>'s <span>current navigation was lazy loaded</span> boolean to
       true.</p></li>

       <li><p><span>Start intersection-observing a lazy loading element</span> for
       <var>element</var>.</p></li>

       <li><p>Return.</p></li>
      </ol>
     </li>

     <li><p><i>Navigate</i>: <span>Navigate an <code>iframe</code> or <code>frame</code></span>
     given <var>element</var>, <var>url</var>, and <var>referrerPolicy</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p id="otherwise-steps-for-iframe-or-frame-elements">The <dfn>shared attribute processing steps
  for <code>iframe</code> and <code>frame</code> elements</dfn>, given an element
  <var>element</var> and a boolean <var>initialInsertion</var>, are:</p>

  <ol>
   <li><p>Let <var>url</var> be the <span>URL record</span> <code>about:blank</code>.</p></li>

   <li>
    <p>If <var>element</var> has a <code data-x="attr-iframe-src">src</code> attribute specified,
    and its value is not the empty string, then:</p>

    <ol>
     <li><p>Let <var>maybeURL</var> be the result of <span>encoding-parsing a URL</span> given that
     attribute's value, relative to <var>element</var>'s <span>node document</span>.</p></li>

     <li><p>If <var>maybeURL</var> is not failure, then set <var>url</var> to
     <var>maybeURL</var>.</p></li>
    </ol>
   </li>

   <li><p>If the <span>inclusive ancestor navigables</span> of <var>element</var>'s <span>node
   navigable</span> contains a <span>navigable</span> whose  <span data-x="nav-document">active
   document</span>'s <span data-x="concept-document-url">URL</span> <span
   data-x="concept-url-equals">equals</span> <var>url</var> with <i data-x="url equals exclude
   fragments">exclude fragments</i> set to true, then return null.</p></li>
   <!-- https://www.hixie.ch/tests/adhoc/html/frames/iframes/ 008.html and 009.html -->
   <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1969 -->

   <li>
    <p>If <var>url</var> <span>matches <code>about:blank</code></span> and
    <var>initialInsertion</var> is true, then perform the <span>URL and history update steps</span>
    given <var>element</var>'s <span>content navigable</span>'s <span data-x="nav-document">active
    document</span> and <var>url</var>.</p>

    <p class="note">This is necessary in case <var>url</var> is something like <code
    data-x="">about:blank?foo</code>. If <var>url</var> is just plain <code
    data-x="">about:blank</code>, this will do nothing.</p>
   </li>

   <li><p>Return <var>url</var>.</p></li>
  </ol>

  <p>To <dfn>navigate an <code>iframe</code> or <code>frame</code></dfn> given an element
  <var>element</var>, a <span>URL</span> <var>url</var>, a <span>referrer policy</span>
  <var>referrerPolicy</var>, and an optional string-or-null <var>srcdocString</var> (default
  null):</p>

  <ol>
   <li><p>Let <var>historyHandling</var> be "<code
   data-x="NavigationHistoryBehavior-auto">auto</code>".</p>

   <li><p>If <var>element</var>'s <span>content navigable</span>'s <span
   data-x="nav-document">active document</span> is not <span>completely loaded</span>, then set
   <var>historyHandling</var> to "<code
   data-x="NavigationHistoryBehavior-replace">replace</code>".</p></li>

   <li><p>If <var>element</var> is an <code>iframe</code>, then set <var>element</var>'s <span
   data-x="iframe-pending-resource-timing-start-time">pending resource-timing start time</span> to
   the <span>current high resolution time</span> given <var>element</var>'s
   <span>node document</span>'s <span>relevant global object</span>.</p></li>

   <li><p><span>Navigate</span><!--DONAV iframe or frame--> <var>element</var>'s <span>content
   navigable</span> to <var>url</var> using <var>element</var>'s <span>node document</span>, with
   <i data-x="navigation-hh">historyHandling</i> set to <var>historyHandling</var>, <i
   data-x="navigation-referrer-policy">referrerPolicy</i> set to <var>referrerPolicy</var>, and <i
   data-x="navigation-resource">documentResource</i> set to <var>srcdocString</var>.</p></li>
  </ol>

  <p>Each <code>Document</code> has an <dfn>iframe load in progress</dfn> flag and a <dfn>mute
  iframe load</dfn> flag. When a <code>Document</code> is created, these flags must be unset for
  that <code>Document</code>.</p>

  <p>To run the <dfn>iframe load event steps</dfn>, given an <code>iframe</code> element
  <var>element</var>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>element</var>'s <span>content navigable</span> is not
   null.</p></li>

   <li><p>Let <var>childDocument</var> be <var>element</var>'s <span>content navigable</span>'s
   <span data-x="nav-document">active document</span>.</p></li>

   <li><p>If <var>childDocument</var> has its <span>mute iframe load</span> flag set, then
   return.</p></li>

   <li>
    <p>If <var>element</var>'s <span
    data-x="iframe-pending-resource-timing-start-time">pending resource-timing start time</span> is
    not null, then:</p>

    <ol>
      <li><p>Let <var>global</var> be <var>element</var>'s <span>node document</span>'s
      <span>relevant global object</span>.</p></li>

      <li><p>Let <var>fallbackTimingInfo</var> be a new <span
      data-x="fetch-timing-info">fetch timing info</span> whose <span
      data-x="fetch-timing-info-start-time">start time</span> is <var>element</var>'s <span
      data-x="iframe-pending-resource-timing-start-time">pending resource-timing start time</span>
      and whose <span data-x="fetch-timing-info-end-time">response end time</span> is the
      <span>current high resolution time</span> given <var>global</var>.</p></li>

      <li><p><span>Mark resource timing</span> given <var>fallbackTimingInfo</var>, <var>url</var>,
      "<code>iframe</code>", <var>global</var>, the empty string, a new
      <span data-x="response-body-info">response body info</span>, and 0.</p></li>

      <li><p>Set <var>element</var>'s <span
      data-x="iframe-pending-resource-timing-start-time">pending resource-timing start time</span>
      to null.</p></li>
    </ol>
   </li>

   <li><p>Set <var>childDocument</var>'s <span>iframe load in progress</span> flag.</p></li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
   data-x="event-load">load</code> at <var>element</var>.</p></li>

   <li><p>Unset <var>childDocument</var>'s <span>iframe load in progress</span> flag.</p></li>
  </ol>

  <p class="warning">This, in conjunction with scripting, can be used to probe the URL space of the
  local network's HTTP servers. User agents may implement <span data-x="origin">cross-origin</span>
  access control policies that are stricter than those described above to mitigate this attack, but
  unfortunately such policies are typically not compatible with existing web content.</p>

  <p>If an element type <dfn>potentially delays the load event</dfn>, then for each element
  <var>element</var> of that type, the user agent must <span>delay the load event</span> of
  <var>element</var>'s <span>node document</span> if <var>element</var>'s <span>content
  navigable</span> is non-null and any of the following are true:</p>

  <ul>
   <li><p><var>element</var>'s <span>content navigable</span>'s <span data-x="nav-document">active
   document</span> is not <span>ready for post-load tasks</span>;</p></li>

   <li><p><var>element</var>'s <span>content navigable</span>'s <span>is delaying <code
   data-x="event-load">load</code> events</span> is true; or</p></li>

   <li><p>anything is <span data-x="delay the load event">delaying the load event</span> of
   <var>element</var>'s <span>content navigable</span>'s <span data-x="nav-document">active
   document</span>.</p></li>
  </ul>

  <p class="note">If, during the handling of the <code data-x="event-load">load</code> event,
  <var>element</var>'s <span>content navigable</span> is again <span
  data-x="navigate">navigated</span>, that will further <span>delay the load event</span>.</p>

  <p>Each <code>iframe</code> element has an associated <dfn>current navigation was lazy
  loaded</dfn> boolean, initially false. It is set and unset in the <span>process the
  <code>iframe</code> attributes</span> algorithm.</p>

  <p>An <code>iframe</code> element whose <span>current navigation was lazy loaded</span> boolean is
  false <span>potentially delays the load event</span>.</p>

  <p>Each <code>iframe</code> element has an associated null or
  <span><code>DOMHighResTimeStamp</code></span> <dfn
  data-x="iframe-pending-resource-timing-start-time">pending resource-timing start time</dfn>,
  initially set to null.</p>
  </div>

  <!-- END of section that's very similar to <frame> -->

  <p class="note">If, when the element is created, the <code
  data-x="attr-iframe-srcdoc">srcdoc</code> attribute is not set, and the <code
  data-x="attr-iframe-src">src</code> attribute is either also not set or set but its value cannot
  be <span data-x="encoding-parsing a URL">parsed</span>, the element's <span>content
  navigable</span> will remain at the <span data-x="is initial about:blank">initial
  <code>about:blank</code></span> <code>Document</code>.</p>

  <p class="note">If the user <span data-x="navigate">navigates</span> away from this page, the
  <code>iframe</code>'s <span>content navigable</span>'s <span data-x="nav-wp">active
  <code>WindowProxy</code></span> object will proxy new <code>Window</code> objects for new
  <code>Document</code> objects, but the <code data-x="attr-iframe-src">src</code> attribute will
  not change.</p>


  <hr> <!-- NAME -->

  <p>The <dfn element-attr for="iframe"><code data-x="attr-iframe-name">name</code></dfn>
  attribute, if present, must be a <span>valid navigable target name</span>. The given value is
  used to name the element's <span>content navigable</span> if present when that is <span
  data-x="create a new child navigable">created</span>.</p>


  <hr> <!-- SANDBOX -->

  <p>The <dfn element-attr for="iframe"><code data-x="attr-iframe-sandbox">sandbox</code></dfn>
  attribute, when specified, enables a set of extra restrictions on any content hosted by the
  <code>iframe</code>. Its value must be an <span>unordered set of unique space-separated
  tokens</span> that are <span>ASCII case-insensitive</span>. The allowed values are:</p>

  <ul class="brief">
   <li><code data-x="attr-iframe-sandbox-allow-downloads">allow-downloads</code></li>
   <li><code data-x="attr-iframe-sandbox-allow-forms">allow-forms</code></li>
   <li><code data-x="attr-iframe-sandbox-allow-modals">allow-modals</code></li>
   <li><code data-x="attr-iframe-sandbox-allow-orientation-lock">allow-orientation-lock</code></li>
   <li><code data-x="attr-iframe-sandbox-allow-pointer-lock">allow-pointer-lock</code></li>
   <li><code data-x="attr-iframe-sandbox-allow-popups">allow-popups</code></li>
   <li><code
   data-x="attr-iframe-sandbox-allow-popups-to-escape-sandbox">allow-popups-to-escape-sandbox</code></li>
   <li><code data-x="attr-iframe-sandbox-allow-presentation">allow-presentation</code></li>
   <li><code data-x="attr-iframe-sandbox-allow-same-origin">allow-same-origin</code></li>
   <li><code data-x="attr-iframe-sandbox-allow-scripts">allow-scripts</code></li>
   <li><code data-x="attr-iframe-sandbox-allow-top-navigation">allow-top-navigation</code></li>
   <li><code
   data-x="attr-iframe-sandbox-allow-top-navigation-by-user-activation">allow-top-navigation-by-user-activation</code></li>
   <li><code
   data-x="attr-iframe-sandbox-allow-top-navigation-to-custom-protocols">allow-top-navigation-to-custom-protocols</code></li>
  </ul>

  <p>When the attribute is set, the content is treated as being from a unique <span
  data-x="concept-origin-opaque">opaque origin</span>, forms, scripts, and various potentially
  annoying APIs are disabled, and links are prevented from targeting other <span
  data-x="navigable">navigables</span>. The <code
  data-x="attr-iframe-sandbox-allow-same-origin">allow-same-origin</code> keyword causes the
  content to be treated as being from its real origin instead of forcing it into an <span
  data-x="concept-origin-opaque">opaque origin</span>; the <code
  data-x="attr-iframe-sandbox-allow-top-navigation">allow-top-navigation</code> keyword allows the
  content to <span>navigate</span> its <span data-x="nav-traversable">traversable navigable</span>;
  the <code
  data-x="attr-iframe-sandbox-allow-top-navigation-by-user-activation">allow-top-navigation-by-user-activation</code>
  keyword behaves similarly but allows such <span data-x="navigate">navigation</span> only when the
  browsing context's <span data-x="nav-window">active window</span> has <span>transient
  activation</span>; the <code
  data-x="attr-iframe-sandbox-allow-top-navigation-to-custom-protocols">allow-top-navigation-to-custom-protocols</code>
  reenables navigations toward non <span>fetch scheme</span> to be <span data-x="hand-off to
  external software">handed off to external software</span>; and the <code
  data-x="attr-iframe-sandbox-allow-forms">allow-forms</code>, <code
  data-x="attr-iframe-sandbox-allow-modals">allow-modals</code>, <code
  data-x="attr-iframe-sandbox-allow-orientation-lock">allow-orientation-lock</code>, <code
  data-x="attr-iframe-sandbox-allow-pointer-lock">allow-pointer-lock</code>, <code
  data-x="attr-iframe-sandbox-allow-popups">allow-popups</code>, <code
  data-x="attr-iframe-sandbox-allow-presentation">allow-presentation</code>, <code
  data-x="attr-iframe-sandbox-allow-scripts">allow-scripts</code>, and <code
  data-x="attr-iframe-sandbox-allow-popups-to-escape-sandbox">allow-popups-to-escape-sandbox</code>
  keywords re-enable forms, modal dialogs, screen orientation lock, the pointer lock API, popups,
  the presentation API, scripts, and the creation of unsandboxed <span
  data-x="auxiliary browsing context">auxiliary browsing contexts</span> respectively. The <code
  data-x="attr-iframe-sandbox-allow-downloads">allow-downloads</code> keyword allows content to
  perform downloads. <ref>POINTERLOCK</ref> <ref>SCREENORIENTATION</ref> <ref>PRESENTATION</ref></p>

  <p>The <code
  data-x="attr-iframe-sandbox-allow-top-navigation">allow-top-navigation</code> and <code
  data-x="attr-iframe-sandbox-allow-top-navigation-by-user-activation">allow-top-navigation-by-user-activation</code>
  keywords must not both be specified, as doing so is redundant; only <code
  data-x="attr-iframe-sandbox-allow-top-navigation">allow-top-navigation</code> will have an effect
  in such non-conformant markup.</p>

  <p>Similarly, the <code
  data-x="attr-iframe-sandbox-allow-top-navigation-to-custom-protocols">allow-top-navigation-to-custom-protocols</code>
  keyword must not be specified if either <code
  data-x="attr-iframe-sandbox-allow-top-navigation">allow-top-navigation</code> or <code
  data-x="attr-iframe-sandbox-allow-popups">allow-popups</code> are specified, as doing so is
  redundant.</p>

  <p class="note">To allow <code data-x="dom-alert">alert()</code>, <code
  data-x="dom-confirm">confirm()</code>, and <code data-x="dom-prompt">prompt()</code> inside
  sandboxed content, both the <code data-x="attr-iframe-sandbox-allow-modals">allow-modals</code>
  and <code data-x="attr-iframe-sandbox-allow-same-origin">allow-same-origin</code> keywords need to
  be specified, and the loaded URL needs to be <span>same origin</span> with the <span>top-level
  origin</span>. Without the <code
  data-x="attr-iframe-sandbox-allow-same-origin">allow-same-origin</code> keyword, the content is
  always treated as cross-origin, and cross-origin content <span>cannot show simple
  dialogs</span>.</p>

  <p class="warning">Setting both the <code
  data-x="attr-iframe-sandbox-allow-scripts">allow-scripts</code> and <code
  data-x="attr-iframe-sandbox-allow-same-origin">allow-same-origin</code> keywords together when the
  embedded page has the <span>same origin</span> as the page containing the <code>iframe</code>
  allows the embedded page to simply remove the <code data-x="attr-iframe-sandbox">sandbox</code>
  attribute and then reload itself, effectively breaking out of the sandbox altogether.</p>

  <p class="warning">These flags only take effect when the <span>content navigable</span> of the
  <code>iframe</code> element is <span data-x="navigate">navigated</span>. Removing them, or
  removing the entire <code data-x="attr-iframe-sandbox">sandbox</code> attribute, has no effect on
  an already-loaded page.</p>

  <p class="warning">Potentially hostile files should not be served from the same server as the file
  containing the <code>iframe</code> element. Sandboxing hostile content is of minimal help if an
  attacker can convince the user to just visit the hostile content directly, rather than in the
  <code>iframe</code>. To limit the damage that can be caused by hostile HTML content, it should be
  served from a separate dedicated domain. Using a different domain ensures that scripts in the
  files are unable to attack the site, even if the user is tricked into visiting those pages
  directly, without the protection of the <code data-x="attr-iframe-sandbox">sandbox</code>
  attribute.</p>

  <div w-nodev>

  <!-- v2: Add a new attribute that enables new restrictions, e.g.:
       - disallow cross-origin loads of any kind (networking
         override that only allows same-origin URLs or about:,
         javascript:, data:)
       - block access to 'parent.frames' from sandbox
  -->

  <p>When an <code>iframe</code> element's <code data-x="attr-iframe-sandbox">sandbox</code>
  attribute is set or changed while it has a non-null <span>content navigable</span>, the user
  agent must <span data-x="parse a sandboxing directive">parse the sandboxing directive</span>
  given the attribute's value and the <code>iframe</code> element's <span><code>iframe</code>
  sandboxing flag set</span>.</p>

  <p>When an <code>iframe</code> element's <code data-x="attr-iframe-sandbox">sandbox</code>
  attribute is removed while it has a non-null <span>content navigable</span>, the user agent must
  empty the <code>iframe</code> element's <span><code>iframe</code> sandboxing flag set</span>.</p>

  </div>

  <div class="example">

   <p>In this example, some completely-unknown, potentially hostile, user-provided HTML content is
   embedded in a page. Because it is served from a separate domain, it is affected by all the normal
   cross-site restrictions. In addition, the embedded page has scripting disabled, plugins disabled,
   forms disabled, and it cannot navigate any frames or windows other than itself (or any frames or
   windows it itself embeds).</p>

   <pre><code class="html">&lt;p>We're not scared of you! Here is your content, unedited:&lt;/p>
&lt;iframe sandbox src="https://usercontent.example.net/getusercontent.cgi?id=12193">&lt;/iframe></code></pre>

   <p class="warning">It is important to use a separate domain so that if the attacker convinces the
   user to visit that page directly, the page doesn't run in the context of the site's origin, which
   would make the user vulnerable to any attack found in the page.</p>

  </div>

  <div class="example">

   <p>In this example, a gadget from another site is embedded. The gadget has scripting and forms
   enabled, and the origin sandbox restrictions are lifted, allowing the gadget to communicate with
   its originating server. The sandbox is still useful, however, as it disables plugins and popups,
   thus reducing the risk of the user being exposed to malware and other annoyances.</p>

   <pre><code class="html">&lt;iframe sandbox="allow-same-origin allow-forms allow-scripts"
        src="https://maps.example.com/embedded.html">&lt;/iframe></code></pre>

  </div>

  <div class="example">

   <p>Suppose a file A contained the following fragment:</p>

   <pre><code class="html">&lt;iframe sandbox="allow-same-origin allow-forms" src=B>&lt;/iframe></code></pre>

   <p>Suppose that file B contained an iframe also:</p>

   <pre><code class="html">&lt;iframe sandbox="allow-scripts" src=C>&lt;/iframe></code></pre>

   <p>Further, suppose that file C contained a link:</p>

   <pre><code class="html">&lt;a href=D>Link&lt;/a></code></pre>

   <p>For this example, suppose all the files were served as <code>text/html</code>.</p>

   <p>Page C in this scenario has all the sandboxing flags set. Scripts are disabled, because the
   <code>iframe</code> in A has scripts disabled, and this overrides the <code
   data-x="attr-iframe-sandbox-allow-scripts">allow-scripts</code> keyword set on the
   <code>iframe</code> in B. Forms are also disabled, because the inner <code>iframe</code> (in B)
   does not have the <code data-x="attr-iframe-sandbox-allow-forms">allow-forms</code> keyword
   set.</p>

   <p>Suppose now that a script in A removes all the <code
   data-x="attr-iframe-sandbox">sandbox</code> attributes in A <!--grammar-check-override--> and B.
   This would change nothing immediately. If the user clicked the link in C, loading page D into
   the <code>iframe</code> in B, page D would now act as if the <code>iframe</code> in B had the
   <code data-x="attr-iframe-sandbox-allow-same-origin">allow-same-origin</code> and <code
   data-x="attr-iframe-sandbox-allow-forms">allow-forms</code> keywords set, because that was the
   state of the <span>content navigable</span> in the <code>iframe</code> in A when page B was
   loaded.</p>

   <p>Generally speaking, dynamically removing or changing the <code
   data-x="attr-iframe-sandbox">sandbox</code> attribute is ill-advised, because it can make it quite
   hard to reason about what will be allowed and what will not.</p>

  </div>


  <hr> <!-- FEATURE POLICY ATTRIBUTES: ALLOW; ALLOW FULLSCREEN / PAYMENT REQUEST / USER MEDIA -->

  <p>The <dfn element-attr for="iframe"><code data-x="attr-iframe-allow">allow</code></dfn>
  attribute, when specified, determines the <span data-x="concept-container-policy">container
  policy</span> that will be used when the <span
  data-x="concept-document-permissions-policy">permissions policy</span> for a
  <code>Document</code> in the <code>iframe</code>'s <span>content navigable</span> is initialized.
  Its value must be a <span data-x="concept-serialized-permissions-policy">serialized permissions
  policy</span>. <ref>PERMISSIONSPOLICY</ref></p>

  <div class="example">
   <p>In this example, an <code>iframe</code> is used to embed a map from an online navigation
   service. The <code data-x="attr-iframe-allow">allow</code> attribute is used to enable the
   Geolocation API within the nested context.</p>

   <pre><code class="html">&lt;iframe src="https://maps.example.com/" allow="geolocation">&lt;/iframe></code></pre>
  </div>

  <p>The <dfn element-attr for="iframe"><code
  data-x="attr-iframe-allowfullscreen">allowfullscreen</code></dfn> attribute is a <span>boolean
  attribute</span>. When specified, it indicates that <code>Document</code> objects in the
  <code>iframe</code> element's <span>content navigable</span> will be initialized with a <span
  data-x="concept-document-permissions-policy">permissions policy</span> which allows the "<code
  data-x="">fullscreen</code>" feature to be used from any <span>origin</span>. This is enforced by
  the <span data-x="process-permissions-policy-attributes">process permissions policy
  attributes</span> algorithm. <ref>PERMISSIONSPOLICY</ref></p>

  <div class="example">

   <p>Here, an <code>iframe</code> is used to embed a player from a video site. The <code
   data-x="attr-iframe-allowfullscreen">allowfullscreen</code> attribute is needed to enable the
   player to show its video fullscreen.</p>

   <pre><code class="html">&lt;article>
 &lt;header>
  &lt;p>&lt;img src="/usericons/1627591962735"> &lt;b>Fred Flintstone&lt;/b>&lt;/p>
  &lt;p>&lt;a href="/posts/3095182851" rel=bookmark>12:44&lt;/a> &mdash; &lt;a href="#acl-3095182851">Private Post&lt;/a>&lt;/p>
 &lt;/header>
 &lt;p>Check out my new ride!&lt;/p>
 <strong>&lt;iframe src="https://video.example.com/embed?id=92469812" allowfullscreen>&lt;/iframe></strong>
&lt;/article></code></pre>

  </div>

  <p class="note">Neither <code data-x="attr-iframe-allow">allow</code> nor <code
  data-x="attr-iframe-allowfullscreen">allowfullscreen</code> can grant access to a feature in an
  <code>iframe</code> element's <span>content navigable</span> if the element's <span>node
  document</span> is not already allowed to use that feature.</p>

  <div w-nodev>
   <p id="fullscreen-logic">To determine whether a <code>Document</code> object <var>document</var>
   is <dfn export>allowed to use</dfn> the policy-controlled-feature <var>feature</var>, run these
   steps:</p>

   <ol>
    <li><p>If <var>document</var>'s <span data-x="concept-document-bc">browsing context</span> is
    null, then return false.</p></li>

    <li><p>If <var>document</var> is not <span>fully active</span>, then return false.</p></li>

    <li><p>If the result of running <span data-x="is-feature-enabled">is feature enabled in document
    for origin</span> on <var>feature</var>, <var>document</var>, and <var>document</var>'s <span
    data-x="concept-document-origin">origin</span> is "<code data-x="">Enabled</code>", then return
    true.</p></li>

    <li><p>Return false.</p></li>
   </ol>
  </div>

  <p class="warning">Because they only influence the <span
  data-x="concept-document-permissions-policy">permissions policy</span> of the <span>content
  navigable</span>'s <span data-x="nav-document">active document</span>, the <code
  data-x="attr-iframe-allow">allow</code> and <code
  data-x="attr-iframe-allowfullscreen">allowfullscreen</code> attributes only take effect when the
  <span>content navigable</span> of the <code>iframe</code> is <span
  data-x="navigate">navigated</span>. Adding or removing them has no effect on an already-loaded
  document.</p>


  <hr> <!-- DIM ATTRIBUTES -->

  <p>The <code>iframe</code> element supports <span>dimension attributes</span> for cases where the
  embedded content has specific dimensions (e.g. ad units have well-defined dimensions).</p>

  <p>An <code>iframe</code> element never has <span>fallback content</span>, as it will always
  <span>create a new child navigable</span>, regardless of whether the specified initial
  contents are successfully used.</p>

  <hr> <!-- REFERRERPOLICY ATTRIBUTE -->

  <p>The <dfn element-attr for="iframe"
  data-x="attr-iframe-referrerpolicy"><code>referrerpolicy</code></dfn> attribute is a
  <span>referrer policy attribute</span>. Its purpose is to set the <span>referrer policy</span>
  used when <span data-x="process the iframe attributes">processing the <code>iframe</code>
  attributes</span>. <ref>REFERRERPOLICY</ref></p>

  <p>The <dfn element-attr for="iframe" data-x="attr-iframe-loading"><code>loading</code></dfn> attribute is a <span>lazy
  loading attribute</span>. Its purpose is to indicate the policy for loading <code>iframe</code>
  elements that are outside the viewport.</p>

  <p>When the <code data-x="attr-iframe-loading">loading</code> attribute's state is changed to the
  <span data-x="attr-loading-eager-state">Eager</span> state, the user agent must run these
  steps:</p>

  <ol>
   <li><p>Let <var>resumptionSteps</var> be the <code>iframe</code> element's <span>lazy load
   resumption steps</span>.</p></li>

   <li><p>If <var>resumptionSteps</var> is null, then return.</p></li>

   <li><p>Set the <code>iframe</code>'s <span>lazy load resumption steps</span> to null.</p></li>

   <li><p>Invoke <var>resumptionSteps</var>.</p></li>
  </ol>

  <hr> <!-- FALLBACK -->

  <p>Descendants of <code>iframe</code> elements represent nothing. (In legacy user agents that do
  not support <code>iframe</code> elements, the contents would be parsed as markup that could act as
  fallback content.)</p>

  <p class="note">The <span>HTML parser</span> treats markup inside <code>iframe</code> elements as
  text.</p>


  <div w-nodev>

  <hr> <!-- DOM -->

  <p>The IDL attributes <dfn attribute for="HTMLIFrameElement"><code
  data-x="dom-iframe-src">src</code></dfn>, <dfn attribute for="HTMLIFrameElement"><code
  data-x="dom-iframe-name">name</code></dfn>, <dfn attribute for="HTMLIFrameElement"><code
  data-x="dom-iframe-sandbox">sandbox</code></dfn>, and <dfn attribute
  for="HTMLIFrameElement"><code data-x="dom-iframe-allow">allow</code></dfn> must
  <span>reflect</span> the respective content attributes of the same name.</p>

  <p>The <dfn attribute for="HTMLIFrameElement"><code
  data-x="dom-iframe-srcdoc">srcdoc</code></dfn> getter steps are:</p>

  <ol>
   <li><p>Let <var>attribute</var> be the result of running <span
   data-x="concept-element-attributes-get-by-namespace">get an attribute by namespace and local
   name</span> given null, <code data-x="attr-iframe-srcdoc">srcdoc</code>'s <span
   data-x="concept-attribute-local-name">local name</span>, and <span>this</span>.</p></li>

   <li><p>If <var>attribute</var> is null, then return the empty string.</p></li>

   <li><p>Return <var>attribute</var>'s <span
   data-x="concept-attribute-value">value</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-iframe-srcdoc">srcdoc</code> setter steps
  are:</p>

  <ol>
   <li><p>Let <var>compliantString</var> be the result of invoking the <span
   data-x="tt-getcompliantstring">Get Trusted Type compliant string</span> algorithm with <code
   data-x="tt-trustedhtml">TrustedHTML</code>, <span>this</span>'s <span>relevant global
   object</span>, the given value, "<code data-x="">HTMLIFrameElement srcdoc</code>", and "<code
   data-x="">script</code>".</p></li>

   <li><p><span data-x="concept-element-attributes-set-value">Set an attribute value</span> given
   <span>this</span>, <code data-x="attr-iframe-srcdoc">srcdoc</code>'s <span
   data-x="concept-attribute-local-name">local name</span>, and
   <var>compliantString</var>.</p></li>
  </ol>

  <p>The <span data-x="concept-supported-tokens">supported tokens</span> for <code
  data-x="dom-iframe-sandbox">sandbox</code>'s <code>DOMTokenList</code> are the allowed
  values defined in the <code data-x="attr-iframe-sandbox">sandbox</code> attribute and supported by
  the user agent.</p>

  <p>The <dfn attribute for="HTMLIFrameElement"><code
  data-x="dom-iframe-allowFullscreen">allowFullscreen</code></dfn> IDL attribute must
  <span>reflect</span> the <code data-x="attr-iframe-allowfullscreen">allowfullscreen</code> content
  attribute.</p>

  <p>The <dfn attribute for="HTMLIFrameElement"><code
  data-x="dom-iframe-referrerPolicy">referrerPolicy</code></dfn> IDL attribute must
  <span>reflect</span> the <code data-x="attr-iframe-referrerpolicy">referrerpolicy</code> content
  attribute, <span>limited to only known values</span>.</p>

  <p>The <dfn attribute for="HTMLIFrameElement"><code
  data-x="dom-iframe-loading">loading</code></dfn> IDL attribute must <span>reflect</span> the <code
  data-x="attr-iframe-loading">loading</code> content attribute, <span>limited to only known
  values</span>.</p>

  <p>The <dfn attribute for="HTMLIFrameElement"><code
  data-x="dom-iframe-contentDocument">contentDocument</code></dfn> getter steps are to return the
  <span>this</span>'s <span>content document</span>.</p>

  <p>The <dfn attribute for="HTMLIFrameElement"><code
  data-x="dom-iframe-contentWindow">contentWindow</code></dfn> getter steps are to return
  <span>this</span>'s <span>content window</span>.</p>

  </div>

  <div class="example">

   <p>Here is an example of a page using an <code>iframe</code> to include advertising from an
   advertising broker:</p>

   <pre><code class="html">&lt;iframe src="https://ads.example.com/?customerid=923513721&amp;amp;format=banner"
        width="468" height="60">&lt;/iframe></code></pre>

  </div>




  <h4>The <dfn element><code>embed</code></dfn> element</h4>

<!-- (v2?)
 we have all kinds of quirks we should define if they come up during
 testing, as e.g. shown in:
 http://mxr.mozilla.org/mozilla-central/source/layout/generic/nsObjectFrame.cpp
 http://trac.webkit.org/browser/trunk/WebCore/html/HTMLEmbedElement.cpp
 http://trac.webkit.org/browser/trunk/WebCore/rendering/RenderPartObject.cpp (updateWidget)
 e.g. - 240x200 default
      - the attributes/params are sent in a name-value pair list as follows (for Gecko):
         + attributes of the element, in source order
         + a synthesized 'src' attribute, if there was no 'src' but
           there was a 'data', with the value of the 'data' attribute
         + the params, in source order
        (WebKit does something different still)
      - the HIDDEN attribute (might be moot now)
-->

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Embedded content</span>.</dd>
   <dd><span>Interactive content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>embedded content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="concept-content-nothing">Nothing</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-embed-src">src</code></dd>
   <dd><code data-x="attr-embed-type">type</code></dd>
   <dd><code data-x="attr-dim-width">width</code></dd>
   <dd><code data-x="attr-dim-height">height</code></dd>
   <dd>Any other attribute that has no namespace (see prose).</dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-embed">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-embed">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLEmbedElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute USVString <span data-x="dom-embed-src">src</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-embed-type">type</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-dim-width">width</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-dim-height">height</span>;
  <span>Document</span>? <span data-x="dom-media-getSVGDocument">getSVGDocument</span>();

  // <a href="#HTMLEmbedElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
  </dl>

  <p>The <code>embed</code> element provides an integration point for an external application or
  interactive content.</p>

  <p>The <dfn element-attr for="embed"><code data-x="attr-embed-src">src</code></dfn> attribute
  gives the <span>URL</span> of the resource being embedded. The attribute, if present, must contain
  a <span>valid non-empty URL potentially surrounded by spaces</span>.</p>

  <p>If the <code data-x="attr-itemprop">itemprop</code> attribute is specified on an
  <code>embed</code> element, then the <code data-x="attr-embed-src">src</code> attribute must also
  be specified.</p>

  <p>The <dfn element-attr for="embed"><code data-x="attr-embed-type">type</code></dfn> attribute,
  if present, gives the <span>MIME type</span> by which the plugin to instantiate is selected. The
  value must be a <span>valid MIME type string</span>. If both the <code
  data-x="attr-embed-type">type</code> attribute and the <code data-x="attr-embed-src">src</code>
  attribute are present, then the <code data-x="attr-embed-type">type</code> attribute must specify
  the same type as the <span data-x="Content-Type">explicit Content-Type metadata</span> of the
  resource given by the <code data-x="attr-embed-src">src</code> attribute.</p>

  <div w-nodev>

  <p>While any of the following conditions are occurring, any <span>plugin</span> instantiated for
  the element must be removed, and the <code>embed</code> element <span>represents</span>
  nothing:</p>

  <ul>
   <li><p>The element has neither a <code data-x="attr-embed-src">src</code> attribute nor a <code
   data-x="attr-embed-type">type</code> attribute.</p></li>

   <li><p>The element has a <span>media element</span> ancestor.</p></li>

   <li><p>The element has an ancestor <code>object</code> element that is <em>not</em> showing its
   <span>fallback content</span>.</p></li>
  </ul>

  <p>An <code>embed</code> element is said to be <dfn data-x="concept-embed-active">potentially
  active</dfn> when the following conditions are all met simultaneously:</p>

  <ul>
   <li><p>The element is <span>in a document</span> or was <span>in a document</span> the last time
   the <span>event loop</span> reached <a href="#step1">step 1</a>.</p></li>

   <li><p>The element's <span>node document</span> is <span>fully active</span>.</p></li>

   <li><p>The element has either a <code data-x="attr-embed-src">src</code> attribute set or a
   <code data-x="attr-embed-type">type</code> attribute set (or both).</p></li>

   <li><p>The element's <code data-x="attr-embed-src">src</code> attribute is either absent or its
   value is not the empty string.</p></li>

   <li><p>The element is not a descendant of a <span>media element</span>.</p></li>

   <li><p>The element is not a descendant of an <code>object</code> element that is not showing its
   <span>fallback content</span>.</p></li>

   <li><p>The element is <span>being rendered</span>, or was <span>being rendered</span> the last
   time the <span>event loop</span> reached <a href="#step1">step 1</a>.</p></li>
  </ul>

  <p>Whenever an <code>embed</code> element that was not <span
  data-x="concept-embed-active">potentially active</span> becomes <span
  data-x="concept-embed-active">potentially active</span>, and whenever a <span
  data-x="concept-embed-active">potentially active</span> <code>embed</code> element that is
  remaining <span data-x="concept-embed-active">potentially active</span> and has its <code
  data-x="attr-embed-type">src</code> attribute set, changed, or removed or its <code
  data-x="attr-embed-type">type</code> attribute set, changed, or removed, the user agent must
  <span>queue an element task</span> on the <dfn>embed task source</dfn> given the element
  to run <span>the <code>embed</code> element setup steps</span> for that element.</p>

  <p><dfn>The <code>embed</code> element setup steps</dfn> for a given <code>embed</code> element
  <var>element</var> are as follows:</p>

  <ol>
   <li><p>If another <span data-x="concept-task">task</span> has since been queued to run <span>the
   <code>embed</code> element setup steps</span> for <var>element</var>, then return.</p></li>

   <li>
    <p>If <var>element</var> has a <code data-x="attr-embed-src">src</code> attribute set, then:</p>

    <ol>
     <li><p>Let <var>url</var> be the result of <span>encoding-parsing a URL</span> given
     <var>element</var>'s <code data-x="attr-embed-src">src</code> attribute's value, relative to
     <var>element</var>'s <span>node document</span>.</p></li>

     <li><p>If <var>url</var> is failure, then return.</p></li>

     <li><p>Let <var>request</var> be a new <span data-x="concept-request">request</span> whose
     <span data-x="concept-request-url">URL</span> is <var>url</var>, <span
     data-x="concept-request-client">client</span> is <var>element</var>'s <span>node
     document</span>'s <span>relevant settings object</span>, <span
     data-x="concept-request-destination">destination</span> is "<code data-x="">embed</code>",
     <span data-x="concept-request-credentials-mode">credentials mode</span> is "<code
     data-x="">include</code>", <span data-x="concept-request-mode">mode</span> is "<code
     data-x="">navigate</code>", <span data-x="concept-request-initiator-type">initiator
     type</span> is "<code data-x="">embed</code>", and whose <span>use-URL-credentials flag</span>
     is set.</p></li>

     <li>
      <p><span data-x="concept-fetch">Fetch</span> <var>request</var>, with <i
      data-x="processResponse">processResponse</i> set to the following steps given <span
      data-x="concept-response">response</span> <var>response</var>:</p>

      <ol>
       <li><p>If another <span data-x="concept-task">task</span> has since been queued to run
       <span>the <code>embed</code> element setup steps</span> for <var>element</var>, then
       return.</p></li>

       <li><p>If <var>response</var> is a <span>network error</span>, then <span
       data-x="concept-event-fire">fire an event</span> named <code
       data-x="event-load">load</code> at <var>element</var>, and return.</p></li>

       <li><p>Let <var>type</var> be the result of determining the <span
       data-x="concept-embed-type">type of content</span> given <var>element</var> and
       <var>response</var>.</p></li>

       <li>
        <p>Switch on <var>type</var>:</p>

        <dl class="switch">
         <dt>null</dt>
         <dd>
          <ol>
           <li><p><span>Display no plugin</span> for <var>element</var>.</p></li>
          </ol>
         </dd>

         <dt>Otherwise</dt>
         <dd>
          <ol>
           <li><p>If <var>element</var>'s <span>content navigable</span> is null, then
           <span>create a new child navigable</span> for <var>element</var>.</p></li>
           <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2291 - dynamic changes
                to 'name' don't do anything -->

           <li>
            <p><span>Navigate</span><!-- DONAV embed--> <var>element</var>'s <span>content
            navigable</span> to <var>response</var>'s <span
            data-x="concept-response-url">URL</span> using <var>element</var>'s <span>node
            document</span>, with <i data-x="navigation-response">response</i> set to
            <var>response</var>, and <i data-x="navigation-hh">historyHandling</i> set to "<code
            data-x="NavigationHistoryBehavior-replace">replace</code>".</p>

            <p class="note"><var>element</var>'s <code data-x="attr-embed-src">src</code> attribute
            does not get updated if the <span>content navigable</span> gets further navigated to
            other locations.</p>
           </li>

           <li><p><var>element</var> now <span>represents</span> its <span>content
           navigable</span>.</p></li>
          </ol>
         </dd>
        </dl>
       </li>
      </ol>

      <p>Fetching the resource must <span>delay the load event</span> of <var>element</var>'s
      <span>node document</span>.</p>
     </li>
    </ol>
   </li>

   <li><p>Otherwise, <span>display no plugin</span> for <var>element</var>.</p></li>
  </ol>

  <p>To determine the <dfn data-x="concept-embed-type">type of the content</dfn> given an
  <code>embed</code> element <var>element</var> and a <span
  data-x="concept-response">response</span> <var>response</var>, run the following steps:</p>

  <ol>
   <li><p>If <var>element</var> has a <code data-x="attr-embed-type">type</code> attribute, and that
   attribute's value is a type that a <span>plugin</span> supports, then return the value of the
   <code data-x="attr-embed-type">type</code> attribute.</p></li>

   <li>
    <p>If the <span data-x="concept-url-path">path</span> component of <var>response</var>'s
    <span data-x="concept-response-url">url</span> matches a pattern that a <span>plugin</span>
    supports, then return the type that that plugin can handle.</p>

    <p class="example">For example, a plugin might say that it can handle URLs with <span
    data-x="concept-url-path">path</span> components that end with the four character string
    "<code data-x="">.swf</code>".</p>

    <!-- it's sad that we have to do extension sniffing. sigh. -->
    <!-- see also <object> which has a similar step -->
   </li>

   <li><p>If <var>response</var> has <span data-x="Content-Type">explicit Content-Type
   metadata</span>, and that value is a type that a <span>plugin</span> supports, then return that
   value.</p></li>

   <li><p>Return null.</p></li>
  </ol>
  <!-- This algorithm is a monument to bad design. Go legacy! -->

  <p class="note">It is intentional that the above algorithm allows <var>response</var> to have a
  non-<span>ok status</span>. This allows servers to return data for plugins even with error
  responses (e.g., HTTP 500 Internal Server Error codes can still contain plugin data).</p>

  <p>To <dfn>display no plugin</dfn> for an <code>embed</code> element <var>element</var>:</p>

  <ol>
   <li><p><span>Destroy a child navigable</span> given <var>element</var>.</p></li>

   <li><p>Display an indication that no <span>plugin</span> could be found for <var>element</var>,
   as the contents of <var>element</var>.</p></li>

   <li><p><var>element</var> now <span>represents</span> nothing.</p></li>
  </ol>

  <p class="note">The <code>embed</code> element has no <span>fallback content</span>; its
  descendants are ignored.</p>

  <p>Whenever an <code>embed</code> element that was <span data-x="concept-embed-active">potentially
  active</span> stops being <span data-x="concept-embed-active">potentially active</span>, any
  <span>plugin</span> that had been instantiated for that element must be unloaded.</p>

  <p>The <code>embed</code> element <span>potentially delays the load event</span>.</p>

  </div>

  <p>The <code>embed</code> element supports <span>dimension attributes</span>.</p>

  <div w-nodev>

  <p>The IDL attributes <dfn attribute for="HTMLEmbedElement"><code
  data-x="dom-embed-src">src</code></dfn> and <dfn attribute for="HTMLEmbedElement"><code
  data-x="dom-embed-type">type</code></dfn> each must <span>reflect</span> the respective content
  attributes of the same name.</p>

  </div>




  <h4>The <dfn element><code>object</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Embedded content</span>.</dd>
   <!-- Interactive content when showing a plugin or a child navigable, but checking that statically is hard... -->
   <dd><span data-x="category-listed">Listed</span> <span>form-associated element</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>embedded content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Transparent</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-object-data">data</code></dd>
   <dd><code data-x="attr-object-type">type</code></dd>
   <dd><code data-x="attr-object-name">name</code></dd>
   <dd><code data-x="attr-fae-form">form</code></dd>
   <dd><code data-x="attr-dim-width">width</code></dd>
   <dd><code data-x="attr-dim-height">height</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-object">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-object">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLObjectElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute USVString <span data-x="dom-object-data">data</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-object-type">type</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-object-name">name</span>;
  readonly attribute <span>HTMLFormElement</span>? <span data-x="dom-fae-form">form</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-dim-width">width</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-dim-height">height</span>;
  readonly attribute <span>Document</span>? <span data-x="dom-object-contentDocument">contentDocument</span>;
  readonly attribute <span>WindowProxy</span>? <span data-x="dom-object-contentWindow">contentWindow</span>;
  <span>Document</span>? <span data-x="dom-media-getSVGDocument">getSVGDocument</span>();

  readonly attribute boolean <span data-x="dom-cva-willValidate">willValidate</span>;
  readonly attribute <span>ValidityState</span> <span data-x="dom-cva-validity">validity</span>;
  readonly attribute DOMString <span data-x="dom-cva-validationMessage">validationMessage</span>;
  boolean <span data-x="dom-cva-checkValidity">checkValidity</span>();
  boolean <span data-x="dom-cva-reportValidity">reportValidity</span>();
  undefined <span data-x="dom-cva-setCustomValidity">setCustomValidity</span>(DOMString error);

  // <a href="#HTMLObjectElement-partial">also has obsolete members</a>
};</code></pre>
    <div w-nodev>
    <p>Depending on the type of content instantiated by the
    <code>object</code> element, the node also supports other
    interfaces.</p>
    </div>
   </dd>
   <dd w-dev>Uses <code>HTMLObjectElement</code>.</dd>
  </dl>

  <p>The <code>object</code> element can represent an external resource, which, depending on the
  type of the resource, will either be treated as an image or as a <span>child
  navigable</span>.</p>

  <p>The <dfn element-attr for="object"><code data-x="attr-object-data">data</code></dfn> attribute
  specifies the <span>URL</span> of the resource. It must be present, and must contain a
  <span>valid non-empty URL potentially surrounded by spaces</span>.</p>

  <p>The <dfn element-attr for="object"><code data-x="attr-object-type">type</code></dfn> attribute,
  if present, specifies the type of the resource. If present, the attribute must be a <span>valid
  MIME type string</span>.</p>

  <p>The <dfn element-attr for="object"><code data-x="attr-object-name">name</code></dfn>
  attribute, if present, must be a <span>valid navigable target name</span>. The given value is
  used to name the element's <span>content navigable</span>, if applicable, and if present when the
  element's <span>content navigable</span> is <span data-x="create a new child
  navigable">created</span>.</p>

  <div w-nodev>

  <p>Whenever one of the following conditions occur:</p>

  <ul>
   <li>the element is created,

   <li>the element is popped off the <span>stack of open elements</span> of an <span>HTML
   parser</span> or <span>XML parser</span>,

   <li>the element is not on the <span>stack of open elements</span> of an <span>HTML parser</span>
   or <span>XML parser</span>, and it is either <span data-x="node is inserted into a
   document">inserted into a document</span> or <span data-x="node is removed from a
   document">removed from a document</span>,

   <li>the element's <span>node document</span> changes whether it is <span>fully active</span>,

   <li>one of the element's ancestor <code>object</code> elements changes to or from showing its
   <span>fallback content</span>,

   <li>the element's <code data-x="attr-object-classid">classid</code> attribute is set, changed, or
   removed,

   <li>the element's <code data-x="attr-object-classid">classid</code> attribute is not present, and
   its <code data-x="attr-object-data">data</code> attribute is set, changed, or removed,

   <li>neither the element's <code data-x="attr-object-classid">classid</code> attribute nor its
   <code data-x="attr-object-data">data</code> attribute are present, and its <code
   data-x="attr-object-type">type</code> attribute is set, changed, or removed,

   <li>the element changes from <span>being rendered</span> to not being rendered, or vice versa,
  </ul> <!-- Changing the base URL doesn't trigger this. -->

  <p>...the user agent must <span>queue an element task</span> on the <span>DOM manipulation task
  source</span> given the <code>object</code> element to run the following steps to (re)determine
  what the <code>object</code> element represents. This <span data-x="concept-task">task</span>
  being <span data-x="queue a task">queued</span> or actively running must <span>delay the load
  event</span> of the element's <span>node document</span>. <!--As described in the algorithm, once
  the algorithm starts fetching a resource, the fetch is what starts delaying the load event. But to
  tide us over from when the parser finds the <object> element and the fetching begins, we have to
  block the load event like this, lest the parse end before this task gets run.--></p>

  <ol>
   <li>
    <p>If the user has indicated a preference that this <code>object</code> element's <span>fallback
    content</span> be shown instead of the element's usual behavior, then jump to the step below
    labeled <i>fallback</i>.</p>

    <p class="note">For example, a user could ask for the element's <span>fallback content</span> to
    be shown because that content uses a format that the user finds more accessible.</p>
   </li>

   <li><p>If the element has an ancestor <span>media element</span>, or has an ancestor
   <code>object</code> element that is <em>not</em> showing its <span>fallback content</span>, or if
   the element is not <span>in a document</span> whose <span data-x="concept-document-bc">browsing
   context</span> is non-null, or if the element's <span>node document</span> is not <span>fully
   active</span>, or if the element is still in the <span>stack of open elements</span> of an
   <span>HTML parser</span> or <span>XML parser</span>, or if the element is not <span>being
   rendered</span>, then jump to the step below labeled <i>fallback</i>.</p></li>

   <!-- (v2?)
   we may have to define magic fallback to <param> if it turns out to
   be needed in testing:
   <hyatt> apparently your url can come from <param>
   <hyatt> not just the data attribute
   <hyatt> our code looks for params with "src", "movie", "code" and "url"
   <hyatt> and also tries to find the type on a param
   <Hixie> oh that's you trying to have hacky activex support
   <Hixie> opera does that too
   <hyatt> yeah we support activex versions of plugins that are common
   <hyatt> like flash and quicktime and realaudio
   <Hixie> that would be a step 1b. if no data attribute, then look for a <param> to get you a URL instead.
   <Hixie> and if you find one, carry on as if that was your data="".
   -->

   <li><p>If the <code data-x="attr-object-data">data</code> attribute is present and its value is
   not the empty string, then:</p>

    <ol>
     <li><p>If the <code data-x="attr-object-type">type</code> attribute is present and its value is
     not a type that the user agent supports, then the user agent may jump to the step below labeled
     <i>fallback</i> without fetching the content to examine its real type.</p></li>

     <li><p>Let <var>url</var> be the result of <span>encoding-parsing a URL</span> given the <code
     data-x="attr-object-data">data</code> attribute's value, relative to the element's <span>node
     document</span>.</p></li>

     <li><p>If <var>url</var> is failure, then <span data-x="concept-event-fire">fire an
     event</span> named <code data-x="event-error">error</code> at the element and jump to the step
     below labeled <i>fallback</i>.</p></li>

     <!-- identical for <embed>/<object> -->
     <li><p>Let <var>request</var> be a new <span data-x="concept-request">request</span> whose
     <span data-x="concept-request-url">URL</span> is <var>url</var>, <span
     data-x="concept-request-client">client</span> is the element's <span>node document</span>'s
     <span>relevant settings object</span>, <span
     data-x="concept-request-destination">destination</span> is "<code data-x="">object</code>",
     <span data-x="concept-request-credentials-mode">credentials mode</span> is "<code
     data-x="">include</code>", <span data-x="concept-request-mode">mode</span> is "<code
     data-x="">navigate</code>", <span data-x="concept-request-initiator-type">initiator
     type</span> is "<code data-x="">object</code>", and whose <span>use-URL-credentials
     flag</span> is set.</p></li>

     <li>
      <p><!--FETCH--><span data-x="concept-fetch">Fetch</span> <var>request</var>.</p>

      <!-- similar text in various places -->
      <p>Fetching the resource must <span>delay the load event</span> of the element's <span>node
      document</span> until the <span data-x="concept-task">task</span> that is <span data-x="queue
      a task">queued</span> by the <span>networking task source</span> once the resource has been
      fetched (defined next) has been run.</p>
     </li>

     <li><p>If the resource is not yet available (e.g. because the resource was not available in the
     cache, so that loading the resource required making a request over the network), then jump to
     the step below labeled <i>fallback</i>. The <span data-x="concept-task">task</span> that is
     <span data-x="queue a task">queued</span> by the <span>networking task source</span> once the
     resource is available must restart this algorithm from this step. Resources can load
     incrementally; user agents may opt to consider a resource "available" whenever enough data has
     been obtained to begin processing the resource.</p></li>

     <li><p>If the load failed (e.g. there was an HTTP 404 error, there was a DNS error), <span
     data-x="concept-event-fire">fire an event</span> named <code data-x="event-error">error</code>
     at the element, then jump to the step below labeled <i>fallback</i>.</p></li>

     <li id="object-type-detection">

      <p>Determine the <var>resource type</var>, as follows:</p>

<!-- Hopefully this step is exactly equivalent to the following:

 START
   |
   V
 Is there a Content-Type and is the UA going to obey it blindly?
   |                   |
   | YES               | NO
   |                   V                                                          YES
   |                 Is there a type="" attribute whose value is a plugin type? ============================================-.
   |                   |                                                                                                     |
   |                   | NO                                                                                                  |
   |                   V                        NO                                       YES                                 |
   |                 Is there a Content type? ========-> Is there a type="" attribute? ==========> Let TYPE be type=""       |
   |                   |                                                           |      attribute value           |
   |                   | YES                                                       | NO                    |                 |
   V            NO     V                                                           |                       V                 |
   +-<============== Is it text/plain or application/octet-stream?                 `==> Let TYPE be  =====>+                 |
   |                   |                                     |                          Sniffed type       |                 |
   |                   | text/plain                          | octet-stream                                V                 |
   |                   V                              YES    V                                          Is TYPE              |
   |                 Does the page sniff as binary? ======> Is there a type="" attribute?       application/octet-stream?    |
   |                   |                                     |              |                      |                |        |
   |                   | NO                                  | YES          | NO                   | YES            | NO     |
   |                   |                                     |       YES    V                      V                |        |
   |                   |                  application/octet-stream? =====> Extension that is plugin type?           |        |
   |                   |                                     |                          |          |                |        |
   |                   |                                     | NO                       | NO       | YES            |        |
   |                   |                                     V                          |          |                |        |
   |                   |                           Type attribute is XML or     YES     V          |                |        |
   |                   |                           doesn't start with image/* ======> FALLBACK     |                |        |
   |                   |                           and is not a plugin type?                       |                |        |
   |                   |                                          |                                |                |        |
   |                   |                                          | NO                             |                |        V
   V                   V                                          V                                V                V       Use
  Use                 Use                                        Use it (will be                  Use              Use      type=""
  Content-Type        text/plain                                 bitmap or plugin)                extension        TYPE     attribute
   |                   |                                          |                                |                |        |
   |                   V                                          V                                V                V        |
   `================->-+========================================>-+==============>-+-<============-+-<==============+-<======'
                                                                                   |
                                                                                   V
                                                                  Continue following rules in the spec, which might
                                                                  result in a plugin, a browsing context, an image,
                                                                  or using fallback, depending on the UA and the type.


   "Extension that is plugin type?" means "Is there an extension that matches one that a plugin supports?".
   Plugins are not allowed to register text/plain or application/octet-stream.

-->

      <ol>
       <li>
        <p>Let the <var>resource type</var> be unknown.</p>
       </li>

       <li>
        <!-- by request: https://www.w3.org/Bugs/Public/show_bug.cgi?id=8479 -->

        <p>If the user agent is configured to strictly obey Content-Type headers for this resource,
        and the resource has <span data-x="Content-Type">associated Content-Type metadata</span>,
        then let the <var>resource type</var> be the type specified in <span
        data-x="Content-Type">the resource's Content-Type metadata</span>, and jump to the step below
        labeled <i>handler</i>.</p>

        <p class="warning">This can introduce a vulnerability, wherein a site is trying to embed a
        resource that uses a particular type, but the remote site overrides that and instead
        furnishes the user agent with a resource that triggers a different type of content with different
        security characteristics. <!-- e.g. the example given above, where the site is expecting
        Flash with allowScriptAccess=never, and instead gets back Java with its unrestricted DOM
        access --></p>
       </li>

       <li>
        <p>Run the appropriate set of steps from the following
        list:</p>

        <dl class="switch">

         <dt>If the resource has <span data-x="Content-Type">associated Content-Type
         metadata</span></dt>

         <dd>
          <ol>
           <li>
            <p>Let <var>binary</var> be false.</p>
           </li>

           <li>
            <p>If the type specified in <span data-x="Content-Type">the resource's Content-Type
            metadata</span> is "<code>text/plain</code>", and the result of applying the <span
            data-x="Content-Type sniffing: text or binary">rules for distinguishing if a resource is
            text or binary</span> to the resource is that the resource is not
            <code>text/plain</code>, then set <var>binary</var> to true.</p>
           </li>

           <li>
            <p>If the type specified in <span data-x="Content-Type">the resource's Content-Type
            metadata</span> is "<code>application/octet-stream</code>", then set <var>binary</var> to true.</p>
           </li>

           <li>
            <p>If <var>binary</var> is false, then let the <var>resource
            type</var> be the type specified in <span data-x="Content-Type">the resource's
            Content-Type metadata</span>, and jump to the step below labeled <i>handler</i>.</p>
           </li>

           <li>
            <p>If there is a <code data-x="attr-object-type">type</code> attribute present on the
            <code>object</code> element, and its value is not <code>application/octet-stream</code>,
            then run the following steps:</p>

            <ol>
             <li>
              <p>If the attribute's value is a type that starts with "<code data-x="">image/</code>" that is
              not also an <span>XML MIME type</span>, then let the <var>resource type</var> be the
              type specified in that <code data-x="attr-object-type">type</code> attribute.</p>
             </li>

             <li>
              <p>Jump to the step below labeled <i>handler</i>.</p>
             </li>
            </ol>
           </li>
          </ol>
         </dd>

         <dt>Otherwise, if the resource does not have <span data-x="Content-Type">associated
         Content-Type metadata</span></dt>

         <dd>
          <ol>
           <li>
            <p>If there is a <code data-x="attr-object-type">type</code> attribute present on the
            <code>object</code> element, then let the <var>tentative type</var> be the type
            specified in that <code data-x="attr-object-type">type</code> attribute.</p>

            <p>Otherwise, let <var>tentative type</var> be the <span data-x="content-type
            sniffing">computed type of the resource</span>.</p>
           </li>

           <li>
            <p>If <var>tentative type</var> is <em>not</em>
            <code>application/octet-stream</code>, then let <var>resource type</var> be
            <var>tentative type</var> and jump to the step below labeled
            <i>handler</i>.</p>
           </li>
          </ol>
         </dd>
        </dl>
       </li>

       <li>
        <!-- if we get to this point we know we can successfully parsed the URL, since this
        algorithm is only used after fetching the resource in the steps above -->

        <p>If applying the <span>URL parser</span> algorithm to the <span>URL</span> of the
        specified resource (after any redirects) results in a <span>URL record</span> whose <span
        data-x="concept-url-path">path</span> component matches a pattern that a <span>plugin</span>
        supports, then let <var>resource type</var> be the type that that plugin can
        handle.</p>

        <p class="example">For example, a plugin might say that it can handle resources with <span
        data-x="concept-url-path">path</span> components that end with the four character string
        "<code data-x="">.swf</code>".</p>

        <!-- it's sad that we have to do extension sniffing. sigh. -->
        <!-- see also <embed> which has a similar step -->
       </li>
      </ol>

      <p class="note">It is possible for this step to finish, or for one of the substeps above to
      jump straight to the next step, with <var>resource type</var> still being unknown. In
      both cases, the next step will trigger fallback.</p>
     </li>

     <li><p><i>Handler</i>: Handle the content as given by the first of the following cases that
     matches:</p>

      <dl class="switch">
       <dt>If the <var>resource type</var> is an <span>XML MIME type</span>, or <!-- (redundant with
       the next one) if the <var>resource type</var> is HTML, or --> if the <var>resource type</var>
       does not start with "<code data-x="">image/</code>"</dt>

       <dd>
        <p>If the <code>object</code> element's <span>content navigable</span> is null, then
        <span>create a new child navigable</span> for the element.</p>

        <p>Let <var>response</var> be the <span data-x="concept-response">response</span> from
        <span data-x="concept-fetch">fetch</span>.</p>

        <p>If <var>response</var>'s <span data-x="concept-response-url">URL</span> does not <span
        data-x="matches about:blank">match <code>about:blank</code></span>, then
        <span>navigate</span><!-- DONAV object --> the element's <span>content navigable</span> to
        <var>response</var>'s <span data-x="concept-response-url">URL</span> using the element's
        <span>node document</span>, with <var data-x="navigation-hh">historyHandling</var> set to
        "<code data-x="NavigationHistoryBehavior-replace">replace</code>".</p>

        <p class="note">The <code data-x="attr-object-data">data</code> attribute of the
        <code>object</code> element doesn't get updated if the <span>content navigable</span> gets
        further <span data-x="navigate">navigated</span> to other locations.</p>

        <p>The <code>object</code> element <span>represents</span> its <span>content
        navigable</span>.</p>

        <!-- note that malformed XML files don't cause fallback -->
       </dd>


       <dt>If the <var>resource type</var> starts with "<code data-x="">image/</code>", and support
       for images has not been disabled</dt>

       <dd>
        <p><span>Destroy a child navigable</span> given the <code>object</code> element.</p>

        <p>Apply the <span data-x="content-type sniffing: image">image sniffing</span> rules to
        determine the type of the image.</p>

        <p>The <code>object</code> element <span>represents</span> the specified image.</p>

        <p>If the image cannot be rendered, e.g. because it is malformed or in an unsupported
        format, jump to the step below labeled <i>fallback</i>.</p>
       </dd>


       <dt>Otherwise</dt>

       <dd>
        <p>The given <var>resource type</var> is not supported. Jump to the step below
        labeled <i>fallback</i>.</p>

        <p class="note">If the previous step ended with the <var>resource type</var> being
        unknown, this is the case that is triggered.</p>
       </dd>
      </dl>
     </li>

     <li><p>The element's contents are not part of what the <code>object</code> element
     represents.</p>

     <li>
      <p>If the <code>object</code> element does not represent its <span>content navigable</span>,
      then once the resource is completely loaded, <span>queue an element task</span> on the
      <span>DOM manipulation task source</span> given the <code>object</code> element to <span
      data-x="concept-event-fire">fire an event</span> named <code data-x="event-load">load</code>
      at the element.</p>

      <p class="note">If the element <em>does</em> represent its <span>content navigable</span>,
      then an analogous task will be queued when the created <code>Document</code> is <span
      data-x="completely finish loading">completely finished loading</span>.</p>
     </li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li><p><i>Fallback</i>: The <code>object</code> element <span>represents</span> the element's
   children. This is the element's <span>fallback content</span>. <span>Destroy a child
   navigable</span> given the element.</p></li>
  </ol>

  <p>Due to the algorithm above, the contents of <code>object</code> elements act as <span>fallback
  content</span>, used only when referenced resources can't be shown (e.g. because it returned a 404
  error). This allows multiple <code>object</code> elements to be nested inside each other,
  targeting multiple user agents with different capabilities, with the user agent picking the first
  one it supports.</p>

  <p>The <code>object</code> element <span>potentially delays the load event</span>.</p>

  </div>

  <p>The <code data-x="attr-fae-form">form</code> attribute is used to explicitly associate the
  <code>object</code> element with its <span>form owner</span>.</p>

  <p>The <code>object</code> element supports <span>dimension attributes</span>.</p>

  <div w-nodev>

  <p>The IDL attributes <dfn attribute for="HTMLObjectElement"><code
  data-x="dom-object-data">data</code></dfn>, <dfn attribute for="HTMLObjectElement"><code
  data-x="dom-object-type">type</code></dfn>, and <dfn attribute for="HTMLObjectElement"><code
  data-x="dom-object-name">name</code></dfn> each must <span>reflect</span> the respective content
  attributes of the same name.</p>

  <p>The <dfn attribute for="HTMLObjectElement"><code
  data-x="dom-object-contentDocument">contentDocument</code></dfn> getter steps are to return
  <span>this</span>'s <span>content document</span>.</p>

  <p>The <dfn attribute for="HTMLObjectElement"><code
  data-x="dom-object-contentWindow">contentWindow</code></dfn> getter steps are to return
  <span>this</span>'s <span>content window</span>.</p>

  <p>The <code data-x="dom-cva-willValidate">willValidate</code>, <code
  data-x="dom-cva-validity">validity</code>, and <code
  data-x="dom-cva-validationMessage">validationMessage</code> attributes, and the <code
  data-x="dom-cva-checkValidity">checkValidity()</code>, <code
  data-x="dom-cva-reportValidity">reportValidity()</code>, and <code
  data-x="dom-cva-setCustomValidity">setCustomValidity()</code> methods, are part of the
  <span>constraint validation API</span>. The <code data-x="dom-fae-form">form</code> IDL attribute
  is part of the element's forms API.</p>

  </div>

  <div class="example">

   <p>In this example, an HTML page is embedded in another using the <code>object</code>
   element.</p>

   <pre><code class="html">&lt;figure>
 &lt;object data="clock.html">&lt;/object>
 &lt;figcaption>My HTML Clock&lt;/figcaption>
&lt;/figure></code></pre>

  </div>


  <h4 split-filename="media" id="the-video-element">The <dfn element id="video"><code>video</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Embedded content</span>.</dd>
   <dd>If the element has a <code data-x="attr-media-controls">controls</code> attribute: <span>Interactive content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>embedded content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>If the element has a <code data-x="attr-media-src">src</code> attribute:
 zero or more <code>track</code> elements, then
 <span>transparent</span>, but with no <span>media element</span> descendants.</dd>
   <dd>If the element does not have a <code data-x="attr-media-src">src</code> attribute: zero or more <code>source</code> elements, then
 zero or more <code>track</code> elements, then
 <span>transparent</span>, but with no <span>media element</span> descendants.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-media-src">src</code></dd>
   <dd><code data-x="attr-media-crossorigin">crossorigin</code></dd>
   <dd><code data-x="attr-video-poster">poster</code></dd>
   <dd><code data-x="attr-media-preload">preload</code></dd>
   <dd><code data-x="attr-media-autoplay">autoplay</code></dd>
   <dd><code data-x="attr-video-playsinline">playsinline</code></dd>
   <dd><code data-x="attr-media-loop">loop</code></dd>
   <dd><code data-x="attr-media-muted">muted</code></dd>
   <dd><code data-x="attr-media-controls">controls</code></dd>
   <dd><code data-x="attr-dim-width">width</code></dd>
   <dd><code data-x="attr-dim-height">height</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-video">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-video">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLVideoElement</dfn> : <span>HTMLMediaElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-dim-width">width</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-dim-height">height</span>;
  readonly attribute unsigned long <span data-x="dom-video-videoWidth">videoWidth</span>;
  readonly attribute unsigned long <span data-x="dom-video-videoHeight">videoHeight</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-video-poster">poster</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-video-playsInline">playsInline</span>;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLVideoElement</code>.</dd>
  </dl>

  <p>A <code>video</code> element is used for playing videos or movies, and audio files with
  captions.</p>

  <p>Content may be provided inside the <code>video</code> element<span w-nodev>. User agents
  should not show this content to the user</span>; it is intended for older web browsers which do
  not support <code>video</code>, so that text can be shown to the users of these older browsers
  informing them of how to access the video contents.</p>

  <p class="note">In particular, this content is not intended to address accessibility concerns. To
  make video content accessible to the partially sighted, the blind, the hard-of-hearing, the deaf,
  and those with other physical or cognitive disabilities, a variety of features are available.
  Captions can be provided, either embedded in the video stream or as external files using the
  <code>track</code> element. Sign-language tracks can be embedded in the video stream. Audio
  descriptions can be embedded in the video stream or in text form using a <span>WebVTT file</span>
  referenced using the <code>track</code> element and synthesized into speech by the user agent.
  WebVTT can also be used to provide chapter titles. For users who would rather not use a media
  element at all, transcripts or other textual alternatives can be provided by simply linking to
  them in the prose near the <code>video</code> element. <ref>WEBVTT</ref></p>

  <p>The <code>video</code> element is a <span>media element</span> whose <span>media data</span> is
  ostensibly video data, possibly with associated audio data.</p>

  <p>The <code data-x="attr-media-src">src</code>, <code data-x="attr-media-crossorigin">crossorigin</code>,
  <code data-x="attr-media-preload">preload</code>, <code data-x="attr-media-autoplay">autoplay</code>,
  <code data-x="attr-media-loop">loop</code>, <code data-x="attr-media-muted">muted</code>, and <code
  data-x="attr-media-controls">controls</code> attributes are <span data-x="media element
  attributes">the attributes common to all media elements</span>.</p>

  <p>The <dfn element-attr for="video"><code data-x="attr-video-poster">poster</code></dfn>
  attribute gives the <span>URL</span> of an image file that the user agent can show while no video
  data is available.  The attribute, if present, must contain a <span>valid non-empty URL
  potentially surrounded by spaces</span>.</p>

  <div w-nodev>

  <p>If the specified resource is to be used, then, when the element is created or when the <code
  data-x="attr-video-poster">poster</code> attribute is set, changed, or removed, the user agent must
  run the following steps to determine the element's <dfn>poster frame</dfn> (regardless of the
  value of the element's <span>show poster flag</span>):</p> <!-- thus it is unaffected by changes
  to the base URL. -->

  <ol>
   <li><p>If there is an existing instance of this algorithm running for this <code>video</code>
   element, abort that instance of this algorithm without changing the <span>poster
   frame</span>.</p></li>

   <li><p>If the <code data-x="attr-video-poster">poster</code> attribute's value is the empty string
   or if the attribute is absent, then there is no <span>poster frame</span>; return.</p></li>

   <li><p>Let <var>url</var> be the result of <span>encoding-parsing a URL</span> given the <code
   data-x="attr-video-poster">poster</code> attribute's value, relative to the element's <span>node
   document</span>.</p></li>

   <li><p>If <var>url</var> is failure, then return. <span class="note">There is no <span>poster
   frame</span>.</span></p></li>

   <li><p>Let <var>request</var> be a new <span data-x="concept-request">request</span> whose <span
   data-x="concept-request-url">URL</span> is <var>url</var>, <span
   data-x="concept-request-client">client</span> is the element's <span>node document</span>'s
   <span>relevant settings object</span>, <span
   data-x="concept-request-destination">destination</span> is "<code data-x="">image</code>", <span
   data-x="concept-request-initiator-type">initiator type</span> is "<code data-x="">video</code>",
   <span data-x="concept-request-credentials-mode">credentials mode</span> is "<code
   data-x="">include</code>", and whose <span>use-URL-credentials flag</span> is set.

   <li><p><!--FETCH--><span data-x="concept-fetch">Fetch</span> <var>request</var>. This must
   <span>delay the load event</span> of the element's <span>node document</span>.</p></li>

   <!-- could define how to sniff for an image here -->

   <li><p>If an image is thus obtained, the <span>poster frame</span> is that image. Otherwise,
   there is no <span>poster frame</span>.</p></li>
  </ol>

  </div>

  <p class="note">The image given by the <code data-x="attr-video-poster">poster</code> attribute,
  the <i data-x="poster frame">poster frame</i>, is intended to be a representative frame of the
  video (typically one of the first non-blank frames) that gives the user an idea of what the video
  is like.</p>


  <p>The <dfn element-attr for="video"><code
  data-x="attr-video-playsinline">playsinline</code></dfn> attribute is a <span>boolean
  attribute</span>. If present, it serves as a hint to the user agent that the video ought to be
  displayed "inline" in the document by default, constrained to the element's playback area, instead
  of being displayed fullscreen or in an independent resizable window.</p>

  <p class="note">The absence of the <code data-x="attr-video-playsinline">playsinline</code>
  attribute does not imply that the video will display fullscreen by default. Indeed, most user
  agents have chosen to play all videos inline by default, and in such user agents the <code
  data-x="attr-video-playsinline">playsinline</code> attribute has no effect.</p>

  <div w-nodev>

  <hr>

  <p>A <code>video</code> element represents what is given for the first matching condition in the
  list below:</p>

  <dl class="switch">

   <dt>When no video data is available (the element's <code
   data-x="dom-media-readyState">readyState</code> attribute is either <code
   data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</code>, or <code
   data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code> but no video data has yet been obtained at
   all, or the element's <code data-x="dom-media-readyState">readyState</code> attribute is any
   subsequent value but the <span>media resource</span> does not have a video channel)</dt>

   <dd>The <code>video</code> element <span>represents</span> its <span>poster frame</span>, if any,
   or else <span>transparent black</span> with no <span>natural dimensions</span>.</dd>

   <dt>When the <code>video</code> element is <span data-x="dom-media-paused">paused</span>, the <span
   data-x="current playback position">current playback position</span> is the first frame of video,
   and the element's <span>show poster flag</span> is set</dt>

   <dd>The <code>video</code> element <span>represents</span> its <span>poster frame</span>, if any,
   or else the first frame of the video.</dd>

   <dt>When the <code>video</code> element is <span data-x="dom-media-paused">paused</span>, and the
   frame of video corresponding to the <span data-x="current playback position">current playback
   position</span> is not available (e.g. because the video is seeking or buffering)</dt>

   <dt>When the <code>video</code> element is neither <span>potentially playing</span> nor <span
   data-x="dom-media-paused">paused</span> (e.g. when seeking or stalled)</dt>

   <dd>The <code>video</code> element <span>represents</span> the last frame of the video to have
   been rendered.</dd>

   <dt>When the <code>video</code> element is <span data-x="dom-media-paused">paused</span></dt>

   <dd>The <code>video</code> element <span>represents</span> the frame of video corresponding to
   the <span data-x="current playback position">current playback position</span>.</dd>

   <dt>Otherwise (the <code>video</code> element has a video channel and is <span>potentially
   playing</span>)</dt>

   <dd>The <code>video</code> element <span>represents</span> the frame of video at the continuously
   increasing <span data-x="current playback position">"current" position</span>. When the
   <span>current playback position</span> changes such that the last frame rendered is no longer the
   frame corresponding to the <span>current playback position</span> in the video, the new frame
   must be rendered.</dd>
  </dl>

  <p>Frames of video must be obtained from the video track that was <span
  data-x="dom-VideoTrack-selected">selected</span> when the <span>event loop</span> last reached
  <a href="#step1">step 1</a>.</p>

  <p class="note">Which frame in a video stream corresponds to a particular playback position is
  defined by the video stream's format.</p>

  <p>The <code>video</code> element also <span>represents</span> any <span data-x="text track
  cue">text track cues</span> whose <span>text track cue active flag</span> is set and whose
  <span>text track</span> is in the <span data-x="text track showing">showing</span> mode, and any
  audio from the <span>media resource</span>, at the <span>current playback position</span>.</p>

  <p>Any audio associated with the <span>media resource</span> must, if played, be played
  synchronized with the <span>current playback position</span>, at the element's <span>effective
  media volume</span>. The user agent must play the audio from audio tracks that were <span
  data-x="dom-AudioTrack-enabled">enabled</span> when the <span>event loop</span> last reached step
  1.</p>

  <p>In addition to the above, the user agent may provide messages to the user (such as "buffering",
  "no video loaded", "error", or more detailed information) by overlaying text or icons on the video
  or other areas of the element's playback area, or in another appropriate manner.</p>

  <p>User agents that cannot render the video may instead make the element <span
  data-x="represents">represent</span> a link to an external video playback utility or to the video
  data itself.</p>

  <p>When a <code>video</code> element's <span>media resource</span> has a video channel, the
  element provides a <span>paint source</span> whose width is the <span>media resource</span>'s
  <span data-x="concept-video-natural-width">natural width</span>, whose height is the
  <span>media resource</span>'s <span data-x="concept-video-natural-height">natural
  height</span>, and whose appearance is the frame of video corresponding to the <span
  data-x="current playback position">current playback position</span>, if that is available, or else
  (e.g. when the video is seeking or buffering) its previous appearance, if any, or else (e.g.
  because the video is still loading the first frame) blackness.</p>

  <hr>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>video</var>.<span subdfn data-x="dom-video-videoWidth">videoWidth</span></code></dt>
   <dt><code data-x=""><var>video</var>.<span subdfn data-x="dom-video-videoHeight">videoHeight</span></code></dt>

   <dd>
    <p>These attributes return the natural dimensions of the video, or 0 if the dimensions are not
    known.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn data-x="concept-video-natural-width" id="concept-video-intrinsic-width">natural
  width</dfn> and <dfn data-x="concept-video-natural-height"
  id="concept-video-intrinsic-height">natural height</dfn> of the <span>media resource</span> are
  the dimensions of the resource in <span data-x="'px'">CSS pixels</span> after taking into account
  the resource's dimensions, aspect ratio, clean aperture, resolution, and so forth, as defined for
  the format used by the resource. If an anamorphic format does not define how to apply the aspect
  ratio to the video data's dimensions to obtain the "correct" dimensions, then the user agent must
  apply the ratio by increasing one dimension and leaving the other unchanged.</p>

  <p>The <dfn attribute for="HTMLVideoElement"><code
  data-x="dom-video-videoWidth">videoWidth</code></dfn> IDL attribute must return the <span
  data-x="concept-video-natural-width">natural width</span> of the video in <span
  data-x="'px'">CSS pixels</span>. The <dfn attribute for="HTMLVideoElement"><code
  data-x="dom-video-videoHeight">videoHeight</code></dfn> IDL attribute must return the <span
  data-x="concept-video-natural-height">natural height</span> of the video in <span
  data-x="'px'">CSS pixels</span>. If the element's <code
  data-x="dom-media-readyState">readyState</code> attribute is <code
  data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</code>, then the attributes must return 0.</p>

  <p id="dimUpdate">Whenever the <span data-x="concept-video-natural-width">natural width</span>
  or <span data-x="concept-video-natural-height">natural height</span> of the video changes
  (including, for example, because the <span data-x="dom-VideoTrack-selected">selected video
  track</span> was changed), if the element's <code data-x="dom-media-readyState">readyState</code>
  attribute is not <code data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</code>, the user agent must
  <span>queue a media element task</span> given the <span>media element</span> to <span
  data-x="concept-event-fire">fire an event</span> named <code
  data-x="event-media-resize">resize</code> at the <span>media element</span>.</p>

  </div>

  <p>The <code>video</code> element supports <span>dimension attributes</span>.</p>

  <div w-nodev>

  <p>In the absence of style rules to the contrary, video content should be rendered inside the
  element's playback area such that the video content is shown centered in the playback area at the
  largest possible size that fits completely within it, with the video content's aspect ratio being
  preserved. Thus, if the aspect ratio of the playback area does not match the aspect ratio of the
  video, the video will be shown letterboxed or pillarboxed. Areas of the element's playback area
  that do not contain the video represent nothing.</p>

  <p class="note">In user agents that implement CSS, the above requirement can be implemented by
  using the <a href="#video-object-fit">style rule suggested in the Rendering section</a>.</p>

  <p>The <span>natural width</span> of a <code>video</code> element's playback area is the
  <span>natural width</span> of the <span>poster frame</span>, if that is available and the
  element currently <span>represents</span> its poster frame; otherwise, it is the <span
  data-x="concept-video-natural-width">natural width</span> of the video resource, if that is
  available; otherwise the <span>natural width</span> is missing.</p>

  <p>The <span>natural height</span> of a <code>video</code> element's playback area is the
  <span>natural height</span> of the <span>poster frame</span>, if that is available and the
  element currently <span>represents</span> its poster frame; otherwise it is the <span
  data-x="concept-video-natural-height">natural height</span> of the video resource, if that is
  available; otherwise the <span>natural height</span> is missing.</p>

  <p>The <span>default object size</span> is a width of 300 <span data-x="'px'">CSS pixels</span>
  and a height of 150 <span data-x="'px'">CSS pixels</span>. <ref>CSSIMAGES</ref></p>

  <hr>

  <p>User agents should provide controls to enable or disable the display of closed captions, audio
  description tracks, and other additional data associated with the video stream, though such
  features should, again, not interfere with the page's normal rendering.</p>

  <p>User agents may allow users to view the video content in manners more suitable to the user,
  such as fullscreen or in an independent resizable window. User agents may even trigger such a
  viewing mode by default upon playing a video, although they should not do so when the <code
  data-x="attr-video-playsinline">playsinline</code> attribute is specified. As with the other user
  interface features, controls to enable this should not interfere with the page's normal rendering
  unless the user agent is <span data-x="expose a user interface to the user">exposing a user
  interface</span>. In such an independent viewing mode, however, user agents may make full user
  interfaces visible, even if the <code data-x="attr-media-controls">controls</code> attribute is
  absent.</p>

  <p>User agents may allow video playback to affect system features that could interfere with the
  user's experience; for example, user agents could disable screensavers while video playback is in
  progress.</p>

  <hr>

  <p>The <dfn attribute for="HTMLVideoElement"><code data-x="dom-video-poster">poster</code></dfn>
  IDL attribute must <span>reflect</span> the <code data-x="attr-video-poster">poster</code> content
  attribute.</p>

  <p>The <dfn attribute for="HTMLVideoElement"><code
  data-x="dom-video-playsInline">playsInline</code></dfn> IDL attribute must <span>reflect</span>
  the <code data-x="attr-video-playsinline">playsinline</code> content attribute.</p>

  </div>

  <div class="example">

   <p>This example shows how to detect when a video has failed to play correctly:</p>

   <pre><code class="html">&lt;script>
 function failed(e) {
   // video playback failed - show a message saying why
   switch (e.target.error.code) {
     case e.target.error.MEDIA_ERR_ABORTED:
       alert('You aborted the video playback.');
       break;
     case e.target.error.MEDIA_ERR_NETWORK:
       alert('A network error caused the video download to fail part-way.');
       break;
     case e.target.error.MEDIA_ERR_DECODE:
       alert('The video playback was aborted due to a corruption problem or because the video used features your browser did not support.');
       break;
     case e.target.error.MEDIA_ERR_SRC_NOT_SUPPORTED:
       alert('The video could not be loaded, either because the server or network failed or because the format is not supported.');
       break;
     default:
       alert('An unknown error occurred.');
       break;
   }
 }
&lt;/script>
&lt;p>&lt;video src="tgif.vid" autoplay controls onerror="failed(event)">&lt;/video>&lt;/p>
&lt;p>&lt;a href="tgif.vid">Download the video file&lt;/a>.&lt;/p></code></pre>

  </div>




  <!--CODECS

  <div w-nodev>

  <h5>Video and audio codecs for <code>video</code> elements</h5>

  <p>User agents may support any video and audio codecs and container formats.</p>

  <p class="note">Certain user agents might support no codecs at all, e.g. text browsers running
  over SSH connections.</p>

  <!- - similar note in audio codecs section - ->
  <p class="note">Implementations are free to implement support for video codecs either natively, or
  using platform-specific APIs: this specification does not specify how codecs are
  to be implemented.</p>

  </div>

  (when replacing this text, also fix "- -" nested comments)-->





  <h4 id="the-audio-element">The <dfn element id="audio"><code>audio</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Embedded content</span>.</dd>
   <dd>If the element has a <code data-x="attr-media-controls">controls</code> attribute: <span>Interactive content</span>.</dd>
   <dd>If the element has a <code data-x="attr-media-controls">controls</code> attribute: <span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>embedded content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>If the element has a <code data-x="attr-media-src">src</code> attribute:
zero or more <code>track</code> elements, then
<span>transparent</span>, but with no <span>media element</span> descendants.</dd>
   <dd>If the element does not have a <code data-x="attr-media-src">src</code> attribute: zero or more <code>source</code> elements, then
 zero or more <code>track</code> elements, then
 <span>transparent</span>, but with no <span>media element</span> descendants.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-media-src">src</code></dd>
   <dd><code data-x="attr-media-crossorigin">crossorigin</code></dd>
   <dd><code data-x="attr-media-preload">preload</code></dd>
   <dd><code data-x="attr-media-autoplay">autoplay</code></dd>
   <dd><code data-x="attr-media-loop">loop</code></dd>
   <dd><code data-x="attr-media-muted">muted</code></dd>
   <dd><code data-x="attr-media-controls">controls</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-audio">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-audio">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window,
 <span>LegacyFactoryFunction</span>=<span data-x="dom-Audio">Audio</span>(optional DOMString src)]
interface <dfn interface>HTMLAudioElement</dfn> : <span>HTMLMediaElement</span> {
  [<span>HTMLConstructor</span>] constructor();
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLAudioElement</code>.</dd>
  </dl>

  <p>An <code>audio</code> element <span>represents</span> a sound or audio stream.</p>

  <!-- v2 (actually v3) suggestions:
    * Audio synthesis. Use cases from Charles Pritchard:
      > Use a sound of varying pitch to hint to a user the location of their
      > mouse (is it hovering over a button, is it x/y pixels away from the edge
      > of the screen, how close is it to the center).
      >
      > Alter the pitch of a sound to make a very cheap midi instrument.
      >
      > Pre-mix a few generated sounds, because the client processor is slow.
      >
      > Alter the pitch of an actual audio recording, and pre-mix it, to give
      > different sounding voices to pre-recorded readings of a single text. As
      > has been tried for "male" "female" sound fonts.
      >
      > Support very simple audio codecs, and programmable synthesizers.
  -->

  <p>Content may be provided inside the <code>audio</code> element<span w-nodev>. User agents
  should not show this content to the user</span>; it is intended for older web browsers which do
  not support <code>audio</code>, so that text can be shown to the users of these older browsers
  informing them of how to access the audio contents.</p>

  <p class="note">In particular, this content is not intended to address accessibility concerns. To
  make audio content accessible to the deaf or to those with other physical or cognitive
  disabilities, a variety of features are available. If captions or a sign language video are
  available, the <code>video</code> element can be used instead of the <code>audio</code> element to
  play the audio, allowing users to enable the visual alternatives. Chapter titles can be provided
  to aid navigation, using the <code>track</code> element and a <span>WebVTT file</span>. And,
  naturally, transcripts or other textual alternatives can be provided by simply linking to them in
  the prose near the <code>audio</code> element. <ref>WEBVTT</ref></p>

  <p>The <code>audio</code> element is a <span>media element</span> whose <span>media data</span> is
  ostensibly audio data.</p>

  <p>The <code data-x="attr-media-src">src</code>, <code data-x="attr-media-crossorigin">crossorigin</code>,
  <code data-x="attr-media-preload">preload</code>, <code data-x="attr-media-autoplay">autoplay</code>,
  <code data-x="attr-media-loop">loop</code>, <code data-x="attr-media-muted">muted</code>, and <code
  data-x="attr-media-controls">controls</code> attributes are <span data-x="media element
  attributes">the attributes common to all media elements</span>.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>audio</var> = new <span subdfn data-x="dom-Audio">Audio</span>([ <var>url</var> ])</code></dt>

   <dd>
    <p>Returns a new <code>audio</code> element, with the <code data-x="attr-media-src">src</code>
    attribute set to the value passed in the argument, if applicable.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>A legacy factory function is provided for creating <code>HTMLAudioElement</code> objects (in
  addition to the factory methods from DOM such as <code
  data-x="dom-Document-createElement">createElement()</code>): <dfn><code
  data-x="dom-Audio">Audio(<var>src</var>)</code></dfn>. When invoked, the legacy factory function
  must perform the following steps:</p>

  <ol>
   <li><p>Let <var>document</var> be the <span>current global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>Let <var>audio</var> be the result of <span data-x="create an element">creating an
   element</span> given <var>document</var>, "<code data-x="">audio</code>", and the <span>HTML
   namespace</span>.</p></li>

   <li><p><span data-x="concept-element-attributes-set-value">Set an attribute value</span> for
   <var>audio</var> using "<code data-x="attr-media-preload">preload</code>" and "<code
   data-x="attr-media-preload-auto">auto</code>".</p></li>

   <li><p>If <var>src</var> is given, then <span data-x="concept-element-attributes-set-value">set
   an attribute value</span> for <var>audio</var> using "<code data-x="attr-media-src">src</code>"
   and <var>src</var>. (This will <a href="#concept-media-load-algorithm-at-creation">cause the user
   agent to invoke</a> the object's <span data-x="concept-media-load-algorithm">resource selection
   algorithm</span> before returning.)</p></li>

   <li><p>Return <var>audio</var>.</p></li>
  </ol>

  </div>



  <!--CODECS

  <div w-nodev>

  <h5>Audio codecs for <code>audio</code> elements</h5>

  <p>User agents may support any audio codecs and container formats.</p>

  <p>User agents must support the WAVE container format with audio encoded using the 16 bit PCM (LE)
  codec, at sampling frequencies of 11.025kHz, 22.050kHz, and 44.100kHz, and for both mono and
  stereo. <a href="#- -refsWAVE">[WAVE]</a></p>

  <!- -
   <dt id="- -refsWAVE">WAVE</dt>
   <dd>https://en.wikipedia.org/wiki/WAV? </dd>
  - ->

  <!- - similar note in video codecs section - ->
  <p class="note">Implementations are free to implement support for audio codecs either natively, or
  using platform-specific APIs: this specification does not specify how codecs are
  to be implemented.</p>

  </div>

  (when replacing this text, also fix "- -" nested comments)-->



  <h4>The <dfn element><code>track</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As a child of a <span>media element</span>, before any <span>flow content</span>.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="concept-content-nothing">Nothing</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-track-kind">kind</code></dd>
   <dd><code data-x="attr-track-src">src</code></dd>
   <dd><code data-x="attr-track-srclang">srclang</code></dd>
   <dd><code data-x="attr-track-label">label</code></dd>
   <dd><code data-x="attr-track-default">default</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-track">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-track">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLTrackElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-track-kind">kind</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-track-src">src</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-track-srclang">srclang</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-track-label">label</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-track-default">default</span>;

  const unsigned short <span data-x="dom-track-NONE">NONE</span> = 0;
  const unsigned short <span data-x="dom-track-LOADING">LOADING</span> = 1;
  const unsigned short <span data-x="dom-track-LOADED">LOADED</span> = 2;
  const unsigned short <span data-x="dom-track-ERROR">ERROR</span> = 3;
  readonly attribute unsigned short <span data-x="dom-track-readyState">readyState</span>;

  readonly attribute <span>TextTrack</span> <span data-x="dom-track-track">track</span>;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLTrackElement</code>.</dd>
  </dl>

  <p>The <code>track</code> element allows authors to specify explicit external timed <span
  data-x="text track">text tracks</span> for <span data-x="media element">media elements</span>. It
  does not <span data-x="represents">represent</span> anything on its own.</p>

  <p>The <dfn element-attr for="track"><code data-x="attr-track-kind">kind</code></dfn> attribute is
  an <span>enumerated attribute</span> with the following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="track/kind"><code
     data-x="attr-track-kind-keyword-subtitles">subtitles</code></dfn>
     <td><dfn data-x="attr-track-kind-subtitles">Subtitles</dfn>
     <td>
      Transcription or translation of the dialogue, suitable for when the sound is available but not understood (e.g. because the user does not understand the language of the <span>media resource</span>'s audio track).
      Overlaid on the video.
    <tr>
     <td><dfn attr-value for="track/kind"><code
     data-x="attr-track-kind-keyword-captions">captions</code></dfn>
     <td><dfn data-x="attr-track-kind-captions">Captions</dfn>
     <td>
      Transcription or translation of the dialogue, sound effects, relevant musical cues, and other relevant audio information, suitable for when sound is unavailable or not clearly audible (e.g. because it is muted, drowned-out by ambient noise, or because the user is deaf).
      Overlaid on the video; labeled as appropriate for the hard-of-hearing.
    <tr>
     <td><dfn attr-value for="track/kind"><code
     data-x="attr-track-kind-keyword-descriptions">descriptions</code></dfn>
     <td><dfn data-x="attr-track-kind-descriptions">Descriptions</dfn>
     <td>
      Textual descriptions of the video component of the <span>media resource</span>, intended for audio synthesis when the visual component is obscured, unavailable, or not usable (e.g. because the user is interacting with the application without a screen while driving, or because the user is blind).
      Synthesized as audio.
    <tr>
     <td><dfn attr-value for="track/kind"><code
     data-x="attr-track-kind-keyword-chapters">chapters</code></dfn>
     <td><dfn data-x="attr-track-kind-chapters">Chapters metadata</dfn>
     <td rowspan="2">
      Tracks intended for use from script.
      Not displayed by the user agent.
    <tr>
     <td><dfn attr-value for="track/kind"><code
     data-x="attr-track-kind-keyword-metadata">metadata</code></dfn>
     <td><dfn data-x="attr-track-kind-metadata">Metadata</dfn>
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> is the <span
  data-x="attr-track-kind-subtitles">subtitles</span> state, and its <i data-x="invalid value
  default">invalid value default</i> is the <span data-x="attr-track-kind-metadata">metadata</span>
  state.</p>

  <p>The <dfn element-attr for="track"><code data-x="attr-track-src">src</code></dfn> attribute
  gives the <span>URL</span> of the text track data. The value must be a <span>valid non-empty URL
  potentially surrounded by spaces</span>. This attribute must be present.</p>

  <div w-nodev>

  <p>The element has an associated <dfn>track URL</dfn> (a string), initially the empty string.</p>

  <p>When the element's <code data-x="attr-track-src">src</code> attribute is set, run these steps:

  <ol>
   <li><p>Let <var>trackURL</var> be failure.</p></li>

   <li><p>Let <var>value</var> be the element's <code data-x="attr-track-src">src</code> attribute
   value.</p></li>

   <li><p>If <var>value</var> is not the empty string, then set <var>trackURL</var> to the result of
   <span>encoding-parsing-and-serializing a URL</span> given <var>value</var>, relative to the
   element's <span>node document</span>.</p></li>

   <li><p>Set the element's <span>track URL</span> to <var>trackURL</var> if it is not failure;
   otherwise to the empty string.</p></li>
  </ol>

  </div>

  <p>If the element's <span>track URL</span> identifies a WebVTT resource, and the element's <code
  data-x="attr-track-kind">kind</code> attribute is not in the <span
  data-x="attr-track-kind-chapters">chapters metadata</span> or <span
  data-x="attr-track-kind-metadata">metadata</span> state, then the WebVTT file must be a
  <span>WebVTT file using cue text</span>. <ref>WEBVTT</ref></p>

  <p>The <dfn element-attr for="track"><code data-x="attr-track-srclang">srclang</code></dfn>
  attribute gives the language of the text track data. The value must be a valid BCP 47 language
  tag. This attribute must be present if the element's <code data-x="attr-track-kind">kind</code>
  attribute is in the <span data-x="attr-track-kind-subtitles">subtitles</span> state.
  <ref>BCP47</ref></p>

  <div w-nodev>

  <p>If the element has a <code data-x="attr-track-srclang">srclang</code> attribute whose value is
  not the empty string, then the element's <dfn>track language</dfn> is the value of the attribute.
  Otherwise, the element has no <span>track language</span>.</p>

  </div>

  <p>The <dfn element-attr for="track"><code data-x="attr-track-label">label</code></dfn> attribute
  gives a user-readable title for the track. This title is used by user agents when listing <span
  data-x="attr-track-kind-subtitles">subtitle</span>, <span
  data-x="attr-track-kind-captions">caption</span>, and <span
  data-x="attr-track-kind-descriptions">audio description</span> tracks in their user interface.</p>

  <p>The value of the <code data-x="attr-track-label">label</code> attribute, if the attribute is
  present, must not be the empty string. Furthermore, there must not be two <code>track</code>
  element children of the same <span>media element</span> whose <code
  data-x="attr-track-kind">kind</code> attributes are in the same state, whose <code
  data-x="attr-track-srclang">srclang</code> attributes are both missing or have values that
  represent the same language, and whose <code data-x="attr-track-label">label</code> attributes are
  again both missing or both have the same value.</p>

  <div w-nodev>

  <p>If the element has a <code data-x="attr-track-label">label</code> attribute whose value is not
  the empty string, then the element's <dfn>track label</dfn> is the value of the attribute.
  Otherwise, the element's <span>track label</span> is an empty string.</p>

  </div>

  <p>The <dfn element-attr for="track"><code data-x="attr-track-default">default</code></dfn>
  attribute is a <span>boolean attribute</span>, which, if specified, indicates that the track is to
  be enabled if the user's preferences do not indicate that another track would be more
  appropriate.</p>

  <p>Each <span>media element</span> must have no more than one <code>track</code> element child
  whose <code data-x="attr-track-kind">kind</code> attribute is in the <span
  data-x="attr-track-kind-subtitles">subtitles</span> or <span
  data-x="attr-track-kind-captions">captions</span> state and whose <code
  data-x="attr-track-default">default</code> attribute is specified.</p>

  <p>Each <span>media element</span> must have no more than one <code>track</code> element child
  whose <code data-x="attr-track-kind">kind</code> attribute is in the <span
  data-x="attr-track-kind-descriptions">description</span> state and whose <code
  data-x="attr-track-default">default</code> attribute is specified.</p>

  <p>Each <span>media element</span> must have no more than one <code>track</code> element child
  whose <code data-x="attr-track-kind">kind</code> attribute is in the <span
  data-x="attr-track-kind-chapters">chapters metadata</span> state and whose <code
  data-x="attr-track-default">default</code> attribute is specified.</p>

  <p class="note">There is no limit on the number of <code>track</code> elements whose <code
  data-x="attr-track-kind">kind</code> attribute is in the <span
  data-x="attr-track-kind-metadata">metadata</span> state and whose <code
  data-x="attr-track-default">default</code> attribute is specified.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>track</var>.<span subdfn data-x="dom-track-readyState">readyState</span></code></dt>
   <dd>
    <p>Returns the <span>text track readiness state</span>, represented by a number from the
    following list:</p>
    <dl>
     <dt><code data-x=""><var>track</var>.<span subdfn data-x="dom-track-NONE">NONE</span> (0)</code></dt>
     <dd><p>The <span>text track not loaded</span> state.</p></dd>
     <dt><code data-x=""><var>track</var>.<span subdfn data-x="dom-track-LOADING">LOADING</span> (1)</code></dt>
     <dd><p>The <span>text track loading</span> state.</p></dd>
     <dt><code data-x=""><var>track</var>.<span subdfn data-x="dom-track-LOADED">LOADED</span> (2)</code></dt>
     <dd><p>The <span>text track loaded</span> state.</p></dd>
     <dt><code data-x=""><var>track</var>.<span subdfn data-x="dom-track-ERROR">ERROR</span> (3)</code></dt>
     <dd><p>The <span>text track failed to load</span> state.</p></dd>
    </dl>
   </dd>

   <dt><code data-x=""><var>track</var>.<span subdfn data-x="dom-track-track">track</span></code></dt>

   <dd>
    <p>Returns the <code>TextTrack</code> object corresponding to the <span>text track</span> of the
    <code>track</code> element.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLTrackElement"><code
  data-x="dom-track-readyState">readyState</code></dfn> attribute must return the numeric value
  corresponding to the <span>text track readiness state</span> of the <code>track</code> element's
  <span>text track</span>, as defined by the following list:</p>

  <dl>
   <dt><dfn const for="HTMLTrackElement"><code data-x="dom-track-NONE">NONE</code></dfn> (numeric value 0)</dt>
   <dd>The <span>text track not loaded</span> state.</dd>
   <dt><dfn const for="HTMLTrackElement"><code data-x="dom-track-LOADING">LOADING</code></dfn> (numeric value 1)</dt>
   <dd>The <span>text track loading</span> state.</dd>
   <dt><dfn const for="HTMLTrackElement"><code data-x="dom-track-LOADED">LOADED</code></dfn> (numeric value 2)</dt>
   <dd>The <span>text track loaded</span> state.</dd>
   <dt><dfn const for="HTMLTrackElement"><code data-x="dom-track-ERROR">ERROR</code></dfn> (numeric value 3)</dt>
   <dd>The <span>text track failed to load</span> state.</dd>
  </dl>

  <p>The <dfn attribute for="HTMLTrackElement"><code data-x="dom-track-track">track</code></dfn> IDL
  attribute must, on getting, return the <code>track</code> element's <span>text track</span>'s
  corresponding <code>TextTrack</code> object.</p>

  <p>The <dfn attribute for="HTMLTrackElement"><code data-x="dom-track-src">src</code></dfn>, <dfn
  attribute for="HTMLTrackElement"><code data-x="dom-track-srclang">srclang</code></dfn>, <dfn
  attribute for="HTMLTrackElement"><code data-x="dom-track-label">label</code></dfn>, and <dfn
  attribute for="HTMLTrackElement"><code data-x="dom-track-default">default</code></dfn> IDL
  attributes must <span>reflect</span> the respective content attributes of the same name. The <dfn
  attribute for="HTMLTrackElement"><code data-x="dom-track-kind">kind</code></dfn> IDL attribute
  must <span>reflect</span> the content attribute of the same name, <span>limited to only known
  values</span>.</p>

  </div>

  <div class="example">

   <p>This video has subtitles in several languages:</p>

   <pre><code class="html">&lt;video src="brave.webm">
 &lt;track kind=subtitles src=brave.en.vtt srclang=en label="English">
 &lt;track kind=captions src=brave.en.hoh.vtt srclang=en label="English for the Hard of Hearing">
 &lt;track kind=subtitles src=brave.fr.vtt srclang=fr lang=fr label="Fran&ccedil;ais">
 &lt;track kind=subtitles src=brave.de.vtt srclang=de lang=de label="Deutsch">
&lt;/video></code></pre>

   <p>(The <code data-x="attr-lang">lang</code> attributes on the last two describe the language of
   the <code data-x="attr-track-label">label</code> attribute, not the language of the subtitles
   themselves. The language of the subtitles is given by the <code
   data-x="attr-track-srclang">srclang</code> attribute.)</p>

  </div>

  <h4>Media elements</h4>

  <p><span>HTMLMediaElement</span> objects (<code>audio</code> and <code>video</code>, in this
  specification) are simply known as <dfn data-x="media element" data-lt="media element"
  export>media elements</dfn>.</p>

  <pre><code class="idl">enum <dfn enum>CanPlayTypeResult</dfn> { "" /* <span data-x="dom-CanPlayTypeResult-nil">empty string</span> */, "<span data-x="dom-CanPlayTypeResult-maybe">maybe</span>", "<span data-x="dom-CanPlayTypeResult-probably">probably</span>" };
typedef (<span>MediaStream</span> or <span>MediaSource</span> or <span>Blob</span>) <dfn typedef>MediaProvider</dfn>;

[Exposed=Window]
interface <dfn interface>HTMLMediaElement</dfn> : <span>HTMLElement</span> {

  // error state
  readonly attribute <span>MediaError</span>? <span data-x="dom-media-error">error</span>;

  // network state
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-media-src">src</span>;
  attribute <span>MediaProvider</span>? <span data-x="dom-media-srcObject">srcObject</span>;
  readonly attribute USVString <span data-x="dom-media-currentSrc">currentSrc</span>;
  [<span>CEReactions</span>] attribute DOMString? <span data-x="dom-media-crossOrigin">crossOrigin</span>;
  const unsigned short <span data-x="dom-media-NETWORK_EMPTY">NETWORK_EMPTY</span> = 0;
  const unsigned short <span data-x="dom-media-NETWORK_IDLE">NETWORK_IDLE</span> = 1;
  const unsigned short <span data-x="dom-media-NETWORK_LOADING">NETWORK_LOADING</span> = 2;
  const unsigned short <span data-x="dom-media-NETWORK_NO_SOURCE">NETWORK_NO_SOURCE</span> = 3;
  readonly attribute unsigned short <span data-x="dom-media-networkState">networkState</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-media-preload">preload</span>;
<!--v3BUF readonly attribute double <span data-x="dom-media-bufferingRate">bufferingRate</span>;
  readonly attribute boolean <span data-x="dom-media-bufferingThrottled">bufferingThrottled</span>;
-->  readonly attribute <span>TimeRanges</span> <span data-x="dom-media-buffered">buffered</span>;
  undefined <span data-x="dom-media-load">load</span>();
  <span>CanPlayTypeResult</span> <span data-x="dom-navigator-canPlayType">canPlayType</span>(DOMString type);

  // ready state
  const unsigned short <span data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</span> = 0;
  const unsigned short <span data-x="dom-media-HAVE_METADATA">HAVE_METADATA</span> = 1;
  const unsigned short <span data-x="dom-media-HAVE_CURRENT_DATA">HAVE_CURRENT_DATA</span> = 2;
  const unsigned short <span data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</span> = 3;
  const unsigned short <span data-x="dom-media-HAVE_ENOUGH_DATA">HAVE_ENOUGH_DATA</span> = 4;
  readonly attribute unsigned short <span data-x="dom-media-readyState">readyState</span>;
  readonly attribute boolean <span data-x="dom-media-seeking">seeking</span>;

  // playback state
  attribute double <span data-x="dom-media-currentTime">currentTime</span>;
  undefined <span data-x="dom-media-fastSeek">fastSeek</span>(double time);
  readonly attribute unrestricted double <span data-x="dom-media-duration">duration</span>;
  <span data-x="idl-object">object</span> <span data-x="dom-media-getStartDate">getStartDate</span>();
  readonly attribute boolean <span data-x="dom-media-paused">paused</span>;
  attribute double <span data-x="dom-media-defaultPlaybackRate">defaultPlaybackRate</span>;
  attribute double <span data-x="dom-media-playbackRate">playbackRate</span>;
  attribute boolean <span data-x="dom-media-preservesPitch">preservesPitch</span>;
  readonly attribute <span>TimeRanges</span> <span data-x="dom-media-played">played</span>;
  readonly attribute <span>TimeRanges</span> <span data-x="dom-media-seekable">seekable</span>;
  readonly attribute boolean <span data-x="dom-media-ended">ended</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-media-autoplay">autoplay</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-media-loop">loop</span>;
  <span data-x="idl-Promise">Promise</span>&lt;undefined&gt; <span data-x="dom-media-play">play</span>();
  undefined <span data-x="dom-media-pause">pause</span>();

  // controls
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-media-controls">controls</span>;
  attribute double <span data-x="dom-media-volume">volume</span>;
  attribute boolean <span data-x="dom-media-muted">muted</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-media-defaultMuted">defaultMuted</span>;

  // tracks
  [SameObject] readonly attribute <span>AudioTrackList</span> <span data-x="dom-media-audioTracks">audioTracks</span>;
  [SameObject] readonly attribute <span>VideoTrackList</span> <span data-x="dom-media-videoTracks">videoTracks</span>;
  [SameObject] readonly attribute <span>TextTrackList</span> <span data-x="dom-media-textTracks">textTracks</span>;
  <span>TextTrack</span> <span data-x="dom-media-addTextTrack">addTextTrack</span>(<span>TextTrackKind</span> kind, optional DOMString label = "", optional DOMString language = "");
};</code></pre>

  <p>The <dfn>media element attributes</dfn>, <code data-x="attr-media-src">src</code>, <code
  data-x="attr-media-crossorigin">crossorigin</code>, <code
  data-x="attr-media-preload">preload</code>, <code data-x="attr-media-autoplay">autoplay</code>,
  <code data-x="attr-media-loop">loop</code>, <code data-x="attr-media-muted">muted</code>, and
  <code data-x="attr-media-controls">controls</code>, apply to all <span data-x="media
  element">media elements</span>. They are defined in this section.</p>

  <!-- proposed v2 (actually v3!) features:
    * frame forward / backwards / step(n) while paused
    * per-frame control: get current frame; set current frame
    * queue of content
      - pause current stream and insert content at front of queue to play immediately
      - pre-download another stream
      - add stream(s) to play at end of current stream
      - pause playback upon reaching a certain time
      - playlists, with the ability to get metadata out of them (e.g. xspf)
    * general meta data, implemented as getters (don't expose the whole thing)
      - getMetadata(key: string, language: string) => HTMLImageElement or string
      - onmetadatachanged (no context info)
      - general custom metadata store (ratings, etc.)
    * video: applying CSS filters
    * an event to notify people of when the video size changes
      (e.g. for chained Ogg streams of multiple independent videos)
      (or for when the current video track changes)
    * balance and 3D position audio
    * audio filters
    * audio synthesis (see <audio> section for use cases)
    * feedback to the script on how well the video is playing
       - frames per second?
       - skipped frames per second?
       - an event that reports playback difficulties?
       - an arbitrary quality metric?
    * bufferingRate/bufferingThrottled (see v3BUF)
    * events for when the user agent's controls get shown or hidden
      so that the author's controls can get away of the UA's
  -->

  <p><span data-x="media element">Media elements</span> are used to present audio data, or video and
  audio data, to the user. This is referred to as <dfn for=HTMLMediaElement export>media data</dfn>
  in this section, since this section applies equally to <span data-x="media element">media
  elements</span> for audio or for video.

  The term <dfn>media resource</dfn> is used to refer to the complete set of media data, e.g. the
  complete video file, or complete audio file.

  </p>

  <p>A <span>media resource</span> has an associated
  <dfn data-x="media-resource-origin">origin</dfn>, which is either "<code data-x="">none</code>",
  "<code data-x="">multiple</code>", "<code data-x="">rewritten</code>", or an
  <span>origin</span>. It is initially set to "<code data-x="">none</code>".</p>

  <p>A <span>media resource</span> can have multiple audio and video tracks. For the purposes of a
  <span>media element</span>, the video data of the <span>media resource</span> is only that of the
  currently selected track (if any) as given by the element's <code
  data-x="dom-media-videoTracks">videoTracks</code> attribute when the <span>event loop</span> last
  reached <a href="#step1">step 1</a>, and the audio data of the <span>media resource</span> is the result of mixing all
  the currently enabled tracks (if any) given by the element's <code
  data-x="dom-media-audioTracks">audioTracks</code> attribute when the <span>event loop</span> last
  reached <a href="#step1">step 1</a>.</p>

  <p class="note">Both <code>audio</code> and <code>video</code> elements can be used for both audio
  and video. The main difference between the two is simply that the <code>audio</code> element has
  no playback area for visual content (such as video or captions), whereas the <code>video</code>
  element does.</p>

  <div w-nodev>

  <p>Each <span>media element</span> has a unique <dfn>media element event task source</dfn>.</p>

  <p>To <dfn export>queue a media element task</dfn> with a <span>media element</span>
  <var>element</var> and a series of steps <var>steps</var>, <span>queue an element task</span> on
  the <span>media element</span>'s <span>media element event task source</span> given
  <var>element</var> and <var>steps</var>.</p>

  </div>



  <h5>Error codes</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-error">error</span></code></dt>

   <dd>
    <p>Returns a <code>MediaError</code> object representing the current error state of the
    element.</p>

    <p>Returns null if there is no error.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>All <span data-x="media element">media elements</span> have an associated error status, which
  records the last error the element encountered since its <span
  data-x="concept-media-load-algorithm">resource selection algorithm</span> was last invoked. The
  <dfn attribute for="HTMLMediaElement"><code data-x="dom-media-error">error</code></dfn> attribute,
  on getting, must return the <code>MediaError</code> object created for this last error, or null if
  there has not been an error.</p>

  </div>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>MediaError</dfn> {
  const unsigned short <span data-x="dom-MediaError-MEDIA_ERR_ABORTED">MEDIA_ERR_ABORTED</span> = 1;
  const unsigned short <span data-x="dom-MediaError-MEDIA_ERR_NETWORK">MEDIA_ERR_NETWORK</span> = 2;
  const unsigned short <span data-x="dom-MediaError-MEDIA_ERR_DECODE">MEDIA_ERR_DECODE</span> = 3;
  const unsigned short <span data-x="dom-MediaError-MEDIA_ERR_SRC_NOT_SUPPORTED">MEDIA_ERR_SRC_NOT_SUPPORTED</span> = 4;

  readonly attribute unsigned short <span data-x="dom-MediaError-code">code</span>;
  readonly attribute DOMString <span data-x="dom-MediaError-message">message</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>media</var>.<span data-x="dom-media-error">error</span>.<span subdfn data-x="dom-MediaError-code">code</span></code></dt>

   <dd><p>Returns the current error's error code, from the list below.</p></dd>

   <dt><code data-x=""><var>media</var>.<span data-x="dom-media-error">error</span>.<span subdfn data-x="dom-MediaError-message">message</span></code></dt>

   <dd>
    <p>Returns a specific informative diagnostic message about the error condition encountered. The
    message and message format are not generally uniform across different user agents. If no such
    message is available, then the empty string is returned.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>Every <code>MediaError</code> object has a <dfn
  data-x="concept-MediaError-message">message</dfn>, which is a string, and a <dfn
  data-x="concept-MediaError-code">code</dfn>, which is one of the following:</p>

  </div>

  <dl>
   <dt><dfn const for="MediaError"><code data-x="dom-MediaError-MEDIA_ERR_ABORTED">MEDIA_ERR_ABORTED</code></dfn> (numeric value 1)</dt>

   <dd>The fetching process for the <span>media resource</span> was aborted by the user agent at the
   user's request.</dd>

   <dt><dfn const for="MediaError"><code data-x="dom-MediaError-MEDIA_ERR_NETWORK">MEDIA_ERR_NETWORK</code></dfn> (numeric value 2)</dt>

   <dd>A network error of some description caused the user agent to stop fetching the <span>media
   resource</span>, after the resource was established to be usable.</dd>

   <dt><dfn const for="MediaError"><code data-x="dom-MediaError-MEDIA_ERR_DECODE">MEDIA_ERR_DECODE</code></dfn> (numeric value 3)</dt>

   <dd>An error of some description occurred while decoding the <span>media resource</span>, after
   the resource was established to be usable.</dd>

   <dt><dfn const for="MediaError"><code data-x="dom-MediaError-MEDIA_ERR_SRC_NOT_SUPPORTED">MEDIA_ERR_SRC_NOT_SUPPORTED</code></dfn> (numeric value 4)</dt>

   <dd>The <span>media resource</span> indicated by the <code data-x="attr-media-src">src</code>
   attribute or <span>assigned media provider object</span> was not suitable.</dd>
  </dl>

  <div w-nodev>

  <p>To <dfn export data-x="creating a MediaError">create a <code>MediaError</code></dfn>, given an
  error code which is one of the above values, return a new <code>MediaError</code> object whose
  <span data-x="concept-MediaError-code">code</span> is the given error code and whose <span
  data-x="concept-MediaError-message">message</span> is a string containing any details the user
  agent is able to supply about the cause of the error condition, or the empty string if the user
  agent is unable to supply such details. This message string must not contain only the information
  already available via the supplied error code; for example, it must not simply be a translation of
  the code into a string format. If no additional information is available beyond that provided by
  the error code, the <span data-x="concept-MediaError-message">message</span> must be set to the
  empty string.</p>

  <p>The <dfn attribute for="MediaError"><code data-x="dom-MediaError-code">code</code></dfn>
  getter steps are to return <span>this</span>'s <span
  data-x="concept-MediaError-code">code</span>.</p>

  <p>The <dfn attribute for="MediaError"><code data-x="dom-MediaError-message">message</code></dfn>
  getter steps are to return <span>this</span>'s <span
  data-x="concept-MediaError-message">message</span>.</p>

  </div>




  <h5>Location of the media resource</h5>

  <p>The <dfn element-attr for="audio,video"><code data-x="attr-media-src">src</code></dfn> content
  attribute on <span data-x="media element">media elements</span> gives the <span>URL</span> of the
  media resource (video, audio) to show. The attribute, if present, must contain a <span>valid
  non-empty URL potentially surrounded by spaces</span>.</p>

  <p>If the <code data-x="attr-itemprop">itemprop</code> attribute is specified on the <span>media
  element</span>, then the <code data-x="attr-media-src">src</code> attribute must also be
  specified.</p>

  <p>The <dfn element-attr for="audio,video"><code
  data-x="attr-media-crossorigin">crossorigin</code></dfn> content attribute on <span data-x="media
  element">media elements</span> is a <span>CORS settings attribute</span>.</p>

  <div w-nodev>

  <p id="concept-media-load-algorithm-at-creation">If a <span>media element</span> is created with a
  <code data-x="attr-media-src">src</code> attribute, the user agent must <span>immediately</span> invoke the
  <span>media element</span>'s <span data-x="concept-media-load-algorithm">resource selection
  algorithm</span>.</p>

  <p>If a <code data-x="attr-media-src">src</code> attribute of a <span>media element</span> is set
  or changed, the user agent must invoke the <span>media element</span>'s <span>media element load
  algorithm</span>. (<em>Removing</em> the <code data-x="attr-media-src">src</code> attribute does
  not do this, even if there are <code>source</code> elements present.)</p>

  <p>The <dfn attribute for="HTMLMediaElement"><code data-x="dom-media-src">src</code></dfn> IDL
  attribute on <span data-x="media element">media elements</span> must <span>reflect</span> the
  content attribute of the same name.</p>

  <p>The <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-crossOrigin">crossOrigin</code></dfn> IDL attribute must <span>reflect</span>
  the <code data-x="attr-media-crossorigin">crossorigin</code> content attribute, <span>limited to
  only known values</span>.</p>

  </div>

  <p>A <dfn>media provider object</dfn> is an object that can represent a <span>media
  resource</span>, separate from a <span>URL</span>. <code>MediaStream</code> objects,
  <code>MediaSource</code> objects, and <code>Blob</code> objects are all <span data-x="media
  provider object">media provider objects</span>.</p>

  <div w-nodev>

  <p>Each <span>media element</span> can have an <dfn>assigned media provider object</dfn>, which is a
  <span>media provider object</span>. When a <span>media element</span> is created, it has no
  <span>assigned media provider object</span>.</p>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-srcObject">srcObject</span> [ = <var>source</var> ]</code></dt>

   <dd>
    <p>Allows the <span>media element</span> to be assigned a <span>media provider
    object</span>.</p>
   </dd>

   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-currentSrc">currentSrc</span></code></dt>

   <dd>
    <p>Returns the <span>URL</span> of the current <span>media resource</span>, if any.</p>

    <p>Returns the empty string when there is no <span>media resource</span>, or it doesn't have a
    <span>URL</span>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-currentSrc">currentSrc</code></dfn> IDL attribute must initially be set to the
  empty string. Its value is changed by the <span data-x="concept-media-load-algorithm">resource
  selection algorithm</span> defined below.</p>

  <p>The <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-srcObject">srcObject</code></dfn> IDL attribute, on getting, must return the
  element's <span>assigned media provider object</span>, if any, or null otherwise.  On setting, it
  must set the element's <span>assigned media provider object</span> to the new value, and then
  invoke the element's <span>media element load algorithm</span>.</p>

  </div>

  <p class="note">There are three ways to specify a <span>media resource</span>: the <code
  data-x="dom-media-srcObject">srcObject</code> IDL attribute, the <code
  data-x="attr-media-src">src</code> content attribute, and <code>source</code> elements. The IDL
  attribute takes priority, followed by the content attribute, followed by the elements.</p>



  <h5>MIME types</h5>

  <p>A <span>media resource</span> can be described in terms of its <em>type</em>, specifically a
  <span>MIME type</span>, in some cases with a <code data-x="">codecs</code> parameter. (Whether the
  <code data-x="">codecs</code> parameter is allowed or not depends on the MIME type.)
  <ref>RFC6381</ref></p>

  <p>Types are usually somewhat incomplete descriptions; for example "<code
  data-x="">video/mpeg</code>" doesn't say anything except what the container type is, and even a
  type like "<code data-x="">video/mp4; codecs=&quot;avc1.42E01E, mp4a.40.2&quot;</code>" doesn't
  include information like the actual bitrate (only the maximum bitrate). Thus, given a type, a user
  agent can often only know whether it <em>might</em> be able to play media of that type (with
  varying levels of confidence), or whether it definitely <em>cannot</em> play media of that
  type.</p>

  <p><dfn>A type that the user agent knows it cannot render</dfn> is one that describes a resource
  that the user agent definitely does not support, for example because it doesn't recognize the
  container type, or it doesn't support the listed codecs.</p>

  <p>The <span>MIME type</span> "<code>application/octet-stream</code>" with no parameters is never
  <span>a type that the user agent knows it cannot render</span>. User agents must treat that type
  as equivalent to the lack of any explicit <span data-x="Content-Type">Content-Type metadata</span>
  when it is used to label a potential <span>media resource</span>.</p>

  <p class="note">Only the <span>MIME type</span> "<code>application/octet-stream</code>" with no
  parameters is special-cased here; if any parameter appears with it, it will be treated just like
  any other <span>MIME type</span>. This is a deviation from the rule <!-- in RFC 2046, section 1,
  paragraph 3 --> that unknown <span>MIME type</span> parameters <!--non-normative-->should be
  ignored.</p> <!-- but not really a "willful violation" since it's not that the types are not being
  ignored, just that before the type is handled as a type, there's a special case for a particular
  set of strings -->

  <dl class="domintro">
   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-navigator-canPlayType">canPlayType</span>(<var>type</var>)</code></dt>

   <dd>
    <p>Returns the empty string (a negative response), "maybe", or "probably" based on how confident
    the user agent is that it can play media resources of the given type.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="HTMLMediaElement"><code
  data-x="dom-navigator-canPlayType">canPlayType(<var>type</var>)</code></dfn> method must return
  <dfn data-x="dom-CanPlayTypeResult-nil">the empty string</dfn> if <var>type</var> is <span>a type
  that the user agent knows it cannot render</span> or is the type
  "<code>application/octet-stream</code>"; it must return "<dfn enum-value
  for="CanPlayTypeResult"><code data-x="dom-CanPlayTypeResult-probably">probably</code></dfn>" if
  the user agent is confident that the type represents a <span>media resource</span> that it can
  render if used in with this <code>audio</code> or <code>video</code> element; and it must return
  "<dfn enum-value for="CanPlayTypeResult"><code
  data-x="dom-CanPlayTypeResult-maybe">maybe</code></dfn>" otherwise. Implementers are encouraged to
  return "<code data-x="dom-CanPlayTypeResult-maybe">maybe</code>" unless the type can be
  confidently established as being supported or not. Generally, a user agent should never return
  "<code data-x="dom-CanPlayTypeResult-probably">probably</code>" for a type that allows the <code
  data-x="">codecs</code> parameter if that parameter is not present.</p>

  </div>

  <div class="example">

   <p>This script tests to see if the user agent supports a (fictional) new format to dynamically
   decide whether to use a <code>video</code> element:</p>

   <pre><code class="html">&lt;section id="video">
 &lt;p>&lt;a href="playing-cats.nfv">Download video&lt;/a>&lt;/p>
&lt;/section>
&lt;script>
 const videoSection = document.getElementById('video');
 const videoElement = document.createElement('video');
 const support = videoElement.canPlayType('video/x-new-fictional-format;codecs="kittens,bunnies"');
 if (support === "probably") {
   videoElement.setAttribute("src", "playing-cats.nfv");
   videoSection.replaceChildren(videoElement);
 }
&lt;/script></code></pre>

  </div>

  <p class="note">The <code data-x="attr-source-type">type</code> attribute of the
  <code>source</code> element allows the user agent to avoid downloading resources that use formats
  it cannot render.</p>


  <h5>Network states</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-networkState">networkState</span></code></dt>

   <dd>
    <p>Returns the current state of network activity for the element, from the codes in the list
    below.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>As <span data-x="media element">media elements</span> interact with the network, their current
  network activity is represented by the <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-networkState">networkState</code></dfn> attribute. On getting, it must return
  the current network state of the element, which must be one of the following values:</p>

  </div>

  <dl>
   <dt><dfn const for="HTMLMediaElement"><code data-x="dom-media-NETWORK_EMPTY">NETWORK_EMPTY</code></dfn> (numeric value 0)</dt>

   <dd>The element has not yet been initialized. All attributes are in their initial states.</dd>

   <dt><dfn const for="HTMLMediaElement"><code data-x="dom-media-NETWORK_IDLE">NETWORK_IDLE</code></dfn> (numeric value 1)</dt>

   <dd>The element<span w-nodev>'s <span data-x="concept-media-load-algorithm">resource
   selection algorithm</span> is active and</span> has selected a <span data-x="media
   resource">resource</span>, but it is not actually using the network at this time.</dd>

   <dt><dfn const for="HTMLMediaElement"><code data-x="dom-media-NETWORK_LOADING">NETWORK_LOADING</code></dfn> (numeric value 2)</dt>

   <dd>The user agent is actively trying to download data.</dd>

   <dt><dfn const for="HTMLMediaElement"><code data-x="dom-media-NETWORK_NO_SOURCE">NETWORK_NO_SOURCE</code></dfn> (numeric value 3)</dt>

   <dd>The element<span w-nodev>'s <span data-x="concept-media-load-algorithm">resource
   selection algorithm</span> is active, but it</span> has not yet found a <span data-x="media
   resource">resource</span> to use.</dd>
  </dl>

  <div w-nodev>

  <p>The <span data-x="concept-media-load-algorithm">resource selection algorithm</span> defined
  below describes exactly when the <code data-x="dom-media-networkState">networkState</code>
  attribute changes value and what events fire to indicate changes in this state.</p>

  </div>




  <h5>Loading the media resource</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-load">load</span>()</code></dt>

   <dd>
    <p>Causes the element to reset and start selecting and loading a new <span>media resource</span>
    from scratch.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>All <span data-x="media element">media elements</span> have a <dfn>can autoplay flag</dfn>,
  which must begin in the true state, and a <dfn>delaying-the-load-event flag</dfn>, which must
  begin in the false state. While the <span>delaying-the-load-event flag</span> is true, the element
  must <span>delay the load event</span> of its document.</p>

  <p>When the <dfn method for="HTMLMediaElement"><code data-x="dom-media-load">load()</code></dfn>
  method on a <span>media element</span> is invoked, the user agent must run the <span>media element
  load algorithm</span>.</p>

  <p>A <span>media element</span> has an associated boolean <dfn>is currently stalled</dfn>, which
  is initially false.</p>

  <p>The <dfn>media element load algorithm</dfn> consists of the following steps.</p>

  <ol>
   <li><p>Set this element's <span>is currently stalled</span> to false.</p></li>

   <li><p>Abort any already-running instance of the <span
   data-x="concept-media-load-algorithm">resource selection algorithm</span> for this
   element.</p></li>

   <li><p>Let <var>pending tasks</var> be a list of all <span data-x="concept-task">tasks</span>
   from the <span>media element</span>'s <span>media element event task source</span> in one of the
   <span data-x="task queue">task queues</span>.</p></li>

   <li><p>For each task in <var>pending tasks</var> that would <span>resolve pending play
   promises</span> or <span>reject pending play promises</span>, immediately resolve or reject those
   promises in the order the corresponding tasks were queued.</p></li>

   <li>
    <p>Remove each <span data-x="concept-task">task</span> in <var>pending tasks</var> from its
    <span>task queue</span>.</p>

    <p class="note">Basically, pending events and callbacks are discarded and promises in-flight to
    be resolved/rejected are resolved/rejected immediately when the media element starts loading a
    new resource.</p>
   </li>


<!--
    <p>If there are any <span data-x="concept-task">tasks</span> that were <span data-x="queue a
    task">queued</span> by the <span data-x="concept-media-load-algorithm">resource selection
    algorithm</span> (including the algorithms that it itself invokes) for this same <span>media
    element</span> from the <span>DOM manipulation task source</span> in one of the <span
    data-x="task queue">task queues</span>, then remove those tasks.</p>
-->

   <li><p>If the <span>media element</span>'s <code
   data-x="dom-media-networkState">networkState</code> is set to <code
   data-x="dom-media-NETWORK_LOADING">NETWORK_LOADING</code> or <code
   data-x="dom-media-NETWORK_IDLE">NETWORK_IDLE</code>, <span>queue a media element task</span>
   given the <span>media element</span> to <span data-x="concept-event-fire">fire an event</span>
   named <code data-x="event-media-abort">abort</code> at the <span>media element</span>.</p></li>

   <li>
    <p>If the <span>media element</span>'s <code data-x="dom-media-networkState">networkState</code>
    is not set to <code data-x="dom-media-NETWORK_EMPTY">NETWORK_EMPTY</code>, then:</p>

    <ol>
     <li><p><span>Queue a media element task</span> given the <span>media element</span> to <span
     data-x="concept-event-fire">fire an event</span> named <code
     data-x="event-media-emptied">emptied</code> at the <span>media element</span>.</p></li>

     <li><p>If a fetching process is in progress for the <span>media
     element</span>, the user agent should stop it.</p></li>

     <li><p>If the <span>media element</span>'s <span>assigned media provider object</span> is a
     <code>MediaSource</code> object, then <span data-x="Detaching from a media
     element">detach</span> it.</p></li>

     <!--(no point doing this, since we always call the resource selection algorithm immediately
     at the end of this algorithm, and that switches it to NETWORK_NO_SOURCE and sets the flag)
     <li><p>Set the <code data-x="dom-media-networkState">networkState</code> attribute to <code
     data-x="dom-media-NETWORK_EMPTY">NETWORK_EMPTY</code>.</p></li>
     <li><p>Set the element's <span>show poster flag</span> to true.</p></li>
     -->

     <li><p><span>Forget the media element's media-resource-specific tracks</span>.</p></li>

     <li><p>If <code data-x="dom-media-readyState">readyState</code> is not set to <code
     data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</code>, then set it to that state.</p></li>

     <li>
     <p>If the <code data-x="dom-media-paused">paused</code> attribute is false, then:</p>

      <ol>
       <li><p>Set the <code data-x="dom-media-paused">paused</code> attribute to true.</p></li>

       <li><p><span>Take pending play promises</span> and <span>reject pending play promises</span>
       with the result and an <span>"<code>AbortError</code>"</span>
       <code>DOMException</code>.</p></li>
      </ol>
     </li>

     <li><p>If <code data-x="dom-media-seeking">seeking</code> is true, set it to false.</p></li>

     <li>
      <p>Set the <span>current playback position</span> to 0.</p>

      <p>Set the <span>official playback position</span> to 0.</p>

      <p>If this changed the <span>official playback position</span>, then <span>queue a media
      element task</span> given the <span>media element</span> to <span
      data-x="concept-event-fire">fire an event</span> named <code
      data-x="event-media-timeupdate">timeupdate</code> at the <span>media element</span>.</p>
     </li>

     <li><p>Set the <span>timeline offset</span> to Not-a-Number (NaN).</p></li>

     <li>
      <p>Update the <code data-x="dom-media-duration">duration</code> attribute to Not-a-Number
      (NaN).</p>

      <p class="note">The user agent <a href="#durationChange">will not</a> fire a <code
      data-x="event-media-durationchange">durationchange</code> event for this particular change of
      the duration.</p>
     </li>
    </ol>
   </li>

   <li><p>Set the <code data-x="dom-media-playbackRate">playbackRate</code> attribute to the value of
   the <code data-x="dom-media-defaultPlaybackRate">defaultPlaybackRate</code> attribute.</p></li>

   <li><p>Set the <code data-x="dom-media-error">error</code> attribute to null and the
   <span>can autoplay flag</span> to true.</p></li>

   <li><p>Invoke the <span>media element</span>'s <span
   data-x="concept-media-load-algorithm">resource selection algorithm</span>.</li>

   <li>
    <p class="note">Playback of any previously playing <span>media resource</span> for this element
    stops.</p>
   </li>
  </ol>

  <p>The <dfn data-x="concept-media-load-algorithm">resource selection algorithm</dfn> for a
  <span>media element</span> is as follows. This algorithm is always invoked as part of a <span data-x="concept-task">task</span>, but one
  of the first steps in the algorithm is to return and continue running the remaining steps
  <span>in parallel</span>. In addition, this algorithm interacts
  closely with the <span>event loop</span> mechanism; in particular, it has <span data-x="synchronous
  section">synchronous sections</span> (which are triggered as part of the <span>event loop</span>
  algorithm). Steps in such sections are marked with &#x231B;.</p>

  <ol>
   <!-- precondition: networkState == NETWORK_EMPTY or we're coming straight from the algorithm above
                      (where setting it to empty is commented out) -->

   <li><p>Set the element's <code data-x="dom-media-networkState">networkState</code> attribute to
   the <code data-x="dom-media-NETWORK_NO_SOURCE">NETWORK_NO_SOURCE</code> value.</p></li>

   <li><p>Set the element's <span>show poster flag</span> to true.</p></li>

   <li><p>Set the <span>media element</span>'s <span>delaying-the-load-event flag</span> to true
   (this <span data-x="delay the load event">delays the load event</span>).</p></li>

   <li><p><span>Await a stable state</span>, allowing the <span
   data-x="concept-task">task</span> that invoked this algorithm to continue. The <span>synchronous
   section</span> consists of all the remaining steps of this algorithm until the algorithm says the
   <span>synchronous section</span> has ended. (Steps in <span data-x="synchronous
   section">synchronous sections</span> are marked with &#x231B;.)</p></li>

   <li>
    <p>&#x231B; If the <span>media element</span>'s <span>blocked-on-parser</span> flag is false,
    then <span>populate the list of pending text tracks</span>.</p>
   </li>

   <li>
    <p>&#x231B; If the <span>media element</span> has an <span>assigned media provider
    object</span>, then let <var>mode</var> be <i>object</i>.</p>

    <p>&#x231B; Otherwise, if the <span>media element</span> has no <span>assigned media provider
    object</span> but has a <code data-x="attr-media-src">src</code>
    attribute, then let <var>mode</var> be <i>attribute</i>.</p>

    <p>&#x231B; Otherwise, if the <span>media element</span> does not have an <span>assigned media provider
    object</span> and does not have a <code
    data-x="attr-media-src">src</code> attribute, but does have a <code>source</code> element child, then
    let <var>mode</var> be <i>children</i> and let <var>candidate</var>
    be the first such <code>source</code> element child in <span>tree order</span>.</p>

    <p>&#x231B; Otherwise, the <span>media element</span> has no <span>assigned media provider
    object</span> and has neither a <code
    data-x="attr-media-src">src</code> attribute nor a <code>source</code> element child:</p>

    <ol>
     <li><p>&#x231B; Set the <code data-x="dom-media-networkState">networkState</code> to <code
     data-x="dom-media-NETWORK_EMPTY">NETWORK_EMPTY</code>.</p></li>

     <li><p>&#x231B; Set the element's <span>delaying-the-load-event flag</span> to false. This stops <span
     data-x="delay the load event">delaying the load event</span>.</p></li>

     <!-- no need to set the <span>show poster flag</span> to true since we set it up there already -->

     <li><p>End the <span>synchronous section</span> and return.</li>
    </ol>
   </li>

   <li><p>&#x231B; Set the <span>media element</span>'s <code
   data-x="dom-media-networkState">networkState</code> to <code
   data-x="dom-media-NETWORK_LOADING">NETWORK_LOADING</code>.</p></li>

   <li><p>&#x231B; <span>Queue a media element task</span> given the <span>media element</span> to
   <span data-x="concept-event-fire">fire an event</span> named <code
   data-x="event-media-loadstart">loadstart</code> at the <span>media element</span>.</p></li>

   <li>
    <p>Run the appropriate steps from the following list:</p>

    <dl class="switch">

     <dt>If <var>mode</var> is <i>object</i></dt>

     <dd>
      <ol>
       <li><p>&#x231B; Set the <code data-x="dom-media-currentSrc">currentSrc</code> attribute to
       the empty string.</p></li>

       <li><p>End the <span>synchronous section</span>, continuing the remaining steps
       <span>in parallel</span>.</p></li>

       <li><p>Run the <span data-x="concept-media-load-resource">resource fetch algorithm</span>
       with the <span>assigned media provider object</span>. If that algorithm returns without
       aborting <em>this</em> one, then the load failed.</p></li>

       <li><p><i>Failed with media provider</i>: Reaching this step indicates that the media
       resource failed to load. <span>Take pending play promises</span> and <span>queue a media
       element task</span> given the <span>media element</span> to run the <span>dedicated media
       source failure steps</span> with the result.</p></li>

       <li><p>Wait for the <span data-x="concept-task">task</span> queued by the previous step to have
       executed.</p></li>

       <li><p>Return. The element won't attempt to load another resource until this
       algorithm is triggered again.</p></li>
      </ol>
     </dd>

     <dt>If <var>mode</var> is <i>attribute</i></dt>

     <dd>
      <ol>
       <li><p>&#x231B; If the <code data-x="attr-media-src">src</code>
       attribute's value is the empty string, then end the <span>synchronous section</span>, and jump
       down to the <i>failed with attribute</i> step below.</p></li>

       <li><p>&#x231B; Let <var>urlRecord</var> be the result of <span>encoding-parsing a URL</span>
       given the <code data-x="attr-media-src">src</code> attribute's value, relative to the
       <span>media element</span>'s <span>node document</span> when the <code
       data-x="attr-media-src">src</code> attribute was last changed.</p>
       <!-- i.e. changing <base> after src="" has no effect, but this is a terrible way of
            specifying that. We should more eagerly parse instead? -->

       <li><p>&#x231B; If <var>urlRecord</var> is not failure, then set the <code
       data-x="dom-media-currentSrc">currentSrc</code> attribute to the result of applying the <span
       data-x="concept-url-serializer">URL serializer</span> to <var>urlRecord</var>.</p></li>

       <li><p>End the <span>synchronous section</span>, continuing the remaining steps
       <span>in parallel</span>.</p></li>

       <li><p>If <var>urlRecord</var> is not failure, then run the <span
       data-x="concept-media-load-resource">resource fetch algorithm</span> with
       <var>urlRecord</var>. If that algorithm returns without aborting <em>this</em> one, then the
       load failed.</p></li>

       <li><p><i>Failed with attribute</i>: Reaching this step indicates that the media resource
       failed to load or that <var>urlRecord</var> is failure. <span>Take pending play
       promises</span> and <span>queue a media element task</span> given the <span>media
       element</span> to run the <span>dedicated media source failure steps</span> with the
       result.</p></li>

       <li><p>Wait for the <span data-x="concept-task">task</span> queued by the previous step to have
       executed.</p></li>

       <li><p>Return. The element won't attempt to load another resource until this
       algorithm is triggered again.</p></li>
       <!-- it took its ball and went home, sulking. -->
      </ol>
     </dd>

     <dt>Otherwise (<var>mode</var> is <i>children</i>)</dt>

     <dd>
      <ol>
       <li>
        <p>&#x231B; Let <var>pointer</var> be a position defined by two adjacent nodes in the
        <span>media element</span>'s child list, treating the start of the list (before the first
        child in the list, if any) and end of the list (after the last child in the list, if any) as
        nodes in their own right. One node is the node before <var>pointer</var>, and the
        other node is the node after <var>pointer</var>. Initially, let <var>pointer</var> be the position between the <var>candidate</var> node and the
        next node, if there are any, or the end of the list, if it is the last node.</p>

        <p>As nodes are <span data-x="concept-node-insert-ext">inserted</span>, <span
        data-x="concept-node-remove-ext">removed</span>, and <span
        data-x="concept-node-move-ext">moved</span> into the <span>media element</span>,
        <var>pointer</var> must be updated as follows:</p>

        <dl>
         <dt>If a new node is <span data-x="concept-node-insert-ext">inserted</span> or <span
         data-x="concept-node-move-ext">moved</span> between the two nodes that define
         <var>pointer</var></dt>
         <dd>Let <var>pointer</var> be the point between the node before <var>pointer</var> and the
         new node. In other words, insertions at <var>pointer</var> go after
         <var>pointer</var>.</dd>

         <dt>If the node before <var>pointer</var> is removed</dt>
         <dd>Let <var>pointer</var> be the point between the node after <var>pointer</var> and the
         node before the node after <var>pointer</var>. In other words, <var>pointer</var> doesn't
         move relative to the remaining nodes.</dd>

         <dt>If the node after <var>pointer</var> is removed</dt>
         <dd>Let <var>pointer</var> be the point between the node before <var>pointer</var> and the
         node after the node before <var>pointer</var>. Just as with the previous case,
         <var>pointer</var> doesn't move relative to the remaining nodes.</dd>
        </dl>

        <p>Other changes don't affect <var>pointer</var>.</p>
       </li>

       <li><p>&#x231B; <i>Process candidate</i>: If <var>candidate</var> does not have a
       <code data-x="attr-source-src">src</code> attribute, or if its <code
       data-x="attr-source-src">src</code> attribute's value is the empty string, then end the
       <span>synchronous section</span>, and jump down to the <i>failed with elements</i> step
       below.</p></li>

       <li><p>&#x231B; If <var>candidate</var> has a <code data-x="attr-source-media">media</code>
       attribute whose value does not <span data-x="matches the environment">match the
       environment</span>, then end the <span>synchronous section</span>, and jump down to the
       <i>failed with elements</i> step below.</p></li>

       <li><p>&#x231B; Let <var>urlRecord</var> be the result of <span>encoding-parsing a URL</span>
       given <var>candidate</var>'s <code data-x="attr-media-src">src</code> attribute's value,
       relative to <var>candidate</var>'s <span>node document</span> when the <code
       data-x="attr-media-src">src</code> attribute was last changed.</p>
       <!-- i.e. changing <base> after src="" has no effect, but this is a terrible way of
            specifying that. We should more eagerly parse instead? -->

       <li><p>&#x231B; If <var>urlRecord</var> is failure, then end the <span>synchronous
       section</span>, and jump down to the <i>failed with elements</i> step below.</p></li>

       <li><p>&#x231B; If <var>candidate</var> has a <code
       data-x="attr-source-type">type</code> attribute whose value, when parsed as a <span>MIME
       type</span> (including any codecs described by the <code data-x="">codecs</code> parameter, for
       types that define that parameter), represents <span>a type that the user agent knows it cannot
       render</span>, then end the <span>synchronous section</span>, and jump down to the <i>failed with elements</i> step below.</p></li>

       <li><p>&#x231B; Set the <code data-x="dom-media-currentSrc">currentSrc</code> attribute to
       the result of applying the <span data-x="concept-url-serializer">URL serializer</span> to
       <var>urlRecord</var>.</p></li>

       <li><p>End the <span>synchronous section</span>, continuing the remaining steps
       <span>in parallel</span>.</p></li>

       <li><p>Run the <span data-x="concept-media-load-resource">resource fetch algorithm</span>
       with <var>urlRecord</var>. If that algorithm returns without aborting <em>this</em> one,
       then the load failed.</p></li>

       <li><p><i>Failed with elements</i>: <span>Queue a media element task</span> given the
       <span>media element</span> to <span data-x="concept-event-fire">fire an event</span> named
       <code data-x="event-source-error">error</code> at <var>candidate</var>.</p></li>

       <li><p><span>Await a stable state</span>. The <span>synchronous section</span>
       consists of all the remaining steps of this algorithm until the algorithm says the
       <span>synchronous section</span> has ended. (Steps in <span data-x="synchronous
       section">synchronous sections</span> are marked with &#x231B;.)</p></li>

       <li><p>&#x231B; <span>Forget the media element's media-resource-specific
       tracks</span>.</p></li>

       <li><p>&#x231B; <i>Find next candidate</i>: Let <var>candidate</var> be
       null.</p></li>

       <li><p>&#x231B; <i>Search loop</i>: If the node after <var>pointer</var> is
       the end of the list, then jump to the <i>waiting</i> step below.</p></li>

       <li><p>&#x231B; If the node after <var>pointer</var> is a <code>source</code> element,
       let <var>candidate</var> be that element.</p></li>

       <li><p>&#x231B; Advance <var>pointer</var> so that the node before <var>pointer</var> is now the node that was after <var>pointer</var>, and the node
       after <var>pointer</var> is the node after the node that used to be after <var>pointer</var>, if any.</p></li>

       <li><p>&#x231B; If <var>candidate</var> is null, jump back to the <i>search
       loop</i> step. Otherwise, jump back to the <i>process candidate</i> step.</p></li>

       <li><p>&#x231B; <i>Waiting</i>: Set the element's <code
       data-x="dom-media-networkState">networkState</code> attribute to the <code
       data-x="dom-media-NETWORK_NO_SOURCE">NETWORK_NO_SOURCE</code> value.</p></li>

       <li><p>&#x231B; Set the element's <span>show poster flag</span> to true.</p></li>

       <li><p>&#x231B; <span>Queue a media element task</span> given the <span>media element</span>
       to set the element's <span>delaying-the-load-event flag</span> to false. This stops <span
       data-x="delay the load event">delaying the load event</span>.</p></li>

       <li><p>End the <span>synchronous section</span>, continuing the remaining steps
       <span>in parallel</span>.</p></li>

       <li><p>Wait until the node after <var>pointer</var> is a node other than the end of
       the list. (This step might wait forever.)</p></li>

       <li><p><span>Await a stable state</span>. The <span>synchronous section</span>
       consists of all the remaining steps of this algorithm until the algorithm says the
       <span>synchronous section</span> has ended. (Steps in <span data-x="synchronous
       section">synchronous sections</span> are marked with &#x231B;.)</p></li>

       <li><p>&#x231B; Set the element's <span>delaying-the-load-event flag</span> back to true (this
       <span data-x="delay the load event">delays the load event</span> again, in case it hasn't been
       fired yet).</p>

       <li><p>&#x231B; Set the <code data-x="dom-media-networkState">networkState</code> back to <code
       data-x="dom-media-NETWORK_LOADING">NETWORK_LOADING</code>.</p></li>

       <li><p>&#x231B; Jump back to the <i>find next candidate</i> step above.</p></li>
      </ol>
     </dd>
    </dl>

    <p>The <dfn>dedicated media source failure steps</dfn> with a list of promises
    <var>promises</var> are the following steps:</p>

    <ol>
     <li><p>Set the <code data-x="dom-media-error">error</code> attribute to the result of
     <span>creating a <code>MediaError</code></span> with <code
     data-x="dom-MediaError-MEDIA_ERR_SRC_NOT_SUPPORTED">MEDIA_ERR_SRC_NOT_SUPPORTED</code>.</p></li>

     <li><p><span>Forget the media element's media-resource-specific tracks</span>.</p></li>

     <li><p>Set the element's <code data-x="dom-media-networkState">networkState</code> attribute to
     the <code data-x="dom-media-NETWORK_NO_SOURCE">NETWORK_NO_SOURCE</code> value.</p></li>

     <li><p>Set the element's <span>show poster flag</span> to true.</p></li>

     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-media-error">error</code> at the <span>media element</span>.</p></li>

     <li><p><span>Reject pending play promises</span> with <var>promises</var> and a
     <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>Set the element's <span>delaying-the-load-event flag</span> to false. This stops <span
     data-x="delay the load event">delaying the load event</span>.</p></li>
    </ol>
   </li>
  </ol>

  <p>To <dfn>verify a media response</dfn> given a <span data-x="concept-response">response</span>
  <var>response</var>, a <span>media resource</span> <var>resource</var>, and
  "<code data-x="">entire resource</code>" or a
  (number, number or "<code data-x="">until end</code>") tuple <var>byteRange</var>:</p>

  <ol>
   <li><p>If <var>response</var> is a <span>network error</span>, then return false.</p></li>

   <li><p>If <var>byteRange</var> is "<code data-x="">entire resource</code>", then return
   true.</p></li>

   <li><p>Let <var>internalResponse</var> be <var>response</var>'s
   <span>unsafe response</span>.</p></li>

   <li><p>If <var>internalResponse</var>'s <span data-x="concept-response-status">status</span> is
   200, then return true.</p></li>

   <li><p>If <var>internalResponse</var>'s <span data-x="concept-response-status">status</span> is
   not 206, then return false.</p></li>

   <li>
    <p>If the result of <span
    data-x="extract content-range values">extracting content-range values</span> from
    <var>internalResponse</var> is failure, then return false.</p>

    <p class="note">Note that the extracted values are not used, and in particular are not compared
    to <var>byteRange</var>. So this step serves as syntactic validation of the `<code
    data-x="http-content-range">Content-Range</code>` header, but if the `<code
    data-x="http-content-range">Content-Range</code>` values on the response mismatch the `<code
    data-x="http-range">Range</code>` values on the request, that is not considered a failure.</p>
   </li>

   <li><p>Let <var>origin</var> be "<code data-x="">rewritten</code>" if
   <var>internalResponse</var>'s <span data-x="concept-response-url">URL</span> is null; otherwise
   <var>internalResponse</var>'s <span data-x="concept-response-url">URL</span>'s
   <span data-x="concept-url-origin">origin</span>.</p></li>

   <li><p>Let <var>previousOrigin</var> be <var>resource</var>'s
   <span data-x="media-resource-origin">origin</span>.</p></li>

   <li>
    <p>If any of the following are true:</p>

    <ul>
     <li><p><var>previousOrigin</var> is "<code data-x="">none</code>";</p></li>

     <li><p><var>origin</var> and <var>previousOrigin</var> are "<code data-x="">rewritten</code>";
     or</p></li>

     <li><p><var>origin</var> and <var>previousOrigin</var> are <span
     data-x="origin">origins</span>, and <var>origin</var> is <span>same origin</span> with
     <var>previousOrigin</var>,</p></li>
    </ul>

    <p>then set <var>resource</var>'s <span data-x="media-resource-origin">origin</span> to
    <var>origin</var>.</p>

    <p>Otherwise, if <var>response</var> is <span>CORS-cross-origin</span>, then return false.</p>

    <p>Otherwise, set <var>resource</var>'s <span data-x="media-resource-origin">origin</span> to
    "<code data-x="">multiple</code>".</p>

    <p class="note">This ensures that opaque responses with range headers do not leak information
    by being patched together with other responses from different origins.</p>
   </li>

   <li><p>Return true.</p></li>
  </ol>

  <p>The <dfn data-x="concept-media-load-resource" export>resource fetch algorithm</dfn> for a
  <span>media element</span> and a given <span>URL record</span> or <span>media provider
  object</span> is as follows:</p>

  <ol>
   <li><p>Let <var>mode</var> be <i>remote</i>.</p></li>

   <li>
    <p>If the algorithm was invoked with <span>media provider object</span>, then set
    <var>mode</var> to <i>local</i>.</p>

    <p>Otherwise:</p>

    <ol>
     <li><p>Let <var>object</var> be the result of <span data-x="blob-url-obtain-object">obtaining
     a blob object</span> using the <span>URL record</span>'s <span
     data-x="concept-url-blob-entry">blob URL entry</span> and the <span>media element</span>'s
     <span>node document</span>'s <span>relevant settings object</span>.</p></li>

     <li><p>If <var>object</var> is a <span>media provider object</span>, then set <var>mode</var>
     to <i>local</i>.</p></li>
    </ol>
   </li>

   <li><p>If <var>mode</var> is <i>remote</i>, then let the <var>current media resource</var> be the
   resource given by the <span>URL record</span> passed to this algorithm; otherwise, let the
   <var>current media resource</var> be the resource given by the <span>media provider
   object</span>. Either way, the <var>current media resource</var> is now the element's <span>media
   resource</span>.</p></li>

   <li><p>Remove all <span data-x="media-resource-specific text track">media-resource-specific text
   tracks</span> from the <span>media element</span>'s <span>list of pending text tracks</span>, if
   any.</p>


   <li>
    <p>Run the appropriate steps from the following list:</p>

    <dl class="switch">

     <dt>If <var>mode</var> is remote</dt>

     <dd>
      <ol>
       <li>
        <p>Optionally, run the following substeps. This is the expected behavior if the user agent
        intends to not attempt to fetch the resource until the user requests it explicitly (e.g. as
        a way to implement the <code data-x="attr-media-preload">preload</code> attribute's <code
        data-x="attr-media-preload-none">none</code> keyword).</p>

        <ol>
         <li><p>Set the <code data-x="dom-media-networkState">networkState</code> to <code
         data-x="dom-media-NETWORK_IDLE">NETWORK_IDLE</code>.</p></li>

         <li><p><span>Queue a media element task</span> given the <span>media element</span> to
         <span data-x="concept-event-fire">fire an event</span> named <code
         data-x="event-media-suspend">suspend</code> at the element.</p></li>

         <li><p><span>Queue a media element task</span> given the <span>media element</span> to set
         the element's <span>delaying-the-load-event flag</span> to false. This stops <span
         data-x="delay the load event">delaying the load event</span>.</p></li>

         <li><p>Wait for the task to be run.</p></li>

         <li><p>Wait for an <span>implementation-defined</span> event (e.g., the user requesting
         that the media element begin playback).</p></li>

         <li><p>Set the element's <span>delaying-the-load-event flag</span> back to true (this <span
         data-x="delay the load event">delays the load event</span> again, in case it hasn't been fired
         yet).</p>

         <li><p>Set the <code data-x="dom-media-networkState">networkState</code> to <code
         data-x="dom-media-NETWORK_LOADING">NETWORK_LOADING</code>.</p></li>
        </ol>
       </li>

       <li><p>Let <var>destination</var> be "<code data-x="">audio</code>" if the <span>media
       element</span> is an <code>audio</code> element, or "<code data-x="">video</code>"
       otherwise.</p>

       <li><p>Let <var>request</var> be the result of <span
       data-x="create a potential-CORS request">creating a potential-CORS request</span> given
       <var>current media resource</var>'s <span>URL record</span>, <var>destination</var>, and the
       current state of <span>media element</span>'s
       <code data-x="attr-script-crossorigin">crossorigin</code> content attribute.</p></li>

       <li><p>Set <var>request</var>'s <span data-x="concept-request-client">client</span> to the
       <span>media element</span>'s <span>node document</span>'s <span>relevant settings
       object</span>.</p></li>

       <li><p>Set <var>request</var>'s <span data-x="concept-request-initiator-type">initiator
       type</span> to <var>destination</var>.</p></li>

       <li><p>Let <var>byteRange</var>, which is "<code data-x="">entire resource</code>" or a
       (number, number or "<code data-x="">until end</code>") tuple, be the byte range required to satisfy
       missing data in <span>media data</span>. This value is <span>implementation-defined</span>
       and may rely on codec, network conditions or other heuristics. The user-agent may determine
       to fetch the resource in full, in which case <var>byteRange</var> would be
       "<code data-x="">entire resource</code>", to fetch from a byte offset until the end,
       in which case <var>byteRange</var> would be
       (number, "<code data-x="">until end</code>"), or to fetch a range between two byte offsets,
       im which case <var>byteRange</var> would be a (number, number) tuple representing the two
       offsets.</p></li>

       <li>
        <p>If <var>byteRange</var> is not "<code data-x="">entire resource</code>", then:</p>
        <ol>
         <li><p>If <var>byteRange</var>[1] is "<code data-x="">until end</code>" then
         <span>add a range header</span> to <var>request</var> given
         <var>byteRange</var>[0].</p></li>

         <li><p>Otherwise, <span>add a range header</span> to <var>request</var> given
         <var>byteRange</var>[0] and <var>byteRange</var>[1].</p></li>
        </ol>
       </li>

       <li>
        <!--FETCH--><p><span data-x="concept-fetch">Fetch</span> <var>request</var>, with
        <i data-x="processResponse">processResponse</i> set to the following steps given
        <span data-x="concept-response">response</span> <var>response</var>:</p>

        <ol>
         <li><p>Let <var>global</var> be the <span>media element</span>'s
         <span>node document</span>'s <span>relevant global object</span>.</p></li>

         <li><p>Let <var>updateMedia</var> be to <span>queue a media element task</span> given
         the <span>media element</span> to run the first appropriate steps from the
         <span>media data processing steps list</span> below. (A new task is used for
         this so that the work described below occurs relative to the appropriate
         <span>media element event task source</span> rather than using the
         <span>networking task source</span>.)</p>

         <li><p>Let <var>processEndOfMedia</var> be the following step: If the fetching process has
         completed without errors, including decoding the media data, and if all of the data is
         available to the user agent without network access, then, the user agent must move on to
         the <i>final step</i> below. This might never happen, e.g. when streaming an infinite
         resource such as web radio, or if the resource is longer than the user agent's ability to
         cache data.</p></li>

         <li><p>If the result of <span data-x="verify a media response">verifying</span>
         <var>response</var> given the <var>current media resource</var> and <var>byteRange</var>
         is false, then abort these steps.</p></li>

         <li><p>Otherwise, <span data-x="body-incrementally-read">incrementally read</span>
         <var>response</var>'s <span data-x="concept-response-body">body</span> given
         <var>updateMedia</var>, <var>processEndOfMedia</var>, an empty algorithm, and
         <var>global</var>.</p></li>

         <li><p>Update the <span>media data</span> with the contents of <var>response</var>'s
         <span>unsafe response</span> obtained in this fashion. <var>response</var> can be
         <span>CORS-same-origin</span> or <span>CORS-cross-origin</span>; this affects whether
         subtitles referenced in the <span>media data</span> are exposed in the API and, for
         <code>video</code> elements, whether a<code>canvas</code> gets tainted when the video is
         drawn on it.</p></li>
        </ol>

        <p>The <dfn export id="stall-timeout">media element stall timeout</dfn> is an
        <span>implementation-defined</span> length of time, which should be about three seconds.
        When a <span>media element</span> that is actively attempting to obtain <span>media
        data</span> has failed to receive any data for a duration equal to the <span>media element
        stall timeout</span>, the user agent must <span>queue a media element task</span> given the
        <span>media element</span> to:</p>

        <ol>
         <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
         data-x="event-media-stalled">stalled</code> at the element.</p></li>

         <li><p>Set the element's <span>is currently stalled</span> to true.</p></li>
        </ol>

        <p>User agents may allow users to selectively block or slow <span>media data</span> downloads.
        When a <span>media element</span>'s download has been blocked altogether, the user agent must
        act as if it was stalled (as opposed to acting as if the connection was closed). The rate of the
        download may also be throttled automatically by the user agent, e.g. to balance the download
        with other connections sharing the same bandwidth.</p>

        <p id="resourceSuspend">User agents may decide to not download more content at any time,
        e.g. after buffering five minutes of a one hour media resource, while waiting for the user
        to decide whether to play the resource or not, while waiting for user input in an
        interactive resource, or when the user navigates away from the page. When a <span>media
        element</span>'s download has been suspended, the user agent must <span>queue a media
        element task</span> given the <span>media element</span> to set the <code
        data-x="dom-media-networkState">networkState</code> to <code
        data-x="dom-media-NETWORK_IDLE">NETWORK_IDLE</code> and <span
        data-x="concept-event-fire">fire an event</span> named <code
        data-x="event-media-suspend">suspend</code> at the element. If and when downloading of the
        resource resumes, the user agent must <span>queue a media element task</span> given the
        <span>media element</span> to set the <code
        data-x="dom-media-networkState">networkState</code> to <code
        data-x="dom-media-NETWORK_LOADING">NETWORK_LOADING</code>. Between the queuing of these
        tasks, the load is suspended (so <code data-x="event-media-progress">progress</code> events
        don't fire, as described above).</p>

        <p class="note">The <code data-x="attr-media-preload">preload</code> attribute provides a hint
        regarding how much buffering the author thinks is advisable, even in the absence of the <code
        data-x="attr-media-autoplay">autoplay</code> attribute.</p>

        <p>When a user agent decides to completely suspend a download, e.g., if it is waiting until
        the user starts playback before downloading any further content, the user agent must
        <span>queue a media element task</span> given the <span>media element</span> to set the
        element's <span>delaying-the-load-event flag</span> to false. This stops <span data-x="delay
        the load event">delaying the load event</span>.</p>

        <p>Although the above steps give an algorithm for issuing requests, the user agent may use
        other means besides those exact ones, especially in the face of error conditions. For
        example, the user agent may reconnect to the server or switch to a streaming protocol. The
        user agent must only consider the resource erroneous, and proceed into the error branches
        of the above steps, if the user agent has given up trying to fetch the resource.</p>

        <p>To determine the format of the <span>media resource</span>, the user agent must use the
        <span data-x="Content-Type sniffing: video">rules for sniffing audio and video specifically</span>.</p>

        <p>While the load is not suspended (see below), every 350ms (&#xB1;200ms) or for every byte
        received, whichever is <em>least</em> frequent, <span>queue a media element task</span>
        given the <span>media element</span> to:

        <ol>
         <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
         data-x="event-media-progress">progress</code> at the element.</p></li>

         <li><p>Set the element's <span>is currently stalled</span> to false.</p></li>
        </ol>

        <p>While the user agent might still need network access to obtain parts of the <span>media
        resource</span>, the user agent must remain on this step.</p>

        <p class="example">For example, if the user agent has discarded the first half of a video, the
        user agent will remain at this step even once the <span data-x="ended playback">playback has
        ended</span>, because there is always the chance the user will seek back to the start. In fact,
        in this situation, once <span data-x="ended playback">playback has ended</span>, the user agent
        will end up firing a <code data-x="event-media-suspend">suspend</code> event, as described
        earlier.</p>
       </li>
      </ol>
     </dd>

     <dt>Otherwise (<var>mode</var> is <i>local</i>)</dt>

     <dd>
      <p>The resource described by the <var>current media resource</var>, if any, contains the
      <span>media data</span>. It is <span>CORS-same-origin</span>. </p>

      <p>If the <var>current media resource</var> is a raw data stream (e.g. from a
      <code>File</code> object), then to determine the format of the <span>media resource</span>,
      the user agent must use the <span data-x="Content-Type sniffing: video">rules for sniffing
      audio and video specifically</span>. Otherwise, if the data stream is pre-decoded, then the
      format is the format given by the relevant specification.</p>

      <p>Whenever new data for the <var>current media resource</var> becomes available, <span>queue
      a media element task</span> given the <span>media element</span> to run the first appropriate
      steps from the <span>media data processing steps list</span> below.</p>

      <p>When the <var>current media resource</var> is permanently exhausted (e.g. all the bytes of
      a <code>Blob</code> have been processed), if there were no decoding errors, then the user
      agent must move on to the <i>final step</i> below. This might never happen, e.g. if the
      <var>current media resource</var> is a <code>MediaStream</code>.</p>
     </dd>
    </dl>

    <p>The <dfn>media data processing steps list</dfn> is as follows:</p>

    <dl class="switch">

     <dt>If the <span>media data</span> cannot be fetched at all, due to network errors, causing the
     user agent to give up trying to fetch the resource</dt>

     <dt>If the <span>media data</span> can be fetched but is found by inspection to be in an
     unsupported format, or can otherwise not be rendered at all</dt>

     <dd>
      <p>DNS errors, HTTP 4xx and 5xx errors (and equivalents in other protocols), and other fatal
      network errors that occur before the user agent has established whether the <var>current media resource</var> is usable, as well as the file using an unsupported
      container format, or using unsupported codecs for all the data, must cause the user agent to
      execute the following steps:</p>

      <ol>
       <li><p>The user agent should cancel the fetching process.</p></li>

       <li><p>Abort this subalgorithm, returning to the <span
       data-x="concept-media-load-algorithm">resource selection algorithm</span>.</p>
      </ol>
     </dd>


     <dt id="found-another-audio-track">If the <span>media resource</span> is found to have an audio
     track</dt>

     <dd>
      <ol>
       <li><p>Create an <code>AudioTrack</code> object to represent the audio track.</p></li>

       <li><p>Update the <span>media element</span>'s <code
       data-x="dom-media-audioTracks">audioTracks</code> attribute's <code>AudioTrackList</code>
       object with the new <code>AudioTrack</code> object.</p></li>

       <li><p>Let <var>enable</var> be <i>unknown</i>.</p></li>

       <li>
        <p>If either the <span>media resource</span> or the <span>URL</span> of the <var>current
        media resource</var> indicate a particular set of audio tracks to enable, or if the user
        agent has information that would facilitate the selection of specific audio tracks to
        improve the user's experience, then: if this audio track is one of the ones to enable, then
        set <var>enable</var> to <i>true</i>, otherwise, set <var>enable</var>
        to <i>false</i>.</p>

        <p class="example">This could be triggered by <span>media fragment syntax</span>, but it
        could also be triggered e.g. by the user agent selecting a 5.1 surround sound audio track
        over a stereo audio track.</p>
       </li>

       <li><p>If <var>enable</var> is still <i>unknown</i>, then, if the <span>media
       element</span> does not yet have an <span data-x="dom-AudioTrack-enabled">enabled</span>
       audio track, then set <var>enable</var> to <i>true</i>, otherwise, set <var>enable</var> to <i>false</i>.</p></li>

       <li><p>If <var>enable</var> is <i>true</i>, then enable this audio track,
       otherwise, do not enable this audio track.</p></li>

       <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
       data-x="event-media-addtrack">addtrack</code> at this <code>AudioTrackList</code> object,
       using <code>TrackEvent</code>, with the <code data-x="dom-TrackEvent-track">track</code>
       attribute initialized to the new <code>AudioTrack</code> object.</p></li>
      </ol>
     </dd>


     <dt id="found-another-video-track">If the <span>media resource</span> is found to have a video
     track</dt>

     <dd>
      <ol>
       <li><p>Create a <code>VideoTrack</code> object to represent the video track.</p></li>

       <li><p>Update the <span>media element</span>'s <code
       data-x="dom-media-videoTracks">videoTracks</code> attribute's <code>VideoTrackList</code>
       object with the new <code>VideoTrack</code> object.</p></li>

       <li><p>Let <var>enable</var> be <i>unknown</i>.</p></li>

       <li>
        <p>If either the <span>media resource</span> or the <span>URL</span> of the <var>current
        media resource</var> indicate a particular set of video tracks to enable, or if the user
        agent has information that would facilitate the selection of specific video tracks to
        improve the user's experience, then: if this video track is the first such video track, then
        set <var>enable</var> to <i>true</i>, otherwise, set <var>enable</var>
        to <i>false</i>.</p>

        <p class="example">This could again be triggered by <span>media fragment syntax</span>.</p>
       </li>

       <li><p>If <var>enable</var> is still <i>unknown</i>, then, if the <span>media
       element</span> does not yet have a <span data-x="dom-VideoTrack-selected">selected</span>
       video track, then set <var>enable</var> to <i>true</i>, otherwise, set <var>enable</var> to <i>false</i>.</p></li>

       <li><p>If <var>enable</var> is <i>true</i>, then select this track and unselect any
       previously selected video tracks, otherwise, do not select this video track. If other tracks
       are unselected, then <a href="#toggle-video-track">a <code
       data-x="event-media-change">change</code> event will be fired</a>.</p></li>

       <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
       data-x="event-media-addtrack">addtrack</code> at this <code>VideoTrackList</code> object,
       using <code>TrackEvent</code>, with the <code data-x="dom-TrackEvent-track">track</code>
       attribute initialized to the new <code>VideoTrack</code> object.</p></li>
      </ol>
     </dd>


     <dt id="getting-media-metadata">Once enough of the <span>media data</span> has been fetched to
     determine the duration of the <span>media resource</span>, its dimensions, and other
     metadata</dt>

     <dd>
      <p>This indicates that the resource is usable. The user agent must follow these substeps:</p>

      <ol>
       <li>
        <p><span>Establish the media timeline</span> for the purposes of the <span>current playback
        position</span> and the <span>earliest possible position</span>, based on the <span>media data</span>.</p>
       </li>

       <li>
        <p>Update the <span>timeline offset</span> to the date and time that corresponds to the zero
        time in the <span>media timeline</span> established in the previous step, if any. If no
        explicit time and date is given by the <span>media resource</span>, the <span>timeline
        offset</span> must be set to Not-a-Number (NaN).</p>
       </li>

       <li><p>Set the <span>current playback position</span> and the <span>official playback
       position</span> to the <span>earliest possible position</span>.</p></li>

       <li>
        <p>Update the <code data-x="dom-media-duration">duration</code> attribute with the time of
        the last frame of the resource, if known, on the <span>media timeline</span> established
        above. If it is not known (e.g. a stream that is in principle infinite), update the <code
        data-x="dom-media-duration">duration</code> attribute to the value positive Infinity.</p>

        <p class="note">The user agent <a href="#durationChange">will</a> <span>queue a media
        element task</span> given the <span>media element</span> to <span
        data-x="concept-event-fire">fire an event</span> named <code
        data-x="event-media-durationchange">durationchange</code> at the element at this point.</p>
       </li>

       <li>
        <p>For <code>video</code> elements, set the <code
        data-x="dom-video-videoWidth">videoWidth</code> and <code
        data-x="dom-video-videoHeight">videoHeight</code> attributes, and <span>queue a media
        element task</span> given the <span>media element</span> to <span
        data-x="concept-event-fire">fire an event</span> named <code
        data-x="event-media-resize">resize</code> at the <span>media element</span>.</p>

        <p class="note">Further <code data-x="event-media-resize">resize</code> events will be fired
        if the dimensions subsequently change.</p>
       </li>

       <li>
        <p>Set the <code data-x="dom-media-readyState">readyState</code> attribute to <code
        data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code>.</p>

        <p class="note">A <code data-x="event-media-loadedmetadata">loadedmetadata</code> DOM event
        <a href="#fire-loadedmetadata">will be fired</a> as part of setting the <code
        data-x="dom-media-readyState">readyState</code> attribute to a new value.</p>
       </li>

       <li><p>Let <var>jumped</var> be false.</p></li>

       <li><p>If the <span>media element</span>'s <span>default playback start position</span> is
       greater than zero, then <span data-x="dom-media-seek">seek</span> to that time, and let <var>jumped</var> be true.</p></li>

       <li><p>Let the <span>media element</span>'s <span>default playback
       start position</span> be zero.</p></li>

       <li><p>Let the <var>initial playback position</var> be zero.</p></li>

       <li>
        <p>If either the <span>media resource</span> or the <span>URL</span> of the <var>current
        media resource</var> indicate a particular start time, then set the <var>initial playback
        position</var> to that time and, if <var>jumped</var> is still false, <span
        data-x="dom-media-seek">seek</span> to that time.</p>

        <p class="example">For example, with media formats that support <span>media fragment
        syntax</span>, the <span data-x="concept-url-fragment">fragment</span> can be used to
        indicate a start position.</p>
       </li>

       <li><p>If there is no <span data-x="dom-AudioTrack-enabled">enabled</span> audio track, then
       enable an audio track. This <a href="#toggle-audio-track">will cause a <code
       data-x="event-media-change">change</code> event to be fired</a>.</p></li>

       <li><p>If there is no <span data-x="dom-VideoTrack-selected">selected</span> video track,
       then select a video track. This <a href="#toggle-video-track">will cause a <code
       data-x="event-media-change">change</code> event to be fired</a>.</p></li>
      </ol>

      <p>Once the <code data-x="dom-media-readyState">readyState</code> attribute reaches <code
      data-x="dom-media-HAVE_CURRENT_DATA">HAVE_CURRENT_DATA</code>, <a href="#fire-loadeddata">after
      the <code data-x="event-media-loadeddata">loadeddata</code> event has been fired</a>, set the
      element's <span>delaying-the-load-event flag</span> to false. This stops <span data-x="delay
      the load event">delaying the load event</span>.</p>

      <p class="note">A user agent that is attempting to reduce network usage while still fetching
      the metadata for each <span>media resource</span> would also stop buffering at this point,
      following <a href="#resourceSuspend">the rules described previously</a>, which involve the
      <code data-x="dom-media-networkState">networkState</code> attribute switching to the <code
      data-x="dom-media-NETWORK_IDLE">NETWORK_IDLE</code> value and a <code
      data-x="event-media-suspend">suspend</code> event firing.</p>

      <p class="note">The user agent is <em>required</em> to determine the duration of the
      <span>media resource</span> and go through this step before playing.</p> <!-- actually defined
      in the 'duration' section -->
     </dd>


     <dt>Once the entire <span>media resource</span> has been fetched (but potentially before any of
     it has been decoded)</dt>

     <dd>
      <p><span data-x="concept-event-fire">Fire an event</span> named <code
      data-x="event-media-progress">progress</code> at the <span>media element</span>.</p>

      <p>Set the <code data-x="dom-media-networkState">networkState</code> to <code
      data-x="dom-media-NETWORK_IDLE">NETWORK_IDLE</code> and <span
      data-x="concept-event-fire">fire an event</span> named
      <code data-x="event-media-suspend">suspend</code> at the <span>media element</span>.</p>

      <p>If the user agent ever discards any <span>media data</span> and then needs to resume the
      network activity to obtain it again, then it must <span>queue a media element task</span>
      given the <span>media element</span> to set the <code
      data-x="dom-media-networkState">networkState</code> to <code
      data-x="dom-media-NETWORK_LOADING">NETWORK_LOADING</code>.</p>

      <p class="note">If the user agent can keep the <span>media resource</span> loaded, then the
      algorithm will continue to its <i>final step</i> below, which aborts the algorithm.</p>
     </dd>


     <dt>If the connection is interrupted after some <span>media data</span> has been received,
     causing the user agent to give up trying to fetch the resource</dt>

     <dd>
      <p>Fatal network errors that occur after the user agent has established whether the <var>current media resource</var> is usable (i.e. once the <span>media element</span>'s
      <code data-x="dom-media-readyState">readyState</code> attribute is no longer <code
      data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</code>) must cause the user agent to execute the
      following steps:</p>

      <ol>
       <li><p>The user agent should cancel the fetching process.</p></li>

       <li><p>Set the <code data-x="dom-media-error">error</code> attribute to the result of
       <span>creating a <code>MediaError</code></span> with <code
       data-x="dom-MediaError-MEDIA_ERR_NETWORK">MEDIA_ERR_NETWORK</code>.</p></li>

       <li><p>Set the element's <code data-x="dom-media-networkState">networkState</code> attribute
       to the <code data-x="dom-media-NETWORK_IDLE">NETWORK_IDLE</code> value.</p></li>

       <li><p>Set the element's <span>delaying-the-load-event flag</span> to false. This stops <span
       data-x="delay the load event">delaying the load event</span>.</p></li>

       <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
       data-x="event-media-error">error</code> at the <span>media element</span>.</p></li>

       <li><p>Abort the overall <span data-x="concept-media-load-algorithm">resource selection
       algorithm</span>.</p></li>
      </ol>
     </dd>


     <dt id="fatal-decode-error">If the <span>media data</span> is corrupted</dt>

     <dd>
      <p>Fatal errors in decoding the <span>media data</span> that occur after the user agent has
      established whether the <var>current media resource</var> is usable (i.e. once the <span>media element</span>'s
      <code data-x="dom-media-readyState">readyState</code> attribute is no longer <code
      data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</code>) must cause the
      user agent to execute the following steps:</p>

      <ol>
       <li><p>The user agent should cancel the fetching process.</p></li>

       <li><p>Set the <code data-x="dom-media-error">error</code> attribute to the result of
       <span>creating a <code>MediaError</code></span> with <code
       data-x="dom-MediaError-MEDIA_ERR_DECODE">MEDIA_ERR_DECODE</code>.</p></li>

       <li><p>Set the element's <code data-x="dom-media-networkState">networkState</code> attribute
       to the <code data-x="dom-media-NETWORK_IDLE">NETWORK_IDLE</code> value.</p></li>

       <li><p>Set the element's <span>delaying-the-load-event flag</span> to false. This stops <span
       data-x="delay the load event">delaying the load event</span>.</p></li>

       <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
       data-x="event-media-error">error</code> at the <span>media element</span>.</p></li>

       <li><p>Abort the overall <span data-x="concept-media-load-algorithm">resource selection
       algorithm</span>.</p></li>
      </ol>
     </dd>


     <dt>If the <span>media data</span> fetching process is aborted by the user</dt>

     <dd>
      <p>The fetching process is aborted by the user, e.g. because the user
      pressed a "stop" button, the user agent must execute the following steps. These steps are not
      followed if the <code data-x="dom-media-load">load()</code> method itself is invoked while
      these steps are running, as the steps above handle that particular kind of abort.</p>

      <ol>
       <li><p>The user agent should cancel the fetching process.</p></li>

       <li><p>Set the <code data-x="dom-media-error">error</code> attribute to the result of
       <span>creating a <code>MediaError</code></span> with <code
       data-x="dom-MediaError-MEDIA_ERR_ABORTED">MEDIA_ERR_ABORTED</code>.</p></li>

       <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
       data-x="event-media-abort">abort</code> at the <span>media element</span>.</p></li>

       <li>
        <p>If the <span>media element</span>'s <code data-x="dom-media-readyState">readyState</code>
        attribute has a value equal to <code data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</code>, set
        the element's <code data-x="dom-media-networkState">networkState</code> attribute to the
        <code data-x="dom-media-NETWORK_EMPTY">NETWORK_EMPTY</code> value, set the element's
        <span>show poster flag</span> to true, and <span data-x="concept-event-fire">fire an
        event</span> named <code data-x="event-media-emptied">emptied</code> at the element.</p>

        <p>Otherwise, set the element's <code data-x="dom-media-networkState">networkState</code>
        attribute to the <code data-x="dom-media-NETWORK_IDLE">NETWORK_IDLE</code> value.</p>
       </li>

       <li><p>Set the element's <span>delaying-the-load-event flag</span> to false. This stops <span
       data-x="delay the load event">delaying the load event</span>.</p></li>

       <li><p>Abort the overall <span data-x="concept-media-load-algorithm">resource selection
       algorithm</span>.</p></li>
      </ol>
     </dd>


     <dt id="non-fatal-media-error">If the <span>media data</span> can be fetched but has non-fatal
     errors or uses, in part, codecs that are unsupported, preventing the user agent from rendering
     the content completely correctly but not preventing playback altogether</dt>

     <dd>
      <p>The server returning data that is partially usable but cannot be optimally rendered must
      cause the user agent to render just the bits it can handle, and ignore the rest.</p>

      <!-- v2: fire a 'warning' event and set the 'error' flag to 'MEDIA_ERR_SUBOPTIMAL' or something -->
     </dd>


     <dt id="found-a-media-resource-specific-timed-track">If the <span>media resource</span> is
     found to declare a <span>media-resource-specific text track</span> that the user agent
     supports</dt>

     <dd>
      <p>If the <span>media data</span> is <span>CORS-same-origin</span>, run the <span>steps to
      expose a media-resource-specific text track</span> with the relevant data.</p>

      <p class="note">Cross-origin videos do not expose their subtitles, since that would allow
      attacks such as hostile sites reading subtitles from confidential videos on a user's
      intranet.</p>
     </dd>
    </dl>
   </li>

   <!-- this step is mentioned above, search for "final step" -->
   <li><p><i>Final step</i>: If the user agent ever reaches this step (which can only happen if the entire resource
   gets loaded and kept available): abort the overall <span
   data-x="concept-media-load-algorithm">resource selection algorithm</span>.</p></li>
  </ol>

  <p>When a <span>media element</span> is to <dfn>forget the media element's media-resource-specific
  tracks</dfn>, the user agent must remove from the <span>media element</span>'s <span>list of text
  tracks</span> all the <span data-x="media-resource-specific text track">media-resource-specific
  text tracks</span>, then empty the <span>media element</span>'s <code
  data-x="dom-media-audioTracks">audioTracks</code> attribute's <code>AudioTrackList</code> object,
  then empty the <span>media element</span>'s <code data-x="dom-media-videoTracks">videoTracks</code>
  attribute's <code>VideoTrackList</code> object. No events (in particular, no <code
  data-x="event-media-removetrack">removetrack</code> events) are fired as part of this; the <code
  data-x="event-media-error">error</code> and <code data-x="event-media-emptied">emptied</code>
  events, fired by the algorithms that invoke this one, can be used instead.</p>

  </div>

  <hr>

  <p>The <dfn element-attr for="audio,video"><code data-x="attr-media-preload">preload</code></dfn>
  attribute is an <span>enumerated attribute</span> with the following keywords and
  states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="audio/preload,video/preload"><code
     data-x="attr-media-preload-auto">auto</code></dfn>
     <td rowspan=2><dfn data-x="attr-media-preload-auto-state">Automatic</dfn>
     <td rowspan=2>Hints to the user agent that the user agent can put the user's needs first without risk to the server, up to and including optimistically downloading the entire resource.
    <tr>
     <td>(the empty string)
    <tr>
     <td><dfn attr-value for="audio/preload,video/preload"><code
     data-x="attr-media-preload-none">none</code></dfn>
     <td><dfn data-x="attr-media-preload-none-state">None</dfn>
     <td>Hints to the user agent that either the author does not expect the user to need the media resource, or that the server wants to minimize unnecessary traffic.
         This state does not provide a hint regarding how aggressively to actually download the media resource if buffering starts anyway (e.g. once the user hits "play").
    <tr>
     <td><dfn attr-value for="audio/preload,video/preload"><code
     data-x="attr-media-preload-metadata">metadata</code></dfn>
     <td><dfn data-x="attr-media-preload-metadata-state">Metadata</dfn>
     <td>Hints to the user agent that the author does not expect the user to need the media resource, but that fetching the resource metadata (dimensions, track list, duration, etc.), and maybe even the first few frames, is reasonable. If the user agent precisely fetches no more than the metadata, then the <span>media element</span> will end up with its <code data-x="dom-media-readyState">readyState</code> attribute set to <code data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code>; typically though, some frames will be obtained as well and it will probably be <code data-x="dom-media-HAVE_CURRENT_DATA">HAVE_CURRENT_DATA</code> or <code data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code>.
         When the media resource is playing, hints to the user agent that bandwidth is to be considered scarce, e.g. suggesting throttling the download so that the media data is obtained at the slowest possible rate that still maintains consistent playback.
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both
  <span>implementation-defined</span>, though the <span
  data-x="attr-media-preload-metadata-state">Metadata</span> state is suggested as a compromise
  between reducing server load and providing an optimal user experience.</p>

  <p>The attribute can be changed even once the <span>media resource</span> is being buffered or
  played; the descriptions in the table above are to be interpreted with that in mind.</p>

  <p class="note">Authors might switch the attribute from "<code
  data-x="attr-media-preload-none">none</code>" or "<code
  data-x="attr-media-preload-metadata">metadata</code>" to "<code
  data-x="attr-media-preload-auto">auto</code>" dynamically once the user begins playback. For
  example, on a page with many videos this might be used to indicate that the many videos are not to
  be downloaded unless requested, but that once one <em>is</em> requested it is to be downloaded
  aggressively.</p>

  <div w-nodev>

  <p>The <code data-x="attr-media-preload">preload</code> attribute is intended to provide a hint to
  the user agent about what the author thinks will lead to the best user experience. The attribute
  may be ignored altogether, for example based on explicit user preferences or based on the
  available connectivity.</p>

  <p>The <dfn attribute for="HTMLMediaElement"><code data-x="dom-media-preload">preload</code></dfn>
  IDL attribute must <span>reflect</span> the content attribute of the same name, <span>limited to
  only known values</span>.</p>

  </div>

  <p class="note">The <code data-x="attr-media-autoplay">autoplay</code> attribute can override the
  <code data-x="attr-media-preload">preload</code> attribute (since if the media plays, it naturally
  has to buffer first, regardless of the hint given by the <code
  data-x="attr-media-preload">preload</code> attribute). Including both is not an error, however.</p>

  <hr>

<!--v3BUF (when readding this, also add a domintro block)
  <p>The <dfn><code data-x="dom-media-bufferingRate">bufferingRate</code></dfn> attribute must return
  the average number of bits received per second for the current download over the past few seconds.
  If there is no download in progress, the attribute must return 0.</p>

  <p>The <dfn><code data-x="dom-media-bufferingThrottled">bufferingThrottled</code></dfn> attribute
  must return true if the user agent is intentionally throttling the bandwidth used by the download
  (including when throttling to zero to pause the download altogether), and false otherwise.</p>

  <hr>
-->

  <dl class="domintro">
   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-buffered">buffered</span></code></dt>

   <dd>
    <p>Returns a <code>TimeRanges</code> object that represents the ranges of the <span>media
    resource</span> that the user agent has buffered.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-buffered">buffered</code></dfn> attribute must return a new static
  <span>normalized <code>TimeRanges</code> object</span> that represents the ranges of the
  <span>media resource</span>, if any, that the user agent has buffered, at the time the attribute
  is evaluated. Users agents must accurately determine the ranges available, even for media streams
  where this can only be determined by tedious inspection.</p>

  <p class="note">Typically this will be a single range anchored at the zero point, but if, e.g. the
  user agent uses HTTP range requests in response to seeking, then there could be multiple
  ranges.</p>

  <p>User agents may discard previously buffered data.</p>

  <p class="note">Thus, a time position included within a range of the objects return by the <code
  data-x="dom-media-buffered">buffered</code> attribute at one time can end up being not included in
  the range(s) of objects returned by the same attribute at later times.</p>

  <p class="warning">Returning a new object each time is a bad pattern for attribute getters and is
  only enshrined here as it would be costly to change it. It is not to be copied to new APIs.</p>

  </div>



  <h5>Offsets into the media resource</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-duration">duration</span></code></dt>

   <dd>
    <p>Returns the length of the <span>media resource</span>, in seconds, assuming that the start of
    the <span>media resource</span> is at time zero.</p>

    <p>Returns NaN if the duration isn't available.</p>

    <p>Returns Infinity for unbounded streams.</p>
   </dd>

   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-currentTime">currentTime</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the <span>official playback position</span>, in seconds.</p>

    <p>Can be set, to seek to the given time.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>A <span>media resource</span> has a <dfn>media timeline</dfn> that maps times (in seconds) to
  positions in the <span>media resource</span>. The origin of a timeline is its earliest defined
  position. The duration of a timeline is its last defined position.</p>

  <p><dfn id="defineTimeline" data-x="establish the media timeline">Establishing the media
  timeline</dfn>: if the <span>media resource</span> somehow specifies an explicit timeline whose
  origin is not negative (i.e. gives each frame a specific time offset and gives the first frame a
  zero or positive offset), then the <span>media timeline</span> should be that timeline. (Whether
  the <span>media resource</span> can specify a timeline or not depends on the <span data-x="media
  resource">media resource's</span> format.) If the <span>media resource</span> specifies an
  explicit start time <em>and date</em>, then that time and date should be considered the zero point
  in the <span>media timeline</span>; the <span>timeline offset</span> will be the time and date,
  exposed using the <code data-x="dom-media-getStartDate">getStartDate()</code> method.</p>

  <p>If the <span>media resource</span> has a discontinuous timeline, the user agent must extend the
  timeline used at the start of the resource across the entire resource, so that the <span>media
  timeline</span> of the <span>media resource</span> increases linearly starting from the
  <span>earliest possible position</span> (as defined below), even if the underlying <span>media
  data</span> has out-of-order or even overlapping time codes.</p>

  <p class="example">For example, if two clips have been concatenated into one video file, but the
  video format exposes the original times for the two clips, the video data might expose a timeline
  that goes, say, 00:15..00:29 and then 00:05..00:38. However, the user agent would not expose those
  times; it would instead expose the times as 00:15..00:29 and 00:29..01:02, as a single video.</p>

  <p>
  <!--INSERT TRACKING-->
  In the rare case of a <span>media resource</span> that does not have an explicit timeline, the
  zero time on the <span>media timeline</span> should correspond to the first frame of the
  <span>media resource</span>. In the even rarer case of a <span>media resource</span> with no
  explicit timings of any kind, not even frame durations, the user agent must itself determine the
  time for each frame in an <span>implementation-defined</span> manner.</p>

  <p class="note">An example of a file format with no explicit timeline but with explicit frame
  durations is the Animated GIF format. An example of a file format with no explicit timings at all
  is the JPEG-push format (<code>multipart/x-mixed-replace</code> with JPEG frames, often
  used as the format for MJPEG streams).</p>

  <p>If, in the case of a resource with no timing information, the user agent will nonetheless be
  able to seek to an earlier point than the first frame originally provided by the server, then the
  zero time should correspond to the earliest seekable time of the <span>media resource</span>;
  otherwise, it should correspond to the first frame received from the server (the point in the
  <span>media resource</span> at which the user agent began receiving the stream).</p>

  <p class="note">At the time of writing, there is no known format that lacks explicit frame time
  offsets yet still supports seeking to a frame before the first frame sent by the server.</p>

  <div class="example">

   <p>Consider a stream from a TV broadcaster, which begins streaming on a sunny Friday afternoon in
   October, and always sends connecting user agents the media data on the same media timeline, with
   its zero time set to the start of this stream. Months later, user agents connecting to this
   stream will find that the first frame they receive has a time with millions of seconds. The <code
   data-x="dom-media-getStartDate">getStartDate()</code> method would always return the date that the
   broadcast started; this would allow controllers to display real times in their scrubber (e.g.
   "2:30pm") rather than a time relative to when the broadcast began ("8 months, 4 hours, 12
   minutes, and 23 seconds").</p>

   <p>Consider a stream that carries a video with several concatenated fragments, broadcast by a
   server that does not allow user agents to request specific times but instead just streams the
   video data in a predetermined order, with the first frame delivered always being identified as
   the frame with time zero. If a user agent connects to this stream and receives fragments defined
   as covering timestamps 2010-03-20 23:15:00 UTC to 2010-03-21 00:05:00 UTC and 2010-02-12 14:25:00
   UTC to 2010-02-12 14:35:00 UTC, it would expose this with a <span>media timeline</span> starting
   at 0s and extending to 3,600s (one hour). Assuming the streaming server disconnected at the end
   of the second clip, the <code data-x="dom-media-duration">duration</code> attribute would then
   return 3,600. The <code data-x="dom-media-getStartDate">getStartDate()</code> method would return a
   <code>Date</code> object with a time corresponding to 2010-03-20 23:15:00 UTC. However, if a
   different user agent connected five minutes later, <em>it</em> would (presumably) receive
   fragments covering timestamps 2010-03-20 23:20:00 UTC to 2010-03-21 00:05:00 UTC and 2010-02-12
   14:25:00 UTC to 2010-02-12 14:35:00 UTC, and would expose this with a <span>media timeline</span>
   starting at 0s and extending to 3,300s (fifty five minutes). In this case, the <code
   data-x="dom-media-getStartDate">getStartDate()</code> method would return a <code>Date</code> object
   with a time corresponding to 2010-03-20 23:20:00 UTC.</p>

   <p>In both of these examples, the <code data-x="dom-media-seekable">seekable</code> attribute
   would give the ranges that the controller would want to actually display in its UI; typically, if
   the servers don't support seeking to arbitrary times, this would be the range of time from the
   moment the user agent connected to the stream up to the latest frame that the user agent has
   obtained; however, if the user agent starts discarding earlier information, the actual range
   might be shorter.</p>

  </div>

  <p>In any case, the user agent must ensure that the <span>earliest possible position</span> (as
  defined below) using the established <span>media timeline</span>, is greater than or equal to
  zero.</p>

  <p>The <span>media timeline</span> also has an associated clock. Which clock is used is user-agent
  defined, and may be <span>media resource</span>-dependent, but it should approximate the user's
  wall clock.</p>

  <p><span data-x="media element">Media elements</span> have a <dfn>current playback position</dfn>,
  which must initially (i.e. in the absence of <span>media data</span>) be zero seconds. The
  <span>current playback position</span> is a time on the <span>media timeline</span>.</p>

  <p><span data-x="media element">Media elements</span> also have an <dfn>official playback
  position</dfn>, which must initially be set to zero seconds. The <span>official playback
  position</span> is an approximation of the <span>current playback position</span> that is kept
  stable while scripts are running.</p>

  <p><span data-x="media element">Media elements</span> also have a <dfn>default playback start
  position</dfn>, which must initially be set to zero seconds. This time is used to allow the
  element to be seeked even before the media is loaded.</p>

  <p>Each <span>media element</span> has a <dfn>show poster flag</dfn>. When a <span>media
  element</span> is created, this flag must be set to true. This flag is used to control when the
  user agent is to show a poster frame for a <code>video</code> element instead of showing the video
  contents.</p>

  <p>The <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-currentTime">currentTime</code></dfn> attribute must, on getting, return the
  <span>media element</span>'s <span>default playback start position</span>, unless that is zero, in
  which case it must return the element's <span>official playback position</span>. The returned
  value must be expressed in seconds. On setting, if the <span>media element</span>'s <code
  data-x="dom-media-readyState">readyState</code> is <code
  data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</code>, then it must set the <span>media
  element</span>'s <span>default playback start position</span> to the new value; otherwise, it must
  set the <span>official playback position</span> to the new value and then <span
  data-x="dom-media-seek">seek</span> to the new value. The new value must be interpreted as being
  in seconds.</p>

  <p>If the <span>media resource</span> is a streaming resource, then the user agent might be unable
  to obtain certain parts of the resource after it has expired from its buffer. Similarly, some
  <span data-x="media resource">media resources</span> might have a <span>media timeline</span> that
  doesn't start at zero. The <dfn>earliest possible position</dfn> is the earliest position in the
  stream or resource that the user agent can ever obtain again. It is also a time on the <span>media
  timeline</span>.</p>

  <p class="note">The <span>earliest possible position</span> is not explicitly exposed in the API;
  it corresponds to the start time of the first range in the <code
  data-x="dom-media-seekable">seekable</code> attribute's <code>TimeRanges</code> object, if any, or
  the <span>current playback position</span> otherwise.</p>

  <p>When the <span>earliest possible position</span> changes, then: if the <span>current playback
  position</span> is before the <span>earliest possible position</span>, the user agent must <span
  data-x="dom-media-seek">seek</span> to the <span>earliest possible position</span>; otherwise, if
  the user agent has not fired a <code data-x="event-media-timeupdate">timeupdate</code> event at
  the element in the past 15 to 250ms and is not still running event handlers for such an event,
  then the user agent must <span>queue a media element task</span> given the <span>media
  element</span> to <span data-x="concept-event-fire">fire an event</span> named <code
  data-x="event-media-timeupdate">timeupdate</code> at the element.</p>

  <p class="note">Because of the above requirement and the requirement in the <span
  data-x="concept-media-load-resource">resource fetch algorithm</span> that kicks in <a
  href="#getting-media-metadata">when the metadata of the clip becomes known</a>, the <span>current
  playback position</span> can never be less than the <span>earliest possible position</span>.</p>

  <!-- see https://www.w3.org/Bugs/Public/show_bug.cgi?id=14492 -->
  <!-- basically this is to handle very-long-running streams that use different video and audio
       tracks per TV show -->

  <p>If at any time the user agent learns that an audio or video track has ended and all <span>media
  data</span> relating to that track corresponds to parts of the <span>media timeline</span> that
  are <em>before</em> the <span>earliest possible position</span>, the user agent may <span>queue a
  media element task</span> given the <span>media element</span> to run these steps:</p>

  <ol>
   <li><p>Remove the track from the <code data-x="dom-media-audioTracks">audioTracks</code>
   attribute's <code>AudioTrackList</code> object or the <code
   data-x="dom-media-videoTracks">videoTracks</code> attribute's <code>VideoTrackList</code> object
   as appropriate.</p></li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
   data-x="event-media-removetrack">removetrack</code> at the <span>media element</span>'s
   aforementioned <code>AudioTrackList</code> or <code>VideoTrackList</code> object, using
   <code>TrackEvent</code>, with the <code data-x="dom-TrackEvent-track">track</code> attribute
   initialized to the <code>AudioTrack</code> or <code>VideoTrack</code> object representing the
   track.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-duration">duration</code></dfn> attribute must return the time of the end of the
  <span>media resource</span>, in seconds, on the <span>media timeline</span>. If no <span>media
  data</span> is available, then the attributes must return the Not-a-Number (NaN) value. If the
  <span>media resource</span> is not known to be bounded (e.g. streaming radio, or a live event with
  no announced end time), then the attribute must return the positive Infinity value.</p>

  <p>The user agent must determine the duration of the <span>media resource</span> before playing
  any part of the <span>media data</span> and before setting <code
  data-x="dom-media-readyState">readyState</code> to a value greater than or equal to <code
  data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code>, even if doing so requires fetching multiple
  parts of the resource.</p>

  <p id="durationChange">When the length of the <span>media resource</span> changes to a known value
  (e.g. from being unknown to known, or from a previously established length to a new length) the
  user agent must <span>queue a media element task</span> given the <span>media element</span> to
  <span data-x="concept-event-fire">fire an event</span> named <code
  data-x="event-media-durationchange">durationchange</code> at the <span>media element</span>. (The
  event is not fired when the duration is reset as part of loading a new media resource.) If the
  duration is changed such that the <span>current playback position</span> ends up being greater
  than the time of the end of the <span>media resource</span>, then the user agent must also <span
  data-x="dom-media-seek">seek</span> to the time of the end of the <span>media resource</span>.</p>

  <p class="example">If an "infinite" stream ends for some reason, then the duration would change
  from positive Infinity to the time of the last frame or sample in the stream, and the <code
  data-x="event-media-durationchange">durationchange</code> event would be fired. Similarly, if the
  user agent initially estimated the <span>media resource</span>'s duration instead of determining
  it precisely, and later revises the estimate based on new information, then the duration would
  change and the <code data-x="event-media-durationchange">durationchange</code> event would be
  fired.</p>

  <p>Some video files also have an explicit date and time corresponding to the zero time in the
  <span>media timeline</span>, known as the <dfn>timeline offset</dfn>. Initially, the
  <span>timeline offset</span> must be set to Not-a-Number (NaN).</p>

  <p>The <dfn method for="HTMLMediaElement"><code
  data-x="dom-media-getStartDate">getStartDate()</code></dfn> method must return <span
  data-x="create a Date object">a new <code>Date</code> object</span> representing the current
  <span>timeline offset</span>.</p>

  </div>

  <hr>

  <p>The <dfn element-attr for="audio,video"><code data-x="attr-media-loop">loop</code></dfn>
  attribute is a <span>boolean attribute</span> that, if specified, indicates that the <span>media
  element</span> is to seek back to the start of the <span>media resource</span> upon reaching the
  end.</p>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLMediaElement"><code data-x="dom-media-loop">loop</code></dfn> IDL
  attribute must <span>reflect</span> the content attribute of the same name.</p>

  </div>



  <h5>Ready states</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-readyState">readyState</span></code></dt>

   <dd>
    <p>Returns a value that expresses the current state of the element with respect to rendering the
    <span>current playback position</span>, from the codes in the list below.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p><span data-x="media element">Media elements</span> have a <i>ready state</i>, which describes to
  what degree they are ready to be rendered at the <span>current playback position</span>. The
  possible values are as follows; the ready state of a media element at any particular time is the
  greatest value describing the state of the element:</p>

  </div>

  <dl>
   <dt><dfn const for="HTMLMediaElement"><code data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</code></dfn> (numeric value 0)</dt>

   <dd><p>No information regarding the <span>media resource</span> is available. No data for the
   <span>current playback position</span> is available. <span data-x="media element">Media
   elements</span> whose <code data-x="dom-media-networkState">networkState</code> attribute are set
   to <code data-x="dom-media-NETWORK_EMPTY">NETWORK_EMPTY</code> are always in the <code
   data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</code> state.</p></dd>

   <dt><dfn const for="HTMLMediaElement"><code data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code></dfn> (numeric value 1)</dt>

   <dd><p>Enough of the resource has been obtained that the duration of the resource is available.
   In the case of a <code>video</code> element, the dimensions of the video are also available. No
   <span>media data</span> is available for the immediate <span>current playback
   position</span>.</p></dd>

   <dt><dfn const for="HTMLMediaElement"><code data-x="dom-media-HAVE_CURRENT_DATA">HAVE_CURRENT_DATA</code></dfn> (numeric value 2)</dt>

   <dd><p>Data for the immediate <span>current playback position</span> is available, but either not
   enough data is available that the user agent could successfully advance the <span>current
   playback position</span> in the <span>direction of playback</span> at all without immediately
   reverting to the <code data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code> state, or there is no
   more data to obtain in the <span>direction of playback</span>. For example, in video this
   corresponds to the user agent having data from the current frame, but not the next frame, when
   the <span>current playback position</span> is at the end of the current frame; and to when <span
   data-x="ended playback">playback has ended</span>.</p></dd>

   <dt><dfn const for="HTMLMediaElement"><code data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code></dfn> (numeric value 3)</dt>

   <dd><p>Data for the immediate <span>current playback position</span> is available, as well as
   enough data for the user agent to advance the <span>current playback position</span> in the
   <span>direction of playback</span> at least a little without immediately reverting to the <code
   data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code> state, and <span>the text tracks are
   ready</span>. For example, in video this corresponds to the user agent having data for at least
   the current frame and the next frame when the <span>current playback position</span> is at the
   instant in time between the two frames, or to the user agent having the video data for the
   current frame and audio data to keep playing at least a little when the <span>current playback
   position</span> is in the middle of a frame. The user agent cannot be in this state if <span
   data-x="ended playback">playback has ended</span>, as the <span>current playback position</span>
   can never advance in this case.</p></dd>

   <dt><dfn const for="HTMLMediaElement"><code data-x="dom-media-HAVE_ENOUGH_DATA">HAVE_ENOUGH_DATA</code></dfn> (numeric value 4)</dt>

   <dd>
   <p>All the conditions described for the <code
    data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code> state are met, and, in addition,
    either of the following conditions is also true:</p>

    <ul>
     <li>The user agent estimates that data is being fetched at a rate where the <span>current
     playback position</span>, if it were to advance at the element's <code
     data-x="dom-media-playbackRate">playbackRate</code>, would not overtake the available data
     before playback reaches the end of the <span>media resource</span>.</li>

     <li>The user agent has entered a state where waiting longer will not result in further data
     being obtained, and therefore nothing would be gained by delaying playback any further. (For
     example, the buffer might be full.)</li>
    </ul>
   </dd>
  </dl>

  <p class="note">In practice, the difference between <code
  data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code> and <code
  data-x="dom-media-HAVE_CURRENT_DATA">HAVE_CURRENT_DATA</code> is negligible. Really the only time
  the difference is relevant is when painting a <code>video</code> element onto a
  <code>canvas</code>, where it distinguishes the case where something will be drawn (<code
  data-x="dom-media-HAVE_CURRENT_DATA">HAVE_CURRENT_DATA</code> or greater) from the case where
  nothing is drawn (<code data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code> or less). Similarly,
  the difference between <code data-x="dom-media-HAVE_CURRENT_DATA">HAVE_CURRENT_DATA</code> (only
  the current frame) and <code data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code> (at least
  this frame and the next) can be negligible (in the extreme, only one frame). The only time that
  distinction really matters is when a page provides an interface for "frame-by-frame"
  navigation.</p>

  <div w-nodev>

  <p>When the ready state of a <span>media element</span> whose <code
  data-x="dom-media-networkState">networkState</code> is not <code
  data-x="dom-media-NETWORK_EMPTY">NETWORK_EMPTY</code> changes, the user agent must follow the steps
  given below:</p>

  <ol>
   <li>
    <p>Apply the first applicable set of substeps from the following list:</p>


    <dl class="switch">

     <!-- going up to metadata -->
     <dt>If the previous ready state was <code data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</code>,
     and the new ready state is <code data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code></dt>

     <dd id="fire-loadedmetadata">

      <p><span>Queue a media element task</span> given the <span>media element</span> to <span
      data-x="concept-event-fire">fire an event</span> named <code
      data-x="event-media-loadedmetadata">loadedmetadata</code> at the element.</p>

      <p class="note">Before this task is run, as part of the <span>event loop</span> mechanism, the
      rendering will have been updated to resize the <code>video</code> element if appropriate.</p>
     </dd>

     <!-- going up to current for the first time -->

     <dt id="handling-first-frame-available">If the previous ready state was <code
     data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code> and the new ready state is <code
     data-x="dom-media-HAVE_CURRENT_DATA">HAVE_CURRENT_DATA</code> or greater</dt>

     <dd>
      <p id="fire-loadeddata">If this is the first time this occurs for this <span>media
      element</span> since the <code data-x="dom-media-load">load()</code> algorithm was last
      invoked, the user agent must <span>queue a media element task</span> given the <span>media
      element</span> to <span data-x="concept-event-fire">fire an event</span> named <code
      data-x="event-media-loadeddata">loadeddata</code> at the element.</p>

      <p>If the new ready state is <code data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code>
      or <code data-x="dom-media-HAVE_ENOUGH_DATA">HAVE_ENOUGH_DATA</code>, then the relevant steps
      below must then be run also.</p>
     </dd>

     <!-- going down -->
     <dt>If the previous ready state was <code
     data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code> or more, and the new ready state is
     <code data-x="dom-media-HAVE_CURRENT_DATA">HAVE_CURRENT_DATA</code> or less</dt>

     <dd>
      <p id="fire-waiting-when-waiting">If the <span>media element</span> was <span>potentially
      playing</span> before its <code data-x="dom-media-readyState">readyState</code> attribute
      changed to a value lower than <code
      data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code>, and the element has not
      <span>ended playback</span>, and playback has not <span>stopped due to errors</span>,
      <span>paused for user interaction</span>, or <span>paused for in-band content</span>, the user
      agent must <span>queue a media element task</span> given the <span>media element</span> to
      <span data-x="concept-event-fire">fire an event</span> named <code
      data-x="event-media-timeupdate">timeupdate</code> at the element, and <span>queue a media
      element task</span> given the <span>media element</span> to <span
      data-x="concept-event-fire">fire an event</span> named <code
      data-x="event-media-waiting">waiting</code> at the element.</p>
     </dd>

     <!-- going up to future -->
     <dt>If the previous ready state was <code
     data-x="dom-media-HAVE_CURRENT_DATA">HAVE_CURRENT_DATA</code> or less, and the new ready state
     is <code data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code></dt>

     <dd>
      <p>The user agent must <span>queue a media element task</span> given the <span>media
      element</span> to <span data-x="concept-event-fire">fire an event</span> named <code
      data-x="event-media-canplay">canplay</code> at the element.</p>

      <p>If the element's <code data-x="dom-media-paused">paused</code> attribute is false, the user
      agent must <span>notify about playing</span> for the element.</p>
     </dd>

     <!-- going up to enough -->
     <dt>If the new ready state is <code
     data-x="dom-media-HAVE_ENOUGH_DATA">HAVE_ENOUGH_DATA</code></dt>

     <dd>
      <p>If the previous ready state was <code
      data-x="dom-media-HAVE_CURRENT_DATA">HAVE_CURRENT_DATA</code> or less, the user agent must
      <span>queue a media element task</span> given the <span>media element</span> to <span
      data-x="concept-event-fire">fire an event</span> named <code
      data-x="event-media-canplay">canplay</code> at the element, and, if the element's <code
      data-x="dom-media-paused">paused</code> attribute is false, <span>notify about playing</span>
      for the element.</p>

      <p>The user agent must <span>queue a media element task</span> given the <span>media
      element</span> to <span data-x="concept-event-fire">fire an event</span> named <code
      data-x="event-media-canplaythrough">canplaythrough</code> at the element.</p>

      <p>If the element is not <span>eligible for autoplay</span>, then the user agent must abort
      these substeps.</p>

      <p>The user agent may run the following substeps:</p>

      <ol>
       <li>Set the <code data-x="dom-media-paused">paused</code> attribute to false.</li>

       <li>If the element's <span>show poster flag</span> is true, set it to false and run the
       <i data-x="time marches on">time marches on</i> steps.</li>

       <li><span>Queue a media element task</span> given the element to <span
       data-x="concept-event-fire">fire an event</span> named <code
       data-x="event-media-play">play</code> at the element.</li>

       <li><span>Notify about playing</span> for the element.</li>
      </ol>

      <p>Alternatively, if the element is a <code>video</code> element, the user agent may start
      observing whether the element <span data-x="intersect the viewport">intersects the
      viewport</span>. When the element starts <span data-x="intersect the viewport">intersecting
      the viewport</span>, if the element is still <span>eligible for autoplay</span>, run the
      substeps above. Optionally, when the element stops <span data-x="intersect the
      viewport">intersecting the viewport</span>, if the <span>can autoplay flag</span> is still
      true and the <code data-x="attr-media-autoplay">autoplay</code> attribute is still specified,
      run the following substeps:</p>

      <ol>
       <li>Run the <span>internal pause steps</span> and set the <span>can autoplay flag</span> to
       true.</li>

       <li><span>Queue a media element task</span> given the element to <span
       data-x="concept-event-fire">fire an event</span> named <code
       data-x="event-media-pause">pause</code> at the element.</li>
      </ol>

      <p class="note">The substeps for playing and pausing can run multiple times as the element
      starts or stops <span data-x="intersect the viewport">intersecting the viewport</span>, as
      long as the <span>can autoplay flag</span> is true.</p>

      <p class="note">User agents do not need to support autoplay, and it is suggested that user
      agents honor user preferences on the matter. Authors are urged to use the <code
      data-x="attr-media-autoplay">autoplay</code> attribute rather than using script to force the
      video to play, so as to allow the user to override the behavior if so desired.</p>
     </dd>
    </dl>
   </li>
  </ol>

  </div>

  <p class="note">It is possible for the ready state of a media element to jump between these states
  discontinuously. For example, the state of a media element can jump straight from <code
  data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code> to <code
  data-x="dom-media-HAVE_ENOUGH_DATA">HAVE_ENOUGH_DATA</code> without passing through the <code
  data-x="dom-media-HAVE_CURRENT_DATA">HAVE_CURRENT_DATA</code> and <code
  data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code> states.</p>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-readyState">readyState</code></dfn> IDL attribute must, on getting, return the
  value described above that describes the current ready state of the <span>media
  element</span>.</p>

  </div>

  <p>The <dfn element-attr for="audio,video"><code
  data-x="attr-media-autoplay">autoplay</code></dfn> attribute is a <span>boolean attribute</span>.
  When present, the user agent <span w-nodev>(as described in the algorithm described herein)</span>
  will automatically begin playback of the <span>media resource</span> as soon as it can do so
  without stopping.</p>

  <p class="note">Authors are urged to use the <code data-x="attr-media-autoplay">autoplay</code>
  attribute rather than using script to trigger automatic playback, as this allows the user to
  override the automatic playback when it is not desired, e.g. when using a screen reader. Authors
  are also encouraged to consider not using the automatic playback behavior at all, and instead to
  let the user agent wait for the user to start playback explicitly.</p>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-autoplay">autoplay</code></dfn> IDL attribute must <span>reflect</span> the
  content attribute of the same name.</p>

  </div>



  <h5>Playing the media resource</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-paused">paused</span></code></dt>

   <dd><p>Returns true if playback is paused; false otherwise.</p></dd>

   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-ended">ended</span></code></dt>

   <dd><p>Returns true if playback has reached the end of the <span>media resource</span>.</p></dd>

   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-defaultPlaybackRate">defaultPlaybackRate</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the default rate of playback, for when the user is not fast-forwarding or reversing
    through the <span>media resource</span>.</p>

    <p>Can be set, to change the default rate of playback.</p>

    <p>The default rate has no direct effect on playback, but if the user switches to a fast-forward
    mode, when they return to the normal playback mode, it is expected that the rate of playback
    will be returned to the default rate of playback.</p>
   </dd>

   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-playbackRate">playbackRate</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current rate playback, where 1.0 is normal speed.</p>

    <p>Can be set, to change the rate of playback.</p>
   </dd>

   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-preservesPitch">preservesPitch</span></code></dt>

   <dd>
    <p>Returns true if pitch-preserving algorithms are used when the <code
    data-x="dom-media-playbackRate">playbackRate</code> is not 1.0. The default value is true.</p>

    <p>Can be set to false to have the <span>media resource</span>'s audio pitch change up or down
    depending on the <code data-x="dom-media-playbackRate">playbackRate</code>. This is useful for
    aesthetic and performance reasons.</p>
   </dd>

   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-played">played</span></code></dt>

   <dd>
    <p>Returns a <code>TimeRanges</code> object that represents the ranges of the <span>media
    resource</span> that the user agent has played.</p>
   </dd>

   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-play">play</span>()</code></dt>

   <dd>
    <p>Sets the <code data-x="dom-media-paused">paused</code> attribute to false, loading the
    <span>media resource</span> and beginning playback if necessary. If the playback had ended, will
    restart it from the start.</p>
   </dd>

   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-pause">pause</span>()</code></dt>

   <dd>
    <p>Sets the <code data-x="dom-media-paused">paused</code> attribute to true, loading the
    <span>media resource</span> if necessary.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLMediaElement"><code data-x="dom-media-paused">paused</code></dfn>
  attribute represents whether the <span>media element</span> is paused or not. The attribute must
  initially be true.</p>

  <p>A <span>media element</span> is a <dfn>blocked media element</dfn> if its <code
  data-x="dom-media-readyState">readyState</code> attribute is in the <code
  data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</code> state, the <code
  data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code> state, or the <code
  data-x="dom-media-HAVE_CURRENT_DATA">HAVE_CURRENT_DATA</code> state, or if the element has
  <span>paused for user interaction</span> or <span>paused for in-band content</span>.</p>

  <p>A <span>media element</span> is said to be <dfn export for="media element">potentially
  playing</dfn> when its <code data-x="dom-media-paused">paused</code> attribute is false, the
  element has not <span>ended playback</span>, playback has not <span>stopped due to errors</span>,
  and the element is not a <span>blocked media element</span>.</p>

  <p class="note">A <code data-x="event-media-waiting">waiting</code> DOM event <a
  href="#fire-waiting-when-waiting">can be fired</a> as a result of an element that is
  <span>potentially playing</span> stopping playback due to its <code
  data-x="dom-media-readyState">readyState</code> attribute changing to a value lower than <code
  data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code>.</p>

  <p>A <span>media element</span> is said to be <dfn>eligible for autoplay</dfn> when all of the
  following are true:</p>

  <ul>
   <li><p>its <span>can autoplay flag</span> is true;</p></li>

   <li><p>its <code data-x="dom-media-paused">paused</code> attribute is true;</p></li>

   <li><p>it has an <code data-x="attr-media-autoplay">autoplay</code> attribute
   specified;</p></li>

   <li><p>its <span>node document</span>'s <span>active sandboxing flag set</span> does not have
   the <span>sandboxed automatic features browsing context flag</span> set; and</p></li>

   <li><p>its <span>node document</span> is <span>allowed to use</span> the "<code
   data-x="autoplay-feature">autoplay</code>" feature.</p></li>
  </ul>

  <p>A <span>media element</span> is said to be <dfn>allowed to play</dfn> if the user agent and the
  system allow media playback in the current context.</p>

  <p class="example">For example, a user agent could allow playback only when the <span>media
  element</span>'s <code>Window</code> object has <span>transient activation</span>, but an
  exception could be made to allow playback while <span
  data-x="concept-media-muted">muted</span>.</p>

  <p>A <span>media element</span> is said to have <dfn>ended playback</dfn> when:</p>

  <ul>
   <li><p>The element's <code data-x="dom-media-readyState">readyState</code> attribute is <code
   data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code> or greater, and</p>

   <li>
    <p>Either:</p>

    <ul>
     <li><p>The <span>current playback position</span> is the end of the <span>media
     resource</span>, and</p></li>

     <li><p>The <span>direction of playback</span> is forwards, and</p></li>

     <li><p>The <span>media element</span> does not have a <code
     data-x="attr-media-loop">loop</code> attribute specified.</p></li>
    </ul>

    <p>Or:</p>

    <ul>
     <li><p>The <span>current playback position</span> is the <span>earliest possible
     position</span>, and</p></li>

     <li><p>The <span>direction of playback</span> is backwards.</p></li>
    </ul>
   </li>
  </ul>

  <p>The <dfn attribute for="HTMLMediaElement"><code data-x="dom-media-ended">ended</code></dfn>
  attribute must return true if, the last time the <span>event loop</span> reached <a
  href="#step1">step 1</a>, the <span>media element</span> had <span>ended playback</span> and the
  <span>direction of playback</span> was forwards, and false otherwise.</p>

  <p>A <span>media element</span> is said to have <dfn>stopped due to errors</dfn> when the
  element's <code data-x="dom-media-readyState">readyState</code> attribute is <code
  data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code> or greater, and the user agent <a
  href="#non-fatal-media-error">encounters a non-fatal error</a> during the processing of the
  <span>media data</span>, and due to that error, is not able to play the content at the
  <span>current playback position</span>.</p>

  <p>A <span>media element</span> is said to have <dfn>paused for user interaction</dfn> when its
  <code data-x="dom-media-paused">paused</code> attribute is false, the <code
  data-x="dom-media-readyState">readyState</code> attribute is either <code
  data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code> or <code
  data-x="dom-media-HAVE_ENOUGH_DATA">HAVE_ENOUGH_DATA</code> and the user agent has reached a point
  in the <span>media resource</span> where the user has to make a selection for the resource to
  continue.</p>

  <p>It is possible for a <span>media element</span> to have both <span>ended playback</span> and
  <span>paused for user interaction</span> at the same time.</p>

  <p>When a <span>media element</span> that is <span>potentially playing</span> stops playing
  because it has <span>paused for user interaction</span>, the user agent must <span>queue a media
  element task</span> given the <span>media element</span> to <span data-x="concept-event-fire">fire
  an event</span> named <code data-x="event-media-timeupdate">timeupdate</code> at the element.</p>

  <p>A <span>media element</span> is said to have <dfn>paused for in-band content</dfn> when its
  <code data-x="dom-media-paused">paused</code> attribute is false, the <code
  data-x="dom-media-readyState">readyState</code> attribute is either <code
  data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code> or <code
  data-x="dom-media-HAVE_ENOUGH_DATA">HAVE_ENOUGH_DATA</code> and the user agent has suspended
  playback of the <span>media resource</span> in order to play content that is temporally anchored
  to the <span>media resource</span> and has a nonzero length, or to play content that is
  temporally anchored to a segment of the <span>media resource</span> but has a length longer than
  that segment.</p>

  <p class="example">One example of when a <span>media element</span> would be <span>paused for
  in-band content</span> is when the user agent is playing <span
  data-x="attr-track-kind-descriptions">audio descriptions</span> from an external WebVTT file, and
  the synthesized speech generated for a cue is longer than the time between the <span>text track
  cue start time</span> and the <span>text track cue end time</span>.</p>

  <hr>

  <p id="reaches-the-end">When the <span>current playback position</span> reaches the end of the
  <span>media resource</span> when the <span>direction of playback</span> is forwards, then the user
  agent must follow these steps:</p>

  <ol>
   <li><p>If the <span>media element</span> has a <code data-x="attr-media-loop">loop</code>
   attribute specified, then <span data-x="dom-media-seek">seek</span> to the <span>earliest
   possible position</span> of the <span>media resource</span> and return.</p></li>
   <!-- v2/v3: We should fire a 'looping' event here to explain why this immediately fires a
   'playing' event, otherwise the 'playing' event that fires from the readyState going from
   HAVE_CURRENT_DATA back to HAVE_FUTURE_DATA will seem inexplicable (since the normally matching
   'ended' given below event doesn't fire in the loop case). -->

   <li><p>As defined above, the <code data-x="dom-media-ended">ended</code> IDL attribute starts
   returning true once the <span>event loop</span> returns to <a href="#step1">step 1</a>.</p></li>

   <li>
    <p><span>Queue a media element task</span> given the <span>media element</span> and the
    following steps:</p>

    <ol>
     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-media-timeupdate">timeupdate</code> at the <span>media element</span>.</p></li>

     <li>
      <p>If the <span>media element</span> has <span>ended playback</span>, the <span>direction of
      playback</span> is forwards, and <span data-x="dom-media-paused">paused</span> is false,
      then:</p>

      <ol>
       <li><p>Set the <code data-x="dom-media-paused">paused</code> attribute to true.</p></li>

       <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
       data-x="event-media-pause">pause</code> at the <span>media element</span>.</p></li>

       <li><p><span>Take pending play promises</span> and <span>reject pending play promises</span>
       with the result and an <span>"<code>AbortError</code>"</span>
       <code>DOMException</code>.</p></li>
      </ol>
     </li>

     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-media-ended">ended</code> at the <span>media element</span>.</p></li>
    </ol>
   </li>
  </ol>

  <p>When the <span>current playback position</span> reaches the <span>earliest possible
  position</span> of the <span>media resource</span> when the <span>direction of playback</span> is
  backwards, then the user agent must only <span>queue a media element task</span> given the
  <span>media element</span> to <span data-x="concept-event-fire">fire an event</span> named <code
  data-x="event-media-timeupdate">timeupdate</code> at the element.</p>

  <p class="note">The word "reaches" here does not imply that the <span>current playback
  position</span> needs to have changed during normal playback; it could be via <span
  data-x="dom-media-seek">seeking</span>, for instance.</p>

  <hr>

  <p>The <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-defaultPlaybackRate">defaultPlaybackRate</code></dfn> attribute gives the
  desired speed at which the <span>media resource</span> is to play, as a multiple of its intrinsic
  speed. The attribute is mutable: on getting it must return the last value it was set to, or 1.0 if
  it hasn't yet been set; on setting the attribute must be set to the new value.</p>

  <p class="note">The <code data-x="dom-media-defaultPlaybackRate">defaultPlaybackRate</code> is used
  by the user agent when it <span data-x="expose a user interface to the user">exposes a user
  interface to the user</span>.</p>

  <p>The <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-playbackRate">playbackRate</code></dfn> attribute gives the effective playback
  rate, which is the speed at which the <span>media resource</span> plays, as a multiple of its
  intrinsic speed. If it is not equal to the <code
  data-x="dom-media-defaultPlaybackRate">defaultPlaybackRate</code>, then the implication is that
  the user is using a feature such as fast forward or slow motion playback. The attribute is
  mutable: on getting it must return the last value it was set to, or 1.0 if it hasn't yet been set;
  on setting, the user agent must follow these steps:</p>

  <ol>
   <li><p>If the given value is not supported by the user agent, then throw a
   <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Set <code data-x="dom-media-playbackRate">playbackRate</code> to the new value, and if the
   element is <span>potentially playing</span>, change the playback speed.</p></li>
  </ol>

  <p id="rateUpdate">When the <code
  data-x="dom-media-defaultPlaybackRate">defaultPlaybackRate</code> or <code
  data-x="dom-media-playbackRate">playbackRate</code> attributes change value (either by being set
  by script or by being changed directly by the user agent, e.g. in response to user control) the
  user agent must <span>queue a media element task</span> given the <span>media element</span> to
  <span data-x="concept-event-fire">fire an event</span> named <code
  data-x="event-media-ratechange">ratechange</code> at the <span>media element</span>. The user
  agent must process attribute changes smoothly and must not introduce any perceivable gaps or
  muting of playback in response.</p>

  <p>The <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-preservesPitch">preservesPitch</code></dfn> getter steps are to return true if a
  pitch-preserving algorithm is in effect during playback. The setter steps are to correspondingly
  switch the pitch-preserving algorithm on or off, without any perceivable gaps or muting of
  playback. By default, such a pitch-preserving algorithm must be in effect (i.e., the getter will
  initially return true).</p>

  <hr>

  <p>The <dfn attribute for="HTMLMediaElement"><code data-x="dom-media-played">played</code></dfn>
  attribute must return a new static <span>normalized <code>TimeRanges</code> object</span> that
  represents the ranges of points on the <span>media timeline</span> of the <span>media
  resource</span> reached through the usual monotonic increase of the <span>current playback
  position</span> during normal playback, if any, at the time the attribute is evaluated.</p>

  <p class="warning">Returning a new object each time is a bad pattern for attribute getters and is
  only enshrined here as it would be costly to change it. It is not to be copied to new APIs.</p>

  <hr>

  <p>Each <span>media element</span> has a <dfn>list of pending play promises</dfn>, which must
  initially be empty.</p>

  <p>To <dfn>take pending play promises</dfn> for a <span>media element</span>, the user agent must
  run the following steps:</p>

  <ol>
   <li><p>Let <var>promises</var> be an empty list of promises.</p></li>

   <li><p>Copy the <span>media element</span>'s <span>list of pending play promises</span> to
   <var>promises</var>.</p></li>

   <li><p>Clear the <span>media element</span>'s <span>list of pending play
   promises</span>.</p></li>

   <li>Return <var>promises</var>.</li>
  </ol>

  <p>To <dfn>resolve pending play promises</dfn> for a <span>media element</span> with a list of
  promises <var>promises</var>, the user agent must resolve each promise in <var>promises</var> with
  undefined.</p>

  <p>To <dfn>reject pending play promises</dfn> for a <span>media element</span> with a list of
  promises <var>promises</var> and an exception name <var>error</var>, the user agent must reject
  each promise in <var>promises</var> with <var>error</var>.</p>

  <p>To <dfn>notify about playing</dfn> for a <span>media element</span>, the user agent must run
  the following steps:</p>

  <ol>
   <li><p><span>Take pending play promises</span> and let <var>promises</var> be the
   result.</p></li>

   <li>
    <p><span>Queue a media element task</span> given the element and the following steps:</p>

    <ol>
     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-media-playing">playing</code> at the element.</p></li>

     <li><p><span>Resolve pending play promises</span> with <var>promises</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>When the <dfn method for="HTMLMediaElement"><code data-x="dom-media-play">play()</code></dfn>
  method on a <span>media element</span> is invoked, the user agent must run the following
  steps.</p>

  <ol>
   <li><p>If the <span>media element</span> is not <span>allowed to play</span>, then return
   <span>a promise rejected with</span> a <span>"<code>NotAllowedError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li>
    <p>If the <span>media element</span>'s <code data-x="dom-media-error">error</code> attribute is
    not null and its <span data-x="concept-MediaError-code">code</span> is <code
    data-x="dom-MediaError-MEDIA_ERR_SRC_NOT_SUPPORTED">MEDIA_ERR_SRC_NOT_SUPPORTED</code>, then
    return <span>a promise rejected with</span> a <span>"<code>NotSupportedError</code>"</span>
    <code>DOMException</code>.</p>

    <p class="note">This means that the <span>dedicated media source failure steps</span> have run.
    Playback is not possible until the <span>media element load algorithm</span> clears the <code
    data-x="dom-media-error">error</code> attribute.</p>
   </li>

   <li><p>Let <var>promise</var> be a new promise and append <var>promise</var> to the <span>list of
   pending play promises</span>.</p></li>

   <li><p>Run the <span>internal play steps</span> for the <span>media element</span>.</p></li>

   <li><p>Return <var>promise</var>.</p></li>
  </ol>

  <p>The <dfn>internal play steps</dfn> for a <span>media element</span> are as follows:</p>

  <ol>
   <li><p>If the <span>media element</span>'s <code
   data-x="dom-media-networkState">networkState</code> attribute has the value <code
   data-x="dom-media-NETWORK_EMPTY">NETWORK_EMPTY</code>, invoke the <span>media element</span>'s
   <span data-x="concept-media-load-algorithm">resource selection algorithm</span>.</p></li>

   <li>
    <p>If the <span data-x="ended playback">playback has ended</span> and the <span>direction of
    playback</span> is forwards, <span data-x="dom-media-seek">seek</span> to the <span>earliest
    possible position</span> of the <span>media resource</span>.</p>

    <p class="note">This <a href="#seekUpdate">will cause</a> the user agent to <span>queue a media
    element task</span> given the <span>media element</span> to <span
    data-x="concept-event-fire">fire an event</span> named <code
    data-x="event-media-timeupdate">timeupdate</code> at the <span>media element</span>.</p>
    <!-- if we're already playing at this point, it might also fire 'waiting' -->
   </li>

   <li>
    <p>If the <span>media element</span>'s <code data-x="dom-media-paused">paused</code> attribute
    is true, then:</p>

    <ol>
     <li><p>Change the value of <code data-x="dom-media-paused">paused</code> to false.</p></li>

     <li><p>If the <span>show poster flag</span> is true, set the element's <span>show poster
     flag</span> to false and run the <i data-x="time marches on">time marches on</i> steps.</p></li>

     <li><p><span>Queue a media element task</span> given the <span>media element</span> to <span
     data-x="concept-event-fire">fire an event</span> named <code
     data-x="event-media-play">play</code> at the element.</p></li>

     <li>
      <p>If the <span>media element</span>'s <code data-x="dom-media-readyState">readyState</code>
      attribute has the value <code data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</code>, <code
      data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code>, or <code
      data-x="dom-media-HAVE_CURRENT_DATA">HAVE_CURRENT_DATA</code>, <span>queue a media element
      task</span> given the <span>media element</span> to <span data-x="concept-event-fire">fire an
      event</span> named <code data-x="event-media-waiting">waiting</code> at the element.</p>

      <p>Otherwise, the <span>media element</span>'s <code
      data-x="dom-media-readyState">readyState</code> attribute has the value <code
      data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code> or <code
      data-x="dom-media-HAVE_ENOUGH_DATA">HAVE_ENOUGH_DATA</code>: <span>notify about playing</span>
      for the element.</p>
     </li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if the <span>media element</span>'s <code
    data-x="dom-media-readyState">readyState</code> attribute has the value <code
    data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code> or <code
    data-x="dom-media-HAVE_ENOUGH_DATA">HAVE_ENOUGH_DATA</code>, <span>take pending play
    promises</span> and <span>queue a media element task</span> given the <span>media element</span>
    to <span>resolve pending play promises</span> with the result.</p>

    <p class="note">The media element is already playing. However, it's possible that
    <var>promise</var> will be <span data-x="reject pending play promises">rejected</span> before
    the queued task is run.</p>
   </li>

   <li><p>Set the <span>media element</span>'s <span>can autoplay flag</span> to false.</p></li>
  </ol>

  <hr>

  <p>When the <dfn method for="HTMLMediaElement"><code data-x="dom-media-pause">pause()</code></dfn>
  method is invoked, and when the user agent is required to pause the <span>media element</span>,
  the user agent must run the following steps:</p>

  <ol>
   <li><p>If the <span>media element</span>'s <code
   data-x="dom-media-networkState">networkState</code> attribute has the value <code
   data-x="dom-media-NETWORK_EMPTY">NETWORK_EMPTY</code>, invoke the <span>media element</span>'s
   <span data-x="concept-media-load-algorithm">resource selection algorithm</span>.</p></li>

   <li><p>Run the <span>internal pause steps</span> for the <span>media element</span>.</p></li>
  </ol>

  <p>The <dfn>internal pause steps</dfn> for a <span>media element</span> are as follows:</p>

  <ol>
   <li><p>Set the <span>media element</span>'s <span>can autoplay flag</span> to false.</p></li>

   <li><p>If the <span>media element</span>'s <code data-x="dom-media-paused">paused</code> attribute
   is false, run the following steps:</p>

    <ol>
     <li><p>Change the value of <code data-x="dom-media-paused">paused</code> to true.</p></li>

     <li><p><span>Take pending play promises</span> and let <var>promises</var> be the
     result.</p></li>

     <li>
      <p><span>Queue a media element task</span> given the <span>media element</span> and the
      following steps:</p>

      <ol>
       <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
       data-x="event-media-timeupdate">timeupdate</code> at the element.</p></li>

       <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
       data-x="event-media-pause">pause</code> at the element.</p></li>

       <li><p><span>Reject pending play promises</span> with <var>promises</var> and an
       <span>"<code>AbortError</code>"</span> <code>DOMException</code>.</p></li>
      </ol>
     </li>

     <li><p>Set the <span>official playback position</span> to the <span>current playback
     position</span>.</p></li>
    </ol>
   </li>
  </ol>

  <hr>

  <p>If the element's <code data-x="dom-media-playbackRate">playbackRate</code> is positive or zero,
  then the <dfn>direction of playback</dfn> is forwards. Otherwise, it is backwards.</p>

  <p id="media-playback">When a <span>media element</span> is <span>potentially playing</span> and
  its <code>Document</code> is a <span>fully active</span> <code>Document</code>, its <span>current
  playback position</span> must increase monotonically at the element's <code
  data-x="dom-media-playbackRate">playbackRate</code> units of media time per unit time of the
  <span>media timeline</span>'s clock. (This specification always refers to this as an
  <i>increase</i>, but that increase could actually be a <em>de</em>crease if the element's <code
  data-x="dom-media-playbackRate">playbackRate</code> is negative.)</p>

  <p class="note">The element's <code data-x="dom-media-playbackRate">playbackRate</code> can be
  0.0, in which case the <span>current playback position</span> doesn't move, despite playback not
  being paused (<code data-x="dom-media-paused">paused</code> doesn't become true, and the <code
  data-x="event-media-pause">pause</code> event doesn't fire).</p>

  <p class="note">This specification doesn't define how the user agent achieves the appropriate
  playback rate &mdash; depending on the protocol and media available, it is plausible that the user
  agent could negotiate with the server to have the server provide the media data at the appropriate
  rate, so that (except for the period between when the rate is changed and when the server updates
  the stream's playback rate) the client doesn't actually have to drop or interpolate any
  frames.</p>

  <p>Any time the user agent <span data-x="await a stable state">provides a stable state</span>,
  the <span>official playback position</span> must be set to the <span>current playback
  position</span>.</p> <!-- cross-ref is intentionally to the other term, to help find this -->

  <p>While the <span>direction of playback</span> is backwards, any corresponding audio must be
  <span data-x="concept-media-muted">muted</span>. While the element's <code
  data-x="dom-media-playbackRate">playbackRate</code> is so low or so high that the user agent
  cannot play audio usefully, the corresponding audio must also be <span
  data-x="concept-media-muted">muted</span>. If the element's <code
  data-x="dom-media-playbackRate">playbackRate</code> is not 1.0 and <code
  data-x="dom-media-preservesPitch">preservesPitch</code> is true, the user agent must apply pitch
  adjustment to preserve the original pitch of the audio. Otherwise, the user agent must speed up
  or slow down the audio without any pitch adjustment.</p>

  <p>When a <span>media element</span> is <span>potentially playing</span>, its audio data played
  must be synchronized with the <span>current playback position</span>, at the element's
  <span>effective media volume</span>. The user agent must play the audio from audio tracks that
  were enabled when the <span>event loop</span> last reached <a href="#step1">step 1</a>.</p>

  <p>When a <span>media element</span> is not <span>potentially playing</span>, audio must not play
  for the element.</p>

  <p><span data-x="media element">Media elements</span> that are <span>potentially playing</span>
  while not <span>in a document</span> must not play any video, but should play any audio component.
  Media elements must not stop playing just because all references to them have been removed; only
  once a media element is in a state where no further audio could ever be played by that element may
  the element be garbage collected.</p>

  <p class="note">It is possible for an element to which no explicit references exist to play audio,
  even if such an element is not still actively playing: for instance, it could be unpaused but
  stalled waiting for content to buffer, or it could be still buffering, but with a
  <code data-x="event-media-suspend">suspend</code> event listener that begins playback. Even a
  media element whose <span>media resource</span> has no audio tracks could eventually play audio
  again if it had an event listener that changes the <span>media resource</span>.</p>

  <hr>

  <p>Each <span>media element</span> has a <dfn>list of newly introduced cues</dfn>, which must be
  initially empty. Whenever a <span>text track cue</span> is added to the <span data-x="text track
  list of cues">list of cues</span> of a <span>text track</span> that is in the <span>list of text
  tracks</span> for a <span>media element</span>, that <span data-x="text track cue">cue</span> must
  be added to the <span>media element</span>'s <span>list of newly introduced cues</span>. Whenever
  a <span>text track</span> is added to the <span>list of text tracks</span> for a <span>media
  element</span>, all of the <span data-x="text track cue">cues</span> in that <span>text
  track</span>'s <span data-x="text track list of cues">list of cues</span> must be added to the
  <span>media element</span>'s <span>list of newly introduced cues</span>. When a <span>media
  element</span>'s <span>list of newly introduced cues</span> has new cues added while the
  <span>media element</span>'s <span>show poster flag</span> is not set, then the user agent must
  run the <i data-x="time marches on">time marches on</i> steps.</p>

  <p>When a <span>text track cue</span> is removed from the <span data-x="text track list of
  cues">list of cues</span> of a <span>text track</span> that is in the <span>list of text
  tracks</span> for a <span>media element</span>, and whenever a <span>text track</span> is removed
  from the <span>list of text tracks</span> of a <span>media element</span>, if the <span>media
  element</span>'s <span>show poster flag</span> is not set, then the user agent must run the
  <i data-x="time marches on">time marches on</i> steps.</p>

  <p>When the <span>current playback position</span> of a <span>media element</span> changes (e.g.
  due to playback or seeking), the user agent must run the <i data-x="time marches on">time marches
  on</i> steps. To support use cases that depend on the timing accuracy of cue event firing, such as
  synchronizing captions with shot changes in a video, user agents should fire cue events as close
  as possible to their position on the media timeline, and ideally within 20 milliseconds. If the
  <span>current playback position</span> changes while the steps are running, then the user agent
  must wait for the steps to complete, and then must immediately rerun the steps. These steps are
  thus run as often as possible or needed.</p>

  <p class="note">If one iteration takes a long time, this can cause short duration
  <span data-x="text track cue">cues</span> to be skipped over as the user agent rushes ahead to
  "catch up", so these cues will not appear in the <code
  data-x="dom-TextTrack-activeCues">activeCues</code> list.</p>

  <p>The <dfn><i>time marches on</i></dfn> steps are as follows:</p>

  <ol>
   <li><p>Let <var>current cues</var> be a list of <span data-x="text track
   cue">cues</span>, initialized to contain all the <span data-x="text track cue">cues</span> of all
   the <span data-x="text track hidden">hidden</span> or <span data-x="text track
   showing">showing</span> <span data-x="text track">text tracks</span> of the <span>media
   element</span> (not the <span data-x="text track disabled">disabled</span> ones) whose <span
   data-x="text track cue start time">start times</span> are less than or equal to the <span>current
   playback position</span> and whose <span data-x="text track cue end time">end times</span> are
   greater than the <span>current playback position</span>.</p></li>

   <li><p>Let <var>other cues</var> be a list of <span data-x="text track cue">cues</span>,
   initialized to contain all the <span data-x="text track cue">cues</span> of <span data-x="text
   track hidden">hidden</span> and <span data-x="text track showing">showing</span> <span data-x="text
   track">text tracks</span> of the <span>media element</span> that are not present in <var>current cues</var>.</p></li>

   <li><p>Let <var>last time</var> be the <span>current playback position</span> at the
   time this algorithm was last run for this <span>media element</span>, if this is not the first
   time it has run.</p></li>

   <li><p>If the <span>current playback position</span> has, since the last time this algorithm was
   run, only changed through its usual monotonic increase during normal playback, then let
   <var>missed cues</var> be the list of <span data-x="text track cue">cues</span> in <var>other
   cues</var> whose <span data-x="text track cue start time">start times</span> are greater than or
   equal to <var>last time</var> and whose <span data-x="text track cue end time">end times</span>
   are less than or equal to the <span>current playback position</span>. Otherwise, let <var>missed
   cues</var> be an empty list.</p></li>

   <li><p>Remove all the <span data-x="text track cue">cues</span> in <var>missed cues</var>
   that are also in the <span>media element</span>'s <span>list of newly introduced cues</span>, and
   then empty the element's <span>list of newly introduced cues</span>.</p></li>

   <li><p>If the time was reached through the usual monotonic increase of the <span>current playback
   position</span> during normal playback, and if the user agent has not fired a <code
   data-x="event-media-timeupdate">timeupdate</code> event at the element in the past 15 to 250ms
   and is not still running event handlers for such an event, then the user agent must <span>queue a
   media element task</span> given the <span>media element</span> to <span
   data-x="concept-event-fire">fire an event</span> named <code
   data-x="event-media-timeupdate">timeupdate</code> at the element. (In the other cases, such as
   explicit seeks, relevant events get fired as part of the overall process of changing the
   <span>current playback position</span>.)</p>

   <p class="note">The event thus is not to be fired faster than about 66Hz or slower than 4Hz
   (assuming the event handlers don't take longer than 250ms to run). User agents are encouraged to
   vary the frequency of the event based on the system load and the average cost of processing the
   event each time, so that the UI updates are not any more frequent than the user agent can
   comfortably handle while decoding the video.</p></li>

   <li><p>If all of the <span data-x="text track cue">cues</span> in <var>current cues</var>
   have their <span>text track cue active flag</span> set, none of the <span data-x="text track
   cue">cues</span> in <var>other cues</var> have their <span>text track cue active
   flag</span> set, and <var>missed cues</var> is empty, then return.</p></li>

   <li>
    <p>If the time was reached through the usual monotonic increase of the <span>current playback
    position</span> during normal playback, and there are <span data-x="text track cue">cues</span>
    in <var>other cues</var> that have their <span>text track cue pause-on-exit flag</span>
    set and that either have their <span>text track cue active flag</span> set or are also in <var>missed cues</var>, then <span>immediately</span> <span data-x="dom-media-pause">pause</span> the
    <span>media element</span>. <!-- "pause" can in theory call load(), but never can it do so as
    part of this invocation, since we wouldn't be in this algorithm if the media element was empty.
    So, no need to couch all this in a task. --></p>

    <p class="note">In the other cases, such as explicit seeks, playback is not paused by going past
    the end time of a <span data-x="text track cue">cue</span>, even if that <span data-x="text track
    cue">cue</span> has its <span>text track cue pause-on-exit flag</span> set.</p>
   </li>

   <li>
    <p>Let <var>events</var> be a list of <span data-x="concept-task">tasks</span>,
    initially empty. Each <span data-x="concept-task">task</span> in this list will be associated
    with a <span>text track</span>, a <span>text track cue</span>, and a time, which are used to
    sort the list before the <span data-x="concept-task">tasks</span> are queued.</p>

    <p>Let <var>affected tracks</var> be a list of <span data-x="text track">text
    tracks</span>, initially empty.</p>

    <p>When the steps below say to <dfn>prepare an event</dfn> named <var>event</var> for a
    <span>text track cue</span> <var>target</var> with a time <var>time</var>, the
    user agent must run these steps:</p>

    <ol>
     <li><p>Let <var>track</var> be the <span>text track</span> with which the <span>text
     track cue</span> <var>target</var> is associated.</p></li>

     <li><p>Create a <span data-x="concept-task">task</span> to <span
     data-x="concept-event-fire">fire an event</span> named <var>event</var> at
     <var>target</var>.</p></li>

     <li><p>Add the newly created <span data-x="concept-task">task</span> to <var>events</var>, associated with the time <var>time</var>, the <span>text
     track</span> <var>track</var>, and the <span>text track cue</span> <var>target</var>.</p></li>

     <li><p>Add <var>track</var> to <var>affected tracks</var>.</p></li>
    </ol>
   </li>

   <li><p>For each <span data-x="text track cue">text track cue</span> in <var>missed
   cues</var>, <span>prepare an event</span> named <code data-x="event-media-enter">enter</code> for the
   <code>TextTrackCue</code> object with the <span>text track cue start time</span>.</p></li>

   <li><p>For each <span data-x="text track cue">text track cue</span> in <var>other
   cues</var> that either has its <span>text track cue active flag</span> set or is in <var>missed cues</var>, <span>prepare an event</span> named <code
   data-x="event-media-exit">exit</code> for the <code>TextTrackCue</code> object with the later of the
   <span>text track cue end time</span> and the <span>text track cue start time</span>.</p></li>

   <li><p>For each <span data-x="text track cue">text track cue</span> in <var>current
   cues</var> that does not have its <span>text track cue active flag</span> set, <span>prepare an
   event</span> named <code data-x="event-media-enter">enter</code> for the <code>TextTrackCue</code>
   object with the <span>text track cue start time</span>.</p></li>

   <li>
    <p>Sort the <span data-x="concept-task">tasks</span> in <var>events</var> in ascending
    time order (<span data-x="concept-task">tasks</span> with earlier times first).</p>

    <p>Further sort <span data-x="concept-task">tasks</span> in <var>events</var> that have
    the same time by the relative <span>text track cue order</span> of the <span data-x="text track
    cue">text track cues</span> associated with these <span data-x="concept-task">tasks</span>.</p>

    <p>Finally, sort <span data-x="concept-task">tasks</span> in <var>events</var> that have the
    same time and same <span>text track cue order</span> by placing <span
    data-x="concept-task">tasks</span> that fire <code data-x="event-media-enter">enter</code>
    events before those that fire <code data-x="event-media-exit">exit</code> events.</p>
   </li>

   <li><p><span>Queue a media element task</span> given the <span>media element</span> for each
   <span data-x="concept-task">task</span> in <var>events</var>, in list order.</p></li>

   <li><p>Sort <var>affected tracks</var> in the same order as the <span data-x="text
   track">text tracks</span> appear in the <span>media element</span>'s <span>list of text
   tracks</span>, and remove duplicates.</p>

   <li><p>For each <span>text track</span> in <var>affected tracks</var>, in the list order,
   <span>queue a media element task</span> given the <span>media element</span> to <span
   data-x="concept-event-fire">fire an event</span> named <code
   data-x="event-media-cuechange">cuechange</code> at the <code>TextTrack</code> object, and, if the
   <span>text track</span> has a corresponding <code>track</code> element, to then <span
   data-x="concept-event-fire">fire an event</span> named <code
   data-x="event-media-cuechange">cuechange</code> at the <code>track</code> element as well.</p></li>

   <li><p>Set the <span>text track cue active flag</span> of all the <span data-x="text track
   cue">cues</span> in the <var>current cues</var>, and unset the <span>text track cue
   active flag</span> of all the <span data-x="text track cue">cues</span> in the <var>other
   cues</var>.</p></li>

   <li><p>Run the <span>rules for updating the text track rendering</span> of each of the <span
   data-x="text track">text tracks</span> in <var>affected tracks</var> that are <span data-x="text
   track showing">showing</span>, providing the <span>text track</span>'s <span>text track
   language</span> as the fallback language if it is not the empty string. For example, for <span
   data-x="text track">text tracks</span> based on WebVTT, the <span>rules for updating the display
   of WebVTT text tracks</span>. <ref>WEBVTT</ref></p></li>
  </ol>

  <p>For the purposes of the algorithm above, a <span>text track cue</span> is considered to be part
  of a <span>text track</span> only if it is listed in the <span>text track list of cues</span>, not
  merely if it is associated with the <span>text track</span>.</p>

  <p class="note">If the <span>media element</span>'s <span>node document</span> stops being a
  <span>fully active</span> document, then the playback will <a href="#media-playback">stop</a>
  until the document is active again.</p>

  <p>When a <span>media element</span> is <span data-x="node is removed from a document">removed
  from a <code>Document</code></span>, the user agent must run the following steps:</p>

  <ol>
   <li><p><span>Await a stable state</span>, allowing the <span
   data-x="concept-task">task</span> that removed the <span>media element</span> from the
   <code>Document</code> to continue. The <span>synchronous section</span> consists of all the
   remaining steps of this algorithm. (Steps in the <span>synchronous section</span> are marked with
   &#x231B;.)</p></li>

   <li><p>&#x231B; If the <span>media element</span> is <span>in a document</span>, return.</p></li>

   <li><p>&#x231B; Run the <span>internal pause steps</span> for the <span>media element</span>.</p>
  </ol>

  </div>



  <h5>Seeking</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-seeking">seeking</span></code></dt>

   <dd><p>Returns true if the user agent is currently seeking.</p></dd>

   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-seekable">seekable</span></code></dt>

   <dd>
    <p>Returns a <code>TimeRanges</code> object that represents the ranges of the <span>media
    resource</span> to which it is possible for the user agent to seek.</p>
   </dd>

   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-fastSeek">fastSeek</span>(<var>time</var>)</code></dt>

   <dd>
    <p>Seeks to near the given <var>time</var> as fast as possible, trading precision for speed. (To
    seek to a precise time, use the <code data-x="dom-media-currentTime">currentTime</code>
    attribute.)</p>

    <p>This does nothing if the media resource has not been loaded.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLMediaElement"><code data-x="dom-media-seeking">seeking</code></dfn>
  attribute must initially have the value false.</p>

  <p>The <dfn method for="HTMLMediaElement"><code
  data-x="dom-media-fastSeek">fastSeek(<var>time</var>)</code></dfn> method must <span
  data-x="dom-media-seek">seek</span> to the time given by <var>time</var>, with the
  <i>approximate-for-speed</i> flag set.</p>

  <p>When the user agent is required to <dfn data-x="dom-media-seek">seek</dfn> to a particular <var>new playback position</var> in the <span>media resource</span>, optionally with the
  <i>approximate-for-speed</i> flag set, it means that the user agent must run the following steps.
  This algorithm interacts closely with the <span>event loop</span> mechanism; in particular, it has
  a <span>synchronous section</span> (which is triggered as part of the <span>event loop</span>
  algorithm). Steps in that section are marked with &#x231B;.</p>

  <ol>
   <li><p>Set the <span>media element</span>'s <span>show poster flag</span> to false.</p></li>

   <li><p>If the <span>media element</span>'s <code data-x="dom-media-readyState">readyState</code>
   is <code data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</code>, return.</p></li>

   <li><p>If the element's <code data-x="dom-media-seeking">seeking</code> IDL attribute is true,
   then another instance of this algorithm is already running. Abort that other instance of the
   algorithm without waiting for the step that it is running to complete.</p></li>

   <li><p>Set the <code data-x="dom-media-seeking">seeking</code> IDL attribute to true.</p></li>

   <li><p>If the seek was in response to a DOM method call or setting of an IDL attribute, then
   continue the script. The remainder of these steps must be run <span>in parallel</span>. With the exception
   of the steps marked with &#x231B;, they could be aborted at any time by another instance of this
   algorithm being invoked.</p></li>

   <li><p>If the <var>new playback position</var> is later than the end of the <span>media
   resource</span>, then let it be the end of the <span>media resource</span> instead.</p></li>

   <li><p>If the <var>new playback position</var> is less than the <span>earliest possible
   position</span>, let it be that position instead.</p></li>

   <li><p>If the (possibly now changed) <var>new playback position</var> is not in one of
   the ranges given in the <code data-x="dom-media-seekable">seekable</code> attribute, then let it
   be the position in one of the ranges given in the <code
   data-x="dom-media-seekable">seekable</code> attribute that is the nearest to the <var>new
   playback position</var>. If two positions both satisfy that constraint (i.e. the <var>new playback position</var> is exactly in the middle between two ranges in the <code
   data-x="dom-media-seekable">seekable</code> attribute) then use the position that is closest to
   the <span>current playback position</span>. If there are no ranges given in the <code
   data-x="dom-media-seekable">seekable</code> attribute then set the <code
   data-x="dom-media-seeking">seeking</code> IDL attribute to false and return.</p></li>

   <li>
    <p>If the <i>approximate-for-speed</i> flag is set, adjust the <var>new playback
    position</var> to a value that will allow for playback to resume promptly. If <var>new
    playback position</var> before this step is before <span>current playback position</span>, then
    the adjusted <var>new playback position</var> must also be before the <span>current
    playback position</span>. Similarly, if the <var>new playback position</var> before
    this step is after <span>current playback position</span>, then the adjusted <var>new
    playback position</var> must also be after the <span>current playback position</span>.</p>

    <p class="example">For example, the user agent could snap to a nearby key frame, so that it
    doesn't have to spend time decoding then discarding intermediate frames before resuming
    playback.</p>
   </li>

   <li><p><span>Queue a media element task</span> given the <span>media element</span> to <span
   data-x="concept-event-fire">fire an event</span> named <code
   data-x="event-media-seeking">seeking</code> at the element.</p></li>

   <li>
    <p>Set the <span>current playback position</span> to the <var>new playback
    position</var>.</p>

    <p class="note">If the <span>media element</span> was <span>potentially playing</span>
    immediately before it started seeking, but seeking caused its <code
    data-x="dom-media-readyState">readyState</code> attribute to change to a value lower than <code
    data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code>, then a <code
    data-x="event-media-waiting">waiting</code> event <a href="#fire-waiting-when-waiting">will be
    fired</a> at the element.</p> <!-- also, timeupdate -->

    <p class="note">This step sets the <span>current playback position</span>, and thus can
    immediately trigger other conditions, such as the rules regarding when playback "<a
    href="#reaches-the-end">reaches the end of the media resource</a>" (part of the logic that
    handles looping), even before the user agent is actually able to render the media data for that
    position (as determined in the next step).</p>

    <p class="note">The <code data-x="dom-media-currentTime">currentTime</code> attribute returns
    the <span>official playback position</span>, not the <span>current playback position</span>, and
    therefore gets updated before script execution, separate from this algorithm.</p>
   </li>

   <li><p>Wait until the user agent has established whether or not the <span>media data</span> for
   the <var>new playback position</var> is available, and, if it is, until it has decoded
   enough data to play back that position.</p></li>

   <li><p><span>Await a stable state</span>. The <span>synchronous section</span> consists of all
   the remaining steps of this algorithm. (Steps in the <span>synchronous section</span> are marked
   with &#x231B;.)</p></li>

   <li><p>&#x231B; Set the <code data-x="dom-media-seeking">seeking</code> IDL attribute to
   false.</p></li>

   <li><p>&#x231B; Run the <span>time marches on</span> steps.</p></li>

   <li id="seekUpdate"><p>&#x231B; <span>Queue a media element task</span> given the <span>media
   element</span> to <span data-x="concept-event-fire">fire an event</span> named <code
   data-x="event-media-timeupdate">timeupdate</code> at the element.</p></li>

   <li><p>&#x231B; <span>Queue a media element task</span> given the <span>media element</span> to
   <span data-x="concept-event-fire">fire an event</span> named <code
   data-x="event-media-seeked">seeked</code> at the element.</p></li>
  </ol>

  <hr>

  <p>The <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-seekable">seekable</code></dfn> attribute must return a new static
  <span>normalized <code>TimeRanges</code> object</span> that represents the ranges of the
  <span>media resource</span>, if any, that the user agent is able to seek to, at the time the
  attribute is evaluated.</p>

  <p class="note">If the user agent can seek to anywhere in the <span>media resource</span>, e.g.
  because it is a simple movie file and the user agent and the server support HTTP Range requests,
  then the attribute would return an object with one range, whose start is the time of the first
  frame (the <span>earliest possible position</span>, typically zero), and whose end is the same as
  the time of the first frame plus the <code data-x="dom-media-duration">duration</code> attribute's
  value (which would equal the time of the last frame, and might be positive Infinity).</p>

  <p class="note">The range might be continuously changing, e.g. if the user agent is buffering a
  sliding window on an infinite stream. This is the behavior seen with DVRs viewing live TV, for
  instance.</p>

  <p class="warning">Returning a new object each time is a bad pattern for attribute getters and is
  only enshrined here as it would be costly to change it. It is not to be copied to new APIs.</p>

  <p>User agents should adopt a very liberal and optimistic view of what is seekable. User
  agents should also buffer recent content where possible to enable seeking to be fast.</p>

  <p class="example">For instance, consider a large video file served on an HTTP server without
  support for HTTP Range requests. A browser <em>could</em> implement this by only buffering the
  current frame and data obtained for subsequent frames, never allow seeking, except for seeking to
  the very start by restarting the playback. However, this would be a poor implementation. A high
  quality implementation would buffer the last few minutes of content (or more, if sufficient
  storage space is available), allowing the user to jump back and rewatch something surprising
  without any latency, and would in addition allow arbitrary seeking by reloading the file from the
  start if necessary, which would be slower but still more convenient than having to literally
  restart the video and watch it all the way through just to get to an earlier unbuffered spot.</p>

  <p><span data-x="media resource">Media resources</span> might be internally scripted or
  interactive. Thus, a <span>media element</span> could play in a non-linear fashion. If this
  happens, the user agent must act as if the algorithm for <span
  data-x="dom-media-seek">seeking</span> was used whenever the <span>current playback position</span>
  changes in a discontinuous fashion (so that the relevant events fire).</p>

  </div>


  <h5>Media resources with multiple media tracks</h5>

  <p>A <span>media resource</span> can have multiple embedded audio and video tracks. For example,
  in addition to the primary video and audio tracks, a <span>media resource</span> could have
  foreign-language dubbed dialogues, director's commentaries, audio descriptions, alternative
  angles, or sign-language overlays.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-audioTracks">audioTracks</span></code></dt>

   <dd>
    <p>Returns an <code>AudioTrackList</code> object representing the audio tracks available in the
    <span>media resource</span>.</p>
   </dd>

   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-videoTracks">videoTracks</span></code></dt>

   <dd>
    <p>Returns a <code>VideoTrackList</code> object representing the video tracks available in the
    <span>media resource</span>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-audioTracks">audioTracks</code></dfn> attribute of a <span>media element</span>
  must return a <span>live</span> <code>AudioTrackList</code> object representing the audio tracks
  available in the <span>media element</span>'s <span>media resource</span>.</p>

  <p>The <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-videoTracks">videoTracks</code></dfn> attribute of a <span>media element</span>
  must return a <span>live</span> <code>VideoTrackList</code> object representing the video tracks
  available in the <span>media element</span>'s <span>media resource</span>.</p>

  <p class="note">There are only ever one <code>AudioTrackList</code> object and one
  <code>VideoTrackList</code> object per <span>media element</span>, even if another <span>media
  resource</span> is loaded into the element: the objects are reused. (The <code>AudioTrack</code>
  and <code>VideoTrack</code> objects are not, though.)</p>

  </div>


  <h6><code>AudioTrackList</code> and <code>VideoTrackList</code> objects</h6>

  <p>The <code>AudioTrackList</code> and <code>VideoTrackList</code> interfaces are used by
  attributes defined in the previous section.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>AudioTrackList</dfn> : <span>EventTarget</span> {
  readonly attribute unsigned long <span data-x="dom-AudioTrackList-length">length</span>;
  <a href="#dom-tracklist-item">getter</a> <span>AudioTrack</span> (unsigned long index);
  <span>AudioTrack</span>? <span data-x="dom-AudioTrackList-getTrackById">getTrackById</span>(DOMString id);

  attribute <span>EventHandler</span> <span data-x="handler-TrackList-onchange">onchange</span>;
  attribute <span>EventHandler</span> <span data-x="handler-TrackList-onaddtrack">onaddtrack</span>;
  attribute <span>EventHandler</span> <span data-x="handler-TrackList-onremovetrack">onremovetrack</span>;
};

[Exposed=Window]
interface <dfn interface>AudioTrack</dfn> {
  readonly attribute DOMString <span data-x="dom-AudioTrack-id">id</span>;
  readonly attribute DOMString <span data-x="dom-AudioTrack-kind">kind</span>;
  readonly attribute DOMString <span data-x="dom-AudioTrack-label">label</span>;
  readonly attribute DOMString <span data-x="dom-AudioTrack-language">language</span>;
  attribute boolean <span data-x="dom-AudioTrack-enabled">enabled</span>;
};

[Exposed=Window]
interface <dfn interface>VideoTrackList</dfn> : <span>EventTarget</span> {
  readonly attribute unsigned long <span data-x="dom-VideoTrackList-length">length</span>;
  <a href="#dom-tracklist-item">getter</a> <span>VideoTrack</span> (unsigned long index);
  <span>VideoTrack</span>? <span data-x="dom-VideoTrackList-getTrackById">getTrackById</span>(DOMString id);
  readonly attribute long <span data-x="dom-VideoTrackList-selectedIndex">selectedIndex</span>;

  attribute <span>EventHandler</span> <span data-x="handler-TrackList-onchange">onchange</span>;
  attribute <span>EventHandler</span> <span data-x="handler-TrackList-onaddtrack">onaddtrack</span>;
  attribute <span>EventHandler</span> <span data-x="handler-TrackList-onremovetrack">onremovetrack</span>;
};

[Exposed=Window]
interface <dfn interface>VideoTrack</dfn> {
  readonly attribute DOMString <span data-x="dom-VideoTrack-id">id</span>;
  readonly attribute DOMString <span data-x="dom-VideoTrack-kind">kind</span>;
  readonly attribute DOMString <span data-x="dom-VideoTrack-label">label</span>;
  readonly attribute DOMString <span data-x="dom-VideoTrack-language">language</span>;
  attribute boolean <span data-x="dom-VideoTrack-selected">selected</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>media</var>.<span data-x="dom-media-audioTracks">audioTracks</span>.<span subdfn data-x="dom-AudioTrackList-length">length</span></code></dt>
   <dt><code data-x=""><var>media</var>.<span data-x="dom-media-videoTracks">videoTracks</span>.<span subdfn data-x="dom-VideoTrackList-length">length</span></code></dt>

   <dd><p>Returns the number of tracks in the list.</p></dd>

   <dt><code data-x=""><var>audioTrack</var> = <var>media</var>.<span data-x="dom-media-audioTracks">audioTracks</span>[<var>index</var>]</code></dt>
   <dt><code data-x=""><var>videoTrack</var> = <var>media</var>.<span data-x="dom-media-videoTracks">videoTracks</span>[<var>index</var>]</code></dt>

   <dd><p>Returns the specified <code>AudioTrack</code> or <code>VideoTrack</code> object.</p></dd>

   <dt><code data-x=""><var>audioTrack</var> = <var>media</var>.<span data-x="dom-media-audioTracks">audioTracks</span>.<span subdfn data-x="dom-AudioTrackList-getTrackById">getTrackById</span>(<var>id</var>)</code></dt>
   <dt><code data-x=""><var>videoTrack</var> = <var>media</var>.<span data-x="dom-media-videoTracks">videoTracks</span>.<span subdfn data-x="dom-VideoTrackList-getTrackById">getTrackById</span>(<var>id</var>)</code></dt>

   <dd>
    <p>Returns the <code>AudioTrack</code> or <code>VideoTrack</code> object with the given
    identifier, or null if no track has that identifier.</p>
   </dd>

   <dt><code data-x=""><var>audioTrack</var>.<span subdfn data-x="dom-AudioTrack-id">id</span></code></dt>
   <dt><code data-x=""><var>videoTrack</var>.<span subdfn data-x="dom-VideoTrack-id">id</span></code></dt>

   <dd>
    <p>Returns the ID of the given track. This is the ID that can be used with a <span
    data-x="concept-url-fragment">fragment</span> if the format supports <span>media fragment
    syntax</span>, and that can be used with the <code data-x="">getTrackById()</code> method.</p>
   </dd>

   <dt><code data-x=""><var>audioTrack</var>.<span subdfn data-x="dom-AudioTrack-kind">kind</span></code></dt>
   <dt><code data-x=""><var>videoTrack</var>.<span subdfn data-x="dom-VideoTrack-kind">kind</span></code></dt>

   <dd><p>Returns the category the given track falls into. The <a
   href="#dom-TrackList-getKind-categories">possible track categories</a> are given below.</p></dd>

   <dt><code data-x=""><var>audioTrack</var>.<span subdfn data-x="dom-AudioTrack-label">label</span></code></dt>
   <dt><code data-x=""><var>videoTrack</var>.<span subdfn data-x="dom-VideoTrack-label">label</span></code></dt>

   <dd><p>Returns the label of the given track, if known, or the empty string otherwise.</p></dd>

   <dt><code data-x=""><var>audioTrack</var>.<span subdfn data-x="dom-AudioTrack-language">language</span></code></dt>
   <dt><code data-x=""><var>videoTrack</var>.<span subdfn data-x="dom-VideoTrack-language">language</span></code></dt>

   <dd><p>Returns the language of the given track, if known, or the empty string otherwise.</p></dd>

   <dt><code data-x=""><var>audioTrack</var>.<span subdfn data-x="dom-AudioTrack-enabled">enabled</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns true if the given track is active, and false otherwise.</p>

    <p>Can be set, to change whether the track is enabled or not. If multiple audio tracks are
    enabled simultaneously, they are mixed.</p>
   </dd>

   <dt><code data-x=""><var>media</var>.<span data-x="dom-media-videoTracks">videoTracks</span>.<span subdfn data-x="dom-VideoTrackList-selectedIndex">selectedIndex</span></code></dt>

   <dd><p>Returns the index of the currently selected track, if any, or &#x2212;1
   otherwise.</p></dd>

   <dt><code data-x=""><var>videoTrack</var>.<span subdfn data-x="dom-VideoTrack-selected">selected</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns true if the given track is active, and false otherwise.</p>

    <p>Can be set, to change whether the track is selected or not. Either zero or one video track is
    selected; selecting a new track while a previous one is selected will unselect the previous
    one.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>An <code>AudioTrackList</code> object represents a dynamic list of zero or more audio tracks,
  of which zero or more can be enabled at a time. Each audio track is represented by an
  <code>AudioTrack</code> object.</p>

  <p>A <code>VideoTrackList</code> object represents a dynamic list of zero or more video tracks, of
  which zero or one can be selected at a time. Each video track is represented by a
  <code>VideoTrack</code> object.</p>

  <p>Tracks in <code>AudioTrackList</code> and <code>VideoTrackList</code> objects must be
  consistently ordered. If the <span>media resource</span> is in a format that defines an order,
  then that order must be used; otherwise, the order must be the relative order in which the tracks
  are declared in the <span>media resource</span>. The order used is called the <i>natural order</i>
  of the list.</p>

  <p class="note">Each track in one of these objects thus has an index; the first has the index
  0, and each subsequent track is numbered one higher than the previous one. If a <span>media
  resource</span> dynamically adds or removes audio or video tracks, then the indices of the tracks
  will change dynamically. If the <span>media resource</span> changes entirely, then all the
  previous tracks will be removed and replaced with new tracks.</p>

  <p>The <code>AudioTrackList</code> <dfn attribute for="AudioTrackList"><code
  data-x="dom-AudioTrackList-length">length</code></dfn> and <code>VideoTrackList</code>
  <dfn attribute for="VideoTrackList"><code data-x="dom-VideoTrackList-length">length</code></dfn>
  attribute getters must return the number of tracks represented by their objects at the time of
  getting.</p>

  <p>The <span>supported property indices</span> of <code>AudioTrackList</code> and
  <code>VideoTrackList</code> objects at any instant are the numbers from zero to the number of
  tracks represented by the respective object minus one, if any tracks are represented. If an
  <code>AudioTrackList</code> or <code>VideoTrackList</code> object represents no tracks, it has no
  <span>supported property indices</span>.</p>

  <p id="dom-tracklist-item">To <span>determine the value of an indexed property</span> for a
  given index <var>index</var> in an <code>AudioTrackList</code> or <code>VideoTrackList</code>
  object <var>list</var>, the user agent must return the <code>AudioTrack</code> or
  <code>VideoTrack</code> object that represents the <var>index</var>th track in <var>list</var>.</p>

  <p>The <code>AudioTrackList</code> <dfn method for="AudioTrackList"><code
  data-x="dom-AudioTrackList-getTrackById">getTrackById(<var>id</var>)</code></dfn> and
  <code>VideoTrackList</code> <dfn method for="VideoTrackList"><code
  data-x="dom-VideoTrackList-getTrackById">getTrackById(<var>id</var>)</code></dfn> methods must
  return the first <code>AudioTrack</code> or <code>VideoTrack</code> object (respectively) in the
  <code>AudioTrackList</code> or <code>VideoTrackList</code> object (respectively) whose identifier
  is equal to the value of the <var>id</var> argument (in the natural order of the list, as defined
  above). When no tracks match the given argument, the methods must return null.</p>

  <p>The <code>AudioTrack</code> and <code>VideoTrack</code> objects represent specific tracks of a
  <span>media resource</span>. Each track can have an identifier, category, label, and language.
  These aspects of a track are permanent for the lifetime of the track; even if a track is removed
  from a <span>media resource</span>'s <code>AudioTrackList</code> or <code>VideoTrackList</code>
  objects, those aspects do not change.</p>

  <p>In addition, <code>AudioTrack</code> objects can each be enabled or disabled; this is the audio
  track's <i>enabled state</i>. When an <code>AudioTrack</code> is created, its <i>enabled state</i>
  must be set to false (disabled). The <span data-x="concept-media-load-resource">resource fetch
  algorithm</span> can override this.</p>

  <p>Similarly, a single <code>VideoTrack</code> object per <code>VideoTrackList</code> object can
  be selected, this is the video track's <i>selection state</i>. When a <code>VideoTrack</code> is
  created, its <i>selection state</i> must be set to false (not selected). The <span
  data-x="concept-media-load-resource">resource fetch algorithm</span> can override this.</p>

  <p>The <code>AudioTrack</code> <dfn attribute for="AudioTrack"><code
  data-x="dom-AudioTrack-id">id</code></dfn> and <code>VideoTrack</code> <dfn attribute
  for="VideoTrack"><code data-x="dom-VideoTrack-id">id</code></dfn> attributes must return the
  identifier of the track, if it has one, or the empty string otherwise. If the <span>media
  resource</span> is in a format that supports <span>media fragment syntax</span>, the identifier
  returned for a particular track must be the same identifier that would enable the track if used as
  the name of a track in the track dimension of such a <span
  data-x="concept-url-fragment">fragment</span>. <ref>INBAND</ref></p>

  <p class="example">For example, in Ogg files, this would be the Name header field of the track.
  <ref>OGGSKELETONHEADERS</ref></p>

  <p>The <code>AudioTrack</code> <dfn attribute for="AudioTrack"><code
  data-x="dom-AudioTrack-kind">kind</code></dfn> and <code>VideoTrack</code> <dfn attribute
  for="VideoTrack"><code data-x="dom-VideoTrack-kind">kind</code></dfn> attributes must return the
  category of the track, if it has one, or the empty string otherwise.</p>

  <p>The category of a track is the string given in the first column of the table below that is the
  most appropriate for the track based on the definitions in the table's second and third columns,
  as determined by the metadata included in the track in the <span>media resource</span>. The cell
  in the third column of a row says what the category given in the cell in the first column of that
  row applies to; a category is only appropriate for an audio track if it applies to audio tracks,
  and a category is only appropriate for video tracks if it applies to video tracks. Categories must
  only be returned for <code>AudioTrack</code> objects if they are appropriate for audio, and must
  only be returned for <code>VideoTrack</code> objects if they are appropriate for video.</p>

  <p>For Ogg files, the Role header field of the track gives the relevant metadata. For DASH media
  resources, the <code data-x="">Role</code> element conveys the information. For WebM, only the
  <code data-x="">FlagDefault</code> element currently maps to a value. <cite>Sourcing In-band
  Media Resource Tracks from Media Containers into HTML</cite> has further details.
  <ref>OGGSKELETONHEADERS</ref> <ref>DASH</ref> <ref>WEBMCG</ref> <ref>INBAND</ref></p>

  </div>

  <table id="dom-TrackList-getKind-categories">
   <caption>Return values for <code>AudioTrack</code>'s <code data-x="dom-AudioTrack-kind">kind</code> and <code>VideoTrack</code>'s <code data-x="dom-VideoTrack-kind">kind</code></caption>
   <thead>
    <tr>
     <th>Category
     <th>Definition
     <th>Applies to...</th>
     <th>Examples
   <tbody>

    <tr>
     <td>"<dfn><code data-x="value-track-kind-alternate">alternative</code></dfn>"
     <td>A possible alternative to the main track, e.g. a different take of a song (audio), or a different angle (video).
     <td>Audio and video.
     <td>Ogg: "audio/alternate" or "video/alternate"; DASH: "alternate" without "main" and "commentary" roles, and, for audio, without the "dub" role (other roles ignored).

    <tr>
     <td>"<dfn><code data-x="value-track-kind-caption">captions</code></dfn>"
     <td>A version of the main video track with captions burnt in. (For legacy content; new content would use text tracks.)
     <td>Video only.
     <td>DASH: "caption" and "main" roles together (other roles ignored).

    <tr>
     <td>"<dfn><code data-x="value-track-kind-descriptions">descriptions</code></dfn>"
     <td>An audio description of a video track.
     <td>Audio only.
     <td>Ogg: "audio/audiodesc".

    <tr>
     <td>"<dfn><code data-x="value-track-kind-main">main</code></dfn>"
     <td>The primary audio or video track.
     <td>Audio and video.
     <td>Ogg: "audio/main" or "video/main"; WebM: the "FlagDefault" element is set; DASH: "main" role without "caption", "subtitle", and "dub" roles (other roles ignored).

    <tr>
     <td>"<dfn><code data-x="value-track-kind-main-desc">main-desc</code></dfn>"
     <td>The primary audio track, mixed with audio descriptions.
     <td>Audio only.
     <td>AC3 audio in MPEG-2 TS: bsmod=2 and full_svc=1. <!-- see https://www.w3.org/Bugs/Public/show_bug.cgi?id=17797#c1 -->

    <tr>
     <td>"<dfn><code data-x="value-track-kind-sign">sign</code></dfn>"
     <td>A sign-language interpretation of an audio track.
     <td>Video only.
     <td>Ogg: "video/sign".

    <tr>
     <td>"<dfn><code data-x="value-track-kind-subtitle">subtitles</code></dfn>"
     <td>A version of the main video track with subtitles burnt in. (For legacy content; new content would use text tracks.)
     <td>Video only.
     <td>DASH: "subtitle" and "main" roles together (other roles ignored).

    <tr>
     <td>"<dfn><code data-x="value-track-kind-translation">translation</code></dfn>"
     <td>A translated version of the main audio track.
     <td>Audio only.
     <td>Ogg: "audio/dub". DASH: "dub" and "main" roles together (other roles ignored).

    <tr>
     <td>"<dfn><code data-x="value-track-kind-commentary">commentary</code></dfn>"
     <td>Commentary on the primary audio or video track, e.g. a director's commentary.
     <td>Audio and video.
     <td>DASH: "commentary" role without "main" role (other roles ignored).

    <tr>
     <td>"<dfn><code data-x="value-track-kind-none"></code></dfn>" (empty string)
     <td>No explicit kind, or the kind given by the track's metadata is not recognized by the user agent.
     <td>Audio and video.
     <td>

  </table>

  <div w-nodev>

  <p>The <code>AudioTrack</code> <dfn attribute for="AudioTrack"><code
  data-x="dom-AudioTrack-label">label</code></dfn> and <code>VideoTrack</code> <dfn attribute
  for="VideoTrack"><code data-x="dom-VideoTrack-label">label</code></dfn> attributes must return the
  label of the track, if it has one, or the empty string otherwise. <ref>INBAND</ref></p>

  <p>The <code>AudioTrack</code> <dfn attribute for="AudioTrack"><code
  data-x="dom-AudioTrack-language">language</code></dfn> and <code>VideoTrack</code> <dfn attribute
  for="VideoTrack"><code data-x="dom-VideoTrack-language">language</code></dfn> attributes must
  return the BCP 47 language tag of the language of the track, if it has one, or the empty string
  otherwise. If the user agent is not able to express that language as a BCP 47 language tag (for
  example because the language information in the <span>media resource</span>'s format is a
  free-form string without a defined interpretation), then the method must return the empty string,
  as if the track had no language. <ref>INBAND</ref></p>

  <p>The <code>AudioTrack</code> <dfn attribute for="AudioTrack"><code
  data-x="dom-AudioTrack-enabled">enabled</code></dfn> attribute, on getting, must return true if
  the track is currently enabled, and false otherwise. On setting, it must enable the track if the
  new value is true, and disable it otherwise. (If the track is no longer in an
  <code>AudioTrackList</code> object, then the track being enabled or disabled has no effect beyond
  changing the value of the attribute on the <code>AudioTrack</code> object.)</p>

  <p id="toggle-audio-track">Whenever an audio track in an <code>AudioTrackList</code> that was
  disabled is enabled, and whenever one that was enabled is disabled, the user agent must
  <span>queue a media element task</span> given the <span>media element</span> to <span
  data-x="concept-event-fire">fire an event</span> named <code
  data-x="event-media-change">change</code> at the <code>AudioTrackList</code> object.</p>

  <p>An audio track that has no data for a particular position on the <span>media timeline</span>,
  or that does not exist at that position, must be interpreted as being silent at that point on the
  timeline.</p>

  <p>The <code>VideoTrackList</code> <dfn attribute for="VideoTrackList"><code
  data-x="dom-VideoTrackList-selectedIndex">selectedIndex</code></dfn> attribute must return the
  index of the currently selected track, if any. If the <code>VideoTrackList</code> object does not
  currently represent any tracks, or if none of the tracks are selected, it must instead return
  &#x2212;1.</p>

  <p>The <code>VideoTrack</code> <dfn attribute for="VideoTrack"><code
  data-x="dom-VideoTrack-selected">selected</code></dfn> attribute, on getting, must return true if
  the track is currently selected, and false otherwise.  On setting, it must select the track if the
  new value is true, and unselect it otherwise. If the track is in a <code>VideoTrackList</code>,
  then all the other <code>VideoTrack</code> objects in that list must be unselected. (If the track
  is no longer in a <code>VideoTrackList</code> object, then the track being selected or unselected
  has no effect beyond changing the value of the attribute on the <code>VideoTrack</code>
  object.)</p>

  <p id="toggle-video-track">Whenever a track in a <code>VideoTrackList</code> that was previously
  not selected is selected, and whenever the selected track in a <code>VideoTrackList</code> is
  unselected without a new track being selected in its stead, the user agent must <span>queue a
  media element task</span> given the <span>media element</span> to <span
  data-x="concept-event-fire">fire an event</span> named <code
  data-x="event-media-change">change</code> at the <code>VideoTrackList</code> object. This <span
  data-x="concept-task">task</span> must be <span data-x="queue an element task">queued</span>
  before the <span data-x="concept-task">task</span> that fires the <code
  data-x="event-media-resize">resize</code> event, if any.</p>

  <p>A video track that has no data for a particular position on the <span>media timeline</span>
  must be interpreted as being <span>transparent black</span> at that point on the timeline, with
  the same dimensions as the last frame before that position, or, if the position is before all the
  data for that track, the same dimensions as the first frame for that track. A track that does not
  exist at all at the current position must be treated as if it existed but had no data.</p>

  <p class="example">For instance, if a video has a track that is only introduced after one hour of
  playback, and the user selects that track then goes back to the start, then the user agent will
  act as if that track started at the start of the <span>media resource</span> but was simply
  transparent until one hour in.</p>

  <hr>

  <p>The following are the <span>event handlers</span> (and their corresponding <span data-x="event
  handler event type">event handler event types</span>) <span w-nodev>that must be</span> supported, as <span>event handler IDL attributes</span>,
  by all objects implementing the <code>AudioTrackList</code> and <code>VideoTrackList</code>
  interfaces:</p>

  <table>
   <thead>
    <tr><th><span data-x="event handlers">Event handler</span> <th><span>Event handler event type</span>
   <tbody>
    <tr><td><dfn attribute for="AudioTrackList,VideoTrackList"><code data-x="handler-TrackList-onchange">onchange</code></dfn> <td> <code data-x="event-media-change">change</code>
    <tr><td><dfn attribute for="AudioTrackList,VideoTrackList"><code data-x="handler-TrackList-onaddtrack">onaddtrack</code></dfn> <td> <code data-x="event-media-addtrack">addtrack</code>
    <tr><td><dfn attribute for="AudioTrackList,VideoTrackList"><code data-x="handler-TrackList-onremovetrack">onremovetrack</code></dfn> <td> <code data-x="event-media-removetrack">removetrack</code>
  </table>

  <!-- v2 should also fire an event when the list changes (but note that that should be on the media element event task source) -->

  </div>


  <h6>Selecting specific audio and video tracks declaratively</h6>

  <p>The <code data-x="dom-media-audioTracks">audioTracks</code> and <code
  data-x="dom-media-videoTracks">videoTracks</code> attributes allow scripts to select which track
  should play, but it is also possible to select specific tracks declaratively, by specifying
  particular tracks in the <span data-x="concept-url-fragment">fragment</span> of the
  <span>URL</span> of the <span>media resource</span>. The format of the <span
  data-x="concept-url-fragment">fragment</span> depends on the <span>MIME type</span> of the
  <span>media resource</span>. <ref>RFC2046</ref> <ref>URL</ref></p>

  <div class="example">

   <p>In this example, a video that uses a format that supports <span>media fragment syntax</span>
   is embedded in such a way that the alternative angles labeled "Alternative" are enabled instead
   of the default video track.</p>

   <pre><code class="html">&lt;video src="myvideo#track=Alternative">&lt;/video></code></pre>

  </div>



  <h5>Timed text tracks</h5>

  <h6>Text track model</h6>

  <p>A <span>media element</span> can have a group of associated <dfn data-x="text track">text
  tracks</dfn>, known as the <span>media element</span>'s <dfn>list of text tracks</dfn>. The <span
  data-x="text track">text tracks</span> are sorted as follows:</p>

  <ol>
   <li><p>The <span data-x="text track">text tracks</span> corresponding to <code>track</code>
   element children of the <span>media element</span>, in <span>tree order</span>.</p></li>

   <li><p>Any <span data-x="text track">text tracks</span> added using the <code
   data-x="dom-media-addTextTrack">addTextTrack()</code> method, in the order they were added,
   oldest first.</p></li>

   <li><p>Any <span data-x="media-resource-specific text track">media-resource-specific text
   tracks</span> (<span data-x="text track">text tracks</span> corresponding to data in the
   <span>media resource</span>), in the order defined by the <span>media resource</span>'s format
   specification.</p></li>
  </ol>

  <p>A <span>text track</span> consists of:</p>

  <dl>
   <dt><dfn data-x="text track kind">The kind of text track</dfn>

   <dd>
    <p>This decides how the track is handled by the user agent. The kind is represented by a string.
    The possible strings are:</p>

    <ul class="brief">
     <li><dfn><code data-x="dom-TextTrack-kind-subtitles">subtitles</code></dfn>
     <li><dfn><code data-x="dom-TextTrack-kind-captions">captions</code></dfn>
     <li><dfn><code data-x="dom-TextTrack-kind-descriptions">descriptions</code></dfn>
     <li><dfn><code data-x="dom-TextTrack-kind-chapters">chapters</code></dfn>
     <li><dfn><code data-x="dom-TextTrack-kind-metadata">metadata</code></dfn>
    </ul>

    <p>The <span data-x="text track kind">kind of track</span> can change dynamically, in the case of
    a <span>text track</span> corresponding to a <code>track</code> element.</p>
   </dd>

   <dt><dfn data-x="text track label">A label</dfn>

   <dd>
    <p>This is a human-readable string intended to identify the track for the user.</p>

    <p>The <span data-x="text track label">label of a track</span> can change dynamically, in the
    case of a <span>text track</span> corresponding to a <code>track</code> element.</p>

    <p>When a <span>text track label</span> is the empty string, the user agent should automatically
    generate an appropriate label from the text track's other properties (e.g. the kind of text
    track and the text track's language) for use in its user interface. This automatically-generated
    label is not exposed in the API.</p>
   </dd>

   <dt><dfn data-x="text track in-band metadata track dispatch type">An in-band metadata track dispatch type</dfn>

   <dd>
    <p>This is a string extracted from the <span>media resource</span> specifically for in-band
    metadata tracks to enable such tracks to be dispatched to different scripts in the document.</p>

    <p class="example">For example, a traditional TV station broadcast streamed on the web and
    augmented with web-specific interactive features could include text tracks with metadata for ad
    targeting, trivia game data during game shows, player states during sports games, recipe
    information during food programs, and so forth. As each program starts and ends, new tracks
    might be added or removed from the stream, and as each one is added, the user agent could bind
    them to dedicated script modules using the value of this attribute.</p>

    <p>Other than for in-band metadata text tracks, the <span data-x="text track in-band metadata
    track dispatch type">in-band metadata track dispatch type</span> is the empty string. How this
    value is populated for different media formats is described in <span>steps to expose a
    media-resource-specific text track</span>.</p>
   </dd>

   <dt><dfn data-x="text track language">A language</dfn>

   <dd>
    <p>This is a string (a BCP 47 language tag) representing the language of the text track's cues.
    <ref>BCP47</ref></p>

    <p>The <span data-x="text track language">language of a text track</span> can change dynamically,
    in the case of a <span>text track</span> corresponding to a <code>track</code> element.</p>
   </dd>

   <dt><dfn data-x="text track readiness state">A readiness state</dfn>

   <dd>
    <p>One of the following:</p>

    <dl>
     <dt><dfn data-x="text track not loaded">Not loaded</dfn>

     <dd>
      <p>Indicates that the text track's cues have not been obtained.</p>
     </dd>

     <dt><dfn data-x="text track loading">Loading</dfn>

     <dd>
      <p>Indicates that the text track is loading and there have been no fatal errors encountered so
      far. Further cues might still be added to the track by the parser.</p>
     </dd>

     <dt><dfn data-x="text track loaded">Loaded</dfn>

     <dd>
      <p>Indicates that the text track has been loaded with no fatal errors.</p>
     </dd>

     <dt><dfn data-x="text track failed to load">Failed to load</dfn>

     <dd>
      <p>Indicates that the text track was enabled, but when the user agent attempted to obtain it,
      this failed in some way (e.g., <span>URL</span> could not be <span data-x="encoding-parsing a
      URL">parsed</span>, network error, unknown text track format). Some or all of the cues are
      likely missing and will not be obtained.</p>
     </dd>
    </dl>

    <p>The <span data-x="text track readiness state">readiness state</span> of a <span>text
    track</span> changes dynamically as the track is obtained.</p>
   </dd>

   <dt><dfn data-x="text track mode">A mode</dfn>

   <dd>
    <p>One of the following:</p>

    <dl>
     <dt><dfn data-x="text track disabled">Disabled</dfn>

     <dd>
      <p>Indicates that the text track is not active. Other than for the purposes of exposing the
      track in the DOM, the user agent is ignoring the text track. No cues are active, no events are
      fired, and the user agent will not attempt to obtain the track's cues.</p>
     </dd>

     <dt><dfn data-x="text track hidden">Hidden</dfn>

     <dd>
      <p>Indicates that the text track is active, but that the user agent is not actively displaying
      the cues. If no attempt has yet been made to obtain the track's cues, the user agent will
      perform such an attempt momentarily. The user agent is maintaining a list of which cues are
      active, and events are being fired accordingly.</p>
     </dd>

     <dt><dfn data-x="text track showing">Showing</dfn>

     <dd>
      <p>Indicates that the text track is active. If no attempt has yet been made to obtain the
      track's cues, the user agent will perform such an attempt momentarily. The user agent is
      maintaining a list of which cues are active, and events are being fired accordingly. In
      addition, for text tracks whose <span data-x="text track kind">kind</span> is <code
      data-x="dom-TextTrack-kind-subtitles">subtitles</code> or <code
      data-x="dom-TextTrack-kind-captions">captions</code>, the cues are being overlaid on the video
      as appropriate; for text tracks whose <span data-x="text track kind">kind</span> is <code
      data-x="dom-TextTrack-kind-descriptions">descriptions</code>, the user agent is making the
      cues available to the user in a non-visual fashion; and for text tracks whose <span
      data-x="text track kind">kind</span> is <code
      data-x="dom-TextTrack-kind-chapters">chapters</code>, the user agent is making available to
      the user a mechanism by which the user can navigate to any point in the <span>media
      resource</span> by selecting a cue.</p>
     </dd>
    </dl>
   </dd>

   <dt><dfn data-x="text track list of cues">A list of zero or more cues</dfn>

   <dd>
    <p>A list of <span data-x="text track cue">text track cues</span>, along with <dfn>rules for
    updating the text track rendering</dfn>. For example, for WebVTT, the <span>rules for updating
    the display of WebVTT text tracks</span>. <ref>WEBVTT</ref> </p>

    <p>The <span data-x="text track list of cues">list of cues of a text track</span> can change
    dynamically, either because the <span>text track</span> has <span data-x="text track not
    loaded">not yet been loaded</span> or is still <span data-x="text track loading">loading</span>,
    or due to DOM manipulation.</p>
   </dd>
  </dl>

  <p>Each <span>text track</span> has a corresponding <code>TextTrack</code> object.</p>

  <hr>

  <p>Each <span>media element</span> has a <dfn>list of pending text tracks</dfn>, which must
  initially be empty, a <dfn>blocked-on-parser</dfn> flag, which must initially be false, and a
  <dfn>did-perform-automatic-track-selection</dfn> flag, which must also initially be false.</p>

  <p>When the user agent is required to <dfn>populate the list of pending text tracks</dfn> of a
  <span>media element</span>, the user agent must add to the element's <span>list of pending text
  tracks</span> each <span>text track</span> in the element's <span>list of text tracks</span> whose
  <span>text track mode</span> is not <span data-x="text track disabled">disabled</span> and whose
  <span>text track readiness state</span> is <!--either <span data-x="text track not loaded">not
  loaded</span> or [there can't be any in the 'not loaded' state that are not 'disabled']--> <span
  data-x="text track loading">loading</span>.</p>

  <p>Whenever a <code>track</code> element's parent node changes, the user agent must remove the
  corresponding <span>text track</span> from any <span>list of pending text tracks</span> that it is
  in.</p>

  <p>Whenever a <span>text track</span>'s <span>text track readiness state</span> changes to either
  <span data-x="text track loaded">loaded</span> or <span data-x="text track failed to load">failed to
  load</span>, the user agent must remove it from any <span>list of pending text tracks</span> that
  it is in.</p>

  <p>When a <span>media element</span> is created by an <span>HTML parser</span> or <span>XML
  parser</span>, the user agent must set the element's <span>blocked-on-parser</span> flag to true.
  When a <span>media element</span> is popped off the <span>stack of open elements</span> of an
  <span>HTML parser</span> or <span>XML parser</span>, the user agent must <span>honor user
  preferences for automatic text track selection</span>, <span>populate the list of pending text
  tracks</span>, and set the element's <span>blocked-on-parser</span> flag to false.</p>

  <p>The <span data-x="text track">text tracks</span> of a <span>media element</span> are <dfn
  data-x="the text tracks are ready">ready</dfn> when both the element's <span>list of pending text
  tracks</span> is empty and the element's <span>blocked-on-parser</span> flag is false.</p>

  <p>Each <span>media element</span> has a <dfn>pending text track change notification flag</dfn>,
  which must initially be unset.</p>

  <p>Whenever a <span>text track</span> that is in a <span>media element</span>'s <span>list of text
  tracks</span> has its <span>text track mode</span> change value, the user agent must run the
  following steps for the <span>media element</span>:</p>

  <ol>
   <li><p>If the <span>media element</span>'s <span>pending text track change notification
   flag</span> is set, return.</p></li>

   <li><p>Set the <span>media element</span>'s <span>pending text track change notification
   flag</span>.</p></li>

   <li>
    <p><span>Queue a media element task</span> given the <span>media element</span> to run these
    steps:</p>

    <ol>
     <li><p>Unset the <span>media element</span>'s <span>pending text track change notification
     flag</span>.</p></li>

     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-media-change">change</code> at the <span>media element</span>'s <code
     data-x="dom-media-textTracks">textTracks</code> attribute's <code>TextTrackList</code>
     object.</p></li>
    </ol>
   </li>

   <li><p>If the <span>media element</span>'s <span>show poster flag</span> is not set, run the
   <i data-x="time marches on">time marches on</i> steps.</p></li>
  </ol>

  <p>The <span>task source</span> for the <span data-x="concept-task">tasks</span> listed in this
  section is the <span>DOM manipulation task source</span>.</p>

  <hr>

  <p>A <dfn>text track cue</dfn> is the unit of time-sensitive data in a <span>text track</span>,
  corresponding for instance for subtitles and captions to the text that appears at a particular
  time and disappears at another time.</p>

  <p>Each <span>text track cue</span> consists of:</p>

  <dl>
   <dt><dfn data-x="text track cue identifier">An identifier</dfn>
   <dd>
    <p>An arbitrary string.</p>
   </dd>

   <dt><dfn data-x="text track cue start time">A start time</dfn>
   <dd>
    <p>The time, in seconds and fractions of a second, that describes the beginning of the range of
    the <span>media data</span> to which the cue applies.</p>
   </dd>

   <dt><dfn data-x="text track cue end time">An end time</dfn>
   <dd>
    <p>The time, in seconds and fractions of a second, that describes the end of the range of the
    <span>media data</span> to which the cue applies, or positive Infinity for an <span>unbounded
    text track cue</span>.</p>
   </dd>

   <dt><dfn data-x="text track cue pause-on-exit flag">A pause-on-exit flag</dfn>
   <dd>
    <p>A boolean indicating whether playback of the <span>media resource</span> is to pause when the
    end of the range to which the cue applies is reached.</p>
   </dd>

   <dt>Some additional format-specific data</dt>
   <dd>
    <p>Additional fields, as needed for the format, including the actual data of the cue. For
    example, WebVTT has a <span>text track cue writing direction</span> and so forth.
    <ref>WEBVTT</ref></p>
   </dd>
  </dl>

  <p>An <dfn data-x="unbounded text track cue">unbounded text track cue</dfn> is a text track cue
  with a <span>text track cue end time</span> set to positive Infinity. An active <span>unbounded
  text track cue</span> cannot become inactive through the usual monotonic increase of the
  <span>current playback position</span> during normal playback (e.g. a metadata cue for a chapter in a live
  event with no announced end time.)</p>

  <p class="note">The <span>text track cue start time</span> and <span>text track cue end
  time</span> can be negative. (The <span>current playback position</span> can never be negative,
  though, so cues entirely before time zero cannot be active.)</p>

  <p>Each <span>text track cue</span> has a corresponding <code>TextTrackCue</code> object (or more
  specifically, an object that inherits from <code>TextTrackCue</code> &mdash; for example, WebVTT
  cues use the <code>VTTCue</code> interface). A <span>text track cue</span>'s in-memory
  representation can be dynamically changed through this <code>TextTrackCue</code> API.
  <ref>WEBVTT</ref></p>

  <p>A <span>text track cue</span> is associated with <span>rules for updating the text track
  rendering</span>, as defined by the specification for the specific kind of <span>text track
  cue</span>. These rules are used specifically when the object representing the cue is added to a
  <code>TextTrack</code> object using the <code data-x="dom-TextTrack-addCue">addCue()</code>
  method.</p>

  <p>In addition, each <span>text track cue</span> has two pieces of dynamic information:</p>

  <dl>
   <dt>The <dfn data-x="text track cue active flag">active flag</dfn>
   <dd>
    <p>This flag must be initially unset. The flag is used to ensure events are fired appropriately
    when the cue becomes active or inactive, and to make sure the right cues are rendered.</p>

    <p>The user agent must synchronously unset this flag whenever the <span>text track cue</span> is
    removed from its <span>text track</span>'s <span>text track list of cues</span>; whenever the
    <span>text track</span> itself is removed from its <span>media element</span>'s <span>list of
    text tracks</span> or has its <span>text track mode</span> changed to <span data-x="text track
    disabled">disabled</span>; and whenever the <span>media element</span>'s <code
    data-x="dom-media-readyState">readyState</code> is changed back to <code
    data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</code>. When the flag is unset in this way for one
    or more cues in <span data-x="text track">text tracks</span> that were <span data-x="text track
    showing">showing</span> prior to the relevant incident, the user agent must, after having unset
    the flag for all the affected cues, apply the <span>rules for updating the text track
    rendering</span> of those <span data-x="text track">text tracks</span>. For example, for <span
    data-x="text track">text tracks</span> based on WebVTT, the <span>rules for updating the display
    of WebVTT text tracks</span>. <ref>WEBVTT</ref></p> </dd>

   <dt>The <dfn data-x="text track cue display state">display state</dfn>
   <dd>
    <p>This is used as part of the rendering model, to keep cues in a consistent position. It must
    initially be empty. Whenever the <span>text track cue active flag</span> is unset, the user
    agent must empty the <span>text track cue display state</span>.</p>
   </dd>
  </dl>

  <p>The <span data-x="text track cue">text track cues</span> of a <span>media element</span>'s
  <span data-x="text track">text tracks</span> are ordered relative to each other in the <dfn>text
  track cue order</dfn>, which is determined as follows: first group the <span data-x="text track
  cue">cues</span> by their <span>text track</span>, with the groups being sorted in the same order
  as their <span data-x="text track">text tracks</span> appear in the <span>media element</span>'s
  <span>list of text tracks</span>; then, within each group, <span data-x="text track
  cue">cues</span> must be sorted by their <span data-x="text track cue start time">start
  time</span>, earliest first; then, any <span data-x="text track cue">cues</span> with the same
  <span data-x="text track cue start time">start time</span> must be sorted by their <span
  data-x="text track cue end time">end time</span>, latest first; and finally, any <span
  data-x="text track cue">cues</span> with identical <span data-x="text track cue end time">end
  times</span> must be sorted in the order they were last added to their respective <span>text track
  list of cues</span>, oldest first (so e.g. for cues from a WebVTT file, that would initially be
  the order in which the cues were listed in the file). <ref>WEBVTT</ref></p>


  <h6>Sourcing in-band text tracks</h6>

  <p>A <dfn>media-resource-specific text track</dfn> is a <span>text track</span> that corresponds
  to data found in the <span>media resource</span>.</p>


  <div w-nodev>

  <p>Rules for processing and rendering such data are defined by the relevant specifications, e.g.
  the specification of the video format if the <span>media resource</span> is a video. Details for
  some legacy formats can be found in <cite>Sourcing In-band Media Resource Tracks from Media
  Containers into HTML</cite>. <ref>INBAND</ref></p>

  <p>When a <span>media resource</span> contains data that the user agent recognizes and supports as
  being equivalent to a <span>text track</span>, the user agent <a
  href="#found-a-media-resource-specific-timed-track">runs</a> the <dfn>steps to expose a
  media-resource-specific text track</dfn> with the relevant data, as follows.</p>

  <!-- this runs synchronously from a fetch task -->

  <ol>
   <li><p>Associate the relevant data with a new <span>text track</span> and its corresponding new
   <code>TextTrack</code> object. The <span>text track</span> is a <span>media-resource-specific
   text track</span>.</p></li>

   <li><p>Set the new <span>text track</span>'s <span data-x="text track kind">kind</span>, <span
   data-x="text track label">label</span>, and <span data-x="text track language">language</span>
   based on the semantics of the relevant data, as defined by the relevant specification. If there
   is no label in that data, then the <span data-x="text track label">label</span> must be set to the
   empty string.</p></li>

   <li><p>Associate the <span>text track list of cues</span> with the <span>rules for updating the
   text track rendering</span> appropriate for the format in question.</p>

   <li>
    <p>If the new <span>text track</span>'s <span data-x="text track kind">kind</span> is <code
    data-x="dom-TextTrack-kind-chapters">chapters</code> or <code
    data-x="dom-TextTrack-kind-metadata">metadata</code>, then set the <span>text track in-band
    metadata track dispatch type</span> as follows, based on the type of the <span>media
    resource</span>:</p>

    <dl class="switch">

     <dt>If the <span>media resource</span> is an Ogg file</dt>

     <dd>The <span>text track in-band metadata track dispatch type</span> must be set to the value
     of the Name header field. <ref>OGGSKELETONHEADERS</ref></dd>

     <dt>If the <span>media resource</span> is a WebM file</dt>

     <dd>The <span>text track in-band metadata track dispatch type</span> must be set to the value
     of the <code data-x="">CodecID</code> element. <ref>WEBMCG</ref></dd>

     <dt>If the <span>media resource</span> is an MPEG-2 file</dt>

     <dd>Let <var>stream type</var> be the value of the "stream_type" field describing the text
     track's type in the file's program map section, interpreted as an 8-bit unsigned integer. Let
     <var>length</var> be the value of the "ES_info_length" field for the track in the same part of
     the program map section, interpreted as an integer as defined by <cite>Generic coding of moving
     pictures and associated audio information</cite>. Let <var>descriptor bytes</var> be the
     <var>length</var> bytes following the "ES_info_length" field. The <span>text track in-band
     metadata track dispatch type</span> must be set to the concatenation of the <var>stream
     type</var> byte and the zero or more <var>descriptor bytes</var> bytes, expressed in
     hexadecimal using <span data-x="ASCII upper hex digit">ASCII upper hex digits</span>.
     <ref>MPEG2</ref>
     </dd>

     <dt>If the <span>media resource</span> is an MPEG-4 file</dt>

     <dd>Let the
     first <code data-x="">stsd</code> box of the
     first <code data-x="">stbl</code> box of the
     first <code data-x="">minf</code> box of the
     first <code data-x="">mdia</code> box of the
     <span>text track</span>'s <code data-x="">trak</code> box in the
     first <code data-x="">moov</code> box
     of the file be the <i>stsd box</i>, if any.

     If the file has no <i>stsd box</i>, or if the <i>stsd box</i> has neither a <code
     data-x="">mett</code> box nor a <code data-x="">metx</code> box, then the <span>text track
     in-band metadata track dispatch type</span> must be set to the empty string.

     Otherwise, if the <i>stsd box</i> has a <code data-x="">mett</code> box then the <span>text
     track in-band metadata track dispatch type</span> must be set to the concatenation of the
     string "<code data-x="">mett</code>", a U+0020 SPACE character, and the value of the first <code
     data-x="">mime_format</code> field of the first <code data-x="">mett</code> box of the <i>stsd
     box</i>, or the empty string if that field is absent in that box.

     Otherwise, if the <i>stsd box</i> has no <code data-x="">mett</code> box but has a <code
     data-x="">metx</code> box then the <span>text track in-band metadata track dispatch type</span>
     must be set to the concatenation of the string "<code data-x="">metx</code>", a U+0020 SPACE
     character, and the value of the first <code data-x="">namespace</code> field of the first <code
     data-x="">metx</code> box of the <i>stsd box</i>, or the empty string if that field is absent in
     that box.

     <ref>MPEG4</ref>
     </dd>
    </dl>
   </li>

   <li><p>Populate the new <span>text track</span>'s <span data-x="text track list of cues">list of
   cues</span> with the cues parsed so far, following the <span>guidelines for exposing
   cues</span>, and begin updating it dynamically as necessary.</p></li>

   <li><p>Set the new <span>text track</span>'s <span data-x="text track readiness state">readiness
   state</span> to <span data-x="text track loaded">loaded</span>.</p></li> <!-- otherwise, you'll
   have to load the whole media file just to start playing the first frame... -->

   <li><p>Set the new <span>text track</span>'s <span data-x="text track mode">mode</span> to the
   mode consistent with the user's preferences and the requirements of the relevant specification
   for the data.</p>

    <p class="note">For instance, if there are no other active subtitles, and this is a forced
    subtitle track (a subtitle track giving subtitles in the audio track's primary language, but
    only for audio that is actually in another language), then those subtitles might be activated
    here.</p>
   </li> <!-- it's too late to apply the normal heuristic, so we don't bother -->

   <li><p>Add the new <span>text track</span> to the <span>media element</span>'s <span>list of text
   tracks</span>.</p></li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
   data-x="event-media-addtrack">addtrack</code> at the <span>media element</span>'s <code
   data-x="dom-media-textTracks">textTracks</code> attribute's <code>TextTrackList</code> object,
   using <code>TrackEvent</code>, with the <code data-x="dom-TrackEvent-track">track</code>
   attribute initialized to the <span>text track</span>'s <code>TextTrack</code> object.</p></li>
  </ol>

  <!-- removetrack: we don't currently ever remove an in-band text track, because it might have been
  manipulated or might be about to be manipulated. If we made in-band text tracks readonly, we could
  probably get away with cleaning them up when they fall into the time before the earliest possible
  position and are known to be over and done with. -->

  <!-- if, after in-band text tracks have been found, there's no out-of-band text tracks, then apply
  user prefs? -->

  </div>



  <div w-nodev>

  <h6>Sourcing out-of-band text tracks</h6>

  <p>When a <code>track</code> element is created, it must be associated with a new <span>text
  track</span> (with its value set as defined below) and its corresponding new
  <code>TextTrack</code> object.</p>

  <p>The <span>text track kind</span> is determined from the state of the element's <code
  data-x="attr-track-kind">kind</code> attribute according to the following table; for a state given
  in a cell of the first column, the <span data-x="text track kind">kind</span> is the string given
  in the second column:</p>

  <table>
   <thead>
    <tr>
     <th>State
     <th>String
   <tbody>
    <tr>
     <td><span data-x="attr-track-kind-subtitles">Subtitles</span>
     <td><code data-x="dom-TextTrack-kind-subtitles">subtitles</code>
    <tr>
     <td><span data-x="attr-track-kind-captions">Captions</span>
     <td><code data-x="dom-TextTrack-kind-captions">captions</code>
    <tr>
     <td><span data-x="attr-track-kind-descriptions">Descriptions</span>
     <td><code data-x="dom-TextTrack-kind-descriptions">descriptions</code>
    <tr>
     <td><span data-x="attr-track-kind-chapters">Chapters metadata</span>
     <td><code data-x="dom-TextTrack-kind-chapters">chapters</code>
    <tr>
     <td><span data-x="attr-track-kind-metadata">Metadata</span>
     <td><code data-x="dom-TextTrack-kind-metadata">metadata</code>
  </table>

  <p>The <span>text track label</span> is the element's <span>track label</span>.</p>

  <p>The <span>text track language</span> is the element's <span>track language</span>, if any, or
  the empty string otherwise.</p>

  <p>As the <code data-x="attr-track-kind">kind</code>, <code data-x="attr-track-label">label</code>,
  and <code data-x="attr-track-srclang">srclang</code> attributes are set, changed, or removed, the
  <span>text track</span> must update accordingly, as per the definitions above.</p>

  <p class="note">Changes to the <span>track URL</span> are handled in the algorithm below.</p>

  <p>The <span>text track readiness state</span> is initially <span data-x="text track not
  loaded">not loaded</span>, and the <span>text track mode</span> is initially <span data-x="text
  track disabled">disabled</span>.</p>

  <p>The <span>text track list of cues</span> is initially empty. It is dynamically modified when
  the referenced file is parsed. Associated with the list are the <span>rules for updating the text
  track rendering</span> appropriate for the format in question; for WebVTT, this is the <span>rules
  for updating the display of WebVTT text tracks</span>. <ref>WEBVTT</ref></p>

  <p>When a <code>track</code> element's parent element changes and the new parent is a <span>media
  element</span>, then the user agent must add the <code>track</code> element's corresponding
  <span>text track</span> to the <span>media element</span>'s <span>list of text tracks</span>, and
  then <span>queue a media element task</span> given the <span>media element</span> to <span
  data-x="concept-event-fire">fire an event</span> named <code
  data-x="event-media-addtrack">addtrack</code> at the <span>media element</span>'s <code
  data-x="dom-media-textTracks">textTracks</code> attribute's <code>TextTrackList</code> object,
  using <code>TrackEvent</code>, with the <code data-x="dom-TrackEvent-track">track</code> attribute
  initialized to the <span>text track</span>'s <code>TextTrack</code> object.</p>

  <p>When a <code>track</code> element's parent element changes and the old parent was a <span>media
  element</span>, then the user agent must remove the <code>track</code> element's corresponding
  <span>text track</span> from the <span>media element</span>'s <span>list of text tracks</span>,
  and then <span>queue a media element task</span> given the <span>media element</span> to <span
  data-x="concept-event-fire">fire an event</span> named <code
  data-x="event-media-removetrack">removetrack</code> at the <span>media element</span>'s <code
  data-x="dom-media-textTracks">textTracks</code> attribute's <code>TextTrackList</code> object,
  using <code>TrackEvent</code>, with the <code data-x="dom-TrackEvent-track">track</code> attribute
  initialized to the <span>text track</span>'s <code>TextTrack</code> object.</p>
  <!-- removetrack -->

  <hr>

  <p>When a <span>text track</span> corresponding to a <code>track</code> element is added to a
  <span>media element</span>'s <span>list of text tracks</span>, the user agent must <span>queue a
  media element task</span> given the <span>media element</span> to run the following steps for the
  <span>media element</span>:</p>

  <ol>
   <li><p>If the element's <span>blocked-on-parser</span> flag is true, then return.</p></li>

   <li><p>If the element's <span>did-perform-automatic-track-selection</span> flag is true, then
   return.</p></li>

   <li><p><span>Honor user preferences for automatic text track selection</span> for this
   element.</p></li>
  </ol>

  <p>When the user agent is required to <dfn>honor user preferences for automatic text track
  selection</dfn> for a <span>media element</span>, the user agent must run the following steps:</p>

  <ol>
   <li><p><span>Perform automatic text track selection</span> for <code
   data-x="dom-TextTrack-kind-subtitles">subtitles</code> and <code
   data-x="dom-TextTrack-kind-captions">captions</code>.</p></li>

   <li><p><span>Perform automatic text track selection</span> for <code
   data-x="dom-TextTrack-kind-descriptions">descriptions</code>.</p></li>

   <li><p>If there are any <span data-x="text track">text tracks</span> in the <span>media
   element</span>'s <span>list of text tracks</span> whose <span>text track kind</span> is <code
   data-x="dom-TextTrack-kind-metadata">chapters</code> or <code
   data-x="dom-TextTrack-kind-metadata">metadata</code> that correspond to <code>track</code>
   elements with a <code data-x="attr-track-default">default</code> attribute set whose <span>text
   track mode</span> is set to <span data-x="text track disabled">disabled</span>, then set the
   <span>text track mode</span> of all such tracks to <span data-x="text track
   hidden">hidden</span>.</p></li>

   <li><p>Set the element's <span>did-perform-automatic-track-selection</span> flag to
   true.</p></li>
  </ol>

  <p>When the steps above say to <dfn>perform automatic text track selection</dfn> for one or more
  <span data-x="text track kind">text track kinds</span>, it means to run the following steps:</p>

  <ol>
   <li><p>Let <var>candidates</var> be a list consisting of the <span data-x="text
   track">text tracks</span> in the <span>media element</span>'s <span>list of text tracks</span>
   whose <span>text track kind</span> is one of the kinds that were passed to the algorithm, if any,
   in the order given in the <span>list of text tracks</span>.</p></li>

   <li><p>If <var>candidates</var> is empty, then return.</p></li>

   <li><p>If any of the <span data-x="text track">text tracks</span> in <var>candidates</var> have a <span>text track mode</span> set to <span data-x="text track
   showing">showing</span>, return.</p></li>

   <li>
    <p>If the user has expressed an interest in having a track from <var>candidates</var>
    enabled based on its <span>text track kind</span>, <span>text track language</span>, and
    <span>text track label</span>, then set its <span>text track mode</span> to <span data-x="text
    track showing">showing</span><!--, and if there are any <span data-x="text track">text
    tracks</span> in <var>candidates</var> that correspond to <code>track</code> elements
    with a <code data-x="attr-track-default">default</code> attribute set whose <span>text track
    mode</span> is set to <span data-x="text track disabled">disabled</span>, then additionally set
    the <span>text track mode</span> of the first such track to <span data-x="text track
    hidden">hidden</span>-->.</p> <!-- the idea behind the commented out stuff is that we still get
    events and stuff: we can expect authors to forget that default doesn't mean that it'll always be
    turned on, and that they'll still rely on events firing even if it doesn't show. But it is
    commented out, because we can equally expect authors to expect only one track is getting events,
    so we've got problems either way, and might as well go with the simpler behavior. -->

    <p class="note">For example, the user could have set a browser preference to the effect of "I
    want French captions whenever possible", or "If there is a subtitle track with 'Commentary' in
    the title, enable it", or "If there are audio description tracks available, enable one, ideally
    in Swiss German, but failing that in Standard Swiss German or Standard German".</p>

    <p>Otherwise, if there are any <span data-x="text track">text tracks</span> in <var>candidates</var> that correspond to <code>track</code> elements with a <code
    data-x="attr-track-default">default</code> attribute set whose <span>text track mode</span> is
    set to <span data-x="text track disabled">disabled</span>, then set the <span>text track
    mode</span> of the first such track to <span data-x="text track showing">showing</span>.</p>
   </li>
  </ol>

  <p>When a <span>text track</span> corresponding to a <code>track</code> element experiences any of
  the following circumstances, the user agent must <span>start the <code>track</code> processing
  model</span> for that <span>text track</span> and its <code>track</code> element:

  <ul>
   <li>The <code>track</code> element is created.</li>

   <li>The <span>text track</span> has its <span>text track mode</span> changed.</li>

   <li>The <code>track</code> element's parent element changes and the new parent is a <span>media
   element</span>.</li>
  </ul>

  <p>When a user agent is to <dfn>start the <code>track</code> processing model</dfn> for a
  <span>text track</span> and its <code>track</code> element, it must run the following algorithm.
  This algorithm interacts closely with the <span>event loop</span> mechanism; in particular, it has
  a <span>synchronous section</span> (which is triggered as part of the <span>event loop</span>
  algorithm). The steps in that section are marked with &#x231B;.</p>

  <ol>
   <li><p>If another occurrence of this algorithm is already running for this <span>text
   track</span> and its <code>track</code> element, return, letting that other algorithm
   take care of this element.</p></li>

   <li><p>If the <span>text track</span>'s <span>text track mode</span> is not set to one of <span
   data-x="text track hidden">hidden</span> or <span data-x="text track showing">showing</span>,
   then return.</p></li>

   <li><p>If the <span>text track</span>'s <code>track</code> element does not have a <span>media
   element</span> as a parent, return.</p></li>

   <li><p>Run the remainder of these steps <span>in parallel</span>, allowing whatever caused these steps to
   run to continue.</p></li>

   <li><p><i>Top</i>: <span>Await a stable state</span>. The <span>synchronous section</span>
   consists of the following steps. (The steps in the <span>synchronous section</span> are marked
   with &#x231B;.)</p></li>

   <li><p>&#x231B; Set the <span>text track readiness state</span> to <span data-x="text track
   loading">loading</span>.</p></li>

   <li><p>&#x231B; Let <var>URL</var> be the <span>track URL</span> of the
   <code>track</code> element.</p></li>

   <li><p>&#x231B; If the <code>track</code> element's parent is a <span>media element</span> then
   let <var>corsAttributeState</var> be the state of the parent <span>media element</span>'s <code
   data-x="attr-media-crossorigin">crossorigin</code> content attribute. Otherwise, let
   <var>corsAttributeState</var> be <span data-x="attr-crossorigin-none">No CORS</span>.</p></li>

   <li><p>End the <span>synchronous section</span>, continuing the remaining steps
   <span>in parallel</span>.</p></li>

   <li>
    <p>If <var>URL</var> is not the empty string, then:

    <ol>
     <li><p>Let <var>request</var> be the result of <span
     data-x="create a potential-CORS request">creating a potential-CORS request</span> given
     <var>URL</var>, "<code data-x="">track</code>", and <var>corsAttributeState</var>, and with the
     <i>same-origin fallback flag</i> set.</p></li>

     <li><p>Set <var>request</var>'s <span data-x="concept-request-client">client</span> to the
     <code>track</code> element's <span>node document</span>'s <span>relevant settings
     object</span>.</p></li>

     <li><p>Set <var>request</var>'s <span data-x="concept-request-initiator-type">initiator
     type</span> to "<code data-x="">track</code>".</p></li>

     <li><p><!--FETCH--><span data-x="concept-fetch">Fetch</span> <var>request</var>.</p></li>
    </ol>

    <p>The <span data-x="concept-task">tasks</span> <span data-x="queue a task">queued</span> by the
    fetching algorithm on the <span>networking task source</span> to process the data as it is being
    fetched must <!--<span data-x="Content-Type sniffing">-->determine the <!--computed--> type of
    the resource<!--</span>-->. If the <!--computed--> type of the resource is not a supported text
    track format, the load will fail, as described below. Otherwise, the resource's data must be
    passed to the appropriate parser (e.g., the <span>WebVTT parser</span><!-- if the file starts
    with the "<code data-x="">WEBVTT</code>" signature-->) as it is received, with the <span>text
    track list of cues</span> being used for that parser's output. <ref>WEBVTT</ref></p><!-- see
    also critical block below, and the word "computed" in the paragraph after that -->

    <p class="note">The appropriate parser will incrementally update the <span>text track list of
    cues</span> during these <span>networking task
    source</span> <span data-x="concept-task">tasks</span>, as each such task is
    run with whatever data has been received from the network).</p>

    <p class="critical">This specification does not currently say whether or how to check the MIME
    types of text tracks, or whether or how to perform file type sniffing using the actual file
    data. Implementers differ in their intentions on this matter and it is therefore unclear what
    the right solution is. In the absence of any requirement here, the HTTP specifications' strict
    requirement to follow the Content-Type header prevails ("Content-Type specifies the media type
    of the underlying data." ... "If and only if the media type is not given by a Content-Type
    field, the recipient MAY attempt to guess the media type via inspection of its content
    and<!---->/<!---->or the name extension(s) of the URI used to identify the resource.").</p>

    <p>If fetching fails for any reason (network error, the server returns an error code, CORS
    fails, etc.), or if <var>URL</var> is the empty string, then <span>queue an element task</span>
    on the <span>DOM manipulation task source</span> given the <span>media element</span> to first
    change the <span>text track readiness state</span> to <span data-x="text track failed to
    load">failed to load</span> and then <span data-x="concept-event-fire">fire an event</span>
    named <code data-x="event-track-error">error</code> at the <code>track</code> element.</p>
    <!-- can't be the media element one, since there might not be a media element by this point -->

    <p>If fetching does not fail, but the <!--computed--> type of the resource is not a supported
    text track format, or the file was not successfully processed (e.g., the format in question is
    an XML format and the file contained a well-formedness error that <cite>XML</cite> requires be
    detected and reported to the application), then the <span data-x="concept-task">task</span> that
    is <span data-x="queue an element task">queued</span> on the <span>networking task source</span>
    in which the aforementioned problem is found must change the <span>text track readiness
    state</span> to <span data-x="text track failed to load">failed to load</span> and <span
    data-x="concept-event-fire">fire an event</span> named <code
    data-x="event-track-error">error</code> at the <code>track</code> element.</p>

    <p>If fetching does not fail, and the file was successfully processed, then the final <span
    data-x="concept-task">task</span> that is <span data-x="queue an element task">queued</span> by
    the <span>networking task source</span>, after it has finished parsing the data, must change the
    <span>text track readiness state</span> to <span data-x="text track loaded">loaded</span>, and
    <span data-x="concept-event-fire">fire an event</span> named <code
    data-x="event-track-load">load</code> at the <code>track</code> element.</p>

    <p>If, while fetching is ongoing, either:</p>

    <ul>
     <li>the <span>track URL</span> changes so that it is no longer equal to <var>URL</var>, while the <span>text track mode</span> is set to <span data-x="text track
     hidden">hidden</span> or <span data-x="text track showing">showing</span>; or</li>

     <li>the <span>text track mode</span> changes to <span data-x="text track hidden">hidden</span>
     or <span data-x="text track showing">showing</span>, while the <span>track URL</span> is not
     equal to <var>URL</var></li>
    </ul>

    <p>...then the user agent must abort <span data-x="concept-fetch">fetching</span>, discarding
    any pending <span data-x="concept-task">tasks</span> generated by that algorithm (and in
    particular, not adding any cues to the <span>text track list of cues</span> after the moment the
    URL changed), and then <span>queue an element task</span> on the <span>DOM manipulation task
    source</span> given the <code>track</code> element that first changes the <span>text track
    readiness state</span> to <span data-x="text track failed to load">failed to load</span> and
    then <span data-x="concept-event-fire">fires an event</span> named <code
    data-x="event-track-error">error</code> at the <code>track</code> element. <!-- can't be the
    media element one, since there might not be a media element by this point --></p>
   </li>

   <li><p>Wait until the <span>text track readiness state</span> is no longer set to <span
   data-x="text track loading">loading</span>.</p></li>

   <li><p>Wait until the <span>track URL</span> is no longer equal to <var>URL</var>, at
   the same time as the <span>text track mode</span> is set to <span data-x="text track
   hidden">hidden</span> or <span data-x="text track showing">showing</span>.</p></li>

   <li><p>Jump to the step labeled <i>top</i>.</p></li>
  </ol>

  <p>Whenever a <code>track</code> element has its <code data-x="attr-track-src">src</code> attribute
  set, changed, or removed, the user agent must <span>immediately</span> empty the element's <span>text
  track</span>'s <span>text track list of cues</span>. (This also causes the algorithm above to stop
  adding cues from the resource being obtained using the previously given URL, if any.)</p>

  </div>


  <div w-nodev>

  <h6><dfn>Guidelines for exposing cues</dfn> in various formats as <span data-x="text track
  cue">text track cues</span></h6>

  <p>How a specific format's text track cues are to be interpreted for the purposes of processing by
  an HTML user agent is defined by that format. In the absence of such a specification, this section
  provides some constraints within which implementations can attempt to consistently expose such
  formats.</p>

  <p>To support the <span>text track</span> model of HTML, each unit of timed data is converted to a
  <span>text track cue</span>. Where the mapping of the format's features to the aspects of a
  <span>text track cue</span> as defined in this specification are not defined, implementations must
  ensure that the mapping is consistent with the definitions of the aspects of a <span>text track
  cue</span> as defined above, as well as with the following constraints:</p>

  <dl>
   <dt>The <span>text track cue identifier</span>
   <dd>
    <p>Should be set to the empty string if the format has no obvious analogue to a per-cue
    identifier.</p>
   </dd>

   <dt>The <span>text track cue pause-on-exit flag</span>
   <dd>
    <p>Should be set to false.</p>
   </dd>
  </dl>

  </div>


  <h6>Text track API</h6>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>TextTrackList</dfn> : <span>EventTarget</span> {
  readonly attribute unsigned long <span data-x="dom-TextTrackList-length">length</span>;
  <a href="#dom-texttracklist-item">getter</a> <span>TextTrack</span> (unsigned long index);
  <span>TextTrack</span>? <span data-x="dom-TextTrackList-getTrackById">getTrackById</span>(DOMString id);

  attribute <span>EventHandler</span> <span data-x="handler-TextTrackList-onchange">onchange</span>;
  attribute <span>EventHandler</span> <span data-x="handler-TextTrackList-onaddtrack">onaddtrack</span>;
  attribute <span>EventHandler</span> <span data-x="handler-TextTrackList-onremovetrack">onremovetrack</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-textTracks">textTracks</span>.<span data-x="">length</span></code></dt>
   <dd>
    <p>Returns the number of <span data-x="text track">text tracks</span> associated with the
    <span>media element</span> (e.g. from <code>track</code> elements). This is the number of <span
    data-x="text track">text tracks</span> in the <span>media element</span>'s <span>list of text
    tracks</span>.</p>
   </dd>

   <dt><code data-x=""><var>media</var>.<span data-x="dom-media-textTracks">textTracks[</span> <var>n</var> <span data-x="">]</span></code></dt>
   <dd>
    <p>Returns the <code>TextTrack</code> object representing the <var>n</var>th <span>text
    track</span> in the <span>media element</span>'s <span>list of text tracks</span>.</p>
   </dd>

   <dt><code data-x=""><var>textTrack</var> = <var>media</var>.<span data-x="dom-media-textTracks">textTracks</span>.<span subdfn data-x="dom-TextTrackList-getTrackById">getTrackById</span>(<var>id</var>)</code></dt>

   <dd>
    <p>Returns the <code>TextTrack</code> object with the given identifier, or null if no track has
    that identifier.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>A <code>TextTrackList</code> object represents a dynamically updating list of <span data-x="text
  track">text tracks</span> in a given order.</p>

  <p>The <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-textTracks">textTracks</code></dfn> attribute of <span data-x="media
  element">media elements</span> must return a <code>TextTrackList</code> object representing the
  <code>TextTrack</code> objects of the <span data-x="text track">text tracks</span> in the
  <span>media element</span>'s <span>list of text tracks</span>, in the same order as in the
  <span>list of text tracks</span>.</p>

  <p>The <dfn attribute for="TextTrackList"><code
  data-x="dom-TextTrackList-length">length</code></dfn> attribute of a <code>TextTrackList</code>
  object must return the number of <span data-x="text track">text tracks</span> in the list
  represented by the <code>TextTrackList</code> object.</p>

  <p>The <span>supported property indices</span> of a <code>TextTrackList</code> object at any
  instant are the numbers from zero to the number of <span data-x="text track">text tracks</span> in
  the list represented by the <code>TextTrackList</code> object minus one, if any. If there are no
  <span data-x="text track">text tracks</span> in the list, there are no <span>supported property
  indices</span>.</p>

  <p id="dom-texttracklist-item">To <span>determine the value of an indexed property</span> of a
  <code>TextTrackList</code> object for a given index <var>index</var>, the user agent must return
  the <var>index</var>th <span>text track</span> in the list represented by the
  <code>TextTrackList</code> object.</p>

  <p>The <dfn method for="TextTrackList"><code
  data-x="dom-TextTrackList-getTrackById">getTrackById(<var>id</var>)</code></dfn> method must
  return the first <code>TextTrack</code> in the <code>TextTrackList</code> object whose <code
  data-x="dom-TextTrack-id">id</code> IDL attribute would return a value equal to the value of the
  <var>id</var> argument. When no tracks match the given argument, the method must return null.</p>

  </div>

  <hr>

  <pre><code class="idl">enum <dfn enum>TextTrackMode</dfn> {<!--
   --> "<span data-x="dom-TextTrack-disabled">disabled</span>", <!--
   --> "<span data-x="dom-TextTrack-hidden">hidden</span>", <!--
   --> "<span data-x="dom-TextTrack-showing">showing</span>" };
enum <dfn enum>TextTrackKind</dfn> {<!--
   --> "<span data-x="dom-TextTrack-kind-subtitles">subtitles</span>", <!--
   --> "<span data-x="dom-TextTrack-kind-captions">captions</span>", <!--
   --> "<span data-x="dom-TextTrack-kind-descriptions">descriptions</span>", <!--
   --> "<span data-x="dom-TextTrack-kind-chapters">chapters</span>", <!--
   --> "<span data-x="dom-TextTrack-kind-metadata">metadata</span>" };

[Exposed=Window]
interface <dfn interface>TextTrack</dfn> : <span>EventTarget</span> {
  readonly attribute <span>TextTrackKind</span> <span data-x="dom-TextTrack-kind">kind</span>;
  readonly attribute DOMString <span data-x="dom-TextTrack-label">label</span>;
  readonly attribute DOMString <span data-x="dom-TextTrack-language">language</span>;

  readonly attribute DOMString <span data-x="dom-TextTrack-id">id</span>;
  readonly attribute DOMString <span data-x="dom-TextTrack-inBandMetadataTrackDispatchType">inBandMetadataTrackDispatchType</span>;

  attribute <span>TextTrackMode</span> <span data-x="dom-TextTrack-mode">mode</span>;

  readonly attribute <span>TextTrackCueList</span>? <span data-x="dom-TextTrack-cues">cues</span>;
  readonly attribute <span>TextTrackCueList</span>? <span data-x="dom-TextTrack-activeCues">activeCues</span>;

  undefined <span data-x="dom-TextTrack-addCue">addCue</span>(<span>TextTrackCue</span> cue);
  undefined <span data-x="dom-TextTrack-removeCue">removeCue</span>(<span>TextTrackCue</span> cue);

  attribute <span>EventHandler</span> <span data-x="handler-TextTrack-oncuechange">oncuechange</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>textTrack</var> = <var>media</var>.<span subdfn data-x="dom-media-addTextTrack">addTextTrack</span>(<var>kind</var> [, <var>label</var> [, <var>language</var> ] ])</code></dt>
   <dd>
    <p>Creates and returns a new <code>TextTrack</code> object, which is also added to the
    <span>media element</span>'s <span>list of text tracks</span>.</p>
   </dd>

   <dt><code data-x=""><var>textTrack</var>.<span subdfn data-x="dom-TextTrack-kind">kind</span></code></dt>
   <dd><p>Returns the <span>text track kind</span> string.</p></dd>

   <dt><code data-x=""><var>textTrack</var>.<span subdfn data-x="dom-TextTrack-label">label</span></code></dt>
   <dd>
    <p>Returns the <span>text track label</span>, if there is one, or the empty string otherwise
    (indicating that a custom label probably needs to be generated from the other attributes of the
    object if the object is exposed to the user).</p>
   </dd>

   <dt><code data-x=""><var>textTrack</var>.<span subdfn data-x="dom-TextTrack-language">language</span></code></dt>
   <dd><p>Returns the <span>text track language</span> string.</p></dd>

   <dt><code data-x=""><var>textTrack</var>.<span subdfn data-x="dom-TextTrack-id">id</span></code></dt>
   <dd>
    <p>Returns the ID of the given track.</p>

    <p>For in-band tracks, this is the ID that can be used with a <span
    data-x="concept-url-fragment">fragment</span> if the format supports <span>media fragment
    syntax</span>, and that can be used with the <code
    data-x="dom-TextTrackList-getTrackById">getTrackById()</code> method.</p>

    <p>For <code>TextTrack</code> objects corresponding to <code>track</code> elements, this is the
    ID of the <code>track</code> element.</p>
   </dd>

   <dt><code data-x=""><var>textTrack</var>.<span subdfn data-x="dom-TextTrack-inBandMetadataTrackDispatchType">inBandMetadataTrackDispatchType</span></code></dt>
   <dd><p>Returns the <span>text track in-band metadata track dispatch type</span> string.</p></dd>

   <dt><code data-x=""><var>textTrack</var>.<span subdfn data-x="dom-TextTrack-mode">mode</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the <span>text track mode</span>, represented by a string from the following
    list:</p>

    <dl>
     <dt>"<code data-x="dom-TextTrack-disabled">disabled</code>"</dt>
     <dd><p>The <span>text track disabled</span> mode.</p></dd>
     <dt>"<code data-x="dom-TextTrack-hidden">hidden</code>"</dt>
     <dd><p>The <span>text track hidden</span> mode.</p></dd>
     <dt>"<code data-x="dom-TextTrack-showing">showing</code>"</dt>
     <dd><p>The <span>text track showing</span> mode.</p></dd>
    </dl>

    <p>Can be set, to change the mode.</p>
   </dd>

   <dt><code data-x=""><var>textTrack</var>.<span subdfn data-x="dom-TextTrack-cues">cues</span></code></dt>
   <dd>
    <p>Returns the <span>text track list of cues</span>, as a <code>TextTrackCueList</code>
    object.</p>
   </dd>

   <dt><code data-x=""><var>textTrack</var>.<span subdfn data-x="dom-TextTrack-activeCues">activeCues</span></code></dt>
   <dd>
    <p>Returns the <span data-x="text track cue">text track cues</span> from the <span>text track
    list of cues</span> that are currently active (i.e. that start before the <span>current playback
    position</span> and end after it), as a <code>TextTrackCueList</code> object.</p>
   </dd>

   <dt><code data-x=""><var>textTrack</var>.<span subdfn data-x="dom-TextTrack-addCue">addCue</span>(<var>cue</var>)</code></dt>
   <dd><p>Adds the given cue to <var>textTrack</var>'s <span>text track list of
   cues</span>.</p></dd>

   <dt><code data-x=""><var>textTrack</var>.<span subdfn data-x="dom-TextTrack-removeCue">removeCue</span>(<var>cue</var>)</code></dt>
   <dd><p>Removes the given cue from <var>textTrack</var>'s <span>text track list of
   cues</span>.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="HTMLMediaElement"><code
  data-x="dom-media-addTextTrack">addTextTrack(<var>kind</var>, <var>label</var>,
  <var>language</var>)</code></dfn> method of <span data-x="media element">media elements</span>,
  when invoked, must run the following steps:</p>

  <ol>
   <li>
    <p>Create a new <code>TextTrack</code> object.</p>
   </li>

   <li>
    <p>Create a new <span>text track</span> corresponding to the new object, and set its <span>text
    track kind</span> to <var>kind</var>, its <span>text track label</span> to <var>label</var>, its <span>text track language</span> to <var>language</var>, its
    <span>text track readiness state</span> to the <span>text track loaded</span> state, its
    <span>text track mode</span> to the <span>text track hidden</span> mode, and its <span>text
    track list of cues</span> to an empty list.</p>

    <p>Initially, the <span>text track list of cues</span> is not associated with any <span>rules
    for updating the text track rendering</span>. When a <span>text track cue</span> is added to it,
    the <span>text track list of cues</span> has its rules permanently set accordingly.</p>
   </li>

   <li>
    <p>Add the new <span>text track</span> to the <span>media element</span>'s <span>list of text
    tracks</span>.</p>
   </li>

   <li>
    <p><span>Queue a media element task</span> given the <span>media element</span> to <span
    data-x="concept-event-fire">fire an event</span> named <code
    data-x="event-media-addtrack">addtrack</code> at the <span>media element</span>'s <code
    data-x="dom-media-textTracks">textTracks</code> attribute's <code>TextTrackList</code> object,
    using <code>TrackEvent</code>, with the <code data-x="dom-TrackEvent-track">track</code>
    attribute initialized to the new <span>text track</span>'s <code>TextTrack</code> object.</p>
   </li>

   <li>
    <p>Return the new <code>TextTrack</code> object.</p>
   </li>
  </ol>

  <hr>

  <p>The <dfn attribute for="TextTrack"><code data-x="dom-TextTrack-kind">kind</code></dfn>
  attribute must return the <span>text track kind</span> of the <span>text track</span> that the
  <code>TextTrack</code> object represents.</p>

  <p>The <dfn attribute for="TextTrack"><code data-x="dom-TextTrack-label">label</code></dfn>
  attribute must return the <span>text track label</span> of the <span>text track</span> that the
  <code>TextTrack</code> object represents.</p>

  <p>The <dfn attribute for="TextTrack"><code data-x="dom-TextTrack-language">language</code></dfn>
  attribute must return the <span>text track language</span> of the <span>text track</span> that the
  <code>TextTrack</code> object represents.</p>

  <p>The <dfn attribute for="TextTrack"><code data-x="dom-TextTrack-id">id</code></dfn> attribute
  returns the track's identifier, if it has one, or the empty string otherwise. For tracks that
  correspond to <code>track</code> elements, the track's identifier is the value of the element's
  <code data-x="attr-id">id</code> attribute, if any. For in-band tracks, the track's identifier is
  specified by the <span>media resource</span>. If the <span>media resource</span> is in a format
  that supports <span>media fragment syntax</span>, the identifier returned for a particular track
  must be the same identifier that would enable the track if used as the name of a track in the
  track dimension of such a <span data-x="concept-url-fragment">fragment</span>.</p>

  <p>The <dfn attribute for="TextTrack"><code
  data-x="dom-TextTrack-inBandMetadataTrackDispatchType">inBandMetadataTrackDispatchType</code></dfn>
  attribute must return the <span>text track in-band metadata track dispatch type</span> of the
  <span>text track</span> that the <code>TextTrack</code> object represents.</p>

  <p>The <dfn attribute for="TextTrack"><code data-x="dom-TextTrack-mode">mode</code></dfn>
  attribute, on getting, must return the string corresponding to the <span>text track mode</span> of
  the <span>text track</span> that the <code>TextTrack</code> object represents, as defined by the
  following list:</p>

  <dl>
   <dt>"<dfn enum-value for="TextTrackMode" data-x="dom-TextTrack-disabled"><code>disabled</code></dfn>"</dt>
   <dd>The <span>text track disabled</span> mode.</dd>
   <dt>"<dfn enum-value for="TextTrackMode" data-x="dom-TextTrack-hidden"><code>hidden</code></dfn>"</dt>
   <dd>The <span>text track hidden</span> mode.</dd>
   <dt>"<dfn enum-value for="TextTrackMode" data-x="dom-TextTrack-showing"><code>showing</code></dfn>"</dt>
   <dd>The <span>text track showing</span> mode.</dd>
  </dl>

  <p>On setting, if the new value isn't equal to what the attribute would currently return, the new
  value must be processed as follows:</p>

  <dl class="switch">

   <dt>If the new value is "<code data-x="dom-TextTrack-disabled">disabled</code>"</dt>

   <dd>
    <p>Set the <span>text track mode</span> of the <span>text track</span> that the
    <code>TextTrack</code> object represents to the <span>text track disabled</span> mode.</p>
   </dd>

   <dt>If the new value is "<code data-x="dom-TextTrack-hidden">hidden</code>"</dt>

   <dd>
    <p>Set the <span>text track mode</span> of the <span>text track</span> that the
    <code>TextTrack</code> object represents to the <span>text track hidden</span> mode.</p>
   </dd>

   <dt>If the new value is "<code data-x="dom-TextTrack-showing">showing</code>"</dt>

   <dd>
    <p>Set the <span>text track mode</span> of the <span>text track</span> that the
    <code>TextTrack</code> object represents to the <span>text track showing</span> mode.</p>
   </dd>
  </dl>

  <p>If the <span>text track mode</span> of the <span>text track</span> that the
  <code>TextTrack</code> object represents is not the <span>text track disabled</span> mode, then
  the <dfn attribute for="TextTrack"><code data-x="dom-TextTrack-cues">cues</code></dfn> attribute
  must return a <span>live</span> <code>TextTrackCueList</code> object that represents the subset of
  the <span>text track list of cues</span> of the <span>text track</span> that the
  <code>TextTrack</code> object represents whose <span data-x="text track cue end time">end
  times</span> occur at or after the <span>earliest possible position when the script
  started</span>, in <span>text track cue order</span>. Otherwise, it must return null. For each
  <code>TextTrack</code> object, when an object is returned, the same <code>TextTrackCueList</code>
  object must be returned each time.</p>

  <p>The <dfn>earliest possible position when the script started</dfn> is whatever the
  <span>earliest possible position</span> was the last time the <span>event loop</span> reached step
  1.</p>

  <p>If the <span>text track mode</span> of the <span>text track</span> that the
  <code>TextTrack</code> object represents is not the <span>text track disabled</span> mode, then
  the <dfn attribute for="TextTrack"><code data-x="dom-TextTrack-activeCues">activeCues</code></dfn>
  attribute must return a <span>live</span> <code>TextTrackCueList</code> object that represents the
  subset of the <span>text track list of cues</span> of the <span>text track</span> that the
  <code>TextTrack</code> object represents whose <span>active flag was set when the script
  started</span>, in <span>text track cue order</span>. Otherwise, it must return null. For each
  <code>TextTrack</code> object, when an object is returned, the same <code>TextTrackCueList</code>
  object must be returned each time.</p>

  <p>A <span>text track cue</span>'s <dfn>active flag was set when the script started</dfn> if its
  <span>text track cue active flag</span> was set the last time the <span>event loop</span> reached
  <a href="#step1">step 1</a>.</p>

  <hr>

  <p>The <dfn method for="TextTrack"><code
  data-x="dom-TextTrack-addCue">addCue(<var>cue</var>)</code></dfn> method of <code>TextTrack</code>
  objects, when invoked, must run the following steps:</p>

  <ol>
   <li><p>If the <span>text track list of cues</span> does not yet have any associated <span>rules
   for updating the text track rendering</span>, then associate the <span>text track list of
   cues</span> with the <span>rules for updating the text track rendering</span> appropriate to <var>cue</var>.</p>

   <li><p>If <span>text track list of cues</span>' associated <span>rules for updating the text
   track rendering</span> are not the same <span>rules for updating the text track rendering</span>
   as appropriate for <var>cue</var>, then throw an <span>"<code>InvalidStateError</code>"</span>
   <code>DOMException</code>.</p>

   <li><p>If the given <var>cue</var> is in a <span>text track list of cues</span>, then
   remove <var>cue</var> from that <span>text track list of cues</span>.</p></li>

   <li><p>Add <var>cue</var> to the <code>TextTrack</code> object's <span>text track</span>'s
   <span>text track list of cues</span>.</p></li>
  </ol>

  <p>The <dfn method for="TextTrack"><code
  data-x="dom-TextTrack-removeCue">removeCue(<var>cue</var>)</code></dfn> method of
  <code>TextTrack</code> objects, when invoked, must run the following steps:</p>

  <ol>
   <li><p>If the given <var>cue</var> is not in the <code>TextTrack</code> object's <span>text
   track</span>'s <span>text track list of cues</span>, then throw a
   <span>"<code>NotFoundError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Remove <var>cue</var> from the <code>TextTrack</code> object's <span>text track</span>'s
   <span>text track list of cues</span>.</p></li>
  </ol>

  </div>

  <div class="example">

   <p>In this example, an <code>audio</code> element is used to play a specific sound-effect from a
   sound file containing many sound effects. A cue is used to pause the audio, so that it ends
   exactly at the end of the clip, even if the browser is busy running some script. If the page had
   relied on script to pause the audio, then the start of the next clip might be heard if the
   browser was not able to run the script at the exact time specified.</p>

   <pre><code class="js">var sfx = new Audio('sfx.wav');
var sounds = sfx.addTextTrack('metadata');

// add sounds we care about
function addFX(start, end, name) {
  var cue = new VTTCue(start, end, '');
  cue.id = name;
  cue.pauseOnExit = true;
  sounds.addCue(cue);
}
addFX(12.783, 13.612, 'dog bark');
addFX(13.612, 15.091, 'kitten mew');

function playSound(id) {
  sfx.currentTime = sounds.getCueById(id).startTime;
  sfx.play();
}

// play a bark as soon as we can
sfx.oncanplaythrough = function () {
  playSound('dog bark');
}
// meow when the user tries to leave,
// and have the browser ask them to stay
window.onbeforeunload = function (e) {
  playSound('kitten mew');
  e.preventDefault();
}</code></pre>

  </div>

  <hr>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>TextTrackCueList</dfn> {
  readonly attribute unsigned long <span data-x="dom-TextTrackCueList-length">length</span>;
  <a href="#dom-texttrackcuelist-item">getter</a> <span>TextTrackCue</span> (unsigned long index);
  <span>TextTrackCue</span>? <span data-x="dom-TextTrackCueList-getCueById">getCueById</span>(DOMString id);
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>cuelist</var>.<span subdfn data-x="dom-TextTrackCueList-length">length</span></code></dt>
   <dd><p>Returns the number of <span data-x="text track cue">cues</span> in the list.</p></dd>

   <dt><code data-x=""><var>cuelist</var>[<var>index</var>]</code></dt>
   <dd>
    <p>Returns the <span>text track cue</span> with index <var>index</var> in the list. The cues are
    sorted in <span>text track cue order</span>.</p>
   </dd>

   <dt><code data-x=""><var>cuelist</var>.<span subdfn data-x="dom-TextTrackCueList-getCueById">getCueById</span>(<var>id</var>)</code></dt>
   <dd>
    <p>Returns the first <span>text track cue</span> (in <span>text track cue order</span>) with
    <span>text track cue identifier</span> <var>id</var>.</p>

    <p>Returns null if none of the cues have the given identifier or if the argument is the empty
    string.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>A <code>TextTrackCueList</code> object represents a dynamically updating list of <span
  data-x="text track cue">text track cues</span> in a given order.</p>

  <p>The <dfn attribute for="TextTrackCueList"><code
  data-x="dom-TextTrackCueList-length">length</code></dfn> attribute must return the number of <span
  data-x="text track cue">cues</span> in the list represented by the <code>TextTrackCueList</code>
  object.</p>

  <p>The <span>supported property indices</span> of a <code>TextTrackCueList</code> object at any
  instant are the numbers from zero to the number of <span data-x="text track cue">cues</span> in the
  list represented by the <code>TextTrackCueList</code> object minus one, if any. If there are no
  <span data-x="text track cue">cues</span> in the list, there are no <span>supported property
  indices</span>.</p>

  <p id="dom-texttrackcuelist-item">To <span>determine the value of an indexed property</span> for a
  given index <var>index</var>, the user agent must return the <var>index</var>th <span>text track
  cue</span> in the list represented by the <code>TextTrackCueList</code> object.</p>

  <p>The <dfn method for="TextTrackCueList"><code
  data-x="dom-TextTrackCueList-getCueById">getCueById(<var>id</var>)</code></dfn> method, when
  called with an argument other than the empty string, must return the first <span>text track
  cue</span> in the list represented by the <code>TextTrackCueList</code> object whose <span>text
  track cue identifier</span> is <var>id</var>, if any, or null otherwise. If the argument is the
  empty string, then the method must return null.</p>

  </div>

  <hr>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>TextTrackCue</dfn> : <span>EventTarget</span> {
  readonly attribute <span>TextTrack</span>? <span data-x="dom-TextTrackCue-track">track</span>;

  attribute DOMString <span data-x="dom-TextTrackCue-id">id</span>;
  attribute double <span data-x="dom-TextTrackCue-startTime">startTime</span>;
  attribute unrestricted double <span data-x="dom-TextTrackCue-endTime">endTime</span>;
  attribute boolean <span data-x="dom-TextTrackCue-pauseOnExit">pauseOnExit</span>;

  attribute <span>EventHandler</span> <span data-x="handler-TextTrackCue-onenter">onenter</span>;
  attribute <span>EventHandler</span> <span data-x="handler-TextTrackCue-onexit">onexit</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>cue</var>.<span subdfn data-x="dom-TextTrackCue-track">track</span></code></dt>
   <dd>
    <p>Returns the <code>TextTrack</code> object to which this <span>text track cue</span> belongs,
    if any, or null otherwise.</p>
   </dd>

   <dt><code data-x=""><var>cue</var>.<span subdfn data-x="dom-TextTrackCue-id">id</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the <span>text track cue identifier</span>.</p>

    <p>Can be set.</p>
   </dd>

   <dt><code data-x=""><var>cue</var>.<span subdfn data-x="dom-TextTrackCue-startTime">startTime</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the <span>text track cue start time</span>, in seconds.</p>

    <p>Can be set.</p>
   </dd>

   <dt><code data-x=""><var>cue</var>.<span subdfn data-x="dom-TextTrackCue-endTime">endTime</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the <span>text track cue end time</span>, in seconds.</p>

    <p>Returns positive Infinity for an <span>unbounded text track cue</span>.</p>

    <p>Can be set.</p>
   </dd>

   <dt><code data-x=""><var>cue</var>.<span subdfn data-x="dom-TextTrackCue-pauseOnExit">pauseOnExit</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns true if the <span>text track cue pause-on-exit flag</span> is set, false
    otherwise.</p>

    <p>Can be set.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="TextTrackCue"><code data-x="dom-TextTrackCue-track">track</code></dfn>
  attribute, on getting, must return the <code>TextTrack</code> object of the <span>text
  track</span> in whose <span data-x="text track list of cues">list of cues</span> the <span>text
  track cue</span> that the <code>TextTrackCue</code> object represents finds itself, if any; or
  null otherwise.</p>

  <p>The <dfn attribute for="TextTrackCue"><code data-x="dom-TextTrackCue-id">id</code></dfn>
  attribute, on getting, must return the <span>text track cue identifier</span> of the <span>text
  track cue</span> that the <code>TextTrackCue</code> object represents. On setting, the <span>text
  track cue identifier</span> must be set to the new value.</p>

  <p>The <dfn attribute for="TextTrackCue"><code
  data-x="dom-TextTrackCue-startTime">startTime</code></dfn> attribute, on getting, must return the
  <span>text track cue start time</span> of the <span>text track cue</span> that the
  <code>TextTrackCue</code> object represents, in seconds. On setting, the <span>text track cue
  start time</span> must be set to the new value, interpreted in seconds; then, if the
  <code>TextTrackCue</code> object's <span>text track cue</span> is in a <span>text track</span>'s
  <span data-x="text track list of cues">list of cues</span>, and that <span>text track</span> is in
  a <span>media element</span>'s <span>list of text tracks</span>, and the <span>media
  element</span>'s <span>show poster flag</span> is not set, then run the <i data-x="time marches
  on">time marches on</i> steps for that <span>media element</span>.</p>

  <p>The <dfn attribute for="TextTrackCue"><code
  data-x="dom-TextTrackCue-endTime">endTime</code></dfn> attribute, on getting, must return the
  <span>text track cue end time</span> of the <span>text track cue</span> that the
  <code>TextTrackCue</code> object represents, in seconds or positive Infinity. On setting, if the
  new value is negative Infinity or a Not-a-Number (NaN) value, then throw a <a
  href="https://tc39.es/ecma262/#sec-native-error-types-used-in-this-standard-typeerror">TypeError
  </a> exception. Otherwise, the <span>text track cue end time</span> must be set to the new value.
  Then, if the <code>TextTrackCue</code> object's <span>text track cue</span> is in a <span>text
  track</span>'s <span data-x="text track list of cues">list of cues</span>, and that <span>text
  track</span> is in a <span>media element</span>'s <span>list of text tracks</span>, and the
  <span>media element</span>'s <span>show poster flag</span> is not set, then run the <i
  data-x="time marches on">time marches on</i> steps for that <span>media element</span>.</p>

  <p>The <dfn attribute for="TextTrackCue"><code
  data-x="dom-TextTrackCue-pauseOnExit">pauseOnExit</code></dfn> attribute, on getting, must return
  true if the <span>text track cue pause-on-exit flag</span> of the <span>text track cue</span> that
  the <code>TextTrackCue</code> object represents is set; or false otherwise.  On setting, the
  <span>text track cue pause-on-exit flag</span> must be set if the new value is true, and must be
  unset otherwise.</p>


  <h6 id="cue-events">Event handlers for objects of the text track APIs</h6>

  <p>The following are the <span>event handlers</span> that (and their corresponding <span
  data-x="event handler event type">event handler event types</span>) <span w-nodev>that must
  be</span> supported, as <span>event handler IDL attributes</span>, by all objects implementing the
  <code>TextTrackList</code> interface:</p>

  <table>
   <thead>
    <tr><th><span data-x="event handlers">Event handler</span> <th><span>Event handler event type</span>
   <tbody>
    <tr><td><dfn attribute for="TextTrackList"><code data-x="handler-TextTrackList-onchange">onchange</code></dfn> <td> <code data-x="event-media-change">change</code>
    <tr><td><dfn attribute for="TextTrackList"><code data-x="handler-TextTrackList-onaddtrack">onaddtrack</code></dfn> <td> <code data-x="event-media-addtrack">addtrack</code>
    <tr><td><dfn attribute for="TextTrackList"><code data-x="handler-TextTrackList-onremovetrack">onremovetrack</code></dfn> <td> <code data-x="event-media-removetrack">removetrack</code>
  </table>

  <p>The following are the <span>event handlers</span> that (and their corresponding <span
  data-x="event handler event type">event handler event types</span>) <span w-nodev>that must
  be</span> supported, as <span>event handler IDL attributes</span>, by all objects implementing the
  <code>TextTrack</code> interface:</p>

  <table>
   <thead>
    <tr><th><span data-x="event handlers">Event handler</span> <th><span>Event handler event type</span>
   <tbody>
    <tr><td><dfn attribute for="TextTrack"><code data-x="handler-TextTrack-oncuechange">oncuechange</code></dfn> <td> <code data-x="event-media-cuechange">cuechange</code>
  </table>

  <p>The following are the <span>event handlers</span> (and their corresponding <span
  data-x="event handler event type">event handler event types</span>) <span w-nodev>that must
  be</span> supported, as <span>event handler IDL attributes</span>, by all objects implementing the
  <code>TextTrackCue</code> interface:</p>

  <table>
   <thead>
    <tr><th><span data-x="event handlers">Event handler</span> <th><span>Event handler event type</span>
   <tbody>
    <tr><td><dfn attribute for="TextTrackCue"><code data-x="handler-TextTrackCue-onenter">onenter</code></dfn> <td> <code data-x="event-media-enter">enter</code>
    <tr><td><dfn attribute for="TextTrackCue"><code data-x="handler-TextTrackCue-onexit">onexit</code></dfn> <td> <code data-x="event-media-exit">exit</code>
  </table>

  </div>



  <h6>Best practices for metadata text tracks</h6>

  <!-- NON-NORMATIVE SECTION -->

  <p>Text tracks can be used for storing data relating to the media data, for interactive or
  augmented views.</p>

  <p>For example, a page showing a sports broadcast could include information about the current
  score. Suppose a robotics competition was being streamed live. The image could be overlaid with
  the scores, as follows:</p>

  <p><iframe src="data:text/html;charset=utf-8,%3C!DOCTYPE%20html%3E%0A%3Cstyle%3E%0A%20body%2C%20html%20%7B%20margin%3A%200%3B%20padding%3A%200%3B%20overflow%3A%20hidden%3B%20%7D%0A%20div%20%7B%20width%3A%20600px%3B%20height%3A%20400px%3B%20position%3A%20relative%3B%20%7D%0A%20p%20%7B%20position%3A%20absolute%3B%20top%3A%200%3B%20margin%3A%200.25em%3B%20font%3A%20small-caps%20900%202em%20sans-serif%3B%20text-shadow%3A%20white%200%200%204px%3B%20%7D%0A%20span%20%7B%20display%3A%20block%3B%20%7D%0A%20.left%20%7B%20color%3A%20red%3B%20left%3A%200%3B%20text-align%3A%20left%3B%20%7D%0A%20.right%20%7B%20color%3A%20blue%3B%20right%3A%200%3B%20text-align%3A%20right%3B%20%7D%0A%20.middle%20%7B%20color%3A%20white%3B%20top%3A%20auto%3B%20bottom%3A%200%3B%20left%3A%200%3B%20right%3A%200%3B%20text-align%3A%20center%3B%20text-shadow%3A%20black%200%200%204px%3B%20%7D%0A%20.middle%20span%20%7B%20display%3A%20inline-block%3B%20margin%3A%200%201em%3B%20font-size%3A%200.75em%3B%20text-transform%3A%20uppercase%3B%20%7D%0A%3C%2Fstyle%3E%0A%3Cdiv%3E%0A%20%3Cimg%20src%3D%22https%3A%2F%2Fhtml.spec.whatwg.org%2Fimages%2Frobots.jpeg%22%3E%0A%20%3Cp%20class%3D%22score%20left%22%3E%3Cspan%3ERed%20Alliance%3C%2Fspan%3E%20%3Cspan%3E78%3C%2Fspan%3E%3C%2Fp%3E%0A%20%3Cp%20class%3D%22score%20right%22%3E%3Cspan%3EBlue%20Alliance%3C%2Fspan%3E%20%3Cspan%3E66%3C%2Fspan%3E%3C%2Fp%3E%0A%20%3Cp%20class%3D%22score%20middle%22%3E%3Cspan%3EQual%20Match%2037%3C%2Fspan%3E%20%3Cspan%3EFriday%2014%3A21%3C%2Fspan%3E%0A%3C%2Fdiv%3E" width="600" height="400"></iframe>

  <p>In order to make the score display render correctly whenever the user seeks to an arbitrary
  point in the video, the metadata text track cues need to be as long as is appropriate for the
  score. For example, in the frame above, there would be maybe one cue that lasts the length of the
  match that gives the match number, one cue that lasts until the blue alliance's score changes, and
  one cue that lasts until the red alliance's score changes. If the video is just a stream of the
  live event, the time in the bottom right would presumably be automatically derived from the
  current video time, rather than based on a cue. However, if the video was just the highlights,
  then that might be given in cues also.</p>

  <p>The following shows what fragments of this could look like in a WebVTT file:</p>

  <pre>WEBVTT

...

05:10:00.000 --> 05:12:15.000
matchtype:qual
matchnumber:37

...

05:11:02.251 --> 05:11:17.198
red:78

05:11:03.672 --> 05:11:54.198
blue:66

05:11:17.198 --> 05:11:25.912
red:80

05:11:25.912 --> 05:11:26.522
red:83

05:11:26.522 --> 05:11:26.982
red:86

05:11:26.982 --> 05:11:27.499
red:89

...</pre>

  <p>The key here is to notice that the information is given in cues that span the length of time to
  which the relevant event applies. If, instead, the scores were given as zero-length (or very
  brief, nearly zero-length) cues when the score changes, for example saying "red+2" at
  05:11:17.198, "red+3" at 05:11:25.912, etc, problems arise: primarily, seeking is much harder to
  implement, as the script has to walk the entire list of cues to make sure that no notifications
  have been missed; but also, if the cues are short it's possible the script will never see that
  they are active unless it listens to them specifically.</p>

  <p>When using cues in this manner, authors are encouraged to use the <code
  data-x="event-media-cuechange">cuechange</code> event to update the current annotations. (In
  particular, using the <code data-x="event-media-timeupdate">timeupdate</code> event would be less
  appropriate as it would require doing work even when the cues haven't changed, and, more
  importantly, would introduce a higher latency between when the metadata cues become active and
  when the display is updated, since <code data-x="event-media-timeupdate">timeupdate</code> events
  are rate-limited.)</p>



  <h5>Identifying a track kind through a URL</h5>

  <p>Other specifications or formats that need a <span>URL</span> to identify the return values of
  the <code>AudioTrack</code> <code data-x="dom-AudioTrack-kind">kind</code> or
  <code>VideoTrack</code> <code data-x="dom-VideoTrack-kind">kind</code> IDL attributes, or identify
  the <span data-x="text track kind">kind of text track</span>, must use the
  <code>about:html-kind</code> <span>URL</span>.</p>



  <h5>User interface</h5>

  <p>The <dfn element-attr for="audio,video"><code
  data-x="attr-media-controls">controls</code></dfn> attribute is a <span>boolean attribute</span>.
  If present, it indicates that the author has not provided a scripted controller and would like the
  user agent to provide its own set of controls.</p>

  <div w-nodev>

  <p>If the attribute is present, or if <span data-x="concept-n-noscript">scripting is
  disabled</span> for the <span>media element</span>, then the user agent should <dfn>expose a user
  interface to the user</dfn>. This user interface should include features to begin playback, pause
  playback, seek to an arbitrary position in the content (if the content supports arbitrary
  seeking), change the volume, change the display of closed captions or embedded sign-language
  tracks, select different audio tracks or turn on audio descriptions, and show the media content in
  manners more suitable to the user (e.g. fullscreen video or in an independent resizable window).
  Other controls may also be made available.</p>

  <p>Even when the attribute is absent, however, user agents may provide controls to affect playback
  of the media resource (e.g. play, pause, seeking, track selection, and volume controls), but such
  features should not interfere with the page's normal rendering. For example, such features could
  be exposed in the <span>media element</span>'s context menu, platform media keys, or a remote
  control. The user agent may implement this simply by <span data-x="expose a user interface to the
  user">exposing a user interface to the user</span> as described above (as if the <code
  data-x="attr-media-controls">controls</code> attribute was present).</p>

  <p>If the user agent <span data-x="expose a user interface to the user">exposes a user interface to
  the user</span> by displaying controls over the <span>media element</span>, then the user agent
  should suppress any user interaction events while the user agent is interacting with this
  interface. (For example, if the user clicks on a video's playback control, <code
  data-x="event-mousedown">mousedown</code> events and so forth would not simultaneously be fired at
  elements on the page.)</p>

  <p>Where possible (specifically, for starting, stopping, pausing, and unpausing playback, for
  seeking, for changing the rate of playback, for fast-forwarding or rewinding, for listing,
  enabling, and disabling text tracks, and for muting or changing the volume of the audio), user
  interface features exposed by the user agent must be implemented in terms of the DOM API described
  above, so that, e.g., all the same events fire.</p>

  <p>Features such as fast-forward or rewind must be implemented by only changing the <code
  data-x="">playbackRate</code> attribute (and not the <code data-x="">defaultPlaybackRate</code>
  attribute).</p>

  <p>Seeking must be implemented in terms of <span data-x="dom-media-seek">seeking</span> to the
  requested position in the <span>media element</span>'s <span>media timeline</span>. For media
  resources where seeking to an arbitrary position would be slow, user agents are encouraged to use
  the <i>approximate-for-speed</i> flag when seeking in response to the user manipulating an
  approximate position interface such as a seek bar.</p>

  <p>The <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-controls">controls</code></dfn> IDL attribute must <span>reflect</span> the
  content attribute of the same name.</p>

  <hr>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-volume">volume</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current playback volume, as a number in the range 0.0 to 1.0, where 0.0 is the
    quietest and 1.0 the loudest.</p>

    <p>Can be set, to change the volume.</p>

    <p>Throws an <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code> if the new
    value is not in the range 0.0 .. 1.0.</p>
   </dd>

   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-media-muted">muted</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns true if audio is muted, overriding the <code data-x="dom-media-volume">volume</code>
    attribute, and false if the <code data-x="dom-media-volume">volume</code> attribute is being
    honored.</p>

    <p>Can be set, to change whether the audio is muted or not.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>A <span>media element</span> has a <dfn data-x="concept-media-volume">playback volume</dfn>, which is a fraction in the range 0.0 (silent) to 1.0 (loudest).
  Initially, the volume should be 1.0, but user agents may remember the last set value across
  sessions, on a per-site basis or otherwise, so the volume may start at other values.</p>

  <p>The <dfn attribute for="HTMLMediaElement"><code data-x="dom-media-volume">volume</code></dfn>
  IDL attribute must return the <span data-x="concept-media-volume">playback volume</span> of any
  audio portions of the <span>media element</span>. On setting, if the new value is in the range 0.0
  to 1.0 inclusive, the <span>media element</span>'s <span data-x="concept-media-volume">playback
  volume</span> must be set to the new value. If the new value is outside the range 0.0 to 1.0
  inclusive, then, on setting, an <span>"<code>IndexSizeError</code>"</span>
  <code>DOMException</code> must be thrown instead.</p>

  <p>A <span>media element</span> can also be <dfn for="media element" data-x="concept-media-muted"
  export>muted</dfn>. If anything is muting the element, then it is muted. (For example, when the
  <span>direction of playback</span> is backwards, the element is muted.)</p>

  <p>The <dfn attribute for="HTMLMediaElement"><code data-x="dom-media-muted">muted</code></dfn> IDL
  attribute must return the value to which it was last set. When a <span>media element</span> is
  created, if the element has a <code data-x="attr-media-muted">muted</code> content attribute
  specified, then the <code data-x="dom-media-muted">muted</code> IDL attribute should be set to
  true; otherwise, the user agents may set the value to the user's preferred value (e.g. remembering
  the last set value across sessions, on a per-site basis or otherwise). While the <code
  data-x="dom-media-muted">muted</code> IDL attribute is set to true, the <span>media element</span>
  must be <span data-x="concept-media-muted">muted</span>.</p>

  <p>Whenever either of the values that would be returned by the <code
  data-x="dom-media-volume">volume</code> and <code data-x="dom-media-muted">muted</code> IDL
  attributes change, the user agent must <span>queue a media element task</span> given the
  <span>media element</span> to <span data-x="concept-event-fire">fire an event</span> named <code
  data-x="event-media-volumechange">volumechange</code> at the <span>media element</span>. Then, if
  the <span>media element</span> is not <span>allowed to play</span>, the user agent must run the
  <span>internal pause steps</span> for the <span>media element</span>.</p>

  <p>A user agent has an associated <dfn>volume locked</dfn> (a boolean). Its value is
  <span>implementation-defined</span> and determines whether the <span
  data-x="concept-media-volume">playback volume</span> takes effect.</p>

  <p>An element's <dfn for=HTMLMediaElement export>effective media volume</dfn> is determined as
  follows:</p>

  <ol>
   <li><p>If the user has indicated that the user agent is to override the volume of the element,
   then return the volume desired by the user.</p></li>

   <li><p>If the user agent's <span>volume locked</span> is true, then return the system
   volume.</p></li>

   <li><p>If the element's audio output is <span data-x="concept-media-muted">muted</span>, then
   return zero.</p></li>

   <li><p>Let <var>volume</var> be the <span data-x="concept-media-volume">playback
   volume</span> of the audio portions of the <span>media element</span>, in range 0.0 (silent) to
   1.0 (loudest).</p></li>

   <li><p>Return <var>volume</var>, interpreted relative to the range 0.0 to 1.0, with 0.0 being
   silent, and 1.0 being the loudest setting, values in between increasing in loudness. The range
   need not be linear. The loudest setting may be lower than the system's loudest possible setting;
   for example the user could have set a maximum volume.</p></li>
  </ol>

  </div>

  <p>The <dfn element-attr for="audio,video"><code data-x="attr-media-muted">muted</code></dfn>
  content attribute on <span data-x="media element">media elements</span> is a <span>boolean
  attribute</span> that controls the default state of the audio output of the <span>media
  resource</span>, potentially overriding user preferences.</p>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLMediaElement"><code
  data-x="dom-media-defaultMuted">defaultMuted</code></dfn> IDL attribute must <span>reflect</span>
  the <code data-x="attr-media-muted">muted</code> content attribute.</p>

  </div>

  <p class="note">This attribute has no dynamic effect (it only controls the default state of the
  element).</p>

  <div class="example">

   <p>This video (an advertisement) autoplays, but to avoid annoying users, it does so without
   sound, and allows the user to turn the sound on. The user agent can pause the video if it's
   unmuted without a user interaction.</p>

   <pre><code class="html">&lt;video src="adverts.cgi?kind=video" controls autoplay loop muted>&lt;/video></code></pre>

  </div>




  <h5>Time ranges</h5>

  <p>Objects implementing the <code>TimeRanges</code> interface
  represent a list of ranges (periods) of time.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>TimeRanges</dfn> {
  readonly attribute unsigned long <span data-x="dom-TimeRanges-length">length</span>;
  double <span data-x="dom-TimeRanges-start">start</span>(unsigned long index);
  double <span data-x="dom-TimeRanges-end">end</span>(unsigned long index);
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>media</var>.<span subdfn data-x="dom-TimeRanges-length">length</span></code></dt>

   <dd><p>Returns the number of ranges in the object.</p></dd>

   <dt><code data-x=""><var>time</var> = <var>media</var>.<span subdfn data-x="dom-TimeRanges-start">start</span>(<var>index</var>)</code></dt>

   <dd>
    <p>Returns the time for the start of the range with the given index.</p>

    <p>Throws an <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code> if the index
    is out of range.</p>
   </dd>

   <dt><code data-x=""><var>time</var> = <var>media</var>.<span subdfn data-x="dom-TimeRanges-end">end</span>(<var>index</var>)</code></dt>

   <dd>
    <p>Returns the time for the end of the range with the given index.</p>

    <p>Throws an <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code> if the index
    is out of range.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="TimeRanges"><code data-x="dom-TimeRanges-length">length</code></dfn>
  IDL attribute must return the number of ranges represented by the object.</p>

  <p>The <dfn method for="TimeRanges"><code
  data-x="dom-TimeRanges-start">start(<var>index</var>)</code></dfn> method must return the position
  of the start of the <var>index</var>th range represented by the object, in seconds measured from
  the start of the timeline that the object covers.</p>

  <p>The <dfn method for="TimeRanges"><code
  data-x="dom-TimeRanges-end">end(<var>index</var>)</code></dfn> method must return the position of
  the end of the <var>index</var>th range represented by the object, in seconds measured from the
  start of the timeline that the object covers.</p>

  <p>These methods must throw <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>s
  if called with an <var>index</var> argument greater than or equal to the number of ranges
  represented by the object.</p>

  <p>When a <code>TimeRanges</code> object is said to be a <dfn
  id="normalised-timeranges-object" export><!-- en-GB -->normalized <code>TimeRanges</code> object</dfn>,
  the ranges it represents must obey the following criteria:</p>

  <ul>
   <li>The start of a range must be greater than the end of all earlier ranges.</li>

   <li>The start of a range must be less than or equal to the end of that same range.</li>
  </ul>

  <p>In other words, the ranges in such an object are ordered, don't overlap, and don't touch
  (adjacent ranges are folded into one bigger range). A range can be empty (referencing just a
  single moment in time), e.g. to indicate that only one frame is currently buffered in the case
  that the user agent has discarded the entire <span>media resource</span> except for the current
  frame, when a <span>media element</span> is paused.</p>

  <p>Ranges in a <code>TimeRanges</code> object must be inclusive.</p>

  <p class="example">Thus, the end of a range would be equal to the start of a following adjacent
  (touching but not overlapping) range. Similarly, a range covering a whole timeline anchored at
  zero would have a start equal to zero and an end equal to the duration of the timeline.</p>

  <p>The timelines used by the objects returned by the <code
  data-x="dom-media-buffered">buffered</code>, <code data-x="dom-media-seekable">seekable</code>, and
  <code data-x="dom-media-played">played</code> IDL attributes of <span data-x="media element">media
  elements</span> must be that element's <span>media timeline</span>.</p>

  </div>


  <h5>The <code>TrackEvent</code> interface</h5>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>TrackEvent</dfn> : <span>Event</span> {
  constructor(DOMString type, optional <span>TrackEventInit</span> eventInitDict = {});

  readonly attribute (<span>VideoTrack</span> or <span>AudioTrack</span> or <span>TextTrack</span>)? <span data-x="dom-TrackEvent-track">track</span>;
};

dictionary <dfn dictionary>TrackEventInit</dfn> : <span>EventInit</span> {
  (<span>VideoTrack</span> or <span>AudioTrack</span> or <span>TextTrack</span>)? track = null;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-TrackEvent-track">track</span></code></dt>

   <dd>
    <p>Returns the track object (<code>TextTrack</code>, <code>AudioTrack</code>, or
    <code>VideoTrack</code>) to which the event relates.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="TrackEvent"><code data-x="dom-TrackEvent-track">track</code></dfn>
  attribute must return the value it was initialized to. It represents the context information for
  the event.</p>

  </div>



  <h5 id="mediaevents">Events summary</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>The following events fire on <span data-x="media element">media elements</span> as part of the
  processing model described above:</p>

  <table>
   <thead>
    <tr>
     <th>Event name
     <th>Interface
     <th>Fired when...
     <th>Preconditions

   <tbody>
    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-loadstart">loadstart</code></dfn>

     <td><code>Event</code>

     <td>The user agent begins looking for <span>media data</span>, as part of the <span
     data-x="concept-media-load-algorithm">resource selection algorithm</span>.

     <td><code data-x="dom-media-networkState">networkState</code> equals <code
     data-x="dom-media-NETWORK_LOADING">NETWORK_LOADING</code>

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-progress">progress</code></dfn>

     <td><code>Event</code>

     <td>The user agent is fetching <span>media data</span>.

     <td><code data-x="dom-media-networkState">networkState</code> equals <code
     data-x="dom-media-NETWORK_LOADING">NETWORK_LOADING</code>

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-suspend">suspend</code></dfn>

     <td><code>Event</code>

     <td>The user agent is intentionally not currently fetching <span>media data</span>.

     <td><code data-x="dom-media-networkState">networkState</code> equals <code
     data-x="dom-media-NETWORK_IDLE">NETWORK_IDLE</code>

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-abort">abort</code></dfn>

     <td><code>Event</code>

     <td>The user agent stops fetching the <span>media data</span> before it is completely
     downloaded, but not due to an error.

     <td><code data-x="dom-media-error">error</code> is an object with the code <code
     data-x="dom-MediaError-MEDIA_ERR_ABORTED">MEDIA_ERR_ABORTED</code>. <code
     data-x="dom-media-networkState">networkState</code> equals either <code
     data-x="dom-media-NETWORK_EMPTY">NETWORK_EMPTY</code> or <code
     data-x="dom-media-NETWORK_IDLE">NETWORK_IDLE</code>, depending on when the download was aborted.

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-error">error</code></dfn>

     <td><code>Event</code>

     <td>An error occurs while fetching the <span>media data</span> or the type of the resource
     is not a supported media format.

     <td><code data-x="dom-media-error">error</code> is an object with the code <code
     data-x="dom-MediaError-MEDIA_ERR_NETWORK">MEDIA_ERR_NETWORK</code> or higher. <code
     data-x="dom-media-networkState">networkState</code> equals either <code
     data-x="dom-media-NETWORK_EMPTY">NETWORK_EMPTY</code> or <code
     data-x="dom-media-NETWORK_IDLE">NETWORK_IDLE</code>, depending on when the download was aborted.

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-emptied">emptied</code></dfn>

     <td><code>Event</code>

     <td>A <span>media element</span> whose <code data-x="dom-media-networkState">networkState</code>
     was previously not in the <code data-x="dom-media-NETWORK_EMPTY">NETWORK_EMPTY</code> state has
     just switched to that state (either because of a fatal error during load that's about to be
     reported, or because the <code data-x="dom-media-load">load()</code> method was invoked while
     the <span data-x="concept-media-load-algorithm">resource selection algorithm</span> was already
     running).

     <td><code data-x="dom-media-networkState">networkState</code> is <code
     data-x="dom-media-NETWORK_EMPTY">NETWORK_EMPTY</code>; all the IDL attributes are in their
     initial states.

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-stalled">stalled</code></dfn>

     <td><code>Event</code>

     <td>The user agent is trying to fetch <span>media data</span>, but data is unexpectedly not
     forthcoming.

     <td><code data-x="dom-media-networkState">networkState</code> is <code
     data-x="dom-media-NETWORK_LOADING">NETWORK_LOADING</code>.

   <tbody>
    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-loadedmetadata">loadedmetadata</code></dfn>

     <td><code>Event</code>

     <td>The user agent has just determined the duration and dimensions of the <span>media
     resource</span> and <span>the text tracks are ready</span>.

     <td><code data-x="dom-media-readyState">readyState</code> is newly equal to <code
     data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code> or greater for the first time.

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-loadeddata">loadeddata</code></dfn>

     <td><code>Event</code>

     <td>The user agent can render the <span>media data</span> at the <span>current playback
     position</span> for the first time.

     <td><code data-x="dom-media-readyState">readyState</code> newly increased to <code
     data-x="dom-media-HAVE_CURRENT_DATA">HAVE_CURRENT_DATA</code> or greater for the first time.

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-canplay">canplay</code></dfn>

     <td><code>Event</code>

     <td>The user agent can resume playback of the <span>media data</span>, but estimates that if
     playback were to be started now, the <span>media resource</span> could not be rendered at the
     current playback rate up to its end without having to stop for further buffering of content.

     <td><code data-x="dom-media-readyState">readyState</code> newly increased to <code
     data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code> or greater.

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-canplaythrough">canplaythrough</code></dfn>

     <td><code>Event</code>

     <td>The user agent estimates that if playback were to be started now, the <span>media
     resource</span> could be rendered at the current playback rate all the way to its end without
     having to stop for further buffering.

     <td><code data-x="dom-media-readyState">readyState</code> is newly equal to <code
     data-x="dom-media-HAVE_ENOUGH_DATA">HAVE_ENOUGH_DATA</code>.

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-playing">playing</code></dfn>

     <td><code>Event</code>

     <td>Playback is ready to start after having been paused or delayed due to lack of <span>media
     data</span>.

     <td><code data-x="dom-media-readyState">readyState</code> is newly greater than or equal to
     <code data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code> and <code
     data-x="dom-media-paused">paused</code> is false, or <code
     data-x="dom-media-paused">paused</code> is newly false and <code
     data-x="dom-media-readyState">readyState</code> is greater than or equal to <code
     data-x="dom-media-HAVE_FUTURE_DATA">HAVE_FUTURE_DATA</code>. Even if this event fires, the
     element might still not be <span>potentially playing</span>, e.g. if the element is
     <span>paused for user interaction</span> or <span>paused for in-band content</span>.

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-waiting">waiting</code></dfn>

     <td><code>Event</code>

     <td>Playback has stopped because the next frame is not available, but the user agent expects
     that frame to become available in due course.

     <td><code data-x="dom-media-readyState">readyState</code> is less than or equal to <code
     data-x="dom-media-HAVE_CURRENT_DATA">HAVE_CURRENT_DATA</code>, and <code
     data-x="dom-media-paused">paused</code> is false. Either <code
     data-x="dom-media-seeking">seeking</code> is true, or the <span>current playback position</span>
     is not contained in any of the ranges in <code data-x="dom-media-buffered">buffered</code>. It
     is possible for playback to stop for other reasons without <code
     data-x="dom-media-paused">paused</code> being false, but those reasons do not fire this event
     (and when those situations resolve, a separate <code data-x="event-media-playing">playing</code>
     event is not fired either): e.g., <span data-x="ended playback">playback has ended</span>, or
     playback <span>stopped due to errors</span>, or the element has <span>paused for user
     interaction</span> or <span>paused for in-band content</span>.

   <tbody>
    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-seeking">seeking</code></dfn>

     <td><code>Event</code>

     <td>The <code data-x="dom-media-seeking">seeking</code> IDL attribute changed to true, and the user agent has started seeking to a new position.

     <td>

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-seeked">seeked</code></dfn>

     <td><code>Event</code>

     <td>The <code data-x="dom-media-seeking">seeking</code> IDL attribute changed to false after the <span>current playback position</span> was changed.

     <td>

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-ended">ended</code></dfn>

     <td><code>Event</code>

     <td>Playback has stopped because the end of the <span>media resource</span> was reached.

     <td><code data-x="dom-media-currentTime">currentTime</code> equals the end of the <span>media
     resource</span>; <code data-x="dom-media-ended">ended</code> is true.

   <tbody>
    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-durationchange">durationchange</code></dfn>

     <td><code>Event</code>

     <td>The <code data-x="dom-media-duration">duration</code> attribute has just been updated.

     <td>

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-timeupdate">timeupdate</code></dfn>

     <td><code>Event</code>

     <td>The <span>current playback position</span> changed as part of normal playback or in an especially interesting way, for example discontinuously.

     <td>

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-play">play</code></dfn>

     <td><code>Event</code>

     <td>The element is no longer paused. Fired after the <code data-x="dom-media-play">play()</code>
     method has returned, or when the <code data-x="attr-media-autoplay">autoplay</code> attribute
     has caused playback to begin.

     <td><code data-x="dom-media-paused">paused</code> is newly false.

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-pause">pause</code></dfn>

     <td><code>Event</code>

     <td>The element has been paused. Fired after the <code data-x="dom-media-pause">pause()</code>
     method has returned.

     <td><code data-x="dom-media-paused">paused</code> is newly true.

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-ratechange">ratechange</code></dfn>

     <td><code>Event</code>

     <td>Either the <code data-x="dom-media-defaultPlaybackRate">defaultPlaybackRate</code> or the
     <code data-x="dom-media-playbackRate">playbackRate</code> attribute has just been updated.

     <td>

   <tbody>
    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-resize">resize</code></dfn>

     <td><code>Event</code>

     <td>One or both of the <code data-x="dom-video-videoWidth">videoWidth</code> and <code
     data-x="dom-video-videoHeight">videoHeight</code> attributes have just been updated.

     <td><span>Media element</span> is a <code>video</code> element; <code
     data-x="dom-media-readyState">readyState</code> is not <code
     data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</code>

    <tr>

     <td><dfn event for="HTMLMediaElement"><code data-x="event-media-volumechange">volumechange</code></dfn>

     <td><code>Event</code>

     <td>Either the <code data-x="dom-media-volume">volume</code> attribute or the <code
     data-x="dom-media-muted">muted</code> attribute has changed. Fired after the relevant
     attribute's setter has returned.

     <td>

  </table>


  <p>The following event fires on <code>source</code> elements:</p>

  <table>

   <thead>
    <tr>

     <th>Event name

     <th>Interface

     <th>Fired when...

   <tbody>
    <tr>

     <td><dfn event for="HTMLSourceElement"><code data-x="event-source-error">error</code></dfn>

     <td><code>Event</code>

     <td>An error occurs while fetching the <span>media data</span> or the type of the resource
     is not a supported media format.

  </table>


  <p>The following events fire on <code>AudioTrackList</code>, <code>VideoTrackList</code>, and
  <code>TextTrackList</code> objects:</p>

  <table>

   <thead>
    <tr>

     <th>Event name

     <th>Interface

     <th>Fired when...

   <tbody>
    <tr>

     <td><dfn event for="AudioTrackList,VideoTrackList,TextTrackList"><code data-x="event-media-change">change</code></dfn>

     <td><code>Event</code>

     <td>One or more tracks in the track list have been enabled or disabled.

    <tr>

     <td><dfn event for="AudioTrackList,VideoTrackList,TextTrackList"><code data-x="event-media-addtrack">addtrack</code></dfn>

     <td><code>TrackEvent</code>

     <td>A track has been added to the track list.

    <tr>

     <td><dfn event for="AudioTrackList,VideoTrackList,TextTrackList"><code data-x="event-media-removetrack">removetrack</code></dfn>

     <td><code>TrackEvent</code>

     <td>A track has been removed from the track list.

  </table>



  <p>The following event fires on <code>TextTrack</code> objects and <code>track</code> elements:</p>

  <table>

   <thead>
    <tr>

     <th>Event name

     <th>Interface

     <th>Fired when...

   <tbody>
    <tr>

     <td><dfn event for="TextTrack,HTMLTrackElement"><code data-x="event-media-cuechange">cuechange</code></dfn>

     <td><code>Event</code>

     <td>One or more cues in the track have become active or stopped being active.

  </table>


  <p>The following events fire on <code>track</code> elements:</p>

  <table>

   <thead>
    <tr>

     <th>Event name

     <th>Interface

     <th>Fired when...

   <tbody>
    <tr>

     <td><dfn event for="HTMLTrackElement"><code data-x="event-track-error">error</code></dfn>

     <td><code>Event</code>

     <td>An error occurs while fetching the track data or the type of the resource is not supported text track format.

    <tr>

     <td><dfn event for="HTMLTrackElement"><code data-x="event-track-load">load</code></dfn>

     <td><code>Event</code>

     <td>A track data has been fetched and successfully processed.
  </table>


  <p>The following events fire on <code>TextTrackCue</code> objects:</p>

  <table>

   <thead>
    <tr>

     <th>Event name

     <th>Interface

     <th>Fired when...

   <tbody>
    <tr>

     <td><dfn event for="TextTrackCue"><code data-x="event-media-enter">enter</code></dfn>

     <td><code>Event</code>

     <td>The cue has become active.

    <tr>

     <td><dfn event for="TextTrackCue"><code data-x="event-media-exit">exit</code></dfn>

     <td><code>Event</code>

     <td>The cue has stopped being active.

  </table>


  <div w-nodev>

  <h5>Security and privacy considerations</h5>

  <p>The main security and privacy implications of the <code>video</code> and <code>audio</code>
  elements come from the ability to embed media cross-origin. There are two directions that threats
  can flow: from hostile content to a victim page, and from a hostile page to victim content.</p>

  <hr>

  <p>If a victim page embeds hostile content, the threat is that the content might contain scripted
  code that attempts to interact with the <code>Document</code> that embeds the content. To avoid
  this, user agents must ensure that there is no access from the content to the embedding page. In
  the case of media content that uses DOM concepts, the embedded content must be treated as if it
  was in its own unrelated <span>top-level traversable</span>.</p>

  <p class="example">For instance, if an SVG animation was embedded in a <code>video</code> element,
  the user agent would not give it access to the DOM of the outer page. From the perspective of
  scripts in the SVG resource, the SVG file would appear to be in a lone top-level traversable
  with no parent.</p>

  <hr>

  <p>If a hostile page embeds victim content, the threat is that the embedding page could obtain
  information from the content that it would not otherwise have access to. The API does expose some
  information: the existence of the media, its type, its duration, its size, and the performance
  characteristics of its host. Such information is already potentially problematic, but in practice
  the same information can more or less be obtained using the <code>img</code> element, and so it
  has been deemed acceptable.</p>

  <p>However, significantly more sensitive information could be obtained if the user agent further
  exposes metadata within the content, such as subtitles. That information is therefore only exposed
  if the video resource uses CORS. The <code data-x="attr-media-crossorigin">crossorigin</code>
  attribute allows authors to enable CORS. <ref>FETCH</ref></p>

  <p class="example">Without this restriction, an attacker could trick a user running within a
  corporate network into visiting a site that attempts to load a video from a previously leaked
  location on the corporation's intranet. If such a video included confidential plans for a new
  product, then being able to read the subtitles would present a serious confidentiality breach.</p>

  </div>


  <h5>Best practices for authors using media elements</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>Playing audio and video resources on small devices such as set-top boxes or mobile phones is
  often constrained by limited hardware resources in the device. For example, a device might only
  support three simultaneous videos. For this reason, it is a good practice to release resources
  held by <span data-x="media element">media elements</span> when they are done playing, either by
  being very careful about removing all references to the element and allowing it to be garbage
  collected, or, even better, by setting the element's <code data-x="attr-media-src">src</code>
  attribute to an empty string. In cases where <code data-x="dom-media-srcObject">srcObject</code>
  was set, instead set the <code data-x="dom-media-srcObject">srcObject</code> to null.</p>

  <p>Similarly, when the playback rate is not exactly 1.0, hardware, software, or format limitations
  can cause video frames to be dropped and audio to be choppy or muted.</p>


  <div w-nodev>

  <h5>Best practices for implementers of media elements</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>How accurately various aspects of the <span>media element</span> API are implemented is
  considered a quality-of-implementation issue.</p>

  <p>For example, when implementing the <code data-x="dom-media-buffered">buffered</code> attribute,
  how precise an implementation reports the ranges that have been buffered depends on how carefully
  the user agent inspects the data. Since the API reports ranges as times, but the data is obtained
  in byte streams, a user agent receiving a variable-bitrate stream might only be able to determine
  precise times by actually decoding all of the data. User agents aren't required to do this,
  however; they can instead return estimates (e.g. based on the average bitrate seen so far) which
  get revised as more information becomes available.</p>

  <p>As a general rule, user agents are urged to be conservative rather than optimistic. For
  example, it would be bad to report that everything had been buffered when it had not.</p>

  <p>Another quality-of-implementation issue would be playing a video backwards when the codec is
  designed only for forward playback (e.g. there aren't many key frames, and they are far apart, and
  the intervening frames only have deltas from the previous frame). User agents could do a poor job,
  e.g. only showing key frames; however, better implementations would do more work and thus do a
  better job, e.g. actually decoding parts of the video forwards, storing the complete frames, and
  then playing the frames backwards.</p>

  <p>Similarly, while implementations are allowed to drop buffered data at any time (there is no
  requirement that a user agent keep all the media data obtained for the lifetime of the media
  element), it is again a quality of implementation issue: user agents with sufficient resources to
  keep all the data around are encouraged to do so, as this allows for a better user experience. For
  example, if the user is watching a live stream, a user agent could allow the user only to view the
  live video; however, a better user agent would buffer everything and allow the user to seek
  through the earlier material, pause it, play it forwards and backwards, etc.</p>

  <hr>

  <p>When a <span>media element</span> that is paused is <span data-x="node is removed from a
  document">removed from a document</span> and not reinserted before the next time the <span>event
  loop</span> reaches <a href="#step1">step 1</a>, implementations that are resource constrained are encouraged to take
  that opportunity to release all hardware resources (like video planes, networking resources, and
  data buffers) used by the <span>media element</span>. (User agents still have to keep track of the
  playback position and so forth, though, in case playback is later restarted.)</p>

  </div>



  <h4 split-filename="image-maps">The <dfn element><code>map</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Transparent</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-map-name">name</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-map">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-map">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLMapElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-map-name">name</span>;
  [SameObject] readonly attribute <span>HTMLCollection</span> <span data-x="dom-map-areas">areas</span>;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLMapElement</code>.</dd>
  </dl>

  <p>The <code>map</code> element, in conjunction with an <code>img</code> element and any
  <code>area</code> element descendants, defines an <span>image map</span>. The element
  <span>represents</span> its children.</p>

  <p>The <dfn element-attr for="map"><code data-x="attr-map-name">name</code></dfn> attribute gives
  the map a name so that it can be <span>referenced</span>. The attribute must be present and must
  have a non-empty value with no <span>ASCII whitespace</span>. The value of the <code
  data-x="attr-map-name">name</code> attribute must not be equal to the value of the <code
  data-x="attr-map-name">name</code> attribute of another <code>map</code> element in the same
  <span>tree</span>. If the <code data-x="attr-id">id</code> attribute is also specified, both
  attributes must have the same value.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>map</var>.<span subdfn data-x="dom-map-areas">areas</span></code></dt>

   <dd>
    <p>Returns an <code>HTMLCollection</code> of the <code>area</code> elements in the
    <code>map</code>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLMapElement"><code data-x="dom-map-areas">areas</code></dfn>
  attribute must return an <code>HTMLCollection</code> rooted at the <code>map</code> element, whose
  filter matches only <code>area</code> elements.</p>

  <p>The IDL attribute <dfn attribute for="HTMLMapElement"><code
  data-x="dom-map-name">name</code></dfn> must <span>reflect</span> the content attribute of the
  same name.</p>

  </div>

  <div class="example">

   <p>Image maps can be defined in conjunction with other content on the page, to ease maintenance.
   This example is of a page with an image map at the top of the page and a corresponding set of
   text links at the bottom.</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;HTML LANG="EN">
&lt;TITLE>Babies&trade;: Toys&lt;/TITLE>
&lt;HEADER>
 &lt;H1>Toys&lt;/H1>
 &lt;IMG SRC="/images/menu.gif"
      ALT="Babies&trade; navigation menu. Select a department to go to its page."
      USEMAP="#NAV">
&lt;/HEADER>
 ...
&lt;FOOTER>
 &lt;MAP NAME="NAV">
  &lt;P>
   &lt;A HREF="/clothes/">Clothes&lt;/A>
   &lt;AREA ALT="Clothes" COORDS="0,0,100,50" HREF="/clothes/"> |
   &lt;A HREF="/toys/">Toys&lt;/A>
   &lt;AREA ALT="Toys" COORDS="100,0,200,50" HREF="/toys/"> |
   &lt;A HREF="/food/">Food&lt;/A>
   &lt;AREA ALT="Food" COORDS="200,0,300,50" HREF="/food/"> |
   &lt;A HREF="/books/">Books&lt;/A>
   &lt;AREA ALT="Books" COORDS="300,0,400,50" HREF="/books/">
  &lt;/P>
 &lt;/MAP>
&lt;/FOOTER></code></pre>

  </div>



  <h4>The <dfn element><code>area</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected, but only if there is a <code>map</code> element ancestor.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="concept-content-nothing">Nothing</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-area-alt">alt</code></dd>
   <dd><code data-x="attr-area-coords">coords</code></dd>
   <dd><code data-x="attr-area-shape">shape</code></dd>
   <dd><code data-x="attr-hyperlink-href">href</code></dd>
   <dd><code data-x="attr-hyperlink-target">target</code></dd>
   <dd><code data-x="attr-hyperlink-download">download</code></dd>
   <dd><code data-x="attr-hyperlink-ping">ping</code></dd>
   <dd><code data-x="attr-hyperlink-rel">rel</code></dd>
   <dd><code data-x="attr-hyperlink-referrerpolicy">referrerpolicy</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd>If the element has an <code data-x="attr-hyperlink-href">href</code> attribute: <a
   href="https://w3c.github.io/html-aria/#el-area">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-area">for implementers</a>.</dd>
   <dd>Otherwise: <a
   href="https://w3c.github.io/html-aria/#el-area-no-href">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-area-no-href">for implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLAreaElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-area-alt">alt</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-area-coords">coords</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-area-shape">shape</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-area-target">target</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-area-download">download</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-area-ping">ping</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-area-rel">rel</span>;
  [SameObject, PutForwards=<span data-x="dom-DOMTokenList-value">value</span>] readonly attribute <span>DOMTokenList</span> <span data-x="dom-area-relList">relList</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-area-referrerPolicy">referrerPolicy</span>;

  // <a href="#HTMLAreaElement-partial">also has obsolete members</a>
};
<span>HTMLAreaElement</span> includes <span>HTMLHyperlinkElementUtils</span>;</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLAreaElement</code>.</dd>
  </dl>

  <p>The <code>area</code> element <span>represents</span> either a hyperlink with some text and a
  corresponding area on an <span>image map</span>, or a dead area on an image map.</p>

  <p>An <code>area</code> element with a parent node must have a <code>map</code> element
  ancestor.</p>

  <p>If the <code>area</code> element has an <code data-x="attr-hyperlink-href">href</code>
  attribute, then the <code>area</code> element represents a <span>hyperlink</span>. In this case,
  the <dfn element-attr for="area"><code data-x="attr-area-alt">alt</code></dfn> attribute must be
  present. It specifies the text of the hyperlink. Its value must be text that, when presented with
  the texts specified for the other hyperlinks of the <span>image map</span>, and with the
  alternative text of the image, but without the image itself, provides the user with the same kind
  of choice as the hyperlink would when used without its text but with its shape applied to the
  image. The <code data-x="attr-area-alt">alt</code> attribute may be left blank if there is another
  <code>area</code> element in the same <span>image map</span> that points to the same resource and
  has a non-blank <code data-x="attr-area-alt">alt</code> attribute.</p>

  <p>If the <code>area</code> element has no <code data-x="attr-hyperlink-href">href</code>
  attribute, then the area represented by the element cannot be selected, and the <code
  data-x="attr-area-alt">alt</code> attribute must be omitted.</p>

  <p>In both cases, the <code data-x="attr-area-shape">shape</code> and <code
  data-x="attr-area-coords">coords</code> attributes specify the area.</p>

  <p>The <dfn element-attr for="area"><code data-x="attr-area-shape">shape</code></dfn> attribute is
  an <span>enumerated attribute</span> with the following keywords and states:</p>

  <table class="yesno">
   <thead>
    <tr>
     <th>Keyword
     <th w-nodev>Conforming
     <th>State
     <th w-nodev>Brief description
   <tbody>
    <tr>
     <td><dfn><code data-x="attr-area-shape-keyword-circle">circle</code></dfn>
     <td w-nodev>
     <td w-nodev rowspan=2><span data-x="attr-area-shape-circle">Circle state</span>
     <td w-dev><span data-x="attr-area-shape-circle">Circle state</span>
     <td w-nodev rowspan=2>Designates a circle, using exactly three integers in the <code
     data-x="attr-area-coords">coords</code> attribute.
    <tr w-nodev>
     <td><dfn><code data-x="attr-area-shape-keyword-circ">circ</code></dfn>
     <td class="no">No
    <tr>
     <td><dfn><code data-x="attr-area-shape-keyword-default">default</code></dfn>
     <td w-nodev>
     <td><span data-x="attr-area-shape-default">Default state</span>
     <td w-nodev>This area is the whole image. (The <code data-x="attr-area-coords">coords</code>
     attribute is not used.)
    <tr>
     <td><dfn><code data-x="attr-area-shape-keyword-poly">poly</code></dfn>
     <td w-nodev>
     <td w-nodev rowspan=2><span data-x="attr-area-shape-poly">Polygon state</span>
     <td w-dev><span data-x="attr-area-shape-poly">Polygon state</span>
     <td w-nodev rowspan=2>Designates a polygon, using at-least six integers in the <code
     data-x="attr-area-coords">coords</code> attribute.
    <tr w-nodev>
     <td><dfn><code data-x="attr-area-shape-keyword-polygon">polygon</code></dfn>
     <td class="no">No
    <tr>
     <td><dfn><code data-x="attr-area-shape-keyword-rect">rect</code></dfn>
     <td w-nodev>
     <td w-nodev rowspan=2><span data-x="attr-area-shape-rect">Rectangle state</span>
     <td w-dev><span data-x="attr-area-shape-rect">Rectangle state</span>
     <td w-nodev rowspan=2>Designates a rectangle, using exactly four integers in the <code
     data-x="attr-area-coords">coords</code> attribute.
    <tr w-nodev>
     <td><dfn><code data-x="attr-area-shape-keyword-rectangle">rectangle</code></dfn>
     <td class="no">No
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <span
  data-x="attr-area-shape-rect">rectangle</span> state.</p>

  <p>The <dfn element-attr for="area"><code data-x="attr-area-coords">coords</code></dfn> attribute
  must, if specified, contain a <span>valid list of floating-point numbers</span>. This attribute
  gives the coordinates for the shape described by the <code data-x="attr-area-shape">shape</code>
  attribute. <span w-nodev>The processing for this attribute is described as part of the <span>image
  map</span> processing model.</span></p>

  <!-- v2: It was suggested by John S. Urban that coords should support percentages as well as
  pixels, so that one could use the same image map for images of various sizes. -->

  <p>In the <dfn attr-value for="area/shape" data-x="attr-area-shape-circle">circle state</dfn>,
  <code>area</code> elements must have a <code data-x="attr-area-coords">coords</code> attribute
  present, with three integers, the last of which must be non-negative. The first integer must be
  the distance in <span data-x="'px'">CSS pixels</span> from the left edge of the image to the
  center of the circle, the second integer must be the distance in <span data-x="'px'">CSS
  pixels</span> from the top edge of the image to the center of the circle, and the third integer
  must be the radius of the circle, again in <span data-x="'px'">CSS pixels</span>.</p>

  <p>In the <dfn attr-value for="area/shape" data-x="attr-area-shape-default">default state</dfn>,
  <code>area</code> elements must not have a <code data-x="attr-area-coords">coords</code>
  attribute. (The area is the whole image.)</p>

  <p>In the <dfn attr-value for="area/shape" data-x="attr-area-shape-poly">polygon state</dfn>,
  <code>area</code> elements must have a <code data-x="attr-area-coords">coords</code> attribute
  with at least six integers, and the number of integers must be even. Each pair of integers must
  represent a coordinate given as the distances from the left and the top of the image in <span
  data-x="'px'">CSS pixels</span> respectively, and all the coordinates together must represent the
  points of the polygon, in order.</p>

  <p>In the <dfn attr-value for="area/shape" data-x="attr-area-shape-rect">rectangle state</dfn>,
  <code>area</code> elements must have a <code data-x="attr-area-coords">coords</code> attribute
  with exactly four integers, the first of which must be less than the third, and the second of
  which must be less than the fourth. The four points must represent, respectively, the distance
  from the left edge of the image to the left side of the rectangle, the distance from the top edge
  to the top side, the distance from the left edge to the right side, and the distance from the top
  edge to the bottom side, all in <span data-x="'px'">CSS pixels</span>.</p>

  <div w-nodev>

  <p>When user agents allow users to <span data-x="following hyperlinks">follow hyperlinks</span> or
  <span data-x="downloading hyperlinks">download hyperlinks</span> created using the
  <code>area</code> element, the <code data-x="attr-hyperlink-href">href</code>, <code
  data-x="attr-hyperlink-target">target</code>, <code
  data-x="attr-hyperlink-download">download</code>, and <code
  data-x="attr-hyperlink-ping">ping</code> attributes decide how the link is followed. The <code
  data-x="attr-hyperlink-rel">rel</code> attribute may be used to indicate to the user the likely
  nature of the target resource before the user follows the link.</p>

  </div>

  <p>The <code data-x="attr-hyperlink-target">target</code>, <code
  data-x="attr-hyperlink-download">download</code>, <code data-x="attr-hyperlink-ping">ping</code>,
  <code data-x="attr-hyperlink-rel">rel</code>, and <code
  data-x="attr-hyperlink-referrerpolicy">referrerpolicy</code> attributes must be omitted if the
  <code data-x="attr-hyperlink-href">href</code> attribute is not present.</p>

  <p>If the <code data-x="attr-itemprop">itemprop</code> attribute is specified on an
  <code>area</code> element, then the <code data-x="attr-hyperlink-href">href</code> attribute must
  also be specified.</p>

  <div w-nodev>

  <p>The IDL attributes <dfn attribute for="HTMLAreaElement"><code
  data-x="dom-area-alt">alt</code></dfn>, <dfn attribute for="HTMLAreaElement"><code
  data-x="dom-area-coords">coords</code></dfn>, <dfn attribute for="HTMLAreaElement"><code
  data-x="dom-area-shape">shape</code></dfn>, <dfn attribute for="HTMLAreaElement"><code
  data-x="dom-area-target">target</code></dfn>, <dfn attribute for="HTMLAreaElement"><code
  data-x="dom-area-download">download</code></dfn>, <dfn attribute for="HTMLAreaElement"><code
  data-x="dom-area-ping">ping</code></dfn>, and <dfn attribute for="HTMLAreaElement"><code
  data-x="dom-area-rel">rel</code></dfn>, each must <span>reflect</span> the respective content
  attributes of the same name.</p>

  <p>The IDL attribute <dfn attribute for="HTMLAreaElement"><code
  data-x="dom-area-rellist">relList</code></dfn> must <span>reflect</span> the <code
  data-x="attr-hyperlink-rel">rel</code> content attribute.</p>

  <p>The IDL attribute <dfn attribute for="HTMLAreaElement"><code
  data-x="dom-area-referrerPolicy">referrerPolicy</code></dfn> must <span>reflect</span> the <code
  data-x="attr-hyperlink-referrerpolicy">referrerpolicy</code> content attribute, <span>limited to
  only known values</span>.</p>

  </div>



  <h4 id="image-maps">Image maps</h4>

  <!-- TESTS
  https://software.hixie.ch/utilities/js/live-dom-viewer/?%3C%21DOCTYPE%20html%3E%0A%3Cimg%20src%3D%22http%3A//hixie.ch/resources/images/smallcats%22%20usemap%3D%23a%20onclick%3Dw%28%27img%27%29%3E%0A%3Cmap%20name%3Da%3E%0A%20%3Carea%20onclick%3Dw%28%271%27%29%20coords%3D%270%25%200%25%20100%25%20100%25%27%20href%3Djavascript%3A%3E%0A%3C/map%3E
  https://software.hixie.ch/utilities/js/live-dom-viewer/?%3C%21DOCTYPE%20html%3E%0A%3Cbody%20onfocus%3D%22w%28document.activeElement.tagName%29%22%3E%0A%3Cimg%20src%3D%22http%3A//hixie.ch/resources/images/smallcats%22%20usemap%3D%23a%20onclick%3Dw%28%27img%27%29%20onfocus%3D%22w%28document.activeElement.tagName%29%22%3E%0A%3Cimg%20src%3D%22http%3A//hixie.ch/resources/images/sample%22%20usemap%3D%23a%20onclick%3Dw%28%27img%27%29%20onfocus%3D%22w%28document.activeElement.tagName%29%22%3E%0A%3Cmap%20name%3Da%20onfocus%3D%22w%28document.activeElement.tagName%29%22%3E%0A%20%3Carea%20onclick%3Dw%28%271%27%29%20coords%3D%270%200%2050%2050%27%20href%3Djavascript%3A%20onfocus%3D%22w%28document.activeElement.tagName%29%22%3E%0A%3C/map%3E%0A%3Cscript%3E%0A%20var%20x%20%3D%20document.getElementsByTagName%28%27img%27%29%5B0%5D%3B%0A%20x.parentNode.appendChild%28x%29%3B%0A%20document.getElementsByTagName%28%27area%27%29%5B0%5D.focus%28%29%3B%0A%3C/script%3E
  https://software.hixie.ch/utilities/js/live-dom-viewer/?%3C%21DOCTYPE%20html%3Ex%3Cmap%3E%3Carea%20shape%3Dpolyg%20coords%3D%221%2C2%203%22%3E%3C/map%3E%0A%3Cscript%3Ex%20%3D%20document.getElementsByTagName%28%27area%27%29%5B0%5D%3B%20w%28x.shape%20+%20%27%20%27%20+%20x.coords%29%3C/script%3E
  https://software.hixie.ch/utilities/js/live-dom-viewer/?%3C%21DOCTYPE%20html%3E%0D%0A%3Cp%3E%3Cimg%20src%3D%22http%3A//hixie.ch/resources/images/astrophy/128%22%20usemap%3D%23a%3E%0D%0A%3Cmap%20name%3Da%3E%3Carea%20shape%3Dcirc%20coords%3D%2220%2C20%2C10%25%22%20href%3D%23%3E%3Carea%20shape%3Dcirc%20coords%3D%2220%2C20%2C10%22%20href%3D%23%3E%3C/map%3E%0D%0A%3Cscript%3Edocument.write%28document.getElementsByTagName%28%27area%27%29%5B0%5D.coords%29%3C/script%3E
  -->

  <div w-nodev>

  <h5>Authoring</h5>

  </div>

  <p>An <dfn>image map</dfn> allows geometric areas on an image to be associated with <span
  data-x="hyperlink">hyperlinks</span>.</p>

  <p>An image, in the form of an <code>img</code> element, may be associated with an image map (in
  the form of a <code>map</code> element) by specifying a <dfn element-attr for="img"><code
  data-x="attr-hyperlink-usemap">usemap</code></dfn> attribute on the <code>img</code> element. The
  <code data-x="attr-hyperlink-usemap">usemap</code> attribute, if specified, must be a <span>valid
  hash-name reference</span> to a <code>map</code> element.</p>

  <div class="example">

   <p>Consider an image that looks as follows:</p>

   <p><img src="/images/sample-usemap.png" width="600" height="150" alt="A line with four shapes in it, equally spaced: a red hollow box, a green circle, a blue triangle, and a yellow four-pointed star."></p>

   <p>If we wanted just the colored areas to be clickable, we could do it as follows:</p>

   <pre><code class="html">&lt;p>
 Please select a shape:
 &lt;img src="shapes.png" usemap="#shapes"
      alt="Four shapes are available: a red hollow box, a green circle, a blue triangle, and a yellow four-pointed star.">
 &lt;map name="shapes">
  &lt;area shape=rect coords="50,50,100,100"> &lt;!-- the hole in the red box -->
  &lt;area shape=rect coords="25,25,125,125" href="red.html" alt="Red box.">
  &lt;area shape=circle coords="200,75,50" href="green.html" alt="Green circle.">
  &lt;area shape=poly coords="325,25,262,125,388,125" href="blue.html" alt="Blue triangle.">
  &lt;area shape=poly coords="450,25,435,60,400,75,435,90,450,125,465,90,500,75,465,60"
        href="yellow.html" alt="Yellow star.">
 &lt;/map>
&lt;/p></code></pre>

  </div>

  <div w-nodev>

  <h5 id="image-map-processing-model"><span id="processing-model"></span>Processing model</h5>

  <p>If an <code>img</code> element has a <code data-x="attr-hyperlink-usemap">usemap</code>
  attribute specified, user agents must process it as follows:</p>

  <ol>
   <li><p>Parse the attribute's value using the <span>rules for parsing a hash-name reference</span>
   to a <code>map</code> element, with the element as the context node. This will return either an
   element (the <var>map</var>) or null.</p></li>

   <li><p>If that returned null, then return. The image is not associated with an image
   map after all.</p></li>

   <li><p>Otherwise, the user agent must collect all the <code>area</code> elements that are
   descendants of the <var>map</var>. Let those be the <var>areas</var>.</p></li>
  </ol>

  <p>Having obtained the list of <code>area</code> elements that form the image map (the <var>areas</var>), interactive user agents must process the list in one of two ways.</p>

  <p>If the user agent intends to show the text that the <code>img</code> element represents, then
  it must use the following steps.</p>

  <ol>
   <li><p>Remove all the <code>area</code> elements in <var>areas</var> that have no <code
   data-x="attr-hyperlink-href">href</code> attribute.</p></li>

   <li><p>Remove all the <code>area</code> elements in <var>areas</var> that have no <code
   data-x="attr-area-alt">alt</code> attribute, or whose <code data-x="attr-area-alt">alt</code>
   attribute's value is the empty string, <em>if</em> there is another <code>area</code> element in
   <var>areas</var> with the same value in the <code
   data-x="attr-hyperlink-href">href</code> attribute and with a non-empty <code
   data-x="attr-area-alt">alt</code> attribute.</li>

   <li><p>Each remaining <code>area</code> element in <var>areas</var> represents a
   <span>hyperlink</span>. Those hyperlinks should all be made available to the user in a manner
   associated with the text of the <code>img</code>.</p>

   <p>In this context, user agents may represent <code>area</code> and <code>img</code> elements
   with no specified <code data-x="">alt</code> attributes, or whose <code data-x="">alt</code>
   attributes are the empty string or some other non-visible text, in an
   <span>implementation-defined</span> fashion intended to indicate the lack of suitable
   author-provided text.</p></li>
  </ol>

  <p>If the user agent intends to show the image and allow interaction with the image to select
  hyperlinks, then the image must be associated with a set of layered shapes, taken from the
  <code>area</code> elements in <var>areas</var>, in reverse <span>tree order</span> (so the last
  specified <code>area</code> element in the <var>map</var> is the bottom-most shape, and
  the first element in the <var>map</var>, in <span>tree order</span>, is the top-most shape).</p>

  <p>Each <code>area</code> element in <var>areas</var> must be processed as follows to
  obtain a shape to layer onto the image:</p>

  <ol>
   <li><p>Find the state that the element's <code data-x="attr-area-shape">shape</code> attribute
   represents.</p></li>

   <li><p>Use the <span>rules for parsing a list of floating-point numbers</span> to parse the
   element's <code data-x="attr-area-coords">coords</code> attribute, if it is present, and let the
   result be the <var>coords</var> list. If the attribute is absent, let the <var>coords</var> list
   be the empty list.</p></li>

   <li>
    <p>If the number of items in the <var>coords</var> list is less than the minimum number
    given for the <code>area</code> element's current state, as per the following table, then the
    shape is empty; return.</p>

    <table>
     <thead>
      <tr>
       <th>State
       <th>Minimum number of items
     <tbody>
      <tr>
       <td><span data-x="attr-area-shape-circle">Circle state</span>
       <td>3
      <tr>
       <td><span data-x="attr-area-shape-default">Default state</span>
       <td>0
      <tr>
       <td><span data-x="attr-area-shape-poly">Polygon state</span>
       <td>6
      <tr>
       <td><span data-x="attr-area-shape-rect">Rectangle state</span>
       <td>4
    </table>
   </li>

   <li>
    <p>Check for excess items in the <var>coords</var> list as per the entry in the
    following list corresponding to the <code data-x="attr-area-shape">shape</code> attribute's
    state:</p>

    <dl class="switch">
     <dt><span data-x="attr-area-shape-circle">Circle state</span></dt>
     <dd>Drop any items in the list beyond the third.</dd>
     <dt><span data-x="attr-area-shape-default">Default state</span></dt>
     <dd>Drop all items in the list.</dd>
     <dt><span data-x="attr-area-shape-poly">Polygon state</span></dt>
     <dd>Drop the last item if there's an odd number of items.</dd>
     <dt><span data-x="attr-area-shape-rect">Rectangle state</span></dt>
     <dd>Drop any items in the list beyond the fourth.</dd>
    </dl>
   </li>

   <li><p>If the <code data-x="attr-area-shape">shape</code> attribute represents the <span
   data-x="attr-area-shape-rect">rectangle state</span>, and the first number in the list is
   numerically greater than the third number in the list, then swap those two numbers around.</p></li>

   <li><p>If the <code data-x="attr-area-shape">shape</code> attribute represents the <span
   data-x="attr-area-shape-rect">rectangle state</span>, and the second number in the list is
   numerically greater than the fourth number in the list, then swap those two numbers around.</p></li>

   <li><p>If the <code data-x="attr-area-shape">shape</code> attribute represents the <span
   data-x="attr-area-shape-circle">circle state</span>, and the third number in the list is less than
   or equal to zero, then the shape is empty; return.</p></li>

   <li><p>Now, the shape represented by the element is the one described for the entry in the list
   below corresponding to the state of the <code data-x="attr-area-shape">shape</code> attribute:</p>

    <dl class="switch">

     <dt><span data-x="attr-area-shape-circle">Circle state</span></dt>
     <dd>
      <p>Let <var>x</var> be the first number in <var>coords</var>, <var>y</var> be the second number, and <var>r</var> be the third number.</p>

      <p>The shape is a circle whose center is <var>x</var> <span data-x="'px'">CSS pixels</span>
      from the left edge of the image and <var>y</var> <span data-x="'px'">CSS pixels</span> from
      the top edge of the image, and whose radius is <var>r</var> <span data-x="'px'">CSS
      pixels</span>.</p>
     </dd>

     <dt><span data-x="attr-area-shape-default">Default state</span></dt>
     <dd>
      <p>The shape is a rectangle that exactly covers the entire image.</p>
     </dd>

     <dt><span data-x="attr-area-shape-poly">Polygon state</span></dt>
     <dd>
      <p>Let <var>x<sub><var>i</var></sub></var> be the <span
      data-x="">(2<var>i</var>)</span>th entry in <var>coords</var>, and <var>y<sub><var>i</var></sub></var> be the <span data-x="">(2<var>i</var>+1)</span>th entry in <var>coords</var> (the first entry in <var>coords</var> being the one with index 0).</p>

      <p>Let <var>the coordinates</var> be (<var>x<sub><var>i</var></sub></var>, <var>y<sub><var>i</var></sub></var>),
      interpreted in <span data-x="'px'">CSS pixels</span> measured from the top left of the image,
      for all integer values of <var>i</var> from 0 to <span data-x="">(<var>N</var>/2)-1</span>,
      where <var>N</var> is the number of items in <var>coords</var>.</p>

      <p>The shape is a polygon whose vertices are given by <var>the coordinates</var>, and whose
      interior is established using the even-odd rule. <ref>GRAPHICS</ref></p>

      <!--
        browsers implement the even-odd rule / even winding rule:
        https://software.hixie.ch/utilities/js/live-dom-viewer/?%3C%21DOCTYPE%20html%3E%0A%3Cimg%20usemap%3D%22%23x%22%20src%3D%22/resources/images/sample%22%3E%0A%3Cmap%20name%3D%22x%22%3E%0A%20%20%3Carea%20shape%3Dpolygon%20coords%3D%220%2C0%200%2C100%20100%2C100%20100%2C2%201%2C2%202%2C1%202%2C99%2099%2C99%2099%2C0%22%20href%3Da%3E%0A%3C/map%3E%0A
       -->
     </dd>

     <dt><span data-x="attr-area-shape-rect">Rectangle state</span></dt>

     <dd>
      <p>Let <var>x<sub>1</sub></var> be the first number in <var>coords</var>, <var>y<sub>1</sub></var> be the second number, <var>x<sub>2</sub></var> be the third number, and <var>y<sub>2</sub></var> be the fourth number.</p>

      <p>The shape is a rectangle whose top-left corner is given by the coordinate (<var>x<sub>1</sub></var>, <var>y<sub>1</sub></var>) and whose
      bottom right corner is given by the coordinate (<var>x<sub>2</sub></var>,
      <var>y<sub>2</sub></var>), those coordinates being interpreted as <span data-x="'px'">CSS
      pixels</span> from the top left corner of the image.</p>
     </dd>
    </dl>

    <p>For historical reasons, the coordinates must be interpreted relative to the
    <em>displayed</em> image after any stretching caused by the CSS <span>'width'</span> and
    <span>'height'</span> properties (or, for non-CSS browsers, the image element's <code
    data-x="">width</code> and <code data-x="">height</code> attributes &mdash; CSS browsers map
    those attributes to the aforementioned CSS properties).</p>

    <p class="note">Browser zoom features and transforms applied using CSS or SVG do not affect the
    coordinates.</p>
   </li>
  </ol>

  <p>Pointing device interaction with an image associated with a set of layered shapes per the above
  algorithm must result in the relevant user interaction events being first fired to the top-most
  shape covering the point that the pointing device indicated, if any, or to the image element
  itself, if there is no shape covering that point. User agents may also allow individual
  <code>area</code> elements representing <span data-x="hyperlink">hyperlinks</span> to be selected
  and activated (e.g. using a keyboard).</p>

  <p class="note">Because a <code>map</code> element (and its <code>area</code> elements) can be
  associated with multiple <code>img</code> elements, it is possible for an <code>area</code>
  element to correspond to multiple <i data-x="focusable area">focusable areas</i> of the
  document.</p>

  <p>Image maps are <span>live</span>; if the DOM is mutated, then the user agent must act as if it
  had rerun the algorithms for image maps.</p>

  </div>



  <h4 split-filename="embedded-content-other">MathML</h4>

  <p>The <span>MathML <code>math</code></span> element falls into the <span>embedded content</span>,
  <span>phrasing content</span>, <span>flow content</span>, and <span>palpable content</span>
  categories for the purposes of the content models in this specification.</p>

  <p>When the <span>MathML <code>annotation-xml</code></span> element contains elements from the
  <span>HTML namespace</span>, such elements must all be <span>flow content</span>.</p>

  <p>When the MathML token elements (<code data-x="MathML mi">mi</code>, <code data-x="MathML
  mo">mo</code>, <code data-x="MathML mn">mn</code>, <code data-x="MathML ms">ms</code>, and <code
  data-x="MathML mtext">mtext</code>) are descendants of HTML elements, they may contain
  <span>phrasing content</span> elements from the <span>HTML namespace</span>.</p>
  <!-- this hooks into https://w3c.github.io/mathml-core/#html-and-svg -->

  <div w-nodev>

  <!-- apparently we get to define error handling, so: -->

  <p>User agents must handle text other than <span>inter-element whitespace</span> found in MathML
  elements whose content models do not allow straight text by pretending for the purposes of MathML
  content models, layout, and rendering that the text is actually wrapped in a <span>MathML
  <code>mtext</code></span> element. (Such text is not, however, conforming.)</p>

  <p>User agents must act as if any MathML element whose contents does not match the element's
  content model was replaced, for the purposes of MathML layout and rendering, by a <span>MathML
  <code>merror</code></span> element containing some appropriate error message.</p>

  </div>

  <p>The semantics of MathML elements are defined by <cite>MathML</cite> and <span>other applicable
  specifications</span>. <ref>MATHML</ref></p>

  <div class="example">

   <p>Here is an example of the use of MathML in an HTML document:</p>

   <pre><code class="html">&lt;!DOCTYPE html>
&lt;html lang="en">
 &lt;head>
  &lt;title>The quadratic formula&lt;/title>
 &lt;/head>
 &lt;body>
  &lt;h1>The quadratic formula&lt;/h1>
  &lt;p>
   &lt;math>
    &lt;mi>x&lt;/mi>
    &lt;mo>=&lt;/mo>
    &lt;mfrac>
     &lt;mrow>
      &lt;mo form="prefix">&#x2212;&lt;/mo> &lt;mi>b&lt;/mi>
      &lt;mo>&#x00B1;&lt;/mo>
      &lt;msqrt>
       &lt;msup> &lt;mi>b&lt;/mi> &lt;mn>2&lt;/mn> &lt;/msup>
       &lt;mo>&#x2212;&lt;/mo>
       &lt;mn>4&lt;/mn> &lt;mo>&#x2062;&lt;/mo> &lt;mi>a&lt;/mi> &lt;mo>&#x2062;&lt;/mo> &lt;mi>c&lt;/mi>
      &lt;/msqrt>
     &lt;/mrow>
     &lt;mrow>
      &lt;mn>2&lt;/mn> &lt;mo>&#x2062;&lt;/mo> &lt;mi>a&lt;/mi>
     &lt;/mrow>
    &lt;/mfrac>
   &lt;/math>
  &lt;/p>
 &lt;/body>
&lt;/html></code></pre>

  </div>



  <h4 id="svg-0">SVG</h4>

  <p>The <span>SVG <code>svg</code></span> element falls into the <span>embedded content</span>,
  <span>phrasing content</span>, <span>flow content</span>, and <span>palpable content</span>
  categories for the purposes of the content models in this specification.</p>

  <p>When the <span>SVG <code>foreignObject</code></span> element contains elements from the
  <span>HTML namespace</span>, such elements must all be <span>flow content</span>.</p>

  <p>The content model for the <span>SVG <code>title</code></span> element inside <span>HTML
  documents</span> is <span>phrasing content</span>. (This further constrains the requirements given
  in <cite>SVG 2</cite>.)</p>

  <p>The semantics of SVG elements are defined by <cite>SVG 2</cite> and <span>other applicable
  specifications</span>. <ref>SVG</ref></p>

  <hr>

  <dl class="domintro">
   <dt><code data-x=""><var>doc</var> = <var>iframe</var>.<span subdfn data-x="dom-media-getSVGDocument">getSVGDocument</span>()</code></dt>
   <dt><code data-x=""><var>doc</var> = <var>embed</var>.<span data-x="dom-media-getSVGDocument">getSVGDocument</span>()</code></dt>
   <dt><code data-x=""><var>doc</var> = <var>object</var>.<span data-x="dom-media-getSVGDocument">getSVGDocument</span>()</code></dt>

   <dd><p>Returns the <code>Document</code> object, in the case of <code>iframe</code>,
   <code>embed</code>, or <code>object</code> elements being used to embed SVG.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="HTMLIFrameElement,HTMLEmbedElement,HTMLObjectElement"><code
  data-x="dom-media-getSVGDocument">getSVGDocument()</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>document</var> be <span>this</span>'s <span>content document</span>.</p></li>

   <li><p>If <var>document</var> is non-null and was created by the <span data-x="navigate-xml">page
   load processing model for XML files</span> section because the <span data-x="Content-Type
   sniffing">computed type of the resource</span> in the <span>navigate</span> algorithm was
   <code>image/svg+xml</code>, then return <var>document</var>.</p></li>
   <!-- Ideally this would check a property of the document. -->

   <li><p>Return null.</p></li>
  </ol>

  </div>



  <h4><dfn>Dimension attributes</dfn></h4>

  <p><span w-nodev><strong>Author requirements</strong>:</span> The <dfn element-attr
  for="img,iframe,embed,object,source,video"><code data-x="attr-dim-width">width</code></dfn> and
  <dfn element-attr for="img,iframe,embed,object,source,video"><code
  data-x="attr-dim-height">height</code></dfn> attributes on <code>img</code>, <code>iframe</code>,
  <code>embed</code>, <code>object</code>, <code>video</code>, <code>source</code> when the parent
  is a <code>picture</code> element and, when their <code data-x="attr-input-type">type</code>
  attribute is in the <span data-x="attr-input-type-image">Image Button</span> state,
  <code>input</code> elements may be specified to give the dimensions of the visual content of the
  element (the width and height respectively, relative to the nominal direction of the output
  medium), in <span data-x="'px'">CSS pixels</span>. The attributes, if specified, must have values
  that are <span data-x="valid non-negative integer">valid non-negative integers</span>.</p>

  <p>The specified dimensions given may differ from the dimensions specified in the resource itself,
  since the resource may have a resolution that differs from the CSS pixel resolution. (On screens,
  <span data-x="'px'">CSS pixels</span> have a resolution of 96ppi, but in general the CSS pixel
  resolution depends on the reading distance.) If both attributes are specified, then one of the
  following statements must be true:</p>

  <ul>
   <li><var>specified width</var> - 0.5 &le;
             <var>specified height</var> * <var>target ratio</var> &le;
             <var>specified width</var> + 0.5</li>

   <li><var>specified height</var> - 0.5 &le;
             <var>specified width</var> / <var>target ratio</var> &le;
             <var>specified height</var> + 0.5</li>

   <li><var>specified height</var> = <var>specified width</var> = 0</li>
  </ul>

  <p>The <var>target ratio</var> is the ratio of the <span>natural width</span> to the
  <span>natural height</span> in the resource. The <var>specified width</var> and <var>specified
  height</var> are the values of the <code data-x="attr-dim-width">width</code> and <code
  data-x="attr-dim-height">height</code> attributes respectively.</p>

  <p>The two attributes must be omitted if the resource in question does not have both a
  <span>natural width</span> and a <span>natural height</span>.</p>

  <p>If the two attributes are both 0, it indicates that the element is not intended for the user
  (e.g. it might be a part of a service to count page views).</p>

  <p class="note">The dimension attributes are not intended to be used to stretch the image.</p>

  <div w-nodev>

  <p><strong>User agent requirements</strong>: User agents are expected to use these attributes <a
  href="#dimRendering">as hints for the rendering</a>.</p>


  <p>The <dfn attribute
  for="HTMLIFrameElement,HTMLEmbedElement,HTMLObjectElement,HTMLSourceElement,HTMLVideoElement"><code
  data-x="dom-dim-width">width</code></dfn> and <dfn attribute
  for="HTMLIFrameElement,HTMLEmbedElement,HTMLObjectElement,HTMLSourceElement,HTMLVideoElement"><code
  data-x="dom-dim-height">height</code></dfn> IDL attributes on the <code>iframe</code>,
  <code>embed</code>, <code>object</code>, <code>source</code>, and <code>video</code> elements must
  <span>reflect</span> the respective content attributes of the same name.</p>

  <p class="note">For <code>iframe</code>, <code>embed</code> and <code>object</code> the IDL
  attributes are <code data-x="idl-DOMString">DOMString</code>; for <code>video</code> and
  <code>source</code> the IDL attributes are <code data-x="idl-unsigned-long">unsigned
  long</code>.</p>

  <p class="note">The corresponding IDL attributes for <code data-x="dom-img-height">img</code> and
  <code data-x="dom-input-height">input</code> elements are defined in those respective elements'
  sections, as they are slightly more specific to those elements' other behaviors.</p>

  </div>



  <h3 split-filename="tables" id="tables">Tabular data</h3>


  <h4>The <dfn element><code>table</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>In this order: optionally a <code>caption</code> element, followed by zero or more
   <code>colgroup</code> elements, followed optionally by a <code>thead</code> element, followed by
   either zero or more <code>tbody</code> elements or one or more <code>tr</code> elements, followed
   optionally by a <code>tfoot</code> element, optionally intermixed with one or more
   <span>script-supporting elements</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-table">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-table">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLTableElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute <span>HTMLTableCaptionElement</span>? <span data-x="dom-table-caption">caption</span>;
  <span>HTMLTableCaptionElement</span> <span data-x="dom-table-createCaption">createCaption</span>();
  [<span>CEReactions</span>] undefined <span data-x="dom-table-deleteCaption">deleteCaption</span>();

  [<span>CEReactions</span>] attribute <span>HTMLTableSectionElement</span>? <span data-x="dom-table-tHead">tHead</span>;
  <span>HTMLTableSectionElement</span> <span data-x="dom-table-createTHead">createTHead</span>();
  [<span>CEReactions</span>] undefined <span data-x="dom-table-deleteTHead">deleteTHead</span>();

  [<span>CEReactions</span>] attribute <span>HTMLTableSectionElement</span>? <span data-x="dom-table-tFoot">tFoot</span>;
  <span>HTMLTableSectionElement</span> <span data-x="dom-table-createTFoot">createTFoot</span>();
  [<span>CEReactions</span>] undefined <span data-x="dom-table-deleteTFoot">deleteTFoot</span>();

  [SameObject] readonly attribute <span>HTMLCollection</span> <span data-x="dom-table-tBodies">tBodies</span>;
  <span>HTMLTableSectionElement</span> <span data-x="dom-table-createTBody">createTBody</span>();

  [SameObject] readonly attribute <span>HTMLCollection</span> <span data-x="dom-table-rows">rows</span>;
  <span>HTMLTableRowElement</span> <span data-x="dom-table-insertRow">insertRow</span>(optional long index = -1);
  [<span>CEReactions</span>] undefined <span data-x="dom-table-deleteRow">deleteRow</span>(long index);

  // <a href="#HTMLTableElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLTableElement</code>.</dd>
  </dl>

  <p>The <code>table</code> element <span>represents</span> data with more than one dimension, in
  the form of a <span data-x="concept-table">table</span>.</p>

  <p><span w-nodev>The <code>table</code> element takes part in the <span>table
  model</span>.</span> Tables have rows, columns, and cells given by their descendants. The rows and
  columns form a grid; a table's cells must completely cover that grid without overlap.</p>

  <div w-nodev>

  <p class="note">Precise rules for determining whether this conformance requirement is met are
  described in the description of the <span>table model</span>.</p>

  </div>

  <p>Authors are encouraged to provide information describing how to interpret complex tables.
  Guidance on how to <a href="#table-descriptions-techniques">provide such information</a> is given
  below.</p>

  <p>Tables must not be used as layout aids. Historically, some web authors have misused tables in
  HTML as a way to control their page layout. This usage is non-conforming, because tools attempting
  to extract tabular data from such documents would obtain very confusing results. In particular,
  users of accessibility tools like screen readers are likely to find it very difficult to navigate
  pages with tables used for layout.</p>

  <p class="note">There are a variety of alternatives to using HTML tables for layout, such as CSS
  grid layout, CSS flexible box layout ("flexbox"), CSS multi-column layout, CSS positioning, and
  the CSS table model. <ref>CSS</ref></p>

  <div w-nodev>

  <hr>

  <p>Tables can be complicated to understand and navigate. To help users with this, user agents
  should clearly delineate cells in a table from each other, unless the user agent has classified
  the table as a (non-conforming) layout table.</p>

  </div>

  <p class="note">Authors <span w-nodev>and implementers</span> are encouraged to consider
  using some of the <a href="#table-layout-techniques">table design techniques</a> described below
  to make tables easier to navigate for users.</p>

  <div w-nodev>

  <p>User agents, especially those that do table analysis on arbitrary content, are encouraged to
  find heuristics to determine which tables actually contain data and which are merely being used
  for layout. This specification does not define a precise heuristic, but the following are
  suggested as possible indicators:</p>

  <table>
   <thead>
    <tr>
     <th>Feature
     <th>Indication
   <tbody>
    <tr>
     <td>The use of the <code data-x="attr-aria-role">role</code> attribute with the value <code
     data-x="attr-aria-role-presentation">presentation</code>
     <td>Probably a layout table
    <tr>
     <td>The use of the non-conforming <code data-x="attr-table-border">border</code> attribute with the non-conforming value 0
     <td>Probably a layout table
    <tr>
     <td>The use of the non-conforming <code data-x="attr-table-cellspacing">cellspacing</code> and
     <code data-x="attr-table-cellpadding">cellpadding</code> attributes with the value 0
     <td>Probably a layout table
   <tbody>
    <tr>
     <td>The use of <code>caption</code>, <code>thead</code>, or <code>th</code> elements
     <td>Probably a non-layout table
    <tr>
     <td>The use of the <code data-x="attr-tdth-headers">headers</code> and <code data-x="attr-th-scope">scope</code> attributes
     <td>Probably a non-layout table
    <tr>
     <td>The use of the non-conforming <code data-x="attr-table-border">border</code> attribute with a value other than 0
     <td>Probably a non-layout table
    <tr>
     <td>Explicit visible borders set using CSS
     <td>Probably a non-layout table
   <tbody>
    <tr>
     <td>The use of the <code data-x="attr-table-summary">summary</code> attribute
     <td>Not a good indicator (both layout and non-layout tables have historically been given this attribute)
  </table>

  <p class="note">It is quite possible that the above suggestions are wrong. Implementers are urged
  to provide feedback elaborating on their experiences with trying to create a layout table
  detection heuristic.</p>

  <p>If a <code>table</code> element has a (non-conforming) <code
  data-x="attr-table-summary">summary</code> attribute, and the user agent has not classified the
  table as a layout table, the user agent may report the contents of that attribute to the user.</p>

  </div>

  <hr>

  <dl class="domintro">
   <dt><code data-x=""><var>table</var>.<span subdfn data-x="dom-table-caption">caption</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the table's <code>caption</code> element.</p>

    <p>Can be set, to replace the <code>caption</code> element.</p>
   </dd>

   <dt><code data-x=""><var>caption</var> = <var>table</var>.<span subdfn data-x="dom-table-createCaption">createCaption</span>()</code></dt>
   <dd><p>Ensures the table has a <code>caption</code> element, and returns it.</p></dd>

   <dt><code data-x=""><var>table</var>.<span subdfn data-x="dom-table-deleteCaption">deleteCaption</span>()</code></dt>
   <dd><p>Ensures the table does not have a <code>caption</code> element.</p></dd>

   <dt><code data-x=""><var>table</var>.<span subdfn data-x="dom-table-tHead">tHead</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the table's <code>thead</code> element.</p>

    <p>Can be set, to replace the <code>thead</code> element. If the new value is not a
    <code>thead</code> element, throws a <span>"<code>HierarchyRequestError</code>"</span>
    <code>DOMException</code>.</p>
   </dd>

   <dt><code data-x=""><var>thead</var> = <var>table</var>.<span subdfn data-x="dom-table-createTHead">createTHead</span>()</code></dt>
   <dd><p>Ensures the table has a <code>thead</code> element, and returns it.</p></dd>

   <dt><code data-x=""><var>table</var>.<span subdfn data-x="dom-table-deleteTHead">deleteTHead</span>()</code></dt>
   <dd><p>Ensures the table does not have a <code>thead</code> element.</p></dd>

   <dt><code data-x=""><var>table</var>.<span subdfn data-x="dom-table-tFoot">tFoot</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the table's <code>tfoot</code> element.</p>

    <p>Can be set, to replace the <code>tfoot</code> element. If the new value is not a
    <code>tfoot</code> element, throws a <span>"<code>HierarchyRequestError</code>"</span>
    <code>DOMException</code>.</p>
   </dd>

   <dt><code data-x=""><var>tfoot</var> = <var>table</var>.<span subdfn data-x="dom-table-createTFoot">createTFoot</span>()</code></dt>
   <dd><p>Ensures the table has a <code>tfoot</code> element, and returns it.</p></dd>

   <dt><code data-x=""><var>table</var>.<span subdfn data-x="dom-table-deleteTFoot">deleteTFoot</span>()</code></dt>
   <dd><p>Ensures the table does not have a <code>tfoot</code> element.</p></dd>

   <dt><code data-x=""><var>table</var>.<span subdfn data-x="dom-table-tBodies">tBodies</span></code></dt>
   <dd><p>Returns an <code>HTMLCollection</code> of the <code>tbody</code> elements of the
   table.</p></dd>

   <dt><code data-x=""><var>tbody</var> = <var>table</var>.<span subdfn data-x="dom-table-createTBody">createTBody</span>()</code></dt>
   <dd><p>Creates a <code>tbody</code> element, inserts it into the table, and returns it.</p></dd>

   <dt><code data-x=""><var>table</var>.<span subdfn data-x="dom-table-rows">rows</span></code></dt>
   <dd><p>Returns an <code>HTMLCollection</code> of the <code>tr</code> elements of the
   table.</p></dd>

   <dt><code data-x=""><var>tr</var> = <var>table</var>.<span subdfn data-x="dom-table-insertRow">insertRow</span>([ <var>index</var> ])</code></dt>
   <dd>
    <p>Creates a <code>tr</code> element, along with a <code>tbody</code> if required, inserts them
    into the table at the position given by the argument, and returns the <code>tr</code>.</p>

    <p>The position is relative to the rows in the table. The index &#x2212;1, which is the default
    if the argument is omitted, is equivalent to inserting at the end of the table.</p>

    <p>If the given position is less than &#x2212;1 or greater than the number of rows, throws an
    <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p>
   </dd>

   <dt><code data-x=""><var>table</var>.<span subdfn data-x="dom-table-deleteRow">deleteRow</span>(<var>index</var>)</code></dt>
   <dd>
    <p>Removes the <code>tr</code> element with the given position in the table.</p>

    <p>The position is relative to the rows in the table. The index &#x2212;1 is equivalent to
    deleting the last row of the table.</p>

    <p>If the given position is less than &#x2212;1 or greater than the index of the last row, or if
    there are no rows, throws an <span>"<code>IndexSizeError</code>"</span>
    <code>DOMException</code>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>In all of the following attribute and method definitions, when an element is to be
  <dfn>table-created</dfn>, that means to <span>create an element</span> given the
  <code>table</code> element's <span>node document</span>, the given local name, and the <span>HTML
  namespace</span>.</p>

  <p>The <dfn attribute for="HTMLTableElement"><code data-x="dom-table-caption">caption</code></dfn>
  IDL attribute must return, on getting, the first <code>caption</code> element child of the
  <code>table</code> element, if any, or null otherwise. On setting, the first <code>caption</code>
  element child of the <code>table</code> element, if any, must be removed, and the new value, if
  not null, must be inserted as the first node of the <code>table</code> element.</p>

  <p>The <dfn method for="HTMLTableElement"><code
  data-x="dom-table-createCaption">createCaption()</code></dfn> method must return the first
  <code>caption</code> element child of the <code>table</code> element, if any; otherwise a new
  <code>caption</code> element must be <span>table-created</span>, inserted as the first node of the
  <code>table</code> element, and then returned.</p>

  <p>The <dfn method for="HTMLTableElement"><code
  data-x="dom-table-deleteCaption">deleteCaption()</code></dfn> method must remove the first
  <code>caption</code> element child of the <code>table</code> element, if any.</p>

  <p>The <dfn attribute for="HTMLTableElement"><code data-x="dom-table-tHead">tHead</code></dfn> IDL
  attribute must return, on getting, the first <code>thead</code> element child of the
  <code>table</code> element, if any, or null otherwise. On setting, if the new value is null or a
  <code>thead</code> element, the first <code>thead</code> element child of the <code>table</code>
  element, if any, must be removed, and the new value, if not null, must be inserted immediately
  before the first element in the <code>table</code> element that is neither a <code>caption</code>
  element nor a <code>colgroup</code> element, if any, or at the end of the table if there are no
  such elements.  If the new value is neither null nor a <code>thead</code> element, then a
  <span>"<code>HierarchyRequestError</code>"</span> <code>DOMException</code> must be thrown
  instead.</p>

  <p>The <dfn method for="HTMLTableElement"><code
  data-x="dom-table-createTHead">createTHead()</code></dfn> method must return the first
  <code>thead</code> element child of the <code>table</code> element, if any; otherwise a new
  <code>thead</code> element must be <span>table-created</span> and inserted immediately before the
  first element in the <code>table</code> element that is neither a <code>caption</code> element nor
  a <code>colgroup</code> element, if any, or at the end of the table if there are no such elements,
  and then that new element must be returned.</p>

  <p>The <dfn method for="HTMLTableElement"><code
  data-x="dom-table-deleteTHead">deleteTHead()</code></dfn> method must remove the first
  <code>thead</code> element child of the <code>table</code> element, if any.</p>

  <p>The <dfn attribute for="HTMLTableElement"><code data-x="dom-table-tFoot">tFoot</code></dfn> IDL
  attribute must return, on getting, the first <code>tfoot</code> element child of the
  <code>table</code> element, if any, or null otherwise. On setting, if the new value is null or a
  <code>tfoot</code> element, the first <code>tfoot</code> element child of the <code>table</code>
  element, if any, must be removed, and the new value, if not null, must be inserted at the end of
  the table. If the new value is neither null nor a <code>tfoot</code> element, then a
  <span>"<code>HierarchyRequestError</code>"</span> <code>DOMException</code> must be thrown
  instead.</p>

  <p>The <dfn method for="HTMLTableElement"><code
  data-x="dom-table-createTFoot">createTFoot()</code></dfn> method must return the first
  <code>tfoot</code> element child of the <code>table</code> element, if any; otherwise a new
  <code>tfoot</code> element must be <span>table-created</span> and inserted at the end of the
  table, and then that new element must be returned.</p>

  <p>The <dfn method for="HTMLTableElement"><code
  data-x="dom-table-deleteTFoot">deleteTFoot()</code></dfn> method must remove the first
  <code>tfoot</code> element child of the <code>table</code> element, if any.</p>

  <p>The <dfn attribute for="HTMLTableElement"><code data-x="dom-table-tBodies">tBodies</code></dfn>
  attribute must return an <code>HTMLCollection</code> rooted at the <code>table</code> node, whose
  filter matches only <code>tbody</code> elements that are children of the <code>table</code>
  element.</p>

  <p>The <dfn method for="HTMLTableElement"><code
  data-x="dom-table-createTBody">createTBody()</code></dfn> method must <span
  data-x="table-created">table-create</span> a new <code>tbody</code> element, insert it immediately
  after the last <code>tbody</code> element child in the <code>table</code> element, if any, or at
  the end of the <code>table</code> element if the <code>table</code> element has no
  <code>tbody</code> element children, and then must return the new <code>tbody</code> element.</p>

  <p>The <dfn attribute for="HTMLTableElement"><code data-x="dom-table-rows">rows</code></dfn>
  attribute must return an <code>HTMLCollection</code> rooted at the <code>table</code> node, whose
  filter matches only <code>tr</code> elements that are either children of the <code>table</code>
  element, or children of <code>thead</code>, <code>tbody</code>, or <code>tfoot</code> elements
  that are themselves children of the <code>table</code> element. The elements in the collection
  must be ordered such that those elements whose parent is a <code>thead</code> are included first,
  in <span>tree order</span>, followed by those elements whose parent is either a <code>table</code>
  or <code>tbody</code> element, again in <span>tree order</span>, followed finally by those
  elements whose parent is a <code>tfoot</code> element, still in <span>tree order</span>.</p>

  <p>The behavior of the <dfn method for="HTMLTableElement"><code
  data-x="dom-table-insertRow">insertRow(<var>index</var>)</code></dfn> method depends on the state
  of the table. When it is called, the method must act as required by the first item in the
  following list of conditions that describes the state of the table and the <var>index</var>
  argument:</p>

  <dl class="switch">

   <dt>If <var>index</var> is less than &#x2212;1 or greater than the number of elements
   in <code data-x="dom-table-rows">rows</code> collection:</dt>

   <dd>The method must throw an <span>"<code>IndexSizeError</code>"</span>
   <code>DOMException</code>.</dd>

   <dt>If the <code data-x="dom-table-rows">rows</code> collection has zero elements in it, and the
   <code>table</code> has no <code>tbody</code> elements in it:</dt>

   <dd>The method must <span data-x="table-created">table-create</span> a <code>tbody</code>
   element, then <span data-x="table-created">table-create</span> a <code>tr</code> element, then
   append the <code>tr</code> element to the <code>tbody</code> element, then append the
   <code>tbody</code> element to the <code>table</code> element, and finally return the
   <code>tr</code> element.</dd>

   <dt>If the <code data-x="dom-table-rows">rows</code> collection has zero elements in it:</dt>

   <dd>The method must <span data-x="table-created">table-create</span> a <code>tr</code> element,
   append it to the last <code>tbody</code> element in the table, and return the <code>tr</code>
   element.</dd>

   <dt>If <var>index</var> is &#x2212;1 or equal to the number of items in <code
   data-x="dom-table-rows">rows</code> collection:</dt>

   <dd>The method must <span data-x="table-created">table-create</span> a <code>tr</code> element,
   and append it to the parent of the last <code>tr</code> element in the <code
   data-x="dom-table-rows">rows</code> collection. Then, the newly created <code>tr</code> element
   must be returned.</dd>

   <dt>Otherwise:</dt>

   <dd>The method must <span data-x="table-created">table-create</span> a <code>tr</code> element,
   insert it immediately before the <var>index</var>th <code>tr</code> element in the <code
   data-x="dom-table-rows">rows</code> collection, in the same parent, and finally must return the
   newly created <code>tr</code> element.</dd>
  </dl>

  <p>When the <dfn method for="HTMLTableElement"><code
  data-x="dom-table-deleteRow">deleteRow(<var>index</var>)</code></dfn> method is called, the user
  agent must run the following steps:</p>

  <ol>
   <li><p>If <var>index</var> is less than &#x2212;1 or greater than or equal to the number of
   elements in the <code data-x="dom-table-rows">rows</code> collection, then throw an
   <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <var>index</var> is &#x2212;1, then <span data-x="concept-node-remove">remove</span>
   the last element in the <code data-x="dom-table-rows">rows</code> collection from its parent, or
   do nothing if the <code data-x="dom-table-rows">rows</code> collection is empty.</p></li>

   <li><p>Otherwise, <span data-x="concept-node-remove">remove</span> the <var>index</var>th element
   in the <code data-x="dom-table-rows">rows</code> collection from its parent.</p></li>
  </ol>

  </div>

  <div class="example">

   <p>Here is an example of a table being used to mark up a Sudoku puzzle. Observe the lack of
   headers, which are not necessary in such a table.</p>

   <pre><code class="html">&lt;style>
 #sudoku { border-collapse: collapse; border: solid thick; }
 #sudoku colgroup, table#sudoku tbody { border: solid medium; }
 #sudoku td { border: solid thin; height: 1.4em; width: 1.4em; text-align: center; padding: 0; }
&lt;/style>
&lt;h1>Today's Sudoku&lt;/h1>
&lt;table id="sudoku">
 &lt;colgroup>&lt;col>&lt;col>&lt;col>
 &lt;colgroup>&lt;col>&lt;col>&lt;col>
 &lt;colgroup>&lt;col>&lt;col>&lt;col>
 &lt;tbody>
  &lt;tr> &lt;td> 1 &lt;td>   &lt;td> 3 &lt;td> 6 &lt;td>   &lt;td> 4 &lt;td> 7 &lt;td>   &lt;td> 9
  &lt;tr> &lt;td>   &lt;td> 2 &lt;td>   &lt;td>   &lt;td> 9 &lt;td>   &lt;td>   &lt;td> 1 &lt;td>
  &lt;tr> &lt;td> 7 &lt;td>   &lt;td>   &lt;td>   &lt;td>   &lt;td>   &lt;td>   &lt;td>   &lt;td> 6
 &lt;tbody>
  &lt;tr> &lt;td> 2 &lt;td>   &lt;td> 4 &lt;td>   &lt;td> 3 &lt;td>   &lt;td> 9 &lt;td>   &lt;td> 8
  &lt;tr> &lt;td>   &lt;td>   &lt;td>   &lt;td>   &lt;td>   &lt;td>   &lt;td>   &lt;td>   &lt;td>
  &lt;tr> &lt;td> 5 &lt;td>   &lt;td>   &lt;td> 9 &lt;td>   &lt;td> 7 &lt;td>   &lt;td>   &lt;td> 1
 &lt;tbody>
  &lt;tr> &lt;td> 6 &lt;td>   &lt;td>   &lt;td>   &lt;td> 5 &lt;td>   &lt;td>   &lt;td>   &lt;td> 2
  &lt;tr> &lt;td>   &lt;td>   &lt;td>   &lt;td>   &lt;td> 7 &lt;td>   &lt;td>   &lt;td>   &lt;td>
  &lt;tr> &lt;td> 9 &lt;td>   &lt;td>   &lt;td> 8 &lt;td>   &lt;td> 2 &lt;td>   &lt;td>   &lt;td> 5
&lt;/table></code></pre>

  </div>




  <h5 id="table-descriptions-techniques">Techniques for describing tables</h5>

  <p id="table-descriptions">For tables that consist of more than just a grid of cells with headers
  in the first row and headers in the first column, and for any table in general where the reader
  might have difficulty understanding the content, authors should include explanatory information
  introducing the table. This information is useful for all users, but is especially useful for
  users who cannot see the table, e.g. users of screen readers.</p>

  <p>Such explanatory information should introduce the purpose of the table, outline its basic cell
  structure, highlight any trends or patterns, and generally teach the user how to use the
  table.</p>

  <!-- Describing the conclusions of the data in a table is useful to everyone; explaining how to
  read the table, if not obvious from the headers alone, is useful to everyone; describing the
  structure of the table, if it is easy to grasp visually, might not be useful to everyone, but it
  might also not be useful to users who can quickly navigate the table with an accessibility tool.
  -->

  <p>For instance, the following table:</p>

  <table>
   <caption>Characteristics with positive and negative sides</caption>
   <thead>
    <tr>
     <th id="n"> Negative
     <th> Characteristic
     <th> Positive
   <tbody>
    <tr>
     <td headers="n r1"> Sad
     <th id=r1> Mood
     <td> Happy
    <tr>
     <td headers="n r2"> Failing
     <th id=r2> Grade
     <td> Passing
  </table>

  <p>...might benefit from a description explaining the way the table is laid out, something like
  "Characteristics are given in the second column, with the negative side in the left column and the
  positive side in the right column".</p>

  <p>There are a variety of ways to include this information, such as:</p>

  <dl>
   <dt>In prose, surrounding the table</dt>

   <dd>
    <div class="example"><pre><code class="html"><strong>&lt;p>In the following table, characteristics are given in the second
column, with the negative side in the left column and the positive
side in the right column.&lt;/p></strong>
&lt;table>
 &lt;caption>Characteristics with positive and negative sides&lt;/caption>
 &lt;thead>
  &lt;tr>
   &lt;th id="n"> Negative
   &lt;th> Characteristic
   &lt;th> Positive
 &lt;tbody>
  &lt;tr>
   &lt;td headers="n r1"> Sad
   &lt;th id="r1"> Mood
   &lt;td> Happy
  &lt;tr>
   &lt;td headers="n r2"> Failing
   &lt;th id="r2"> Grade
   &lt;td> Passing
&lt;/table></code></pre></div>
   </dd>

   <dt>In the table's <code>caption</code></dt>

   <dd>
    <div class="example"><pre><code class="html">&lt;table>
<strong> &lt;caption>
  &lt;strong>Characteristics with positive and negative sides.&lt;/strong>
  &lt;p>Characteristics are given in the second column, with the
  negative side in the left column and the positive side in the right
  column.&lt;/p></strong>
 &lt;/caption>
 &lt;thead>
  &lt;tr>
   &lt;th id="n"> Negative
   &lt;th> Characteristic
   &lt;th> Positive
 &lt;tbody>
  &lt;tr>
   &lt;td headers="n r1"> Sad
   &lt;th id="r1"> Mood
   &lt;td> Happy
  &lt;tr>
   &lt;td headers="n r2"> Failing
   &lt;th id="r2"> Grade
   &lt;td> Passing
&lt;/table></code></pre></div>
   </dd>

   <dt>In the table's <code>caption</code>, in a <code>details</code> element</dt>

   <dd>
    <div class="example"><pre><code class="html">&lt;table>
 &lt;caption>
  &lt;strong>Characteristics with positive and negative sides.&lt;/strong>
<strong>  &lt;details>
   &lt;summary>Help&lt;/summary>
   &lt;p>Characteristics are given in the second column, with the
   negative side in the left column and the positive side in the right
   column.&lt;/p>
  &lt;/details></strong>
 &lt;/caption>
 &lt;thead>
  &lt;tr>
   &lt;th id="n"> Negative
   &lt;th> Characteristic
   &lt;th> Positive
 &lt;tbody>
  &lt;tr>
   &lt;td headers="n r1"> Sad
   &lt;th id="r1"> Mood
   &lt;td> Happy
  &lt;tr>
   &lt;td headers="n r2"> Failing
   &lt;th id="r2"> Grade
   &lt;td> Passing
&lt;/table></code></pre></div>
   </dd>

   <dt>Next to the table, in the same <code>figure</code></dt>

   <dd>
    <div class="example"><pre><code class="html">&lt;figure>
 &lt;figcaption>Characteristics with positive and negative sides&lt;/figcaption>
<strong> &lt;p>Characteristics are given in the second column, with the
 negative side in the left column and the positive side in the right
 column.&lt;/p></strong>
 &lt;table>
  &lt;thead>
   &lt;tr>
    &lt;th id="n"> Negative
    &lt;th> Characteristic
    &lt;th> Positive
  &lt;tbody>
   &lt;tr>
    &lt;td headers="n r1"> Sad
    &lt;th id="r1"> Mood
    &lt;td> Happy
   &lt;tr>
    &lt;td headers="n r2"> Failing
    &lt;th id="r2"> Grade
    &lt;td> Passing
 &lt;/table>
&lt;/figure></code></pre></div>
   </dd>

   <dt>Next to the table, in a <code>figure</code>'s <code>figcaption</code></dt>

   <dd>
    <div class="example"><pre><code class="html">&lt;figure>
 &lt;figcaption>
  &lt;strong>Characteristics with positive and negative sides&lt;/strong>
<strong>  &lt;p>Characteristics are given in the second column, with the
  negative side in the left column and the positive side in the right
  column.&lt;/p></strong>
 &lt;/figcaption>
 &lt;table>
  &lt;thead>
   &lt;tr>
    &lt;th id="n"> Negative
    &lt;th> Characteristic
    &lt;th> Positive
  &lt;tbody>
   &lt;tr>
    &lt;td headers="n r1"> Sad
    &lt;th id="r1"> Mood
    &lt;td> Happy
   &lt;tr>
    &lt;td headers="n r2"> Failing
    &lt;th id="r2"> Grade
    &lt;td> Passing
 &lt;/table>
&lt;/figure></code></pre></div>
   </dd>
  </dl>

  <p>Authors may also use other techniques, or combinations of the above techniques, as
  appropriate.</p>

  <p>The best option, of course, rather than writing a description explaining the way the table is
  laid out, is to adjust the table such that no explanation is needed.</p>

  <div class="example">

   <p>In the case of the table used in the examples above, a simple rearrangement of the table so
   that the headers are on the top and left sides removes the need for an explanation as well as
   removing the need for the use of <code data-x="attr-tdth-headers">headers</code> attributes:</p>

   <pre><code class="html">&lt;table>
 &lt;caption>Characteristics with positive and negative sides&lt;/caption>
 &lt;thead>
  &lt;tr>
   &lt;th> Characteristic
   &lt;th> Negative
   &lt;th> Positive
 &lt;tbody>
  &lt;tr>
   &lt;th> Mood
   &lt;td> Sad
   &lt;td> Happy
  &lt;tr>
   &lt;th> Grade
   &lt;td> Failing
   &lt;td> Passing
&lt;/table></code></pre>

  </div>


  <h5 id="table-layout-techniques">Techniques for table design</h5>

  <p>Good table design is key to making tables more readable and usable.</p>

  <p>In visual media, providing column and row borders and alternating row backgrounds can be very
  effective to make complicated tables more readable.</p>

  <p>For tables with large volumes of numeric content, using monospaced fonts can help users see
  patterns, especially in situations where a user agent does not render the borders. (Unfortunately,
  for historical reasons, not rendering borders on tables is a common default.)</p>

  <p>In speech media, table cells can be distinguished by reporting the corresponding headers before
  reading the cell's contents, and by allowing users to navigate the table in a grid fashion, rather
  than serializing the entire contents of the table in source order.</p>

  <p>Authors are encouraged to use CSS to achieve these effects.</p>

  <div w-nodev>

  <p>User agents are encouraged to render tables using these techniques whenever the page does not
  use CSS and the table is not classified as a layout table.</p>

  </div>




  <h4>The <dfn element><code>caption</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As the first element child of a <code>table</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>, but with no descendant <code>table</code> elements.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-caption">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-caption">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLTableCaptionElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  // <a href="#HTMLTableCaptionElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLTableCaptionElement</code>.</dd>
  </dl>

  <p>The <code>caption</code> element <span>represents</span> the title of the <code>table</code>
  that is its parent, if it has a parent and that is a <code>table</code> element.</p>

  <div w-nodev>

  <p>The <code>caption</code> element takes part in the <span>table model</span>.</p>

  </div>

  <p>When a <code>table</code> element is the only content in a <code>figure</code> element other
  than the <code>figcaption</code>, the <code>caption</code> element should be omitted in favor of
  the <code>figcaption</code>.</p>

  <p>A caption can introduce context for a table, making it significantly easier to understand.</p>

  <div class="example">

   <p>Consider, for instance, the following table:</p>

   <table class="dice-example">
    <tr> <th>   <th> 1 <th> 2 <th> 3 <th> 4 <th> 5 <th> 6
    <tr> <th> 1 <td> 2 <td> 3 <td> 4 <td> 5 <td> 6 <td> 7
    <tr> <th> 2 <td> 3 <td> 4 <td> 5 <td> 6 <td> 7 <td> 8
    <tr> <th> 3 <td> 4 <td> 5 <td> 6 <td> 7 <td> 8 <td> 9
    <tr> <th> 4 <td> 5 <td> 6 <td> 7 <td> 8 <td> 9 <td> 10
    <tr> <th> 5 <td> 6 <td> 7 <td> 8 <td> 9 <td> 10 <td> 11
    <tr> <th> 6 <td> 7 <td> 8 <td> 9 <td> 10 <td> 11 <td> 12
   </table>

   <p>In the abstract, this table is not clear. However, with a caption giving the table's number
   (for <span data-x="referenced">reference</span> in the main prose) and explaining its use, it
   makes more sense:</p>

   <pre><code class="html">&lt;caption>
&lt;p>Table 1.
&lt;p>This table shows the total score obtained from rolling two
six-sided dice. The first row represents the value of the first die,
the first column the value of the second die. The total is given in
the cell that corresponds to the values of the two dice.
&lt;/caption></code></pre>

   <p>This provides the user with more context:</p>

   <table class="dice-example">
    <caption>
     <p>Table 1.
     <p>This table shows the total score obtained from rolling two
     six-sided dice. The first row represents the value of the first
     die, the first column the value of the second die. The total is
     given in the cell that corresponds to the values of the two dice.
    </caption>
    <tr> <th>   <th> 1 <th> 2 <th> 3 <th> 4 <th> 5 <th> 6
    <tr> <th> 1 <td> 2 <td> 3 <td> 4 <td> 5 <td> 6 <td> 7
    <tr> <th> 2 <td> 3 <td> 4 <td> 5 <td> 6 <td> 7 <td> 8
    <tr> <th> 3 <td> 4 <td> 5 <td> 6 <td> 7 <td> 8 <td> 9
    <tr> <th> 4 <td> 5 <td> 6 <td> 7 <td> 8 <td> 9 <td> 10
    <tr> <th> 5 <td> 6 <td> 7 <td> 8 <td> 9 <td> 10 <td> 11
    <tr> <th> 6 <td> 7 <td> 8 <td> 9 <td> 10 <td> 11 <td> 12
   </table>

  </div>




  <h4>The <dfn element><code>colgroup</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As a child of a <code>table</code> element, after any
   <code>caption</code> elements and before any <code>thead</code>,
   <code>tbody</code>, <code>tfoot</code>, and <code>tr</code>
   elements.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>If the <code data-x="attr-colgroup-span">span</code> attribute is present: <span data-x="concept-content-nothing">Nothing</span>.</dd>
   <dd>If the <code data-x="attr-colgroup-span">span</code> attribute is absent: Zero or more <code>col</code> and <code>template</code> elements.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-colgroup-span">span</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-colgroup">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-colgroup">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLTableColElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-colgroup-span">span</span>;

  // <a href="#HTMLTableColElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLTableColElement</code>.</dd>
  </dl>

  <p>The <code>colgroup</code> element <span>represents</span> a <span
  data-x="concept-column-group">group</span> of one or more <span
  data-x="concept-column">columns</span> in the <code>table</code> that is its parent, if it has a
  parent and that is a <code>table</code> element.</p>

  <p>If the <code>colgroup</code> element contains no <code>col</code> elements, then the element
  may have a <dfn element-attr for="colgroup"><code data-x="attr-colgroup-span">span</code></dfn>
  content attribute specified, whose value must be a <span>valid non-negative integer</span> greater
  than zero and less than or equal to 1000.</p>

  <div w-nodev>

  <p>The <code>colgroup</code> element and its <code data-x="attr-colgroup-span">span</code>
  attribute take part in the <span>table model</span>.</p>

  <p>The <dfn attribute for="HTMLTableColElement"><code data-x="dom-colgroup-span">span</code></dfn>
  IDL attribute must <span>reflect</span> the content attribute of the same name. It is
  <span>clamped to the range</span> [1, 1000], and its <span>default value</span> is 1.</p>

  </div>


  <h4>The <dfn element><code>col</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As a child of a <code>colgroup</code> element that doesn't have
   a <code data-x="attr-col-span">span</code> attribute.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="concept-content-nothing">Nothing</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-col-span">span</code></dd>
   <!-- v2: char, to specify the decimal character used in numeric data cells in the column (not header cells) -->
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-col">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-col">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLTableColElement</code>, as defined for <code>colgroup</code> elements.</dd>
  </dl>

  <p>If a <code>col</code> element has a parent and that is a <code>colgroup</code> element that
  itself has a parent that is a <code>table</code> element, then the <code>col</code> element
  <span>represents</span> one or more <span data-x="concept-column">columns</span> in the <span
  data-x="concept-column-group">column group</span> represented by that <code>colgroup</code>.</p>

  <p>The element may have a <dfn element-attr for="col"><code
  data-x="attr-col-span">span</code></dfn> content attribute specified, whose value must be a
  <span>valid non-negative integer</span> greater than zero and less than or equal to 1000.</p>

  <div w-nodev>

  <p>The <code>col</code> element and its <code data-x="attr-col-span">span</code> attribute take
  part in the <span>table model</span>.</p>

  </div>


  <h4>The <dfn element><code>tbody</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As a child of a <code>table</code> element, after any
   <code>caption</code>, <code>colgroup</code>, and
   <code>thead</code> elements, but only if there are no
   <code>tr</code> elements that are children of the
   <code>table</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>Zero or more <code>tr</code> and <span data-x="script-supporting elements">script-supporting</span> elements.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-tbody">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-tbody">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLTableSectionElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [SameObject] readonly attribute <span>HTMLCollection</span> <span data-x="dom-tbody-rows">rows</span>;
  <span>HTMLTableRowElement</span> <span data-x="dom-tbody-insertRow">insertRow</span>(optional long index = -1);
  [<span>CEReactions</span>] undefined <span data-x="dom-tbody-deleteRow">deleteRow</span>(long index);

  // <a href="#HTMLTableSectionElement-partial">also has obsolete members</a>
};</code></pre>
    <p><span w-dev>Uses <code>HTMLTableSectionElement</code>. </span>The
    <code>HTMLTableSectionElement</code> interface is also used for <code>thead</code> and
    <code>tfoot</code> elements.</p>
   </dd>
  </dl>

  <p>The <code>tbody</code> element <span>represents</span> a <span
  data-x="concept-row-group">block</span> of <span data-x="concept-row">rows</span> that consist of a
  body of data for the parent <code>table</code> element, if the <code>tbody</code> element has a
  parent and it is a <code>table</code>.</p>

  <div w-nodev>

  <p>The <code>tbody</code> element takes part in the <span>table model</span>.</p>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>tbody</var>.<span subdfn data-x="dom-tbody-rows">rows</span></code></dt>
   <dd>
    <p>Returns an <code>HTMLCollection</code> of the <code>tr</code> elements of the table
    section.</p>
   </dd>

   <dt><code data-x=""><var>tr</var> = <var>tbody</var>.<span subdfn data-x="dom-tbody-insertRow">insertRow</span>([ <var>index</var> ])</code></dt>
   <dd>
    <p>Creates a <code>tr</code> element, inserts it into the table section at the position given by
    the argument, and returns the <code>tr</code>.</p>

    <p>The position is relative to the rows in the table section. The index &#x2212;1, which is the
    default if the argument is omitted, is equivalent to inserting at the end of the table
    section.</p>

    <p>If the given position is less than &#x2212;1 or greater than the number of rows, throws an
    <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p>
   </dd>

   <dt><code data-x=""><var>tbody</var>.<span subdfn data-x="dom-tbody-deleteRow">deleteRow</span>(<var>index</var>)</code></dt>
   <dd>
    <p>Removes the <code>tr</code> element with the given position in the table section.</p>

    <p>The position is relative to the rows in the table section. The index &#x2212;1 is equivalent
    to deleting the last row of the table section.</p>

    <p>If the given position is less than &#x2212;1 or greater than the index of the last row, or if
    there are no rows, throws an <span>"<code>IndexSizeError</code>"</span>
    <code>DOMException</code>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLTableSectionElement"><code
  data-x="dom-tbody-rows">rows</code></dfn> attribute must return an <code>HTMLCollection</code>
  rooted at this element, whose filter matches only <code>tr</code> elements that are children of
  this element.</p>

  <p>The <dfn method for="HTMLTableSectionElement"><code
  data-x="dom-tbody-insertRow">insertRow(<var>index</var>)</code></dfn> method must act as
  follows:</p>

  <ol>
   <li><p>If <var>index</var> is less than &#x2212;1 or greater than the number of elements in the
   <code data-x="dom-tbody-rows">rows</code> collection, throw an
   <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>table row</var> be the result of <span data-x="create an element">creating an
   element</span> given this element's <span>node document</span>, "<code data-x="">tr</code>", and
   the <span>HTML namespace</span>.</p></li>

   <li><p>If <var>index</var> is &#x2212;1 or equal to the number of items in the <code
   data-x="dom-tbody-rows">rows</code> collection, then <span
   data-x="concept-node-append">append</span> <var>table row</var> to this element.</p></li>

   <li><p>Otherwise, <span data-x="concept-node-insert">insert</span> <var>table row</var> as a
   child of this element, immediately before the <var>index</var>th <code>tr</code> element in the
   <code data-x="dom-tbody-rows">rows</code> collection.</p></li>

   <li><p>Return <var>table row</var>.</p></li>
  </ol>

  <p>The <dfn method for="HTMLTableSectionElement"><code
  data-x="dom-tbody-deleteRow">deleteRow(<var>index</var>)</code></dfn> method must, when invoked,
  act as follows:

  <ol>
   <li><p>If <var>index</var> is less than &#x2212;1 or greater than or equal to the number of
   elements in the <code data-x="dom-tbody-rows">rows</code> collection, then throw an
   <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <var>index</var> is &#x2212;1, then <span data-x="concept-node-remove">remove</span>
   the last element in the <code data-x="dom-tbody-rows">rows</code> collection from this
   element, or do nothing if the <code data-x="dom-tbody-rows">rows</code> collection is
   empty.</p></li>

   <li><p>Otherwise, <span data-x="concept-node-remove">remove</span> the <var>index</var>th element
   in the <code data-x="dom-tbody-rows">rows</code> collection from this element.</p></li>
  </ol>

  </div>


  <h4>The <dfn element><code>thead</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As a child of a <code>table</code> element, after any
   <code>caption</code>, and <code>colgroup</code>
   elements and before any <code>tbody</code>, <code>tfoot</code>, and
   <code>tr</code> elements, but only if there are no other
   <code>thead</code> elements that are children of the
   <code>table</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>Zero or more <code>tr</code> and <span data-x="script-supporting elements">script-supporting</span> elements.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-thead">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-thead">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLTableSectionElement</code>, as defined for <code>tbody</code> elements.</dd>
  </dl>

  <p>The <code>thead</code> element <span>represents</span> the <span
  data-x="concept-row-group">block</span> of <span data-x="concept-row">rows</span> that consist of
  the column labels (headers) and any ancillary non-header cells for the parent <code>table</code>
  element, if the <code>thead</code> element has a parent and it is a <code>table</code>.</p>

  <div w-nodev>

  <p>The <code>thead</code> element takes part in the <span>table model</span>.</p>

  </div>

  <div class="example">

   <p>This example shows a <code>thead</code> element being used. Notice the use of both
   <code>th</code> and <code>td</code> elements in the <code>thead</code> element: the first row is
   the headers, and the second row is an explanation of how to fill in the table.</p>

   <pre><code class="html">&lt;table>
 &lt;caption> School auction sign-up sheet &lt;/caption>
<strong> &lt;thead>
  &lt;tr>
   &lt;th>&lt;label for=e1>Name&lt;/label>
   &lt;th>&lt;label for=e2>Product&lt;/label>
   &lt;th>&lt;label for=e3>Picture&lt;/label>
   &lt;th>&lt;label for=e4>Price&lt;/label>
  &lt;tr>
   &lt;td>Your name here
   &lt;td>What are you selling?
   &lt;td>Link to a picture
   &lt;td>Your reserve price
</strong> &lt;tbody>
  &lt;tr>
   &lt;td>Ms Danus
   &lt;td>Doughnuts
   &lt;td>&lt;img src="https://example.com/mydoughnuts.png" title="Doughnuts from Ms Danus">
   &lt;td>$45
  &lt;tr>
   &lt;td>&lt;input id=e1 type=text name=who required form=f>
   &lt;td>&lt;input id=e2 type=text name=what required form=f>
   &lt;td>&lt;input id=e3 type=url name=pic form=f>
   &lt;td>&lt;input id=e4 type=number step=0.01 min=0 value=0 required form=f>
&lt;/table>
&lt;form id=f action="/auction.cgi">
 &lt;input type=button name=add value="Submit">
&lt;/form></code></pre>

  </div>


  <h4>The <dfn element><code>tfoot</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As a child of a <code>table</code> element, after any
   <code>caption</code>, <code>colgroup</code>, <code>thead</code>,
   <code>tbody</code>, and <code>tr</code> elements, but only if there
   are no other <code>tfoot</code> elements that are children of the
   <code>table</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>Zero or more <code>tr</code> and <span data-x="script-supporting elements">script-supporting</span> elements.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-tfoot">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-tfoot">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLTableSectionElement</code>, as defined for <code>tbody</code> elements.</dd>
  </dl>

  <p>The <code>tfoot</code> element <span>represents</span> the <span
  data-x="concept-row-group">block</span> of <span data-x="concept-row">rows</span> that consist of
  the column summaries (footers) for the parent <code>table</code> element, if the
  <code>tfoot</code> element has a parent and it is a <code>table</code>.</p>

  <div w-nodev>

  <p>The <code>tfoot</code> element takes part in the <span>table
  model</span>.</p>

  </div>


  <h4>The <dfn element><code>tr</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As a child of a <code>thead</code> element.</dd>
   <dd>As a child of a <code>tbody</code> element.</dd>
   <dd>As a child of a <code>tfoot</code> element.</dd>
   <dd>As a child of a <code>table</code> element, after any
   <code>caption</code>, <code>colgroup</code>, and <code>thead</code>
   elements, but only if there are no <code>tbody</code> elements that
   are children of the <code>table</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>Zero or more <code>td</code>, <code>th</code>, and <span data-x="script-supporting elements">script-supporting</span> elements.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-tr">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-tr">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLTableRowElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  readonly attribute long <span data-x="dom-tr-rowIndex">rowIndex</span>;
  readonly attribute long <span data-x="dom-tr-sectionRowIndex">sectionRowIndex</span>;
  [SameObject] readonly attribute <span>HTMLCollection</span> <span data-x="dom-tr-cells">cells</span>;
  <span>HTMLTableCellElement</span> <span data-x="dom-tr-insertCell">insertCell</span>(optional long index = -1);
  [<span>CEReactions</span>] undefined <span data-x="dom-tr-deleteCell">deleteCell</span>(long index);

  // <a href="#HTMLTableRowElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLTableRowElement</code>.</dd>
  </dl>

  <p>The <code>tr</code> element <span>represents</span> a <span data-x="concept-row">row</span> of
  <span data-x="concept-cell">cells</span> in a <span data-x="concept-table">table</span>.</p>

  <div w-nodev>

  <p>The <code>tr</code> element takes part in the <span>table model</span>.</p>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>tr</var>.<span subdfn data-x="dom-tr-rowIndex">rowIndex</span></code></dt>

   <dd>
    <p>Returns the position of the row in the table's <code data-x="dom-table-rows">rows</code>
    list.</p>

    <p>Returns &#x2212;1 if the element isn't in a table.</p>
   </dd>

   <dt><code data-x=""><var>tr</var>.<span subdfn data-x="dom-tr-sectionRowIndex">sectionRowIndex</span></code></dt>

   <dd>
    <p>Returns the position of the row in the table section's <code
    data-x="dom-tbody-rows">rows</code> list.</p>

    <p>Returns &#x2212;1 if the element isn't in a table section.</p>
   </dd>

   <dt><code data-x=""><var>tr</var>.<span subdfn data-x="dom-tr-cells">cells</span></code></dt>
   <dd>
    <p>Returns an <code>HTMLCollection</code> of the <code>td</code> and <code>th</code> elements of
    the row.</p>
   </dd>

   <dt><code data-x=""><var>cell</var> = <var>tr</var>.<span subdfn data-x="dom-tr-insertCell">insertCell</span>([ <var>index</var> ])</code></dt>

   <dd>
    <p>Creates a <code>td</code> element, inserts it into the table row at the position given by the
    argument, and returns the <code>td</code>.</p>

    <p>The position is relative to the cells in the row. The index &#x2212;1, which is the default
    if the argument is omitted, is equivalent to inserting at the end of the row.</p>

    <p>If the given position is less than &#x2212;1 or greater than the number of cells, throws an
    <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p>
   </dd>

   <dt><code data-x=""><var>tr</var>.<span subdfn data-x="dom-tr-deleteCell">deleteCell</span>(<var>index</var>)</code></dt>
   <dd>
    <p>Removes the <code>td</code> or <code>th</code> element with the given position in the
    row.</p>

    <p>The position is relative to the cells in the row. The index &#x2212;1 is equivalent to
    deleting the last cell of the row.</p>

    <p>If the given position is less than &#x2212;1 or greater than the index of the last cell, or
    if there are no cells, throws an <span>"<code>IndexSizeError</code>"</span>
    <code>DOMException</code>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLTableRowElement"><code
  data-x="dom-tr-rowIndex">rowIndex</code></dfn> attribute must, if this element has a parent
  <code>table</code> element, or a parent <code>tbody</code>, <code>thead</code>, or
  <code>tfoot</code> element and a <em>grandparent</em> <code>table</code> element, return the index
  of this <code>tr</code> element in that <code>table</code> element's <code
  data-x="dom-table-rows">rows</code> collection. If there is no such <code>table</code> element,
  then the attribute must return &#x2212;1.</p>

  <p>The <dfn attribute for="HTMLTableRowElement"><code
  data-x="dom-tr-sectionRowIndex">sectionRowIndex</code></dfn> attribute must, if this element has a
  parent <code>table</code>, <code>tbody</code>, <code>thead</code>, or <code>tfoot</code> element,
  return the index of the <code>tr</code> element in the parent element's <code
  data-x="">rows</code> collection (for tables, that's <code>HTMLTableElement</code>'s <code
  data-x="dom-table-rows">rows</code> collection; for table sections, that's
  <code>HTMLTableSectionElement</code>'s <code data-x="dom-tbody-rows">rows</code>
  collection). If there is no such parent element, then the attribute must return &#x2212;1.</p>

  <p>The <dfn attribute for="HTMLTableRowElement"><code data-x="dom-tr-cells">cells</code></dfn>
  attribute must return an <code>HTMLCollection</code> rooted at this <code>tr</code> element, whose
  filter matches only <code>td</code> and <code>th</code> elements that are children of the
  <code>tr</code> element.</p>

  <p>The <dfn method for="HTMLTableRowElement"><code
  data-x="dom-tr-insertCell">insertCell(<var>index</var>)</code></dfn> method must act as
  follows:</p>

  <ol>
   <li><p>If <var>index</var> is less than &#x2212;1 or greater than the number of elements in
   the <code data-x="dom-tr-cells">cells</code> collection, then throw an
   <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>table cell</var> be the result of <span data-x="create an element">creating an
   element</span> given this <code>tr</code> element's <span>node document</span>, "<code
   data-x="">td</code>", and the <span>HTML namespace</span>.</p></li>

   <li><p>If <var>index</var> is equal to &#x2212;1 or equal to the number of items in <code
   data-x="dom-tr-cells">cells</code> collection, then <span
   data-x="concept-node-append">append</span> <var>table cell</var> to this <code>tr</code>
   element.</p></li>

   <li><p>Otherwise, <span data-x="concept-node-insert">insert</span> <var>table cell</var> as a
   child of this <code>tr</code> element, immediately before the <var>index</var>th <code>td</code>
   or <code>th</code> element in the <code data-x="dom-tr-cells">cells</code> collection.</p></li>

   <li><p>Return <var>table cell</var>.</p></li>
  </ol>

  <p>The <dfn method for="HTMLTableRowElement"><code
  data-x="dom-tr-deleteCell">deleteCell(<var>index</var>)</code></dfn> method must act as
  follows:</p>

  <ol>
   <li><p>If <var>index</var> is less than &#x2212;1 or greater than or equal to the number of
   elements in the <code data-x="dom-tr-cells">cells</code> collection, then throw an
   <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <var>index</var> is &#x2212;1, then <span data-x="concept-node-remove">remove</span>
   the last element in the <code data-x="dom-tr-cells">cells</code> collection from its
   parent, or do nothing if the <code data-x="dom-tr-cells">cells</code> collection is
   empty.</p></li>

   <li><p>Otherwise, <span data-x="concept-node-remove">remove</span> the <var>index</var>th element
   in the <code data-x="dom-tr-cells">cells</code> collection from its parent.</p></li>
  </ol>

  </div>


  <h4>The <dfn element><code>td</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As a child of a <code>tr</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-tdth-colspan">colspan</code></dd>
   <dd><code data-x="attr-tdth-rowspan">rowspan</code></dd>
   <dd><code data-x="attr-tdth-headers">headers</code></dd>
   <!-- v2 char, to specify the decimal character used in numeric cells -->
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-td">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-td">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLTableCellElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-tdth-colSpan">colSpan</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-tdth-rowSpan">rowSpan</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-tdth-headers">headers</span>;
  readonly attribute long <span data-x="dom-tdth-cellIndex">cellIndex</span>;

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-th-scope">scope</span>; // only conforming for th elements
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-th-abbr">abbr</span>;  // only conforming for th elements

  // <a href="#HTMLTableCellElement-partial">also has obsolete members</a>
};</code></pre>

    <p><span w-dev>Uses <code>HTMLTableCellElement</code>. </span>The
    <code>HTMLTableCellElement</code> interface is also used for <code>th</code> elements.</p>
   </dd>
  </dl>

  <p>The <code>td</code> element <span>represents</span> a data <span
  data-x="concept-cell">cell</span> in a table.</p>

  <div w-nodev>

  <p>The <code>td</code> element and its <code data-x="attr-tdth-colspan">colspan</code>, <code
  data-x="attr-tdth-rowspan">rowspan</code>, and <code data-x="attr-tdth-headers">headers</code>
  attributes take part in the <span>table model</span>.</p>

  <p>User agents, especially in non-visual environments or where displaying the table as a 2D grid
  is impractical, may give the user context for the cell when rendering the contents of a cell; for
  instance, giving its position in the <span>table model</span>, or listing the cell's header cells
  (as determined by the <span>algorithm for assigning header cells</span>). When a cell's header
  cells are being listed, user agents may use the value of <code data-x="attr-th-abbr">abbr</code>
  attributes on those header cells, if any, instead of the contents of the header cells
  themselves.</p>

  </div>

  <div class="example">

   <p>In this example, we see a snippet of a web application consisting of a grid of editable cells
   (essentially a simple spreadsheet). One of the cells has been configured to show the sum of the
   cells above it. Three have been marked as headings, which use <code>th</code> elements instead of
   <code>td</code> elements. A script would attach event handlers to these elements to maintain the
   total.</p>

   <pre><code class="html">&lt;table>
 &lt;tr>
  &lt;th>&lt;input value="Name">
  &lt;th>&lt;input value="Paid ($)">
 &lt;tr>
  &lt;td>&lt;input value="Jeff">
  &lt;td>&lt;input value="14">
 &lt;tr>
  &lt;td>&lt;input value="Britta">
  &lt;td>&lt;input value="9">
 &lt;tr>
  &lt;td>&lt;input value="Abed">
  &lt;td>&lt;input value="25">
 &lt;tr>
  &lt;td>&lt;input value="Shirley">
  &lt;td>&lt;input value="2">
 &lt;tr>
  &lt;td>&lt;input value="Annie">
  &lt;td>&lt;input value="5">
 &lt;tr>
  &lt;td>&lt;input value="Troy">
  &lt;td>&lt;input value="5">
 &lt;tr>
  &lt;td>&lt;input value="Pierce">
  &lt;td>&lt;input value="1000">
 &lt;tr>
  &lt;th>&lt;input value="Total">
  &lt;td>&lt;output value="1060">
&lt;/table></code></pre>

  </div>


  <h4>The <dfn element><code>th</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As a child of a <code>tr</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>, but with no <code>header</code>, <code>footer</code>,
   <span>sectioning content</span>, or <span>heading content</span> descendants.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-tdth-colspan">colspan</code></dd>
   <dd><code data-x="attr-tdth-rowspan">rowspan</code></dd>
   <dd><code data-x="attr-tdth-headers">headers</code></dd>
   <dd><code data-x="attr-th-scope">scope</code></dd>
   <dd><code data-x="attr-th-abbr">abbr</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-th">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-th">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLTableCellElement</code>, as defined for <code>td</code> elements.</dd>
  </dl>

  <p>The <code>th</code> element <span>represents</span> a header <span
  data-x="concept-cell">cell</span> in a table.</p>

  <p>The <code>th</code> element may have a <dfn element-attr for="th"><code
  data-x="attr-th-scope">scope</code></dfn> content attribute specified.</p>

  <p>The <code data-x="attr-th-scope">scope</code> attribute is an <span>enumerated attribute</span>
  with the following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   </thead>
   <tbody>
    <tr>
     <td><dfn attr-value for="th/scope"><code data-x="attr-th-scope-row">row</code></dfn>
     <td><dfn data-x="attr-th-scope-row-state">row</dfn>
     <td>The header cell applies to some of the subsequent cells in the same row(s).
    <tr>
     <td><dfn attr-value for="th/scope"><code data-x="attr-th-scope-col">col</code></dfn>
     <td><dfn data-x="attr-th-scope-column-state">column</dfn>
     <td>The header cell applies to some of the subsequent cells in the same column(s).
    <tr>
     <td><dfn attr-value for="th/scope"><code data-x="attr-th-scope-rowgroup">rowgroup</code></dfn>
     <td><dfn data-x="attr-th-scope-rowgroup-state">row group</dfn>
     <td>The header cell applies to all the remaining cells in the row group.
    <tr>
     <td><dfn attr-value for="th/scope"><code data-x="attr-th-scope-colgroup">colgroup</code></dfn>
     <td><dfn data-x="attr-th-scope-colgroup-state">column group</dfn>
     <td>The header cell applies to all the remaining cells in the column group.
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <dfn
  data-x="attr-th-scope-auto-state">auto</dfn> state. (In this state the header cell applies to a
  set of cells selected based on context.)</p>

  <p>A <code>th</code> element's <code data-x="attr-th-scope">scope</code> attribute must not be in
  the <span data-x="attr-th-scope-rowgroup-state">row group</span> state if the element is not anchored in
  a <span data-x="concept-row-group">row group</span>, nor in the <span
  data-x="attr-th-scope-colgroup-state">column group</span> state if the element is not anchored in a
  <span data-x="concept-column-group">column group</span>.

  <p>The <code>th</code> element may have an <dfn element-attr for="th"><code
  data-x="attr-th-abbr">abbr</code></dfn> content attribute specified. Its value must be an
  alternative label for the header cell, to be used when referencing the cell in other contexts
  (e.g. when describing the header cells that apply to a data cell). It is typically an abbreviated
  form of the full header cell, but can also be an expansion, or merely a different phrasing.</p>

  <div w-nodev>

  <p>The <code>th</code> element and its <code data-x="attr-tdth-colspan">colspan</code>, <code
  data-x="attr-tdth-rowspan">rowspan</code>, <code data-x="attr-tdth-headers">headers</code>, and
  <code data-x="attr-th-scope">scope</code> attributes take part in the <span>table model</span>.</p>

  </div>

  <div class="example">

   <p>The following example shows how the <code data-x="attr-th-scope">scope</code> attribute's <code
   data-x="attr-th-scope-rowgroup">rowgroup</code> value affects which data cells a header cell
   applies to.</p>

   <p>Here is a markup fragment showing a table:</p>

   <pre><code class="html">&lt;table>
 &lt;thead>
  &lt;tr> &lt;th> ID &lt;th> Measurement &lt;th> Average &lt;th> Maximum
 &lt;tbody>
  &lt;tr> &lt;td> &lt;th scope=rowgroup> Cats &lt;td> &lt;td>
  &lt;tr> &lt;td> 93 &lt;th> Legs &lt;td> 3.5 &lt;td> 4
  &lt;tr> &lt;td> 10 &lt;th> Tails &lt;td> 1 &lt;td> 1
 &lt;tbody>
  &lt;tr> &lt;td> &lt;th scope=rowgroup> English speakers &lt;td> &lt;td>
  &lt;tr> &lt;td> 32 &lt;th> Legs &lt;td> 2.67 &lt;td> 4
  &lt;tr> &lt;td> 35 &lt;th> Tails &lt;td> 0.33 &lt;td> 1
&lt;/table></code></pre>

   <p>This would result in the following table:</p>

   <table>
    <thead>
     <tr> <th> ID <th> Measurement <th> Average <th> Maximum
    <tbody>
     <tr> <td> <th scope=rowgroup> Cats <td> <td>
     <tr> <td> 93 <th> Legs <td> 3.5 <td> 4
     <tr> <td> 10 <th> Tails <td> 1 <td> 1
    <tbody>
     <tr> <td> <th scope=rowgroup> English speakers <td> <td>
     <tr> <td> 32 <th> Legs <td> 2.67 <td> 4
     <tr> <td> 35 <th> Tails <td> 0.33 <td> 1
   </table>

   <p>The headers in the first row all apply directly down to the rows in their column.</p>

   <p>The headers with a <code data-x="attr-th-scope">scope</code> attribute in the <span
   data-x="attr-th-scope-rowgroup-state">rowgroup</span> state apply to all the cells in their row
   group other than the cells in the first column.</p>

   <p>The remaining headers apply just to the cells to the right of them.</p>

   <!-- image source: https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=151 -->
   <img src="/images/table-scope-diagram.png" width="459" height="256" alt=""><!-- (alt is empty because the diagram is completely described by the previous paragraphs) -->

  </div>



  <h4>Attributes common to <code>td</code> and <code>th</code> elements</h4>

  <p>The <code>td</code> and <code>th</code> elements may have a <dfn element-attr for="td,th"><code
  data-x="attr-tdth-colspan">colspan</code></dfn> content attribute specified, whose value must be a
  <span>valid non-negative integer</span> greater than zero and less than or equal to 1000.</p>

  <p>The <code>td</code> and <code>th</code> elements may also have a <dfn element-attr
  for="td,th"><code data-x="attr-tdth-rowspan">rowspan</code></dfn> content attribute specified,
  whose value must be a <span>valid non-negative integer</span> less than or equal to 65534.  For
  this attribute, the value zero means that the cell is to span all the remaining rows in the row
  group.</p> <!-- only in no-quirks and limited-quirks mode though, for back compat! -->

  <p>These attributes give the number of columns and rows respectively that the cell is to span.
  These attributes must not be used to overlap cells<span w-nodev>, as described in the
  description of the <span>table model</span></span>.</p> <!-- conformance criteria for determining
  when this is violated are given in the processing model -->

  <hr>

  <p>The <code>td</code> and <code>th</code> element may have a <dfn element-attr for="td,th"><code
  data-x="attr-tdth-headers">headers</code></dfn> content attribute specified. The <code
  data-x="attr-tdth-headers">headers</code> attribute, if specified, must contain a string
  consisting of an <span>unordered set of unique space-separated tokens</span>, none of which are
  <span>identical to</span> another token and each of which must have the value of an <span
  data-x="concept-id">ID</span> of a <code>th</code> element taking part in the same <span
  data-x="concept-table">table</span> as the <code>td</code> or <code>th</code> element<span
  w-nodev> (as defined by the <span>table model</span>)</span>.</p>

  <p>A <code>th</code> element with <span data-x="concept-id">ID</span> <var>id</var> is
  said to be <i>directly targeted</i> by all <code>td</code> and <code>th</code> elements in the
  same <span data-x="concept-table">table</span> that have <code
  data-x="attr-tdth-headers">headers</code> attributes whose values include as one of their tokens
  the <span data-x="concept-id">ID</span> <var>id</var>. A <code>th</code> element <var>A</var> is said to be <i>targeted</i> by a <code>th</code> or <code>td</code> element
  <var>B</var> if either <var>A</var> is <i>directly targeted</i> by <var>B</var> or if there exists an element <var>C</var> that is itself
  <i>targeted</i> by the element <var>B</var> and <var>A</var> is <i>directly
  targeted</i> by <var>C</var>.</p>

  <p>A <code>th</code> element must not be <i>targeted</i> by itself.</p>

  <div w-nodev>

  <p>The <code data-x="attr-tdth-colspan">colspan</code>, <code
  data-x="attr-tdth-rowspan">rowspan</code>, and <code data-x="attr-tdth-headers">headers</code>
  attributes take part in the <span>table model</span>.</p>

  </div>

  <hr>

  <dl class="domintro">
   <dt><code data-x=""><var>cell</var>.<span subdfn data-x="dom-tdth-cellIndex">cellIndex</span></code></dt>

   <dd>
    <p>Returns the position of the cell in the row's <code data-x="dom-tr-cells">cells</code> list.
    This does not necessarily correspond to the <var>x</var>-position of the cell in the table,
    since earlier cells might cover multiple rows or columns.</p>

    <p>Returns &#x2212;1 if the element isn't in a row.</p>
   </dd>
  </dl>

  <div w-nodev>

  <!--
   https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1699
   https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1700
  -->

  <!-- For colSpan/rowSpan rationale: https://github.com/whatwg/html/issues/1198 -->

  <p>The <dfn attribute for="HTMLTableCellElement"><code
  data-x="dom-tdth-colSpan">colSpan</code></dfn> IDL attribute must <span>reflect</span> the <code
  data-x="attr-tdth-colspan">colspan</code> content attribute. It is <span>clamped to the
  range</span> [1, 1000], and its <span>default value</span> is 1.</p>

  <p>The <dfn attribute for="HTMLTableCellElement"><code
  data-x="dom-tdth-rowSpan">rowSpan</code></dfn> IDL attribute must <span>reflect</span> the <code
  data-x="attr-tdth-rowspan">rowspan</code> content attribute. It is <span>clamped to the
  range</span> [0, 65534], and its <span>default value</span> is 1.</p>

  <p>The <dfn attribute for="HTMLTableCellElement"><code
  data-x="dom-tdth-headers">headers</code></dfn> IDL attribute must <span>reflect</span> the content
  attribute of the same name.</p>

  <p>The <dfn attribute for="HTMLTableCellElement"><code
  data-x="dom-tdth-cellIndex">cellIndex</code></dfn> IDL attribute must, if the element has a parent
  <code>tr</code> element, return the index of the cell's element in the parent element's <code
  data-x="dom-tr-cells">cells</code> collection. If there is no such parent element, then the
  attribute must return &#x2212;1.</p>

  <p>The <dfn attribute for="HTMLTableCellElement"><code data-x="dom-th-scope">scope</code></dfn>
  IDL attribute must <span>reflect</span> the content attribute of the same name, <span>limited to
  only known values</span>.</p>

  <p>The <dfn attribute for="HTMLTableCellElement"><code data-x="dom-th-abbr">abbr</code></dfn> IDL
  attribute must <span>reflect</span> the content attribute of the same name.</p>

  </div>


  <div w-nodev>

  <h4 id="table-processing-model"><span id="processing-model-2"></span>Processing model</h4>

  <p>The various table elements and their content attributes together define the <dfn>table
  model</dfn>.</p>

  <p>A <dfn data-x="concept-table">table</dfn> consists of cells aligned on a two-dimensional grid of
  <dfn data-x="concept-slots">slots</dfn> with coordinates (<var>x</var>, <var>y</var>). The grid is finite, and is either empty or has one or more slots. If the grid
  has one or more slots, then the <var>x</var> coordinates are always in the range <span
  data-x="">0&nbsp;&le;&nbsp;<var>x</var>&nbsp;&lt;&nbsp;<var>x<sub>width</sub></var></span>, and the <var>y</var> coordinates are always in the
  range <span data-x="">0&nbsp;&le;&nbsp;<var>y</var>&nbsp;&lt;&nbsp;<var>y<sub>height</sub></var></span>. If one or both of <var>x<sub>width</sub></var> and <var>y<sub>height</sub></var> are zero, then the
  table is empty (has no slots). Tables correspond to <code>table</code> elements.</p>

  <p>A <dfn data-x="concept-cell">cell</dfn> is a set of slots anchored at a slot (<var>cell<sub>x</sub></var>, <var>cell<sub>y</sub></var>), and with
  a particular <var>width</var> and <var>height</var> such that the cell covers
  all the slots with coordinates (<var>x</var>, <var>y</var>) where <span
  data-x=""><var>cell<sub>x</sub></var>&nbsp;&le;&nbsp;<var>x</var>&nbsp;&lt;&nbsp;<var>cell<sub>x</sub></var>+<var>width</var></span> and <var>cell<sub>y</sub></var>&nbsp;&le;&nbsp;<var>y</var>&nbsp;&lt;&nbsp;<var>cell<sub>y</sub></var>+<var>height</var>. Cells can either be <em>data cells</em>
  or <em>header cells</em>. Data cells correspond to <code>td</code> elements, and header cells
  correspond to <code>th</code> elements. Cells of both types can have zero or more associated
  header cells.</p>

  <p>It is possible, in certain error cases, for two cells to occupy the same slot.</p>

  <p>A <dfn data-x="concept-row">row</dfn> is a complete set of slots from <var>x</var>=0 to <var>x</var>=<var>x<sub>width</sub></var>-1, for a particular value of <var>y</var>. Rows usually
  correspond to <code>tr</code> elements, though a <span data-x="concept-row-group">row group</span>
  can have some implied <span data-x="concept-row">rows</span> at the end in some cases involving
  <span data-x="concept-cell">cells</span> spanning multiple rows.</p>

  <p>A <dfn data-x="concept-column">column</dfn> is a complete set of slots from <var>y</var>=0 to <var>y</var>=<var>y<sub>height</sub></var>-1, for a particular value of <var>x</var>. Columns can
  correspond to <code>col</code> elements. In the absence of <code>col</code> elements, columns are
  implied.</p>

  <p>A <dfn data-x="concept-row-group">row group</dfn> is a set of <span
  data-x="concept-row">rows</span> anchored at a slot (0, <var>group<sub>y</sub></var>) with a particular <var>height</var> such that the row group
  covers all the slots with coordinates (<var>x</var>, <var>y</var>) where <span
  data-x="">0&nbsp;&le;&nbsp;<var>x</var>&nbsp;&lt;&nbsp;<var>x<sub>width</sub></var></span> and <var>group<sub>y</sub></var>&nbsp;&le;&nbsp;<var>y</var>&nbsp;&lt;&nbsp;<var>group<sub>y</sub></var>+<var>height</var>. Row groups correspond to
  <code>tbody</code>, <code>thead</code>, and <code>tfoot</code> elements. Not every row is
  necessarily in a row group.</p>

  <p>A <dfn data-x="concept-column-group">column group</dfn> is a set of <span
  data-x="concept-column">columns</span> anchored at a slot (<var>group<sub>x</sub></var>, 0) with a particular <var>width</var> such that the column group
  covers all the slots with coordinates (<var>x</var>, <var>y</var>) where <span
  data-x=""><var>group<sub>x</sub></var>&nbsp;&le;&nbsp;<var>x</var>&nbsp;&lt;&nbsp;<var>group<sub>x</sub></var>+<var>width</var></span> and <span data-x="">0&nbsp;&le;&nbsp;<var>y</var>&nbsp;&lt;&nbsp;<var>y<sub>height</sub></var></span>. Column
  groups correspond to <code>colgroup</code> elements. Not every column is necessarily in a column
  group.</p>

  <p><span data-x="concept-row-group">Row groups</span> cannot overlap each other. Similarly, <span
  data-x="concept-column-group">column groups</span> cannot overlap each other.</p>

  <p>A <span data-x="concept-cell">cell</span> cannot cover slots that are from two or more <span
  data-x="concept-row-group">row groups</span>. It is, however, possible for a cell to be in multiple
  <span data-x="concept-column-group">column groups</span>. All the slots that form part of one cell
  are part of zero or one <span data-x="concept-row-group">row groups</span> and zero or more <span
  data-x="concept-column-group">column groups</span>.</p>

  <p>In addition to <span data-x="concept-cell">cells</span>, <span
  data-x="concept-column">columns</span>, <span data-x="concept-row">rows</span>, <span
  data-x="concept-row-group">row groups</span>, and <span data-x="concept-column-group">column
  groups</span>, <span data-x="concept-table">tables</span> can have a <code>caption</code> element
  associated with them. This gives the table a heading, or legend.</p>

  <p>A <dfn>table model error</dfn> is an error with the data represented by <code>table</code>
  elements and their descendants. Documents must not have table model errors.</p>


  <h5>Forming a table</h5>

  <p>To determine which elements correspond to which slots in a <span
  data-x="concept-table">table</span> associated with a <code>table</code> element, to determine the
  dimensions of the table (<var>x<sub>width</sub></var> and <var>y<sub>height</sub></var>), and to determine if there are any <span data-x="table model
  error">table model errors</span>, user agents must use the following algorithm:</p>

  <ol>
   <li>
    <p>Let <var>x<sub>width</sub></var> be zero.</p>
   </li>

   <li>
    <p>Let <var>y<sub>height</sub></var> be zero.</p>
   </li>

   <li>
    <p>Let <var>pending <code>tfoot</code> elements</var> be a list of <code>tfoot</code>
    elements, initially empty.</p>
   </li>

   <li>
    <p>Let <var>the table</var> be the <span data-x="concept-table">table</span> represented
    by the <code>table</code> element. The <var>x<sub>width</sub></var> and <var>y<sub>height</sub></var> variables give <var>the table</var>'s
    dimensions. <var>The table</var> is initially empty.</p>
   </li>

   <li>
    <!-- this step is here just so that we can be sure to have a first element in the next step, so
    that we can set up the "advance" exception-handling thingy below; otherwise it'd be more
    complicated. it's not a perf optimization per se. -->

    <p>If the <code>table</code> element has no children elements, then return <var>the
    table</var> (which will be empty).</p>
   </li>

   <li>
    <p>Associate the first <code>caption</code> element child of the <code>table</code> element with
    <var>the table</var>. If there are no such children, then it has no associated
    <code>caption</code> element.</p>
   </li>

   <li>
    <p>Let the <var>current element</var> be the first element child of the
    <code>table</code> element.</p>

    <p>If a step in this algorithm ever requires the <var>current element</var> to be <dfn
    data-x="concept-table-advance">advanced to the next child of the <code>table</code></dfn> when
    there is no such next child, then the user agent must jump to the step labeled <i>end</i>, near
    the end of this algorithm.</p>
   </li>

   <li>
    <p>While the <var>current element</var> is not one of the following elements, <span
    data-x="concept-table-advance">advance</span> the <var>current element</var> to the next
    child of the <code>table</code>:</p>

    <ul class="brief">
     <li><code>colgroup</code></li>
     <li><code>thead</code></li>
     <li><code>tbody</code></li>
     <li><code>tfoot</code></li>
     <li><code>tr</code></li>
    </ul>
   </li>

   <li>
    <p>If the <var>current element</var> is a <code>colgroup</code>, follow these
    substeps:</p>

    <ol>
     <li>
      <p><i>Column groups</i>: Process the <var>current element</var> according to the
      appropriate case below:</p>

      <dl class="switch">

       <dt>If the <var>current element</var> has any <code>col</code> element children</dt>

       <dd>
        <p>Follow these steps:</p>

        <ol>
         <li>
          <p>Let <var>x<sub>start</sub></var> have the value of <span
          data-x=""><var>x<sub>width</sub></var></span>.</p>
         </li>

         <li>
          <p>Let the <var>current column</var> be the first <code>col</code> element child
          of the <code>colgroup</code> element.</p>
         </li>

         <li>
          <p><i>Columns</i>: If the <var>current column</var> <code>col</code> element has
          a <code data-x="attr-col-span">span</code> attribute, then parse its value using the
          <span>rules for parsing non-negative integers</span>.</p>

          <p>If the result of parsing the value is not an error or zero, then let <var>span</var> be that value.</p>

          <p>Otherwise, if the <code>col</code> element has no <code
          data-x="attr-col-span">span</code> attribute, or if trying to parse the attribute's value
          resulted in an error or zero, then let <var>span</var> be 1.</p>

          <p>If <var>span</var> is greater than 1000, let it be 1000 instead.</p>
         </li>

         <li>
          <p>Increase <var>x<sub>width</sub></var> by <var>span</var>.</p>
         </li>

         <li>
          <p>Let the last <var>span</var> <span data-x="concept-column">columns</span> in
          <var>the table</var> correspond to the <var>current column</var>
          <code>col</code> element.</p>
         </li>

         <li>
          <p>If <var>current column</var> is not the last <code>col</code> element child of
          the <code>colgroup</code> element, then let the <var>current column</var> be the
          next <code>col</code> element child of the <code>colgroup</code> element, and return to
          the step labeled <i>columns</i>.</p>
         </li>

         <li>
          <p>Let all the last <span data-x="concept-column">columns</span> in <var>the
          table</var> from <span data-x="">x=<var>x<sub>start</sub></var></span> to
          <span data-x="">x=<var>x<sub>width</sub></var>-1</span> form a new <span
          data-x="concept-column-group">column group</span>, anchored at the slot (<var>x<sub>start</sub></var>, 0), with width <var>x<sub>width</sub></var>-<var>x<sub>start</sub></var>, corresponding to the <code>colgroup</code> element.</p>
         </li>
        </ol>
       </dd>


       <dt>If the <var>current element</var> has no <code>col</code> element children</dt>

       <dd>
        <ol>
         <li>
          <p>If the <code>colgroup</code> element has a <code data-x="attr-colgroup-span">span</code>
 attribute, then parse its value using the <span>rules for parsing non-negative
          integers</span>.</p>

          <p>If the result of parsing the value is not an error or zero, then let <var>span</var> be that value.</p>

          <p>Otherwise, if the <code>colgroup</code> element has no <code
          data-x="attr-colgroup-span">span</code> attribute, or if trying to parse the attribute's
          value resulted in an error or zero, then let <var>span</var> be 1.</p>

          <p>If <var>span</var> is greater than 1000, let it be 1000 instead.</p>
         </li>

         <li>
          <p>Increase <var>x<sub>width</sub></var> by <var>span</var>.</p>
         </li>

         <li>
          <p>Let the last <var>span</var> <span data-x="concept-column">columns</span> in
          <var>the table</var> form a new <span data-x="concept-column-group">column
          group</span>, anchored at the slot (<var>x<sub>width</sub></var>-<var>span</var>, 0), with width <var>span</var>, corresponding to the <code>colgroup</code> element.</p>
         </li>
        </ol>
       </dd>
      </dl>
     </li>

     <li>
      <p><span data-x="concept-table-advance">Advance</span> the <var>current element</var>
      to the next child of the <code>table</code>.</p>
     </li>

     <li>
      <p>While the <var>current element</var> is not one of the following elements, <span
      data-x="concept-table-advance">advance</span> the <var>current element</var> to the
      next child of the <code>table</code>:</p>

      <ul class="brief">
       <li><code>colgroup</code></li>
       <li><code>thead</code></li>
       <li><code>tbody</code></li>
       <li><code>tfoot</code></li>
       <li><code>tr</code></li>
      </ul>
     </li>

     <li>
      <p>If the <var>current element</var> is a <code>colgroup</code> element, jump to the
      step labeled <i>column groups</i> above.</p>
     </li>
    </ol>
   </li>

   <li>
    <p>Let <var>y<sub>current</sub></var> be zero.</p>
   </li>

   <li>
    <p>Let the <var>list of downward-growing cells</var> be an empty list.</p>
   </li>

   <li>
    <p><i>Rows</i>: While the <var>current element</var> is not one of the following
    elements, <span data-x="concept-table-advance">advance</span> the <var>current
    element</var> to the next child of the <code>table</code>:</p>

    <ul class="brief">
     <li><code>thead</code></li>
     <li><code>tbody</code></li>
     <li><code>tfoot</code></li>
     <li><code>tr</code></li>
    </ul>
   </li>

   <li>
    <p>If the <var>current element</var> is a <code>tr</code>, then run the <span>algorithm
    for processing rows</span>, <span data-x="concept-table-advance">advance</span> the <var>current element</var> to the next child of the <code>table</code>, and return to the
    step labeled <i>rows</i>.</p>
   </li>

   <li>
    <p>Run the <span>algorithm for ending a row group</span>.</p>
   </li>

   <li>
    <p>If the <var>current element</var> is a <code>tfoot</code>, then add that element to
    the list of <var>pending <code>tfoot</code> elements</var>, <span
    data-x="concept-table-advance">advance</span> the <var>current element</var> to the next
    child of the <code>table</code>, and return to the step labeled <i>rows</i>.</p>
   </li>

   <li>
    <p>The <var>current element</var> is either a <code>thead</code> or a
    <code>tbody</code>.</p>

    <p>Run the <span>algorithm for processing row groups</span>.</p>
   </li>

   <li>
    <p><span data-x="concept-table-advance">Advance</span> the <var>current element</var> to
    the next child of the <code>table</code>.</p>
   </li>

   <li>
    <p>Return to the step labeled <i>rows</i>.</p>
   </li>

   <li>
    <p><i>End</i>: For each <code>tfoot</code> element in the list of <var>pending
    <code>tfoot</code> elements</var>, in <span>tree order</span>, run the <span>algorithm for processing row
    groups</span>.</p>
   </li>

   <li>
    <p>If there exists a <span data-x="concept-row">row</span> or <span
    data-x="concept-column">column</span> in <var>the table</var> containing only <span
    data-x="concept-slots">slots</span> that do not have a <span data-x="concept-cell">cell</span>
    anchored to them, then this is a <span>table model error</span>.</p>
   </li>

   <li>
    <p>Return <var>the table</var>.</p>
   </li>
  </ol>

  <p>The <dfn>algorithm for processing row groups</dfn>, which is invoked by the set of steps above
  for processing <code>thead</code>, <code>tbody</code>, and <code>tfoot</code> elements, is:</p>

  <ol>
   <li>
    <p>Let <var>y<sub>start</sub></var> have the value of <var>y<sub>height</sub></var>.</p>
   </li>

   <li>
    <p>For each <code>tr</code> element that is a child of the element being processed, in tree
    order, run the <span>algorithm for processing rows</span>.</p>
   </li>

   <li>
    <!-- if we added any rows, make them part of a row group -->
    <p>If <var>y<sub>height</sub></var>&nbsp;&gt;&nbsp;<var>y<sub>start</sub></var>, then let all the last <span
    data-x="concept-row">rows</span> in <var>the table</var> from <span data-x="">y=<var>y<sub>start</sub></var></span> to <span data-x="">y=<var>y<sub>height</sub></var>-1</span> form a new <span data-x="concept-row-group">row
    group</span>, anchored at the slot with coordinate (0, <var>y<sub>start</sub></var>), with height <var>y<sub>height</sub></var>-<var>y<sub>start</sub></var>, corresponding
    to the element being processed.</p>
   </li>

   <li>
    <p>Run the <span>algorithm for ending a row group</span>.</p>
   </li>
  </ol>

  <p>The <dfn>algorithm for ending a row group</dfn>, which is invoked by the set of steps above
  when starting and ending a block of rows, is:</p>

  <ol>
   <li>
    <p>While <var>y<sub>current</sub></var> is less than <var>y<sub>height</sub></var>, follow these steps:</p>

    <ol>
     <li>
      <p>Run the <span>algorithm for growing downward-growing cells</span>.</p>
     </li>

     <li>
      <p>Increase <var>y<sub>current</sub></var> by 1.</p>
     </li>
    </ol>
   </li>

   <li>
    <p>Empty the <var>list of downward-growing cells</var>.</p>
   </li>
  </ol>


  <p>The <dfn>algorithm for processing rows</dfn>, which is invoked by the set of steps above for
  processing <code>tr</code> elements, is:</p>

  <ol>
   <li>
    <p>If <var>y<sub>height</sub></var> is equal to <var>y<sub>current</sub></var>, then increase <var>y<sub>height</sub></var> by
    1. (<var>y<sub>current</sub></var> is never <em>greater</em> than <var>y<sub>height</sub></var>.)</p>
   </li>

   <li>
    <p>Let <var>x<sub>current</sub></var> be 0.</p>
   </li>

   <li>
    <p>Run the <span>algorithm for growing downward-growing cells</span>.</p>
   </li>

   <li>
    <p>If the <code>tr</code> element being processed has no <code>td</code> or <code>th</code>
    element children, then increase <var>y<sub>current</sub></var> by 1, abort
    this set of steps, and return to the algorithm above.</p>
   </li>

   <li>
    <p>Let <var>current cell</var> be the first <code>td</code> or <code>th</code> element child
    in the <code>tr</code> element being processed.</p>
   </li>

   <li>
    <p><i>Cells</i>: While <var>x<sub>current</sub></var> is less than <var>x<sub>width</sub></var> and the slot with coordinate (<var>x<sub>current</sub></var>, <var>y<sub>current</sub></var>) already has a
    cell assigned to it, increase <var>x<sub>current</sub></var> by 1.</p>
   </li>

   <li>
    <p>If <var>x<sub>current</sub></var> is equal to <var>x<sub>width</sub></var>, increase <var>x<sub>width</sub></var> by 1. (<var>x<sub>current</sub></var> is never <em>greater</em> than <var>x<sub>width</sub></var>.)</p>
   </li>

   <li>
    <p>If the <var>current cell</var> has a <code data-x="attr-tdth-colspan">colspan</code>
    attribute, then <span data-x="rules for parsing non-negative integers">parse that attribute's
    value</span>, and let <var>colspan</var> be the result.</p>

    <p>If parsing that value failed, or returned zero, or if the attribute is absent, then let <var>colspan</var> be 1, instead.</p>

    <p>If <var>colspan</var> is greater than 1000, let it be 1000 instead.</p>
   </li>

   <li>
    <p>If the <var>current cell</var> has a <code data-x="attr-tdth-rowspan">rowspan</code>
    attribute, then <span data-x="rules for parsing non-negative integers">parse that attribute's
    value</span>, and let <var>rowspan</var> be the result.</p>

    <p>If parsing that value failed or if the attribute is absent, then let <var>rowspan</var> be 1, instead.</p>

    <p>If <var>rowspan</var> is greater than 65534, let it be 65534 instead.</p>
   </li>

   <li>
    <p>If <var>rowspan</var> is zero and the <code>table</code> element's
    <span>node document</span> is not set to <span>quirks mode</span>, then let <var>cell grows
    downward</var> be true, and set <var>rowspan</var> to 1. Otherwise, let <var>cell grows downward</var> be false.</p>
   </li>

   <li>
    <p>If <var>x<sub>width</sub></var>&nbsp;&lt;&nbsp;<var>x<sub>current</sub></var>+<var>colspan</var>, then let <var>x<sub>width</sub></var> be <var>x<sub>current</sub></var>+<var>colspan</var>.</p>
   </li>

   <li>
    <p>If <var>y<sub>height</sub></var>&nbsp;&lt;&nbsp;<var>y<sub>current</sub></var>+<var>rowspan</var>, then let <var>y<sub>height</sub></var> be <var>y<sub>current</sub></var>+<var>rowspan</var>.</p>
   </li>

   <li>
    <p>Let the slots with coordinates (<var>x</var>, <var>y</var>) such that <span
    data-x=""><var>x<sub>current</sub></var>&nbsp;&le;&nbsp;<var>x</var>&nbsp;&lt;&nbsp;<var>x<sub>current</sub></var>+<var>colspan</var></span> and <var>y<sub>current</sub></var>&nbsp;&le;&nbsp;<var>y</var>&nbsp;&lt;&nbsp;<var>y<sub>current</sub></var>+<var>rowspan</var> be covered by a
    new <span data-x="concept-cell">cell</span> <var>c</var>, anchored at (<var>x<sub>current</sub></var>, <var>y<sub>current</sub></var>),
    which has width <var>colspan</var> and height <var>rowspan</var>,
    corresponding to the <var>current cell</var> element.</p>

    <p>If the <var>current cell</var> element is a <code>th</code> element, let this new
    cell <var>c</var> be a header cell; otherwise, let it be a data cell.</p>

    <p>To establish which header cells apply to the <var>current cell</var> element, use
    the <span>algorithm for assigning header cells</span> described in the next section.</p>

    <p>If any of the slots involved already had a <span data-x="concept-cell">cell</span> covering
    them, then this is a <span>table model error</span>. Those slots now have two cells
    overlapping.</p>
   </li>

   <li>
    <p>If <var>cell grows downward</var> is true, then add the tuple {<var>c</var>, <var>x<sub>current</sub></var>, <var>colspan</var>}
    to the <var>list of downward-growing cells</var>.</p>
   </li>

   <li>
    <p>Increase <var>x<sub>current</sub></var> by <var>colspan</var>.</p>
   </li>

   <li>
    <p>If <var>current cell</var> is the last <code>td</code> or <code>th</code> element child in
    the <code>tr</code> element being processed, then increase <var>y<sub>current</sub></var> by 1, abort this set of steps, and return to the algorithm
    above.</p>
   </li>

   <li>
    <p>Let <var>current cell</var> be the next <code>td</code> or <code>th</code> element child
    in the <code>tr</code> element being processed.</p>
   </li>

   <li>
    <p>Return to the step labeled <i>cells</i>.</p>
   </li>
  </ol>


  <p>When the algorithms above require the user agent to run the <dfn>algorithm for growing
  downward-growing cells</dfn>, the user agent must, for each {<var>cell</var>, <var>cell<sub>x</sub></var>, <var>width</var>} tuple in the <var>list of downward-growing cells</var>, if any, extend the <span
  data-x="concept-cell">cell</span> <var>cell</var> so that it also covers the slots with
  coordinates (<var>x</var>, <var>y<sub>current</sub></var>), where <span
  data-x=""><var>cell<sub>x</sub></var>&nbsp;&le;&nbsp;<var>x</var>&nbsp;&lt;&nbsp;<var>cell<sub>x</sub></var>+<var>width</var></span>.</p>




  <h5 id="header-and-data-cell-semantics">Forming relationships between data cells and header cells</h5>

  <p>Each cell can be assigned zero or more header cells. The <dfn>algorithm for assigning header
  cells</dfn> to a cell <var>principal cell</var> is as follows.</p>

  <ol>
   <!-- INITIALIZATION -->

   <li>
    <p>Let <var>header list</var> be an empty list of cells.</p>
   </li>

   <li>
    <p>Let (<var>principal<sub>x</sub></var>, <var>principal<sub>y</sub></var>) be the coordinate of the slot to which the <var>principal
    cell</var> is anchored.</p>
   </li>

   <li>
    <dl class="switch">

     <dt>If the <var>principal cell</var> has a <code
     data-x="attr-tdth-headers">headers</code> attribute specified</dt>

     <dd>
      <!-- HEADERS="" -->

      <ol>
       <li>
        <p>Take the value of the <var>principal cell</var>'s <code
        data-x="attr-tdth-headers">headers</code> attribute and <span data-x="split a string on
        ASCII whitespace">split it on ASCII whitespace</span>, letting <var>id list</var> be the
        list of tokens obtained.</p>
       </li>

       <li>
        <!-- support headers="" on <td> element for legacy compat -->
        <!-- note that it's not conforming though -->
        <p>For each token in the <var>id list</var>, if the
        first element in the <code>Document</code> with an <span data-x="concept-id">ID</span> equal to
        the token is a cell in the same <span
        data-x="concept-table">table</span>, and that cell is not the
        <var>principal cell</var>, then add that cell to <var>header list</var>.</p>
       </li>
      </ol>
     </dd>


     <dt>If <var>principal cell</var> does not have a <code
     data-x="attr-tdth-headers">headers</code> attribute specified</dt>

     <dd>
      <ol>
       <li>
        <p>Let <var>principal<sub>width</sub></var> be the width of the <var>principal cell</var>.</p>
       </li>

       <li>
        <p>Let <var>principal<sub>height</sub></var> be the height of the <var>principal cell</var>.</p>
       </li>


       <!-- HORIZONTAL -->

       <li>
        <p>For each value of <var>y</var> from <var>principal<sub>y</sub></var> to <var>principal<sub>y</sub></var>+<var>principal<sub>height</sub></var>-1, run
        the <span>internal algorithm for scanning and assigning header cells</span>, with the <var>principal cell</var>, the <var>header list</var>, the initial coordinate
        (<var>principal<sub>x</sub></var>, <var>y</var>), and the
        increments <span data-x="">&Delta;<var>x</var>=&#x2212;1</span> and <span
        data-x="">&Delta;<var>y</var>=0</span>.</p>
       </li>


       <!-- VERTICAL -->

       <li>
        <p>For each value of <var>x</var> from <var>principal<sub>x</sub></var> to <var>principal<sub>x</sub></var>+<var>principal<sub>width</sub></var>-1, run
        the <span>internal algorithm for scanning and assigning header cells</span>, with the <var>principal cell</var>, the <var>header list</var>, the initial coordinate
        (<var>x</var>, <var>principal<sub>y</sub></var>), and the
        increments <span data-x="">&Delta;<var>x</var>=0</span> and <span
        data-x="">&Delta;<var>y</var>=&#x2212;1</span>.</p>
       </li>


       <!-- ROW GROUP HEADERS -->

       <li>
        <p>If the <var>principal cell</var> is anchored in a <span
        data-x="concept-row-group">row group</span>, then add all header cells that are <span
        data-x="row group header">row group headers</span> and are anchored in the same row group
        with an <var>x</var>-coordinate less than or equal to <var>principal<sub>x</sub></var>+<var>principal<sub>width</sub></var>-1 and a <var>y</var>-coordinate less than or
        equal to <var>principal<sub>y</sub></var>+<var>principal<sub>height</sub></var>-1 to <var>header
        list</var>.</p>

        <!-- this might introduce principal accidentally; fixed below -->
       </li>


       <!-- COLUMN GROUP HEADERS -->

       <li>
        <p>If the <var>principal cell</var> is anchored in a <span
        data-x="concept-column-group">column group</span>, then add all header cells that are <span
        data-x="column group header">column group headers</span> and are anchored in the same column
        group with an <var>x</var>-coordinate less than or equal to <var>principal<sub>x</sub></var>+<var>principal<sub>width</sub></var>-1 and a <var>y</var>-coordinate less than or
        equal to <var>principal<sub>y</sub></var>+<var>principal<sub>height</sub></var>-1 to <var>header
        list</var>.</p>

        <!-- this might introduce principal accidentally; fixed below -->
       </li>
      </ol>
     </dd>
    </dl>
   </li>


   <!--CLEANING UP-->

   <li>
    <p>Remove all the <span data-x="empty cell">empty cells</span> from the <var>header
    list</var>.</p>
   </li>

   <li>
    <p>Remove any duplicates from the <var>header list</var>.</p>
   </li>

   <li>
    <p>Remove <var>principal cell</var> from the <var>header list</var> if it is
    there.</p> <!-- see "might introduce principal accidentally" above -->
   </li>

   <li>
    <p>Assign the headers in the <var>header list</var> to the <var>principal
    cell</var>.</p>
   </li>
  </ol>

  <p>The <dfn>internal algorithm for scanning and assigning header cells</dfn>, given a <var>principal cell</var>, a <var>header list</var>, an initial coordinate (<var>initial<sub>x</sub></var>, <var>initial<sub>y</sub></var>),
  and &Delta;<var>x</var> and &Delta;<var>y</var> increments, is as follows:</p>

  <ol>
   <li>
    <p>Let <var>x</var> equal <var>initial<sub>x</sub></var>.</p>
   </li>

   <li>
    <p>Let <var>y</var> equal <var>initial<sub>y</sub></var>.</p>
   </li>

   <li>
    <p>Let <var>opaque headers</var> be an empty list of cells.</p>
   </li>

   <li>
    <dl class="switch">

     <dt>If <var>principal cell</var> is a header cell</dt>

     <dd><p>Let <var>in header block</var> be true, and let <var>headers from
     current header block</var> be a list of cells containing just the <var>principal
     cell</var>.</p></dd>

     <dt>Otherwise</dt>

     <dd><p>Let <var>in header block</var> be false and let <var>headers from
     current header block</var> be an empty list of cells.</p>
    </dl>
   </li>

   <li>
    <p><i>Loop</i>: Increment <var>x</var> by &Delta;<var>x</var>; increment <var>y</var> by &Delta;<var>y</var>.</p>

    <p class="note">For each invocation of this algorithm, one of &Delta;<var>x</var> and
    &Delta;<var>y</var> will be &#x2212;1, and the other will be 0.</p>
   </li>

   <li>
    <p>If either <var>x</var> or <var>y</var> are less than 0, then abort this
    internal algorithm.</p>
   </li>

   <li>
    <p>If there is no cell covering slot (<var>x</var>, <var>y</var>), or if there
    is more than one cell covering slot (<var>x</var>, <var>y</var>), return to
    the substep labeled <i>loop</i>.</p>
   </li>

   <li>
    <p>Let <var>current cell</var> be the cell covering slot (<var>x</var>, <var>y</var>).</p>
   </li>

   <li>
    <dl class="switch">

     <dt>If <var>current cell</var> is a header cell</dt>

     <dd>
      <ol>
       <li><p>Set <var>in header block</var> to true.</p></li>

       <li><p>Add <var>current cell</var> to <var>headers from current header
       block</var>.</p></li>

       <li><p>Let <var>blocked</var> be false.</p></li>

       <li>
        <dl class="switch">

         <dt>If &Delta;<var>x</var> is 0</dt>

         <dd>
          <p>If there are any cells in the <var>opaque headers</var> list anchored with the
          same <var>x</var>-coordinate as the <var>current cell</var>, and with
          the same width as <var>current cell</var>, then let <var>blocked</var>
          be true.</p>

          <p>If the <var>current cell</var> is not a <span>column header</span>, then let
          <var>blocked</var> be true.</p>
         </dd>

         <dt>If &Delta;<var>y</var> is 0</dt>

         <dd>
          <p>If there are any cells in the <var>opaque headers</var> list anchored with the
          same <var>y</var>-coordinate as the <var>current cell</var>, and with
          the same height as <var>current cell</var>, then let <var>blocked</var>
          be true.</p>

          <p>If the <var>current cell</var> is not a <span>row header</span>, then let <var>blocked</var> be true.</p>
         </dd>
        </dl>
       </li>

       <li><p>If <var>blocked</var> is false, then add the <var>current cell</var>
       to the <var>header list</var>.</p></li>
      </ol>
     </dd>

     <dt>If <var>current cell</var> is a data cell and <var>in header block</var> is true</dt>

     <dd><p>Set <var>in header block</var> to false. Add all the cells in <var>headers from current header block</var> to the <var>opaque headers</var>
     list, and empty the <var>headers from current header block</var> list.</p>
    </dl>
   </li>

   <li>
    <p>Return to the step labeled <i>loop</i>.</p>
   </li>
  </ol>

  <p>A header cell anchored at the slot with coordinate (<var>x</var>, <var>y</var>) with width
  <var>width</var> and height <var>height</var> is said to be a <dfn>column header</dfn> if any of
  the following are true:</p>

  <ul>
   <li><p>the cell's <code data-x="attr-th-scope">scope</code> attribute is in the <span
   data-x="attr-th-scope-column-state">column</span> state; or</p></li>

   <li><p>the cell's <code data-x="attr-th-scope">scope</code> attribute is in the <span
   data-x="attr-th-scope-auto-state">auto</span> state, and there are no data cells in any of the
   cells covering slots with <var>y</var>-coordinates <var>y</var> .. <span
   data-x=""><var>y</var>+<var>height</var>-1</span>.</p></li>
  </ul>

  <p>A header cell anchored at the slot with coordinate (<var>x</var>, <var>y</var>) with width
  <var>width</var> and height <var>height</var> is said to be a <dfn>row header</dfn> if any of the
  following are true:</p>

  <ul>
   <li><p>the cell's <code data-x="attr-th-scope">scope</code> attribute is in the <span
   data-x="attr-th-scope-row-state">row</span> state; or</p></li>

   <li><p>the cell's <code data-x="attr-th-scope">scope</code> attribute is in the <span
   data-x="attr-th-scope-auto-state">auto</span> state, the cell is not a <span>column
   header</span>, and there are no data cells in any of the cells covering slots with
   <var>x</var>-coordinates <var>x</var> .. <var>x</var>+<var>width</var>-1.</p></li>
  </ul>

  <p>A header cell is said to be a <dfn>column group header</dfn> if its <code
  data-x="attr-th-scope">scope</code> attribute is in the <span
  data-x="attr-th-scope-colgroup-state">column group</span> state.</p>

  <p>A header cell is said to be a <dfn>row group header</dfn> if its <code
  data-x="attr-th-scope">scope</code> attribute is in the <span
  data-x="attr-th-scope-rowgroup-state">row group</span> state.</p>

  <p>A cell is said to be an <dfn>empty cell</dfn> if it contains no elements and its <span>child
  text content</span>, if any, consists only of <span>ASCII whitespace</span>.</p>

  </div>

  <h4 id="table-examples">Examples</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>The following shows how one might mark up the bottom part of table 45 of the <cite>Smithsonian
  physical tables, Volume 71</cite>:</p>

   <!-- Smithsonian physical tables, Volume 71: By Smithsonian Institution, Frederick Eugene Fowle; page 76 -->
   <!-- from the reprint of the seventh revised edition; publication 2539, published 1921 -->
  <pre><code class="html">&lt;table>
 &lt;caption>Specification values: &lt;b>Steel&lt;/b>, &lt;b>Castings&lt;/b>,
 Ann. A.S.T.M. A27-16, Class B;* P max. 0.06; S max. 0.05.&lt;/caption>
 &lt;thead>
  &lt;tr>
   &lt;th rowspan=2>Grade.&lt;/th>
   &lt;th rowspan=2>Yield Point.&lt;/th>
   &lt;th colspan=2>Ultimate tensile strength&lt;/th>
   &lt;th rowspan=2>Per cent elong. 50.8&amp;nbsp;mm or 2&amp;nbsp;in.&lt;/th>
   &lt;th rowspan=2>Per cent reduct. area.&lt;/th>
  &lt;/tr>
  &lt;tr>
   &lt;th>kg/mm&lt;sup>2&lt;/sup>&lt;/th>
   &lt;th>lb/in&lt;sup>2&lt;/sup>&lt;/th>
  &lt;/tr>
 &lt;/thead>
 &lt;tbody>
  &lt;tr>
   &lt;td>Hard&lt;/td>
   &lt;td>0.45 ultimate&lt;/td>
   &lt;td>56.2&lt;/td>
   &lt;td>80,000&lt;/td>
   &lt;td>15&lt;/td>
   &lt;td>20&lt;/td>
  &lt;/tr>
  &lt;tr>
   &lt;td>Medium&lt;/td>
   &lt;td>0.45 ultimate&lt;/td>
   &lt;td>49.2&lt;/td>
   &lt;td>70,000&lt;/td>
   &lt;td>18&lt;/td>
   &lt;td>25&lt;/td>
  &lt;/tr>
  &lt;tr>
   &lt;td>Soft&lt;/td>
   &lt;td>0.45 ultimate&lt;/td>
   &lt;td>42.2&lt;/td>
   &lt;td>60,000&lt;/td>
   &lt;td>22&lt;/td>
   &lt;td>30&lt;/td>
  &lt;/tr>
 &lt;/tbody>
&lt;/table></code></pre>

  <p>This table could look like this:</p>

  <table id="table-example-1">
   <caption>Specification values: <b>Steel</b>, <b>Castings</b>,
   Ann. A.S.T.M. A27-16, Class B;* P max. 0.06; S max. 0.05.</caption>
   <thead>
    <tr>
     <th rowspan=2>Grade.</th>
     <th rowspan=2>Yield Point.</th>
     <th colspan=2>Ultimate tensile strength</th>
     <th rowspan=2>Per cent elong. 50.8&nbsp;mm or&nbsp;2&nbsp;in.</th>
     <th rowspan=2>Per cent reduct. area.</th>
    </tr>
    <tr>
     <th>kg/mm<sup>2</sup></th>
     <th>lb/in<sup>2</sup></th>
    </tr>
   </thead>
   <tbody>
    <tr>
     <td>Hard</td>
     <td>0.45 ultimate</td>
     <td>56.2</td>
     <td>80,000</td>
     <td>15</td>
     <td>20</td>
    </tr>
    <tr>
     <td>Medium</td>
     <td>0.45 ultimate</td>
     <td>49.2</td>
     <td>70,000</td>
     <td>18</td>
     <td>25</td>
    </tr>
    <tr>
     <td>Soft</td>
     <td>0.45 ultimate</td>
     <td>42.2</td>
     <td>60,000</td>
     <td>22</td>
     <td>30</td>
    </tr>
   </tbody>
  </table>

  <hr>

  <p>The following shows how one might mark up the gross margin table on page 46 of Apple, Inc's
  10-K filing for fiscal year 2008:</p>

  <pre><code class="html">&lt;table>
 &lt;thead>
  &lt;tr>
   &lt;th>
   &lt;th>2008
   &lt;th>2007
   &lt;th>2006
 &lt;tbody>
  &lt;tr>
   &lt;th>Net sales
   &lt;td>$ 32,479
   &lt;td>$ 24,006
   &lt;td>$ 19,315
  &lt;tr>
   &lt;th>Cost of sales
   &lt;td>  21,334
   &lt;td>  15,852
   &lt;td>  13,717
 &lt;tbody>
  &lt;tr>
   &lt;th>Gross margin
   &lt;td>$ 11,145
   &lt;td>$  8,154
   &lt;td>$  5,598
 &lt;tfoot>
  &lt;tr>
   &lt;th>Gross margin percentage
   &lt;td>34.3%
   &lt;td>34.0%
   &lt;td>29.0%
&lt;/table></code></pre>

  <p>This table could look like this:</p>

  <table class="apple-table-examples e1">
   <thead>
    <tr>
     <th>
     <th>2008
     <th>2007
     <th>2006
   <tbody>
    <tr>
     <th>Net sales
     <td>$ 32,479
     <td>$ 24,006
     <td>$ 19,315
    <tr>
     <th>Cost of sales
     <td>  21,334
     <td>  15,852
     <td>  13,717
   <tbody>
    <tr>
     <th>Gross margin
     <td>$ 11,145
     <td>$  8,154
     <td>$  5,598
   <tfoot>
    <tr>
     <th>Gross margin percentage
     <td>34.3%
     <td>34.0%
     <td>29.0%
  </table>

  <hr>

  <p>The following shows how one might mark up the operating expenses table from lower on the same
  page of that document:</p>

  <pre><code class="html">&lt;table>
 &lt;colgroup> &lt;col>
 &lt;colgroup> &lt;col> &lt;col> &lt;col>
 &lt;thead>
  &lt;tr> &lt;th> &lt;th>2008 &lt;th>2007 &lt;th>2006
 &lt;tbody>
  &lt;tr> &lt;th scope=rowgroup> Research and development
       &lt;td> $ 1,109 &lt;td> $ 782 &lt;td> $ 712
  &lt;tr> &lt;th scope=row> Percentage of net sales
       &lt;td> 3.4% &lt;td> 3.3% &lt;td> 3.7%
 &lt;tbody>
  &lt;tr> &lt;th scope=rowgroup> Selling, general, and administrative
       &lt;td> $ 3,761 &lt;td> $ 2,963 &lt;td> $ 2,433
  &lt;tr> &lt;th scope=row> Percentage of net sales
       &lt;td> 11.6% &lt;td> 12.3% &lt;td> 12.6%
&lt;/table></code></pre>

  <p>This table could look like this:</p>

  <table class="apple-table-examples e2">
   <thead>
    <tr> <th> <th>2008 <th>2007 <th>2006
   <tbody>
    <tr> <th scope=rowgroup> Research and development
         <td> $ 1,109 <td> $ 782 <td> $ 712
    <tr> <th scope=row> Percentage of net sales
         <td> 3.4% <td> 3.3% <td> 3.7%
   <tbody>
    <tr> <th scope=rowgroup> Selling, general, and administrative
         <td> $ 3,761 <td> $ 2,963 <td> $ 2,433
    <tr> <th scope=row> Percentage of net sales
         <td> 11.6% <td> 12.3% <td> 12.6%
  </table>



  <h3 split-filename="forms" id="forms">Forms</h3>

  <h4>Introduction</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>A form is a component of a web page that has form controls, such as text, buttons, checkboxes,
  range, or color picker controls. A user can interact with such a form, providing data that can
  then be sent to the server for further processing (e.g. returning the results of a search or
  calculation). No client-side scripting is needed in many cases, though an API is available so that
  scripts can augment the user experience or use forms for purposes other than submitting data to a
  server.</p>

  <p>Writing a form consists of several steps, which can be performed in any order: writing the user
  interface, implementing the server-side processing, and configuring the user interface to
  communicate with the server.</p>


  <h5>Writing a form's user interface</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>For the purposes of this brief introduction, we will create a pizza ordering form.</p>

  <p>Any form starts with a <code>form</code> element, inside which are placed the controls. Most
  controls are represented by the <code>input</code> element, which by default provides a text
  control. To label a control, the <code>label</code> element is used; the label text and the
  control itself go inside the <code>label</code> element. Each part of a form is considered a
  <span>paragraph</span>, and is typically separated from other parts using <code>p</code> elements.
  Putting this together, here is how one might ask for the customer's name:</p>

  <pre><code class="html"><strong>&lt;form>
 &lt;p>&lt;label>Customer name: &lt;input>&lt;/label>&lt;/p>
&lt;/form></strong></code></pre>

  <p>To let the user select the size of the pizza, we can use a set of radio buttons. Radio buttons
  also use the <code>input</code> element, this time with a <code
  data-x="attr-input-type">type</code> attribute with the value <code
  data-x="attr-input-type-radio">radio</code>. To make the radio buttons work as a group, they are
  given a common name using the <code data-x="attr-fe-name">name</code> attribute. To group a batch
  of controls together, such as, in this case, the radio buttons, one can use the
  <code>fieldset</code> element. The title of such a group of controls is given by the first element
  in the <code>fieldset</code>, which has to be a <code>legend</code> element.</p>

  <pre><code class="html">&lt;form>
 &lt;p>&lt;label>Customer name: &lt;input>&lt;/label>&lt;/p>
<strong> &lt;fieldset>
  &lt;legend> Pizza Size &lt;/legend>
  &lt;p>&lt;label> &lt;input type=radio name=size> Small &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size> Medium &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size> Large &lt;/label>&lt;/p>
 &lt;/fieldset></strong>
&lt;/form></code></pre>

  <p class="note">Changes from the previous step are highlighted.</p>

  <p>To pick toppings, we can use checkboxes. These use the <code>input</code> element with a <code
  data-x="attr-input-type">type</code> attribute with the value <code
  data-x="attr-input-type-checkbox">checkbox</code>:</p>

  <pre><code class="html">&lt;form>
 &lt;p>&lt;label>Customer name: &lt;input>&lt;/label>&lt;/p>
 &lt;fieldset>
  &lt;legend> Pizza Size &lt;/legend>
  &lt;p>&lt;label> &lt;input type=radio name=size> Small &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size> Medium &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size> Large &lt;/label>&lt;/p>
 &lt;/fieldset>
<strong> &lt;fieldset>
  &lt;legend> Pizza Toppings &lt;/legend>
  &lt;p>&lt;label> &lt;input type=checkbox> Bacon &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox> Extra Cheese &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox> Onion &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox> Mushroom &lt;/label>&lt;/p>
 &lt;/fieldset></strong>
&lt;/form></code></pre>

  <p>The pizzeria for which this form is being written is always making mistakes, so it needs a way
  to contact the customer. For this purpose, we can use form controls specifically for telephone
  numbers (<code>input</code> elements with their <code data-x="attr-input-type">type</code>
  attribute set to <code data-x="attr-input-type-tel">tel</code>) and email addresses
  (<code>input</code> elements with their <code data-x="attr-input-type">type</code> attribute set
  to <code data-x="attr-input-type-email">email</code>):</p>

  <pre><code class="html">&lt;form>
 &lt;p>&lt;label>Customer name: &lt;input>&lt;/label>&lt;/p>
<strong> &lt;p>&lt;label>Telephone: &lt;input type=tel>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Email address: &lt;input type=email>&lt;/label>&lt;/p></strong>
 &lt;fieldset>
  &lt;legend> Pizza Size &lt;/legend>
  &lt;p>&lt;label> &lt;input type=radio name=size> Small &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size> Medium &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size> Large &lt;/label>&lt;/p>
 &lt;/fieldset>
 &lt;fieldset>
  &lt;legend> Pizza Toppings &lt;/legend>
  &lt;p>&lt;label> &lt;input type=checkbox> Bacon &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox> Extra Cheese &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox> Onion &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox> Mushroom &lt;/label>&lt;/p>
 &lt;/fieldset>
&lt;/form></code></pre>

  <p>We can use an <code>input</code> element with its <code data-x="attr-input-type">type</code>
  attribute set to <code data-x="attr-input-type-time">time</code> to ask for a delivery time. Many
  of these form controls have attributes to control exactly what values can be specified; in this
  case, three attributes of particular interest are <code data-x="attr-input-min">min</code>, <code
  data-x="attr-input-max">max</code>, and <code data-x="attr-input-step">step</code>. These set the
  minimum time, the maximum time, and the interval between allowed values (in seconds). This
  pizzeria only delivers between 11am and 9pm, and doesn't promise anything better than 15 minute
  increments, which we can mark up as follows:</p>

  <pre><code class="html">&lt;form>
 &lt;p>&lt;label>Customer name: &lt;input>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Telephone: &lt;input type=tel>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Email address: &lt;input type=email>&lt;/label>&lt;/p>
 &lt;fieldset>
  &lt;legend> Pizza Size &lt;/legend>
  &lt;p>&lt;label> &lt;input type=radio name=size> Small &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size> Medium &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size> Large &lt;/label>&lt;/p>
 &lt;/fieldset>
 &lt;fieldset>
  &lt;legend> Pizza Toppings &lt;/legend>
  &lt;p>&lt;label> &lt;input type=checkbox> Bacon &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox> Extra Cheese &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox> Onion &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox> Mushroom &lt;/label>&lt;/p>
 &lt;/fieldset>
<strong> &lt;p>&lt;label>Preferred delivery time: &lt;input type=time min="11:00" max="21:00" step="900">&lt;/label>&lt;/p></strong>
&lt;/form></code></pre>

  <p>The <code>textarea</code> element can be used to provide a multiline text control. In this
  instance, we are going to use it to provide a space for the customer to give delivery
  instructions:</p>

  <pre><code class="html">&lt;form>
 &lt;p>&lt;label>Customer name: &lt;input>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Telephone: &lt;input type=tel>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Email address: &lt;input type=email>&lt;/label>&lt;/p>
 &lt;fieldset>
  &lt;legend> Pizza Size &lt;/legend>
  &lt;p>&lt;label> &lt;input type=radio name=size> Small &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size> Medium &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size> Large &lt;/label>&lt;/p>
 &lt;/fieldset>
 &lt;fieldset>
  &lt;legend> Pizza Toppings &lt;/legend>
  &lt;p>&lt;label> &lt;input type=checkbox> Bacon &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox> Extra Cheese &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox> Onion &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox> Mushroom &lt;/label>&lt;/p>
 &lt;/fieldset>
 &lt;p>&lt;label>Preferred delivery time: &lt;input type=time min="11:00" max="21:00" step="900">&lt;/label>&lt;/p>
<strong> &lt;p>&lt;label>Delivery instructions: &lt;textarea>&lt;/textarea>&lt;/label>&lt;/p></strong>
&lt;/form></code></pre>

  <p>Finally, to make the form submittable we use the <code>button</code> element:</p>

  <pre><code class="html">&lt;form>
 &lt;p>&lt;label>Customer name: &lt;input>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Telephone: &lt;input type=tel>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Email address: &lt;input type=email>&lt;/label>&lt;/p>
 &lt;fieldset>
  &lt;legend> Pizza Size &lt;/legend>
  &lt;p>&lt;label> &lt;input type=radio name=size> Small &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size> Medium &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size> Large &lt;/label>&lt;/p>
 &lt;/fieldset>
 &lt;fieldset>
  &lt;legend> Pizza Toppings &lt;/legend>
  &lt;p>&lt;label> &lt;input type=checkbox> Bacon &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox> Extra Cheese &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox> Onion &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox> Mushroom &lt;/label>&lt;/p>
 &lt;/fieldset>
 &lt;p>&lt;label>Preferred delivery time: &lt;input type=time min="11:00" max="21:00" step="900">&lt;/label>&lt;/p>
 &lt;p>&lt;label>Delivery instructions: &lt;textarea>&lt;/textarea>&lt;/label>&lt;/p>
<strong> &lt;p>&lt;button>Submit order&lt;/button>&lt;/p></strong>
&lt;/form></code></pre>


  <h5>Implementing the server-side processing for a form</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>The exact details for writing a server-side processor are out of scope for this specification.
  For the purposes of this introduction, we will assume that the script at <code
  data-x="">https://pizza.example.com/order.cgi</code> is configured to accept submissions using the
  <code data-x="attr-fs-enctype-urlencoded">application/x-www-form-urlencoded</code> format,
  expecting the following parameters sent in an HTTP POST body:</p>

  <dl>
   <dt><code data-x="">custname</code></dt>
   <dd>Customer's name</dd>

   <dt><code data-x="">custtel</code></dt>
   <dd>Customer's telephone number</dd>

   <dt><code data-x="">custemail</code></dt>
   <dd>Customer's email address</dd>

   <dt><code data-x="">size</code></dt>
   <dd>The pizza size, either <code data-x="">small</code>, <code data-x="">medium</code>, or <code data-x="">large</code></dd>

   <dt><code data-x="">topping</code></dt>
   <dd>A topping, specified once for each selected topping, with the allowed values being <code data-x="">bacon</code>, <code data-x="">cheese</code>, <code data-x="">onion</code>, and <code data-x="">mushroom</code></dd>

   <dt><code data-x="">delivery</code></dt>
   <dd>The requested delivery time</dd>

   <dt><code data-x="">comments</code></dt>
   <dd>The delivery instructions</dd>
  </dl>


  <h5>Configuring a form to communicate with a server</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>Form submissions are exposed to servers in a variety of ways, most commonly as HTTP GET or POST
  requests. To specify the exact method used, the <code data-x="attr-fs-method">method</code>
  attribute is specified on the <code>form</code> element. This doesn't specify how the form data is
  encoded, though; to specify that, you use the <code data-x="attr-fs-enctype">enctype</code>
  attribute. You also have to specify the <span>URL</span> of the service that will handle the
  submitted data, using the <code data-x="attr-fs-action">action</code> attribute.</p>

  <p>For each form control you want submitted, you then have to give a name that will be used to
  refer to the data in the submission. We already specified the name for the group of radio buttons;
  the same attribute (<code data-x="attr-fe-name">name</code>) also specifies the submission name.
  Radio buttons can be distinguished from each other in the submission by giving them different
  values, using the <code data-x="attr-input-value">value</code> attribute.</p>

  <p>Multiple controls can have the same name; for example, here we give all the checkboxes the same
  name, and the server distinguishes which checkbox was checked by seeing which values are submitted
  with that name &mdash; like the radio buttons, they are also given unique values with the <code
  data-x="attr-input-value">value</code> attribute.</p>

  <p>Given the settings in the previous section, this all becomes:</p>

  <pre><code class="html">&lt;form<strong> method="post"
      enctype="application/x-www-form-urlencoded"
      action="https://pizza.example.com/order.cgi"</strong>>
 &lt;p>&lt;label>Customer name: &lt;input<strong> name="custname"</strong>>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Telephone: &lt;input type=tel<strong> name="custtel"</strong>>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Email address: &lt;input type=email<strong> name="custemail"</strong>>&lt;/label>&lt;/p>
 &lt;fieldset>
  &lt;legend> Pizza Size &lt;/legend>
  &lt;p>&lt;label> &lt;input type=radio name=size<strong> value="small"</strong>> Small &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size<strong> value="medium"</strong>> Medium &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size<strong> value="large"</strong>> Large &lt;/label>&lt;/p>
 &lt;/fieldset>
 &lt;fieldset>
  &lt;legend> Pizza Toppings &lt;/legend>
  &lt;p>&lt;label> &lt;input type=checkbox<strong> name="topping" value="bacon"</strong>> Bacon &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox<strong> name="topping" value="cheese"</strong>> Extra Cheese &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox<strong> name="topping" value="onion"</strong>> Onion &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox<strong> name="topping" value="mushroom"</strong>> Mushroom &lt;/label>&lt;/p>
 &lt;/fieldset>
 &lt;p>&lt;label>Preferred delivery time: &lt;input type=time min="11:00" max="21:00" step="900"<strong> name="delivery"</strong>>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Delivery instructions: &lt;textarea<strong> name="comments"</strong>>&lt;/textarea>&lt;/label>&lt;/p>
 &lt;p>&lt;button>Submit order&lt;/button>&lt;/p>
&lt;/form></code></pre>

  <p class="note">There is no particular significance to the way some of the attributes have their
  values quoted and others don't. The HTML syntax allows a variety of equally valid ways to specify
  attributes, as discussed <span data-x="syntax-attributes">in the syntax section</span>.</p>

  <p>For example, if the customer entered "Denise Lawrence" as their name, "555-321-8642" as their
  telephone number, did not specify an email address, asked for a medium-sized pizza, selected the
  Extra Cheese and Mushroom toppings, entered a delivery time of 7pm, and left the delivery
  instructions text control blank, the user agent would submit the following to the online web
  service:</p>

  <pre><code class="html">custname=Denise+Lawrence&amp;custtel=555-321-8642&amp;custemail=&amp;size=medium&amp;topping=cheese&amp;topping=mushroom&amp;delivery=19%3A00&amp;comments=</code></pre>


  <h5>Client-side form validation</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>Forms can be annotated in such a way that the user agent will check the user's input before the
  form is submitted. The server still has to verify the input is valid (since hostile users can
  easily bypass the form validation), but it allows the user to avoid the wait incurred by having
  the server be the sole checker of the user's input.</p>

  <p>The simplest annotation is the <code data-x="attr-input-required">required</code> attribute,
  which can be specified on <code>input</code> elements to indicate that the form is not to be
  submitted until a value is given. By adding this attribute to the customer name, pizza size, and
  delivery time fields, we allow the user agent to notify the user when the user submits the form
  without filling in those fields:</p>

  <pre><code class="html">&lt;form method="post"
      enctype="application/x-www-form-urlencoded"
      action="https://pizza.example.com/order.cgi">
 &lt;p>&lt;label>Customer name: &lt;input name="custname"<strong> required</strong>>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Telephone: &lt;input type=tel name="custtel">&lt;/label>&lt;/p>
 &lt;p>&lt;label>Email address: &lt;input type=email name="custemail">&lt;/label>&lt;/p>
 &lt;fieldset>
  &lt;legend> Pizza Size &lt;/legend>
  &lt;p>&lt;label> &lt;input type=radio name=size<strong> required</strong> value="small"> Small &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size<strong> required</strong> value="medium"> Medium &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size<strong> required</strong> value="large"> Large &lt;/label>&lt;/p>
 &lt;/fieldset>
 &lt;fieldset>
  &lt;legend> Pizza Toppings &lt;/legend>
  &lt;p>&lt;label> &lt;input type=checkbox name="topping" value="bacon"> Bacon &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox name="topping" value="cheese"> Extra Cheese &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox name="topping" value="onion"> Onion &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox name="topping" value="mushroom"> Mushroom &lt;/label>&lt;/p>
 &lt;/fieldset>
 &lt;p>&lt;label>Preferred delivery time: &lt;input type=time min="11:00" max="21:00" step="900" name="delivery"<strong> required</strong>>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Delivery instructions: &lt;textarea name="comments">&lt;/textarea>&lt;/label>&lt;/p>
 &lt;p>&lt;button>Submit order&lt;/button>&lt;/p>
&lt;/form></code></pre>

  <p>It is also possible to limit the length of the input, using the <code
  data-x="attr-fe-maxlength">maxlength</code> attribute. By adding this to the <code>textarea</code>
  element, we can limit users to 1000 characters, preventing them from writing huge essays to the
  busy delivery drivers instead of staying focused and to the point:</p>

  <pre><code class="html">&lt;form method="post"
      enctype="application/x-www-form-urlencoded"
      action="https://pizza.example.com/order.cgi">
 &lt;p>&lt;label>Customer name: &lt;input name="custname" required>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Telephone: &lt;input type=tel name="custtel">&lt;/label>&lt;/p>
 &lt;p>&lt;label>Email address: &lt;input type=email name="custemail">&lt;/label>&lt;/p>
 &lt;fieldset>
  &lt;legend> Pizza Size &lt;/legend>
  &lt;p>&lt;label> &lt;input type=radio name=size required value="small"> Small &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size required value="medium"> Medium &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size required value="large"> Large &lt;/label>&lt;/p>
 &lt;/fieldset>
 &lt;fieldset>
  &lt;legend> Pizza Toppings &lt;/legend>
  &lt;p>&lt;label> &lt;input type=checkbox name="topping" value="bacon"> Bacon &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox name="topping" value="cheese"> Extra Cheese &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox name="topping" value="onion"> Onion &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox name="topping" value="mushroom"> Mushroom &lt;/label>&lt;/p>
 &lt;/fieldset>
 &lt;p>&lt;label>Preferred delivery time: &lt;input type=time min="11:00" max="21:00" step="900" name="delivery" required>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Delivery instructions: &lt;textarea name="comments"<strong> maxlength=1000</strong>>&lt;/textarea>&lt;/label>&lt;/p>
 &lt;p>&lt;button>Submit order&lt;/button>&lt;/p>
&lt;/form></code></pre>

  <p class="note">When a form is submitted, <code data-x="event-invalid">invalid</code> events are
  fired at each form control that is invalid. This can be useful for displaying a summary of the
  problems with the form, since typically the browser itself will only report one problem at a
  time.</p>



  <h5>Enabling client-side automatic filling of form controls</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>Some browsers attempt to aid the user by automatically filling form controls rather than having
  the user reenter their information each time. For example, a field asking for the user's telephone
  number can be automatically filled with the user's phone number.</p>

  <p>To help the user agent with this, the <code data-x="attr-fe-autocomplete">autocomplete</code>
  attribute can be used to describe the field's purpose. In the case of this form, we have three
  fields that can be usefully annotated in this way: the information about who the pizza is to be
  delivered to. Adding this information looks like this:</p>

  <pre><code class="html">&lt;form method="post"
      enctype="application/x-www-form-urlencoded"
      action="https://pizza.example.com/order.cgi">
 &lt;p>&lt;label>Customer name: &lt;input name="custname" required <strong>autocomplete="shipping name"</strong>>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Telephone: &lt;input type=tel name="custtel" <strong>autocomplete="shipping tel"</strong>>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Email address: &lt;input type=email name="custemail" <strong>autocomplete="shipping email"</strong>>&lt;/label>&lt;/p>
 &lt;fieldset>
  &lt;legend> Pizza Size &lt;/legend>
  &lt;p>&lt;label> &lt;input type=radio name=size required value="small"> Small &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size required value="medium"> Medium &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size required value="large"> Large &lt;/label>&lt;/p>
 &lt;/fieldset>
 &lt;fieldset>
  &lt;legend> Pizza Toppings &lt;/legend>
  &lt;p>&lt;label> &lt;input type=checkbox name="topping" value="bacon"> Bacon &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox name="topping" value="cheese"> Extra Cheese &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox name="topping" value="onion"> Onion &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox name="topping" value="mushroom"> Mushroom &lt;/label>&lt;/p>
 &lt;/fieldset>
 &lt;p>&lt;label>Preferred delivery time: &lt;input type=time min="11:00" max="21:00" step="900" name="delivery" required>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Delivery instructions: &lt;textarea name="comments" maxlength=1000>&lt;/textarea>&lt;/label>&lt;/p>
 &lt;p>&lt;button>Submit order&lt;/button>&lt;/p>
&lt;/form></code></pre>


  <h5>Improving the user experience on mobile devices</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>Some devices, in particular those with virtual keyboards can provide the user with multiple
  input modalities. For example, when typing in a credit card number the user may wish to only see
  keys for digits 0-9, while when typing in their name they may wish to see a form field that by
  default capitalizes each word.</p>

  <p>Using the <code data-x="attr-inputmode">inputmode</code> attribute we can select appropriate
  input modalities:</p>

  <pre><code class="html">&lt;form method="post"
      enctype="application/x-www-form-urlencoded"
      action="https://pizza.example.com/order.cgi">
 &lt;p>&lt;label>Customer name: &lt;input name="custname" required autocomplete="shipping name">&lt;/label>&lt;/p>
 &lt;p>&lt;label>Telephone: &lt;input type=tel name="custtel" autocomplete="shipping tel">&lt;/label>&lt;/p>
 &lt;p>&lt;label>Buzzer code: &lt;input name="custbuzz" <strong>inputmode="numeric"</strong>>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Email address: &lt;input type=email name="custemail" autocomplete="shipping email">&lt;/label>&lt;/p>
 &lt;fieldset>
  &lt;legend> Pizza Size &lt;/legend>
  &lt;p>&lt;label> &lt;input type=radio name=size required value="small"> Small &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size required value="medium"> Medium &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=radio name=size required value="large"> Large &lt;/label>&lt;/p>
 &lt;/fieldset>
 &lt;fieldset>
  &lt;legend> Pizza Toppings &lt;/legend>
  &lt;p>&lt;label> &lt;input type=checkbox name="topping" value="bacon"> Bacon &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox name="topping" value="cheese"> Extra Cheese &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox name="topping" value="onion"> Onion &lt;/label>&lt;/p>
  &lt;p>&lt;label> &lt;input type=checkbox name="topping" value="mushroom"> Mushroom &lt;/label>&lt;/p>
 &lt;/fieldset>
 &lt;p>&lt;label>Preferred delivery time: &lt;input type=time min="11:00" max="21:00" step="900" name="delivery" required>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Delivery instructions: &lt;textarea name="comments" maxlength=1000>&lt;/textarea>&lt;/label>&lt;/p>
 &lt;p>&lt;button>Submit order&lt;/button>&lt;/p>
&lt;/form></code></pre>



  <h5>The difference between the field type, the autofill field name, and the input modality</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>The <code data-x="attr-input-type">type</code>, <code
  data-x="attr-fe-autocomplete">autocomplete</code>, and <code
  data-x="attr-inputmode">inputmode</code> attributes can seem confusingly similar. For instance,
  in all three cases, the string "<code data-x="">email</code>" is a valid value. This section
  attempts to illustrate the difference between the three attributes and provides advice suggesting
  how to use them.</p>

  <p>The <code data-x="attr-input-type">type</code> attribute on <code>input</code> elements decides
  what kind of control the user agent will use to expose the field. Choosing between different
  values of this attribute is the same choice as choosing whether to use an <code>input</code>
  element, a <code>textarea</code> element, a <code>select</code> element, etc.</p>

  <p>The <code data-x="attr-fe-autocomplete">autocomplete</code> attribute, in contrast, describes
  what the value that the user will enter actually represents. Choosing between different values of
  this attribute is the same choice as choosing what the label for the element will be.</p>

  <p>First, consider telephone numbers. If a page is asking for a telephone number from the user,
  the right form control to use is <code data-x="attr-input-type-tel">&lt;input type=tel></code>.
  However, which <code data-x="attr-fe-autocomplete">autocomplete</code> value to use depends on
  which phone number the page is asking for, whether they expect a telephone number in the
  international format or just the local format, and so forth.</p>

  <p>For example, a page that forms part of a checkout process on an e-commerce site for a customer
  buying a gift to be shipped to a friend might need both the buyer's telephone number (in case of
  payment issues) and the friend's telephone number (in case of delivery issues). If the site
  expects international phone numbers (with the country code prefix), this could thus look like
  this:</p>

  <pre><code class="html">&lt;p>&lt;label>Your phone number: &lt;input type=tel name=custtel autocomplete="billing tel">&lt;/label>
&lt;p>&lt;label>Recipient's phone number: &lt;input type=tel name=shiptel autocomplete="shipping tel">&lt;/label>
&lt;p>Please enter complete phone numbers including the country code prefix, as in "+1 555 123 4567".</code></pre>

  <p>But if the site only supports British customers and recipients, it might instead look like this
  (notice the use of <code data-x="attr-fe-autocomplete-tel-national">tel-national</code> rather than
  <code data-x="attr-fe-autocomplete-tel">tel</code>):</p>

  <pre><code class="html">&lt;p>&lt;label>Your phone number: &lt;input type=tel name=custtel autocomplete="billing tel-national">&lt;/label>
&lt;p>&lt;label>Recipient's phone number: &lt;input type=tel name=shiptel autocomplete="shipping tel-national">&lt;/label>
&lt;p>Please enter complete UK phone numbers, as in "(01632) 960 123".</code></pre>

  <p>Now, consider a person's preferred languages. The right <code
  data-x="attr-fe-autocomplete">autocomplete</code> value is <code
  data-x="attr-fe-autocomplete-language">language</code>. However, there could be a number of
  different form controls used for the purpose: a text control (<code
  data-x="attr-input-type-text">&lt;input type=text></code>), a drop-down list (<code
  data-x="select">&lt;select></code>), radio buttons (<code data-x="attr-input-type-radio">&lt;input
  type=radio></code>), etc. It only depends on what kind of interface is desired.</p>

  <p>Finally, consider names. If a page just wants one name from the user, then the relevant control
  is <code data-x="attr-input-type-text">&lt;input type=text></code>. If the page is asking for the
  user's full name, then the relevant <code data-x="attr-fe-autocomplete">autocomplete</code> value
  is <code data-x="attr-fe-autocomplete-name">name</code>.

  <pre><code class="html">&lt;p>&lt;label>Japanese name: &lt;input name="j" type="text" autocomplete="section-jp name">&lt;/label>
&lt;label>Romanized name: &lt;input name="e" type="text" autocomplete="section-en name">&lt;/label></code></pre>

  <p>In this example, the "<code data-x="attr-fe-autocomplete-section">section-*</code>" keywords in
  the <code data-x="attr-fe-autocomplete">autocomplete</code> attributes' values tell the user agent
  that the two fields expect <em>different</em> names. Without them, the user agent could
  automatically fill the second field with the value given in the first field when the user gave a
  value to the first field.</p>

  <p class="note">The "<code data-x="">-jp</code>" and "<code data-x="">-en</code>" parts of the
  keywords are opaque to the user agent; the user agent cannot guess, from those, that the two names
  are expected to be in Japanese and English respectively.</p>

  <p>Separate from the choices regarding <code data-x="attr-input-type">type</code> and <code
  data-x="attr-fe-autocomplete">autocomplete</code>, the <code
  data-x="attr-inputmode">inputmode</code> attribute decides what kind of input modality (e.g.,
  virtual keyboard) to use, when the control is a text control.</p>

  <p>Consider credit card numbers. The appropriate input type is <em>not</em> <code
  data-x="attr-input-type-number">&lt;input type=number></code>, <a
  href="#when-number-is-not-appropriate">as explained below</a>; it is instead <code
  data-x="attr-input-type-text">&lt;input type=text></code>. To encourage the user agent to use a
  numeric input modality anyway (e.g., a virtual keyboard displaying only digits), the page would
  use</p>

  <pre><code class="html">&lt;p>&lt;label>Credit card number:
                &lt;input name="cc" type="text" inputmode="numeric" pattern="[0-9]{8,19}" autocomplete="cc-number">
&lt;/label>&lt;/p></code></pre>



  <h5 id="input-author-notes">Date, time, and number formats</h5>
  <!-- ID referenced from other parts of the spec -->

  <!-- NON-NORMATIVE SECTION -->

  <p>In this pizza delivery example, the times are specified in the format "HH:MM": two digits for
  the hour, in 24-hour format, and two digits for the time. (Seconds could also be specified, though
  they are not necessary in this example.)</p>

  <p>In some locales, however, times are often expressed differently when presented to users. For
  example, in the United States, it is still common to use the 12-hour clock with an am/pm
  indicator, as in "2pm". In France, it is common to separate the hours from the minutes using an
  "h" character, as in "14h00".</p>

  <p>Similar issues exist with dates, with the added complication that even the order of the
  components is not always consistent &mdash; for example, in Cyprus the first of February 2003
  would typically be written "1/2/03", while that same date in Japan would typically be written as
  "2003&#x5E74;02&#x6708;01&#x65E5;" &mdash; and even with numbers, where locales differ, for
  example, in what punctuation is used as the decimal separator and the thousands separator.</p>

  <p>It is therefore important to distinguish the time, date, and number formats used in HTML and in
  form submissions, which are always the formats defined in this specification (and based on the
  well-established ISO 8601 standard for computer-readable date and time formats), from the time,
  date, and number formats presented to the user by the browser and accepted as input from the user
  by the browser.</p>

  <p>The format used "on the wire", i.e., in HTML markup and in form submissions, is intended to be
  computer-readable and consistent irrespective of the user's locale. Dates, for instance, are
  always written in the format "YYYY-MM-DD", as in "2003-02-01". While some users might see this
  format, others might see it as "01.02.2003" or "February 1, 2003".</p>

  <p>The time, date, or number given by the page in the wire format is then translated to the user's
  preferred presentation (based on user preferences or on the locale of the page itself), before
  being displayed to the user. Similarly, after the user inputs a time, date, or number using their
  preferred format, the user agent converts it back to the wire format before putting it in the DOM
  or submitting it.</p>

  <p>This allows scripts in pages and on servers to process times, dates, and numbers in a
  consistent manner without needing to support dozens of different formats, while still supporting
  the users' needs.</p>

  <div w-nodev>
  <p class="note">See also the <a
  href="#input-impl-notes">implementation notes</a> regarding
  localization of form controls.</p>
  </div>


  <h4>Categories</h4>

  <p>Mostly for historical reasons, elements in this section fall into several overlapping (but
  subtly different) categories in addition to the usual ones like <span>flow content</span>,
  <span>phrasing content</span>, and <span>interactive content</span>.</p>

  <p>A number of the elements are <dfn data-x="form-associated element">form-associated
  elements</dfn>, which means they can have a <span>form owner</span>.

  <!-- when updating this also update the category index -->
  <ul class="brief category-list">
   <li><code>button</code></li>
   <li><code>fieldset</code></li>
   <li><code>input</code></li>
   <li><code>object</code></li>
   <li><code>output</code></li>
   <li><code>select</code></li>
   <li><code>textarea</code></li>
   <li><code>img</code></li>
   <li><span data-x="form-associated custom element">form-associated custom elements</span></li>
  </ul>

  <p>The <span data-x="form-associated element">form-associated elements</span> fall into several
  subcategories:</p>

  <dl>
   <dt><dfn data-x="category-listed">Listed elements</dfn></dt>

   <dd>
    <p>Denotes elements that are listed in the <code
    data-x="dom-form-elements"><var>form</var>.elements</code> and <code
    data-x="dom-fieldset-elements"><var>fieldset</var>.elements</code> APIs. These elements also
    have a <code data-x="attr-fae-form">form</code> content attribute, and a matching <code
    data-x="dom-fae-form">form</code> IDL attribute, that allow authors to specify an explicit
    <span>form owner</span>.</p>

    <!-- when updating this also update the category index -->
    <ul class="brief category-list">
     <li><code>button</code></li>
     <li><code>fieldset</code></li>
     <li><code>input</code></li>
     <li><code>object</code></li>
     <li><code>output</code></li>
     <li><code>select</code></li>
     <li><code>textarea</code></li>
     <li><span data-x="form-associated custom element">form-associated custom elements</span></li>
    </ul>
   </dd>

   <dt><dfn data-x="category-submit">Submittable elements</dfn></dt>

   <dd>
    <p>Denotes elements that can be used for <span>constructing the entry list</span> when a
    <code>form</code> element is <span data-x="concept-form-submit">submitted</span>.</p>

    <!-- when updating this also update the category index -->
    <ul class="brief category-list">
     <li><code>button</code></li>
     <li><code>input</code></li>
     <li><code>select</code></li>
     <li><code>textarea</code></li>
     <li><span data-x="form-associated custom element">form-associated custom elements</span></li>
    </ul>

    <p>Some <span data-x="category-submit">submittable elements</span> can be, depending on their
    attributes, <dfn data-x="concept-button">buttons</dfn>. The prose below defines when an element
    is a button. Some buttons are specifically <dfn export data-lt="submit button"
    data-x="concept-submit-button">submit buttons</dfn>.</p>
   </dd>

   <dt><dfn data-x="category-reset">Resettable elements</dfn></dt>

   <dd>
    <p>Denotes elements that can be affected when a <code>form</code> element is <span
    data-x="concept-form-reset">reset</span>.</p>

    <!-- when updating this also update the category index -->
    <ul class="brief category-list">
     <li><code>input</code></li>
     <li><code>output</code></li>
     <li><code>select</code></li>
     <li><code>textarea</code></li>
     <li><span data-x="form-associated custom element">form-associated custom elements</span></li>
    </ul>
   </dd>

   <dt><dfn id="category-autocapitalize"
   data-x="category-autocapitalize-and-autocorrect">Autocapitalize-and-autocorrect-inheriting
   elements</dfn></dt>

   <dd>
    <p>Denotes elements that inherit the <code data-x="attr-autocapitalize">autocapitalize</code>
    and <code data-x="attr-autocorrect">autocorrect</code> attributes from their <span>form
    owner</span>.</p>

    <!-- when updating this also update the category index -->
    <ul class="brief category-list">
     <li><code>button</code></li>
     <li><code>fieldset</code></li>
     <li><code>input</code></li>
     <li><code>output</code></li>
     <li><code>select</code></li>
     <li><code>textarea</code></li>
    </ul>
   </dd>
  </dl>

  <p>Some elements, not all of them <span data-x="form-associated element">form-associated</span>,
  are categorized as <dfn data-x="category-label">labelable elements</dfn>. These are elements that
  can be associated with a <code>label</code> element.

  <!-- when updating this also update the category index -->
  <ul class="brief category-list">
   <li><code>button</code></li>
   <li><code>input</code> (if the <code data-x="attr-input-type">type</code> attribute is <em>not</em> in the <span data-x="attr-input-type-hidden">Hidden</span> state)</li>
   <li><code>meter</code></li>
   <li><code>output</code></li>
   <li><code>progress</code></li>
   <li><code>select</code></li>
   <li><code>textarea</code></li>
   <li><span data-x="form-associated custom element">form-associated custom elements</span></li>
  </ul>


  <h4>The <dfn element><code>form</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>, but with no <code>form</code> element descendants.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-form-accept-charset">accept-charset</code></dd>
   <dd><code data-x="attr-fs-action">action</code></dd>
   <dd><code data-x="attr-form-autocomplete">autocomplete</code></dd>
   <dd><code data-x="attr-fs-enctype">enctype</code></dd>
   <dd><code data-x="attr-fs-method">method</code></dd>
   <dd><code data-x="attr-form-name">name</code></dd>
   <dd><code data-x="attr-fs-novalidate">novalidate</code></dd>
   <dd><code data-x="attr-fs-target">target</code></dd>
   <dd><code data-x="attr-form-rel">rel</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-form">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-form">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window,
 <span>LegacyOverrideBuiltIns</span>,
 <span>LegacyUnenumerableNamedProperties</span>]
interface <dfn interface>HTMLFormElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-form-acceptCharset">acceptCharset</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-fs-action">action</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-form-autocomplete">autocomplete</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fs-enctype">enctype</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fs-encoding">encoding</span>;<!-- historical artefact -->
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fs-method">method</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-form-name">name</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-fs-noValidate">noValidate</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fs-target">target</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-form-rel">rel</span>;
  [SameObject, PutForwards=<span data-x="dom-DOMTokenList-value">value</span>] readonly attribute <span>DOMTokenList</span> <span data-x="dom-form-relList">relList</span>;

  [SameObject] readonly attribute <span>HTMLFormControlsCollection</span> <span data-x="dom-form-elements">elements</span>;
  readonly attribute unsigned long <span data-x="dom-form-length">length</span>;
  <a href="#dom-form-item">getter</a> <span>Element</span> (unsigned long index);
  <a href="#dom-form-nameditem">getter</a> (<span>RadioNodeList</span> or <span>Element</span>) (DOMString name);

  undefined <span data-x="dom-form-submit">submit</span>();
  undefined <span data-x="dom-form-requestSubmit">requestSubmit</span>(optional <span>HTMLElement</span>? submitter = null);
  [<span>CEReactions</span>] undefined <span data-x="dom-form-reset">reset</span>();
  boolean <span data-x="dom-form-checkValidity">checkValidity</span>();
  boolean <span data-x="dom-form-reportValidity">reportValidity</span>();
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLFormElement</code>.</dd>
  </dl>

  <p>The <code>form</code> element <span>represents</span> a <span>hyperlink</span> that can be
  manipulated through a collection of <span data-x="form-associated element">form-associated
  elements</span>, some of which can represent editable values that can be submitted to a server for
  processing.</p>

  <p>The <dfn element-attr for="form"><code
  data-x="attr-form-accept-charset">accept-charset</code></dfn> attribute gives the character
  encodings that are to be used for the submission. If specified, the value must be an <span>ASCII
  case-insensitive</span> match for "<code data-x="">UTF-8</code>". <ref>ENCODING</ref></p>

  <p>The <dfn element-attr for="form"><code data-x="attr-form-name">name</code></dfn> attribute
  represents the <code>form</code>'s name within the <code data-x="dom-document-forms">forms</code>
  collection. The value must not be the empty string, and the value must be unique amongst the
  <code>form</code> elements in the <code data-x="dom-document-forms">forms</code> collection that
  it is in, if any.</p>

  <p>The <dfn element-attr for="form"><code
  data-x="attr-form-autocomplete">autocomplete</code></dfn> attribute is an <span>enumerated
  attribute</span> with the following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="form/autocomplete"><code data-x="attr-form-autocomplete-on">on</code></dfn>
     <td><dfn data-x="attr-form-autocomplete-on-state">on</dfn>
     <td>Form controls will have their <span>autofill field name</span> set to "<code
     data-x="attr-fe-autocomplete-on">on</code>" by default.
    <tr>
     <td><dfn attr-value for="form/autocomplete"><code data-x="attr-form-autocomplete-off">off</code></dfn>
     <td><dfn data-x="attr-form-autocomplete-off-state">off</dfn>
     <td>Form controls will have their <span>autofill field name</span> set to "<code
     data-x="attr-fe-autocomplete-off">off</code>" by default.
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <span
  data-x="attr-form-autocomplete-on-state">on</span> state.</p>

  <p>The <code data-x="attr-fs-action">action</code>, <code data-x="attr-fs-enctype">enctype</code>,
  <code data-x="attr-fs-method">method</code>, <code data-x="attr-fs-novalidate">novalidate</code>,
  and <code data-x="attr-fs-target">target</code> attributes are <span>attributes for form
  submission</span>.</p>

  <p>The <dfn element-attr for="form"><code data-x="attr-form-rel">rel</code></dfn> attribute on
  <code>form</code> elements controls what kinds of links the elements create. The attribute's value
  must be a <span>unordered set of unique space-separated tokens</span>. The <a
  href="#linkTypes">allowed keywords and their meanings</a> are defined in an earlier section.</p>

  <p><code data-x="attr-form-rel">rel</code>'s <span data-x="concept-supported-tokens">supported
  tokens</span> are the keywords defined in <a href="#linkTypes">HTML link types</a> which are
  allowed on <code>form</code> elements, impact the processing model, and are supported by the user
  agent. The possible <span data-x="concept-supported-tokens">supported tokens</span> are <code
  data-x="rel-noreferrer">noreferrer</code>, <code data-x="rel-noopener">noopener</code>, and <code
  data-x="rel-opener">opener</code>. <code data-x="attr-form-rel">rel</code>'s <span
  data-x="concept-supported-tokens">supported tokens</span> must only include the tokens from this
  list that the user agent implements the processing model for.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>form</var>.<span subdfn data-x="dom-form-elements">elements</span></code></dt>

   <dd>
    <p>Returns an <code>HTMLFormControlsCollection</code> of the form controls in the form
    (excluding image buttons for historical reasons).</p>
   </dd>

   <dt><code data-x=""><var>form</var>.<span subdfn data-x="dom-form-length">length</span></code></dt>

   <dd>
    <p>Returns the number of form controls in the form (excluding image buttons for historical
    reasons).</p>
   </dd>

   <dt><code data-x=""><var>form</var>[<var>index</var>]</code></dt>

   <dd>
    <p>Returns the <var>index</var>th element in the form (excluding image buttons for historical
    reasons).</p>
   </dd>

   <dt><code data-x=""><var>form</var>[<var>name</var>]</code></dt>

   <dd>
    <p>Returns the form control (or, if there are several, a <code>RadioNodeList</code> of the form
    controls) in the form with the given <span data-x="concept-id">ID</span> or <code
    data-x="attr-fe-name">name</code> (excluding image buttons for historical reasons); or, if there
    are none, returns the <code>img</code> element with the given ID.</p>

    <p>Once an element has been referenced using a particular name, that name will continue being
    available as a way to reference that element in this method, even if the element's actual <span
    data-x="concept-id">ID</span> or <code data-x="attr-fe-name">name</code> changes, for as long as
    the element remains in the <span>tree</span>.</p>

    <p>If there are multiple matching items, then a <code>RadioNodeList</code> object containing all
    those elements is returned.</p>
   </dd>

   <dt><code data-x=""><var>form</var>.<span subdfn data-x="dom-form-submit">submit</span>()</code></dt>

   <dd>
    <p>Submits the form, bypassing <span data-x="interactively validate the constraints">interactive
    constraint validation</span> and without firing a <code data-x="event-submit">submit</code>
    event.</p>
   </dd>

   <dt><code data-x=""><var>form</var>.<span subdfn data-x="dom-form-requestSubmit">requestSubmit</span>([ <var>submitter</var> ])</code></dt>

   <dd>
    <p>Requests to submit the form. Unlike <code data-x="dom-form-submit">submit()</code>, this
    method includes <span data-x="interactively validate the constraints">interactive constraint
    validation</span> and firing a <code data-x="event-submit">submit</code> event, either of which
    can cancel submission.</p>

    <p>The <var>submitter</var> argument can be used to point to a specific <span
    data-x="concept-submit-button">submit button</span>, whose <code
    data-x="attr-fs-formaction">formaction</code>, <code
    data-x="attr-fs-formenctype">formenctype</code>, <code
    data-x="attr-fs-formmethod">formmethod</code>, <code
    data-x="attr-fs-formnovalidate">formnovalidate</code>, and <code
    data-x="attr-fs-formtarget">formtarget</code> attributes can impact submission. Additionally,
    the submitter will be included when <span>constructing the entry list</span> for submission;
    normally, buttons are excluded.</p>
   </dd>

   <dt><code data-x=""><var>form</var>.<span subdfn data-x="dom-form-reset">reset</span>()</code></dt>

   <dd><p>Resets the form.</p></dd>

   <dt><code data-x=""><var>form</var>.<span subdfn data-x="dom-form-checkValidity">checkValidity</span>()</code></dt>

   <dd><p>Returns true if the form's controls are all valid; otherwise, returns false.</p></dd>

   <dt><code data-x=""><var>form</var>.<span subdfn data-x="dom-form-reportValidity">reportValidity</span>()</code></dt>

   <dd>
    <p>Returns true if the form's controls are all valid; otherwise, returns false and informs the
    user.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLFormElement"><code
  data-x="dom-form-autocomplete">autocomplete</code></dfn> IDL attribute must <span>reflect</span>
  the content attribute of the same name, <span>limited to only known values</span>.</p>

  <p>The <dfn attribute for="HTMLFormElement"><code data-x="dom-form-name">name</code></dfn> and
  <dfn attribute for="HTMLFormElement"><code data-x="dom-form-rel">rel</code></dfn> IDL attributes
  must <span>reflect</span> the content attribute of the same name.</p>

  <p>The <dfn attribute for="HTMLFormElement"><code
  data-x="dom-form-acceptCharset">acceptCharset</code></dfn> IDL attribute must <span>reflect</span>
  the <code data-x="attr-form-accept-charset">accept-charset</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLFormElement"><code data-x="dom-form-relList">relList</code></dfn>
  IDL attribute must <span>reflect</span> the <code data-x="attr-form-rel">rel</code> content
  attribute.</p>

  <hr>

  <p>The <dfn attribute for="HTMLFormElement"><code data-x="dom-form-elements">elements</code></dfn>
  IDL attribute must return an <code>HTMLFormControlsCollection</code> rooted at the
  <code>form</code> element's <span>root</span>, whose filter matches <span
  data-x="category-listed">listed elements</span> whose <span>form owner</span> is the
  <code>form</code> element, with the exception of <code>input</code> elements whose <code
  data-x="attr-input-type">type</code> attribute is in the <span
  data-x="attr-input-type-image">Image Button</span> state, which must, for historical reasons, be
  excluded from this particular collection.</p>

  <p>The <dfn attribute for="HTMLFormElement"><code data-x="dom-form-length">length</code></dfn> IDL
  attribute must return the number of nodes <span data-x="represented by the
  collection">represented</span> by the <code data-x="dom-form-elements">elements</code>
  collection.</p>

  <p>The <span>supported property indices</span> at any instant are the indices supported by the
  object returned by the <code data-x="dom-form-elements">elements</code> attribute at that
  instant.</p>

  <p id="dom-form-item">To <span>determine the value of an indexed property</span> for a
  <code>form</code> element, the user agent must return the value returned by the <code
  data-x="dom-HTMLCollection-item">item</code> method on the <code
  data-x="dom-form-elements">elements</code> collection, when invoked with the given index as its
  argument.</p>

  <hr>

  <p>Each <code>form</code> element has a mapping of names to elements called the <dfn>past names
  map</dfn>. It is used to persist names of controls even when they change names.</p>

  <p>The <span>supported property names</span> consist of the names obtained from the following
  algorithm, in the order obtained from this algorithm:</p>

  <ol>
   <li><p>Let <var>sourced names</var> be an initially empty ordered list of tuples
   consisting of a string, an element, a source, where the source is either <i>id</i>, <i>name</i>,
   or <i>past</i>, and, if the source is <i>past</i>, an age.</p></li>

   <li>
    <p>For each <span data-x="category-listed">listed element</span> <var>candidate</var>
    whose <span>form owner</span> is the <code>form</code> element, with the exception of any
    <code>input</code> elements whose <code data-x="attr-input-type">type</code> attribute is in the
    <span data-x="attr-input-type-image">Image Button</span> state:</p>

    <ol>
     <li><p>If <var>candidate</var> has an <code data-x="attr-id">id</code> attribute, add
     an entry to <var>sourced names</var> with that <code data-x="attr-id">id</code>
     attribute's value as the string, <var>candidate</var> as the element, and <i>id</i> as
     the source.</p></li>

     <li><p>If <var>candidate</var> has a <code data-x="attr-fe-name">name</code> attribute,
     add an entry to <var>sourced names</var> with that <code
     data-x="attr-fe-name">name</code> attribute's value as the string, <var>candidate</var>
     as the element, and <i>name</i> as the source.</p></li>
    </ol>
   </li>

   <li>
    <p>For each <code>img</code> element <var>candidate</var> whose <span>form owner</span> is the
    <code>form</code> element:</p>

    <ol><!-- this is the same as the above list, except 'name' xrefs to something different -->

     <li><p>If <var>candidate</var> has an <code data-x="attr-id">id</code> attribute, add
     an entry to <var>sourced names</var> with that <code data-x="attr-id">id</code>
     attribute's value as the string, <var>candidate</var> as the element, and <i>id</i> as
     the source.</p></li>

     <li><p>If <var>candidate</var> has a <code data-x="attr-img-name">name</code> attribute,
     add an entry to <var>sourced names</var> with that <code
     data-x="attr-img-name">name</code> attribute's value as the string, <var>candidate</var>
     as the element, and <i>name</i> as the source.</p></li>
    </ol>
   </li>

   <li>
    <p>For each entry <var>past entry</var> in the <span>past names map</span>, add an entry
    to <var>sourced names</var> with the <var>past entry</var>'s name as the
    string, <var>past entry</var>'s element as the element, <i>past</i> as the source, and
    the length of time <var>past entry</var> has been in the <span>past names map</span> as
    the age.</p>
   </li>

   <li><p>Sort <var>sourced names</var> by <span>tree order</span> of the element entry of
   each tuple, sorting entries with the same element by putting entries whose source is <i>id</i>
   first, then entries whose source is <i>name</i>, and finally entries whose source is <i>past</i>,
   and sorting entries with the same element and source by their age, oldest first.</p></li>

   <li><p>Remove any entries in <var>sourced names</var> that have the empty string as
   their name.</p></li>

   <li><p>Remove any entries in <var>sourced names</var> that have the same name as an
   earlier entry in the map.</p></li>

   <li><p>Return the list of names from <var>sourced names</var>, maintaining their
   relative order.</p></li>
  </ol>

  <p id="dom-form-nameditem">To <span>determine the value of a named property</span> <var>name</var>
  for a <code>form</code> element, the user agent must run the following steps:</p>

  <ol>
   <li><p>Let <var>candidates</var> be a <span>live</span> <code>RadioNodeList</code> object
   containing all the <span data-x="category-listed">listed elements</span>, whose <span>form
   owner</span> is the <code>form</code> element, that have either an <code
   data-x="attr-id">id</code> attribute or a <code data-x="attr-fe-name">name</code> attribute equal
   to <var>name</var>, with the exception of <code>input</code> elements whose <code
   data-x="attr-input-type">type</code> attribute is in the <span
   data-x="attr-input-type-image">Image Button</span> state, in <span>tree order</span>.</p></li>

   <!-- we return RadioNodeList here for consistency -->
   <li><p>If <var>candidates</var> is empty, let <var>candidates</var> be a <span>live</span>
   <code>RadioNodeList</code> object containing all the <code>img</code> elements, whose <span>form
   owner</span> is the <code>form</code> element, that have either an <code
   data-x="attr-id">id</code> attribute or a <code data-x="attr-img-name">name</code> attribute
   equal to <var>name</var>, in <span>tree order</span>.</p></li>

   <li><p>If <var>candidates</var> is empty, <var>name</var> is the name of one of
   the entries in the <code>form</code> element's <span>past names map</span>: return the object
   associated with <var>name</var> in that map.</p></li>

   <li><p>If <var>candidates</var> contains more than one node, return <var>candidates</var>.</p></li>

   <li><p>Otherwise, <var>candidates</var> contains exactly one node. Add a mapping from
   <var>name</var> to the node in <var>candidates</var> in the <code>form</code>
   element's <span>past names map</span>, replacing the previous entry with the same name, if
   any.</p></li>

   <li><p>Return the node in <var>candidates</var>.</p></li>
  </ol>

  <p>If an element listed in a <code>form</code> element's <span>past names map</span> changes
  <span>form owner</span>, then its entries must be removed from that map.</p>

  <!--
    This ridiculous setup is intended to do as much of the right thing while still supporting code
    written to work in IE7. IE versions prior to IE8 do not update the names on the <form> element
    collection to match new names when elements are renamed, and there are enough pages that rename
    elements and then access them by their old name that we have to support this.

    But we still want to expose them using the new names, so as far as possible we pretend the
    legacy names aren't there except if there's no other element actually named that way.

    Removing the element did remove the legacy name in IE7:
    https://software.hixie.ch/utilities/js/live-dom-viewer/?%3C!DOCTYPE%20html%3E...%3Cform%20action%3D%22%2F%22%3E%3Cinput%20name%3Dsubmit%3E%3C%2Fform%3E%0A%3Cscript%3E%0A%20w(document.forms%5B0%5D.submit)%3B%0A%20w(document.forms%5B0%5D.removeChild(document.getElementsByTagName('input')%5B0%5D))%3B%0A%20w(document.forms%5B0%5D.length)%3B%0A%20try%20%7B%20document.forms%5B0%5D.submit()%3B%20%7D%20catch%20(e)%20%20%7B%20w(e.message)%20%7D%0A%3C%2Fscript%3E

    There's no interop on what happens when the name was originally a duplicate name, so we don't
    persist such accesses - at the time of writing, Safari returned the first element, Firefox
    returned null (as we do), and IE7 returned the original collection:
    https://software.hixie.ch/utilities/js/live-dom-viewer/?%3C!DOCTYPE%20html%3E...%3Cform%20action%3D%22%2F%22%3E%3Cinput%20name%3Da%20id%3Dfirst%3E%3Cinput%20name%3Da%20id%3Dsecond%3E%3C%2Fform%3E%0A%3Cscript%3E%0A%20w%28document.forms[0].a%29%3B%0A%20document.getElementsByTagName%28%27input%27%29[0].name%20%3D%20%27b%27%3B%0A%20document.getElementsByTagName%28%27input%27%29[1].name%20%3D%20%27b%27%3B%0A%20w%28document.forms[0].length%29%3B%0A%20w%28document.forms[0].a.id%29%3B%0A%3C%2Fscript%3E

    In addition, the <img> fallback nonsense is similarly here for legacy reasons. As is the
    exclusion of <input type=image>.

    Also, check these out:
    https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1220
    https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2388

  -->

  <hr>

  <p>The <dfn method for="HTMLFormElement"><code data-x="dom-form-submit">submit()</code></dfn>
  method steps are to <span data-x="concept-form-submit">submit</span> <span>this</span> from
  <span>this</span>, with <i data-x="submit-subbmitted-from-method">submitted from <code
  data-x="dom-form-submit">submit()</code> method</i> set to true.</p>

  <p>The <dfn method for="HTMLFormElement"><code
  data-x="dom-form-requestSubmit">requestSubmit(<var>submitter</var>)</code></dfn> method, when
  invoked, must run the following steps:</p>

  <ol>
   <li>
    <p>If <var>submitter</var> is not null, then:</p>

    <ol>
     <li><p>If <var>submitter</var> is not a <span data-x="concept-submit-button">submit
     button</span>, then throw a <code>TypeError</code>.</p></li>

     <li><p>If <var>submitter</var>'s <span>form owner</span> is not this <code>form</code> element,
     then throw a <span>"<code>NotFoundError</code>"</span> <code>DOMException</code>.</p></li>
    </ol>
   </li>

   <li><p>Otherwise, set <var>submitter</var> to this <code>form</code> element.</p></li>

   <li><p><span data-x="concept-form-submit">Submit</span> this <code>form</code> element, from
   <var>submitter</var>.</p></li>
  </ol>

  <p>The <dfn method for="HTMLFormElement"><code data-x="dom-form-reset">reset()</code></dfn>
  method, when invoked, must run the following steps:</p>

  <ol>
   <li><p>If the <code>form</code> element is marked as <i data-x="locked for reset">locked for reset</i>, then return.</p></li>

   <li><p>Mark the <code>form</code> element as <dfn>locked for reset</dfn>.</p></li>

   <li><p><span data-x="concept-form-reset">Reset</span> the <code>form</code> element.</p></li>

   <li><p>Unmark the <code>form</code> element as <i data-x="locked for reset">locked for reset</i>.</p></li>
  </ol>

  <p>If the <dfn method for="HTMLFormElement"><code
  data-x="dom-form-checkValidity">checkValidity()</code></dfn> method is invoked, the user agent
  must <span>statically validate the constraints</span> of the <code>form</code> element, and return
  true if the constraint validation return a <i>positive</i> result, and false if it returned a
  <i>negative</i> result.</p>

  <p>If the <dfn method for="HTMLFormElement"><code
  data-x="dom-form-reportValidity">reportValidity()</code></dfn> method is invoked, the user agent
  must <span>interactively validate the constraints</span> of the <code>form</code> element, and
  return true if the constraint validation return a <i>positive</i> result, and false if it returned
  a <i>negative</i> result.</p>

  </div>

  <div class="example">

   <p>This example shows two search forms:</p>

   <pre><code class="html">&lt;form action="https://www.google.com/search" method="get">
 &lt;label>Google: &lt;input type="search" name="q">&lt;/label> &lt;input type="submit" value="Search...">
&lt;/form>
&lt;form action="https://www.bing.com/search" method="get">
 &lt;label>Bing: &lt;input type="search" name="q">&lt;/label> &lt;input type="submit" value="Search...">
&lt;/form></code></pre>

  </div>



  <h4>The <dfn element><code>label</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Interactive content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>, but with no descendant <span data-x="category-label">labelable elements</span> unless it is the element's <span>labeled control</span>, and no descendant <code>label</code> elements.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-label-for">for</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-label">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-label">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLLabelElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  readonly attribute <span>HTMLFormElement</span>? <span data-x="dom-label-form">form</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-label-htmlFor">htmlFor</span>;
  readonly attribute <span>HTMLElement</span>? <span data-x="dom-label-control">control</span>;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLLabelElement</code>.</dd>
  </dl>

  <p>The <code>label</code> element <span>represents</span> a caption in a user interface. The
  caption can be associated with a specific form control<span w-nodev>, known as the
  <code>label</code> element's <dfn>labeled control</dfn></span>, either using the <code
  data-x="attr-label-for">for</code> attribute, or by putting the form control inside the
  <code>label</code> element itself.</p>

  <div w-nodev>

  <p>Except where otherwise specified by the following rules, a <code>label</code> element has no
  <span>labeled control</span>.</p>

  </div>

  <p>The <dfn element-attr for="label"><code data-x="attr-label-for">for</code></dfn> attribute may
  be specified to indicate a form control with which the caption is to be associated. If the
  attribute is specified, the attribute's value must be the <span data-x="concept-id">ID</span> of a
  <span data-x="category-label">labelable element</span> in the same <span>tree</span> as the
  <code>label</code> element. <span w-nodev>If the attribute is specified and there is an element in
  the <span>tree</span> whose <span data-x="concept-id">ID</span> is equal to the value of the <code
  data-x="attr-label-for">for</code> attribute, and the first such element in <span>tree
  order</span> is a <span data-x="category-label">labelable element</span>, then that element is the
  <code>label</code> element's <span>labeled control</span>.</span></p>

  <div w-nodev>

  <p>If the <code data-x="attr-label-for">for</code> attribute is not specified, but the
  <code>label</code> element has a <span data-x="category-label">labelable element</span> descendant,
  then the first such descendant in <span>tree order</span> is the <code>label</code> element's
  <span>labeled control</span>.</p>

  <p>The <code>label</code> element's exact default presentation and behavior, in particular what
  its <span>activation behavior</span> might be, if anything, should match the platform's label
  behavior. The <span>activation behavior</span> of a <code>label</code> element for events targeted
  at <span>interactive content</span> descendants of a <code>label</code> element, and any
  descendants of those <span>interactive content</span> descendants, must be to do nothing.</p>

  <p class="note"><span data-x="form-associated custom element">Form-associated custom
  elements</span> are <span data-x="category-label">labelable elements</span>, so for user agents
  where the <code>label</code> element's <span>activation behavior</span> impacts the <span>labeled
  control</span>, both built-in and custom elements will be impacted.</p>

  <!-- activation behavior need not be dependent on whether the
  labeled control is being rendered:
   https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=371
  -->

  <div class="example">

   <p>For example, on platforms where clicking a label activates the form control, clicking the
   <code>label</code> in the following snippet could trigger the user agent to <span>fire a <code
   data-x="event-click">click</code> event</span> at the <code>input</code> element, as if the
   element itself had been triggered by the user:</p>

   <pre><code class="html">&lt;label>&lt;input type=checkbox name=lost> Lost&lt;/label></code></pre>

   <p>Similarly, assuming <code data-x="">my-checkbox</code> was declared as a
   <span>form-associated custom element</span> (like in <a href="#custom-elements-face-example">this
   example</a>), then the code</p>

   <pre><code class="html">&lt;label>&lt;my-checkbox name=lost>&lt;/my-checkbox> Lost&lt;/label></code></pre>

   <p>would have the same behavior, <span data-x="fire a click event">firing a <code
   data-x="event-click">click</code> event</span> at the <code data-x="">my-checkbox</code>
   element.</p>

   <p>On other platforms, the behavior in both cases might be just to focus the control, or to do
   nothing.</p>
  </div>

  </div>

  <div class="example">

   <p>The following example shows three form controls each with a label, two of which have small
   text showing the right format for users to use.</p>

   <pre><code class="html">&lt;p>&lt;label>Full name: &lt;input name=fn> &lt;small>Format: First Last&lt;/small>&lt;/label>&lt;/p>
&lt;p>&lt;label>Age: &lt;input name=age type=number min=0>&lt;/label>&lt;/p>
&lt;p>&lt;label>Post code: &lt;input name=pc> &lt;small>Format: AB12 3CD&lt;/small>&lt;/label>&lt;/p></code></pre>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>label</var>.<span subdfn data-x="dom-label-control">control</span></code></dt>

   <dd><p>Returns the form control that is associated with this element.</p></dd>

   <dt><code data-x=""><var>label</var>.<span subdfn data-x="dom-label-form">form</span></code></dt>

   <dd>
    <p>Returns the <span>form owner</span> of the form control that is associated with this
    element.</p>

    <p>Returns null if there isn't one.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLLabelElement"><code data-x="dom-label-htmlFor">htmlFor</code></dfn>
  IDL attribute must <span>reflect</span> the <code data-x="attr-label-for">for</code> content
  attribute.</p>

  <p>The <dfn attribute for="HTMLLabelElement"><code data-x="dom-label-control">control</code></dfn>
  IDL attribute must return the <code>label</code> element's <span>labeled control</span>, if any,
  or null if there isn't one.</p>

  <p>The <dfn attribute for="HTMLLabelElement"><code data-x="dom-label-form">form</code></dfn> IDL
  attribute must run the following steps:</p>

  <ol>
   <li><p>If the <code>label</code> element has no <span>labeled control</span>, then return
   null.</p></li>

   <li><p>If the <code>label</code> element's <span>labeled control</span> is not a
   <span>form-associated element</span>, then return null.</p></li>

   <li><p>Return the <code>label</code> element's <span>labeled control</span>'s <span>form
   owner</span> (which can still be null).</p></li>
  </ol>

  </div>

  <p class="note">The <code data-x="dom-label-form">form</code> IDL attribute on the
  <code>label</code> element is different from the <code data-x="attr-fae-form">form</code> IDL
  attribute on <span data-x="category-listed">listed</span> <span data-x="form-associated
  element">form-associated elements</span>, and the <code>label</code> element does not have a <code
  data-x="attr-fae-form">form</code> content attribute.</p>

  <hr>

  <dl class="domintro">
   <dt><code data-x=""><var>control</var>.<span subdfn data-x="dom-lfe-labels">labels</span></code></dt>

   <dd>
    <p>Returns a <code>NodeList</code> of all the <code>label</code> elements that the form control
    is associated with.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p><span data-x="category-label">Labelable elements</span> and all <code>input</code> elements
  have a <span>live</span> <code>NodeList</code> object associated with them that represents the
  list of <code>label</code> elements, in <span>tree order</span>, whose <span>labeled
  control</span> is the element in question. The <dfn attribute
  for="HTMLInputElement,HTMLButtonElement,HTMLSelectElement,HTMLTextAreaElement,HTMLOutputElement,HTMLProgressElement,HTMLMeterElement"><code
  data-x="dom-lfe-labels">labels</code></dfn> IDL attribute of <span
  data-x="category-label">labelable elements</span> that are not <span data-x="form-associated
  custom element">form-associated custom elements</span>, and the <code
  data-x="dom-lfe-labels">labels</code> IDL attribute of <code>input</code> elements, on getting,
  must return that <code>NodeList</code> object, and that same value must always be returned, unless
  this element is an <code>input</code> element whose <code data-x="attr-input-type">type</code>
  attribute is in the <span data-x="attr-input-type-hidden">Hidden</span> state, in which case it
  must instead return null.</p>

  <p><span data-x="form-associated custom element">Form-associated custom elements</span> don't have
  a <code data-x="dom-lfe-labels">labels</code> IDL attribute. Instead, their
  <code>ElementInternals</code> object has a <dfn attribute for="ElementInternals"><code
  data-x="dom-ElementInternals-labels">labels</code></dfn> IDL attribute. On getting, it must throw
  a <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code> if the <span
  data-x="internals-target">target element</span> is not a <span>form-associated custom
  element</span>. Otherwise, it must return that <code>NodeList</code> object, and that same value
  must always be returned.</p>

  <div class="example">
   <p>This (non-conforming) example shows what happens to the <code>NodeList</code> and what <code
   data-x="dom-lfe-labels">labels</code> returns when an <code>input</code> element has its <code
   data-x="attr-input-type">type</code> attribute changed.</p>

   <pre><code class="html">&lt;!doctype html>
&lt;p>&lt;label>&lt;input>&lt;/label>&lt;/p>
&lt;script>
 const input = document.querySelector('input');
 const labels = input.labels;
 console.assert(labels.length === 1);

 input.type = 'hidden';
 console.assert(labels.length === 0); // the input is no longer the label's <span>labeled control</span>
 console.assert(input.labels === null);

 input.type = 'checkbox';
 console.assert(labels.length === 1); // the input is once again the label's <span>labeled control</span>
 console.assert(input.labels === labels); // same value as returned originally
&lt;/script></code></pre>
  </div>

  </div>


  <h4 split-filename="input">The <dfn element><code>input</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd>If the <code data-x="attr-input-type">type</code> attribute is <em>not</em> in the <span data-x="attr-input-type-hidden">Hidden</span> state: <span>Interactive content</span>.</dd>
   <dd>If the <code data-x="attr-input-type">type</code> attribute is <em>not</em> in the <span data-x="attr-input-type-hidden">Hidden</span> state: <span data-x="category-listed">Listed</span>, <span data-x="category-label">labelable</span>, <span data-x="category-submit">submittable</span>, <span data-x="category-reset">resettable</span>, and <span data-x="category-autocapitalize-and-autocorrect">autocapitalize-and-autocorrect inheriting</span> <span>form-associated element</span>.</dd>
   <dd>If the <code data-x="attr-input-type">type</code> attribute is in the <span data-x="attr-input-type-hidden">Hidden</span> state: <span data-x="category-listed">Listed</span>, <span data-x="category-submit">submittable</span>, <span data-x="category-reset">resettable</span>, and <span data-x="category-autocapitalize-and-autocorrect">autocapitalize-and-autocorrect inheriting</span> <span>form-associated element</span>.</dd>
   <dd>If the <code data-x="attr-input-type">type</code> attribute is <em>not</em> in the <span data-x="attr-input-type-hidden">Hidden</span> state: <span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="concept-content-nothing">Nothing</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-input-accept">accept</code></dd>
   <dd><code data-x="attr-input-alpha">alpha</code></dd>
   <dd><code data-x="attr-input-alt">alt</code></dd>
   <dd><code data-x="attr-fe-autocomplete">autocomplete</code></dd>
   <dd><code data-x="attr-input-checked">checked</code></dd>
   <dd><code data-x="attr-input-colorspace">colorspace</code></dd>
   <dd><code data-x="attr-fe-dirname">dirname</code></dd>
   <dd><code data-x="attr-fe-disabled">disabled</code></dd>
   <dd><code data-x="attr-fae-form">form</code></dd>
   <dd><code data-x="attr-fs-formaction">formaction</code></dd>
   <dd><code data-x="attr-fs-formenctype">formenctype</code></dd>
   <dd><code data-x="attr-fs-formmethod">formmethod</code></dd>
   <dd><code data-x="attr-fs-formnovalidate">formnovalidate</code></dd>
   <dd><code data-x="attr-fs-formtarget">formtarget</code></dd>
   <dd><code data-x="attr-dim-height">height</code></dd>
   <dd><code data-x="attr-input-list">list</code></dd>
   <dd><code data-x="attr-input-max">max</code></dd>
   <dd><code data-x="attr-input-maxlength">maxlength</code></dd>
   <dd><code data-x="attr-input-min">min</code></dd>
   <dd><code data-x="attr-input-minlength">minlength</code></dd>
   <dd><code data-x="attr-input-multiple">multiple</code></dd>
   <dd><code data-x="attr-fe-name">name</code></dd>
   <dd><code data-x="attr-input-pattern">pattern</code></dd>
   <dd><code data-x="attr-input-placeholder">placeholder</code></dd>
   <dd><code data-x="attr-popovertarget">popovertarget</code></dd>
   <dd><code data-x="attr-popovertargetaction">popovertargetaction</code></dd>
   <dd><code data-x="attr-input-readonly">readonly</code></dd>
   <dd><code data-x="attr-input-required">required</code></dd>
   <dd><code data-x="attr-input-size">size</code></dd>
   <dd><code data-x="attr-input-src">src</code></dd>
   <dd><code data-x="attr-input-step">step</code></dd>
   <dd><code data-x="attr-input-type">type</code></dd>
   <dd><code data-x="attr-input-value">value</code></dd>
   <dd><code data-x="attr-dim-width">width</code></dd>
   <dd>Also, the <code data-x="attr-input-title">title</code> attribute <span data-x="attr-input-title">has special semantics</span> on this element.</dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-hidden">Hidden</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-hidden">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-hidden">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-text">Text</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-text">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-text">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-search">Search</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-search">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-search">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-tel">Telephone</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-tel">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-tel">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-url">URL</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-url">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-url">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-email">Email</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-email">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-email">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-password">Password</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-password">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-password">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-date">Date</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-date">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-date">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-month">Month</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-month">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-month">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-week">Week</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-week">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-week">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-time">Time</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-time">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-time">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-datetime-local">Local Date and Time</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-datetime-local">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-datetime-local">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-number">Number</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-number">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-number">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-range">Range</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-range">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-range">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-color">Color</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-color">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-color">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-checkbox">Checkbox</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-checkbox">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-checkbox">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-radio">Radio Button</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-radio">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-radio">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-file">File Upload</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-file">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-file">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-submit">Submit Button</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-submit">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-submit">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-image">Image Button</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-image">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-image">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-reset">Reset Button</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-reset">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-reset">for implementers</a>.</dd>
   <dd><code data-x="attr-input-type">type</code> attribute in the <span
   data-x="attr-input-type-button">Button</span> state: <a
   href="https://w3c.github.io/html-aria/#el-input-button">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-input-button">for implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLInputElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-input-accept">accept</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-input-alpha">alpha</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-input-alt">alt</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fe-autocomplete">autocomplete</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-input-defaultChecked">defaultChecked</span>;
  attribute boolean <span data-x="dom-input-checked">checked</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-input-colorspace">colorSpace</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-input-dirName">dirName</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-fe-disabled">disabled</span>;
  readonly attribute <span>HTMLFormElement</span>? <span data-x="dom-fae-form">form</span>;
  attribute <span>FileList</span>? <span data-x="dom-input-files">files</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-fs-formAction">formAction</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fs-formEnctype">formEnctype</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fs-formMethod">formMethod</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-fs-formNoValidate">formNoValidate</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fs-formTarget">formTarget</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-input-height">height</span>;
  attribute boolean <span data-x="dom-input-indeterminate">indeterminate</span>;
  readonly attribute <span>HTMLDataListElement</span>? <span data-x="dom-input-list">list</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-input-max">max</span>;
  [<span>CEReactions</span>] attribute long <span data-x="dom-input-maxLength">maxLength</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-input-min">min</span>;
  [<span>CEReactions</span>] attribute long <span data-x="dom-input-minLength">minLength</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-input-multiple">multiple</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fe-name">name</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-input-pattern">pattern</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-input-placeholder">placeholder</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-input-readOnly">readOnly</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-input-required">required</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-input-size">size</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-input-src">src</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-input-step">step</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-input-type">type</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-input-defaultValue">defaultValue</span>;
  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-input-value">value</span>;
  attribute <span data-x="idl-object">object</span>? <span data-x="dom-input-valueAsDate">valueAsDate</span>;
  attribute unrestricted double <span data-x="dom-input-valueAsNumber">valueAsNumber</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-input-width">width</span>;

  undefined <span data-x="dom-input-stepUp">stepUp</span>(optional long n = 1);
  undefined <span data-x="dom-input-stepDown">stepDown</span>(optional long n = 1);

  readonly attribute boolean <span data-x="dom-cva-willValidate">willValidate</span>;
  readonly attribute <span>ValidityState</span> <span data-x="dom-cva-validity">validity</span>;
  readonly attribute DOMString <span data-x="dom-cva-validationMessage">validationMessage</span>;
  boolean <span data-x="dom-cva-checkValidity">checkValidity</span>();
  boolean <span data-x="dom-cva-reportValidity">reportValidity</span>();
  undefined <span data-x="dom-cva-setCustomValidity">setCustomValidity</span>(DOMString error);

  readonly attribute <span>NodeList</span>? <span data-x="dom-lfe-labels">labels</span>;

  undefined <span data-x="dom-textarea/input-select">select</span>();
  attribute unsigned long? <span data-x="dom-textarea/input-selectionStart">selectionStart</span>;
  attribute unsigned long? <span data-x="dom-textarea/input-selectionEnd">selectionEnd</span>;
  attribute DOMString? <span data-x="dom-textarea/input-selectionDirection">selectionDirection</span>;
  undefined <span data-x="dom-textarea/input-setRangeText">setRangeText</span>(DOMString replacement);
  undefined <span data-x="dom-textarea/input-setRangeText">setRangeText</span>(DOMString replacement, unsigned long start, unsigned long end, optional <span>SelectionMode</span> selectionMode = "preserve");
  undefined <span data-x="dom-textarea/input-setSelectionRange">setSelectionRange</span>(unsigned long start, unsigned long end, optional DOMString direction);

  undefined <span data-x="dom-input-showPicker">showPicker</span>();

  // <a href="#HTMLInputElement-partial">also has obsolete members</a>
};
<span>HTMLInputElement</span> includes <span>PopoverInvokerElement</span>;</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLInputElement</code>.</dd>
  </dl>

  <p>The <code>input</code> element <span>represents</span> a typed data field, usually with a form
  control to allow the user to edit the data.</p>

  <p>The <dfn element-attr for="input"><code data-x="attr-input-type">type</code></dfn> attribute
  controls the data type (and associated control) of the element. It is an <span>enumerated
  attribute</span> with the following keywords and states:</p>

  <table id="attr-input-type-keywords">
   <thead>
    <tr>
     <th> Keyword
     <th> State
     <th> Data type
     <th> Control type
   <tbody>
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-hidden-keyword">hidden</code></dfn>
     <td> <span data-x="attr-input-type-hidden">Hidden</span>
     <td> An arbitrary string
     <td> n/a
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-text-keyword">text</code></dfn>
     <td> <span data-x="attr-input-type-text">Text</span>
     <td> Text with no line breaks
     <td> A text control
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-search-keyword">search</code></dfn>
     <td> <span data-x="attr-input-type-search">Search</span>
     <td> Text with no line breaks
     <td> Search control
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-tel-keyword">tel</code></dfn>
     <td> <span data-x="attr-input-type-tel">Telephone</span>
     <td> Text with no line breaks
     <td> A text control
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-url-keyword">url</code></dfn>
     <td> <span data-x="attr-input-type-url">URL</span>
     <td> An absolute URL
     <td> A text control
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-email-keyword">email</code></dfn>
     <td> <span data-x="attr-input-type-email">Email</span>
     <td> An email address or list of email addresses
     <td> A text control
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-password-keyword">password</code></dfn>
     <td> <span data-x="attr-input-type-password">Password</span>
     <td> Text with no line breaks (sensitive information)
     <td> A text control that obscures data entry
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-date-keyword">date</code></dfn>
     <td> <span data-x="attr-input-type-date">Date</span>
     <td> A date (year, month, day) with no time zone
     <td> A date control
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-month-keyword">month</code></dfn>
     <td> <span data-x="attr-input-type-month">Month</span>
     <td> A date consisting of a year and a month with no time zone
     <td> A month control
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-week-keyword">week</code></dfn>
     <td> <span data-x="attr-input-type-week">Week</span>
     <td> A date consisting of a week-year number and a week number with no time zone
     <td> A week control
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-time-keyword">time</code></dfn>
     <td> <span data-x="attr-input-type-time">Time</span>
     <td> A time (hour, minute, seconds, fractional seconds) with no time zone
     <td> A time control
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-datetime-local-keyword">datetime-local</code></dfn>
     <td> <span data-x="attr-input-type-datetime-local">Local Date and Time</span>
     <td> A date and time (year, month, day, hour, minute, second, fraction of a second) with no time zone
     <td> A date and time control
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-number-keyword">number</code></dfn>
     <td> <span data-x="attr-input-type-number">Number</span>
     <td> A numerical value
     <td> A text control or spinner control
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-range-keyword">range</code></dfn>
     <td> <span data-x="attr-input-type-range">Range</span>
     <td> A numerical value, with the extra semantic that the exact value is not important
     <td> A slider control or similar
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-color-keyword">color</code></dfn>
     <td> <span data-x="attr-input-type-color">Color</span>
     <td> An sRGB color with 8-bit red, green, and blue components
     <td> A color picker
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-checkbox-keyword">checkbox</code></dfn>
     <td> <span data-x="attr-input-type-checkbox">Checkbox</span>
     <td> A set of zero or more values from a predefined list
     <td> A checkbox
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-radio-keyword">radio</code></dfn>
     <td> <span data-x="attr-input-type-radio">Radio Button</span>
     <td> An enumerated value
     <td> A radio button
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-file-keyword">file</code></dfn>
     <td> <span data-x="attr-input-type-file">File Upload</span>
     <td> Zero or more files each with a <span>MIME type</span> and optionally a filename
     <td> A label and a button
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-submit-keyword">submit</code></dfn>
     <td> <span data-x="attr-input-type-submit">Submit Button</span>
     <td> An enumerated value, with the extra semantic that it must be the last value selected and initiates form submission
     <td> A button
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-image-keyword">image</code></dfn>
     <td> <span data-x="attr-input-type-image">Image Button</span>
     <td> A coordinate, relative to a particular image's size, with the extra semantic that it must
     be the last value selected and initiates form submission
     <td> Either a clickable image, or a button
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-reset-keyword">reset</code></dfn>
     <td> <span data-x="attr-input-type-reset">Reset Button</span>
     <td> n/a
     <td> A button
    <tr>
     <td> <dfn attr-value for="input/type"><code data-x="attr-input-type-button-keyword">button</code></dfn>
     <td> <span data-x="attr-input-type-button">Button</span>
     <td> n/a
     <td> A button
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <span
  data-x="attr-input-type-text">Text</span> state.</p>

  <p>Which of the
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>,
   <code data-x="attr-input-size">size</code>,
   <code data-x="attr-input-src">src</code>,
   <code data-x="attr-input-step">step</code>, and
   <code data-x="attr-dim-width">width</code> content attributes, the
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>,
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code>, and
   <code data-x="dom-input-list">list</code> IDL attributes, the
   <code data-x="dom-textarea/input-select">select()</code> method, the
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>, and
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>, IDL attributes, the
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code> and
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code> methods, the
   <code data-x="dom-input-stepUp">stepUp()</code> and
   <code data-x="dom-input-stepDown">stepDown()</code> methods, and the
   <code data-x="event-input">input</code> and
   <code data-x="event-change">change</code> events <dfn data-x="concept-input-apply">apply</dfn> to an
   <code>input</code> element depends on the state of its
   <code data-x="attr-input-type">type</code> attribute.
  The subsections that define each type also clearly define in normative "bookkeeping" sections
  which of these feature apply, and which <dfn>do not apply</dfn>, to each type. The behavior of
  these features depends on whether they apply or not, as defined in their various sections (q.v.
  for <a href="#common-input-element-attributes">content attributes</a>, for <a
  href="#common-input-element-apis">APIs</a>, for <a
  href="#common-input-element-events">events</a>).</p>

  <p>The following table <span w-nodev>is non-normative and</span> summarizes which of those
  content attributes, IDL attributes, methods, and events <span
  data-x="concept-input-apply">apply</span> to each state:</p>

  <table class="applies" id="input-type-attr-summary">
   <thead>
    <tr>
     <th>
     <th> <span data-x=""><span data-x="attr-input-type-hidden">Hidden</span></span>
     <th> <span data-x=""><span data-x="attr-input-type-text">Text</span>,</span>
          <span data-x=""><span data-x="attr-input-type-search">Search</span></span>
     <th> <span data-x=""><span data-x="attr-input-type-tel">Telephone</span>,</span>
          <span data-x=""><span data-x="attr-input-type-url">URL</span></span>
     <th> <span data-x=""><span data-x="attr-input-type-email">Email</span></span>
     <th> <span data-x=""><span data-x="attr-input-type-password">Password</span></span>
     <th> <span data-x=""><span data-x="attr-input-type-date">Date</span>,</span>
          <span data-x=""><span data-x="attr-input-type-month">Month</span>,</span>
          <span data-x=""><span data-x="attr-input-type-week">Week</span>,</span>
          <span data-x=""><span data-x="attr-input-type-time">Time</span></span>
     <th> <span data-x=""><span data-x="attr-input-type-datetime-local">Local Date and Time</span></span>
     <th> <span data-x=""><span data-x="attr-input-type-number">Number</span></span>
     <th> <span data-x=""><span data-x="attr-input-type-range">Range</span></span>
     <th> <span data-x=""><span data-x="attr-input-type-color">Color</span></span>
     <th> <span data-x=""><span data-x="attr-input-type-checkbox">Checkbox</span>,</span>
          <span data-x=""><span data-x="attr-input-type-radio">Radio Button</span></span>
     <th> <span data-x=""><span data-x="attr-input-type-file">File Upload</span></span>
     <th> <span data-x=""><span data-x="attr-input-type-submit">Submit Button</span></span>
     <th> <span data-x=""><span data-x="attr-input-type-image">Image Button</span></span>
     <th> <span data-x=""><span data-x="attr-input-type-reset">Reset Button</span>,</span>
          <span data-x=""><span data-x="attr-input-type-button">Button</span></span>

   <tbody>
    <tr>
     <th scope="rowgroup" colspan="16">Content attributes

    <tr>
     <th> <code data-x="attr-input-accept">accept</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="yes"> Yes     <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-input-alpha">alpha</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="yes"> Yes     <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-input-alt">alt</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="yes"> Yes     <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-fe-autocomplete">autocomplete</code>
     <td class="yes"> Yes     <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="yes"> Yes     <!-- Email -->
     <td class="yes"> Yes     <!-- Password -->
     <td class="yes"> Yes     <!-- Date -->
<!-- <td class="yes"> Yes          Month -->
<!-- <td class="yes"> Yes          Week -->
<!-- <td class="yes"> Yes          Time -->
     <td class="yes"> Yes     <!-- Local Date and Time -->
     <td class="yes"> Yes     <!-- Number -->
     <td class="yes"> Yes     <!-- Range -->
     <td class="yes"> Yes     <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-input-checked">checked</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="yes"> Yes     <!-- Checkbox -->
<!-- <td class="yes"> Yes          Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-input-colorspace">colorspace</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="yes"> Yes     <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-fe-dirname">dirname</code>
     <td class="yes"> Yes     <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="yes"> Yes     <!-- Email -->
     <td class="yes"> Yes     <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="yes"> Yes     <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-fs-formaction">formaction</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="yes"> Yes     <!-- Submit Button -->
     <td class="yes"> Yes     <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-fs-formenctype">formenctype</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="yes"> Yes     <!-- Submit Button -->
     <td class="yes"> Yes     <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-fs-formmethod">formmethod</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="yes"> Yes     <!-- Submit Button -->
     <td class="yes"> Yes     <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-fs-formnovalidate">formnovalidate</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="yes"> Yes     <!-- Submit Button -->
     <td class="yes"> Yes     <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-fs-formtarget">formtarget</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="yes"> Yes     <!-- Submit Button -->
     <td class="yes"> Yes     <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-dim-height">height</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="yes"> Yes     <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-input-list">list</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="yes"> Yes     <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="yes"> Yes     <!-- Date -->
<!-- <td class="yes"> Yes          Month -->
<!-- <td class="yes"> Yes          Week -->
<!-- <td class="yes"> Yes          Time -->
     <td class="yes"> Yes     <!-- Local Date and Time -->
     <td class="yes"> Yes     <!-- Number -->
     <td class="yes"> Yes     <!-- Range -->
     <td class="yes"> Yes     <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-input-max">max</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="yes"> Yes     <!-- Date -->
<!-- <td class="yes"> Yes          Month -->
<!-- <td class="yes"> Yes          Week -->
<!-- <td class="yes"> Yes          Time -->
     <td class="yes"> Yes     <!-- Local Date and Time -->
     <td class="yes"> Yes     <!-- Number -->
     <td class="yes"> Yes     <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-input-maxlength">maxlength</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="yes"> Yes     <!-- Email -->
     <td class="yes"> Yes     <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-input-min">min</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="yes"> Yes     <!-- Date -->
<!-- <td class="yes"> Yes          Month -->
<!-- <td class="yes"> Yes          Week -->
<!-- <td class="yes"> Yes          Time -->
     <td class="yes"> Yes     <!-- Local Date and Time -->
     <td class="yes"> Yes     <!-- Number -->
     <td class="yes"> Yes     <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-input-minlength">minlength</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="yes"> Yes     <!-- Email -->
     <td class="yes"> Yes     <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-input-multiple">multiple</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL --> <!-- if you change this, you can merge this line with the next to save room on the chart -->
     <td class="yes"> Yes     <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="yes"> Yes     <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-input-pattern">pattern</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="yes"> Yes     <!-- Email -->
     <td class="yes"> Yes     <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-input-placeholder">placeholder</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="yes"> Yes     <!-- Email -->
     <td class="yes"> Yes     <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="yes"> Yes     <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-popovertarget">popovertarget</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="yes"> Yes     <!-- Submit Button -->
     <td class="yes"> Yes     <!-- Image Button -->
     <td class="yes"> Yes     <!-- Reset Button -->
<!-- <td class="yes"> Yes          Button -->

    <tr>
     <th> <code data-x="attr-popovertargetaction">popovertargetaction</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="yes"> Yes     <!-- Submit Button -->
     <td class="yes"> Yes     <!-- Image Button -->
     <td class="yes"> Yes     <!-- Reset Button -->
<!-- <td class="yes"> Yes          Button -->

    <tr>
     <th> <code data-x="attr-input-readonly">readonly</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="yes"> Yes     <!-- Email -->
     <td class="yes"> Yes     <!-- Password -->
     <td class="yes"> Yes     <!-- Date -->
<!-- <td class="yes"> Yes          Month -->
<!-- <td class="yes"> Yes          Week -->
<!-- <td class="yes"> Yes          Time -->
     <td class="yes"> Yes     <!-- Local Date and Time -->
     <td class="yes"> Yes     <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-input-required">required</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="yes"> Yes     <!-- Email -->
     <td class="yes"> Yes     <!-- Password -->
     <td class="yes"> Yes     <!-- Date -->
<!-- <td class="yes"> Yes          Month -->
<!-- <td class="yes"> Yes          Week -->
<!-- <td class="yes"> Yes          Time -->
     <td class="yes"> Yes     <!-- Local Date and Time -->
     <td class="yes"> Yes     <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="yes"> Yes     <!-- Checkbox -->
<!-- <td class="yes"> Yes          Radio Button -->
     <td class="yes"> Yes     <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-input-size">size</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="yes"> Yes     <!-- Email -->
     <td class="yes"> Yes     <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-input-src">src</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="yes"> Yes     <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-input-step">step</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="yes"> Yes     <!-- Date -->
<!-- <td class="yes"> Yes          Month -->
<!-- <td class="yes"> Yes          Week -->
<!-- <td class="yes"> Yes          Time -->
     <td class="yes"> Yes     <!-- Local Date and Time -->
     <td class="yes"> Yes     <!-- Number -->
     <td class="yes"> Yes     <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="attr-dim-width">width</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="yes"> Yes     <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

   <tbody>
    <tr>
     <th scope="rowgroup" colspan="16">IDL attributes and methods

    <tr>
     <th> <code data-x="dom-input-checked">checked</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="yes"> Yes     <!-- Checkbox -->
<!-- <td class="yes"> Yes          Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="dom-input-files">files</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="yes"> Yes     <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr w-nodev>
     <th> <code data-x="dom-input-value">value</code>
     <td class="yes"> <span data-x="dom-input-value-default">default</span> <!-- Hidden -->
     <td class="yes"> <span data-x="dom-input-value-value">value</span> <!-- Text -->
<!-- <td class="yes"> <span data-x="dom-input-value-value">value</span>      Search -->
     <td class="yes"> <span data-x="dom-input-value-value">value</span> <!-- Telephone, URL -->
     <td class="yes"> <span data-x="dom-input-value-value">value</span> <!-- Email -->
     <td class="yes"> <span data-x="dom-input-value-value">value</span> <!-- Password -->
     <td class="yes"> <span data-x="dom-input-value-value">value</span> <!-- Date -->
<!-- <td class="yes"> <span data-x="dom-input-value-value">value</span>      Month -->
<!-- <td class="yes"> <span data-x="dom-input-value-value">value</span>      Week -->
<!-- <td class="yes"> <span data-x="dom-input-value-value">value</span>      Time -->
     <td class="yes"> <span data-x="dom-input-value-value">value</span> <!-- Local Date and Time -->
     <td class="yes"> <span data-x="dom-input-value-value">value</span> <!-- Number -->
     <td class="yes"> <span data-x="dom-input-value-value">value</span> <!-- Range -->
     <td class="yes"> <span data-x="dom-input-value-value">value</span> <!-- Color -->
     <td class="yes"> <span data-x="dom-input-value-default-on">default/on</span> <!-- Checkbox -->
<!-- <td class="yes"> <span data-x="dom-input-value-default-on">default/on</span>      Radio Button -->
     <td class="yes"> <span data-x="dom-input-value-filename">filename</span> <!-- File Upload -->
     <td class="yes"> <span data-x="dom-input-value-default">default</span> <!-- Submit Button -->
     <td class="yes"> <span data-x="dom-input-value-default">default</span> <!-- Image Button -->
     <td class="yes"> <span data-x="dom-input-value-default">default</span> <!-- Reset Button -->
<!-- <td class="yes"> <span data-x="dom-input-value-default">default</span>      Button -->

    <tr>
     <th> <code data-x="dom-input-valueAsDate">valueAsDate</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="yes"> Yes     <!-- Date -->
<!-- <td class="yes"> Yes          Month -->
<!-- <td class="yes"> Yes          Week -->
<!-- <td class="yes"> Yes          Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="dom-input-valueAsNumber">valueAsNumber</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="yes"> Yes     <!-- Date -->
<!-- <td class="yes"> Yes          Month -->
<!-- <td class="yes"> Yes          Week -->
<!-- <td class="yes"> Yes          Time -->
     <td class="yes"> Yes     <!-- Local Date and Time -->
     <td class="yes"> Yes     <!-- Number -->
     <td class="yes"> Yes     <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="dom-input-list">list</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="yes"> Yes     <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="yes"> Yes     <!-- Date -->
<!-- <td class="yes"> Yes          Month -->
<!-- <td class="yes"> Yes          Week -->
<!-- <td class="yes"> Yes          Time -->
     <td class="yes"> Yes     <!-- Local Date and Time -->
     <td class="yes"> Yes     <!-- Number -->
     <td class="yes"> Yes     <!-- Range -->
     <td class="yes"> Yes     <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="dom-textarea/input-select">select()</code>
     <td class="no"> &middot;     <!-- Hidden -->
     <td class="yes"> Yes         <!-- Text -->
<!-- <td class="yes"> Yes              Search -->
     <td class="yes"> Yes         <!-- Telephone, URL -->
     <td class="yes"> Yes&dagger; <!-- Email -->
     <td class="yes"> Yes         <!-- Password -->
     <td class="yes"> Yes&dagger; <!-- Date -->
<!-- <td class="yes"> Yes              Month -->
<!-- <td class="yes"> Yes              Week -->
<!-- <td class="yes"> Yes              Time -->
     <td class="yes"> Yes&dagger; <!-- Local Date and Time -->
     <td class="yes"> Yes&dagger; <!-- Number -->
     <td class="no"> &middot;     <!-- Range -->
     <td class="yes"> Yes&dagger; <!-- Color -->
     <td class="no"> &middot;     <!-- Checkbox -->
<!-- <td class="no"> &middot;          Radio Button -->
     <td class="yes"> Yes&dagger; <!-- File Upload -->
     <td class="no"> &middot;     <!-- Submit Button -->
     <td class="no"> &middot;     <!-- Image Button -->
     <td class="no"> &middot;     <!-- Reset Button -->
<!-- <td class="no"> &middot;          Button -->

    <tr>
     <th> <code data-x="dom-textarea/input-selectionStart">selectionStart</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="yes"> Yes     <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="yes"> Yes     <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="yes"> Yes     <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="yes"> Yes     <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="yes"> Yes     <!-- Password -->
     <td class="no"> &middot; <!-- Date -->
<!-- <td class="no"> &middot;      Month -->
<!-- <td class="no"> &middot;      Week -->
<!-- <td class="no"> &middot;      Time -->
     <td class="no"> &middot; <!-- Local Date and Time -->
     <td class="no"> &middot; <!-- Number -->
     <td class="no"> &middot; <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="dom-input-stepDown">stepDown()</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="yes"> Yes     <!-- Date -->
<!-- <td class="yes"> Yes          Month -->
<!-- <td class="yes"> Yes          Week -->
<!-- <td class="yes"> Yes          Time -->
     <td class="yes"> Yes     <!-- Local Date and Time -->
     <td class="yes"> Yes     <!-- Number -->
     <td class="yes"> Yes     <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <code data-x="dom-input-stepUp">stepUp()</code>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="no"> &middot; <!-- Text -->
<!-- <td class="no"> &middot;      Search -->
     <td class="no"> &middot; <!-- Telephone, URL -->
     <td class="no"> &middot; <!-- Email -->
     <td class="no"> &middot; <!-- Password -->
     <td class="yes"> Yes     <!-- Date -->
<!-- <td class="yes"> Yes          Month -->
<!-- <td class="yes"> Yes          Week -->
<!-- <td class="yes"> Yes          Time -->
     <td class="yes"> Yes     <!-- Local Date and Time -->
     <td class="yes"> Yes     <!-- Number -->
     <td class="yes"> Yes     <!-- Range -->
     <td class="no"> &middot; <!-- Color -->
     <td class="no"> &middot; <!-- Checkbox -->
<!-- <td class="no"> &middot;      Radio Button -->
     <td class="no"> &middot; <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

   <tbody>
    <tr>
     <th scope="rowgroup" colspan="16">Events

    <tr>
     <th> <span data-x=""><code data-x="event-input">input</code> event</span>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="yes"> Yes     <!-- Email -->
     <td class="yes"> Yes     <!-- Password -->
     <td class="yes"> Yes     <!-- Date -->
<!-- <td class="yes"> Yes          Month -->
<!-- <td class="yes"> Yes          Week -->
<!-- <td class="yes"> Yes          Time -->
     <td class="yes"> Yes     <!-- Local Date and Time -->
     <td class="yes"> Yes     <!-- Number -->
     <td class="yes"> Yes     <!-- Range -->
     <td class="yes"> Yes     <!-- Color -->
     <td class="yes"> Yes     <!-- Checkbox -->
<!-- <td class="yes"> Yes          Radio Button -->
     <td class="yes"> Yes     <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

    <tr>
     <th> <span data-x=""><code data-x="event-change">change</code> event</span>
     <td class="no"> &middot; <!-- Hidden -->
     <td class="yes"> Yes     <!-- Text -->
<!-- <td class="yes"> Yes          Search -->
     <td class="yes"> Yes     <!-- Telephone, URL -->
     <td class="yes"> Yes     <!-- Email -->
     <td class="yes"> Yes     <!-- Password -->
     <td class="yes"> Yes     <!-- Date -->
<!-- <td class="yes"> Yes          Month -->
<!-- <td class="yes"> Yes          Week -->
<!-- <td class="yes"> Yes          Time -->
     <td class="yes"> Yes     <!-- Local Date and Time -->
     <td class="yes"> Yes     <!-- Number -->
     <td class="yes"> Yes     <!-- Range -->
     <td class="yes"> Yes     <!-- Color -->
     <td class="yes"> Yes     <!-- Checkbox -->
<!-- <td class="yes"> Yes          Radio Button -->
     <td class="yes"> Yes     <!-- File Upload -->
     <td class="no"> &middot; <!-- Submit Button -->
     <td class="no"> &middot; <!-- Image Button -->
     <td class="no"> &middot; <!-- Reset Button -->
<!-- <td class="no"> &middot;      Button -->

  </table>

  <p class="tablenote"><small>&dagger; If the control has no selectable text, the <code
  data-x="dom-textarea/input-select">select()</code> method results in a no-op, with no
  <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</small></p>

  <div w-nodev>

  <p>Some states of the <code data-x="attr-input-type">type</code> attribute define a <dfn>value
  sanitization algorithm</dfn>.</p>

  <p>Each <code>input</code> element has a <span data-x="concept-fe-value">value</span>, which is
  exposed by the <code data-x="dom-input-value">value</code> IDL attribute. Some states define an
  <dfn data-x="concept-input-value-string-number">algorithm to convert a string to a number</dfn>,
  an <dfn data-x="concept-input-value-number-string">algorithm to convert a number to a
  string</dfn>, an <dfn data-x="concept-input-value-string-date">algorithm to convert a string to a
  <code>Date</code> object</dfn>, and an <dfn data-x="concept-input-value-date-string">algorithm to
  convert a <code>Date</code> object to a string</dfn>, which are used by <code
  data-x="attr-input-max">max</code>, <code data-x="attr-input-min">min</code>, <code
  data-x="attr-input-step">step</code>, <code data-x="dom-input-valueAsDate">valueAsDate</code>,
  <code data-x="dom-input-valueAsNumber">valueAsNumber</code>, and <code
  data-x="dom-input-stepUp">stepUp()</code>.</p>

  <p>An <code>input</code> element's <span data-x="concept-fe-dirty">dirty value flag</span> must be
  set to true whenever the user interacts with the control in a way that changes the <span
  data-x="concept-fe-value">value</span>. (It is also set to true when the value is programmatically
  changed, as described in the definition of the <code data-x="dom-input-value">value</code> IDL
  attribute.)</p>

  </div>

  <p>The <dfn element-attr for="input"><code data-x="attr-input-value">value</code></dfn> content
  attribute gives the default <span data-x="concept-fe-value">value</span> of the <code>input</code>
  element. <span w-nodev>When the <code data-x="attr-input-value">value</code> content attribute is
  added, set, or removed, if the control's <span data-x="concept-fe-dirty">dirty value flag</span>
  is false, the user agent must set the <span data-x="concept-fe-value">value</span> of the element
  to the value of the <code data-x="attr-input-value">value</code> content attribute, if there is
  one, or the empty string otherwise, and then run the current <span>value sanitization
  algorithm</span>, if one is defined.</span></p>

  <div w-nodev>

  <p>Each <code>input</code> element has a <span data-x="concept-fe-checked">checkedness</span>,
  which is exposed by the <code data-x="dom-input-checked">checked</code> IDL attribute.</p>

  <p>Each <code>input</code> element has a boolean <dfn
  data-x="concept-input-checked-dirty-flag">dirty checkedness flag</dfn>. When it is true, the
  element is said to have a <dfn data-x="concept-input-checked-dirty"><i>dirty checkedness</i></dfn>.
  The <span data-x="concept-input-checked-dirty-flag">dirty checkedness flag</span> must be initially
  set to false when the element is created, and must be set to true whenever the user interacts with
  the control in a way that changes the <span data-x="concept-fe-checked">checkedness</span>.</p>

  </div>

  <p>The <dfn element-attr for="input"><code data-x="attr-input-checked">checked</code></dfn>
  content attribute is a <span>boolean attribute</span> that gives the default <span
  data-x="concept-fe-checked">checkedness</span> of the <code>input</code> element. <span
  w-nodev>When the <code data-x="attr-input-checked">checked</code> content attribute is added, if
  the control does not have <i data-x="concept-input-checked-dirty">dirty checkedness</i>, the user
  agent must set the <span data-x="concept-fe-checked">checkedness</span> of the element to true;
  when the <code data-x="attr-input-checked">checked</code> content attribute is removed, if the
  control does not have <i data-x="concept-input-checked-dirty">dirty checkedness</i>, the user
  agent must set the <span data-x="concept-fe-checked">checkedness</span> of the element to
  false.</span></p>

  <div w-nodev>

  <p>The <span data-x="concept-form-reset-control">reset algorithm</span> for <code>input</code>
  elements is to set its <span>user validity</span>,
  <span data-x="concept-fe-dirty">dirty value flag</span>, and
  <span data-x="concept-input-checked-dirty-flag">dirty checkedness flag</span>
  back to false, set the <span data-x="concept-fe-value">value</span> of the element to the value
  of the <code data-x="attr-input-value">value</code> content attribute, if there is one, or the
  empty string otherwise, set the <span data-x="concept-fe-checked">checkedness</span> of the
  element to true if the element has a <code data-x="attr-input-checked">checked</code> content
  attribute and false if it does not, empty the list of <span
  data-x="concept-input-type-file-selected">selected files</span>, and then invoke the <span>value
  sanitization algorithm</span>, if the <code data-x="attr-input-type">type</code> attribute's
  current state defines one.</p>

  <p>Each <code>input</code> element can be <i data-x="concept-fe-mutable">mutable</i>. Except where
  otherwise specified, an <code>input</code> element is always <i
  data-x="concept-fe-mutable">mutable</i>. Similarly, except where otherwise specified, the user
  agent should not allow the user to modify the element's <span
  data-x="concept-fe-value">value</span> or <span data-x="concept-fe-checked">checkedness</span>.</p>

  <p>When an <code>input</code> element is <span
  data-x="concept-fe-disabled">disabled</span>, it is not <i
  data-x="concept-fe-mutable">mutable</i>.</p>

  <p class="note">The <code data-x="attr-input-readonly">readonly</code> attribute can also in some
  cases (e.g. for the <span data-x="attr-input-type-date">Date</span> state, but not the <span
  data-x="attr-input-type-checkbox">Checkbox</span> state) stop an <code>input</code> element from
  being <i data-x="concept-fe-mutable">mutable</i>.</p>

  <p>The <code>input</code> element can <dfn data-x="input-support-picker">support a picker</dfn>. A
  picker is a user interface element that allows the end user to choose a value. Whether an
  <code>input</code> element supports a picker depends on the <code
  data-x="attr-input-type">type</code> attribute state and <span>implementation-defined</span>
  behavior. An <code>input</code> element must support a picker when its <code
  data-x="attr-input-type">type</code> attribute is in the <span data-x="attr-input-type-file">File
  Upload</span> state.</p>

  <div class="note">
   <p>As of the time of this writing, typical browser implementations show such picker UI for:</p>

   <ul>
    <li><p><code>input</code> elements whose <code data-x="attr-input-type">type</code>
    attributes are in the <span data-x="attr-input-type-date">Date</span>, <span
    data-x="attr-input-type-month">Month</span>, <span data-x="attr-input-type-week">Week</span>,
    <span data-x="attr-input-type-time">Time</span>, <span
    data-x="attr-input-type-datetime-local">Local Date and Time</span>, and <span
    data-x="attr-input-type-color">Color</span> states;</p></li>

    <li><p><code>input</code> elements in various states that have a <span
    data-x="concept-input-list">suggestions source element</span>;</p></li>

    <li><p><code>input</code> elements whose <code data-x="attr-input-type">type</code> attribute
    is in the <span data-x="attr-input-type-file">File Upload</span> state; and</p></li>

    <li><p><code>select</code> elements.</p></li>
   </ul>
  </div>

  <p>The <span data-x="concept-node-clone-ext">cloning steps</span> for <code>input</code> elements
  given <var>node</var>, <var>copy</var>, and <var>subtree</var> are to propagate the <span
  data-x="concept-fe-value">value</span>, <span data-x="concept-fe-dirty">dirty value flag</span>,
  <span data-x="concept-fe-checked">checkedness</span>, and <span
  data-x="concept-input-checked-dirty-flag">dirty checkedness flag</span> from <var>node</var> to
  <var>copy</var>.</p>

  <p>The <span>activation behavior</span> for <code>input</code> elements <var>element</var>, given
  <var>event</var>, are these steps:</p>

  <ol>
   <li><p>If <var>element</var> is not <i data-x="concept-fe-mutable">mutable</i>, and
   <var>element</var>'s <span data-x="attr-input-type">type</span> attribute is neither in the <span
   data-x="attr-input-type-checkbox">Checkbox</span> nor in the <span
   data-x="attr-input-type-radio">Radio</span> state, then return.</p></li>

   <li><p>Run <var>element</var>'s <dfn>input activation behavior</dfn>, if any, and do nothing
   otherwise.</p></li>

   <li><p>If <var>element</var> has a <span>form owner</span> and <var>element</var>'s <span
   data-x="attr-input-type">type</span> attribute is not in the <span
   data-x="attr-input-type-button">Button</span> state, then return.</p></li>

   <li><p>Run the <span>popover target attribute activation behavior</span> given <var>element</var>
   and <var>event</var>'s <span data-x="concept-event-target">target</span>.</p></li>
  </ol>

  <p class="note">Recall that an element's <span>activation behavior</span> runs for both
  user-initiated activations and for synthetic activations (e.g., via <code
  data-x="">el.click()</code>). User agents might also have behaviors for a given control — not
  specified here — that are triggered only by true user-initiated activations. A common choice is to
  <span>show the picker, if applicable</span>, for the control. In contrast, the <span>input
  activation behavior</span> only shows pickers for the special historical cases of the <span
  data-x="attr-input-type-file">File Upload</span> and <span
  data-x="attr-input-type-color">Color</span> states.</p>

  <p>The <span>legacy-pre-activation behavior</span> for <code>input</code> elements are these
  steps:</p>

  <ol>
   <li><p>If this element's <code data-x="attr-input-type">type</code> attribute is in the <span
   data-x="attr-input-type-checkbox">Checkbox state</span>, then set this element's <span
   data-x="concept-fe-checked">checkedness</span> to its opposite value (i.e. true if it is false,
   false if it is true) and set this element's <code
   data-x="dom-input-indeterminate">indeterminate</code> IDL attribute to false.</p></li>

   <li><p>If this element's <code data-x="attr-input-type">type</code> attribute is in the <span
   data-x="attr-input-type-radio">Radio Button state</span>, then get a reference to the element in
   this element's <span>radio button group</span> that has its <span
   data-x="concept-fe-checked">checkedness</span> set to true, if any, and then set this element's
   <span data-x="concept-fe-checked">checkedness</span> to true.</p></li>
  </ol>

  <p>The <span>legacy-canceled-activation behavior</span> for <code>input</code> elements are these
  steps:</p>

  <ol>
   <li><p>If the element's <code data-x="attr-input-type">type</code> attribute is in the <span
   data-x="attr-input-type-checkbox">Checkbox state</span>, then set the element's <span
   data-x="concept-fe-checked">checkedness</span> and the element's <code
   data-x="dom-input-indeterminate">indeterminate</code> IDL attribute back to the values they had
   before the <span>legacy-pre-activation behavior</span> was run.</p></li>

   <li><p>If this element's <code data-x="attr-input-type">type</code> attribute is in the <span
   data-x="attr-input-type-radio">Radio Button state</span>, then if the element to which a
   reference was obtained in the <span>legacy-pre-activation behavior</span>, if any, is still in
   what is now this element's <span>radio button group</span>, if it still has one, and if so,
   setting that element's <span data-x="concept-fe-checked">checkedness</span> to true; or else, if
   there was no such element, or that element is no longer in this element's <span>radio button
   group</span>, or if this element no longer has a <span>radio button group</span>, setting this
   element's <span data-x="concept-fe-checked">checkedness</span> to false.
  </ol>

  <!--
    http://software.hixie.ch/utilities/js/live-dom-viewer/?saved=3094
    http://software.hixie.ch/utilities/js/live-dom-viewer/?saved=3095
    http://software.hixie.ch/utilities/js/live-dom-viewer/?saved=3096
  -->

  <hr>

  <p>When an <code>input</code> element is first created, the element's rendering and behavior must
  be set to the rendering and behavior defined for the <code data-x="attr-input-type">type</code>
  attribute's state, and the <span>value sanitization algorithm</span>, if one is defined for the
  <code data-x="attr-input-type">type</code> attribute's state, must be invoked.</p>

  </div>

  <div id="input-type-change" w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute
  changes state, the user agent must run the following steps:</p>

  <ol>
   <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=602 -->

   <li><p>If the previous state of the element's <code data-x="attr-input-type">type</code> attribute
   put the <code data-x="dom-input-value">value</code> IDL attribute in the <i
   data-x="dom-input-value-value">value</i> mode, and the element's <span
   data-x="concept-fe-value">value</span> is not the empty string, and the new state of the element's
   <code data-x="attr-input-type">type</code> attribute puts the <code
   data-x="dom-input-value">value</code> IDL attribute in either the <i
   data-x="dom-input-value-default">default</i> mode or the <i
   data-x="dom-input-value-default-on">default/on</i> mode, then set the element's <code
   data-x="attr-input-value">value</code> content attribute to the element's <span
   data-x="concept-fe-value">value</span>.</p></li>

   <li><p>Otherwise, if the previous state of the element's <code
   data-x="attr-input-type">type</code> attribute put the <code data-x="dom-input-value">value</code>
   IDL attribute in any mode other than the <i data-x="dom-input-value-value">value</i> mode, and the
   new state of the element's <code data-x="attr-input-type">type</code> attribute puts the <code
   data-x="dom-input-value">value</code> IDL attribute in the <i
   data-x="dom-input-value-value">value</i> mode, then set the <span
   data-x="concept-fe-value">value</span> of the element to the value of the <code
   data-x="attr-input-value">value</code> content attribute, if there is one, or the empty string
   otherwise, and then set the control's <span data-x="concept-fe-dirty">dirty value flag</span> to
   false.</p></li>

   <!-- This step is present for the sake of clarity and safety so that the previous internal value
   can't be accidentally referenced while we're in the type=file state. This step's effect isn't
   (or at least shouldn't be) observable by script, due to the #input-type-change and
   #dom-input-value state machines. -->
   <li><p>Otherwise, if the previous state of the element's <code
   data-x="attr-input-type">type</code> attribute put the <code
   data-x="dom-input-value">value</code> IDL attribute in any mode other than the <i
   data-x="dom-input-value-filename">filename</i> mode, and the new state of the element's <code
   data-x="attr-input-type">type</code> attribute puts the <code
   data-x="dom-input-value">value</code> IDL attribute in the <i
   data-x="dom-input-value-filename">filename</i> mode, then set the <span
   data-x="concept-fe-value">value</span> of the element to the empty string.</p></li>

   <li><p>Update the element's rendering and behavior to the new state's.</p></li>

   <li><p><dfn>Signal a type change</dfn> for the element. (The <span
   data-x="attr-input-type-radio">Radio Button</span> state uses this, in particular.)</p></li>

   <li><p>Invoke the <span>value sanitization algorithm</span>, if one is defined for the <code
   data-x="attr-input-type">type</code> attribute's new state.</p></li>

   <li><p>Let <var>previouslySelectable</var> be true if <code
   data-x="dom-textarea/input-setRangeText">setRangeText()</code> previously <span
   data-x="concept-input-apply">applied</span> to the element, and false otherwise.</p></li>

   <li><p>Let <var>nowSelectable</var> be true if <code
   data-x="dom-textarea/input-setRangeText">setRangeText()</code> now <span
   data-x="concept-input-apply">applies</span> to the element, and false otherwise.</p></li>

   <li><p>If <var>previouslySelectable</var> is false and <var>nowSelectable</var> is true, set the
   element's <span data-x="concept-textarea/input-cursor">text entry cursor position</span> to the
   beginning of the text control, and <span data-x="set the selection direction">set its selection
   direction</span> to "<code data-x="">none</code>".</p></li>
  </ol>

  </div>

  <hr>

  <p>
  The <code data-x="attr-fe-name">name</code> attribute represents the element's name.
  The <code data-x="attr-fe-dirname">dirname</code> attribute controls how the element's <span data-x="the directionality">directionality</span> is submitted.
  The <code data-x="attr-fe-disabled">disabled</code> attribute is used to make the control non-interactive and to prevent its value from being submitted.
  The <code data-x="attr-fae-form">form</code> attribute is used to explicitly associate the <code>input</code> element with its <span>form owner</span>.
  The <code data-x="attr-fe-autocomplete">autocomplete</code> attribute controls how the user agent provides autofill behavior.
  </p>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLInputElement"><code
  data-x="dom-input-indeterminate">indeterminate</code></dfn> IDL attribute must initially be set to
  false. On getting, it must return the last value it was set to. On setting, it must be set to the
  new value. It has no effect except for changing the appearance of <span
  data-x="attr-input-type-checkbox">checkbox</span> controls.</p>

  <p>The <dfn attribute for="HTMLInputElement"><code data-x="dom-input-accept">accept</code></dfn>,
  <dfn attribute for="HTMLInputElement"><code data-x="dom-input-alpha">alpha</code></dfn>, <dfn
  attribute for="HTMLInputElement"><code data-x="dom-input-alt">alt</code></dfn>, <dfn attribute
  for="HTMLInputElement"><code data-x="dom-input-max">max</code></dfn>, <dfn attribute
  for="HTMLInputElement"><code data-x="dom-input-min">min</code></dfn>, <dfn attribute
  for="HTMLInputElement"><code data-x="dom-input-multiple">multiple</code></dfn>, <dfn attribute
  for="HTMLInputElement"><code data-x="dom-input-pattern">pattern</code></dfn>, <dfn attribute
  for="HTMLInputElement"><code data-x="dom-input-placeholder">placeholder</code></dfn>, <dfn
  attribute for="HTMLInputElement"><code data-x="dom-input-required">required</code></dfn>, <dfn
  attribute for="HTMLInputElement"><code data-x="dom-input-size">size</code></dfn>, <dfn attribute
  for="HTMLInputElement"><code data-x="dom-input-src">src</code></dfn>, and <dfn attribute
  for="HTMLInputElement"><code data-x="dom-input-step">step</code></dfn> IDL attributes must
  <span>reflect</span> the respective content attributes of the same name. The <dfn attribute
  for="HTMLInputElement"><code data-x="dom-input-dirName">dirName</code></dfn> IDL attribute must
  <span>reflect</span> the <code data-x="attr-fe-dirname">dirname</code> content attribute. The <dfn
  attribute for="HTMLInputElement"><code data-x="dom-input-readOnly">readOnly</code></dfn> IDL
  attribute must <span>reflect</span> the <code data-x="attr-input-readonly">readonly</code> content
  attribute. The <dfn attribute for="HTMLInputElement"><code
  data-x="dom-input-defaultChecked">defaultChecked</code></dfn> IDL attribute must
  <span>reflect</span> the <code data-x="attr-input-checked">checked</code> content attribute. The
  <dfn attribute for="HTMLInputElement"><code
  data-x="dom-input-defaultValue">defaultValue</code></dfn> IDL attribute must <span>reflect</span>
  the <code data-x="attr-input-value">value</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLInputElement"><code
  data-x="dom-input-colorSpace">colorSpace</code></dfn> IDL attribute must <span>reflect</span> the
  <code data-x="attr-input-colorspace">colorspace</code> content attribute, <span>limited to only
  known values</span>. The <dfn attribute for="HTMLInputElement"><code
  data-x="dom-input-type">type</code></dfn> IDL attribute must <span>reflect</span> the respective
  content attribute of the same name, <span>limited to only known values</span>. The <dfn attribute
  for="HTMLInputElement"><code data-x="dom-input-maxLength">maxLength</code></dfn> IDL attribute
  must <span>reflect</span> the <code data-x="attr-input-maxlength">maxlength</code> content
  attribute, <span>limited to only non-negative numbers</span>.  The <dfn
  attribute for="HTMLInputElement"><code data-x="dom-input-minLength">minLength</code></dfn> IDL
  attribute must <span>reflect</span> the <code data-x="attr-input-minlength">minlength</code>
  content attribute, <span>limited to only non-negative numbers</span>.</p>

  <p>The IDL attributes <dfn attribute for="HTMLInputElement"><code
  data-x="dom-input-width">width</code></dfn> and <dfn attribute for="HTMLInputElement"><code
  data-x="dom-input-height">height</code></dfn> must return the rendered width and height of the
  image, in <span data-x="'px'">CSS pixels</span>, if an image is <span>being rendered</span>; or
  else the <span data-x="natural dimensions">natural width and height</span> of the image, in
  <span data-x="'px'">CSS pixels</span>, if an image is <i
  data-x="input-img-available">available</i> but not <span>being rendered</span>; or else 0,
  if no image is <i data-x="input-img-available">available</i>. When the <code>input</code>
  element's <code data-x="attr-input-type">type</code> attribute is not in the <span
  data-x="attr-input-type-image">Image Button</span> state, then no image is <i
  data-x="input-img-available">available</i>. <ref>CSS</ref></p>

  <p>On setting, they must act as if they <span data-x="reflect">reflected</span> the respective
  content attributes of the same name.</p>

  <p>The <code data-x="dom-cva-willValidate">willValidate</code>, <code
  data-x="dom-cva-validity">validity</code>, and <code
  data-x="dom-cva-validationMessage">validationMessage</code> IDL attributes, and the <code
  data-x="dom-cva-checkValidity">checkValidity()</code>, <code
  data-x="dom-cva-reportValidity">reportValidity()</code>, and <code
  data-x="dom-cva-setCustomValidity">setCustomValidity()</code> methods, are part of the
  <span>constraint validation API</span>. The <code data-x="dom-lfe-labels">labels</code> IDL
  attribute provides a list of the element's <code>label</code>s. The <code
  data-x="dom-textarea/input-select">select()</code>, <code
  data-x="dom-textarea/input-selectionStart">selectionStart</code>, <code
  data-x="dom-textarea/input-selectionEnd">selectionEnd</code>, <code
  data-x="dom-textarea/input-selectionDirection">selectionDirection</code>, <code
  data-x="dom-textarea/input-setRangeText">setRangeText()</code>, and <code
  data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code> methods and IDL attributes
  expose the element's text selection. The <code data-x="dom-fe-disabled">disabled</code>,
  <code data-x="dom-fae-form">form</code>, and <code data-x="dom-fe-name">name</code> IDL attributes
  are part of the element's forms API.</p>

  </div>



  <h5>States of the <code data-x="attr-input-type">type</code> attribute</h5>


  <h6><dfn element-state for="input" data-x="attr-input-type-hidden">Hidden</dfn> state (<code data-x="">type=hidden</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-hidden">Hidden</span> state, the rules in this section apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> a value that is not intended to be
  examined or manipulated by the user.</p>

  <div w-nodev>

  <p><strong>Constraint validation</strong>: If an <code>input</code> element's <code
  data-x="attr-input-type">type</code> attribute is in the <span
  data-x="attr-input-type-hidden">Hidden</span> state, it is <span>barred from constraint
  validation</span>.</p>

  </div>

  <p>If the <code data-x="attr-fe-name">name</code> attribute is present and has a value that is an
  <span>ASCII case-insensitive</span> match for "<code
  data-x="attr-fe-name-charset">_charset_</code>", then the element's <code
  data-x="attr-input-value">value</code> attribute must be omitted.</p>

  <div class="bookkeeping">

   <p>The
   <code data-x="attr-fe-autocomplete">autocomplete</code> and
   <code data-x="attr-fe-dirname">dirname</code>
   content attributes <span data-x="concept-input-apply">apply</span> to this element.</p>

   <p>The
   <code data-x="dom-input-value">value</code>
   IDL attribute <span data-x="concept-input-apply">applies</span> to this element and is
   in mode <span data-x="dom-input-value-default">default</span>.</p>

   <p>The following content attributes must not be specified and <span>do not
   apply</span> to the element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>,
   <code data-x="attr-input-size">size</code>,
   <code data-x="attr-input-src">src</code>,
   <code data-x="attr-input-step">step</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the
   element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>,
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>,
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code>,
   <code data-x="dom-input-stepDown">stepDown()</code>, and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

   <p>The <code
   data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span>do not apply</span>.</p>

  </div>


  <h6><dfn element-state for="input" data-x="attr-input-type-text">Text</dfn> (<code data-x="">type=text</code>) state and <dfn element-state for="input" data-x="attr-input-type-search">Search</dfn> state (<code data-x="">type=search</code>)</h6>

  <!-- v2 idea: applying input masks to <input>, e.g. for entering
       data with slashes and dashes (ack Greg Kilwein)
  -->

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-text">Text</span> state or the <span
  data-x="attr-input-type-search">Search</span> state, the rules in this section apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> a one line plain text edit control for
  the element's <span data-x="concept-fe-value">value</span>.</p>

  <p class="note">The difference between the <span data-x="attr-input-type-text">Text</span> state
  and the <span data-x="attr-input-type-search">Search</span> state is primarily stylistic: on
  platforms where search controls are distinguished from regular text controls, the <span
  data-x="attr-input-type-search">Search</span> state might result in an appearance consistent with
  the platform's search controls rather than appearing like a regular text control.</p>

  <div w-nodev>

  <p>If the element is <i data-x="concept-fe-mutable">mutable</i>, its <span
  data-x="concept-fe-value">value</span> should be editable by the user. User agents must not allow
  users to insert U+000A LINE FEED (LF) or U+000D CARRIAGE RETURN (CR) characters into the element's
  <span data-x="concept-fe-value">value</span>.</p>

  <!-- this next bit is also in the <textarea> section -->
  <!-- and something similar is in the session history section -->
  <p>If the element is <i data-x="concept-fe-mutable">mutable</i>, the user agent should allow the
  user to change the writing direction of the element, setting it either to a left-to-right writing
  direction or a right-to-left writing direction. If the user does so, the user agent must then run
  the following steps:</p>

  <ol>
   <li><p>Set the element's <code data-x="attr-dir">dir</code> attribute to "<code
   data-x="attr-dir-ltr">ltr</code>" if the user selected a left-to-right writing direction, and
   "<code data-x="attr-dir-rtl">rtl</code>" if the user selected a right-to-left writing
   direction.</p></li>

   <li><p><span>Queue an element task</span> on the <span>user interaction task source</span> given
   the element to <span data-x="concept-event-fire">fire an event</span> named <code
   data-x="event-input">input</code> at the element, with the <code
   data-x="dom-Event-bubbles">bubbles</code> and <code data-x="dom-Event-composed">composed</code>
   attributes initialized to true.</p></li>
  </ol>

  </div>

  <p>The <code data-x="attr-input-value">value</code> attribute, if specified, must have a value that
  contains no U+000A LINE FEED (LF) or U+000D CARRIAGE RETURN (CR) characters.</p>

  <div w-nodev>

  <p><strong>The <span>value sanitization algorithm</span> is as follows</strong>: <span>Strip
  newlines</span> from the <span data-x="concept-fe-value">value</span>.</p>

  </div>

  <div class="bookkeeping">

   <p>The following common <code>input</code> element content
   attributes, IDL attributes, and methods <span data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>, and
   <code data-x="attr-input-size">size</code> content attributes;
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>,
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>, and
   <code data-x="dom-input-value">value</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>, and
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code> methods.</p>

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is
   in mode <span data-x="dom-input-value-value">value</span>.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span data-x="concept-input-apply">apply</span>.</p>

   <p>The following content attributes must not be specified and <span>do not
   apply</span> to the element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-src">src</code>,
   <code data-x="attr-input-step">step</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the
   element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-input-stepDown">stepDown()</code> and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

  </div>


  <h6><dfn element-state for="input" data-x="attr-input-type-tel">Telephone</dfn> state (<code data-x="">type=tel</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-tel">Telephone</span> state, the rules in this section apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> a control for editing a telephone number
  given in the element's <span data-x="concept-fe-value">value</span>.</p>

  <div w-nodev>

  <p>If the element is <i data-x="concept-fe-mutable">mutable</i>, its <span
  data-x="concept-fe-value">value</span> should be editable by the user. User agents may change the
  spacing and, with care, the punctuation of <span data-x="concept-fe-value">values</span> that the
  user enters. User agents must not allow users to insert U+000A LINE FEED (LF) or U+000D CARRIAGE
  RETURN (CR) characters into the element's <span data-x="concept-fe-value">value</span>.</p>

  </div>

  <p>The <code data-x="attr-input-value">value</code> attribute, if specified, must have a value that
  contains no U+000A LINE FEED (LF) or U+000D CARRIAGE RETURN (CR) characters.</p>

  <div w-nodev>

  <p><strong>The <span>value sanitization algorithm</span> is as follows</strong>: <span>Strip
  newlines</span> from the <span data-x="concept-fe-value">value</span>.</p>

  </div>

  <p class="note">Unlike the <span data-x="attr-input-type-url">URL</span> and <span
  data-x="attr-input-type-email">Email</span> types, the <span
  data-x="attr-input-type-tel">Telephone</span> type does not enforce a particular syntax. This is
  intentional; in practice, telephone number fields tend to be free-form fields, because there are a
  wide variety of valid phone numbers. Systems that need to enforce a particular format are
  encouraged to use the <code data-x="attr-input-pattern">pattern</code> attribute or the <code
  data-x="dom-cva-setCustomValidity">setCustomValidity()</code> method to hook into the client-side
  validation mechanism.</p>


  <div class="bookkeeping">

   <p>The following common <code>input</code> element content
   attributes, IDL attributes, and methods <span data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>, and
   <code data-x="attr-input-size">size</code> content attributes;
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>,
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>, and
   <code data-x="dom-input-value">value</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>, and
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code> methods.</p>

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is
   in mode <span data-x="dom-input-value-value">value</span>.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span data-x="concept-input-apply">apply</span>.</p>

   <p>The following content attributes must not be specified and <span>do not
   apply</span> to the element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-src">src</code>,
   <code data-x="attr-input-step">step</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the
   element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-input-stepDown">stepDown()</code> and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

  </div>


  <h6><dfn element-state for="input" data-x="attr-input-type-url">URL</dfn> state (<code data-x="">type=url</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-url">URL</span> state, the rules in this section apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> a control for editing a single
  <span>absolute URL</span> given in the element's <span data-x="concept-fe-value">value</span>.</p>

  <div w-nodev>

  <p>If the element is <i data-x="concept-fe-mutable">mutable</i>, the user agent should allow the
  user to change the URL represented by its <span data-x="concept-fe-value">value</span>. User agents
  may allow the user to set the <span data-x="concept-fe-value">value</span> to a string that is not
  a <span data-x="valid URL string">valid</span> <span>absolute URL</span>, but may also or instead
  automatically escape characters entered by the user so that the <span
  data-x="concept-fe-value">value</span> is always a <span data-x="valid URL string">valid</span>
  <span>absolute URL</span> (even if that isn't the actual value seen and edited by the user in the
  interface). User agents should allow the user to set the <span
  data-x="concept-fe-value">value</span> to the empty string. User agents must not allow users to
  insert U+000A LINE FEED (LF) or U+000D CARRIAGE RETURN (CR) characters into the <span
  data-x="concept-fe-value">value</span>.</p>

  </div>

  <p>The <code data-x="attr-input-value">value</code> attribute, if specified and not empty, must
  have a value that is a <span>valid URL potentially surrounded by spaces</span> that is also an
  <span>absolute URL</span>.</p>

  <div w-nodev>

  <p><strong>The <span>value sanitization algorithm</span> is as follows</strong>: <span>Strip
  newlines</span> from the <span data-x="concept-fe-value">value</span>, then <span>strip leading
  and trailing ASCII whitespace</span> from the <span data-x="concept-fe-value">value</span>.</p>

  <p><strong>Constraint validation</strong>: While the <span data-x="concept-fe-value">value</span>
  of the element is neither the empty string nor a <span data-x="valid URL string">valid</span>
  <span>absolute URL</span>, the element is <span>suffering from a type mismatch</span>.</p>

  </div>

  <div class="bookkeeping">

   <p>The following common <code>input</code> element content
   attributes, IDL attributes, and methods <span data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>, and
   <code data-x="attr-input-size">size</code> content attributes;
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>,
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>, and
   <code data-x="dom-input-value">value</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>, and
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code> methods.</p>

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is
   in mode <span data-x="dom-input-value-value">value</span>.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span data-x="concept-input-apply">apply</span>.</p>

   <p>The following content attributes must not be specified and <span>do not
   apply</span> to the element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-src">src</code>,
   <code data-x="attr-input-step">step</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the
   element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-input-stepDown">stepDown()</code> and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

  </div>

  <div class="example">

   <p>If a document contained the following markup:</p>

   <pre><code class="html">&lt;input type="url" name="location" list="urls"&gt;
&lt;datalist id="urls"&gt;
 &lt;option label="MIME: Format of Internet Message Bodies" value="https://www.rfc-editor.org/rfc/rfc2045"&gt;
 &lt;option label="HTML" value="https://html.spec.whatwg.org/"&gt;
 &lt;option label="DOM" value="https://dom.spec.whatwg.org/"&gt;
 &lt;option label="Fullscreen" value="https://fullscreen.spec.whatwg.org/"&gt;
 &lt;option label="Media Session" value="https://mediasession.spec.whatwg.org/"&gt;
 &lt;option label="The Single UNIX Specification, Version 3" value="http://www.unix.org/version3/"&gt;
&lt;/datalist&gt;</code></pre>

   <p>...and the user had typed "<kbd>spec.w</kbd>", and the user agent had also found that the user
   had visited <code data-x="">https://url.spec.whatwg.org/#url-parsing</code> and <code
   data-x="">https://streams.spec.whatwg.org/</code> in the recent past, then the rendering might look
   like this:</p>

   <p><img src="/images/sample-url.svg" width="480" height="150" alt="A text box with an icon on the left followed by the text &quot;spec.w&quot; and a cursor, with a drop down button on the right hand side; with, below, a drop down box containing a list of six URLs on the left, with the first four having grayed out labels on the right; and a scrollbar to the right of the drop down box, indicating further values are available." class="darkmode-aware"></p>

   <p>The first four URLs in this sample consist of the four URLs in the author-specified list that
   match the text the user has entered, sorted in some <span>implementation-defined</span> manner
   (maybe by how frequently the user refers to those URLs). Note how the UA is using the knowledge
   that the values are URLs to allow the user to omit the scheme part and perform intelligent
   matching on the domain name.</p>

   <p>The last two URLs (and probably many more, given the scrollbar's indications of more values
   being available) are the matches from the user agent's session history data. This data is not
   made available to the page DOM. In this particular case, the UA has no titles to provide for
   those values.</p>

  </div>


  <h6><dfn element-state for="input" data-x="attr-input-type-email">Email</dfn> state (<code data-x="">type=email</code>)<span
  id="e-mail-state-(type=email)"></span></h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-email">Email</span> state, the rules in this section apply.</p>

  </div>

  <p>How the <span data-x="attr-input-type-email">Email</span> state operates depends on whether the
  <code data-x="attr-input-multiple">multiple</code> attribute is specified or not.</p>

  <dl class="switch">

   <dt>When the <code data-x="attr-input-multiple">multiple</code> attribute is not specified on the
   element</dt>

   <dd>
    <p>The <code>input</code> element <span>represents</span> a control for editing an email
    address given in the element's <span data-x="concept-fe-value">value</span>.</p>

    <div w-nodev>

    <p>If the element is <i data-x="concept-fe-mutable">mutable</i>, the user agent should allow the
    user to change the email address represented by its <span
    data-x="concept-fe-value">value</span>. User agents may allow the user to set the <span
    data-x="concept-fe-value">value</span> to a string that is not a <span>valid email
    address</span>. The user agent should act in a manner consistent with expecting the user to
    provide a single email address. User agents should allow the user to set the <span
    data-x="concept-fe-value">value</span> to the empty string. User agents must not allow users to
    insert U+000A LINE FEED (LF) or U+000D CARRIAGE RETURN (CR) characters into the <span
    data-x="concept-fe-value">value</span>. User agents may transform the <span
    data-x="concept-fe-value">value</span> for display and editing; in particular, user agents should
    convert punycode in the domain labels of the <span data-x="concept-fe-value">value</span> to IDN in the display and
    vice versa.</p>

    <p><strong>Constraint validation</strong>: While the user interface is representing input that
    the user agent cannot convert to punycode, the control is <span>suffering from bad
    input</span>.</p>

    </div>

    <p>The <code data-x="attr-input-value">value</code> attribute, if specified and not empty, must
    have a value that is a single <span>valid email address</span>.</p>

    <div w-nodev>

    <p><strong>The <span>value sanitization algorithm</span> is as follows</strong>: <span>Strip
    newlines</span> from the <span data-x="concept-fe-value">value</span>, then <span>strip leading
    and trailing ASCII whitespace</span> from the <span data-x="concept-fe-value">value</span>.</p>

    <p><strong>Constraint validation</strong>: While the <span data-x="concept-fe-value">value</span>
    of the element is neither the empty string nor a single <span>valid email address</span>, the
    element is <span>suffering from a type mismatch</span>.</p>

    </div>
   </dd>

   <dt>When the <code data-x="attr-input-multiple">multiple</code> attribute <em>is</em> specified on
   the element</dt>

   <dd>
    <p>The <code>input</code> element <span>represents</span> a control for adding, removing, and
    editing the email addresses given in the element's <span
    data-x="concept-fe-values">value<em>s</em></span>.</p>

    <div w-nodev>

    <p>If the element is <i data-x="concept-fe-mutable">mutable</i>, the user agent should allow the
    user to add, remove, and edit the email addresses represented by its <span
    data-x="concept-fe-values">values</span>. User agents may allow the user to set any
    individual value in the list of <span data-x="concept-fe-value">value<em>s</em></span> to a
    string that is not a <span>valid email address</span>, but must not allow users to set any
    individual value to a string containing U+002C COMMA (,), U+000A LINE FEED (LF), or U+000D
    CARRIAGE RETURN (CR) characters. User agents should allow the user to remove all the addresses
    in the element's <span data-x="concept-fe-values">values</span>. User agents may
    transform the <span data-x="concept-fe-values">values</span> for display and editing; in
    particular, user agents should convert punycode in the domain labels of the <span
    data-x="concept-fe-value">value</span> to IDN in the display and vice versa.</p>

    <p><strong>Constraint validation</strong>: While the user interface describes a situation where
    an individual value contains a U+002C COMMA (,) or is representing input that the user agent
    cannot convert to punycode, the control is <span>suffering from bad input</span>.</p>

    <p>Whenever the user changes the element's <span
    data-x="concept-fe-value">value<em>s</em></span>, the user agent must run the following
    steps:</p>

    <ol>
     <li><p>Let <var>latest values</var> be a copy of the element's <span
     data-x="concept-fe-value">value<em>s</em></span>.</p></li>

     <!-- It's a copy because /values/ might include leading and trailing spaces that we don't
     necessarily want to remove from the UI but that we do want to remove before serializing. -->

     <li><p><span>Strip leading and trailing ASCII whitespace</span> from each value in <var>latest
     values</var>.</p></li>

     <li><p>Let the element's <span data-x="concept-fe-value">value</span> be the result of
     concatenating all the values in <var>latest values</var>, separating each value from
     the next by a single U+002C COMMA character (,), maintaining the list's order.</p></li>
    </ol>

    </div>

    <p>The <code data-x="attr-input-value">value</code> attribute, if specified, must have a value
    that is a <span>valid email address list</span>.</p>

    <div w-nodev>

    <p><strong>The <span>value sanitization algorithm</span> is as follows</strong>:</p>

    <ol>
     <li><p><span data-x="split a string on commas">Split on commas</span> the element's <span
     data-x="concept-fe-value">value</span>, <span>strip leading and trailing ASCII
     whitespace</span> from each resulting token, if any, and let the element's <span
     data-x="concept-fe-values">values</span> be the (possibly empty) resulting list of
     (possibly empty) tokens, maintaining the original order.</p></li>

     <li><p>Let the element's <span data-x="concept-fe-value">value</span> be the result of
     concatenating the element's <span data-x="concept-fe-values">values</span>, separating
     each value from the next by a single U+002C COMMA character (,), maintaining the list's
     order.</p></li>
    </ol>

    <p><strong>Constraint validation</strong>: While the <span data-x="concept-fe-value">value</span>
    of the element is not a <span>valid email address list</span>, the element is <span>suffering
    from a type mismatch</span>.</p>

    </div>
   </dd>
  </dl>

  <p>When the <code data-x="attr-input-multiple">multiple</code> attribute is set or removed, the
  user agent must run the <span>value sanitization algorithm</span>.</p>

  <p>A <dfn id="valid-e-mail-address">valid email address</dfn> is a string that matches the <code
  data-x="">email</code> production of the following ABNF, the character set for which is Unicode.
  This ABNF implements the extensions described in RFC 1123. <ref>ABNF</ref> <ref>RFC5322</ref>
  <ref>RFC1034</ref> <ref>RFC1123</ref></p>

  <pre><code data-x="" class="abnf">email         = 1*( atext / "." ) "@" label *( "." label )
label         = let-dig [ [ ldh-str ] let-dig ]  ; limited to a length of 63 characters by <a href="https://www.rfc-editor.org/rfc/rfc1034#section-3.5">RFC 1034 section 3.5</a>
atext         = &lt; as defined in <a href="https://www.rfc-editor.org/rfc/rfc5322#section-3.2.3">RFC 5322 section 3.2.3</a> >
let-dig       = &lt; as defined in <a href="https://www.rfc-editor.org/rfc/rfc1034#section-3.5">RFC 1034 section 3.5</a> >
ldh-str       = &lt; as defined in <a href="https://www.rfc-editor.org/rfc/rfc1034#section-3.5">RFC 1034 section 3.5</a> ></code></pre>

  <!-- Domain syntax based on section 3.5 of [RFC1034] and section 2.1 of [RFC1123] -->

  <p class="note">This requirement is a <span>willful violation</span> of RFC 5322, which defines a
  syntax for email addresses that is simultaneously too strict (before the "@" character), too
  vague (after the "@" character), and too lax (allowing comments, whitespace characters, and quoted
  strings in manners unfamiliar to most users) to be of practical use here.</p>

  <div class="note">

   <p>The following JavaScript- and Perl-compatible regular expression is an implementation of the
   above definition.</p>

   <pre>/^[a-zA-Z0-9.!#$%&amp;'*+\/=?^_`{|}~-]+@[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?(?:\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)*$/</pre>

   <!-- based on: https://blog.gerv.net/2011/05/html5_email_address_regexp/ -->

  </div>

  <p>A <dfn id="valid-e-mail-address-list">valid email address list</dfn> is a <span>set of
  comma-separated tokens</span>, where each token is itself a <span>valid email address</span>.
  <span w-nodev>To obtain the list of tokens from a <span>valid email address list</span>, an
  implementation must <span data-x="split a string on commas">split the string on
  commas</span>.</span></p>

  <div class="bookkeeping">

   <p>The following common <code>input</code> element content
   attributes, IDL attributes, and methods <span data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>, and
   <code data-x="attr-input-size">size</code> content attributes;
   <code data-x="dom-input-list">list</code> and
   <code data-x="dom-input-value">value</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code> method.</p>

<!--
   <p>The following common <code>input</code> element content
   attributes, IDL attributes, and methods <span data-x="concept-input-apply">apply</span> to the element when
   the <code data-x="attr-input-multiple">multiple</code> attribute is
   not specified:
   .</p>
-->

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is
   in mode <span data-x="dom-input-value-value">value</span>.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span data-x="concept-input-apply">apply</span>.</p>

   <p>The following content attributes must not be specified and <span>do not
   apply</span> to the element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-src">src</code>,
   <code data-x="attr-input-step">step</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the
   element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>,
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>,
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code>,
   <code data-x="dom-input-stepDown">stepDown()</code> and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

  </div>


  <h6><dfn element-state for="input" data-x="attr-input-type-password">Password</dfn> state (<code data-x="">type=password</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-password">Password</span> state, the rules in this section
  apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> a one line plain text edit control for
  the element's <span data-x="concept-fe-value">value</span>. The user agent should obscure the value
  so that people other than the user cannot see it.</p>

  <div w-nodev>

  <p>If the element is <i data-x="concept-fe-mutable">mutable</i>, its <span
  data-x="concept-fe-value">value</span> should be editable by the user. User agents must not allow
  users to insert U+000A LINE FEED (LF) or U+000D CARRIAGE RETURN (CR) characters into the <span
  data-x="concept-fe-value">value</span>.</p>

  </div>

  <p>The <code data-x="attr-input-value">value</code> attribute, if specified, must have a value that
  contains no U+000A LINE FEED (LF) or U+000D CARRIAGE RETURN (CR) characters.</p>

  <div w-nodev>

  <p><strong>The <span>value sanitization algorithm</span> is as follows</strong>: <span>Strip
  newlines</span> from the <span data-x="concept-fe-value">value</span>.</p>

  </div>

  <div class="bookkeeping">

   <p>The following common <code>input</code> element content
   attributes, IDL attributes, and methods <span data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>, and
   <code data-x="attr-input-size">size</code> content attributes;
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>,
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>, and
   <code data-x="dom-input-value">value</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>, and
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code> methods.</p>

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is
   in mode <span data-x="dom-input-value-value">value</span>.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span data-x="concept-input-apply">apply</span>.</p>

   <p>The following content attributes must not be specified and <span>do not
   apply</span> to the element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-src">src</code>,
   <code data-x="attr-input-step">step</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the
   element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-input-stepDown">stepDown()</code> and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

  </div>


  <h6 id="date-state-(type=date)"><dfn element-state for="input" data-x="attr-input-type-date">Date</dfn> state (<code data-x="">type=date</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-date">Date</span> state, the rules in this section apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> a control for setting the element's
  <span data-x="concept-fe-value">value</span> to a string representing a specific <span
  data-x="concept-date">date</span>.</p>

  <div w-nodev>

  <p>If the element is <i data-x="concept-fe-mutable">mutable</i>, the user agent should allow the
  user to change the <span data-x="concept-date">date</span> represented by its <span
  data-x="concept-fe-value">value</span>, as obtained by <span data-x="parse a date string">parsing a
  date</span> from it. User agents must not allow the user to set the <span
  data-x="concept-fe-value">value</span> to a non-empty string that is not a <span>valid date
  string</span>. If the user agent provides a user interface for selecting a <span
  data-x="concept-date">date</span>, then the <span data-x="concept-fe-value">value</span> must be set
  to a <span>valid date string</span> representing the user's selection. User agents should allow
  the user to set the <span data-x="concept-fe-value">value</span> to the empty string.</p>

  <p><strong>Constraint validation</strong>: While the user interface describes input that the user
  agent cannot convert to a <span>valid date string</span>, the control is <span>suffering from bad
  input</span>.</p>

  </div>

  <p class="note">See the <a href="#input-author-notes">introduction section</a> for a discussion of
  the difference between the input format and submission format for date, time, and number form
  controls<span w-nodev>, and the <a href="#input-impl-notes">implementation notes</a>
  regarding localization of form controls</span>.</p>

  <p>The <code data-x="attr-input-value">value</code> attribute, if specified and not empty, must
  have a value that is a <span>valid date string</span>.</p> <!-- ok to set out-of-range value, we
  never know when we might have to represent bogus input -->

  <div w-nodev>

  <p><strong>The <span>value sanitization algorithm</span> is as follows</strong>: If the <span
  data-x="concept-fe-value">value</span> of the element is not a <span>valid date string</span>, then
  set it to the empty string instead.</p>

  </div>

  <p>The <code data-x="attr-input-min">min</code> attribute, if specified, must have a value that is
  a <span>valid date string</span>. The <code data-x="attr-input-max">max</code> attribute, if
  specified, must have a value that is a <span>valid date string</span>.</p>

  <p>The <code data-x="attr-input-step">step</code> attribute is expressed in days. <span
  w-nodev>The <span data-x="concept-input-step-scale">step scale factor</span> is 86,400,000
  (which converts the days to milliseconds, as used in the other algorithms).</span> The <span
  data-x="concept-input-step-default">default step</span> is 1 day.</p>

  <div w-nodev>

  <p>When the element is <span>suffering from a step mismatch</span>, the user agent may round the
  element's <span data-x="concept-fe-value">value</span> to the nearest <span
  data-x="concept-date">date</span> for which the element would not <span data-x="suffering from a
  step mismatch">suffer from a step mismatch</span>.</p>

  <p><strong>The <span data-x="concept-input-value-string-number">algorithm to convert a string to a
  number</span>, given a string <var>input</var>, is as follows</strong>: If <span
  data-x="parse a date string">parsing a date</span> from <var>input</var> results in an
  error, then return an error; otherwise, return the number of milliseconds elapsed from midnight
  UTC on the morning of 1970-01-01 (the time represented by the value "<code
  data-x="">1970-01-01T00:00:00.0Z</code>") to midnight UTC on the morning of the parsed <span
  data-x="concept-date">date</span>, ignoring leap seconds.</p>

  <p><strong>The <span data-x="concept-input-value-number-string">algorithm to convert a number to a
  string</span>, given a number <var>input</var>, is as follows</strong>: Return a
  <span>valid date string</span> that represents the <span data-x="concept-date">date</span> that, in
  UTC, is current <var>input</var> milliseconds after midnight UTC on the morning of
  1970-01-01 (the time represented by the value "<code data-x="">1970-01-01T00:00:00.0Z</code>").</p>

  <p><strong>The <span data-x="concept-input-value-string-date">algorithm to convert a string to a
  <code>Date</code> object</span>, given a string <var>input</var>, is as follows</strong>:
  If <span data-x="parse a date string">parsing a date</span> from <var>input</var> results
  in an error, then return an error; otherwise, return <span data-x="create a Date object">a new
  <code>Date</code> object</span> representing midnight UTC on the morning of the parsed <span
  data-x="concept-date">date</span>.</p>

  <p><strong>The <span data-x="concept-input-value-date-string">algorithm to convert a
  <code>Date</code> object to a string</span>, given a <code>Date</code> object <var>input</var>, is
  as follows</strong>: Return a <span>valid date string</span> that represents the <span
  data-x="concept-date">date</span> current at the time represented by <var>input</var> in the UTC
  time zone.</p>

  </div>

  <div class="note" id="only-contemporary-times">

   <p>The <span data-x="attr-input-type-date">Date</span> state (and other date- and time-related
   states described in subsequent sections) is not intended for the entry of values for which a
   precise date and time relative to the contemporary calendar cannot be established. For example,
   it would be inappropriate for the entry of times like "one millisecond after the big bang", "the
   early part of the Jurassic period", or "a winter around 250 BCE".</p>

   <p>For the input of dates before the introduction of the Gregorian calendar, authors are
   encouraged to not use the <span data-x="attr-input-type-date">Date</span> state (and the other
   date- and time-related states described in subsequent sections), as user agents are not required
   to support converting dates and times from earlier periods to the Gregorian calendar, and asking
   users to do so manually puts an undue burden on users. (This is complicated by the manner in
   which the Gregorian calendar was phased in, which occurred at different times in different
   countries, ranging from partway through the 16th century all the way to early in the 20th.)
   Instead, authors are encouraged to provide fine-grained input controls using the
   <code>select</code> element and <code>input</code> elements with the <span
   data-x="attr-input-type-number">Number</span> state.</p>

  </div>

  <div class="bookkeeping">

   <p>The following common <code>input</code> element content
   attributes, IDL attributes, and methods <span data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>, and
   <code data-x="attr-input-step">step</code> content attributes;
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-input-value">value</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-input-stepDown">stepDown()</code>, and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is
   in mode <span data-x="dom-input-value-value">value</span>.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span data-x="concept-input-apply">apply</span>.</p>

   <p>The following content attributes must not be specified and <span>do not
   apply</span> to the element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-size">size</code>,
   <code data-x="attr-input-src">src</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the
   element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>, and
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code> IDL attributes;
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>, and
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code> methods.</p>

  </div>


  <h6><dfn element-state for="input" data-x="attr-input-type-month">Month</dfn> state (<code data-x="">type=month</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-month">Month</span> state, the rules in this section apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> a control for setting the element's
  <span data-x="concept-fe-value">value</span> to a string representing a specific <span
  data-x="concept-month">month</span>.</p>

  <div w-nodev>

  <p>If the element is <i data-x="concept-fe-mutable">mutable</i>, the user agent should allow the
  user to change the <span data-x="concept-month">month</span> represented by its <span
  data-x="concept-fe-value">value</span>, as obtained by <span data-x="parse a month string">parsing a
  month</span> from it. User agents must not allow the user to set the <span
  data-x="concept-fe-value">value</span> to a non-empty string that is not a <span>valid month
  string</span>. If the user agent provides a user interface for selecting a <span
  data-x="concept-month">month</span>, then the <span data-x="concept-fe-value">value</span> must be
  set to a <span>valid month string</span> representing the user's selection. User agents should
  allow the user to set the <span data-x="concept-fe-value">value</span> to the empty string.</p>

  <p><strong>Constraint validation</strong>: While the user interface describes input that the user
  agent cannot convert to a <span>valid month string</span>, the control is <span>suffering from bad
  input</span>.</p>

  </div>

  <p class="note">See the <a href="#input-author-notes">introduction section</a> for a discussion of
  the difference between the input format and submission format for date, time, and number form
  controls<span w-nodev>, and the <a href="#input-impl-notes">implementation notes</a>
  regarding localization of form controls</span>.</p>

  <p>The <code data-x="attr-input-value">value</code> attribute, if specified and not empty, must
  have a value that is a <span>valid month string</span>.</p> <!-- ok to set out-of-range value, we
  never know when we might have to represent bogus input -->

  <div w-nodev>

  <p><strong>The <span>value sanitization algorithm</span> is as follows</strong>: If the <span
  data-x="concept-fe-value">value</span> of the element is not a <span>valid month string</span>,
  then set it to the empty string instead.</p>

  </div>

  <p>The <code data-x="attr-input-min">min</code> attribute, if specified, must have a value that is
  a <span>valid month string</span>. The <code data-x="attr-input-max">max</code> attribute, if
  specified, must have a value that is a <span>valid month string</span>.</p>

  <p>The <code data-x="attr-input-step">step</code> attribute is expressed in months. <span
  w-nodev>The <span data-x="concept-input-step-scale">step scale factor</span> is 1 (there is no
  conversion needed as the algorithms use months).</span> The <span
  data-x="concept-input-step-default">default step</span> is 1 month.</p>

  <div w-nodev>

  <p>When the element is <span>suffering from a step mismatch</span>, the user agent may round the
  element's <span data-x="concept-fe-value">value</span> to the nearest <span
  data-x="concept-month">month</span> for which the element would not <span data-x="suffering from a
  step mismatch">suffer from a step mismatch</span>.</p>

  <p><strong>The <span data-x="concept-input-value-string-number">algorithm to convert a string to a
  number</span>, given a string <var>input</var>, is as follows</strong>: If <span
  data-x="parse a month string">parsing a month</span> from <var>input</var> results in an
  error, then return an error; otherwise, return the number of months between January 1970 and the
  parsed <span data-x="concept-month">month</span>.</p>

  <p><strong>The <span data-x="concept-input-value-number-string">algorithm to convert a number to a
  string</span>, given a number <var>input</var>, is as follows</strong>: Return a
  <span>valid month string</span> that represents the <span data-x="concept-month">month</span> that
  has <var>input</var> months between it and January 1970.</p>

  <!-- note - it doesn't matter exactly how many months are "between" two months, so long as the UA
  implements this consistently. The number is never actually exposed. -->

  <p><strong>The <span data-x="concept-input-value-string-date">algorithm to convert a string to a
  <code>Date</code> object</span>, given a string <var>input</var>, is as follows</strong>:
  If <span data-x="parse a month string">parsing a month</span> from <var>input</var>
  results in an error, then return an error; otherwise, return <span data-x="create a Date object">a
  new <code>Date</code> object</span> representing midnight UTC on the morning of the first day of
  the parsed <span data-x="concept-month">month</span>.</p>

  <p><strong>The <span data-x="concept-input-value-date-string">algorithm to convert a
  <code>Date</code> object to a string</span>, given a <code>Date</code> object <var>input</var>, is
  as follows</strong>: Return a <span>valid month string</span> that represents the <span
  data-x="concept-month">month</span> current at the time represented by <var>input</var> in the UTC
  time zone.</p>

  </div>

  <div class="bookkeeping">

   <p>The following common <code>input</code> element content
   attributes, IDL attributes, and methods <span data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>, and
   <code data-x="attr-input-step">step</code> content attributes;
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-input-value">value</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-input-stepDown">stepDown()</code>, and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is
   in mode <span data-x="dom-input-value-value">value</span>.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span data-x="concept-input-apply">apply</span>.</p>

   <p>The following content attributes must not be specified and <span>do not
   apply</span> to the element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-size">size</code>,
   <code data-x="attr-input-src">src</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the
   element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>, and
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code> IDL attributes;
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>, and
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code> methods.</p>

  </div>


  <h6><dfn element-state for="input" data-x="attr-input-type-week">Week</dfn> state (<code data-x="">type=week</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-week">Week</span> state, the rules in this section apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> a control for setting the element's
  <span data-x="concept-fe-value">value</span> to a string representing a specific <span
  data-x="concept-week">week</span>.</p>

  <div w-nodev>

  <p>If the element is <i data-x="concept-fe-mutable">mutable</i>, the user agent should allow the
  user to change the <span data-x="concept-week">week</span> represented by its <span
  data-x="concept-fe-value">value</span>, as obtained by <span data-x="parse a week string">parsing a
  week</span> from it. User agents must not allow the user to set the <span
  data-x="concept-fe-value">value</span> to a non-empty string that is not a <span>valid week
  string</span>. If the user agent provides a user interface for selecting a <span
  data-x="concept-week">week</span>, then the <span data-x="concept-fe-value">value</span> must be set
  to a <span>valid week string</span> representing the user's selection. User agents should allow
  the user to set the <span data-x="concept-fe-value">value</span> to the empty string.</p>

  <p><strong>Constraint validation</strong>: While the user interface describes input that the user
  agent cannot convert to a <span>valid week string</span>, the control is <span>suffering from bad
  input</span>.</p>

  </div>

  <p class="note">See the <a href="#input-author-notes">introduction section</a> for a discussion of
  the difference between the input format and submission format for date, time, and number form
  controls<span w-nodev>, and the <a href="#input-impl-notes">implementation notes</a>
  regarding localization of form controls</span>.</p>

  <p>The <code data-x="attr-input-value">value</code> attribute, if specified and not empty, must
  have a value that is a <span>valid week string</span>.</p> <!-- ok to set out-of-range value, we
  never know when we might have to represent bogus input -->

  <div w-nodev>

  <p><strong>The <span>value sanitization algorithm</span> is as follows</strong>: If the <span
  data-x="concept-fe-value">value</span> of the element is not a <span>valid week string</span>, then
  set it to the empty string instead.</p>

  </div>

  <p>The <code data-x="attr-input-min">min</code> attribute, if specified, must have a value that is
  a <span>valid week string</span>. The <code data-x="attr-input-max">max</code> attribute, if
  specified, must have a value that is a <span>valid week string</span>.</p>

  <p>The <code data-x="attr-input-step">step</code> attribute is expressed in weeks. <span
  w-nodev>The <span data-x="concept-input-step-scale">step scale factor</span> is 604,800,000
  (which converts the weeks to milliseconds, as used in the other algorithms).</span> The <span
  data-x="concept-input-step-default">default step</span> is 1 week. <span w-nodev>The <span
  data-x="concept-input-step-default-base">default step base</span> is &#x2212;259,200,000 (the start
  of week 1970-W01).</span></p>

  <div w-nodev>

  <p>When the element is <span>suffering from a step mismatch</span>, the user agent may round the
  element's <span data-x="concept-fe-value">value</span> to the nearest <span
  data-x="concept-week">week</span> for which the element would not <span data-x="suffering from a
  step mismatch">suffer from a step mismatch</span>.</p>

  <p><strong>The <span data-x="concept-input-value-string-number">algorithm to convert a string to a
  number</span>, given a string <var>input</var>, is as follows</strong>: If <span
  data-x="parse a week string">parsing a week string</span> from <var>input</var> results in
  an error, then return an error; otherwise, return the number of milliseconds elapsed from midnight
  UTC on the morning of 1970-01-01 (the time represented by the value "<code
  data-x="">1970-01-01T00:00:00.0Z</code>") to midnight UTC on the morning of the Monday of the
  parsed <span data-x="concept-week">week</span>, ignoring leap seconds.</p>

  <p><strong>The <span data-x="concept-input-value-number-string">algorithm to convert a number to a
  string</span>, given a number <var>input</var>, is as follows</strong>: Return a
  <span>valid week string</span> that represents the <span data-x="concept-week">week</span> that, in
  UTC, is current <var>input</var> milliseconds after midnight UTC on the morning of
  1970-01-01 (the time represented by the value "<code data-x="">1970-01-01T00:00:00.0Z</code>").</p>

  <p><strong>The <span data-x="concept-input-value-string-date">algorithm to convert a string to a
  <code>Date</code> object</span>, given a string <var>input</var>, is as follows</strong>: If <span
  data-x="parse a week string">parsing a week</span> from <var>input</var> results in an error, then
  return an error; otherwise, return <span data-x="create a Date object">a new <code>Date</code>
  object</span> representing midnight UTC on the morning of the Monday of the parsed <span
  data-x="concept-week">week</span>.</p>

  <p><strong>The <span data-x="concept-input-value-date-string">algorithm to convert a
  <code>Date</code> object to a string</span>, given a <code>Date</code> object <var>input</var>, is
  as follows</strong>: Return a <span>valid week string</span> that represents the <span
  data-x="concept-week">week</span> current at the time represented by <var>input</var> in the UTC
  time zone.</p>

  </div>

  <div class="bookkeeping">

   <p>The following common <code>input</code> element content attributes, IDL attributes, and
   methods <span data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>, and
   <code data-x="attr-input-step">step</code> content attributes;
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-input-value">value</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-input-stepDown">stepDown()</code>, and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is in mode <span
   data-x="dom-input-value-value">value</span>.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span
   data-x="concept-input-apply">apply</span>.</p>

   <p>The following content attributes must not be specified and <span>do not apply</span> to the
   element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-size">size</code>,
   <code data-x="attr-input-src">src</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>, and
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code> IDL attributes;
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>, and
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code> methods.</p>

  </div>


  <h6><dfn element-state for="input" data-x="attr-input-type-time">Time</dfn> state (<code data-x="">type=time</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-time">Time</span> state, the rules in this section apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> a control for setting the element's
  <span data-x="concept-fe-value">value</span> to a string representing a specific <span
  data-x="concept-time">time</span>.</p>

  <div w-nodev>

  <p>If the element is <i data-x="concept-fe-mutable">mutable</i>, the user agent should allow the
  user to change the <span data-x="concept-time">time</span> represented by its <span
  data-x="concept-fe-value">value</span>, as obtained by <span data-x="parse a time string">parsing a
  time</span> from it. User agents must not allow the user to set the <span
  data-x="concept-fe-value">value</span> to a non-empty string that is not a <span>valid time
  string</span>. If the user agent provides a user interface for selecting a <span
  data-x="concept-time">time</span>, then the <span data-x="concept-fe-value">value</span> must be set
  to a <span>valid time string</span> representing the user's selection. User agents should allow
  the user to set the <span data-x="concept-fe-value">value</span> to the empty string.</p>

  <p><strong>Constraint validation</strong>: While the user interface describes input that the user
  agent cannot convert to a <span>valid time string</span>, the control is <span>suffering from bad
  input</span>.</p>

  </div>

  <p class="note">See the <a href="#input-author-notes">introduction section</a> for a discussion of
  the difference between the input format and submission format for date, time, and number form
  controls<span w-nodev>, and the <a href="#input-impl-notes">implementation notes</a>
  regarding localization of form controls</span>.</p>

  <p>The <code data-x="attr-input-value">value</code> attribute, if specified and not empty, must
  have a value that is a <span>valid time string</span>.</p> <!-- ok to set out-of-range value, we
  never know when we might have to represent bogus input -->

  <div w-nodev>

  <p><strong>The <span>value sanitization algorithm</span> is as follows</strong>: If the <span
  data-x="concept-fe-value">value</span> of the element is not a <span>valid time string</span>, then
  set it to the empty string instead.</p>

  <p>The form control <span>has a periodic domain</span>.</p>

  </div>

  <p>The <code data-x="attr-input-min">min</code> attribute, if specified, must have a value that is
  a <span>valid time string</span>. The <code data-x="attr-input-max">max</code> attribute, if
  specified, must have a value that is a <span>valid time string</span>.</p>

  <p>The <code data-x="attr-input-step">step</code> attribute is expressed in seconds. <span
  w-nodev>The <span data-x="concept-input-step-scale">step scale factor</span> is 1000 (which
  converts the seconds to milliseconds, as used in the other algorithms).</span> The <span
  data-x="concept-input-step-default">default step</span> is 60 seconds.</p>

  <div w-nodev>

  <p>When the element is <span>suffering from a step mismatch</span>, the user agent may round the
  element's <span data-x="concept-fe-value">value</span> to the nearest <span
  data-x="concept-time">time</span> for which the element would not <span data-x="suffering from a
  step mismatch">suffer from a step mismatch</span>.</p>

  <p><strong>The <span data-x="concept-input-value-string-number">algorithm to convert a string to a
  number</span>, given a string <var>input</var>, is as follows</strong>: If <span
  data-x="parse a time string">parsing a time</span> from <var>input</var> results in an
  error, then return an error; otherwise, return the number of milliseconds elapsed from midnight to
  the parsed <span data-x="concept-time">time</span> on a day with no time changes.</p>

  <p><strong>The <span data-x="concept-input-value-number-string">algorithm to convert a number to a
  string</span>, given a number <var>input</var>, is as follows</strong>: Return a
  <span>valid time string</span> that represents the <span data-x="concept-time">time</span> that is
  <var>input</var> milliseconds after midnight on a day with no time changes.</p>

  <p><strong>The <span data-x="concept-input-value-string-date">algorithm to convert a string to a
  <code>Date</code> object</span>, given a string <var>input</var>, is as
  follows</strong>: If <span data-x="parse a time string">parsing a time</span> from
  <var>input</var> results
  in an error, then return an error; otherwise, return <span data-x="create a Date object">a new
  <code>Date</code> object</span> representing the parsed <span data-x="concept-time">time</span> in
  UTC on 1970-01-01.</p>

  <p><strong>The <span data-x="concept-input-value-date-string">algorithm to convert a
  <code>Date</code> object to a string</span>, given a <code>Date</code> object <var>input</var>, is
  as follows</strong>: Return a <span>valid time string</span> that represents the UTC <span
  data-x="concept-time">time</span> component that is represented by <var>input</var>.</p>

  </div>

  <div class="bookkeeping">

   <p>The following common <code>input</code> element content attributes, IDL attributes, and
   methods <span data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>, and
   <code data-x="attr-input-step">step</code> content attributes;
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-input-value">value</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-input-stepDown">stepDown()</code>, and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is in mode <span
   data-x="dom-input-value-value">value</span>.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span
   data-x="concept-input-apply">apply</span>.</p>

   <p>The following content attributes must not be specified and <span>do not apply</span> to the
   element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-size">size</code>,
   <code data-x="attr-input-src">src</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>, and
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code> IDL attributes;
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>, and
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code> methods.</p>

  </div>


  <h6><dfn element-state for="input" data-x="attr-input-type-datetime-local">Local Date and Time</dfn> state (<code data-x="">type=datetime-local</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-datetime-local">Local Date and Time</span> state, the rules in
  this section apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> a control for setting the element's
  <span data-x="concept-fe-value">value</span> to a string representing a <span
  data-x="concept-datetime-local">local date and time</span>, with no time-zone offset
  information.</p>

  <div w-nodev>

  <p>If the element is <i data-x="concept-fe-mutable">mutable</i>, the user agent should allow the
  user to change the <span data-x="concept-datetime-local">date and time</span> represented by its
  <span data-x="concept-fe-value">value</span>, as obtained by <span data-x="parse a local date and
  time string">parsing a date and time</span> from it. User agents must not allow the user to set
  the <span data-x="concept-fe-value">value</span> to a non-empty string that is not a <span>valid
  normalized local date and time string</span>. If the user agent provides a user interface for
  selecting a <span data-x="concept-datetime-local">local date and time</span>, then the <span
  data-x="concept-fe-value">value</span> must be set to a <span>valid normalized local date and time
  string</span> representing the user's selection. User agents should allow the user to set the
  <span data-x="concept-fe-value">value</span> to the empty string.</p>

  <p><strong>Constraint validation</strong>: While the user interface describes input that the user
  agent cannot convert to a <span>valid normalized local date and time string</span>, the control is
  <span>suffering from bad input</span>.</p>

  </div>

  <p class="note">See the <a href="#input-author-notes">introduction section</a> for a discussion of
  the difference between the input format and submission format for date, time, and number form
  controls<span w-nodev>, and the <a href="#input-impl-notes">implementation notes</a>
  regarding localization of form controls</span>.</p>

  <p>The <code data-x="attr-input-value">value</code> attribute, if specified and not empty, must
  have a value that is a <span>valid local date and time string</span>.</p> <!-- ok to set
  out-of-range value, we never know when we might have to represent bogus input -->

  <div w-nodev>

  <p><strong>The <span>value sanitization algorithm</span> is as follows</strong>: If the <span
  data-x="concept-fe-value">value</span> of the element is a <span>valid local date and time
  string</span>, then set it to a <span>valid normalized local date and time string</span>
  representing the same date and time; otherwise, set it to the empty string instead.</p>

  </div>

  <p>The <code data-x="attr-input-min">min</code> attribute, if specified, must have a value that is
  a <span>valid local date and time string</span>. The <code data-x="attr-input-max">max</code>
  attribute, if specified, must have a value that is a <span>valid local date and time
  string</span>.</p>

  <p>The <code data-x="attr-input-step">step</code> attribute is expressed in seconds. <span
  w-nodev>The <span data-x="concept-input-step-scale">step scale factor</span> is 1000 (which
  converts the seconds to milliseconds, as used in the other algorithms).</span> The <span
  data-x="concept-input-step-default">default step</span> is 60 seconds.</p>

  <div w-nodev>

  <p>When the element is <span>suffering from a step mismatch</span>, the user agent may round the
  element's <span data-x="concept-fe-value">value</span> to the nearest <span
  data-x="concept-datetime-local">local date and time</span> for which the element would not <span
  data-x="suffering from a step mismatch">suffer from a step mismatch</span>.</p>

  <p><strong>The <span data-x="concept-input-value-string-number">algorithm to convert a string to a
  number</span>, given a string <var>input</var>, is as follows</strong>: If <span
  data-x="parse a local date and time string">parsing a date and time</span> from <var>input</var> results in an error, then return an error; otherwise, return the number of
  milliseconds elapsed from midnight on the morning of 1970-01-01 (the time represented by the value
  "<code data-x="">1970-01-01T00:00:00.0</code>") to the parsed <span
  data-x="concept-datetime-local">local date and time</span>, ignoring leap seconds.</p>

  <p><strong>The <span data-x="concept-input-value-number-string">algorithm to convert a number to a
  string</span>, given a number <var>input</var>, is as follows</strong>: Return a
  <span>valid normalized local date and time string</span> that represents the date and time that is
  <var>input</var> milliseconds after midnight on the morning of 1970-01-01 (the time
  represented by the value "<code data-x="">1970-01-01T00:00:00.0</code>").</p>

  </div>

  <p class="note">See <a href="#only-contemporary-times">the note on historical dates</a> in the
  <span data-x="attr-input-type-date">Date</span> state section.</p>

  <div class="bookkeeping">

   <p>The following common <code>input</code> element content
   attributes, IDL attributes, and methods <span data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>, and
   <code data-x="attr-input-step">step</code> content attributes;
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-input-value">value</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-input-stepDown">stepDown()</code>, and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is
   in mode <span data-x="dom-input-value-value">value</span>.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span data-x="concept-input-apply">apply</span>.</p>

   <p>The following content attributes must not be specified and <span>do not
   apply</span> to the element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-size">size</code>,
   <code data-x="attr-input-src">src</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the
   element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>,
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>, and
   <code data-x="dom-input-valueAsDate">valueAsDate</code> IDL attributes;
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>, and
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code> methods.</p>

  </div>

  <div class="example">

   <p>The following example shows part of a flight booking application. The application uses an
   <code>input</code> element with its <code data-x="attr-input-type">type</code> attribute set to
   <code data-x="attr-input-type-datetime-local">datetime-local</code>, and it then interprets the
   given date and time in the time zone of the selected airport.</p>

   <pre><code class="html">&lt;fieldset>
 &lt;legend>Destination&lt;/legend>
 &lt;p>&lt;label>Airport: &lt;input type=text name=to list=airports>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Departure time: &lt;input type=datetime-local name=totime step=3600>&lt;/label>&lt;/p>
&lt;/fieldset>
&lt;datalist id=airports>
 &lt;option value=ATL label="Atlanta">
 &lt;option value=MEM label="Memphis">
 &lt;option value=LHR label="London Heathrow">
 &lt;option value=LAX label="Los Angeles">
 &lt;option value=FRA label="Frankfurt">
&lt;/datalist></code></pre>

  </div>


  <h6><dfn element-state for="input" id="number-state" data-x="attr-input-type-number">Number</dfn> state (<code data-x="">type=number</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-number">Number</span> state, the rules in this section apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> a control for setting the element's
  <span data-x="concept-fe-value">value</span> to a string representing a number.</p>

  <div w-nodev>

  <p>If the element is <i data-x="concept-fe-mutable">mutable</i>, the user agent should allow the
  user to change the number represented by its <span data-x="concept-fe-value">value</span>, as
  obtained from applying the <span>rules for parsing floating-point number values</span> to it. User
  agents must not allow the user to set the <span data-x="concept-fe-value">value</span> to a
  non-empty string that is not a <span>valid floating-point number</span>. If the user agent
  provides a user interface for selecting a number, then the <span
  data-x="concept-fe-value">value</span> must be set to the <span data-x="best representation of the
  number as a floating-point number">best representation of the number representing the user's
  selection as a floating-point number</span>. User agents should allow the user to set the <span
  data-x="concept-fe-value">value</span> to the empty string.</p>

  <p><strong>Constraint validation</strong>: While the user interface describes input that the user
  agent cannot convert to a <span>valid floating-point number</span>, the control is <span>suffering
  from bad input</span>.</p>

  </div>

  <p class="note">This specification does not define what user interface user agents are to use;
  user agent vendors are encouraged to consider what would best serve their users' needs. For
  example, a user agent in Persian or Arabic markets might support Persian and Arabic numeric input
  (converting it to the format required for submission as described above). Similarly, a user agent
  designed for Romans might display the value in Roman numerals rather than in decimal; or (more
  realistically) a user agent designed for the French market might display the value with
  apostrophes between thousands and commas before the decimals, and allow the user to enter a value
  in that manner, internally converting it to the submission format described above.</p>

  <p>The <code data-x="attr-input-value">value</code> attribute, if specified and not empty, must
  have a value that is a <span>valid floating-point number</span>.</p> <!-- ok to set out-of-range
  value, we never know when we might have to represent bogus input -->

  <div w-nodev>

  <p><strong>The <span>value sanitization algorithm</span> is as follows</strong>: If the <span
  data-x="concept-fe-value">value</span> of the element is not a <span>valid floating-point
  number</span>, then set it to the empty string instead.</p>

  </div>

  <p>The <code data-x="attr-input-min">min</code> attribute, if specified, must have a value that is
  a <span>valid floating-point number</span>. The <code data-x="attr-input-max">max</code> attribute,
  if specified, must have a value that is a <span>valid floating-point number</span>.</p>

  <p><span w-nodev>The <span data-x="concept-input-step-scale">step scale factor</span> is
  1.</span> The <span data-x="concept-input-step-default">default step</span> is 1 (allowing only
  integers to be selected by the user, unless the <span data-x="concept-input-min-zero">step
  base</span> has a non-integer value).</p>

  <div w-nodev>

  <p>When the element is <span>suffering from a step mismatch</span>, the user agent may round the
  element's <span data-x="concept-fe-value">value</span> to the nearest number for which the element
  would not <span data-x="suffering from a step mismatch">suffer from a step mismatch</span>. If
  there are two such numbers, user agents are encouraged to pick the one nearest positive
  infinity.</p>

  <p><strong>The <span data-x="concept-input-value-string-number">algorithm to convert a string to a
  number</span>, given a string <var>input</var>, is as follows</strong>: If applying the
  <span>rules for parsing floating-point number values</span> to <var>input</var> results
  in an error, then return an error; otherwise, return the resulting number.</p>

  <p><strong>The <span data-x="concept-input-value-number-string">algorithm to convert a number to a
  string</span>, given a number <var>input</var>, is as follows</strong>: Return a
  <span>valid floating-point number</span> that represents <var>input</var>.</p>

  </div>

  <div class="bookkeeping">

   <p>The following common <code>input</code> element content attributes, IDL attributes, and
   methods <span data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>, and
   <code data-x="attr-input-step">step</code> content attributes;
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-input-value">value</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-input-stepDown">stepDown()</code>, and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is in mode <span
   data-x="dom-input-value-value">value</span>.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span
   data-x="concept-input-apply">apply</span>.</p>

   <p>The following content attributes must not be specified and <span>do not apply</span> to the
   element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-size">size</code>,
   <code data-x="attr-input-src">src</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>,
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>, and
   <code data-x="dom-input-valueAsDate">valueAsDate</code> IDL attributes;
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>, and
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code> methods.</p>

  </div>

  <div class="example">

   <p>Here is an example of using a numeric input control:</p>

   <pre><code class="html">&lt;label>How much do you want to charge? $&lt;input type=number min=0 step=0.01 name=price>&lt;/label></code></pre>

   <p>As described above, a user agent might support numeric input in the user's local format,
   converting it to the format required for submission as described above. This might include
   handling grouping separators (as in "872,000,000,000") and various decimal separators (such as
   "3,99" vs "3.99") or using local digits (such as those in Arabic, Devanagari, Persian, and
   Thai).</p>

  </div>

  <p class="note" id="when-number-is-not-appropriate">The <code data-x="">type=number</code> state
  is not appropriate for input that happens to only consist of numbers but isn't strictly speaking a
  number. For example, it would be inappropriate for credit card numbers or US postal codes. A
  simple way of determining whether to use <code data-x="">type=number</code> is to consider whether
  it would make sense for the input control to have a spinbox interface (e.g. with "up" and "down"
  arrows). Getting a credit card number wrong by 1 in the last digit isn't a minor mistake, it's as
  wrong as getting every digit incorrect. So it would not make sense for the user to select a credit
  card number using "up" and "down" buttons. When a spinbox interface is not appropriate, <code
  data-x="">type=text</code> is probably the right choice (possibly with an <code
  data-x="attr-inputmode">inputmode</code> or <code data-x="attr-input-pattern">pattern</code>
  attribute).</p>



  <h6><dfn element-state for="input" data-x="attr-input-type-range">Range</dfn> state (<code data-x="">type=range</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-range">Range</span> state, the rules in this section apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> a control for setting the element's
  <span data-x="concept-fe-value">value</span> to a string representing a number, but with the
  caveat that the exact value is not important, letting UAs provide a simpler interface than they
  do for the <span data-x="attr-input-type-number">Number</span> state.</p>

  <div w-nodev>

  <p>If the element is <i data-x="concept-fe-mutable">mutable</i>, the user agent should allow the
  user to change the number represented by its <span data-x="concept-fe-value">value</span>, as
  obtained from applying the <span>rules for parsing floating-point number values</span> to it.
  User agents must not allow the user to set the <span data-x="concept-fe-value">value</span> to a
  string that is not a <span>valid floating-point number</span>. If the user agent provides a user
  interface for selecting a number, then the <span data-x="concept-fe-value">value</span> must be
  set to a <span data-x="best representation of the number as a floating-point number">best
  representation of the number representing the user's selection as a floating-point
  number</span>. User agents must not allow the user to set the <span
  data-x="concept-fe-value">value</span> to the empty string.</p>

  <p><strong>Constraint validation</strong>: While the user interface describes input that the
  user agent cannot convert to a <span>valid floating-point number</span>, the control is
  <span>suffering from bad input</span>.</p>

  </div>

  <p>The <code data-x="attr-input-value">value</code> attribute, if specified, must have a value
  that is a <span>valid floating-point number</span>.</p> <!-- ok to set out-of-range value, we
  never know when we might have to represent bogus input; not ok to not have a value if the
  attribute is present, since you can't not have a value (attribute missing is treated as implying
  a default value) -->

  <div w-nodev>

  <p><strong>The <span>value sanitization algorithm</span> is as follows</strong>: If the <span
  data-x="concept-fe-value">value</span> of the element is not a <span>valid floating-point
  number</span>, then set it to the <span data-x="best representation of the number as a
  floating-point number">best representation, as a floating-point number</span>, of the <span
  data-x="concept-input-value-default-range">default value</span>.</p>

  </div>

  <p>The <dfn data-x="concept-input-value-default-range">default value</dfn> is the <span
  data-x="concept-input-min">minimum</span> plus half the difference between the <span
  data-x="concept-input-min">minimum</span> and the <span
  data-x="concept-input-max">maximum</span>, unless the <span
  data-x="concept-input-max">maximum</span> is less than the <span
  data-x="concept-input-min">minimum</span>, in which case the <span
  data-x="concept-input-value-default-range">default value</span> is the <span
  data-x="concept-input-min">minimum</span>.</p>

  <div w-nodev>

  <p>When the element is <span>suffering from an underflow</span>, the user agent must set the
  element's <span data-x="concept-fe-value">value</span> to the <span data-x="best representation
  of the number as a floating-point number">best representation, as a floating-point
  number</span>, of the <span data-x="concept-input-min">minimum</span>.</p>

  <p>When the element is <span>suffering from an overflow</span>, if the <span
  data-x="concept-input-max">maximum</span> is not less than the <span
  data-x="concept-input-min">minimum</span>, the user agent must set the element's <span
  data-x="concept-fe-value">value</span> to a <span>valid floating-point number</span> that
  represents the <span data-x="concept-input-max">maximum</span>.</p>

  <p>When the element is <span>suffering from a step mismatch</span>, the user agent must round
  the element's <span data-x="concept-fe-value">value</span> to the nearest number for which the
  element would not <span data-x="suffering from a step mismatch">suffer from a step
  mismatch</span>, and which is greater than or equal to the <span
  data-x="concept-input-min">minimum</span>, and, if the <span
  data-x="concept-input-max">maximum</span> is not less than the <span
  data-x="concept-input-min">minimum</span>, which is less than or equal to the <span
  data-x="concept-input-max">maximum</span>, if there is a number that matches these constraints.
  If two numbers match these constraints, then user agents must use the one nearest to positive
  infinity.</p>

  <p class="example">For example, the markup
  <code data-x="">&lt;input&nbsp;type="range"&nbsp;min=0&nbsp;max=100&nbsp;step=20&nbsp;value=50></code>
  results in a range control whose initial value is 60.</p>

  </div>

  <div class="example">

   <p>Here is an example of a range control using an autocomplete list with the <code
   data-x="attr-input-list">list</code> attribute. This could be useful if there are values along
   the full range of the control that are especially important, such as preconfigured light levels
   or typical speed limits in a range control used as a speed control. The following markup
   fragment:</p>

   <pre><code class="html">&lt;input type="range" min="-100" max="100" value="0" step="10" name="power" list="powers"&gt;
&lt;datalist id="powers"&gt;
&lt;option value="0"&gt;
&lt;option value="-30"&gt;
&lt;option value="30"&gt;
<span class="bad"> &lt;option value="++50"&gt;</span>
&lt;/datalist&gt;</code></pre>

   <p>...with the following style sheet applied:</p>

   <pre><code class="css">input { writing-mode: vertical-lr; height: 75px; width: 49px; background: #D5CCBB; color: black; }</code></pre>

   <p>...might render as:</p>

   <p><img src="/images/sample-range.png" width="49" height="75" alt="A vertical slider control whose primary color is black and whose background color is beige, with the slider having five tick marks, one long one at each extremity, and three short ones clustered around the midpoint.">

   <p>Note how the UA determined the orientation of the control from the ratio of the
   style-sheet-specified height and width properties. The colors were similarly derived from the
   style sheet. The tick marks, however, were derived from the markup. In particular, the <code
   data-x="attr-input-step">step</code> attribute has not affected the placement of tick marks,
   the UA deciding to only use the author-specified completion values and then adding longer tick
   marks at the extremes.</p>

   <p>Note also how the invalid value <code data-x="">++50</code> was ignored.</p>

  </div>

  <div class="example">

   <p>For another example, consider the following markup fragment:</p>

   <pre><code class="html">&lt;input name=x type=range min=100 max=700 step=9.09090909 value=509.090909></code></pre>

   <p>A user agent could display in a variety of ways, for instance:</p>

   <p><img src="/images/sample-range-2a.png" width="231" height="57" alt="As a dial."></p>

   <p>Or, alternatively, for instance:</p>

   <p><img src="/images/sample-range-2b.png" width="445" height="56" alt="As a long horizontal slider with tick marks."></p>

   <p>The user agent could pick which one to display based on the dimensions given in the style
   sheet. This would allow it to maintain the same resolution for the tick marks, despite the
   differences in width.</p>

  </div>

  <div class="example">

   <p>Finally, here is an example of a range control with two labeled values:</p>

   <pre><code class="html">&lt;input type="range" name="a" list="a-values"&gt;
&lt;datalist id="a-values"&gt;
&lt;option value="10" label="Low"&gt;
&lt;option value="90" label="High"&gt;
&lt;/datalist&gt;</code></pre>

   <p>With styles that make the control draw vertically, it might look as follows:</p>

   <p><img src="/images/sample-range-labels.png" width="103" height="164" alt="A vertical slider control with two tick marks, one near the top labeled 'High', and one near the bottom labeled 'Low'.">

  </div>

  <p class="note">In this state, the range and step constraints are enforced even during user input,
  and there is no way to set the value to the empty string.</p>

  <p>The <code data-x="attr-input-min">min</code> attribute, if specified, must have a value that is
  a <span>valid floating-point number</span>. The <span data-x="concept-input-min-default">default
  minimum</span> is 0. The <code data-x="attr-input-max">max</code> attribute, if specified, must
  have a value that is a <span>valid floating-point number</span>. The <span
  data-x="concept-input-max-default">default maximum</span> is 100.</p>

  <p><span w-nodev>The <span data-x="concept-input-step-scale">step scale factor</span> is
  1.</span> The <span data-x="concept-input-step-default">default step</span> is 1 (allowing only
  integers, unless the <code data-x="attr-input-min">min</code> attribute has a non-integer
  value).</p>

  <div w-nodev>

  <p><strong>The <span data-x="concept-input-value-string-number">algorithm to convert a string to a
  number</span>, given a string <var>input</var>, is as follows</strong>: If applying the
  <span>rules for parsing floating-point number values</span> to <var>input</var> results
  in an error, then return an error; otherwise, return the resulting number.</p>

  <p><strong>The <span data-x="concept-input-value-number-string">algorithm to convert a number to a
  string</span>, given a number <var>input</var>, is as follows</strong>: Return the <span
  data-x="best representation of the number as a floating-point number">best representation, as a
  floating-point number</span>, of <var>input</var>.</p>

  </div>

  <div class="bookkeeping">

   <p>The following common <code>input</code> element content attributes, IDL attributes, and
   methods <span data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-min">min</code>, and
   <code data-x="attr-input-step">step</code> content attributes;
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-input-value">value</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-input-stepDown">stepDown()</code> and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is in mode <span
   data-x="dom-input-value-value">value</span>.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span
   data-x="concept-input-apply">apply</span>.</p>

   <p>The following content attributes must not be specified and <span>do not apply</span> to the
   element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>,
   <code data-x="attr-input-size">size</code>,
   <code data-x="attr-input-src">src</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>,
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>, and
   <code data-x="dom-input-valueAsDate">valueAsDate</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>, and
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code> methods.</p>

  </div>



  <h6 id="color-state-(type=color)"><dfn element-state for="input" data-x="attr-input-type-color">Color</dfn> state (<code data-x="">type=color</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-color">Color</span> state, the rules in this section apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> a color well control, for setting the
  element's <span data-x="concept-fe-value">value</span> to a string representing the serialization
  of a CSS color.</p>

  <p class="note">In this state, there is always a CSS color picked, and there is no way for the end
  user to set the value to the empty string.</p>

  <p>The <dfn element-attr for="input"><code data-x="attr-input-alpha">alpha</code></dfn> attribute
  is a <span>boolean attribute</span>. If present, it indicates the CSS color's alpha component can
  be manipulated by the end user and does not have to be fully opaque.</p>

  <p>The <dfn element-attr for="input"><code data-x="attr-input-colorspace">colorspace</code></dfn>
  attribute indicates the color space of the serialized CSS color. It also hints at the desired user
  interface for selecting a CSS color. It is an <span>enumerated attribute</span> with the following
  keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th> Keyword
     <th> State
     <th> Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="input/colorspace"><code
     data-x="attr-input-colorspace-limited-srgb">limited-srgb</code></dfn>
     <td><dfn data-x="attr-input-colorspace-limited-srgb-state">Limited sRGB</dfn>
     <td>The CSS color is converted to the <span>'srgb'</span> color space and limited to 8-bits per
     component, e.g., "<code data-x="">#123456</code>" or "<code
     data-x="">color(srgb 0 1 0 / 0.5)</code>".
    <tr>
     <td><dfn attr-value for="input/colorspace"><code
     data-x="attr-input-colorspace-display-p3">display-p3</code></dfn>
     <td><dfn data-x="attr-input-colorspace-display-p3-state">Display P3</dfn>
     <td>The CSS color is converted to the <span>'display-p3'</span> color space, e.g., "<code
     data-x="">color(display-p3 1.84 -0.19 0.72 / 0.6)</code>".
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <span
  data-x="attr-input-colorspace-limited-srgb-state">Limited sRGB</span> state.</p>

  <div w-nodev>

  <p>Whenever the element's <code data-x="attr-input-alpha">alpha</code> or <code
  data-x="attr-input-colorspace">colorspace</code> attributes are changed, the user agent must run
  <span>update a color well control color</span> given the element.</p>
  <!-- As per the statement earlier in this section this only applies when the type attribute is in
       the Color state. -->

  <p>If the element is <i data-x="concept-fe-mutable">mutable</i>, the user agent should allow the
  user to change the color represented by its <span data-x="concept-fe-value">value</span>, as
  obtained from <span data-x="parse a CSS &lt;color> value">parsing</span> it. User agents must not
  allow the user to set the <span data-x="concept-fe-value">value</span> to a string that running
  <span>update a color well control color</span> for the element would not set it to. If the user
  agent provides a user interface for selecting a CSS color, then the <span
  data-x="concept-fe-value">value</span> must be set to the result of <span data-x="serialize a
  color well control color">serializing a color well control color</span> given the element and the
  end user's selection.</p>

  <p>The <span>input activation behavior</span> for such an element <var>element</var> is to
  <span>show the picker, if applicable</span>, for <var>element</var>.</p>

  <p><strong>Constraint validation</strong>: While the element's <span
  data-x="concept-fe-value">value</span> is not the empty string and <span data-x="parse a CSS
  &lt;color> value">parsing</span> it returns failure, the control is <span>suffering from bad
  input</span>.</p>

  </div>

  <p>The <code data-x="attr-input-value">value</code> attribute, if specified and not the empty
  string, must have a value that is a CSS color.</p>

  <div w-nodev>

  <p><strong>The <span>value sanitization algorithm</span> is as follows</strong>: Run <span>update
  a color well control color</span> for the element.</p>

  <p>To <dfn>update a color well control color</dfn>, given an element <var>element</var>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>element</var> is an <code>input</code> element whose <code
   data-x="attr-input-type">type</code> attribute is in the <span
   data-x="attr-input-type-color">Color</span> state.</p></li>

   <li><p>Let <var>color</var> be the result of <span data-x="parse a CSS &lt;color>
   value">parsing</span> <var>element</var>'s <span data-x="concept-fe-value">value</span>.</p></li>

   <li><p>If <var>color</var> is failure, then set <var>color</var> to <span>opaque
   black</span>.</p></li>

   <li><p>Set <var>element</var>'s <span data-x="concept-fe-value">value</span> to the result of <span
   data-x="serialize a color well control color">serializing a color well control color</span> given
   <var>element</var> and <var>color</var>.</p></li>
  </ol>

  <p>To <dfn>serialize a color well control color</dfn>, given an element <var>element</var> and a
  CSS color <var>color</var>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>element</var> is an <code>input</code> element whose <code
   data-x="attr-input-type">type</code> attribute is in the <span
   data-x="attr-input-type-color">Color</span> state.</p></li>

   <li><p>Let <var>htmlCompatible</var> be false.</p></li>

   <li><p>If <var>element</var>'s <code data-x="attr-input-alpha">alpha</code> attribute is not
   specified, then set <var>color</var>'s alpha component to be fully opaque.</p></li>

   <li>
    <p>If <var>element</var>'s <code data-x="attr-input-colorspace">colorspace</code> attribute is
    in the <span data-x="attr-input-colorspace-limited-srgb-state">Limited sRGB</span> state:</p>

    <ol>
     <li><p>Set <var>color</var> to <var>color</var> <span data-x="Converting
     Colors">converted</span> to the <span>'srgb'</span> color space.</p></li>

     <li><p>Round each of <var>color</var>'s components so they are in the range 0 to 255,
     inclusive. Components are to be <a
     href="https://drafts.csswg.org/css-values-4/#combine-integers">rounded towards +∞</a>.</p></li>
     <!-- This matches similar wording in CSS Color. -->

     <li>
      <p>If <var>element</var>'s <code data-x="attr-input-alpha">alpha</code> attribute is not
      specified, then set <var>htmlCompatible</var> to true.</p>

      <p class="note">This intentionally uses a different serialization path for compatibility with
      an earlier version of the color well control.</p>
     </li>

     <li><p>Otherwise, set <var>color</var> to <var>color</var> converted to using the
     <span>'color()'</span> function.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p><span>Assert</span>: <var>element</var>'s <code
     data-x="attr-input-colorspace">colorspace</code> attribute is in the <span
     data-x="attr-input-colorspace-display-p3-state">Display P3</span> state.</p></li>

     <li><p>Set <var>color</var> to <var>color</var> <span data-x="Converting
     Colors">converted</span> to the <span>'display-p3'</span> color space.</p></li>
    </ol>
   </li>

   <li><p>Return the result of <span data-x="serialize a CSS &lt;color> value">serializing</span>
   <var>color</var>. If <var>htmlCompatible</var> is true, then do so with <span
   data-x="HTML-compatible serialization is requested">HTML-compatible serialization
   requested</span>.</p></li>
  </ol>

  </div>

  <div class="bookkeeping">

   <p>The following common <code>input</code> element content attributes and IDL attributes <span
   data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-input-colorspace">colorspace</code>, and
   <code data-x="attr-input-list">list</code> content attributes;
   <code data-x="dom-input-list">list</code> and
   <code data-x="dom-input-value">value</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code> method.</p>

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is in mode <span
   data-x="dom-input-value-value">value</span>.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span
   data-x="concept-input-apply">apply</span>.</p>

   <p>The following content attributes must not be specified and <span>do not apply</span> to the
   element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>,
   <code data-x="attr-input-size">size</code>,
   <code data-x="attr-input-src">src</code>,
   <code data-x="attr-input-step">step</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>,
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code> and,
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>,
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code>,
   <code data-x="dom-input-stepDown">stepDown()</code>, and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

  </div>


  <h6><dfn element-state for="input" data-x="attr-input-type-checkbox">Checkbox</dfn> state (<code data-x="">type=checkbox</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-checkbox">Checkbox</span> state, the rules in this section
  apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> a two-state control that represents the
  element's <span data-x="concept-fe-checked">checkedness</span> state. If the element's <span
  data-x="concept-fe-checked">checkedness</span> state is true, the control represents a positive
  selection, and if it is false, a negative selection. If the element's <code
  data-x="dom-input-indeterminate">indeterminate</code> IDL attribute is set to true, then the
  control's selection should be obscured as if the control was in a third, indeterminate, state.</p>

  <p class="note">The control is never a true tri-state control, even if the element's <code
  data-x="dom-input-indeterminate">indeterminate</code> IDL attribute is set to true. The <code
  data-x="dom-input-indeterminate">indeterminate</code> IDL attribute only gives the appearance of a
  third state.</p>

  <div w-nodev>

  <p>The <span>input activation behavior</span> is to run the following steps:</p>
  <ol>
   <li><p>If the element is not <span>connected</span>, then return.</p></li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
   data-x="event-input">input</code> at the element with the <code
   data-x="dom-Event-bubbles">bubbles</code> and <code data-x="dom-Event-composed">composed</code>
   attributes initialized to true.</p></li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
   data-x="event-change">change</code> at the element with the <code
   data-x="dom-Event-bubbles">bubbles</code> attribute initialized to true.</p></li>
  </ol>
  <!-- These are not cancelable. Once this fires, the control is checked, end of story. -->

  <p><strong>Constraint validation</strong>: If the element is <i
  data-x="concept-input-required">required</i> and its <span
  data-x="concept-fe-checked">checkedness</span> is false, then the element is <span>suffering from
  being missing</span>.</p>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>input</var>.<span subdfn data-x="dom-input-indeterminate">indeterminate</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>When set, overrides the rendering of <span data-x="attr-input-type-checkbox">checkbox</span>
    controls so that the current value is not visible.</p>
   </dd>
  </dl>

  <div class="bookkeeping">

   <p>The following common <code>input</code> element content attributes and IDL attributes <span
   data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-input-checked">checked</code>, and
   <code data-x="attr-input-required">required</code> content attributes;
   <code data-x="dom-input-checked">checked</code> and
   <code data-x="dom-input-value">value</code> IDL attributes.</p>

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is in mode <span
   data-x="dom-input-value-default-on">default/on</span>.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span
   data-x="concept-input-apply">apply</span>.</p>

   <p>The following content attributes must not be specified and <span>do not apply</span> to the
   element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-size">size</code>,
   <code data-x="attr-input-src">src</code>,
   <code data-x="attr-input-step">step</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the element:
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>,
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>,
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code>,
   <code data-x="dom-input-stepDown">stepDown()</code>, and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

  </div>



  <h6><dfn element-state for="input" data-x="attr-input-type-radio">Radio Button</dfn> state (<code data-x="">type=radio</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-radio">Radio Button</span> state, the rules in this section
  apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> a control that, when used in conjunction
  with other <code>input</code> elements, forms a <i data-x="radio button group">radio button group</i> in which only one
  control can have its <span data-x="concept-fe-checked">checkedness</span> state set to true. If
  the element's <span data-x="concept-fe-checked">checkedness</span> state is true, the control
  represents the selected control in the group, and if it is false, it indicates a control in the
  group that is not selected.</p>

  <p>The <dfn><i>radio button group</i></dfn> that contains an <code>input</code> element
  <var>a</var> also contains all the other <code>input</code> elements <var>b</var> that fulfill all
  of the following conditions:</p>

  <ul>
   <li>The <code>input</code> element <var>b</var>'s <code
   data-x="attr-input-type">type</code> attribute is in the <span data-x="attr-input-type-radio">Radio
   Button</span> state.</li>

   <li>Either <var>a</var> and <var>b</var> have the same <span>form owner</span>,
   or they both have no <span>form owner</span>.</li>

   <li>Both <var>a</var> and <var>b</var> are in the same <span>tree</span>.</li>

   <li>They both have a <code data-x="attr-fe-name">name</code> attribute, their <code
   data-x="attr-fe-name">name</code> attributes are not empty, and the value of <var>a</var>'s <code
   data-x="attr-fe-name">name</code> attribute equals the value of <var>b</var>'s <code
   data-x="attr-fe-name">name</code> attribute.</li>

   <!-- Historical details of case-sensitivity: https://github.com/whatwg/html/issues/1666 -->
  </ul>

  <p>A <span>tree</span> must not contain an <code>input</code> element whose <i data-x="radio
  button group">radio button group</i> contains only that element.</p>

  <div w-nodev>

  <p>When any of the following phenomena occur, if the element's <span
  data-x="concept-fe-checked">checkedness</span> state is true after the occurrence, the <span
  data-x="concept-fe-checked">checkedness</span> state of all the other elements in the same <i data-x="radio button group">radio
  button group</i> must be set to false:</p>

  <ul>
   <li>The element's <span data-x="concept-fe-checked">checkedness</span> state is set to true (for
   whatever reason).</li>

   <li>The element's <code data-x="attr-fe-name">name</code> attribute is set, changed, or
   removed.</li>

   <li>The element's <span>form owner</span> changes.</li>

   <li><span data-x="signal a type change">A type change is signalled</span> for the element.</li>

   <li>The element <span>becomes connected</span>.</li>
  </ul>

  <p>The <span>input activation behavior</span> is to run the following steps:</p>
  <ol>
   <li><p>If the element is not <span>connected</span>, then return.</p></li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
   data-x="event-input">input</code> at the element with the <code
   data-x="dom-Event-bubbles">bubbles</code> and <code data-x="dom-Event-composed">composed</code>
   attributes initialized to true.</p></li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
   data-x="event-change">change</code> at the element with the <code
   data-x="dom-Event-bubbles">bubbles</code> attribute initialized to true.</p></li>
  </ol>
  <!-- These are not cancelable. Once this fires, the control is checked, end of story. -->

  <p><strong>Constraint validation</strong>: If an element in the <i data-x="radio button
  group">radio button group</i> is <i data-x="concept-input-required">required</i>, and all of the
  <code>input</code> elements in the <i data-x="radio button group">radio button group</i> have a
  <span data-x="concept-fe-checked">checkedness</span> that is false, then the element is
  <span>suffering from being missing</span>.</p>

  <div class="example">
   <p>The following example, for some reason, has specified that puppers are both <span
   data-x="concept-input-required">required</span> and <span
   data-x="concept-fe-disabled">disabled</span>:</p>

   <pre><code class="html">&lt;form>
 &lt;p>&lt;label>&lt;input type="radio" name="dog-type" value="pupper" required disabled> Pupper&lt;/label>
 &lt;p>&lt;label>&lt;input type="radio" name="dog-type" value="doggo"> Doggo&lt;/label>
 &lt;p>&lt;button>Make your choice&lt;/button>
&lt;/form></code></pre>

   <p>If the user tries to submit this form without first selecting "Doggo", then <em>both</em>
   <code>input</code> elements will be <span>suffering from being missing</span>, since an element
   in the <span>radio button group</span> is <span data-x="concept-input-required">required</span>
   (viz. the first element), and both of the elements in the radio button group have a false <span
   data-x="concept-fe-checked">checkedness</span>.</p>

   <p>On the other hand, if the user selects "Doggo" and then submits the form, then neither
   <code>input</code> element will be <span>suffering from being missing</span>, since while one of
   them is <span data-x="concept-input-required">required</span>, not all of them have a false <span
   data-x="concept-fe-checked">checkedness</span>.</p>
  </div>

  </div>

  <p class="note">If none of the radio buttons in a <span>radio button group</span> are checked,
  then they will all be initially unchecked in the interface, until such time as one of them is
  checked (either by the user or by script).</p>

  <div class="bookkeeping">

   <p>The following common <code>input</code> element content attributes and IDL attributes <span
   data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-input-checked">checked</code> and
   <code data-x="attr-input-required">required</code> content attributes;
   <code data-x="dom-input-checked">checked</code> and
   <code data-x="dom-input-value">value</code> IDL attributes.</p>

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is in mode <span
   data-x="dom-input-value-default-on">default/on</span>.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span
   data-x="concept-input-apply">apply</span>.</p>

   <p>The following content attributes must not be specified and <span>do not apply</span> to the
   element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-size">size</code>,
   <code data-x="attr-input-src">src</code>,
   <code data-x="attr-input-step">step</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the element:
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>,
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>,
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code>,
   <code data-x="dom-input-stepDown">stepDown()</code>, and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

  </div>



  <h6><dfn element-state for="input" data-x="attr-input-type-file">File Upload</dfn> state (<code data-x="">type=file</code>)</h6>

  <!-- v2 ideas:
   * maximum height/width or required ratio for image uploads? - Leons Petrazickis
   * maximum per-file upload size - Alfonso Mart&iacute;nez de Lizarrondo
  -->

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-file">File Upload</span> state, the rules in this section
  apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> a list of <dfn
  data-x="concept-input-type-file-selected">selected files</dfn>, each file consisting of a
  filename, a file type, and a file body (the contents of the file).</p>

  <div w-nodev>

  <p>Filenames must not contain <span data-x="concept-input-file-path">path components</span>, even
  in the case that a user has selected an entire directory hierarchy or multiple files with the same
  name from different directories. <dfn data-x="concept-input-file-path">Path components</dfn>, for
  the purposes of the <span data-x="attr-input-type-file">File Upload</span> state, are those parts
  of filenames that are separated by U+005C REVERSE SOLIDUS character (\) characters.</p>

  <p>Unless the <code data-x="attr-input-multiple">multiple</code> attribute is set, there must be
  no more than one file in the list of <span data-x="concept-input-type-file-selected">selected
  files</span>.</p>

  <p>The <span>input activation behavior</span> for such an element <var>element</var> is to
  <span>show the picker, if applicable</span>, for <var>element</var>.</p>

  <p>If the element is <i data-x="concept-fe-mutable">mutable</i>, the user agent should allow the
  user to change the files on the list in other ways also, e.g., adding or removing files by
  drag-and-drop. When the user does so, the user agent must <span>update the file selection</span>
  for the element.</p>

  <p>If the element is not <i data-x="concept-fe-mutable">mutable</i>, the user agent must not allow
  the user to change the element's selection.</p>

  <p>To <dfn>update the file selection</dfn> for an element <var>element</var>:</p>

  <ol>
   <li>
    <p><span>Queue an element task</span> on the <span>user interaction task source</span> given
    <var>element</var> and the following steps:</p>

    <ol>
     <li><p>Update <var>element</var>'s <span data-x="concept-input-type-file-selected">selected
     files</span> so that it represents the user's selection.</p></li>

     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-input">input</code> at the <code>input</code> element, with the <code
     data-x="dom-Event-bubbles">bubbles</code> and <code data-x="dom-Event-composed">composed</code>
     attributes initialized to true.</p></li>

     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-change">change</code> at the <code>input</code> element, with the <code
     data-x="dom-Event-bubbles">bubbles</code> attribute initialized to true.</p></li>
    </ol>
   </li>
  </ol>

  <p><strong>Constraint validation</strong>: If the element is <i
  data-x="concept-input-required">required</i> and the list of <span
  data-x="concept-input-type-file-selected">selected files</span> is empty, then the element is
  <span>suffering from being missing</span>.</p>

  </div>

  <hr>

  <p>The <dfn element-attr for="input"><code data-x="attr-input-accept">accept</code></dfn>
  attribute may be specified to provide user agents with a hint of what file types will be
  accepted.</p>

  <p>If specified, the attribute must consist of a <span>set of comma-separated tokens</span>, each
  of which must be an <span>ASCII case-insensitive</span> match for one of the following:</p>

  <dl>
   <dt>The string "<code data-x="">audio/*</code>"</dt>

   <dd>Indicates that sound files are accepted.</dd>

   <dt>The string "<code data-x="">video/*</code>"</dt>

   <dd>Indicates that video files are accepted.</dd>

   <dt>The string "<code data-x="">image/*</code>"</dt>

   <dd>Indicates that image files are accepted.</dd>

   <dt>A <span>valid MIME type string with no parameters</span></dt>

   <dd>Indicates that files of the specified type are accepted.</dd>

   <dt>A string whose first character is a U+002E FULL STOP character (.)</dt>

   <dd>Indicates that files with the specified file extension are accepted.</dd>
  </dl>

  <p>The tokens must not be <span>ASCII case-insensitive</span> matches for any of the other tokens
  (i.e. duplicates are not allowed). <span w-nodev>To obtain the list of tokens from the
  attribute, the user agent must <span data-x="split a string on commas">split the attribute value on
  commas</span>.</span></p>

  <p>User agents may use the value of this attribute to display a more appropriate user interface
  than a generic file picker. For instance, given the value <code data-x="">image/*</code>, a user
  agent could offer the user the option of using a local camera or selecting a photograph from their
  photo collection; given the value <code data-x="">audio/*</code>, a user agent could offer the user
  the option of recording a clip using a headset microphone.</p>

  <div w-nodev>

  <p>User agents should prevent the user from selecting files that are not accepted by one (or more)
  of these tokens.</p>

  </div>

  <p class="note">Authors are encouraged to specify both any MIME types and any corresponding
  extensions when looking for data in a specific format.</p>

  <div class="example">

   <p>For example, consider an application that converts Microsoft Word documents to Open Document
   Format files. Since Microsoft Word documents are described with a wide variety of MIME types and
   extensions, the site can list several, as follows:</p>

   <pre><code class="html">&lt;input type="file" accept=".doc,.docx,.xml,application/msword,application/vnd.openxmlformats-officedocument.wordprocessingml.document"></code></pre>

   <p>On platforms that only use file extensions to describe file types, the extensions listed here
   can be used to filter the allowed documents, while the MIME types can be used with the system's
   type registration table (mapping MIME types to extensions used by the system), if any, to
   determine any other extensions to allow. Similarly, on a system that does not have filenames or
   extensions but labels documents with MIME types internally, the MIME types can be used to pick
   the allowed files, while the extensions can be used if the system has an extension registration
   table that maps known extensions to MIME types used by the system.</p>

  </div>

  <p class="warning">Extensions tend to be ambiguous (e.g. there are an untold number of formats
  that use the "<code data-x="">.dat</code>" extension, and users can typically quite easily rename
  their files to have a "<code data-x="">.doc</code>" extension even if they are not Microsoft Word
  documents), and MIME types tend to be unreliable (e.g. many formats have no formally registered
  types, and many formats are in practice labeled using a number of different MIME types). Authors
  are reminded that, as usual, data received from a client should be treated with caution, as it may
  not be in an expected format even if the user is not hostile and the user agent fully obeyed the
  <code data-x="attr-input-accept">accept</code> attribute's requirements.</p>

  <div class="example" id="fakepath-srsly">

   <p>For historical reasons, the <code data-x="dom-input-value">value</code> IDL attribute prefixes
   the filename with the string "<code data-x="">C:\fakepath\</code>". Some legacy user agents
   actually included the full path (which was a security vulnerability). As a result of this,
   obtaining the filename from the <code data-x="dom-input-value">value</code> IDL attribute in a
   backwards-compatible way is non-trivial. The following function extracts the filename in a
   suitably compatible manner:</p>

   <pre><code class="js">function extractFilename(path) {
  if (path.substr(0, 12) == "C:\\fakepath\\")
    return path.substr(12); // modern browser
  var x;
  x = path.lastIndexOf('/');
  if (x >= 0) // Unix-based path
    return path.substr(x+1);
  x = path.lastIndexOf('\\');
  if (x >= 0) // Windows-based path
    return path.substr(x+1);
  return path; // just the filename
}</code></pre>

   <p>This can be used as follows:</p>

   <pre><code class="html">&lt;p>&lt;input type=file name=image onchange="updateFilename(this.value)">&lt;/p>
&lt;p>The name of the file you picked is: &lt;span id="filename">(none)&lt;/span>&lt;/p>
&lt;script>
 function updateFilename(path) {
   var name = extractFilename(path);
   document.getElementById('filename').textContent = name;
 }
&lt;/script></code></pre>

   <!-- How useful this actually is... is unclear. -->

  </div>

  <hr>

  <div class="bookkeeping">

   <p>The following common <code>input</code> element content attributes and IDL attributes <span
   data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-multiple">multiple</code>, and
   <code data-x="attr-input-required">required</code> content attributes;
   <code data-x="dom-input-files">files</code> and
   <code data-x="dom-input-value">value</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code> method.</p>

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is in mode <span
   data-x="dom-input-value-filename">filename</span>.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span
   data-x="concept-input-apply">apply</span>.</p>

   <p>The following content attributes must not be specified and <span>do not apply</span> to the
   element:
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-size">size</code>,
   <code data-x="attr-input-src">src</code>,
   <code data-x="attr-input-step">step</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The element's <code data-x="attr-input-value">value</code> attribute must be omitted.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>,
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>,
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code>,
   <code data-x="dom-input-stepDown">stepDown()</code>, and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

  </div>



  <h6><dfn element-state for="input" data-x="attr-input-type-submit">Submit Button</dfn> state (<code data-x="">type=submit</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-submit">Submit Button</span> state, the rules in this section
  apply.</p>

  </div>

  <p>
  <!--INSERT TRACKING-->
  The <code>input</code> element <span>represents</span> a button that, when activated, submits the
  form. <span w-nodev>If the element has a <code data-x="attr-input-value">value</code> attribute,
  the button's label must be the value of that attribute; otherwise, it must be an
  <span>implementation-defined</span> string that means "Submit" or some such.</span> The element is
  a <span data-x="concept-button">button</span>, specifically a <span
  data-x="concept-submit-button">submit button</span>.</p>

  <p class="note">Since the default label is <span>implementation-defined</span>, and the width of
  the button typically depends on the button's label, the button's width can leak a few bits of
  fingerprintable information. These bits are likely to be strongly correlated to the identity of
  the user agent and the user's locale.</p>

  <div w-nodev>

  <p>The element's <span>input activation behavior</span> given <var>event</var> is as follows:</p>

  <ol>
   <li><p>If the element does not have a <span>form owner</span>, then return.</p></li>

   <li><p>If the element's <span>node document</span> is not <span>fully active</span>, then
   return.</p></li>

   <li><p><span data-x="concept-form-submit">Submit</span> the element's <span>form owner</span>
   from the element with <i data-x="submit-user-involvement">userInvolvement</i> set to
   <var>event</var>'s <span data-x="event-uni">user navigation involvement</span>.</p>
  </ol>

  </div>

  <p>The <code data-x="attr-fs-formaction">formaction</code>, <code
  data-x="attr-fs-formenctype">formenctype</code>, <code
  data-x="attr-fs-formmethod">formmethod</code>, <code
  data-x="attr-fs-formnovalidate">formnovalidate</code>, and <code
  data-x="attr-fs-formtarget">formtarget</code> attributes are <span>attributes for form
  submission</span>.</p>

  <p class="note">The <code data-x="attr-fs-formnovalidate">formnovalidate</code> attribute can be
  used to make submit buttons that do not trigger the constraint validation.</p>

  <div class="bookkeeping">

   <p>The following common <code>input</code> element content attributes and IDL attributes <span
   data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-popovertarget">popovertarget</code>, and
   <code data-x="attr-popovertargetaction">popovertargetaction</code> content
   attributes; <code data-x="dom-input-value">value</code> IDL attribute.</p>

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is in mode <span
   data-x="dom-input-value-default">default</span>.</p>

   <p>The following content attributes must not be specified and <span>do not apply</span> to the
   element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>,
   <code data-x="attr-input-size">size</code>,
   <code data-x="attr-input-src">src</code>,
   <code data-x="attr-input-step">step</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>,
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>,
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code>,
   <code data-x="dom-input-stepDown">stepDown()</code>, and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span>do not apply</span>.</p>

  </div>


  <h6><dfn element-state for="input" data-x="attr-input-type-image">Image Button</dfn> state (<code data-x="">type=image</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-image">Image Button</span> state, the rules in this section
  apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> either an image from which a user can
  select a coordinate and submit the form, or alternatively a button from which the user can submit
  the form. The element is a <span data-x="concept-button">button</span>, specifically a <span
  data-x="concept-submit-button">submit button</span>.</p>

  <p class="note">The coordinate is sent to the server <span data-x="constructing the entry
  list">during form submission</span> by sending two entries for the element, derived from the name
  of the control but with "<code data-x="">.x</code>" and "<code data-x="">.y</code>" appended to
  the name with the <var>x</var> and <var>y</var> components of the coordinate respectively.</p>

  <hr>

  <p>The image is given by the <dfn element-attr for="input"><code
  data-x="attr-input-src">src</code></dfn> attribute. The <code data-x="attr-input-src">src</code>
  attribute must be present, and must contain a <span>valid non-empty URL potentially surrounded by
  spaces</span> referencing a non-interactive, optionally animated, image resource that is neither
  paged nor scripted.</p>

  <div w-nodev>

  <p>When any of the these events occur

  <ul>
   <li>the <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is
   first set to the <span data-x="attr-input-type-image">Image Button</span> state (possibly when
   the element is first created), and the <code data-x="attr-input-src">src</code> attribute is
   present</li>

   <li>the <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is
   changed back to the <span data-x="attr-input-type-image">Image Button</span> state, and the <code
   data-x="attr-input-src">src</code> attribute is present, and its value has changed since the last
   time the <code data-x="attr-input-type">type</code> attribute was in the <span
   data-x="attr-input-type-image">Image Button</span> state</li>

   <li>the <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
   the <span data-x="attr-input-type-image">Image Button</span> state, and the <code
   data-x="attr-input-src">src</code> attribute is set or changed</li>
  </ul> <!-- Note how this does NOT happen when the base URL changes. -->

  <p>then unless the user agent cannot support images, or its support for images has been disabled,
  or the user agent only fetches images on demand, or the <code data-x="attr-input-src">src</code>
  attribute's value is the empty string, run these steps:

  <ol>
   <li><p>Let <var>url</var> be the result of <span>encoding-parsing a URL</span> given the <code
   data-x="attr-input-src">src</code> attribute's value, relative to the element's <span>node
   document</span>.</p></li>

   <li><p>If <var>url</var> is failure, then return.</p></li>

   <li><p>Let <var>request</var> be a new <span data-x="concept-request">request</span> whose <span
   data-x="concept-request-url">URL</span> is <var>url</var>, <span
   data-x="concept-request-client">client</span> is the element's <span>node document</span>'s
   <span>relevant settings object</span>, <span
   data-x="concept-request-destination">destination</span> is "<code data-x="">image</code>", <span
   data-x="concept-request-initiator-type">initiator type</span> is "<code data-x="">input</code>",
   <span data-x="concept-request-credentials-mode">credentials mode</span> is "<code
   data-x="">include</code>", and whose <span>use-URL-credentials flag</span> is set.</p></li>

   <li>
    <p><!--FETCH--><span data-x="concept-fetch">Fetch</span> <var>request</var>, with
    <i data-x="processResponseEndOfBody">processResponseEndOfBody</i> set to the following steps
    given <span data-x="concept-response">response</span> <var>response</var>:</p>

    <ol>
      <li><p>If the download was successful and the image is <i
      data-x="input-img-available">available</i>, <span>queue an element task</span> on the
      <span>user interaction task source</span> given the <code>input</code> element to <span
      data-x="concept-event-fire">fire an event</span> named <code data-x="event-load">load</code>
      at the <code>input</code> element.</p></li>

      <li><p>Otherwise, if the fetching process fails without a response from the remote server, or
      completes but the image is not a valid or supported image, then <span>queue an element
      task</span> on the <span>user interaction task source</span> given the <code>input</code>
      element to <span data-x="concept-event-fire">fire an event</span> named <code
      data-x="event-error">error</code> on the <code>input</code> element.</p></li>
    </ol>
   </li>
  </ol>

  <!-- same text in <img> section and similar text elsewhere -->
  <p>Fetching the image must <span>delay the load event</span> of the element's <span>node document</span> until the
  <span data-x="concept-task">task</span> that is <span data-x="queue a task">queued</span> by the
  <span>networking task source</span> once the resource has been fetched (defined below) has been
  run.</p>

  <p>If the image was successfully obtained, with no network errors, and the image's type is a
  supported image type, and the image is a valid image of that type, then the image is said to be
  <dfn data-x="input-img-available"><i>available</i></dfn>. If this is true before the image is
  completely downloaded, each <span data-x="concept-task">task</span> that is <span data-x="queue a
  task">queued</span> by the <span>networking task source</span> while the image is being fetched
  must update the presentation of the image appropriately.</p>

  <p>The user agent should apply the <span data-x="Content-Type sniffing: image">image sniffing
  rules</span> to determine the type of the image, with the image's <span
  data-x="Content-Type">associated Content-Type headers</span> giving the <var>official
  type</var>. If these rules are not applied, then the type of the image must be the type given by
  the image's <span data-x="Content-Type">associated Content-Type headers</span>.</p>

  <p>User agents must not support non-image resources with the <code>input</code> element. User
  agents must not run executable code embedded in the image resource. User agents must only display
  the first page of a multipage resource. User agents must not allow the resource to act in an
  interactive fashion, but should honor any animation in the resource.</p>
  <hr>

  </div>

  <p>The <dfn element-attr for="input"><code data-x="attr-input-alt">alt</code></dfn> attribute
  provides the textual label for the button for users and user agents who cannot use the image. The
  <code data-x="attr-input-alt">alt</code> attribute must be present, and must contain a non-empty
  string giving the label that would be appropriate for an equivalent button if the image was
  unavailable.</p>

  <p>The <code>input</code> element supports <span>dimension attributes</span>.</p>

  <hr>

  <p>If the <code data-x="attr-input-src">src</code> attribute is set, and the image is <i
  data-x="input-img-available">available</i> and the user agent is configured to display that image,
  then the element <span>represents</span> a control for selecting a <span
  data-x="concept-input-type-image-coordinate">coordinate</span> from the image specified by the
  <code data-x="attr-input-src">src</code> attribute. In that case, if the element is <i
  data-x="concept-fe-mutable">mutable</i>, the user agent should allow the user to select this <span
  data-x="concept-input-type-image-coordinate">coordinate</span>.</p>

  <p>Otherwise, the element <span>represents</span> a submit button whose label is given by the
  value of the <code data-x="attr-input-alt">alt</code> attribute.</p>

  <div w-nodev>

  <p>The element's <span>input activation behavior</span> given <var>event</var> is as follows:</p>

  <ol>
   <li><p>If the element does not have a <span>form owner</span>, then return.</p></li>

   <li><p>If the element's <span>node document</span> is not <span>fully active</span>, then
   return.</p></li>

   <li>
    <p>If the user activated the control while explicitly selecting a coordinate, then set the
    element's <span data-x="concept-input-type-image-coordinate">selected coordinate</span> to that
    coordinate.</p>

    <p class="note">This is only possible under the conditions outlined above, when the element
    <span>represents</span> a control for selecting such a coordinate. Even then, the user
    might activate the control without explicitly selecting a coordinate.</p>
   </li>

   <li><p><span data-x="concept-form-submit">Submit</span> the element's <span>form owner</span>
   from the element with <i data-x="submit-user-involvement">userInvolvement</i> set to
   <var>event</var>'s <span data-x="event-uni">user navigation involvement</span>.</p></li>
  </ol>

  <p>The element's <dfn data-x="concept-input-type-image-coordinate">selected coordinate</dfn>
  consists of an <var>x</var>-component and a <var>y</var>-component. It is initially (0, 0). The
  coordinates represent the position relative to the edge of the element's image, with the
  coordinate space having the positive <var>x</var> direction to the right, and the positive
  <var>y</var> direction downwards.</p>

  <p>The <var>x</var>-component must be a <span>valid integer</span> representing a number
  <var>x</var> in the range <span data-x="">&#x2212;(<var>border<sub>left</sub></var>+<var>padding<sub>left</sub></var>) &le; <var>x</var> &le; <var>width</var>+<var>border<sub>right</sub></var>+<var>padding<sub>right</sub></var></span>, where <var>width</var> is the rendered width of the image, <var>border<sub>left</sub></var> is the width of the border on the left of the image, <var>padding<sub>left</sub></var> is the width of the padding on the left of the
  image, <var>border<sub>right</sub></var> is the width of the border on the right
  of the image, and <var>padding<sub>right</sub></var> is the width of the padding
  on the right of the image, with all dimensions given in <span data-x="'px'">CSS pixels</span>.</p>

  <p>The <var>y</var>-component must be a <span>valid integer</span> representing a number
  <var>y</var> in the range <span data-x="">&#x2212;(<var>border<sub>top</sub></var>+<var>padding<sub>top</sub></var>) &le; <var>y</var> &le; <var>height</var>+<var>border<sub>bottom</sub></var>+<var>padding<sub>bottom</sub></var></span>, where
  <var>height</var> is the rendered height of the image, <var>border<sub>top</sub></var> is the width of the border above the image, <var>padding<sub>top</sub></var> is the width of the padding above the image, <var>border<sub>bottom</sub></var> is the width of the border below the image, and <var>padding<sub>bottom</sub></var> is the width of the padding below the image, with
  all dimensions given in <span data-x="'px'">CSS pixels</span>.</p>

  <p>Where a border or padding is missing, its width is zero <span data-x="'px'">CSS
  pixels</span>.</p>

  <hr>

  </div>

  <p>The <code data-x="attr-fs-formaction">formaction</code>, <code
  data-x="attr-fs-formenctype">formenctype</code>, <code
  data-x="attr-fs-formmethod">formmethod</code>, <code
  data-x="attr-fs-formnovalidate">formnovalidate</code>, and <code
  data-x="attr-fs-formtarget">formtarget</code> attributes are <span>attributes for form
  submission</span>.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>image</var>.<span subdfn data-x="dom-input-width">width</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>image</var>.<span subdfn data-x="dom-input-height">height</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>These attributes return the actual rendered dimensions of the image, or 0 if the dimensions
    are not known.</p>

    <p>They can be set, to change the corresponding content attributes.</p>
   </dd>
  </dl>

  <div class="bookkeeping">

   <p>The following common <code>input</code> element content attributes and IDL attributes <span
   data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-popovertarget">popovertarget</code>,
   <code data-x="attr-popovertargetaction">popovertargetaction</code>,
   <code data-x="attr-input-src">src</code>, and
   <code data-x="attr-dim-width">width</code> content attributes;
   <code data-x="dom-input-value">value</code> IDL attribute.</p>

   <p>The <code data-x="dom-input-value">value</code> IDL attribute is in mode <span
   data-x="dom-input-value-default">default</span>.</p>

   <p>The following content attributes must not be specified and <span>do not apply</span> to the
   element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>,
   <code data-x="attr-input-size">size</code>, and
   <code data-x="attr-input-step">step</code>.</p>

   <p>The element's <code data-x="attr-input-value">value</code> attribute must be omitted.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>,
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>,
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code>,
   <code data-x="dom-input-stepDown">stepDown()</code>, and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span>do not apply</span>.</p>

  </div>

  <p class="note">Many aspects of this state's behavior are similar to the behavior of the
  <code>img</code> element. Readers are encouraged to read that section, where many of the same
  requirements are described in more detail.</p>

  <div class="example">

   <p>Take the following form:</p>

   <pre><code class="html">&lt;form action="process.cgi">
 &lt;input type=image src=map.png name=where alt="Show location list">
&lt;/form></code></pre>

   <p>If the user clicked on the image at coordinate (127,40) then the URL used to submit the form
   would be "<code data-x="">process.cgi?where.x=127&amp;where.y=40</code>".</p>

   <p>(In this example, it's assumed that for users who don't see the map, and who instead just see
   a button labeled "Show location list", clicking the button will cause the server to show a list
   of locations to pick from instead of the map.)</p>

  </div>



  <h6><dfn element-state for="input" data-x="attr-input-type-reset">Reset Button</dfn> state (<code data-x="">type=reset</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-reset">Reset Button</span> state, the rules in this section
  apply.</p>

  </div>

  <p>
  <!--INSERT TRACKING-->
  The <code>input</code> element <span>represents</span> a button that, when activated, resets the
  form. <span w-nodev>If the element has a <code data-x="attr-input-value">value</code> attribute,
  the button's label must be the value of that attribute; otherwise, it must be an
  <span>implementation-defined</span> string that means "Reset" or some such.</span> The element is
  a <span data-x="concept-button">button</span>.</p>

  <p class="note">Since the default label is <span>implementation-defined</span>, and the width of
  the button typically depends on the button's label, the button's width can leak a few bits of
  fingerprintable information. These bits are likely to be strongly correlated to the identity of
  the user agent and the user's locale.</p>

  <div w-nodev>

  <p>The element's <span>input activation behavior</span> is as follows:</p>

  <ol>
   <li><p>If the element does not have a <span>form owner</span>, then return.</p></li>

   <li><p>If the element's <span>node document</span> is not <span>fully active</span>, then
   return.</p></li>

   <li><p><span data-x="concept-form-reset">Reset</span> the <span>form owner</span> from the
   element.</p>
  </ol>

  <p><strong>Constraint validation</strong>: The element is <span>barred from constraint
  validation</span>.</p>

  </div>

  <div class="bookkeeping">

   <p>The <code data-x="dom-input-value">value</code> IDL attribute <span
   data-x="concept-input-apply">applies</span> to this element and is in mode <span
   data-x="dom-input-value-default">default</span>.</p>

   <p>The following common <code>input</code> element content attributes <span
   data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-popovertarget">popovertarget</code> and
   <code data-x="attr-popovertargetaction">popovertargetaction</code>.</p>

   <p>The following content attributes must not be specified and <span>do not apply</span> to the
   element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>,
   <code data-x="attr-input-size">size</code>,
   <code data-x="attr-input-src">src</code>,
   <code data-x="attr-input-step">step</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>,
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>,
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code>,
   <code data-x="dom-input-stepDown">stepDown()</code>, and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span>do not apply</span>.</p>

  </div>


  <h6><dfn element-state for="input" data-x="attr-input-type-button">Button</dfn> state (<code data-x="">type=button</code>)</h6>

  <div w-nodev>

  <p>When an <code>input</code> element's <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-button">Button</span> state, the rules in this section apply.</p>

  </div>

  <p>The <code>input</code> element <span>represents</span> a button with no default behavior. A
  label for the button must be provided in the <code data-x="attr-input-value">value</code>
  attribute, though it may be the empty string. <span w-nodev>If the element has a <code
  data-x="attr-input-value">value</code> attribute, the button's label must be the value of that
  attribute; otherwise, it must be the empty string.</span> The element is a <span
  data-x="concept-button">button</span>.</p>

  <div w-nodev>

  <p>The element has no <span>input activation behavior</span>.</p>

  <p><strong>Constraint validation</strong>: The element is <span>barred from constraint
  validation</span>.</p>

  </div>

  <div class="bookkeeping">

   <p>The <code data-x="dom-input-value">value</code> IDL attribute <span
   data-x="concept-input-apply">applies</span> to this element and is in mode <span
   data-x="dom-input-value-default">default</span>.</p>

   <p>The following common <code>input</code> element content attributes <span
   data-x="concept-input-apply">apply</span> to the element:
   <code data-x="attr-popovertarget">popovertarget</code> and
   <code data-x="attr-popovertargetaction">popovertargetaction</code>.</p>

   <p>The following content attributes must not be specified and <span>do not apply</span> to the
   element:
   <code data-x="attr-input-accept">accept</code>,
   <code data-x="attr-input-alpha">alpha</code>,
   <code data-x="attr-input-alt">alt</code>,
   <code data-x="attr-fe-autocomplete">autocomplete</code>,
   <code data-x="attr-input-checked">checked</code>,
   <code data-x="attr-input-colorspace">colorspace</code>,
   <code data-x="attr-fe-dirname">dirname</code>,
   <code data-x="attr-fs-formaction">formaction</code>,
   <code data-x="attr-fs-formenctype">formenctype</code>,
   <code data-x="attr-fs-formmethod">formmethod</code>,
   <code data-x="attr-fs-formnovalidate">formnovalidate</code>,
   <code data-x="attr-fs-formtarget">formtarget</code>,
   <code data-x="attr-dim-height">height</code>,
   <code data-x="attr-input-list">list</code>,
   <code data-x="attr-input-max">max</code>,
   <code data-x="attr-input-maxlength">maxlength</code>,
   <code data-x="attr-input-min">min</code>,
   <code data-x="attr-input-minlength">minlength</code>,
   <code data-x="attr-input-multiple">multiple</code>,
   <code data-x="attr-input-pattern">pattern</code>,
   <code data-x="attr-input-placeholder">placeholder</code>,
   <code data-x="attr-input-readonly">readonly</code>,
   <code data-x="attr-input-required">required</code>,
   <code data-x="attr-input-size">size</code>,
   <code data-x="attr-input-src">src</code>,
   <code data-x="attr-input-step">step</code>, and
   <code data-x="attr-dim-width">width</code>.</p>

   <p>The following IDL attributes and methods <span>do not apply</span> to the element:
   <code data-x="dom-input-checked">checked</code>,
   <code data-x="dom-input-files">files</code>,
   <code data-x="dom-input-list">list</code>,
   <code data-x="dom-textarea/input-selectionStart">selectionStart</code>,
   <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code>,
   <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>,
   <code data-x="dom-input-valueAsDate">valueAsDate</code>, and
   <code data-x="dom-input-valueAsNumber">valueAsNumber</code> IDL attributes;
   <code data-x="dom-textarea/input-select">select()</code>,
   <code data-x="dom-textarea/input-setRangeText">setRangeText()</code>,
   <code data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code>,
   <code data-x="dom-input-stepDown">stepDown()</code>, and
   <code data-x="dom-input-stepUp">stepUp()</code> methods.</p>

   <p>The <code data-x="event-input">input</code> and <code
   data-x="event-change">change</code> events <span>do not apply</span>.</p>

  </div>


  <div w-nodev>

  <h5 id="input-impl-notes">Implementation notes regarding localization of form controls</h5>
  <!-- ID referenced from other parts of the spec -->

  <!-- NON-NORMATIVE SECTION -->

  <p>The formats shown to the user in date, time, and number controls is independent of the format
  used for form submission.</p>

  <p>Browsers are encouraged to use user interfaces that present dates, times, and numbers according
  to the conventions of either the locale implied by the <code>input</code> element's
  <span>language</span> or the user's preferred locale. Using the page's locale will ensure
  consistency with page-provided data.

  <p class="example">For example, it would be confusing to users if an American English page claimed
  that a Cirque De Soleil show was going to be showing on 02/03<!-- Feb 3 (year) -->, but their
  browser, configured to use the British English locale, only showed the date 03/02<!-- 3 Feb (year)
  --> in the ticket purchase date picker. Using the page's locale would at least ensure that the
  date was presented in the same format everywhere. (There's still a risk that the user would end up
  arriving a month late, of course, but there's only so much that can be done about such cultural
  differences...)</p>

  </div>



  <h5 id="common-input-element-attributes">Common <code>input</code> element attributes</h5>

  <div w-nodev>

  <p>These attributes only <span data-x="concept-input-apply">apply</span> to an <code>input</code>
  element if its <code data-x="attr-input-type">type</code> attribute is in a state whose definition
  declares that the attribute <span data-x="concept-input-apply">applies</span>. When an attribute
  <span data-x="do not apply">doesn't apply</span> to an <code>input</code> element, user agents must
  <span>ignore</span> the attribute, regardless of the requirements and definitions below.</p>

  </div>


  <h6>The <code data-x="attr-input-maxlength">maxlength</code> and <code data-x="attr-input-minlength">minlength</code> attributes</h6>

  <p>The <dfn element-attr for="input"><code data-x="attr-input-maxlength">maxlength</code></dfn>
  attribute<span w-nodev>, when it <span data-x="concept-input-apply">applies</span>,</span> is a
  <span data-x="attr-fe-maxlength">form control <code data-x="">maxlength</code>
  attribute</span>.</p>

  <p>The <dfn element-attr for="input"><code data-x="attr-input-minlength">minlength</code></dfn>
  attribute<span w-nodev>, when it <span data-x="concept-input-apply">applies</span>,</span> is a
  <span data-x="attr-fe-minlength">form control <code data-x="">minlength</code>
  attribute</span>.</p>

  <p>If the <code>input</code> element has a <span>maximum allowed value length</span>, then the
  <span>length</span> of the value of the element's <code data-x="attr-input-value">value</code>
  attribute must be less than or equal to the element's <span>maximum allowed value
  length</span>.</p>

  <!-- we allow the default to be smaller than the minimum, so that you can have incomplete, but not
  empty, input already in the form, like a template that needs to be filled in -->

  <div class="example">

   <p>The following extract shows how a messaging client's text entry could be arbitrarily
   restricted to a fixed number of characters, thus forcing any conversation through this medium to
   be terse and discouraging intelligent discourse.</p>

   <pre><code class="html">&lt;label>What are you doing? &lt;input name=status maxlength=140>&lt;/label></code></pre>

  </div>

  <div class="example">

   <p>Here, a password is given a minimum length:</p>

   <pre><code class="html">&lt;p>&lt;label>Username: &lt;input name=u required>&lt;/label>
&lt;p>&lt;label>Password: &lt;input name=p required minlength=12>&lt;/label></code></pre>

  </div>



  <h6>The <code data-x="attr-input-size">size</code> attribute</h6>

  <p>The <dfn element-attr for="input"><code data-x="attr-input-size">size</code></dfn> attribute
  gives the number of characters that, in a visual rendering, the user agent is to allow the user to
  see while editing the element's <span data-x="concept-fe-value">value</span>.</p>

  <p>The <code data-x="attr-input-size">size</code> attribute, if specified, must have a value that
  is a <span>valid non-negative integer</span> greater than zero.</p>

  <div w-nodev>

  <p>If the attribute is present, then its value must be parsed using the <span>rules for parsing
  non-negative integers</span>, and if the result is a number greater than zero, then the user agent
  should ensure that at least that many characters are visible.</p>

  <p>The <code data-x="dom-input-size">size</code> IDL attribute is <span>limited to only positive
  numbers</span> and has a <span>default value</span> of 20.</p>

  </div>



  <h6>The <code data-x="attr-input-readonly">readonly</code> attribute</h6>

  <p>The <dfn element-attr for="input"><code data-x="attr-input-readonly">readonly</code></dfn>
  attribute is a <span>boolean attribute</span> that controls whether or not the user can edit the
  form control. <span w-nodev>When specified, the element is not <i
  data-x="concept-fe-mutable">mutable</i>.</span></p>

  <div w-nodev>

  <p><strong>Constraint validation</strong>: If the <code
  data-x="attr-input-readonly">readonly</code> attribute is specified on an <code>input</code>
  element, the element is <span>barred from constraint validation</span>.</p>

  </div>

  <div class="note">
   <p>The difference between <code data-x="attr-fe-disabled">disabled</code> and <code
   data-x="attr-input-readonly">readonly</code> is that read-only controls can still function,
   whereas disabled controls generally do not function as controls until they are enabled. This is
   spelled out in more detail elsewhere in this specification with normative requirements that refer
   to the <span data-x="concept-fe-disabled">disabled</span> concept (for example, the element's
   <span>activation behavior</span>, whether or not it is a <span>focusable area</span>, or when
   <span>constructing the entry list</span>). Any other behavior related to user interaction with
   disabled controls, such as whether text can be selected or copied, is not defined in this
   standard.</p>

   <p>Only text controls can be made read-only, since for other controls (such as checkboxes and
   buttons) there is no useful distinction between being read-only and being disabled, so the
   <code data-x="attr-input-readonly">readonly</code> attribute <span data-x="do not apply">does not
   apply</span>.</p>
  </div>

  <div class="example">

   <p>In the following example, the existing product identifiers cannot be modified, but they are
   still displayed as part of the form, for consistency with the row representing a new product
   (where the identifier is not yet filled in).</p>

   <pre><code class="html">&lt;form action="products.cgi" method="post" enctype="multipart/form-data">
 &lt;table>
  &lt;tr> &lt;th> Product ID &lt;th> Product name &lt;th> Price &lt;th> Action
  &lt;tr>
   &lt;td> &lt;input readonly="readonly" name="1.pid" value="H412">
   &lt;td> &lt;input required="required" name="1.pname" value="Floor lamp Ulke">
   &lt;td> $&lt;input required="required" type="number" min="0" step="0.01" name="1.pprice" value="49.99">
   &lt;td> &lt;button formnovalidate="formnovalidate" name="action" value="delete:1">Delete&lt;/button>
  &lt;tr>
   &lt;td> &lt;input readonly="readonly" name="2.pid" value="FG28">
   &lt;td> &lt;input required="required" name="2.pname" value="Table lamp Ulke">
   &lt;td> $&lt;input required="required" type="number" min="0" step="0.01" name="2.pprice" value="24.99">
   &lt;td> &lt;button formnovalidate="formnovalidate" name="action" value="delete:2">Delete&lt;/button>
  &lt;tr>
   &lt;td> &lt;input required="required" name="3.pid" value="" pattern="[A-Z0-9]+">
   &lt;td> &lt;input required="required" name="3.pname" value="">
   &lt;td> $&lt;input required="required" type="number" min="0" step="0.01" name="3.pprice" value="">
   &lt;td> &lt;button formnovalidate="formnovalidate" name="action" value="delete:3">Delete&lt;/button>
 &lt;/table>
 &lt;p> &lt;button formnovalidate="formnovalidate" name="action" value="add">Add&lt;/button> &lt;/p>
 &lt;p> &lt;button name="action" value="update">Save&lt;/button> &lt;/p>
&lt;/form></code></pre>

  </div>



  <h6>The <code data-x="attr-input-required">required</code> attribute</h6>

  <p>The <dfn element-attr for="input"><code data-x="attr-input-required">required</code></dfn>
  attribute is a <span>boolean attribute</span>. When specified, the element is <dfn
  data-x="concept-input-required"><i>required</i></dfn>.</p>

  <div w-nodev>

  <p><strong>Constraint validation</strong>: If the element is <i
  data-x="concept-input-required">required</i>, and its <code data-x="dom-input-value">value</code>
  IDL attribute <span data-x="concept-input-apply">applies</span> and is in the mode <span
  data-x="dom-input-value-value">value</span>, and the element is <i
  data-x="concept-fe-mutable">mutable</i>, and the element's <span
  data-x="concept-fe-value">value</span> is the empty string, then the element is <span>suffering
  from being missing</span>.</p>

  </div>

  <div class="example">

   <p>The following form has two required fields, one for an email address and one for a password.
   It also has a third field that is only considered valid if the user types the same password in
   the password field and this third field.</p>

   <pre><code class="html">&lt;h1>Create new account&lt;/h1>
&lt;form action="/newaccount" method=post
      oninput="up2.setCustomValidity(up2.value != up.value ? 'Passwords do not match.' : '')">
 &lt;p>
  &lt;label for="username">Email address:&lt;/label>
  &lt;input id="username" type=email required name=un>
 &lt;p>
  &lt;label for="password1">Password:&lt;/label>
  &lt;input id="password1" type=password required name=up>
 &lt;p>
  &lt;label for="password2">Confirm password:&lt;/label>
  &lt;input id="password2" type=password name=up2>
 &lt;p>
  &lt;input type=submit value="Create account">
&lt;/form></code></pre>

  </div>

  <div class="example">

   <p>For radio buttons, the <code data-x="attr-input-required">required</code> attribute is
   satisfied if any of the radio buttons in the <span data-x="radio button group">group</span> is
   selected. Thus, in the following example, any of the radio buttons can be checked, not just the
   one marked as required:</p>

   <pre><code class="html">&lt;fieldset>
 &lt;legend>Did the movie pass the Bechdel test?&lt;/legend>
 &lt;p>&lt;label>&lt;input type="radio" name="bechdel" value="no-characters"> No, there are not even two female characters in the movie. &lt;/label>
 &lt;p>&lt;label>&lt;input type="radio" name="bechdel" value="no-names"> No, the female characters never talk to each other. &lt;/label>
 &lt;p>&lt;label>&lt;input type="radio" name="bechdel" value="no-topic"> No, when female characters talk to each other it's always about a male character. &lt;/label>
 &lt;p>&lt;label>&lt;input type="radio" name="bechdel" value="yes" <strong>required</strong>> Yes. &lt;/label>
 &lt;p>&lt;label>&lt;input type="radio" name="bechdel" value="unknown"> I don't know. &lt;/label>
&lt;/fieldset></code></pre>

   <p>To avoid confusion as to whether a <span>radio button group</span> is required or not, authors
   are encouraged to specify the attribute on all the radio buttons in a group. Indeed, in general,
   authors are encouraged to avoid having radio button groups that do not have any initially checked
   controls in the first place, as this is a state that the user cannot return to, and is therefore
   generally considered a poor user interface.</p>

  </div>


  <h6>The <code data-x="attr-input-multiple">multiple</code> attribute</h6>

  <p>The <dfn element-attr for="input"><code data-x="attr-input-multiple">multiple</code></dfn>
  attribute is a <span>boolean attribute</span> that indicates whether the user is to be allowed to
  specify more than one value.</p>

  <div class="example">

   <p>The following extract shows how an email client's "To" field could accept multiple email
   addresses.</p>

   <pre><code class="html">&lt;label>To: &lt;input type=email multiple name=to>&lt;/label></code></pre>

   <p>If the user had, amongst many friends in their user contacts database, two friends
   "Spider-Man" (with address "spider@parker.example.net") and "Scarlet Witch" (with address
   "scarlet@avengers.example.net"), then, after the user has typed "<kbd>s</kbd>", the user agent
   might suggest these two email addresses to the user.</p>

   <!-- Having alternative text in the following image would be bogus and non-conforming use of
   HTML, since the image is redundant with the previous paragraph. In particular, just describing
   the image would be especially bad for accessibility as it would be disorienting, there having
   been no mention that an image is present. -->
   <p><img src="/images/sample-email-1.svg" width="330" height="100" alt="" class="darkmode-aware"></p>

   <p>The page could also link in the user's contacts database from the site:</p>

   <pre><code class="html">&lt;label>To: &lt;input type=email multiple name=to list=contacts>&lt;/label>
...
&lt;datalist id="contacts">
 &lt;option value="hedral@damowmow.com">
 &lt;option value="pillar@example.com">
 &lt;option value="astrophy@cute.example">
 &lt;option value="astronomy@science.example.org">
&lt;/datalist></code></pre>

   <p>Suppose the user had entered "<kbd>bob@example.net</kbd>" into this text control, and then
   started typing a second email address starting with "<kbd>s</kbd>". The user agent might show
   both the two friends mentioned earlier, as well as the "astrophy" and "astronomy" values given in
   the <code>datalist</code> element.</p>

   <!-- As above, having a non-empty alt="" attribute on the following image would be dumb, which is
   why it's left blank. The pertinent parts of the image are already described in the prose, and the
   non-pertinent parts aren't relevant to users who can't see the image, obviously. -->
   <p><img src="/images/sample-email-2.svg" width="330" height="140" alt="" class="darkmode-aware"></p>

  </div>

  <div class="example">

   <p>The following extract shows how an email client's "Attachments" field could accept multiple
   files for upload.</p>

   <pre><code class="html">&lt;label>Attachments: &lt;input type=file multiple name=att>&lt;/label></code></pre>

  </div>



  <h6>The <code data-x="attr-input-pattern">pattern</code> attribute</h6>

  <!-- if more text control types start supporting multiple="", this will need to be refactored -->

  <p>The <dfn element-attr for="input"><code data-x="attr-input-pattern">pattern</code></dfn>
  attribute specifies a regular expression against which the control's <span
  data-x="concept-fe-value">value</span>, or, when the <code
  data-x="attr-input-multiple">multiple</code> attribute <span
  data-x="concept-input-apply">applies</span> and is set, the control's <span
  data-x="concept-fe-values">values</span>, are to be checked.</p>

  <p>If specified, the attribute's value must match the JavaScript <i
  data-x="js-prod-Pattern">Pattern</i><sub><code data-x="">[+UnicodeSetsMode, +N]</code></sub> production.</p>

  <div w-nodev>

  <p>The <dfn>compiled pattern regular expression</dfn> of an <code>input</code> element, if it
  exists, is a JavaScript <code>RegExp</code> object. It is determined as follows:</p>

  <ol>
   <li><p>If the element does not have a <code data-x="attr-input-pattern">pattern</code> attribute
   specified, then return nothing. The element has no <span>compiled pattern regular
   expression</span>.</p></li>

   <li><p>Let <var>pattern</var> be the value of the <code
   data-x="attr-input-pattern">pattern</code> attribute of the element.</p></li>

   <li><p>Let <var>regexpCompletion</var> be <span>RegExpCreate</span>(<var>pattern</var>,
   "<code data-x="">v</code>").</p></li>

   <li>
    <p>If <var>regexpCompletion</var> is an <span data-x="Completion Record">abrupt
    completion</span>, then return nothing. The element has no <span>compiled pattern regular
    expression</span>.</p>

    <p class="note">User agents are encouraged to log this error in a developer console, to aid
    debugging.</p>
   </li>

   <li><p>Let <var>anchoredPattern</var> be the string "<code data-x="">^(?:</code>", followed by
   <var>pattern</var>, followed by "<code data-x="">)$</code>".</p></li>

   <li><p>Return ! <span>RegExpCreate</span>(<var>anchoredPattern</var>, "<code
   data-x="">v</code>").</p></li>
  </ol>

  <p class="note">The reasoning behind these steps, instead of just using the value of the <code
  data-x="attr-input-pattern">pattern</code> attribute directly, is twofold. First, we want to
  ensure that when matched against a string, the regular expression's start is anchored to the start
  of the string and its end to the end of the string. Second, we want to ensure that the regular
  expression is valid in standalone form, instead of only becoming valid after being surrounded by
  the "<code data-x="">^(?:</code>" and "<code data-x="">)$</code>" anchors.</p>

  <p>A <code>RegExp</code> object <var>regexp</var> <dfn data-x="RegExp match a
  string">matches</dfn> a string <var>input</var>, if !
  <span>RegExpBuiltinExec</span>(<var>regexp</var>, <var>input</var>) is not null.</p>

  <p><strong>Constraint validation</strong>: If the element's <span
  data-x="concept-fe-value">value</span> is not the empty string, and either the element's <code
  data-x="attr-input-multiple">multiple</code> attribute is not specified or it <span data-x="do not
  apply">does not apply</span> to the <code>input</code> element given its <code
  data-x="attr-input-type">type</code> attribute's current state, and the element has a
  <span>compiled pattern regular expression</span> but that regular expression does not <span
  data-x="RegExp match a string">match</span> the element's <span
  data-x="concept-fe-value">value</span>, then the element is <span>suffering from a pattern
  mismatch</span>.</p>

  <p><strong>Constraint validation</strong>: If the element's <span
  data-x="concept-fe-value">value</span> is not the empty string, and the element's <code
  data-x="attr-input-multiple">multiple</code> attribute is specified and <span
  data-x="concept-input-apply">applies</span> to the <code>input</code> element, and the element has
  a <span>compiled pattern regular expression</span> but that regular expression does not <span
  data-x="RegExp match a string">match</span> each of the element's <span
  data-x="concept-fe-values">values</span>, then the element is <span>suffering from a pattern
  mismatch</span>.</p>

  </div>

  <p>When an <code>input</code> element has a <code data-x="attr-input-pattern">pattern</code>
  attribute specified, authors should include a <dfn element-attr for="input"><code
  data-x="attr-input-title">title</code></dfn> attribute to give a description of the pattern.  User
  agents may use the contents of this attribute, if it is present, when informing the user that the
  pattern is not matched, or at any other suitable time, such as in a tooltip or read out by
  assistive technology when the control <span>gains focus</span>.</p>

  <div class="example">
   <p>For example, the following snippet:</p>
   <pre><code class="html">&lt;label&gt; Part number:
 &lt;input pattern="[0-9][A-Z]{3}" name="part"
        title="A part number is a digit followed by three uppercase letters."/&gt;
&lt;/label&gt;</code></pre>
   <p>...could cause the UA to display an alert such as:</p>
   <pre><samp>A part number is a digit followed by three uppercase letters.
You cannot submit this form when the field is incorrect.</samp></pre>
  </div>

  <p>When a control has a <code data-x="attr-input-pattern">pattern</code> attribute, the <code
  data-x="attr-input-title">title</code> attribute, if used, must describe the pattern. Additional
  information could also be included, so long as it assists the user in filling in the control.
  Otherwise, assistive technology would be impaired.</p>

  <p class="example">For instance, if the title attribute contained the caption of the control,
  assistive technology could end up saying something like <samp>The text you have entered does not
  match the required pattern. Birthday</samp>, which is not useful.</p>

  <p>UAs may still show the <code data-x="attr-title">title</code> in non-error situations (for
  example, as a tooltip when hovering over the control), so authors should be careful not to word
  <code data-x="attr-input-title">title</code>s as if an error has necessarily occurred.</p>


  <h6>The <code data-x="attr-input-min">min</code> and <code data-x="attr-input-max">max</code> attributes</h6>

  <!-- if more numeric control types start supporting multiple="", this will need to be refactored -->

  <p>Some form controls can have explicit constraints applied limiting the allowed range of values
  that the user can provide. Normally, such a range would be linear and continuous. A form control
  can <dfn data-x="has a periodic domain">have a periodic domain</dfn>, however, in which case the
  form control's broadest possible range is finite, and authors can specify explicit ranges within
  it that span the boundaries.</p>

  <p class="example">Specifically, the broadest range of a <code
  data-x="attr-input-type-time">type=time</code> control is midnight to midnight (24 hours), and
  authors can set both continuous linear ranges (such as 9pm to 11pm) and discontinuous ranges
  spanning midnight (such as 11pm to 1am).</p>

  <p>The <dfn element-attr for="input"><code data-x="attr-input-min">min</code></dfn> and <dfn
  element-attr for="input"><code data-x="attr-input-max">max</code></dfn> attributes indicate the
  allowed range of values for the element.</p>

  <div w-nodev>

  <p>Their syntax is defined by the section that defines the <code
  data-x="attr-input-type">type</code> attribute's current state.</p>

  <p>If the element has a <code data-x="attr-input-min">min</code> attribute, and the result of
  applying the <span data-x="concept-input-value-string-number">algorithm to convert a string to a
  number</span> to the value of the <code data-x="attr-input-min">min</code> attribute is a number,
  then that number is the element's <dfn data-x="concept-input-min">minimum</dfn>; otherwise, if the
  <code data-x="attr-input-type">type</code> attribute's current state defines a <dfn
  data-x="concept-input-min-default">default minimum</dfn>, then that is the <span
  data-x="concept-input-min">minimum</span>; otherwise, the element has no <span
  data-x="concept-input-min">minimum</span>.</p>

  <p>The <code data-x="attr-input-min">min</code> attribute also defines the <span
  data-x="concept-input-min-zero">step base</span>.</p>

  <p>If the element has a <code data-x="attr-input-max">max</code> attribute, and the result of
  applying the <span data-x="concept-input-value-string-number">algorithm to convert a string to a
  number</span> to the value of the <code data-x="attr-input-max">max</code> attribute is a number,
  then that number is the element's <dfn data-x="concept-input-max">maximum</dfn>; otherwise, if the
  <code data-x="attr-input-type">type</code> attribute's current state defines a <dfn
  data-x="concept-input-max-default">default maximum</dfn>, then that is the <span
  data-x="concept-input-max">maximum</span>; otherwise, the element has no <span
  data-x="concept-input-max">maximum</span>.</p>

  </div>

  <p>If the element does not <span data-x="has a periodic domain">have a periodic domain</span>, the
  <code data-x="attr-input-max">max</code> attribute's value (the <span
  data-x="concept-input-max">maximum</span>) must not be less than the <code
  data-x="attr-input-min">min</code> attribute's value (its <span
  data-x="concept-input-min">minimum</span>).</p>

  <div w-nodev>

  <p class="note">If an element that does not <span data-x="has a periodic domain">have a periodic
  domain</span> has a <span data-x="attr-input-max">maximum</span> that is less than its <span
  data-x="attr-input-min">minimum</span>, then so long as the element has a <span
  data-x="concept-fe-value">value</span>, it will either be <span>suffering from an underflow</span>
  or <span>suffering from an overflow</span>.</p>

  <p>An element <dfn>has a reversed range</dfn> if it <span>has a periodic domain</span> and its
  <span data-x="concept-input-max">maximum</span> is less than its <span
  data-x="concept-input-min">minimum</span>.</p>

  </div>

  <p>An element <dfn data-x="have range limitations">has range limitations</dfn> if it has a defined
  <span data-x="concept-input-min">minimum</span> or a defined <span
  data-x="concept-input-max">maximum</span>.</p>

  <div w-nodev>

  <p><strong>Constraint validation</strong>: When the element has a <span
  data-x="attr-input-min">minimum</span> and does not <span data-x="has a reversed range">have a
  reversed range</span>, and the result of applying the <span
  data-x="concept-input-value-string-number">algorithm to convert a string to a number</span> to
  the string given by the element's <span data-x="concept-fe-value">value</span> is a number, and
  the number obtained from that algorithm is less than the <span
  data-x="attr-input-min">minimum</span>, the element is <span>suffering from an
  underflow</span>.</p>

  <p><strong>Constraint validation</strong>: When the element has a <span
  data-x="attr-input-max">maximum</span> and does not <span data-x="has a reversed range">have a
  reversed range</span>, and the result of applying the <span
  data-x="concept-input-value-string-number">algorithm to convert a string to a number</span> to
  the string given by the element's <span data-x="concept-fe-value">value</span> is a number, and
  the number obtained from that algorithm is more than the <span
  data-x="attr-input-max">maximum</span>, the element is <span>suffering from an
  overflow</span>.</p>

  <p><strong>Constraint validation</strong>: When an element <span>has a reversed range</span>,
  and the result of applying the <span data-x="concept-input-value-string-number">algorithm to
  convert a string to a number</span> to the string given by the element's <span
  data-x="concept-fe-value">value</span> is a number, and the number obtained from that algorithm
  is more than the <span data-x="attr-input-max">maximum</span> <em>and</em> less than the <span
  data-x="attr-input-min">minimum</span>, the element is simultaneously <span>suffering from an
  underflow</span> and <span>suffering from an overflow</span>.</p>

  </div>

  <div class="example">

   <p>The following date control limits input to dates that are before the 1980s:</p>

   <pre><code class="html">&lt;input name=bday type=date max="1979-12-31"></code></pre>

  </div>

  <div class="example">

   <p>The following number control limits input to whole numbers greater than zero:</p>

   <pre><code class="html">&lt;input name=quantity required="" type="number" min="1" value="1"></code></pre>

  </div>

  <div class="example">

   <p>The following time control limits input to those minutes that occur between 9pm and 6am,
   defaulting to midnight:</p>

   <pre><code class="html">&lt;input name="sleepStart" type=time min="21:00" max="06:00" step="60" value="00:00"></code></pre>

  </div>



  <h6>The <code data-x="attr-input-step">step</code> attribute</h6>

  <!-- if more numeric control types start supporting multiple="", this will need to be refactored -->

  <p>The <dfn element-attr for="input"><code data-x="attr-input-step">step</code></dfn> attribute
  indicates the granularity that is expected (and required) of the <span
  data-x="concept-fe-value">value</span> or <span data-x="concept-fe-values">values</span>, by
  limiting the allowed values. <span w-nodev>The section that defines the <code
  data-x="attr-input-type">type</code> attribute's current state also defines the <dfn
  data-x="concept-input-step-default">default step</dfn>, the <dfn
  data-x="concept-input-step-scale">step scale factor</dfn>, and in some cases the <dfn
  data-x="concept-input-step-default-base">default step base</dfn>, which are used in processing the
  attribute as described below.</span></p>

  <p>The <code data-x="attr-input-step">step</code> attribute, if specified, must either have a
  value that is a <span>valid floating-point number</span> that <span data-x="rules for parsing
  floating-point number values">parses</span> to a number that is greater than zero, or must have a
  value that is an <span>ASCII case-insensitive</span> match for the string "<code
  data-x="">any</code>".</p>

  <div w-nodev>

  <p>The attribute provides the <dfn data-x="concept-input-step">allowed value step</dfn> for the
  element, as follows:</p>

  <ol>
   <li><p>If the attribute does not <span data-x="concept-input-apply">apply</span>, then there is
   no <span data-x="concept-input-step">allowed value step</span>.</p></li>

   <li><p>Otherwise, if the attribute is absent, then the <span data-x="concept-input-step">allowed
   value step</span> is the <span data-x="concept-input-step-default">default step</span> multiplied
   by the <span data-x="concept-input-step-scale">step scale factor</span>.</p></li>

   <li><p>Otherwise, if the attribute's value is an <span>ASCII case-insensitive</span> match for the
   string "<code data-x="">any</code>", then there is no <span data-x="concept-input-step">allowed
   value step</span>.</p></li>

   <li><p>Otherwise, if the <span>rules for parsing floating-point number values</span>, when they are
   applied to the attribute's value, return an error, zero, or a number less than zero, then the
   <span data-x="concept-input-step">allowed value step</span> is the <span
   data-x="concept-input-step-default">default step</span> multiplied by the <span
   data-x="concept-input-step-scale">step scale factor</span>.</p></li>

   <li><p>Otherwise, the <span data-x="concept-input-step">allowed value step</span> is the number
   returned by the <span>rules for parsing floating-point number values</span> when they are applied
   to the attribute's value, multiplied by the <span data-x="concept-input-step-scale">step scale
   factor</span>.</p></li>
  </ol>

  <p>The <dfn data-x="concept-input-min-zero">step base</dfn> is the value returned by the following
  algorithm:</p>

  <ol>
   <li><p>If the element has a <code data-x="attr-input-min">min</code> content attribute, and the
   result of applying the <span data-x="concept-input-value-string-number">algorithm to convert a
   string to a number</span> to the value of the <code data-x="attr-input-min">min</code> content
   attribute is not an error, then return that result.</p></li>

   <li><p>If the element has a <code data-x="attr-input-value">value</code> content attribute, and
   the result of applying the <span data-x="concept-input-value-string-number">algorithm to convert
   a string to a number</span> to the value of the <code data-x="attr-input-value">value</code>
   content attribute is not an error, then return that result.</p></li>

   <li><p>If a <span data-x="concept-input-step-default-base">default step base</span> is defined for
   this element given its <code data-x="attr-input-type">type</code> attribute's state, then return
   it.</p>

   <li><p>Return zero.</p></li>
  </ol>

  <p><strong>Constraint validation</strong>: When the element has an <span
  data-x="concept-input-step">allowed value step</span>, and the result of applying the <span
  data-x="concept-input-value-string-number">algorithm to convert a string to a number</span> to
  the string given by the element's <span data-x="concept-fe-value">value</span> is a number, and
  that number subtracted from the <span data-x="concept-input-min-zero">step base</span> is not an
  integral multiple of the <span data-x="concept-input-step">allowed value step</span>, the
  element is <span>suffering from a step mismatch</span>.</p>

  </div>

  <div class="example">

   <p>The following range control only accepts values in the range 0..1, and allows 256 steps in
   that range:</p>

   <pre><code class="html">&lt;input name=opacity type=range min=0 max=1 step=0.00392156863></code></pre>

  </div>

  <div class="example">

   <p>The following control allows any time in the day to be selected, with any accuracy (e.g.
   thousandth-of-a-second accuracy or more):</p>

   <pre><code class="html">&lt;input name=favtime type=time step=any></code></pre>

   <p>Normally, time controls are limited to an accuracy of one minute.</p>

  </div>



  <h6>The <code data-x="attr-input-list">list</code> attribute</h6>

  <!-- if we add any more types of controls, the handling of multiple="" below needs refactoring -->

  <p>The <dfn element-attr for="input"><code data-x="attr-input-list">list</code></dfn> attribute is
  used to identify an element that lists predefined options suggested to the user.</p>

  <p>If present, its value must be the <span data-x="concept-id">ID</span> of a <code>datalist</code>
  element in the same <span>tree</span>.</p>

  <div w-nodev>

  <p>The <dfn data-x="concept-input-list">suggestions source element</dfn> is the first element in
  the <span>tree</span> in <span>tree order</span> to have an <span data-x="concept-id">ID</span>
  equal to the value of the <code data-x="attr-input-list">list</code> attribute, if that element
  is a <code>datalist</code> element. If there is no <code data-x="attr-input-list">list</code>
  attribute, or if there is no element with that <span data-x="concept-id">ID</span>, or if the
  first element with that <span data-x="concept-id">ID</span> is not a <code>datalist</code>
  element, then there is no <span data-x="concept-input-list">suggestions source
  element</span>.</p>

  <p>If there is a <span data-x="concept-input-list">suggestions source element</span>, then, when
  the user agent is allowing the user to edit the <code>input</code> element's <span
  data-x="concept-fe-value">value</span>, the user agent should offer the suggestions represented by
  the <span data-x="concept-input-list">suggestions source element</span> to the user in a manner
  suitable for the type of control used. If appropriate, the user agent should use the suggestion's
  <span data-x="concept-option-label">label</span> and <span
  data-x="concept-option-value">value</span> to identify the suggestion to the user.</p>

  <p>User agents are encouraged to filter the suggestions represented by the <span
  data-x="concept-input-list">suggestions source element</span> when the number of suggestions is
  large, including only the most relevant ones (e.g. based on the user's input so far). No precise
  threshold is defined, but capping the list at four to seven values is reasonable. If filtering
  based on the user's input, user agents should search within both the <span
  data-x="concept-option-label">label</span> and <span data-x="concept-option-value">value</span>
  of the suggestions for matches. User agents should consider how input variations affect the
  matching process. Unicode normalization should be applied so that different underlying Unicode
  code point sequences, caused by different keyboard- or input-specific mechanisms, do not
  interfere with the matching process. Case variations should be ignored, which may require
  language-specific case mapping. For examples of these, see <cite>Character Model for the World
  Wide Web: String Matching</cite>. User agents may also provide other matching features: for
  illustration, a few examples include matching different forms of kana to each other (or to kanji),
  ignoring accents, or applying spelling correction. <ref>CHARMODNORM</ref></p>

  <div class="example">
   <p>This text field allows you to choose a type of JavaScript function.</p>

   <pre><code class="html">&lt;input type="text" list="function-types">
&lt;datalist id="function-types">
  &lt;option value="function">function&lt;/option>
  &lt;option value="async function">async function&lt;/option>
  &lt;option value="function*">generator function&lt;/option>
  &lt;option value="=>">arrow function&lt;/option>
  &lt;option value="async =>">async arrow function&lt;/option>
  &lt;option value="async function*">async generator function&lt;/option>
&lt;/datalist></code></pre>

   <p>For user agents that follow the above suggestions, both the <span
   data-x="concept-option-label">label</span> and <span data-x="concept-option-value">value</span>
   would be shown:</p>

   <p><img src="/images/sample-datalist.svg" width="280" height="150" alt="A text box with a drop down button on the right hand side; with, below, a drop down box containing a list of the six values the left and the six labels on the right." class="darkmode-aware"></p>

   <p>Then, typing "<kbd>arrow</kbd>" or "<kbd>=></kbd>" would filter the list to the entries with
   labels "arrow function" and "async arrow function". Typing "<kbd>generator</kbd>" or
   "<kbd>*</kbd>" would filter the list to the entries with labels "generator function" and "async
   generator function".</p>
  </div>

  <p class="note">As always, user agents are free to make user interface decisions which are
  appropriate for their particular requirements and for the user's particular circumstances. However,
  this has historically been an area of confusion for implementers, web developers, and users alike,
  so we've given some "should" suggestions above.</p>


  <p>How user selections of suggestions are handled depends on whether the element is a control
  accepting a single value only, or whether it accepts multiple values:</p>

  <dl class="switch">

   <dt>If the element does not have a <code data-x="attr-input-multiple">multiple</code> attribute
   specified or if the <code data-x="attr-input-multiple">multiple</code> attribute <span data-x="do
   not apply">does not apply</span></dt>

   <dd>
    <p>When the user selects a suggestion, the <code>input</code> element's <span
    data-x="concept-fe-value">value</span> must be set to the selected suggestion's <span
    data-x="concept-option-value">value</span>, as if the user had written that value themself.</p>
   </dd>

   <!-- multiple types where there's a potentially growing number of values -->
   <dt>If the element's <code data-x="attr-input-type">type</code> attribute is in the <span
   data-x="attr-input-type-email">Email</span> state and the element has a <code
   data-x="attr-input-multiple">multiple</code> attribute specified<!--, and the <code
   data-x="attr-input-multiple">multiple</code> attribute does <span
   data-x="concept-input-apply">apply</span>--></dt>

   <dd>
    <p>When the user selects a suggestion, the user agent must either add a new entry to the
    <code>input</code> element's <span data-x="concept-fe-values">values</span>, whose value
    is the selected suggestion's <span data-x="concept-option-value">value</span>, or change an
    existing entry in the <code>input</code> element's <span
    data-x="concept-fe-values">values</span> to have the value given by the selected
    suggestion's <span data-x="concept-option-value">value</span>, as if the user had themself added
    an entry with that value, or edited an existing entry to be that value. Which behavior is to be
    applied depends on the user interface in an <span>implementation-defined</span> manner.</p>
   </dd>
  </dl>

  <hr>

  <p>If the <code data-x="attr-input-list">list</code> attribute <span data-x="do not apply">does not
  apply</span>, there is no <span data-x="concept-input-list">suggestions source element</span>.</p>

  </div>

  <div class="example">

   <p>This URL field offers some suggestions.</p>

   <pre><code class="html">&lt;label>Homepage: &lt;input name=hp type=url list=hpurls>&lt;/label>
&lt;datalist id=hpurls>
 &lt;option value="https://www.google.com/" label="Google">
 &lt;option value="https://www.reddit.com/" label="Reddit">
&lt;/datalist></code></pre>

   <p>Other URLs from the user's history might show also; this is up to the user agent.</p>

  </div>

  <div class="example">

   <p>This example demonstrates how to design a form that uses the autocompletion list feature while
   still degrading usefully in legacy user agents.</p>

   <p>If the autocompletion list is merely an aid, and is not important to the content, then simply
   using a <code>datalist</code> element with children <code>option</code> elements is enough. To
   prevent the values from being rendered in legacy user agents, they need to be placed inside the
   <code data-x="attr-option-value">value</code> attribute instead of inline.</p>

   <pre><code class="html">&lt;p&gt;
 &lt;label&gt;
  Enter a breed:
  &lt;input type="text" name="breed" list="breeds"&gt;
  &lt;datalist id="breeds"&gt;
   &lt;option value="Abyssinian"&gt;
   &lt;option value="Alpaca"&gt;
   &lt;!-- ... --&gt;
  &lt;/datalist&gt;
 &lt;/label&gt;
&lt;/p&gt;</code></pre>

   <p>However, if the values need to be shown in legacy UAs, then fallback content can be placed
   inside the <code>datalist</code> element, as follows:</p>

   <pre><code class="html">&lt;p&gt;
 &lt;label&gt;
  Enter a breed:
  &lt;input type="text" name="breed" list="breeds"&gt;
 &lt;/label&gt;
 &lt;datalist id="breeds"&gt;
  &lt;label&gt;
   or select one from the list:
   &lt;select name="breed"&gt;
    &lt;option value=""&gt; (none selected)
    &lt;option&gt;Abyssinian
    &lt;option&gt;Alpaca
    &lt;!-- ... --&gt;
   &lt;/select&gt;
  &lt;/label&gt;
 &lt;/datalist&gt;
&lt;/p&gt;</code></pre>

   <p>The fallback content will only be shown in UAs that don't support <code>datalist</code>. The
   options, on the other hand, will be detected by all UAs, even though they are not children of the
   <code>datalist</code> element.</p>

   <p>Note that if an <code>option</code> element used in a <code>datalist</code> is <code
   data-x="attr-option-selected">selected</code>, it will be selected by default by legacy UAs
   (because it affects the <code>select</code> element), but it will not have any effect on the
   <code>input</code> element in UAs that support <code>datalist</code>.</p>

  </div>



  <h6>The <code data-x="attr-input-placeholder">placeholder</code> attribute</h6>

  <!-- substantially similar text in the <textarea> section -->

  <p>The <dfn element-attr for="input"><code
  data-x="attr-input-placeholder">placeholder</code></dfn> attribute represents a <em>short</em>
  hint (a word or short phrase) intended to aid the user with data entry when the control has no
  value. A hint could be a sample value or a brief description of the expected format. The
  attribute, if specified, must have a value that contains no U+000A LINE FEED (LF) or U+000D
  CARRIAGE RETURN (CR) characters.</p>

  <p>The <code data-x="attr-input-placeholder">placeholder</code> attribute should not be used as an
  alternative to a <code>label</code>. For a longer hint or other advisory text, the <code
  data-x="attr-title">title</code> attribute is more appropriate.</p>

  <p class="note">These mechanisms are very similar but subtly different: the hint given by the
  control's <code>label</code> is shown at all times; the short hint given in the <code
  data-x="attr-input-placeholder">placeholder</code> attribute is shown before the user enters a
  value; and the hint in the <code data-x="attr-title">title</code> attribute is shown when the user
  requests further help.</p>

  <div w-nodev>

  <p>User agents should present this hint to the user, after having <span data-x="strip
  newlines">stripped newlines</span> from it, when the element's <span
  data-x="concept-fe-value">value</span> is the empty string, especially if the control is not
  <span>focused</span>.</p>

  <p>If a user agent normally doesn't show this hint to the user when the control is
  <span>focused</span>, then the user agent should nonetheless show the hint for the control if it
  was focused as a result of the <code data-x="attr-fe-autofocus">autofocus</code> attribute, since
  in that case the user will not have had an opportunity to examine the control before focusing
  it.</p>

  </div>

  <div class="example">

   <p>Here is an example of a mail configuration user interface that uses the <code
   data-x="attr-input-placeholder">placeholder</code> attribute:</p>

   <pre><code class="html">&lt;fieldset>
 &lt;legend>Mail Account&lt;/legend>
 &lt;p>&lt;label>Name: &lt;input type="text" name="fullname" placeholder="John Ratzenberger">&lt;/label>&lt;/p>
 &lt;p>&lt;label>Address: &lt;input type="email" name="address" placeholder="john@example.net">&lt;/label>&lt;/p>
 &lt;p>&lt;label>Password: &lt;input type="password" name="password">&lt;/label>&lt;/p>
 &lt;p>&lt;label>Description: &lt;input type="text" name="desc" placeholder="My Email Account">&lt;/label>&lt;/p>
&lt;/fieldset></code></pre>

  </div>

  <div class="example">

   <p>In situations where the control's content has one directionality but the placeholder needs to
   have a different directionality, Unicode's bidirectional-algorithm formatting characters can be
   used in the attribute value:</p>

   <pre><code class="html">&lt;input name=t1 type=tel placeholder="<strong>&amp;#x202B;</strong>&#x2009;<bdo dir=rtl>&#1585;&#1602;&#1605;&nbsp;&#1575;&#1604;&#1607;&#1575;&#1578;&#1601;&nbsp;1</bdo>&#x2009;<strong>&amp;#x202E;</strong>">
&lt;input name=t2 type=tel placeholder="<strong>&amp;#x202B;</strong>&#x2009;<bdo dir=rtl>&#1585;&#1602;&#1605;&nbsp;&#1575;&#1604;&#1607;&#1575;&#1578;&#1601;&nbsp;2</bdo>&#x2009;<strong>&amp;#x202E;</strong>"></code></pre>

   <p>For slightly more clarity, here's the same example using numeric character references instead of inline Arabic:</p>

   <pre><code class="html">&lt;input name=t1 type=tel placeholder="<strong>&amp;#x202B;</strong>&amp;#1585;&amp;#1602;&amp;#1605;&nbsp;&amp;#1575;&amp;#1604;&amp;#1607;&amp;#1575;&amp;#1578;&amp;#1601;&nbsp;1<strong>&amp;#x202E;</strong>">
&lt;input name=t2 type=tel placeholder="<strong>&amp;#x202B;</strong>&amp;#1585;&amp;#1602;&amp;#1605;&nbsp;&amp;#1575;&amp;#1604;&amp;#1607;&amp;#1575;&amp;#1578;&amp;#1601;&nbsp;2<strong>&amp;#x202E;</strong>"></code></pre>

  </div>




  <h5 id="common-input-element-apis">Common <code>input</code> element APIs</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>input</var>.<span subdfn data-x="dom-input-value">value</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current <span data-x="concept-fe-value">value</span> of the form control.</p>

    <p>Can be set, to change the value.</p>

    <p>Throws an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> if it is
    set to any value other than the empty string when the control is a file upload control.</p>
   </dd>

   <dt><code data-x=""><var>input</var>.<span subdfn data-x="dom-input-checked">checked</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current <span data-x="concept-fe-checked">checkedness</span> of the form
    control.</p>

    <p>Can be set, to change the <span data-x="concept-fe-checked">checkedness</span>.</p>
   </dd>

   <dt><code data-x=""><var>input</var>.<span subdfn data-x="dom-input-files">files</span> [ = <var>files</var> ]</code></dt>

   <dd>
    <p>Returns a <code>FileList</code> object listing the <span
    data-x="concept-input-type-file-selected">selected files</span> of the form control.</p>

    <p>Returns null if the control isn't a file control.</p>

    <p>Can be set to a <code>FileList</code> object to change the <span
    data-x="concept-input-type-file-selected">selected files</span> of the form control. For
    instance, as the result of a drag-and-drop operation.</p>
   </dd>

   <dt><code data-x=""><var>input</var>.<span subdfn data-x="dom-input-valueAsDate">valueAsDate</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns a <code>Date</code> object representing the form control's <span
    data-x="concept-fe-value">value</span>, if applicable; otherwise, returns null.</p>

    <p>Can be set, to change the value.</p>

    <p>Throws an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> if the
    control isn't date- or time-based.</p>
   </dd>

   <dt><code data-x=""><var>input</var>.<span subdfn data-x="dom-input-valueAsNumber">valueAsNumber</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns a number representing the form control's <span
    data-x="concept-fe-value">value</span>, if applicable; otherwise, returns NaN.</p>

    <p>Can be set, to change the value. Setting this to NaN will set the underlying value to the
    empty string.</p>

    <p>Throws an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> if the
    control is neither date- or time-based nor numeric.</p>
   </dd>

   <dt><code data-x=""><var>input</var>.<span subdfn data-x="dom-input-stepUp">stepUp</span>([ <var>n</var> ])</code></dt>
   <dt><code data-x=""><var>input</var>.<span subdfn data-x="dom-input-stepDown">stepDown</span>([ <var>n</var> ])</code></dt>

   <dd>
    <p>Changes the form control's <span data-x="concept-fe-value">value</span> by the value given in
    the <code data-x="attr-input-step">step</code> attribute, multiplied by <var>n</var>. The
    default value for <var>n</var> is 1.</p>

    <p>Throws <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> if the control
    is neither date- or time-based nor numeric, or if the <code data-x="attr-input-step">step</code>
    attribute's value is "<code data-x="">any</code>".</p>
   </dd>

   <dt><code data-x=""><var>input</var>.<span subdfn data-x="dom-input-list">list</span></code></dt>

   <dd><p>Returns the <code>datalist</code> element indicated by the <code
   data-x="attr-input-list">list</code> attribute.</p></dd>

   <dt><code data-x=""><var>input</var>.<span subdfn data-x="dom-input-showPicker">showPicker</span>()</code></dt>

   <dd>
    <p>Shows any applicable picker UI for <var>input</var>, so that the user can select a value.</p>

    <p>If <var>input</var> does not <span data-x="input-support-picker">support a picker</span>,
    this method does nothing.</p>

    <p>Throws an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> if
    <var>input</var> is not <span data-x="concept-fe-mutable">mutable</span>.</p>

    <p>Throws a <span>"<code>NotAllowedError</code>"</span> <code>DOMException</code> if called
    without <span data-x="transient activation">transient user activation</span>.</p>

    <p>Throws a <span>"<code>SecurityError</code>"</span> <code>DOMException</code> if
    <var>input</var> is inside a cross-origin <code>iframe</code>, unless <var>input</var> is in the
    <span data-x="attr-input-type-file">File Upload</span> or <span
    data-x="attr-input-type-color">Color</span> states.</p>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLInputElement"><code data-x="dom-input-value">value</code></dfn> IDL
  attribute allows scripts to manipulate the <span data-x="concept-fe-value">value</span> of an
  <code>input</code> element. The attribute is in one of the following modes, which define its
  behavior:</p>

  <dl>
   <dt><dfn data-x="dom-input-value-value">value</dfn>

   <dd>
    <p>On getting, return the current <span data-x="concept-fe-value">value</span> of the
    element.</p>

    <p>On setting:</p>

    <ol>
     <li><p>Let <var>oldValue</var> be the element's <span
     data-x="concept-fe-value">value</span>.</p></li>

     <li><p>Set the element's <span data-x="concept-fe-value">value</span> to the new
     value.</p></li>

     <li><p>Set the element's <span data-x="concept-fe-dirty">dirty value flag</span> to
     true.</p></li>

     <li><p>Invoke the <span>value sanitization algorithm</span>, if the element's <code
     data-x="attr-input-type">type</code> attribute's current state defines one.</p></li>

     <li><p>If the element's <span data-x="concept-fe-value">value</span> (after applying the
     <span>value sanitization algorithm</span>) is different from <var>oldValue</var>, and the
     element has a <span data-x="concept-textarea/input-cursor">text entry cursor position</span>,
     move the <span data-x="concept-textarea/input-cursor">text entry cursor position</span> to the
     end of the text control, unselecting any selected text and <span data-x="set the selection
     direction">resetting the selection direction</span> to "<code data-x="">none</code>".</p></li>
    </ol>
   </dd>

   <dt><dfn data-x="dom-input-value-default">default</dfn>

   <dd>
    <p>On getting, if the element has a <code data-x="attr-input-value">value</code> content
    attribute, return that attribute's value; otherwise, return the empty string.</p>

    <p>On setting, set the value of the element's <code data-x="attr-input-value">value</code>
    content attribute to the new value.</p>
   </dd>

   <dt><dfn data-x="dom-input-value-default-on">default/on</dfn>

   <dd>
    <p>On getting, if the element has a <code data-x="attr-input-value">value</code> content
    attribute, return that attribute's value; otherwise, return the string "<code
    data-x="">on</code>".</p>

    <p>On setting, set the value of the element's <code data-x="attr-input-value">value</code>
    content attribute to the new value.</p>
   </dd>

   <dt><dfn data-x="dom-input-value-filename">filename</dfn>

   <dd id="fakepath-orly">

    <p>On getting, return the string "<code data-x="">C:\fakepath\</code>" followed by the name of
    the first file in the list of <span data-x="concept-input-type-file-selected">selected
    files</span>, if any, or the empty string if the list is empty.</p>

    <p>On setting, if the new value is the empty string, empty the list of <span
    data-x="concept-input-type-file-selected">selected files</span>; otherwise, throw an
    <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p>

    <p class="note">This "fakepath" requirement is a sad accident of history. See <a
    href="#fakepath-srsly">the example in the File Upload state section</a> for more
    information.</p>

    <p class="note">Since <span data-x="concept-input-file-path">path components</span> are not
    permitted in filenames in the list of <span data-x="concept-input-type-file-selected">selected
    files</span>, the "\fakepath\" cannot be mistaken for a path component.</p>
   </dd>
  </dl>

  <hr>

  <p>The <dfn attribute for="HTMLInputElement"><code data-x="dom-input-checked">checked</code></dfn>
  IDL attribute allows scripts to manipulate the <span
  data-x="concept-fe-checked">checkedness</span> of an <code>input</code> element. On getting, it
  must return the current <span data-x="concept-fe-checked">checkedness</span> of the element; and
  on setting, it must set the element's <span data-x="concept-fe-checked">checkedness</span> to the
  new value and set the element's <span data-x="concept-input-checked-dirty-flag">dirty checkedness
  flag</span> to true.</p>

  <hr>

  <p>The <dfn attribute for="HTMLInputElement"><code data-x="dom-input-files">files</code></dfn> IDL
  attribute allows scripts to access the element's <span
  data-x="concept-input-type-file-selected">selected files</span>.

  <p>On getting, if the IDL attribute <span data-x="concept-input-apply">applies</span>, it must
  return a <code>FileList</code> object that represents the current <span
  data-x="concept-input-type-file-selected">selected files</span>. The same object must be returned
  until the list of <span data-x="concept-input-type-file-selected">selected files</span> changes.
  If the IDL attribute <span data-x="do not apply">does not apply</span>, then it must instead
  return null. <ref>FILEAPI</ref></p>

  <p>On setting, it must run these steps:

  <ol>
   <li><p>If the IDL attribute <span data-x="do not apply">does not apply</span> or the given value
   is null, then return.</p></li>
   <!-- This is inconsistent with other attributes, but Chrome and Safari already shipped so doesn't
        seem worth the churn. -->

   <li><p>Replace the element's <span data-x="concept-input-type-file-selected">selected
   files</span> with the given value.</p></li>
  </ol>

  <hr>

  <p>The <dfn attribute for="HTMLInputElement"><code
  data-x="dom-input-valueAsDate">valueAsDate</code></dfn> IDL attribute represents the <span
  data-x="concept-fe-value">value</span> of the element, interpreted as a date.</p>

  <p>On getting, if the <code data-x="dom-input-valueAsDate">valueAsDate</code> attribute <span
  data-x="do not apply">does not apply</span>, as defined for the <code>input</code> element's <code
  data-x="attr-input-type">type</code> attribute's current state, then return null. Otherwise, run
  the <span data-x="concept-input-value-string-date">algorithm to convert a string to a
  <code>Date</code> object</span> defined for that state to the element's <span
  data-x="concept-fe-value">value</span>; if the algorithm returned a <code>Date</code> object, then
  return it, otherwise, return null.</p>

  <p>On setting, if the <code data-x="dom-input-valueAsDate">valueAsDate</code> attribute <span
  data-x="do not apply">does not apply</span>, as defined for the <code>input</code> element's <code
  data-x="attr-input-type">type</code> attribute's current state, then throw an
  <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>; otherwise, if the new
  value is not null and not a <code >Date</code> object throw a <code>TypeError</code> exception;
  otherwise, if the new value is null or a <code>Date</code> object representing the NaN time value,
  then set the <span data-x="concept-fe-value">value</span> of the element to the empty string;
  otherwise, run the <span data-x="concept-input-value-date-string">algorithm to convert a
  <code>Date</code> object to a string</span>, as defined for that state, on the new value, and set
  the <span data-x="concept-fe-value">value</span> of the element to the resulting string.</p>

  <hr>

  <p>The <dfn attribute for="HTMLInputElement"><code
  data-x="dom-input-valueAsNumber">valueAsNumber</code></dfn> IDL attribute represents the <span
  data-x="concept-fe-value">value</span> of the element, interpreted as a number.</p>

  <p>On getting, if the <code data-x="dom-input-valueAsNumber">valueAsNumber</code> attribute <span
  data-x="do not apply">does not apply</span>, as defined for the <code>input</code> element's <code
  data-x="attr-input-type">type</code> attribute's current state, then return a Not-a-Number (NaN)
  value. Otherwise, run the <span data-x="concept-input-value-string-number">algorithm to convert a
  string to a number</span> defined for that state to the element's <span
  data-x="concept-fe-value">value</span>; if the algorithm returned a number, then return it,
  otherwise, return a Not-a-Number (NaN) value.</p>

  <p>On setting, if the new value is infinite, then throw a <code>TypeError</code> exception.
  Otherwise, if the <code data-x="dom-input-valueAsNumber">valueAsNumber</code> attribute <span
  data-x="do not apply">does not apply</span>, as defined for the <code>input</code> element's <code
  data-x="attr-input-type">type</code> attribute's current state, then throw an
  <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>. Otherwise, if the new
  value is a Not-a-Number (NaN) value, then set the <span data-x="concept-fe-value">value</span> of
  the element to the empty string. Otherwise, run the <span
  data-x="concept-input-value-number-string">algorithm to convert a number to a string</span>, as
  defined for that state, on the new value, and set the <span data-x="concept-fe-value">value</span>
  of the element to the resulting string.</p>

  <hr>

  <p>The <dfn method for="HTMLInputElement"><code
  data-x="dom-input-stepDown">stepDown(<var>n</var>)</code></dfn> and <dfn method
  for="HTMLInputElement"><code data-x="dom-input-stepUp">stepUp(<var>n</var>)</code></dfn> methods,
  when invoked, must run the following algorithm:</p>

  <ol>
   <li><p>If the <code data-x="dom-input-stepDown">stepDown()</code> and <code
   data-x="dom-input-stepUp">stepUp()</code> methods <span>do not apply</span>, as defined for the
   <code>input</code> element's <code data-x="attr-input-type">type</code> attribute's current state,
   then throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If the element has no <span data-x="concept-input-step">allowed value step</span>, then
   throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If the element has a <span data-x="concept-input-min">minimum</span> and a <span
   data-x="concept-input-max">maximum</span> and the <span data-x="concept-input-min">minimum</span>
   is greater than the <span data-x="concept-input-max">maximum</span>, then return.</p>

   <li><p>If the element has a <span data-x="concept-input-min">minimum</span> and a <span
   data-x="concept-input-max">maximum</span> and there is no value greater than or equal to the
   element's <span data-x="concept-input-min">minimum</span> and less than or equal to the element's
   <span data-x="concept-input-max">maximum</span> that, when subtracted from the <span
   data-x="concept-input-min-zero">step base</span>, is an integral multiple of the <span
   data-x="concept-input-step">allowed value step</span>, then return.</p>

   <li><p>If applying the <span data-x="concept-input-value-string-number">algorithm to convert a
   string to a number</span> to the string given by the element's <span
   data-x="concept-fe-value">value</span> does not result in an error, then let <var>value</var> be the result of that algorithm. Otherwise, let <var>value</var>
   be zero.</p></li>

   <li><p>Let <var>valueBeforeStepping</var> be <var>value</var>.</p></li>

   <li>
    <p>If <var>value</var> subtracted from the <span data-x="concept-input-min-zero">step
    base</span> is not an integral multiple of the <span data-x="concept-input-step">allowed value
    step</span>, then set <var>value</var> to the nearest value that, when subtracted from
    the <span data-x="concept-input-min-zero">step base</span>, is an integral multiple of the <span
    data-x="concept-input-step">allowed value step</span>, and that is less than <var>value</var> if
    the method invoked was the <code data-x="dom-input-stepDown">stepDown()</code> method, and more
    than <var>value</var> otherwise.</p>

    <p>Otherwise (<var>value</var> subtracted from the <span
    data-x="concept-input-min-zero">step base</span> is an integral multiple of the <span
    data-x="concept-input-step">allowed value step</span>):</p>

    <ol>
     <li><p>Let <var>n</var> be the argument.</p></li>

     <li><p>Let <var>delta</var> be the <span data-x="concept-input-step">allowed value
     step</span> multiplied by <var>n</var>.</p></li>

     <li><p>If the method invoked was the <code data-x="dom-input-stepDown">stepDown()</code> method,
     negate <var>delta</var>.</p></li>

     <li><p>Let <var>value</var> be the result of adding <var>delta</var> to <var>value</var>.</p></li>
    </ol>
   </li>

   <li><p>If the element has a <span data-x="concept-input-min">minimum</span>, and <var>value</var>
   is less than that <span data-x="concept-input-min">minimum</span>, then set
   <var>value</var> to the smallest value that, when subtracted from the <span
   data-x="concept-input-min-zero">step base</span>, is an integral multiple of the <span
   data-x="concept-input-step">allowed value step</span>, and that is more than or equal to
   that <span data-x="concept-input-min">minimum</span>.</p></li>

   <li><p>If the element has a <span data-x="concept-input-max">maximum</span>, and <var>value</var>
   is greater than that <span data-x="concept-input-max">maximum</span>, then
   set <var>value</var> to the largest value that, when subtracted from the <span
   data-x="concept-input-min-zero">step base</span>, is an integral multiple of the <span
   data-x="concept-input-step">allowed value step</span>, and that is less than or equal to
   that <span data-x="concept-input-max">maximum</span>.</p></li>

   <li>
    <p>If either the method invoked was the <code data-x="dom-input-stepDown">stepDown()</code>
    method and <var>value</var> is greater than <var>valueBeforeStepping</var>, or the method
    invoked was the <code data-x="dom-input-stepUp">stepUp()</code> method and <var>value</var> is
    less than <var>valueBeforeStepping</var>, then return.</p>

    <div class="example">
     <p>This ensures that invoking the <code data-x="dom-input-stepUp">stepUp()</code> method on the
     <code>input</code> element in the following example does not change the <span
     data-x="concept-fe-value">value</span> of that element:</p>

     <pre><code class="html">&lt;input type=number value=1 max=0></code></pre>
    </div>
   </li>

   <li><p>Let <var>value as string</var> be the result of running the <span
   data-x="concept-input-value-number-string">algorithm to convert a number to a string</span>, as
   defined for the <code>input</code> element's <code data-x="attr-input-type">type</code>
   attribute's current state, on <var>value</var>.</p></li>

   <li><p>Set the <span data-x="concept-fe-value">value</span> of the element to <var>value
   as string</var>.</p></li>
  </ol>

  <hr>

  <p>The <dfn attribute for="HTMLInputElement"><code data-x="dom-input-list">list</code></dfn> IDL
  attribute must return the current <span data-x="concept-input-list">suggestions source
  element</span>, if any, or null otherwise.</p>

  <hr>

  <p>The <code>HTMLInputElement</code> <dfn method for="HTMLInputElement"><code
  data-x="dom-input-showPicker">showPicker()</code></dfn> and <code>HTMLSelectElement</code> <dfn
  method for="HTMLSelectElement"><code data-x="dom-select-showPicker">showPicker()</code></dfn>
  method steps are:</p>

  <ol>
   <li><p>If <span>this</span> is not <span data-x="concept-fe-mutable">mutable</span>, then throw
   an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li>
    <p>If <span>this</span>'s <span>relevant settings object</span>'s <span
    data-x="concept-settings-object-origin">origin</span> is not <span>same origin</span> with
    <span>this</span>'s <span>relevant settings object</span>'s <span>top-level origin</span>, and
    <span>this</span> is a <code>select</code> element, or
    <span>this</span>'s <code data-x="attr-input-type">type</code> attribute is not in the <span
    data-x="attr-input-type-file">File Upload</span> state or <span
    data-x="attr-input-type-color">Color</span> state, then throw a
    <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p>

    <p class="note"><span data-x="attr-input-type-file">File</span> and <span
    data-x="attr-input-type-color">Color</span> inputs are exempted from this check for historical
    reason: their <span>input activation behavior</span> also shows their pickers, and has never
    been guarded by an origin check.</p>
   </li>

   <li><p>If <span>this</span>'s <span>relevant global object</span> does not have <span>transient
   activation</span>, then throw a <span>"<code>NotAllowedError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>If <span>this</span> is a <code>select</code> element, and <span>this</span> is not
   <span>being rendered</span>, then throw a <span>"<code>NotSupportedError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p><span>Show the picker, if applicable</span>, for <span>this</span>.</p></li>
  </ol>

  <p>To <dfn>show the picker, if applicable</dfn> for an <code>input</code> or <code>select</code>
  element <var>element</var>:</p>

  <ol>
   <li><p>If <var>element</var>'s <span>relevant global object</span> does not have <span>transient
   activation</span>, then return.</p></li>

   <li><p>If <var>element</var> is not <i data-x="concept-fe-mutable">mutable</i>, then
   return.</p></li>

   <li><p><span>Consume user activation</span> given <span>element</span>'s <span>relevant global
   object</span>.</p></li>

   <li><p>If <var>element</var> does not <span data-x="input-support-picker">support a
   picker</span>, then return.</p></li>

   <li>
    <p>If <var>element</var> is an <code>input</code> element and <var>element</var>'s <code
    data-x="attr-input-type">type</code> attribute is in the <span
    data-x="attr-input-type-file">File Upload</span> state, then run these steps <span>in
    parallel</span>:</p>

    <ol>
     <li><p>Optionally, wait until any prior execution of this algorithm has terminated.</p></li>

     <li><p>Let <var>dismissed</var> be the result of <span>WebDriver BiDi file dialog opened</span>
     with <var>element</var>.</p></li>

     <li>
      <p>If <var>dismissed</var> is false:</p>

      <ol>
       <li><p>Display a prompt to the user requesting that the user specify some files. If the <code
       data-x="attr-input-multiple">multiple</code> attribute is not set on <var>element</var>, there
       must be no more than one file selected; otherwise, any number may be selected. Files can be
       from the filesystem or created on the fly, e.g., a picture taken from a camera connected to the
       user's device.</p></li>

       <li><p>Wait for the user to have made their selection.</p></li>
      </ol>
     </li>

     <li><p>If <var>dismissed</var> is true or if the user dismissed the prompt without changing
     their selection, then <span>queue an element task</span> on the <span>user interaction task
     source</span> given <var>element</var> to <span data-x="concept-event-fire">fire an
     event</span> named <code data-x="event-cancel">cancel</code> at <var>element</var>, with the
     <code data-x="dom-Event-bubbles">bubbles</code> attribute initialized to true.</p></li>

     <li><p>Otherwise, <span>update the file selection</span> for <var>element</var>.</p></li>
    </ol>

    <p class="note">As with all user interface specifications, user agents have a good deal of
    freedom in how they interpret these requirements. The above text implies that a user either
    dismisses the prompt or changes their selection; exactly one of these will be true. But the
    mapping of these possibilities to specific user interface elements is not mandated by the
    standard. For example, a user agent might interpret clicking the "Cancel" button when files were
    previously selected as a change of selection to select zero files, thus firing <code
    data-x="event-input">input</code> and <code data-x="event-change">change</code>. Or it might
    interpret such a click as a dismissal that leaves the selection unchanged, thus firing <code
    data-x="event-cancel">cancel</code>. Similarly, it's up to the user agent whether re-selecting
    the same files as were previously selected counts as a dismissal, or as a change of
    selection.</p>
   </li>

   <li>
    <p>Otherwise, the user agent should show the relevant user interface for selecting a value for
    <var>element</var>, in the way it normally would when the user interacts with the control.</p>

    <p>When showing such a user interface, it must respect the requirements stated in the relevant
    parts of the specification for how <var>element</var> behaves given its <code
    data-x="attr-input-type">type</code> attribute state. (For example, various sections describe
    restrictions on the resulting <span data-x="concept-fe-value">value</span> string.)</p>

    <p>This step can have side effects, such as closing other pickers that were previously shown by
    this algorithm. (If this closes a file selection picker, then per the above that will lead to
    firing either <code data-x="event-input">input</code> and <code
    data-x="event-change">change</code> events, or a <code data-x="event-cancel">cancel</code>
    event.)</p>
   </li>
  </ol>

  </div>


  <div w-nodev>

  <h5 id="common-input-element-events">Common event behaviors</h5>

  <p>When the <code data-x="event-input">input</code> and <code
  data-x="event-change">change</code> events <span data-x="concept-input-apply">apply</span>
  (which is the case for all <code>input</code> controls other than <span
  data-x="concept-button">buttons</span> and those with the <code
  data-x="attr-input-type">type</code> attribute in the <span
  data-x="attr-input-type-hidden">Hidden</span> state), the events are fired to indicate that the
  user has interacted with the control. The <code data-x="event-input">input</code> event fires
  whenever the user has modified the data of the control. The <code
  data-x="event-change">change</code> event fires when the value is committed, if that makes sense
  for the control, or else when the control <a href="#unfocus-causes-change-event">loses focus</a>.
  In all cases, the <code data-x="event-input">input</code> event comes before the corresponding
  <code data-x="event-change">change</code> event (if any).</p>

  <p>When an <code>input</code> element has a defined <span>input activation behavior</span>, the
  rules for dispatching these events, if they <span data-x="concept-input-apply">apply</span>, are
  given in the section above that defines the <code data-x="attr-input-type">type</code> attribute's
  state. (This is the case for all <code>input</code> controls with the <code
  data-x="attr-input-type">type</code> attribute in the <span
  data-x="attr-input-type-checkbox">Checkbox</span> state, the <span
  data-x="attr-input-type-radio">Radio Button</span> state, or the <span
  data-x="attr-input-type-file">File Upload</span> state.)</p>

  <p>For <code>input</code> elements without a defined <span>input activation behavior</span>, but
  to which these events <span data-x="concept-input-apply">apply</span>, and for which the user
  interface involves both interactive manipulation and an explicit commit action, then when the
  user changes the element's <span data-x="concept-fe-value">value</span>, the user agent must
  <span>queue an element task</span> on the <span>user interaction task source</span> given the
  <code>input</code> element to <span data-x="concept-event-fire">fire an event</span> named <code
  data-x="event-input">input</code> at the <code>input</code> element, with the <code
  data-x="dom-Event-bubbles">bubbles</code> and <code data-x="dom-Event-composed">composed</code>
  attributes initialized to true, and any time the user commits the change, the user agent must
  <span>queue an element task</span> on the <span>user interaction task source</span> given the
  <code>input</code> element to set its <span>user validity</span> to true and <span
  data-x="concept-event-fire">fire an event</span> named <code data-x="event-change">change</code>
  at the <code>input</code> element, with the <code data-x="dom-Event-bubbles">bubbles</code>
  attribute initialized to true.</p>

  <p class="example">An example of a user interface involving both interactive manipulation and a
  commit action would be a <span data-x="attr-input-type-range">Range</span> controls that use a
  slider, when manipulated using a pointing device. While the user is dragging the control's knob,
  <code data-x="event-input">input</code> events would fire whenever the position changed,
  whereas the <code data-x="event-change">change</code> event would only fire when the user
  let go of the knob, committing to a specific value.</p>

  <p>For <code>input</code> elements without a defined <span>input activation behavior</span>, but
  to which these events <span data-x="concept-input-apply">apply</span>, and for which the user
  interface involves an explicit commit action but no intermediate manipulation, then any time the
  user commits a change to the element's <span data-x="concept-fe-value">value</span>, the user
  agent must <span>queue an element task</span> on the <span>user interaction task source</span>
  given the <code>input</code> element to first <span data-x="concept-event-fire">fire an
  event</span> named <code data-x="event-input">input</code> at the <code>input</code> element, with
  the <code data-x="dom-Event-bubbles">bubbles</code> and <code
  data-x="dom-Event-composed">composed</code> attributes initialized to true, and then <span
  data-x="concept-event-fire">fire an event</span> named <code data-x="event-change">change</code>
  at the <code>input</code> element, with the <code data-x="dom-Event-bubbles">bubbles</code>
  attribute initialized to true.</p>

  <p class="example">An example of a user interface with a commit action would be a <span
  data-x="attr-input-type-color">Color</span> control that consists of a single button that brings
  up a color wheel: if the <span data-x="concept-fe-value">value</span> only changes when the dialog
  is closed, then that would be the explicit commit action. On the other hand, if manipulating the
  control changes the color interactively, then there might be no commit action.</p>

  <p class="example">Another example of a user interface with a commit action would be a <span
  data-x="attr-input-type-date">Date</span> control that allows both text-based user input and user
  selection from a drop-down calendar: while text input might not have an explicit commit step,
  selecting a date from the drop down calendar and then dismissing the drop down would be a commit
  action.</p>

  <p>For <code>input</code> elements without a defined <span>input activation behavior</span>, but
  to which these events <span data-x="concept-input-apply">apply</span>, any time the user causes
  the element's <span data-x="concept-fe-value">value</span> to change without an explicit commit
  action, the user agent must <span>queue an element task</span> on the <span>user interaction task
  source</span> given the <code>input</code> element to <span data-x="concept-event-fire">fire an
  event</span> named <code data-x="event-input">input</code> at the <code>input</code> element, with
  the <code data-x="dom-Event-bubbles">bubbles</code> and <code
  data-x="dom-Event-composed">composed</code> attributes initialized to true. The corresponding
  <code data-x="event-change">change</code> event, if any, will be fired when the control <a
  href="#unfocus-causes-change-event">loses focus</a>.</p>

  <p class="example">Examples of a user changing the element's <span
  data-x="concept-fe-value">value</span> would include the user typing into a text control, pasting
  a new value into the control, or undoing an edit in that control. Some user interactions do not
  cause changes to the value, e.g., hitting the "delete" key in an empty text control, or replacing
  some text in the control with text from the clipboard that happens to be exactly the same
  text.</p>

  <p class="example">A <span data-x="attr-input-type-range">Range</span> control in the form of a
  slider that the user has <span>focused</span> and is interacting with using a keyboard would be
  another example of the user changing the element's <span data-x="concept-fe-value">value</span>
  without a commit step.</p>

  <p>In the case of <span data-x="concept-task">tasks</span> that just fire an <code
  data-x="event-input">input</code> event, user agents may wait for a suitable break in the user's
  interaction before <span data-x="queue an element task">queuing</span> the tasks; for example, a
  user agent could wait for the user to have not hit a key for 100ms, so as to only fire the event
  when the user pauses, instead of continuously for each keystroke. <!-- similar text is present in
  the <textarea> section -->

  <p>When the user agent is to change an <code>input</code> element's <span
  data-x="concept-fe-value">value</span> on behalf of the user (e.g. as part of a form prefilling
  feature), the user agent must <span>queue an element task</span> on the <span>user interaction
  task source</span> given the <code>input</code> element to first update the <span
  data-x="concept-fe-value">value</span> accordingly, then <span data-x="concept-event-fire">fire an
  event</span> named <code data-x="event-input">input</code> at the <code>input</code> element, with
  the <code data-x="dom-Event-bubbles">bubbles</code> and <code
  data-x="dom-Event-composed">composed</code> attributes initialized to true, then <span
  data-x="concept-event-fire">fire an event</span> named <code data-x="event-change">change</code>
  at the <code>input</code> element, with the <code data-x="dom-Event-bubbles">bubbles</code>
  attribute initialized to true.</p>

  <p class="note">These events are not fired in response to changes made to the values of form
  controls by scripts. (This is to make it easier to update the values of form controls in response
  to the user manipulating the controls, without having to then filter out the script's own changes
  to avoid an infinite loop.)</p>

  <p class="note">These events are also not fired when the browser changes the values of form
  controls as part of <span data-x="restore persisted state">state restoration during
  navigation</span>.</p>

  </div>



  <h4 split-filename="form-elements">The <dfn element><code>button</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Interactive content</span>.</dd>
   <dd><span data-x="category-listed">Listed</span>, <span data-x="category-label">labelable</span>, <span data-x="category-submit">submittable</span>, and <span data-x="category-autocapitalize-and-autocorrect">autocapitalize-and-autocorrect inheriting</span> <span>form-associated element</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>, but there must be no <span>interactive content</span>
   descendant and no descendant with the <code data-x="attr-tabindex">tabindex</code> attribute
   specified.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-button-command">command</code></dd>
   <dd><code data-x="attr-button-commandfor">commandfor</code></dd>
   <dd><code data-x="attr-fe-disabled">disabled</code></dd>
   <dd><code data-x="attr-fae-form">form</code></dd>
   <dd><code data-x="attr-fs-formaction">formaction</code></dd>
   <dd><code data-x="attr-fs-formenctype">formenctype</code></dd>
   <dd><code data-x="attr-fs-formmethod">formmethod</code></dd>
   <dd><code data-x="attr-fs-formnovalidate">formnovalidate</code></dd>
   <dd><code data-x="attr-fs-formtarget">formtarget</code></dd>
   <dd><code data-x="attr-fe-name">name</code></dd>
   <dd><code data-x="attr-popovertarget">popovertarget</code></dd>
   <dd><code data-x="attr-popovertargetaction">popovertargetaction</code></dd>
   <dd><code data-x="attr-button-type">type</code></dd>
   <dd><code data-x="attr-button-value">value</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-button">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-button">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLButtonElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-button-command">command</span>;
  [<span>CEReactions</span>] attribute Element? <span data-x="dom-button-commandForElement">commandForElement</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-fe-disabled">disabled</span>;
  readonly attribute <span>HTMLFormElement</span>? <span data-x="dom-fae-form">form</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-fs-formAction">formAction</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fs-formEnctype">formEnctype</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fs-formMethod">formMethod</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-fs-formNoValidate">formNoValidate</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fs-formTarget">formTarget</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fe-name">name</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-button-type">type</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-button-value">value</span>;

  readonly attribute boolean <span data-x="dom-cva-willValidate">willValidate</span>;
  readonly attribute <span>ValidityState</span> <span data-x="dom-cva-validity">validity</span>;
  readonly attribute DOMString <span data-x="dom-cva-validationMessage">validationMessage</span>;
  boolean <span data-x="dom-cva-checkValidity">checkValidity</span>();
  boolean <span data-x="dom-cva-reportValidity">reportValidity</span>();
  undefined <span data-x="dom-cva-setCustomValidity">setCustomValidity</span>(DOMString error);

  readonly attribute <span>NodeList</span> <span data-x="dom-lfe-labels">labels</span>;
};
<span>HTMLButtonElement</span> includes <span>PopoverInvokerElement</span>;</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLButtonElement</code>.</dd>
  </dl>

  <p>The <code>button</code> element <span>represents</span> a button labeled by its contents.</p>

  <p>The element is a <span data-x="concept-button">button</span>.</p>

  <p>The <dfn element-attr for="button"><code data-x="attr-button-type">type</code></dfn> attribute
  controls the behavior of the button when it is activated. It is an <span>enumerated
  attribute</span> with the following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th> Keyword
     <th> State
     <th> Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="button/type"><code
     data-x="attr-button-type-submit">submit</code></dfn>
     <td><dfn data-x="attr-button-type-submit-state">Submit Button</dfn>
     <td>Submits the form.
    <tr>
     <td><dfn attr-value for="button/type"><code data-x="attr-button-type-reset">reset</code></dfn>
     <td><dfn data-x="attr-button-type-reset-state">Reset Button</dfn>
     <td>Resets the form.
    <tr>
     <td><dfn attr-value for="button/type"><code
     data-x="attr-button-type-button">button</code></dfn>
     <td><dfn data-x="attr-button-type-button-state">Button</dfn>
     <td>Does nothing.
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <dfn
  data-x="attr-button-type-auto-state">Auto</dfn> state.</p>

  <p>A <code>button</code> element is said to be a <span
  data-x="concept-submit-button">submit button</span> if any of the following are true:</p>

  <ul>
   <li><p>the <code data-x="attr-button-type">type</code> attribute is in the <span
   data-x="attr-button-type-auto-state">Auto</span> state and both the <code
   data-x="attr-button-command">command</code> and <code
   data-x="attr-button-commandfor">commandfor</code> content attributes are not present;
   or</p></li>

   <li><p>the <code data-x="attr-button-type">type</code> attribute is in the <span
   data-x="attr-button-type-submit-state">Submit Button</span> state.</p></li>
  </ul>

  <p><strong>Constraint validation</strong>: If the element is not a <span
  data-x="concept-submit-button">submit button</span>, the element is <span>barred from constraint
  validation</span>.</p>

  <p>If specified, the <dfn element-attr for="button"><code
  data-x="attr-button-commandfor">commandfor</code></dfn> attribute value must be the <span
  data-x="concept-ID">ID</span> of an element in the same <span>tree</span> as the <span
  data-x="concept-button">button</span> with the <code
  data-x="attr-button-commandfor">commandfor</code> attribute.</p>

  <p>The <dfn element-attr for="button"><code data-x="attr-button-command">command</code></dfn>
  attribute is an <span>enumerated attribute</span> with the following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="html-global/command"
     data-x="attr-button-command-toggle-popover"><code>toggle-popover</code></dfn>
     <td><dfn data-x="attr-button-command-toggle-popover-state">Toggle Popover</dfn>
     <td>Shows or hides the targeted <code data-x="attr-popover">popover</code> element.
    <tr>
     <td><dfn attr-value for="html-global/command"
     data-x="attr-button-command-show-popover"><code>show-popover</code></dfn>
     <td><dfn data-x="attr-button-command-show-popover-state">Show Popover</dfn>
     <td>Shows the targeted <code data-x="attr-popover">popover</code> element.
    <tr>
     <td><dfn attr-value for="html-global/command"
     data-x="attr-button-command-hide-popover"><code>hide-popover</code></dfn>
     <td><dfn data-x="attr-button-command-hide-popover-state">Hide Popover</dfn>
     <td>Hides the targeted <code data-x="attr-popover">popover</code> element.
    <tr>
     <td><dfn attr-value for="html-global/command"
     data-x="attr-button-command-close"><code>close</code></dfn>
     <td><dfn data-x="attr-button-command-close-state">Close</dfn>
     <td>Closes the targeted <code>dialog</code> element.
    <tr>
     <td><dfn attr-value for="html-global/command"
     data-x="attr-button-command-show-modal"><code>show-modal</code></dfn>
     <td><dfn data-x="attr-button-command-show-modal-state">Show Modal</dfn>
     <td>Opens the targeted <code>dialog</code> element as modal.
    <tr>
     <td>A <span data-x="attr-button-command-custom">custom command keyword</span>
     <td><dfn data-x="attr-button-command-custom-state">Custom</dfn>
     <td>Only dispatches the <code data-x="event-command">command</code> event on the targeted
     element.
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <dfn
  data-x="attr-button-command-unknown-state">Unknown</dfn> state.</p>

  <p>A <dfn data-x="attr-button-command-custom">custom command keyword</dfn> is a string that
  <span>starts with</span> "<code data-x="">--</code>".</p>

  <div w-nodev>

  <p>A <code>button</code> element <var>element</var>'s <span>activation behavior</span> given
  <var>event</var> is:</p>

  <ol>
   <li><p>If <var>element</var> is <span data-x="concept-fe-disabled">disabled</span>, then
   return.</p></li>

   <li><p>If <var>element</var>'s <span>node document</span> is not <span>fully active</span>, then
   return.</p></li>

   <li>
    <p>If <var>element</var> has a <span>form owner</span>:</p>

    <ol>
     <li><p>If <var>element</var> is a <span data-x="concept-submit-button">submit button</span>,
     then <span data-x="concept-form-submit">submit</span> <var>element</var>'s
     <span>form owner</span> from <var>element</var> with <i
     data-x="submit-user-involvement">userInvolvement</i> set to <var>event</var>'s <span
     data-x="event-uni">user navigation involvement</span>, and return.</p></li>

     <li><p>If <var>element</var>'s <code data-x="attr-button-type">type</code> attribute is in the
     <span data-x="attr-button-type-reset-state">Reset Button</span> state, then <span
     data-x="concept-form-reset">reset</span> <var>element</var>'s <span>form owner</span>, and
     return.</p></li>

     <li><p>If <var>element</var>'s <code data-x="attr-button-type">type</code> attribute is in the
     <span data-x="attr-button-type-auto-state">Auto</span> state, then return.</p></li>
     <!-- We do this for compatability and plan to remove this restriction in future, see
          https://github.com/whatwg/html/issues/10832 -->
    </ol>
   </li>

   <li><p>Let <var>target</var> be the result of running <var>element</var>'s <span
   data-x="get the attr-associated element">get the <code data-x="">commandfor</code> associated
   element</span>.</p></li>

   <li>
    <p>If <var>target</var> is not null:</p>

    <ol>
     <li><p>Let <var>command</var> be <var>element</var>'s <code
     data-x="attr-button-command">command</code> attribute.</p></li>

     <li><p>If <var>command</var> is in the <span
     data-x="attr-button-command-unknown-state">Unknown</span> state, then return.</p></li>

     <li><p>Let <var>isPopover</var> be true if <var>target</var>'s <code
     data-x="attr-popover">popover</code> attribute is not in the <span
     data-x="attr-popover-none-state">no popover</span> state; otherwise false.</p></li>

     <li>
      <p>If <var>isPopover</var> is false and <var>command</var> is not in the <span
      data-x="attr-button-command-custom-state">Custom</span> state:</p>

      <ol>
       <li><p><span>Assert</span>: <var>target</var>'s <span
       data-x="concept-element-namespace">namespace</span> is the
       <span>HTML namespace</span>.</p></li>

       <li><p>If this standard does not define <span>is valid invoker command steps</span> for
       <var>target</var>'s <span data-x="concept-element-local-name">local name</span>, then
       return.</p></li>

       <li><p>Otherwise, if the result of running <var>target</var>'s corresponding <span>is valid
       invoker command steps</span> given <var>command</var> is false, then
       return.</p></li>
      </ol>
     </li>

     <li>
      <p>Let <var>continue</var> be the result of <span data-x="concept-event-fire">firing an
      event</span> named <code data-x="event-command">command</code> at <var>target</var>, using
      <code>CommandEvent</code>, with its <code data-x="dom-CommandEvent-command">command</code>
      attribute initialized to <var>command</var>, its <code
      data-x="dom-CommandEvent-source">source</code> attribute initialized to <var>element</var>,
      and its <code data-x="dom-Event-cancelable">cancelable</code> and <code
      data-x="dom-Event-composed">composed</code> attributes initialized to true.</p>

      <p class="XXX"><a href="https://github.com/whatwg/dom/issues/1328">DOM standard issue
      #1328</a> tracks how to better standardize associated event data in a way which makes sense on
      Events. Currently an event attribute initialized to a value cannot also have a getter, and so
      an internal slot (or map of additional fields) is required to properly specify this.</p>
     </li>

     <li><p>If <var>continue</var> is false, then return.</p></li>

     <li><p>If <var>target</var> is not <span>connected</span>, then return.</p></li>

     <li><p>If <var>command</var> is in the <span
     data-x="attr-button-command-custom-state">Custom</span> state, then return.</p></li>

     <li>
      <p>If <var>command</var> is in the <span
      data-x="attr-button-command-hide-popover-state">Hide Popover</span> state:</p>

      <ol>
       <li><p>If the result of running <span>check popover validity</span> given
       <var>target</var>, true, false, and null is true, then run the
       <span>hide popover algorithm</span> given <var>target</var>, true, true, and false.</p></li>
      </ol>
     </li>

     <li>
      <p>Otherwise, if <var>command</var> is in the <span
      data-x="attr-button-command-toggle-popover-state">Toggle Popover</span> state:</p>

      <ol>
       <li><p>If the result of running <span>check popover validity</span> given
       <var>target</var>, false, false, and null is true, then run the
       <span data-x="show popover">show popover algorithm</span> given <var>target</var>, true,
       true, and false.</p></li>

       <li><p>Otheriwse, if the result of running <span>check popover validity</span> given
       <var>target</var>, true, false, and null is true, then run the
       <span>hide popover algorithm</span> given <var>target</var>, true,
       true, and false.</p></li>
      </ol>
     </li>

     <li>
      <p>Otherwise, if <var>command</var> is in the <span
      data-x="attr-button-command-show-popover-state">Show Popover</span> state:</p>

      <ol>
       <li><p>If the result of running <span>check popover validity</span> given <var>target</var>,
       false, false, and null is true, then run the <span
       data-x="show popover">show popover algorithm</span> given <var>target</var>, true, true,
       and false.</p></li>
      </ol>
     </li>

     <li><p>Otherwise, if this standard defines <span>invoker command steps</span> for
     <var>target</var>'s <span data-x="concept-element-local-name">local name</span>, then run the
     corresponding <span>invoker command steps</span> given <var>target</var>, <var>element</var>
     and <var>command</var>.</p></li>
    </ol>
   </li>

   <li><p>Otherwise, run the <span>popover target attribute activation behavior</span> given
   <var>element</var> and <var>event</var>'s <span
   data-x="concept-event-target">target</span>.</p></li>
  </ol>

  <p>An <span data-x="HTML Elements">HTML element</span> can have specific
  <dfn>is valid invoker command steps</dfn> and <dfn>invoker command steps</dfn> defined for the
  element's <span data-x="concept-element-local-name">local name</span>.</p>

  </div>

  <p>The <code data-x="attr-fae-form">form</code> attribute is used to explicitly associate the
  <code>button</code> element with its <span>form owner</span>. The <code
  data-x="attr-fe-name">name</code> attribute represents the element's name. The <code
  data-x="attr-fe-disabled">disabled</code> attribute is used to make the control non-interactive and
  to prevent its value from being submitted. The <code data-x="attr-fs-formaction">formaction</code>,
  <code data-x="attr-fs-formenctype">formenctype</code>, <code
  data-x="attr-fs-formmethod">formmethod</code>, <code
  data-x="attr-fs-formnovalidate">formnovalidate</code>, and <code
  data-x="attr-fs-formtarget">formtarget</code> attributes are <span>attributes for form
  submission</span>.</p>

  <p class="note">The <code data-x="attr-fs-formnovalidate">formnovalidate</code> attribute can be
  used to make submit buttons that do not trigger the constraint validation.</p>

  <p>The <code data-x="attr-fs-formaction">formaction</code>, <code
  data-x="attr-fs-formenctype">formenctype</code>, <code
  data-x="attr-fs-formmethod">formmethod</code>, <code
  data-x="attr-fs-formnovalidate">formnovalidate</code>, and <code
  data-x="attr-fs-formtarget">formtarget</code> must not be specified if the element is not
  a <span data-x="concept-submit-button">submit button</span>.</p>

  <p>The <dfn attribute for="HTMLButtonElement"><code
  data-x="dom-button-commandForElement">commandForElement</code></dfn> IDL attribute must
  <span>reflect</span> the element's <code data-x="attr-button-commandfor">commandfor</code>
  content attribute.</p>

  <p>The <dfn attribute for="HTMLButtonElement"><code
  data-x="dom-button-command">command</code></dfn> getter steps are:</p>

  <ol>
   <li><p>Let <var>command</var> be <span>this</span>'s <code
   data-x="attr-button-command">command</code> attribute.</p></li>

   <li><p>If <var>command</var> is in the <span
   data-x="attr-button-command-custom-state">Custom</span> state, then return <var>command</var>'s
   value.</p></li>

   <li><p>If <var>command</var> is in the <span
   data-x="attr-button-command-unknown-state">Unknown</span> state, then return the empty
   string.</p></li>

   <li><p>Return the keword corresponding to the value of <var>command</var>.</p></li>
  </ol>

  <p>The <code data-x="dom-button-command">command</code> setter steps are to set the <code
  data-x="attr-button-command">command</code> content attribute to the given value.</p>

  <p>The <dfn element-attr for="button"><code data-x="attr-button-value">value</code></dfn>
  attribute gives the element's value for the purposes of form submission. The element's <span
  data-x="concept-fe-value">value</span> is the value of the element's <code
  data-x="attr-button-value">value</code> attribute, if there is one, or the empty string
  otherwise.</p>

  <p class="note">A button (and its value) is only included in the form submission if the button
  itself was used to initiate the form submission.</p>

  <hr>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLButtonElement"><code data-x="dom-button-value">value</code></dfn>
  IDL attribute must <span>reflect</span> the content attribute of the same name.</p>

  <p>The <dfn attribute for="HTMLButtonElement"><code data-x="dom-button-type">type</code></dfn>
  getter steps are:</p>

  <ol>
   <li><p>If <var>this</var> is a <span data-x="concept-submit-button">submit button</span>, then
   return "<code data-x="">submit</code>".</p></li>

   <li><p>Let <var>state</var> be <span>this</span>'s <code data-x="attr-button-type">type</code>
   attribute.</p></li>

   <li><p><span>Assert</span>: <var>state</var> is not in the <span
   data-x="attr-button-type-submit-state">Submit Button</span> state.</p></li>

   <li><p>If <var>state</var> is in the <span data-x="attr-button-type-auto-state">Auto</span>
   state, then return "<code data-x="">button</code>".</p></li>

   <li><p>Return the keyword value corresponding to <var>state</var>.</p></li>
  </ol>

  <p>The <code data-x="dom-button-type">type</code> setter steps are to set the <code
  data-x="attr-button-type">type</code> content attribute to the given value.</p>

  <p>The <code data-x="dom-cva-willValidate">willValidate</code>, <code
  data-x="dom-cva-validity">validity</code>, and <code
  data-x="dom-cva-validationMessage">validationMessage</code> IDL attributes, and the <code
  data-x="dom-cva-checkValidity">checkValidity()</code>, <code
  data-x="dom-cva-reportValidity">reportValidity()</code>, and <code
  data-x="dom-cva-setCustomValidity">setCustomValidity()</code> methods, are part of the
  <span>constraint validation API</span>. The <code data-x="dom-lfe-labels">labels</code> IDL
  attribute provides a list of the element's <code>label</code>s. The <code
  data-x="dom-fe-disabled">disabled</code>, <code data-x="dom-fae-form">form</code>, and
  <code data-x="dom-fe-name">name</code> IDL attributes are part of the element's forms API.</p>

  </div>

  <div class="example">

   <p>The following button is labeled "Show hint" and pops up a dialog box when activated:</p>

   <pre><code class="html">&lt;button type=button
        onclick="alert('This 15-20 minute piece was composed by George Gershwin.')">
 Show hint
&lt;/button></code></pre>

  </div>

  <div class="example">
   <p>The following shows how <span data-x="concept-button">buttons</span> can use <code
   data-x="attr-button-commandfor">commandfor</code> to show and hide an element with
   the <code data-x="attr-popover">popover</code> attribute when activated:</p>

   <pre><code class="html">&lt;button type=button
        commandfor="the-popover"
        command="show-popover">
 Show menu
&lt;/button>
&lt;div popover
     id="the-popover">
 &lt;button commandfor="the-popover"
         command="hide-popover">
  Hide menu
 &lt;/button>
&lt;/div>
    </code></pre>
  </div>

  <div class="example">
   <p>The following shows how buttons can use <code
   data-x="attr-button-commandfor">commandfor</code> with a <span
   data-x="attr-button-command-custom">custom command keyword</span> on an element, demonstrating
   how one could use custom commands for unspecified behavior:</p>

   <pre><code class="html">&lt;button type=button
        commandfor="the-image"
        command="--rotate-landscape">
 Rotate Left
&lt;/button>
&lt;button type=button
        commandfor="the-image"
        command="--rotate-portrait">
 Rotate Right
&lt;/button>
&lt;img id="the-image"
     src="photo.jpg">
&lt;script>
  const image = document.getElementById("the-image");
  image.addEventListener("command", (event) => {
   if ( event.command == "--rotate-landscape" ) {
    event.target.style.rotate = "-90deg"
   } else if ( event.command == "--rotate-portrait" ) {
    event.target.style.rotate = "0deg"
   }
  });
&lt;/script>
    </code></pre>
  </div>

  <h4>The <dfn element><code>select</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Interactive content</span>.</dd>
   <dd><span data-x="category-listed">Listed</span>, <span data-x="category-label">labelable</span>, <span data-x="category-submit">submittable</span>, <span data-x="category-reset">resettable</span>, and <span data-x="category-autocapitalize-and-autocorrect">autocapitalize-and-autocorrect inheriting</span> <span>form-associated element</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>Zero or more <code>option</code>, <code>optgroup</code>, <code>hr</code>, and <span data-x="script-supporting elements">script-supporting</span> elements.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-fe-autocomplete">autocomplete</code></dd>
   <dd><code data-x="attr-fe-disabled">disabled</code></dd>
   <dd><code data-x="attr-fae-form">form</code></dd>
   <dd><code data-x="attr-select-multiple">multiple</code></dd>
   <dd><code data-x="attr-fe-name">name</code></dd>
   <dd><code data-x="attr-select-required">required</code></dd>
   <dd><code data-x="attr-select-size">size</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd>If the element has a <code data-x="attr-select-multiple">multiple</code> attribute or a <code
   data-x="attr-select-size">size</code> attribute with a value > 1: <a
   href="https://w3c.github.io/html-aria/#el-select-multiple-or-size-greater-1">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-select-listbox">for implementers</a>.</dd>
   <dd>Otherwise: <a
   href="https://w3c.github.io/html-aria/#el-select">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-select-combobox">for implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLSelectElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fe-autocomplete">autocomplete</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-fe-disabled">disabled</span>;
  readonly attribute <span>HTMLFormElement</span>? <span data-x="dom-fae-form">form</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-select-multiple">multiple</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fe-name">name</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-select-required">required</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-select-size">size</span>;

  readonly attribute DOMString <span data-x="dom-select-type">type</span>;

  [SameObject] readonly attribute <span>HTMLOptionsCollection</span> <span data-x="dom-select-options">options</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-select-length">length</span>;
  getter <span>HTMLOptionElement</span>? <span data-x="dom-select-item">item</span>(unsigned long index);
  <span>HTMLOptionElement</span>? <span data-x="dom-select-namedItem">namedItem</span>(DOMString name);
  [<span>CEReactions</span>] undefined <span data-x="dom-select-add">add</span>((<span>HTMLOptionElement</span> or <span>HTMLOptGroupElement</span>) element, optional (<span>HTMLElement</span> or long)? before = null);
  [<span>CEReactions</span>] undefined <span data-x="dom-select-remove">remove</span>(); // ChildNode overload
  [<span>CEReactions</span>] undefined <span data-x="dom-select-remove">remove</span>(long index);
  [<span>CEReactions</span>] <a href="#dom-select-setter">setter</a> undefined (unsigned long index, <span>HTMLOptionElement</span>? option);

  [SameObject] readonly attribute <span>HTMLCollection</span> <span data-x="dom-select-selectedOptions">selectedOptions</span>;
  attribute long <span data-x="dom-select-selectedIndex">selectedIndex</span>;
  attribute DOMString <span data-x="dom-select-value">value</span>;

  readonly attribute boolean <span data-x="dom-cva-willValidate">willValidate</span>;
  readonly attribute <span>ValidityState</span> <span data-x="dom-cva-validity">validity</span>;
  readonly attribute DOMString <span data-x="dom-cva-validationMessage">validationMessage</span>;
  boolean <span data-x="dom-cva-checkValidity">checkValidity</span>();
  boolean <span data-x="dom-cva-reportValidity">reportValidity</span>();
  undefined <span data-x="dom-cva-setCustomValidity">setCustomValidity</span>(DOMString error);

  undefined <span data-x="dom-select-showPicker">showPicker</span>();

  readonly attribute <span>NodeList</span> <span data-x="dom-lfe-labels">labels</span>;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLSelectElement</code>.</dd>
  </dl>
  <!-- Proposals for v2:
       * <select value=""> as an alternative to <option selected>
         http://developers.slashdot.org/comments.pl?sid=426306&cid=22142072
         <Philip`> http://www.ipiao.com.cn/ does <select size="1" name="to"><script>City_Default="PEK"</script><SCRIPT language=javascript
                   src="/js/flightcity.js"></SCRIPT></select> which is the kind of thing that could be much simpler with <select value="PEK">
    -->

  <p>The <code>select</code> element represents a control for selecting amongst a set of
  options.</p>

  <p>The <dfn element-attr for="select"><code data-x="attr-select-multiple">multiple</code></dfn>
  attribute is a <span>boolean attribute</span>. If the attribute is present, then the
  <code>select</code> element <span>represents</span> a control for selecting zero or more options
  from the <span data-x="concept-select-option-list">list of options</span>. If the attribute is
  absent, then the <code>select</code> element <span>represents</span> a control for selecting a
  single option from the <span data-x="concept-select-option-list">list of options</span>.</p>

  <p>The <dfn element-attr for="select"><code data-x="attr-select-size">size</code></dfn> attribute
  gives the number of options to show to the user. The <code data-x="attr-select-size">size</code>
  attribute, if specified, must have a value that is a <span>valid non-negative integer</span>
  greater than zero.</p>

  <div w-nodev>

  <p>The <dfn data-x="concept-select-size">display size</dfn> of a <code>select</code> element is the
  result of applying the <span>rules for parsing non-negative integers</span> to the value of
  element's <code data-x="attr-select-size">size</code> attribute, if it has one and parsing it is
  successful. If applying those rules to the attribute's value is not successful, or if the <code
  data-x="attr-select-size">size</code> attribute is absent, then the element's <span
  data-x="concept-select-size">display size</span> is 4 if the element's <code
  data-x="attr-select-multiple">multiple</code> content attribute is present, and 1 otherwise.</p>

  </div>

  <p>The <dfn data-x="concept-select-option-list">list of options</dfn> for a <code>select</code>
  element consists of all the <code>option</code> element children of the <code>select</code>
  element, and all the <code>option</code> element children of all the <code>optgroup</code> element
  children of the <code>select</code> element, in <span>tree order</span>.</p>

  <p>The <dfn element-attr for="select"><code data-x="attr-select-required">required</code></dfn>
  attribute is a <span>boolean attribute</span>. When specified, the user will be required to select
  a value before submitting the form.</p>

  <p>If a <code>select</code> element has a <code data-x="attr-select-required">required</code>
  attribute specified, does not have a <code data-x="attr-select-multiple">multiple</code> attribute
  specified, and has a <span data-x="concept-select-size">display size</span> of 1; and if the <span
  data-x="concept-option-value">value</span> of the first <code>option</code> element in the
  <code>select</code> element's <span data-x="concept-select-option-list">list of options</span> (if
  any) is the empty string, and that <code>option</code> element's parent node is the
  <code>select</code> element (and not an <code>optgroup</code> element), then that
  <code>option</code> is the <code>select</code> element's <dfn>placeholder label option</dfn>.</p>

  <p>If a <code>select</code> element has a <code data-x="attr-select-required">required</code>
  attribute specified, does not have a <code data-x="attr-select-multiple">multiple</code> attribute
  specified, and has a <span data-x="concept-select-size">display size</span> of 1, then the
  <code>select</code> element must have a <span>placeholder label option</span>.</p>

  <p class="note">In practice, the requirement stated in the paragraph above can only apply when a
  <code>select</code> element does not have a <code data-x="attr-select-size">size</code> attribute
  with a value greater than 1.</p>

  <div w-nodev>

  <p><strong>Constraint validation</strong>: If the element has its <code
  data-x="attr-select-required">required</code> attribute specified, and either none of the
  <code>option</code> elements in the <code>select</code> element's <span
  data-x="concept-select-option-list">list of options</span> have their <span
  data-x="concept-option-selectedness">selectedness</span> set to true, or the only
  <code>option</code> element in the <code>select</code> element's <span
  data-x="concept-select-option-list">list of options</span> with its <span
  data-x="concept-option-selectedness">selectedness</span> set to true is the <span>placeholder label
  option</span>, then the element is <span>suffering from being missing</span>.</p>

  </div>

  <div w-nodev>

  <p>If the <code data-x="attr-select-multiple">multiple</code> attribute is absent, and the element
  is not <span data-x="concept-fe-disabled">disabled</span>, then the user agent should allow the
  user to pick an <code>option</code> element in its <span data-x="concept-select-option-list">list
  of options</span> that is itself not <span data-x="concept-option-disabled">disabled</span>. Upon
  this <code>option</code> element being <dfn data-x="concept-select-pick">picked</dfn> (either
  through a click, or through unfocusing the element after changing its value, or through a <span
  data-x="option-command">menu command</span>, or through any other mechanism), and before the
  relevant user interaction event <!-- interaction event spec point --> is queued (e.g. before the
  <code data-x="event-click">click</code> event), the user agent must set the <span
  data-x="concept-option-selectedness">selectedness</span> of the picked <code>option</code> element
  to true, set its <span data-x="concept-option-dirtiness">dirtiness</span> to true, and then
  <span>send <code>select</code> update notifications</span>.</p>

  <p>If the <code data-x="attr-select-multiple">multiple</code> attribute is absent, whenever an
  <code>option</code> element in the <code>select</code> element's <span
  data-x="concept-select-option-list">list of options</span> has its <span
  data-x="concept-option-selectedness">selectedness</span> set to true, and whenever an
  <code>option</code> element with its <span
  data-x="concept-option-selectedness">selectedness</span> set to true is added to the
  <code>select</code> element's <span data-x="concept-select-option-list">list of options</span>,
  the user agent must set the <span data-x="concept-option-selectedness">selectedness</span> of all
  the other <code>option</code> elements in its <span data-x="concept-select-option-list">list of
  options</span> to false.</p>

  <p>If the <code data-x="attr-select-multiple">multiple</code> attribute is absent and the
  element's <span data-x="concept-select-size">display size</span> is greater than 1, then the user
  agent should also allow the user to request that the <code>option</code> whose <span
  data-x="concept-option-selectedness">selectedness</span> is true, if any, be unselected. Upon this
  request being conveyed to the user agent, and before the relevant user interaction event <!--
  interaction event spec point --> is queued (e.g. before the <code
  data-x="event-click">click</code> event), the user agent must set the <span
  data-x="concept-option-selectedness">selectedness</span> of that <code>option</code> element to
  false, set its <span data-x="concept-option-dirtiness">dirtiness</span> to true, and then
  <span>send <code>select</code> update notifications</span>.</p>

  <p>The <dfn>selectedness setting algorithm</dfn>, given a <code>select</code> element
  <var>element</var>, is to run the following steps:</p>

  <ol>
   <li><p>If <var>element</var>'s <code data-x="attr-select-multiple">multiple</code> attribute is
   absent, and <var>element</var>'s <span data-x="concept-select-size">display size</span> is 1,
   and no <code>option</code> elements in the <var>element</var>'s <span
   data-x="concept-select-option-list">list of options</span> have their <span
   data-x="concept-option-selectedness">selectedness</span> set to true, then set the <span
   data-x="concept-option-selectedness">selectedness</span> of the first <code>option</code>
   element in the <span data-x="concept-select-option-list">list of options</span> in
   <span>tree order</span> that is not <span data-x="concept-option-disabled">disabled</span>,
   if any, to true, and return.</p></li>

   <li><p>If <var>element</var>'s <code data-x="attr-select-multiple">multiple</code> attribute is
   absent, and two or more <code>option</code> elements in <var>element</var>'s <span
   data-x="concept-select-option-list">list of options</span> have their <span
   data-x="concept-option-selectedness">selectedness</span> set to true, then set the <span
   data-x="concept-option-selectedness">selectedness</span> of all but the last <code>option</code>
   element with its <span data-x="concept-option-selectedness">selectedness</span> set to true in
   the <span data-x="concept-select-option-list">list of options</span> in <span>tree order</span>
   to false.</p></li>
  </ol>

  <p>The <code>option</code> <span data-x="html element insertion steps">HTML element insertion
  steps</span>, given <var>insertedNode</var>, are:</p>

  <ol>
   <li><p>If <var>insertedNode</var>'s parent is a <code>select</code> element, or
   <var>insertedNode</var>'s parent is an <code>optgroup</code> element whose parent is a
   <code>select</code> element, then run that <code>select</code> element's <span>selectedness
   setting algorithm</span>.</p></li>
  </ol>

  <p>The <code>option</code> <span data-x="html element removing steps">HTML element removing
  steps</span>, given <var>removedNode</var> and <var>oldParent</var>, are:</p>

  <ol>
   <li><p>If <var>oldParent</var> is a <code>select</code> element, or <var>oldParent</var> is an
   <code>optgroup</code> element whose parent is a <code>select</code> element, then run that
   <code>select</code> element's <span>selectedness setting algorithm</span>.</p></li>
  </ol>

  <p>The <code>option</code> <span>HTML element moving steps</span>, given <var>movedNode</var> and
  <var>oldParent</var>, are:</p>

  <ol>
   <li><p>Run the <code>option</code> <span>HTML element removing steps</span> given
   <var>movedNode</var> and <var>oldParent</var>.</p></li>

   <li><p>Run the <code>option</code> <span>HTML element insertion steps</span> given
   <var>movedNode</var>.</p></li>
  </ol>

  <p>The <code>optgroup</code> <span data-x="html element removing steps">HTML element removing
  steps</span>, given <var>removedNode</var> and <var>oldParent</var>, are:</p>

  <ol>
   <li><p>If <var>oldParent</var> is a <code>select</code> element and <var>removedNode</var> has an
   <code>option</code> child, then run <var>oldParent</var>'s <span>selectedness setting
   algorithm</span>.</p></li>
  </ol>

  <p>The <code>optgroup</code> <span>HTML element moving steps</span>, given <var>movedNode</var>
  and <var>oldParent</var>, are:</p>

  <ol>
   <li><p>Run the <code>optgroup</code> <span>HTML element removing steps</span> given
   <var>movedNode</var> and <var>oldParent</var>.</p></li>
  </ol>

  <p>If an <code>option</code> element in the <span data-x="concept-select-option-list">list of
  options</span> <dfn data-x="ask for a reset">asks for a reset</dfn>, then run that
  <code>select</code> element's <span>selectedness setting algorithm</span>.</p>

  <p>If the <code data-x="attr-select-multiple">multiple</code> attribute is present, and the
  element is not <span data-x="concept-fe-disabled">disabled</span>, then the user agent should
  allow the user to <dfn data-x="concept-select-toggle">toggle</dfn> the <span
  data-x="concept-option-selectedness">selectedness</span> of the <code>option</code> elements in
  its <span data-x="concept-select-option-list">list of options</span> that are themselves not <span
  data-x="concept-option-disabled">disabled</span>. Upon such an element being <span
  data-x="concept-select-toggle">toggled</span> (either through a click, or through a <span
  data-x="option-command">menu command</span>, or any other mechanism), and before the relevant user
  interaction event <!-- interaction event spec point --> is queued (e.g. before a related <code
  data-x="event-click">click</code> event), the <span
  data-x="concept-option-selectedness">selectedness</span> of the <code>option</code> element must
  be changed (from true to false or false to true), the <span
  data-x="concept-option-dirtiness">dirtiness</span> of the element must be set to true, and the
  user agent must <span>send <code>select</code> update notifications</span>.</p>

  <p>When the user agent is to <dfn>send <code>select</code> update notifications</dfn>, <span>queue
  an element task</span> on the <span>user interaction task source</span> given the <code>select</code>
  element to run these steps:</p>

  <ol>
   <li>Set the <code>select</code> element's <span>user validity</span> to true.</li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
   data-x="event-input">input</code> at the <code>select</code> element, with the <code
   data-x="dom-Event-bubbles">bubbles</code> and <code data-x="dom-Event-composed">composed</code>
   attributes initialized to true.</p></li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
   data-x="event-change">change</code> at the <code>select</code> element, with the <code
   data-x="dom-Event-bubbles">bubbles</code> attribute initialized to true.</p></li>
  </ol>

  <p>The <span data-x="concept-form-reset-control">reset algorithm</span> for a <code>select</code>
  element <var>selectElement</var> is:</p>

  <ol>
   <li><p>Set <var>selectElement</var>'s <span>user validity</span> to false.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>optionElement</var> of
    <var>selectElement</var>'s <span data-x="concept-select-option-list">list of
    options</span>:</p>

    <ol>
     <li><p>If <var>optionElement</var> has a <code data-x="attr-option-selected">selected</code>
     attribute, then set <var>optionElement</var>'s <span
     data-x="concept-option-selectedness">selectedness</span> to true; otherwise set it to
     false.</p></li>

     <li><p>Set <var>optionElement</var>'s <span data-x="concept-option-dirtiness">dirtiness</span>
     to false.</p></li>
    </ol>
   </li>

   <li><p>Run the <span>selectedness setting algorithm</span> given
   <var>selectElement</var>.</p></li>
  </ol>

  </div>

  <p>
  The <code data-x="attr-fae-form">form</code> attribute is used to explicitly associate the <code>select</code> element with its <span>form owner</span>.
  The <code data-x="attr-fe-name">name</code> attribute represents the element's name.
  The <code data-x="attr-fe-disabled">disabled</code> attribute is used to make the control non-interactive and to prevent its value from being submitted.
  The <code data-x="attr-fe-autocomplete">autocomplete</code> attribute controls how the user agent provides autofill behavior.
  </p>

  <p>A <code>select</code> element that is not <span data-x="concept-fe-disabled">disabled</span> is
  <i data-x="concept-fe-mutable">mutable</i>.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>select</var>.<span subdfn data-x="dom-select-type">type</span></code></dt>

   <dd>
    <p>Returns "<code data-x="">select-multiple</code>" if the element has a <code
    data-x="attr-select-multiple">multiple</code> attribute, and "<code data-x="">select-one</code>"
    otherwise.</p>
   </dd>

   <dt><code data-x=""><var>select</var>.<span subdfn data-x="dom-select-options">options</span></code></dt>

   <dd><p>Returns an <code>HTMLOptionsCollection</code> of the <span
   data-x="concept-select-option-list">list of options</span>.</p></dd>

   <dt><code data-x=""><var>select</var>.<span subdfn data-x="dom-select-length">length</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the number of elements in the <span data-x="concept-select-option-list">list of
    options</span>.</p>

    <p>When set to a smaller number, truncates the number of <code>option</code> elements in the
    <code>select</code>.</p>

    <p>When set to a greater number, adds new blank <code>option</code> elements to the
    <code>select</code>.</p>
   </dd>

   <dt><code data-x=""><var>element</var> = <var>select</var>.<span subdfn data-x="dom-select-item">item</span>(<var>index</var>)</code></dt>
   <dt><code data-x=""><var>select</var>[<var>index</var>]</code></dt>
   <dd>
    <p>Returns the item with index <var>index</var> from the <span
    data-x="concept-select-option-list">list of options</span>. The items are sorted in <span>tree
    order</span>.</p>
   </dd>

   <dt><code data-x=""><var>element</var> = <var>select</var>.<span subdfn data-x="dom-select-namedItem">namedItem</span>(<var>name</var>)</code></dt>
   <dd>
    <p>Returns the first item with <span data-x="concept-id">ID</span> or <code
    data-x="attr-option-name">name</code> <var>name</var> from the <span
    data-x="concept-select-option-list">list of options</span>.</p>

    <p>Returns null if no element with that <span data-x="concept-id">ID</span> could be found.</p>
   </dd>

   <dt><code data-x=""><var>select</var>.<span subdfn data-x="dom-select-add">add</span>(<var>element</var> [, <var>before</var> ])</code></dt>
   <dd>
    <p>Inserts <var>element</var> before the node given by <var>before</var>.</p>

    <p>The <var>before</var> argument can be a number, in which case <var>element</var> is inserted
    before the item with that number, or an element from the <span
    data-x="concept-select-option-list">list of options</span>, in which case <var>element</var> is
    inserted before that element.</p>

    <p>If <var>before</var> is omitted, null, or a number out of range, then <var>element</var> will
    be added at the end of the list.</p>

    <p>This method will throw a <span>"<code>HierarchyRequestError</code>"</span>
    <code>DOMException</code> if <var>element</var> is an ancestor of the element into which it is
    to be inserted.</p>
   </dd>

   <dt><code data-x=""><var>select</var>.<span subdfn data-x="dom-select-selectedOptions">selectedOptions</span></code></dt>

   <dd>
    <p>Returns an <code>HTMLCollection</code> of the <span data-x="concept-select-option-list">list
    of options</span> that are selected.</p>
   </dd>

   <dt><code data-x=""><var>select</var>.<span subdfn data-x="dom-select-selectedIndex">selectedIndex</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the index of the first selected item, if any, or &#x2212;1 if there is no selected
    item.</p>

    <p>Can be set, to change the selection.</p>
   </dd>

   <dt><code data-x=""><var>select</var>.<span subdfn data-x="dom-select-value">value</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the <span data-x="concept-option-value">value</span> of the first selected item, if
    any, or the empty string if there is no selected item.</p>

    <p>Can be set, to change the selection.</p>
   </dd>

   <dt><code data-x=""><var>select</var>.<span subdfn data-x="dom-select-showPicker">showPicker</span>()</code></dt>

   <dd>
    <p>Shows any applicable picker UI for <var>select</var>, so that the user can select a value.

    <p>Throws an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> if
    <var>select</var> is not <span data-x="concept-fe-mutable">mutable</span>.</p>

    <p>Throws a <span>"<code>NotAllowedError</code>"</span> <code>DOMException</code> if called
    without <span data-x="transient activation">transient user activation</span>.</p>

    <p>Throws a <span>"<code>SecurityError</code>"</span> <code>DOMException</code> if
    <var>select</var> is inside a cross-origin <code>iframe</code>.</p>

    <p>Throws a <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code> if
    <var>select</var> is not <span>being rendered</span>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLSelectElement"><code data-x="dom-select-type">type</code></dfn> IDL
  attribute, on getting, must return the string "<code data-x="">select-one</code>" if the <code
  data-x="attr-select-multiple">multiple</code> attribute is absent, and the string "<code
  data-x="">select-multiple</code>" if the <code data-x="attr-select-multiple">multiple</code>
  attribute is present.</p>

  <p>The <dfn attribute for="HTMLSelectElement"><code
  data-x="dom-select-options">options</code></dfn> IDL attribute must return an
  <code>HTMLOptionsCollection</code> rooted at the <code>select</code> node, whose filter matches
  the elements in the <span data-x="concept-select-option-list">list of options</span>.</p>

  <p>The <code data-x="dom-select-options">options</code> collection is also mirrored on the
  <code>HTMLSelectElement</code> object. The <span>supported property indices</span> at any instant
  are the indices supported by the object returned by the <code
  data-x="dom-select-options">options</code> attribute at that instant.</p>

  <p>The <dfn attribute for="HTMLSelectElement"><code data-x="dom-select-length">length</code></dfn>
  IDL attribute must return the number of nodes <span data-x="represented by the
  collection">represented</span> by the <code data-x="dom-select-options">options</code> collection.
  On setting, it must act like the attribute of the same name on the <code
  data-x="dom-select-options">options</code> collection.</p>

  <p>The <dfn method for="HTMLSelectElement"><code
  data-x="dom-select-item">item(<var>index</var>)</code></dfn> method must return the value returned
  by <span data-x="dom-HTMLCollection-item">the method of the same name</span> on the <code
  data-x="dom-select-options">options</code> collection, when invoked with the same argument.</p>

  <p>The <dfn method for="HTMLSelectElement"><code
  data-x="dom-select-namedItem">namedItem(<var>name</var>)</code></dfn> method must return the value
  returned by <span data-x="dom-HTMLCollection-namedItem">the method of the same name</span> on the
  <code data-x="dom-select-options">options</code> collection, when invoked with the same
  argument.</p>

  <p id="dom-select-setter">When the user agent is to <span>set the value of a new indexed
  property</span> or <span>set the value of an existing indexed property</span> for a
  <code>select</code> element, it must instead run <a href="#dom-htmloptionscollection-setter">the
  corresponding algorithm</a> on the <code>select</code> element's <code
  data-x="dom-select-options">options</code> collection.</p>

  <p>Similarly, the <dfn method for="HTMLSelectElement"><code
  data-x="dom-select-add">add(<var>element</var>, <var>before</var>)</code></dfn> method must act
  like its namesake method on that same <code data-x="dom-select-options">options</code>
  collection.</p>

  <p>The <dfn method for="HTMLSelectElement"><code data-x="dom-select-remove">remove()</code></dfn>
  method must act like its namesake method on that same <code
  data-x="dom-select-options">options</code> collection when it has arguments, and like its namesake
  method on the <code>ChildNode</code> interface implemented by the <code>HTMLSelectElement</code>
  ancestor interface <code>Element</code> when it has no arguments.</p>

  <p>The <dfn attribute for="HTMLSelectElement"><code
  data-x="dom-select-selectedOptions">selectedOptions</code></dfn> IDL attribute must return an
  <code>HTMLCollection</code> rooted at the <code>select</code> node, whose filter matches the
  elements in the <span data-x="concept-select-option-list">list of options</span> that have their
  <span data-x="concept-option-selectedness">selectedness</span> set to true.</p>

  <p>The <dfn attribute for="HTMLSelectElement"><code
  data-x="dom-select-selectedIndex">selectedIndex</code></dfn> IDL attribute, on getting, must
  return the <span data-x="concept-option-index">index</span> of the first <code>option</code>
  element in the <span data-x="concept-select-option-list">list of options</span> in <span>tree
  order</span> that has its <span data-x="concept-option-selectedness">selectedness</span> set to
  true, if any. If there isn't one, then it must return &#x2212;1.</p>

  <p>On setting, the <code data-x="dom-select-selectedIndex">selectedIndex</code> attribute must set
  the <span data-x="concept-option-selectedness">selectedness</span> of all the <code>option</code>
  elements in the <span data-x="concept-select-option-list">list of options</span> to false, and
  then the <code>option</code> element in the <span data-x="concept-select-option-list">list of
  options</span> whose <span data-x="concept-option-index">index</span> is the given new value, if
  any, must have its <span data-x="concept-option-selectedness">selectedness</span> set to true and
  its <span data-x="concept-option-dirtiness">dirtiness</span> set to true.</p>

  <p class="note">This can result in no element having a <span
  data-x="concept-option-selectedness">selectedness</span> set to true even in the case of the
  <code>select</code> element having no <code data-x="attr-select-multiple">multiple</code>
  attribute and a <span data-x="concept-select-size">display size</span> of 1.</p>

  <p>The <dfn attribute for="HTMLSelectElement"><code data-x="dom-select-value">value</code></dfn>
  IDL attribute, on getting, must return the <span data-x="concept-option-value">value</span> of the
  first <code>option</code> element in the <span data-x="concept-select-option-list">list of
  options</span> in <span>tree order</span> that has its <span
  data-x="concept-option-selectedness">selectedness</span> set to true, if any. If there isn't one,
  then it must return the empty string.</p>

  <p>On setting, the <code data-x="dom-select-value">value</code> attribute must set the <span
  data-x="concept-option-selectedness">selectedness</span> of all the <code>option</code> elements
  in the <span data-x="concept-select-option-list">list of options</span> to false, and then the
  first <code>option</code> element in the <span data-x="concept-select-option-list">list of
  options</span>, in <span>tree order</span>, whose <span data-x="concept-option-value">value</span>
  is equal to the given new value, if any, must have its <span
  data-x="concept-option-selectedness">selectedness</span> set to true and its <span
  data-x="concept-option-dirtiness">dirtiness</span> set to true.</p>

  <p class="note">This can result in no element having a <span
  data-x="concept-option-selectedness">selectedness</span> set to true even in the case of the
  <code>select</code> element having no <code data-x="attr-select-multiple">multiple</code>
  attribute and a <span data-x="concept-select-size">display size</span> of 1.</p>

  <p>The <dfn attribute for="HTMLSelectElement"><code
  data-x="dom-select-multiple">multiple</code></dfn>, <dfn attribute for="HTMLSelectElement"><code
  data-x="dom-select-required">required</code></dfn>, and <dfn attribute
  for="HTMLSelectElement"><code data-x="dom-select-size">size</code></dfn> IDL attributes must
  <span>reflect</span> the respective content attributes of the same name. The <code
  data-x="dom-select-size">size</code> IDL attribute has a <span>default value</span> of 0.</p>
  <!-- see https://www.w3.org/Bugs/Public/show_bug.cgi?id=12288 for compat reasons why .size allows
       0 (it's not limited to values greater than zero) even though that value really makes no sense
       -->

  <p class="note">For historical reasons, the default value of the <code
  data-x="dom-select-size">size</code> IDL attribute does not return the actual size used, which, in
  the absence of the <code data-x="attr-select-size">size</code> content attribute, is either 1 or 4
  depending on the presence of the <code data-x="attr-select-multiple">multiple</code>
  attribute.</p>

  <p>The <code data-x="dom-cva-willValidate">willValidate</code>, <code
  data-x="dom-cva-validity">validity</code>, and <code
  data-x="dom-cva-validationMessage">validationMessage</code> IDL attributes, and the <code
  data-x="dom-cva-checkValidity">checkValidity()</code>, <code
  data-x="dom-cva-reportValidity">reportValidity()</code>, and <code
  data-x="dom-cva-setCustomValidity">setCustomValidity()</code> methods, are part of the
  <span>constraint validation API</span>. The <code data-x="dom-lfe-labels">labels</code> IDL
  attribute provides a list of the element's <code>label</code>s. The <code
  data-x="dom-fe-disabled">disabled</code>, <code data-x="dom-fae-form">form</code>, and
  <code data-x="dom-fe-name">name</code> IDL attributes are part of the element's forms API.</p>

  </div>

  <div class="example">

   <p>The following example shows how a <code>select</code> element can be used to offer the user
   with a set of options from which the user can select a single option. The default option is
   preselected.</p>

   <pre><code class="html">&lt;p>
 &lt;label for="unittype">Select unit type:&lt;/label>
 &lt;select id="unittype" name="unittype">
  &lt;option value="1"> Miner &lt;/option>
  &lt;option value="2"> Puffer &lt;/option>
  &lt;option value="3" selected> Snipey &lt;/option>
  &lt;option value="4"> Max &lt;/option>
  &lt;option value="5"> Firebot &lt;/option>
 &lt;/select>
&lt;/p></code></pre>

   <p>When there is no default option, a placeholder can be used instead:</p>

   <pre><code class="html">&lt;select name="unittype" <strong>required</strong>>
 <strong>&lt;option value=""> Select unit type &lt;/option></strong>
 &lt;option value="1"> Miner &lt;/option>
 &lt;option value="2"> Puffer &lt;/option>
 &lt;option value="3"> Snipey &lt;/option>
 &lt;option value="4"> Max &lt;/option>
 &lt;option value="5"> Firebot &lt;/option>
&lt;/select></code></pre>

  </div>

  <div class="example">

   <p>Here, the user is offered a set of options from which they can select any number. By default,
   all five options are selected.</p>

   <pre><code class="html">&lt;p>
 &lt;label for="allowedunits">Select unit types to enable on this map:&lt;/label>
 &lt;select id="allowedunits" name="allowedunits" multiple>
  &lt;option value="1" selected> Miner &lt;/option>
  &lt;option value="2" selected> Puffer &lt;/option>
  &lt;option value="3" selected> Snipey &lt;/option>
  &lt;option value="4" selected> Max &lt;/option>
  &lt;option value="5" selected> Firebot &lt;/option>
 &lt;/select>
&lt;/p></code></pre>

  </div>

  <div class="example">

   <p>Sometimes, a user has to select one or more items. This example shows such an interface.</p>

   <pre><code class="html">&lt;label>
 Select the songs from that you would like on your Act II Mix Tape:
 &lt;select multiple required name="act2">
  &lt;option value="s1">It Sucks to Be Me (Reprise)
  &lt;option value="s2">There is Life Outside Your Apartment
  &lt;option value="s3">The More You Ruv Someone
  &lt;option value="s4">Schadenfreude
  &lt;option value="s5">I Wish I Could Go Back to College
  &lt;option value="s6">The Money Song
  &lt;option value="s7">School for Monsters
  &lt;option value="s8">The Money Song (Reprise)
  &lt;option value="s9">There's a Fine, Fine Line (Reprise)
  &lt;option value="s10">What Do You Do With a B.A. in English? (Reprise)
  &lt;option value="s11">For Now
 &lt;/select>
&lt;/label></code></pre>

  </div>

  <div class="example">
   <p>Occasionally it can be useful to have a separator:

   <pre><code class="html">&lt;label>
 Select the song to play next:
 &lt;select required name="next">
  &lt;option value="sr">Random
  &lt;hr>
  &lt;option value="s1">It Sucks to Be Me (Reprise)
  &lt;option value="s2">There is Life Outside Your Apartment
  &hellip;</code></pre>
  </div>



  <h4>The <dfn element><code>datalist</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>Either: <span>phrasing content</span>.</dd> <!-- we don't require a <select> element descendant, since there might be zero suggestions -->
   <dd>Or: Zero or more <code>option</code> and <span data-x="script-supporting elements">script-supporting</span> elements.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-datalist">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-datalist">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLDataListElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [SameObject] readonly attribute <span>HTMLCollection</span> <span data-x="dom-datalist-options">options</span>;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLDataListElement</code>.</dd>
  </dl>

  <p>The <code>datalist</code> element represents a set of <code>option</code> elements that
  represent predefined options for other controls. In the rendering, the <code>datalist</code>
  element <span>represents</span> nothing<span w-nodev> and it, along with its children, should
  be hidden</span>.</p>

  <p>The <code>datalist</code> element can be used in two ways. In the simplest case, the
  <code>datalist</code> element has just <code>option</code> element children.</p>

  <div class="example">

   <pre><code class="html">&lt;label>
 Animal:
 &lt;input name=animal list=animals>
 &lt;datalist id=animals>
  &lt;option value="Cat">
  &lt;option value="Dog">
 &lt;/datalist>
&lt;/label></code></pre>

  </div>

  <p>In the more elaborate case, the <code>datalist</code> element can be given contents that are to
  be displayed for down-level clients that don't support <code>datalist</code>. In this case, the
  <code>option</code> elements are provided inside a <code>select</code> element inside the
  <code>datalist</code> element.</p>

  <div class="example">

   <pre><code class="html">&lt;label>
 Animal:
 &lt;input name=animal list=animals>
&lt;/label>
&lt;datalist id=animals>
 &lt;label>
  or select from the list:
  &lt;select name=animal>
   &lt;option value="">
   &lt;option>Cat
   &lt;option>Dog
  &lt;/select>
 &lt;/label>
&lt;/datalist></code></pre>

  </div>

  <p>The <code>datalist</code> element is hooked up to an <code>input</code> element using the <code
  data-x="attr-input-list">list</code> attribute on the <code>input</code> element.</p>

  <p>Each <code>option</code> element that is a descendant of the <code>datalist</code> element,
  that is not <span data-x="concept-option-disabled">disabled</span>, and whose <span
  data-x="concept-option-value">value</span> is a string that isn't the empty string, represents a
  suggestion. Each suggestion has a <span data-x="concept-option-value">value</span> and a <span
  data-x="concept-option-label">label</span>.

  <dl class="domintro">
   <dt><code data-x=""><var>datalist</var>.<span subdfn data-x="dom-datalist-options">options</span></code></dt>

   <dd>
    <p>Returns an <code>HTMLCollection</code> of the <code>option</code> elements of the
    <code>datalist</code> element.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLDataListElement"><code
  data-x="dom-datalist-options">options</code></dfn> IDL attribute must return an
  <code>HTMLCollection</code> rooted at the <code>datalist</code> node, whose filter matches
  <code>option</code> elements.</p>

  <p><strong>Constraint validation</strong>: If an element has a <code>datalist</code> element
  ancestor, it is <span>barred from constraint validation</span>.</p>

  </div>


  <h4>The <dfn element><code>optgroup</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As a child of a <code>select</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>Zero or more <code>option</code> and <span data-x="script-supporting elements">script-supporting</span> elements.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-optgroup-disabled">disabled</code></dd>
   <dd><code data-x="attr-optgroup-label">label</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-optgroup">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-optgroup">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLOptGroupElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute boolean <span data-x="dom-optgroup-disabled">disabled</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-optgroup-label">label</span>;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLOptGroupElement</code>.</dd>
  </dl>

  <p>The <code>optgroup</code> element <span>represents</span> a group of <code>option</code>
  elements with a common label.</p>

  <p>The element's group of <code>option</code> elements consists of the <code>option</code>
  elements that are children of the <code>optgroup</code> element.</p>

  <div w-nodev>

  <p>When showing <code>option</code> elements in <code>select</code> elements, user agents should
  show the <code>option</code> elements of such groups as being related to each other and separate
  from other <code>option</code> elements.</p>

  </div>

  <p>The <dfn element-attr for="optgroup"><code
  data-x="attr-optgroup-disabled">disabled</code></dfn> attribute is a <span>boolean
  attribute</span> and can be used to <span data-x="concept-option-disabled">disable</span> a group
  of <code>option</code> elements together.</p>

  <p>The <dfn element-attr for="optgroup"><code data-x="attr-optgroup-label">label</code></dfn>
  attribute must be specified. Its value gives the name of the group, for the purposes of the user
  interface. <span w-nodev>User agents should use this attribute's value when labeling the group of
  <code>option</code> elements in a <code>select</code> element.</span></p>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLOptGroupElement"><code
  data-x="dom-optgroup-disabled">disabled</code></dfn> and <dfn attribute
  for="HTMLOptGroupElement"><code data-x="dom-optgroup-label">label</code></dfn> attributes must
  <span>reflect</span> the respective content attributes of the same name.</p>

  </div>

  <p class="note">There is no way to select an <code>optgroup</code> element. Only
  <code>option</code> elements can be selected. An <code>optgroup</code> element merely provides a
  label for a group of <code>option</code> elements.</p>

  <div class="example">

   <p>The following snippet shows how a set of lessons from three courses could be offered in a
   <code>select</code> drop-down widget:</p>

   <pre><code class="html">&lt;form action="courseselector.dll" method="get">
 &lt;p>Which course would you like to watch today?
 &lt;p>&lt;label>Course:
  &lt;select name="c">
   &lt;optgroup label="8.01 Physics I: Classical Mechanics">
    &lt;option value="8.01.1">Lecture 01: Powers of Ten
    &lt;option value="8.01.2">Lecture 02: 1D Kinematics
    &lt;option value="8.01.3">Lecture 03: Vectors
   &lt;optgroup label="8.02 Electricity and Magnetism">
    &lt;option value="8.02.1">Lecture 01: What holds our world together?
    &lt;option value="8.02.2">Lecture 02: Electric Field
    &lt;option value="8.02.3">Lecture 03: Electric Flux
   &lt;optgroup label="8.03 Physics III: Vibrations and Waves">
    &lt;option value="8.03.1">Lecture 01: Periodic Phenomenon
    &lt;option value="8.03.2">Lecture 02: Beats
    &lt;option value="8.03.3">Lecture 03: Forced Oscillations with Damping
  &lt;/select>
 &lt;/label>
 &lt;p>&lt;input type=submit value="&#x25B6; Play">
&lt;/form></code></pre>

  </div>



  <h4>The <dfn element><code>option</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As a child of a <code>select</code> element.</dd>
   <dd>As a child of a <code>datalist</code> element.</dd>
   <dd>As a child of an <code>optgroup</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>If the element has a <code data-x="attr-option-label">label</code> attribute and a <code data-x="attr-option-value">value</code> attribute: <span data-x="concept-content-nothing">Nothing</span>.</dd>
   <dd>If the element has a <code data-x="attr-option-label">label</code> attribute but no <code data-x="attr-option-value">value</code> attribute: <span data-x="text content">Text</span>.</dd>
   <dd>If the element has no <code data-x="attr-option-label">label</code> attribute and is not a
   child of a <code>datalist</code> element: <span data-x="text content">Text</span> that is not
   <span>inter-element whitespace</span>.</dd>
   <dd>If the element has no <code data-x="attr-option-label">label</code> attribute and is a child
   of a <code>datalist</code> element: <span data-x="text content">Text</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-option-disabled">disabled</code></dd>
   <dd><code data-x="attr-option-label">label</code></dd>
   <dd><code data-x="attr-option-selected">selected</code></dd>
   <dd><code data-x="attr-option-value">value</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-option">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-option">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window,
 <span>LegacyFactoryFunction</span>=<span data-x="dom-option">Option</span>(optional DOMString text = "", optional DOMString value, optional boolean defaultSelected = false, optional boolean selected = false)]
interface <dfn interface>HTMLOptionElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute boolean <span data-x="dom-option-disabled">disabled</span>;
  readonly attribute <span>HTMLFormElement</span>? <span data-x="dom-option-form">form</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-option-label">label</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-option-defaultSelected">defaultSelected</span>;
  attribute boolean <span data-x="dom-option-selected">selected</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-option-value">value</span>;

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-option-text">text</span>;
  readonly attribute long <span data-x="dom-option-index">index</span>;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLOptionElement</code>.</dd>
  </dl>

  <p>The <code>option</code> element <span>represents</span> an option in a <code>select</code>
  element or as part of a list of suggestions in a <code>datalist</code> element.</p>

  <p>In certain circumstances described in the definition of the <code>select</code> element, an
  <code>option</code> element can be a <code>select</code> element's <span>placeholder label
  option</span>. A <span>placeholder label option</span> does not represent an actual option, but
  instead represents a label for the <code>select</code> control.</p>

  <p>The <dfn element-attr for="option"><code data-x="attr-option-disabled">disabled</code></dfn>
  attribute is a <span>boolean attribute</span>. An <code>option</code> element is <dfn
  data-x="concept-option-disabled">disabled</dfn> if its <code
  data-x="attr-option-disabled">disabled</code> attribute is present or if it is a child of an
  <code>optgroup</code> element whose <code data-x="attr-optgroup-disabled">disabled</code>
  attribute is present.</p>

  <div w-nodev>

  <p>An <code>option</code> element that is <span data-x="attr-option-disabled">disabled</span> must
  prevent any <code data-x="event-click">click</code> events that are <span data-x="queue a
  task">queued</span> on the <span>user interaction task source</span> from being dispatched on the
  element.</p>

  </div>

  <p>The <dfn element-attr for="option"><code data-x="attr-option-label">label</code></dfn>
  attribute provides a label for element. The <dfn data-x="concept-option-label">label</dfn> of an
  <code>option</code> element is the value of the <code data-x="attr-option-label">label</code>
  content attribute, if there is one and its value is not the empty string, or, otherwise, the value
  of the element's <code data-x="dom-option-text">text</code> IDL attribute.</p>

  <p>The <code data-x="attr-option-label">label</code> content attribute, if specified, must not be
  empty.</p>

  <p>The <dfn element-attr for="option"><code data-x="attr-option-value">value</code></dfn>
  attribute provides a value for element. The <dfn data-x="concept-option-value">value</dfn> of an
  <code>option</code> element is the value of the <code data-x="attr-option-value">value</code>
  content attribute, if there is one, or, if there is not, the value of the element's <code
  data-x="dom-option-text">text</code> IDL attribute.</p>

  <p>The <dfn element-attr for="option"><code data-x="attr-option-selected">selected</code></dfn>
  attribute is a <span>boolean attribute</span>. It represents the default <span
  data-x="concept-option-selectedness">selectedness</span> of the element.</p>

  <div w-nodev>

  <p>The <dfn data-x="concept-option-dirtiness">dirtiness</dfn> of an <code>option</code> element is
  a boolean state, initially false. It controls whether adding or removing the <code
  data-x="attr-option-selected">selected</code> content attribute has any effect.</p>

  <p>The <dfn data-x="concept-option-selectedness">selectedness</dfn> of an <code>option</code>
  element is a boolean state, initially false. Except where otherwise specified, when the element is
  created, its <span data-x="concept-option-selectedness">selectedness</span> must be set to true if
  the element has a <code data-x="attr-option-selected">selected</code> attribute. Whenever an
  <code>option</code> element's <code data-x="attr-option-selected">selected</code> attribute is
  added, if its <span data-x="concept-option-dirtiness">dirtiness</span> is false, its <span
  data-x="concept-option-selectedness">selectedness</span> must be set to true. Whenever an
  <code>option</code> element's <code data-x="attr-option-selected">selected</code> attribute is
  <em>removed</em>, if its <span data-x="concept-option-dirtiness">dirtiness</span> is false, its
  <span data-x="concept-option-selectedness">selectedness</span> must be set to false.</p>

  <p class="note">The <code data-x="dom-option">Option()</code> constructor, when called with three
  or fewer arguments, overrides the initial state of the <span
  data-x="concept-option-selectedness">selectedness</span> state to always be false even if the third
  argument is true (implying that a <code data-x="attr-option-selected">selected</code> attribute is
  to be set). The fourth argument can be used to explicitly set the initial <span
  data-x="concept-option-selectedness">selectedness</span> state when using the constructor.</p>

  <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=604 -->

  </div>

  <p>A <code>select</code> element whose <code data-x="attr-select-multiple">multiple</code>
  attribute is not specified must not have more than one descendant <code>option</code> element with
  its <code data-x="attr-option-selected">selected</code> attribute set.</p>

  <div w-nodev>

  <p>An <code>option</code> element's <dfn data-x="concept-option-index">index</dfn> is the number of
  <code>option</code> elements that are in the same <span data-x="concept-select-option-list">list of
  options</span> but that come before it in <span>tree order</span>. If the <code>option</code>
  element is not in a <span data-x="concept-select-option-list">list of options</span>, then the
  <code>option</code> element's <span data-x="concept-option-index">index</span> is zero.</p>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>option</var>.<span subdfn data-x="dom-option-selected">selected</span></code></dt>

   <dd>
    <p>Returns true if the element is selected, and false otherwise.</p>

    <p>Can be set, to override the current state of the element.</p>
   </dd>

   <dt><code data-x=""><var>option</var>.<span subdfn data-x="dom-option-index">index</span></code></dt>

   <dd><p>Returns the index of the element in its <code>select</code> element's <code
   data-x="dom-select-options">options</code> list.</p></dd>

   <dt><code data-x=""><var>option</var>.<span subdfn data-x="dom-option-form">form</span></code></dt>

   <dd><p>Returns the element's <code>form</code> element, if any, or null otherwise.</p></dd>

   <dt><code data-x=""><var>option</var>.<span subdfn data-x="dom-option-text">text</span></code></dt>

   <dd>
    <p>Same as <code>textContent</code>, except that spaces are collapsed and <code>script</code>
    elements are skipped.</p>
   </dd>

   <dt><code data-x=""><var>option</var> = new <span subdfn data-x="dom-option">Option</span>([ <var>text</var> [, <var>value</var> [, <var>defaultSelected</var> [, <var>selected</var> ] ] ] ])</code></dt>

   <dd>
    <p>Returns a new <code>option</code> element.</p>

    <p>The <var>text</var> argument sets the contents of the element.</p>

    <p>The <var>value</var> argument sets the <code data-x="attr-option-value">value</code>
    attribute.</p>

    <p>The <var>defaultSelected</var> argument sets the <code
    data-x="attr-option-selected">selected</code> attribute.</p>

    <p>The <var>selected</var> argument sets whether or not the element is selected. If it is
    omitted, even if the <var>defaultSelected</var> argument is true, the element is not
    selected.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLOptionElement"><code
  data-x="dom-option-disabled">disabled</code></dfn> IDL attribute must <span>reflect</span> the
  content attribute of the same name. The <dfn attribute for="HTMLOptionElement"><code
  data-x="dom-option-defaultSelected">defaultSelected</code></dfn> IDL attribute must
  <span>reflect</span> the <code data-x="attr-option-selected">selected</code> content
  attribute.</p>

  <p>The <dfn attribute for="HTMLOptionElement"><code data-x="dom-option-label">label</code></dfn>
  IDL attribute, on getting, if there is a <code data-x="attr-option-label">label</code> content
  attribute, must return that attribute's value; otherwise, it must return the element's <span
  data-x="concept-option-label">label</span>.  On setting, the element's <code
  data-x="attr-option-label">label</code> content attribute must be set to the new value.</p>

  <p>The <dfn attribute for="HTMLOptionElement"><code data-x="dom-option-value">value</code></dfn>
  IDL attribute, on getting, must return the element's <span
  data-x="concept-option-value">value</span>. On setting, the element's <code
  data-x="attr-option-value">value</code> content attribute must be set to the new value.</p>

  <p>The <dfn attribute for="HTMLOptionElement"><code
  data-x="dom-option-selected">selected</code></dfn> IDL attribute, on getting, must return true if
  the element's <span data-x="concept-option-selectedness">selectedness</span> is true, and false
  otherwise. On setting, it must set the element's <span
  data-x="concept-option-selectedness">selectedness</span> to the new value, set its <span
  data-x="concept-option-dirtiness">dirtiness</span> to true, and then cause the element to
  <span>ask for a reset</span>.</p>

  <p>The <dfn attribute for="HTMLOptionElement"><code data-x="dom-option-index">index</code></dfn>
  IDL attribute must return the element's <span data-x="concept-option-index">index</span>.</p>

  <p>The <dfn attribute for="HTMLOptionElement"><code data-x="dom-option-text">text</code></dfn> IDL
  attribute, on getting, must return the result of <span data-x="strip and collapse ASCII
  whitespace">stripping and collapsing ASCII whitespace</span> from the concatenation of <span
  data-x="concept-cd-data">data</span> of all the <code>Text</code> node descendants of the
  <code>option</code> element, in <span>tree order</span>, excluding any that are descendants of
  descendants of the <code>option</code> element that are themselves <code>script</code> or
  <span>SVG <code>script</code></span> elements.</p>

  <p>The <code data-x="dom-option-text">text</code> attribute's setter must <span>string replace
  all</span> with the given value within this element.</p>

  <p>The <dfn attribute for="HTMLOptionElement"><code data-x="dom-option-form">form</code></dfn> IDL
  attribute's behavior depends on whether the <code>option</code> element is in a
  <code>select</code> element or not. If the <code>option</code> has a <code>select</code> element
  as its parent, or has an <code>optgroup</code> element as its parent and that
  <code>optgroup</code> element has a <code>select</code> element as its parent, then the <code
  data-x="dom-option-form">form</code> IDL attribute must return the same value as the <code
  data-x="dom-fae-form">form</code> IDL attribute on that <code>select</code> element. Otherwise, it
  must return null.</p>

  <p>A legacy factory function is provided for creating <code>HTMLOptionElement</code> objects (in
  addition to the factory methods from DOM such as <code
  data-x="dom-Document-createElement">createElement()</code>): <dfn constructor
  for="HTMLOptionElement"><code data-x="dom-option">Option(<var>text</var>, <var>value</var>,
  <var>defaultSelected</var>, <var>selected</var>)</code></dfn>.  When invoked, the legacy factory
  function must perform the following steps:</p>

  <ol>
   <li><p>Let <var>document</var> be the <span>current global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>Let <var>option</var> be the result of <span data-x="create an element">creating an
   element</span> given <var>document</var>, "<code data-x="">option</code>", and the <span>HTML
   namespace</span>.</p></li>

   <li><p>If <var>text</var> is not the empty string, then append to <var>option</var> a new
   <code>Text</code> node whose data is <var>text</var>.</p></li>

   <li><p>If <var>value</var> is given, then <span data-x="concept-element-attributes-set-value">set
   an attribute value</span> for <var>option</var> using "<code
   data-x="attr-option-value">value</code>" and <var>value</var>.</p></li>

   <li><p>If <var>defaultSelected</var> is true, then <span
   data-x="concept-element-attributes-set-value">set an attribute value</span> for <var>option</var>
   using "<code data-x="attr-option-selected">selected</code>" and the empty string.</p></li>

   <li><p>If <var>selected</var> is true, then set <var>option</var>'s <span
   data-x="concept-option-selectedness">selectedness</span> to true; otherwise set its <span
   data-x="concept-option-selectedness">selectedness</span> to false (even if
   <var>defaultSelected</var> is true).</p></li>

   <li><p>Return <var>option</var>.</p></li>
  </ol>

  </div>



  <h4>The <dfn element><code>textarea</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Interactive content</span>.</dd>
   <dd><span data-x="category-listed">Listed</span>, <span data-x="category-label">labelable</span>, <span data-x="category-submit">submittable</span>, <span data-x="category-reset">resettable</span>, and <span data-x="category-autocapitalize-and-autocorrect">autocapitalize-and-autocorrect inheriting</span> <span>form-associated element</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="text content">Text</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-fe-autocomplete">autocomplete</code></dd>
   <dd><code data-x="attr-textarea-cols">cols</code></dd>
   <dd><code data-x="attr-fe-dirname">dirname</code></dd>
   <dd><code data-x="attr-fe-disabled">disabled</code></dd>
   <dd><code data-x="attr-fae-form">form</code></dd>
   <dd><code data-x="attr-textarea-maxlength">maxlength</code></dd>
   <dd><code data-x="attr-textarea-minlength">minlength</code></dd>
   <dd><code data-x="attr-fe-name">name</code></dd>
   <dd><code data-x="attr-textarea-placeholder">placeholder</code></dd>
   <dd><code data-x="attr-textarea-readonly">readonly</code></dd>
   <dd><code data-x="attr-textarea-required">required</code></dd>
   <dd><code data-x="attr-textarea-rows">rows</code></dd>
   <dd><code data-x="attr-textarea-wrap">wrap</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-textarea">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-textarea">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLTextAreaElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fe-autocomplete">autocomplete</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-textarea-cols">cols</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-textarea-dirName">dirName</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-fe-disabled">disabled</span>;
  readonly attribute <span>HTMLFormElement</span>? <span data-x="dom-fae-form">form</span>;
  [<span>CEReactions</span>] attribute long <span data-x="dom-textarea-maxLength">maxLength</span>;
  [<span>CEReactions</span>] attribute long <span data-x="dom-textarea-minLength">minLength</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fe-name">name</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-textarea-placeholder">placeholder</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-textarea-readOnly">readOnly</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-textarea-required">required</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-textarea-rows">rows</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-textarea-wrap">wrap</span>;

  readonly attribute DOMString <span data-x="dom-textarea-type">type</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-textarea-defaultValue">defaultValue</span>;
  attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-textarea-value">value</span>;
  readonly attribute unsigned long <span data-x="dom-textarea-textLength">textLength</span>;

  readonly attribute boolean <span data-x="dom-cva-willValidate">willValidate</span>;
  readonly attribute <span>ValidityState</span> <span data-x="dom-cva-validity">validity</span>;
  readonly attribute DOMString <span data-x="dom-cva-validationMessage">validationMessage</span>;
  boolean <span data-x="dom-cva-checkValidity">checkValidity</span>();
  boolean <span data-x="dom-cva-reportValidity">reportValidity</span>();
  undefined <span data-x="dom-cva-setCustomValidity">setCustomValidity</span>(DOMString error);

  readonly attribute <span>NodeList</span> <span data-x="dom-lfe-labels">labels</span>;

  undefined <span data-x="dom-textarea/input-select">select</span>();
  attribute unsigned long <span data-x="dom-textarea/input-selectionStart">selectionStart</span>;
  attribute unsigned long <span data-x="dom-textarea/input-selectionEnd">selectionEnd</span>;
  attribute DOMString <span data-x="dom-textarea/input-selectionDirection">selectionDirection</span>;
  undefined <span data-x="dom-textarea/input-setRangeText">setRangeText</span>(DOMString replacement);
  undefined <span data-x="dom-textarea/input-setRangeText">setRangeText</span>(DOMString replacement, unsigned long start, unsigned long end, optional <span>SelectionMode</span> selectionMode = "preserve");
  undefined <span data-x="dom-textarea/input-setSelectionRange">setSelectionRange</span>(unsigned long start, unsigned long end, optional DOMString direction);
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLTextAreaElement</code>.</dd>
  </dl>

  <p>The <code>textarea</code> element <span>represents</span> a multiline plain text edit
  control<span w-nodev> for the element's <dfn data-x="concept-textarea-raw-value">raw
  value</dfn></span>. The contents of the control represent the control's default value.</p>

  <div w-nodev>

  <p>The <span data-x="concept-textarea-raw-value">raw value</span> of a <code>textarea</code>
  control must be initially the empty string.</p>

  <p class="note">This element <a href="#bidireq">has rendering requirements involving the
  bidirectional algorithm</a>.</p>

  </div>

  <p>The <dfn element-attr for="textarea"><code
  data-x="attr-textarea-readonly">readonly</code></dfn> attribute is a <span>boolean
  attribute</span> used to control whether the text can be edited by the user or not.</p>

  <div class="example">

   <p>In this example, a text control is marked read-only because it represents a read-only
   file:</p>

   <pre><code class="html">Filename: &lt;code>/etc/bash.bashrc&lt;/code>
&lt;textarea name="buffer" readonly>
# System-wide .bashrc file for interactive bash(1) shells.

# To enable the settings / commands in this file for login shells as well,
# this file has to be sourced in /etc/profile.

# If not running interactively, don't do anything
[ -z "$PS1" ] &amp;amp;&amp;amp; return

...&lt;/textarea></code></pre>

  </div>

  <div w-nodev>

  <p><strong>Constraint validation</strong>: If the <code
  data-x="attr-textarea-readonly">readonly</code> attribute is specified on a <code>textarea</code>
  element, the element is <span>barred from constraint validation</span>.</p>

  <p>A <code>textarea</code> element is <span data-x="concept-fe-mutable">mutable</span> if it is
  neither <span data-x="concept-fe-disabled">disabled</span> nor has a <code
  data-x="attr-textarea-readonly">readonly</code> attribute specified.</p>

  <p>When a <code>textarea</code> is <span data-x="concept-fe-mutable">mutable</span>, its <span
  data-x="concept-textarea-raw-value">raw value</span> should be editable by the user: the user
  agent should allow the user to edit, insert, and remove text, and to insert and remove line breaks
  in the form of U+000A LINE FEED (LF) characters. Any time the user causes the element's <span
  data-x="concept-textarea-raw-value">raw value</span> to change, the user agent must <span>queue an
  element task</span> on the <span>user interaction task source</span> given the
  <code>textarea</code> element to <span data-x="concept-event-fire">fire an event</span> named
  <code data-x="event-input">input</code> at the <code>textarea</code> element, with the <code
  data-x="dom-Event-bubbles">bubbles</code> and <code data-x="dom-Event-composed">composed</code>
  attributes initialized to true. User agents may wait for a suitable break in the user's
  interaction before queuing the task; for example, a user agent could wait for the user to have not
  hit a key for 100ms, so as to only fire the event when the user pauses, instead of continuously
  for each keystroke.</p>
  <!-- same text is present in the <input> section -->

  <p>A <code>textarea</code> element's <span data-x="concept-fe-dirty">dirty value flag</span> must
  be set to true whenever the user interacts with the control in a way that changes the <span
  data-x="concept-textarea-raw-value">raw value</span>.</p>

  <p>The <span data-x="concept-node-clone-ext">cloning steps</span> for <code>textarea</code>
  elements given <var>node</var>, <var>copy</var>, and <var>subtree</var> are to propagate the <span
  data-x="concept-textarea-raw-value">raw value</span> and <span data-x="concept-fe-dirty">dirty
  value flag</span> from <var>node</var> to <var>copy</var>.</p>

  <p>The <span>children changed steps</span> for <code>textarea</code> elements must, if the
  element's <span data-x="concept-fe-dirty">dirty value flag</span> is false, set the element's
  <span data-x="concept-textarea-raw-value">raw value</span> to its <span>child text
  content</span>.</p>

  <p>The <span data-x="concept-form-reset-control">reset algorithm</span> for <code>textarea</code>
  elements is to set the <span>user validity</span> to false, <span
  data-x="concept-fe-dirty">dirty value flag</span> back to false, and set the <span
  data-x="concept-textarea-raw-value">raw value</span> of element to its <span>child text
  content</span>.</p>

  <p>When a <code>textarea</code> element is popped off the <span>stack of open elements</span> of
  an <span>HTML parser</span> or <span>XML parser</span>, then the user agent must invoke the
  element's <span data-x="concept-form-reset-control">reset algorithm</span>.</p>

  <!-- this next bit is also in the <input> Text/Search section -->
  <!-- and something similar is in the session history section -->
  <p>If the element is <span data-x="concept-fe-mutable">mutable</span>, the user agent should allow
  the user to change the writing direction of the element, setting it either to a left-to-right
  writing direction or a right-to-left writing direction. If the user does so, the user agent must
  then run the following steps:</p>

  <ol>
   <li><p>Set the element's <code data-x="attr-dir">dir</code> attribute to "<code
   data-x="attr-dir-ltr">ltr</code>" if the user selected a left-to-right writing direction, and
   "<code data-x="attr-dir-rtl">rtl</code>" if the user selected a right-to-left writing
   direction.</p></li>

   <li><p><span>Queue an element task</span> on the <span>user interaction task source</span> given
   the <code>textarea</code> element to <span data-x="concept-event-fire">fire an event</span> named
   <code data-x="event-input">input</code> at the <code>textarea</code> element, with the <code
   data-x="dom-Event-bubbles">bubbles</code> and <code data-x="dom-Event-composed">composed</code>
   attributes initialized to true.</p></li>
  </ol>

  </div>

  <p>The <dfn element-attr for="textarea"><code data-x="attr-textarea-cols">cols</code></dfn>
  attribute specifies the expected maximum number of characters per line. If the <code
  data-x="attr-textarea-cols">cols</code> attribute is specified, its value must be a <span>valid
  non-negative integer</span> greater than zero. <span w-nodev>If applying the <span>rules for
  parsing non-negative integers</span> to the attribute's value results in a number greater than
  zero, then the element's <dfn data-x="attr-textarea-cols-value">character width</dfn> is that
  value; otherwise, it is 20.</span></p>

  <div w-nodev>

  <p>The user agent may use the <code>textarea</code> element's <span
  data-x="attr-textarea-cols-value">character width</span> as a hint to the user as to how many
  characters the server prefers per line (e.g. for visual user agents by making the width of the
  control be that many characters). In visual renderings, the user agent should wrap the user's
  input in the rendering so that each line is no wider than this number of characters.</p>

  </div>

  <p>The <dfn element-attr for="textarea"><code data-x="attr-textarea-rows">rows</code></dfn>
  attribute specifies the number of lines to show. If the <code
  data-x="attr-textarea-rows">rows</code> attribute is specified, its value must be a <span>valid
  non-negative integer</span> greater than zero. <span w-nodev>If applying the <span>rules for
  parsing non-negative integers</span> to the attribute's value results in a number greater than
  zero, then the element's <dfn data-x="attr-textarea-rows-value">character height</dfn> is that
  value; otherwise, it is 2.</span></p>

  <div w-nodev>

  <p>Visual user agents should set the height of the control to the number of lines given by <span
  data-x="attr-textarea-rows-value">character height</span>.</p>

  </div>

  <p>The <dfn element-attr for="textarea"><code data-x="attr-textarea-wrap">wrap</code></dfn>
  attribute is an <span>enumerated attribute</span> with the following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="textarea/wrap"><code
     data-x="attr-textarea-wrap-soft">soft</code></dfn>
     <td><dfn data-x="attr-textarea-wrap-soft-state">Soft</dfn>
     <td>Text is not to be wrapped when submitted (though can still be wrapped in the rendering).
    <tr>
     <td><dfn attr-value for="textarea/wrap"><code
     data-x="attr-textarea-wrap-hard">hard</code></dfn>
     <td><dfn data-x="attr-textarea-wrap-hard-state">Hard</dfn>
     <td>Text is to have newlines added by the user agent so that the text is wrapped when it is
     submitted.
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <span
  data-x="attr-textarea-wrap-soft-state">Soft</span> state.</p>

  <p>If the element's <code data-x="attr-textarea-wrap">wrap</code> attribute is in the <span
  data-x="attr-textarea-wrap-hard-state">Hard</span> state, the <code
  data-x="attr-textarea-cols">cols</code> attribute must be specified.</p>

  <!-- attr-textarea-wrap-off (attr-textarea-wrap-off-state): not conforming; turns off the
  rendering of wrapping but otherwise acts like 'soft' -->

  <div w-nodev>

  <p>For historical reasons, the element's value is normalized in three different ways for three
  different purposes. The <span data-x="concept-textarea-raw-value">raw value</span> is the value as
  it was originally set. It is not normalized. The <span data-x="concept-fe-api-value">API
  value</span> is the value used in the <code data-x="dom-textarea-value">value</code> IDL
  attribute, <code data-x="dom-textarea-textLength">textLength</code> IDL attribute, and by the
  <code data-x="attr-fe-maxlength">maxlength</code> and <code
  data-x="attr-fe-minlength">minlength</code> content attributes. It is normalized so that line
  breaks use U+000A LINE FEED (LF) characters. Finally, there is the <span
  data-x="concept-fe-value">value</span>, as used in form submission and other processing models in
  this specification. It is normalized as for the <span data-x="concept-fe-api-value">API
  value</span>, and in addition, if necessary given the element's <code
  data-x="attr-textarea-wrap">wrap</code> attribute, additional line breaks are inserted to wrap the
  text at the given width.</p>

  <!--en-GB--><p id="textarea-line-break-normalisation-transformation">The algorithm for obtaining
  the element's <span data-x="concept-fe-api-value">API value</span> is to return the element's
  <span data-x="concept-textarea-raw-value">raw value</span>, with <span data-x="normalize
  newlines">newlines normalized</span>.</p>

  <p>The element's <span data-x="concept-fe-value">value</span> is defined to be the element's <span
  data-x="concept-textarea-raw-value">API value</span> with the <span>textarea wrapping
  transformation</span> applied. The <dfn>textarea wrapping transformation</dfn> is the following
  algorithm, as applied to a string:</p>

  <ol>
   <li><p>If the element's <code data-x="attr-textarea-wrap">wrap</code> attribute is in the <span
   data-x="attr-textarea-wrap-hard-state">Hard</span> state, insert U+000A LINE FEED (LF) characters
   into the string using an <span>implementation-defined</span> algorithm so that each line has no
   more than <span data-x="attr-textarea-cols-value">character width</span> characters. For the
   purposes of this requirement, lines are delimited by the start of the string, the end of the
   string, and U+000A LINE FEED (LF) characters.</p></li>
  </ol>

  </div>

  <p>The <dfn element-attr for="textarea"><code
  data-x="attr-textarea-maxlength">maxlength</code></dfn> attribute is a <span
  data-x="attr-fe-maxlength">form control <code data-x="">maxlength</code> attribute</span>.</p>

  <p>If the <code>textarea</code> element has a <span>maximum allowed value length</span>, then the
  element's children must be such that the <span>length</span> of the value of the element's
  <span>descendant text content</span> with <span data-x="normalize newlines">newlines
  normalized</span> is less than or equal to the element's <span>maximum allowed value
  length</span>.</p>

  <p>The <dfn element-attr for="textarea"><code
  data-x="attr-textarea-minlength">minlength</code></dfn> attribute is a <span
  data-x="attr-fe-minlength">form control <code data-x="">minlength</code> attribute</span>.</p>

  <!-- we allow the default to be smaller than the minimum, so that you can have incomplete, but not
  empty, input already in the form, like a template that needs to be filled in -->

  <p>The <dfn element-attr for="textarea"><code
  data-x="attr-textarea-required">required</code></dfn> attribute is a <span>boolean
  attribute</span>. When specified, the user will be required to enter a value before submitting the
  form.</p>

  <div w-nodev>

  <p><strong>Constraint validation</strong>: If the element has its <code
  data-x="attr-textarea-required">required</code> attribute specified, and the element is <span
  data-x="concept-fe-mutable">mutable</span>, and the element's <span
  data-x="concept-fe-value">value</span> is the empty string, then the element is <span>suffering
  from being missing</span>.</p>

  </div>


  <!-- substantially similar text in the <input> section -->

  <p>The <dfn element-attr for="textarea"><code
  data-x="attr-textarea-placeholder">placeholder</code></dfn> attribute represents a <em>short</em>
  hint (a word or short phrase) intended to aid the user with data entry when the control has no
  value. A hint could be a sample value or a brief description of the expected format.</p>

  <p>The <code data-x="attr-textarea-placeholder">placeholder</code> attribute should not be used as
  an alternative to a <code>label</code>. For a longer hint or other advisory text, the <code
  data-x="attr-title">title</code> attribute is more appropriate.</p>

  <p class="note">These mechanisms are very similar but subtly different: the hint given by the
  control's <code>label</code> is shown at all times; the short hint given in the <code
  data-x="attr-textarea-placeholder">placeholder</code> attribute is shown before the user enters a
  value; and the hint in the <code data-x="attr-title">title</code> attribute is shown when the user
  requests further help.</p>

  <div w-nodev>

  <p>User agents should present this hint to the user when the element's <span
  data-x="concept-fe-value">value</span> is the empty string and the control is not
  <span>focused</span> (e.g. by displaying it inside a blank unfocused control). All U+000D CARRIAGE
  RETURN U+000A LINE FEED character pairs (CRLF) in the hint, as well as all other U+000D CARRIAGE
  RETURN (CR) and U+000A LINE FEED (LF) characters in the hint, must be treated as line breaks when
  rendering the hint.</p>

  <p>If a user agent normally doesn't show this hint to the user when the control is
  <span>focused</span>, then the user agent should nonetheless show the hint for the control if it
  was focused as a result of the <code data-x="attr-fe-autofocus">autofocus</code> attribute, since
  in that case the user will not have had an opportunity to examine the control before focusing
  it.</p>

  </div>

  <p>
  The <code data-x="attr-fe-name">name</code> attribute represents the element's name.
  The <code data-x="attr-fe-dirname">dirname</code> attribute controls how the element's <span
  data-x="the directionality">directionality</span> is submitted.
  The <code data-x="attr-fe-disabled">disabled</code> attribute is used to make the control
  non-interactive and to prevent its value from being submitted.
  The <code data-x="attr-fae-form">form</code> attribute is used to explicitly associate the
  <code>textarea</code> element with its <span>form owner</span>.
  The <code data-x="attr-fe-autocomplete">autocomplete</code> attribute controls how the user agent
  provides autofill behavior.
  </p>

  <dl class="domintro">
   <dt><code data-x=""><var>textarea</var>.<span subdfn data-x="dom-textarea-type">type</span></code></dt>

   <dd><p>Returns the string "<code data-x="">textarea</code>".</p></dd>

   <dt><code data-x=""><var>textarea</var>.<span subdfn data-x="dom-textarea-value">value</span></code></dt>

   <dd>
    <p>Returns the current value of the element.</p>

    <p>Can be set, to change the value.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLTextAreaElement"><code
  data-x="dom-textarea-cols">cols</code></dfn>, <dfn attribute for="HTMLTextAreaElement"><code
  data-x="dom-textarea-placeholder">placeholder</code></dfn>, <dfn attribute
  for="HTMLTextAreaElement"><code data-x="dom-textarea-required">required</code></dfn>, <dfn
  attribute for="HTMLTextAreaElement"><code data-x="dom-textarea-rows">rows</code></dfn>, and <dfn
  attribute for="HTMLTextAreaElement"><code data-x="dom-textarea-wrap">wrap</code></dfn> IDL
  attributes must <span>reflect</span> the respective content attributes of the same name. The <code
  data-x="dom-textarea-cols">cols</code> and <code data-x="dom-textarea-rows">rows</code> attributes
  are <span>limited to only positive numbers with fallback</span>. The <code
  data-x="dom-textarea-cols">cols</code> IDL attribute's <span>default value</span> is 20. The <code
  data-x="dom-textarea-rows">rows</code> IDL attribute's <span>default value</span> is 2. The <dfn
  attribute for="HTMLTextAreaElement"><code data-x="dom-textarea-dirName">dirName</code></dfn> IDL
  attribute must <span>reflect</span> the <code data-x="attr-fe-dirname">dirname</code> content
  attribute. The <dfn attribute for="HTMLTextAreaElement"><code
  data-x="dom-textarea-maxLength">maxLength</code></dfn> IDL attribute must <span>reflect</span> the
  <code data-x="attr-textarea-maxlength">maxlength</code> content attribute, <span>limited to only
  non-negative numbers</span>. The <dfn attribute for="HTMLTextAreaElement"><code
  data-x="dom-textarea-minLength">minLength</code></dfn> IDL attribute must <span>reflect</span> the
  <code data-x="attr-textarea-minlength">minlength</code> content attribute, <span>limited to only
  non-negative numbers</span>. The <dfn attribute for="HTMLTextAreaElement"><code
  data-x="dom-textarea-readOnly">readOnly</code></dfn> IDL attribute must <span>reflect</span> the
  <code data-x="attr-textarea-readonly">readonly</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLTextAreaElement"><code data-x="dom-textarea-type">type</code></dfn>
  IDL attribute must return the value "<code data-x="">textarea</code>".</p>

  <p>The <dfn attribute for="HTMLTextAreaElement"><code
  data-x="dom-textarea-defaultValue">defaultValue</code></dfn> attribute's getter must return the
  element's <span>child text content</span>.

  <p>The <code data-x="dom-textarea-defaultValue">defaultValue</code> attribute's setter must
  <span>string replace all</span> with the given value within this element.</p>

  <p>The <dfn attribute for="HTMLTextAreaElement"><code
  data-x="dom-textarea-value">value</code></dfn> IDL attribute must, on getting, return the
  element's <span data-x="concept-fe-api-value">API value</span>. On setting, it must perform the
  following steps:</p>

  <ol>
   <li><p>Let <var>oldAPIValue</var> be this element's <span
   data-x="concept-fe-api-value">API value</span>.</p></li>

   <li><p>Set this element's <span data-x="concept-textarea-raw-value">raw value</span> to the new
   value.</p></li>

   <li><p>Set this element's <span data-x="concept-fe-dirty">dirty value flag</span> to
   true.</p></li>

   <li><p>If the new <span data-x="concept-fe-api-value">API value</span> is different from
   <var>oldAPIValue</var>, then move the <span data-x="concept-textarea/input-cursor">text entry
   cursor position</span> to the end of the text control, unselecting any selected text and <span
   data-x="set the selection direction">resetting the selection direction</span> to "<code
   data-x="">none</code>".</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLTextAreaElement"><code
  data-x="dom-textarea-textLength">textLength</code></dfn> IDL attribute must return the
  <span>length</span> of the element's <span data-x="concept-fe-api-value">API value</span>.</p>

  <p>The <code data-x="dom-cva-willValidate">willValidate</code>, <code
  data-x="dom-cva-validity">validity</code>, and <code
  data-x="dom-cva-validationMessage">validationMessage</code> IDL attributes, and the <code
  data-x="dom-cva-checkValidity">checkValidity()</code>, <code
  data-x="dom-cva-reportValidity">reportValidity()</code>, and <code
  data-x="dom-cva-setCustomValidity">setCustomValidity()</code> methods, are part of the
  <span>constraint validation API</span>. The <code data-x="dom-lfe-labels">labels</code> IDL
  attribute provides a list of the element's <code>label</code>s. The <code
  data-x="dom-textarea/input-select">select()</code>, <code
  data-x="dom-textarea/input-selectionStart">selectionStart</code>, <code
  data-x="dom-textarea/input-selectionEnd">selectionEnd</code>, <code
  data-x="dom-textarea/input-selectionDirection">selectionDirection</code>, <code
  data-x="dom-textarea/input-setRangeText">setRangeText()</code>, and <code
  data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code> methods and IDL attributes
  expose the element's text selection. The <code data-x="dom-fe-disabled">disabled</code>,
  <code data-x="dom-fae-form">form</code>, and <code data-x="dom-fe-name">name</code> IDL attributes
  are part of the element's forms API.</p>

  </div>

  <div class="example">

   <p>Here is an example of a <code>textarea</code> being used for unrestricted free-form text input
   in a form:</p>

   <pre><code class="html">&lt;p>If you have any comments, please let us know: &lt;textarea cols=80 name=comments>&lt;/textarea>&lt;/p></code></pre>

   <p>To specify a maximum length for the comments, one can use the <code
   data-x="attr-textarea-maxlength">maxlength</code> attribute:</p>

   <pre><code class="html">&lt;p>If you have any short comments, please let us know: &lt;textarea cols=80 name=comments maxlength=200>&lt;/textarea>&lt;/p></code></pre>

   <p>To give a default value, text can be included inside the element:</p>

   <pre><code class="html">&lt;p>If you have any comments, please let us know: &lt;textarea cols=80 name=comments>You rock!&lt;/textarea>&lt;/p></code></pre>

   <p>You can also give a minimum length. Here, a letter needs to be filled out by the user; a
   template (which is shorter than the minimum length) is provided, but is insufficient to submit
   the form:</p>

   <pre><code class="html">&lt;textarea required minlength="500">Dear Madam Speaker,

Regarding your letter dated ...

...

Yours Sincerely,

...&lt;/textarea></code></pre>

   <p>A placeholder can be given as well, to suggest the basic form to the user, without providing
   an explicit template:</p>

   <pre><code class="html">&lt;textarea placeholder="Dear Francine,

They closed the parks this week, so we won't be able to
meet your there. Should we just have dinner?

Love,
Daddy">&lt;/textarea></code></pre>

   <p>To have the browser submit <span>the directionality</span> of the element along with the
   value, the <code data-x="attr-fe-dirname">dirname</code> attribute can be specified:</p>

   <pre><code class="html">&lt;p>If you have any comments, please let us know (you may use either English or Hebrew for your comments):
&lt;textarea cols=80 name=comments dirname=comments.dir>&lt;/textarea>&lt;/p></code></pre>

  </div>



  <h4>The <dfn element><code>output</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span data-x="category-listed">Listed</span>, <span data-x="category-label">labelable</span>, <span data-x="category-reset">resettable</span>, and <span data-x="category-autocapitalize-and-autocorrect">autocapitalize-and-autocorrect inheriting</span> <span>form-associated element</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-output-for">for</code></dd>
   <dd><code data-x="attr-fae-form">form</code></dd>
   <dd><code data-x="attr-fe-name">name</code> <!-- variant --> </dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-output">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-output">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLOutputElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [SameObject, PutForwards=<span data-x="dom-DOMTokenList-value">value</span>] readonly attribute <span>DOMTokenList</span> <span data-x="dom-output-htmlFor">htmlFor</span>;
  readonly attribute <span>HTMLFormElement</span>? <span data-x="dom-fae-form">form</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fe-name">name</span>;

  readonly attribute DOMString <span data-x="dom-output-type">type</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-output-defaultValue">defaultValue</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-output-value">value</span>;

  readonly attribute boolean <span data-x="dom-cva-willValidate">willValidate</span>;
  readonly attribute <span>ValidityState</span> <span data-x="dom-cva-validity">validity</span>;
  readonly attribute DOMString <span data-x="dom-cva-validationMessage">validationMessage</span>;
  boolean <span data-x="dom-cva-checkValidity">checkValidity</span>();
  boolean <span data-x="dom-cva-reportValidity">reportValidity</span>();
  undefined <span data-x="dom-cva-setCustomValidity">setCustomValidity</span>(DOMString error);

  readonly attribute <span>NodeList</span> <span data-x="dom-lfe-labels">labels</span>;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLOutputElement</code>.</dd>
  </dl>

  <p>The <code>output</code> element <span>represents</span> the result of a calculation performed
  by the application, or the result of a user action.</p>

  <p class="note">This element can be contrasted with the <code>samp</code> element, which is the
  appropriate element for quoting the output of other programs run previously.</p>

  <p>The <dfn element-attr for="output"><code data-x="attr-output-for">for</code></dfn> content
  attribute allows an explicit relationship to be made between the result of a calculation and the
  elements that represent the values that went into the calculation or that otherwise influenced the
  calculation. The <code data-x="attr-output-for">for</code> attribute, if specified, must contain a
  string consisting of an <span>unordered set of unique space-separated tokens</span>, none of which
  are <span>identical to</span> another token and each of which must have the value of an <span
  data-x="concept-id">ID</span> of an element in the same <span>tree</span>.</p>

  <p>The <code data-x="attr-fae-form">form</code> attribute is used to explicitly associate the
  <code>output</code> element with its <span>form owner</span>. The <code
  data-x="attr-fe-name">name</code> attribute represents the element's name. The <code>output</code>
  element is associated with a form so that it can be easily <span>referenced</span> from the event
  handlers of form controls; the element's value itself is not submitted when the form is
  submitted.</p>

  <div w-nodev>

  <p>The element has a <dfn data-x="concept-output-default-value-override">default value
  override</dfn> (null or a string). Initially it must be null.</p>

  <p>The element's <dfn data-x="concept-output-default-value">default value</dfn> is determined by
  the following steps:</p>

  <ol>
   <li><p>If this element's <span data-x="concept-output-default-value-override">default value
   override</span> is non-null, then return it.</p></li>

   <li><p>Return this element's <span>descendant text content</span>.</p></li>
  </ol>

  <p>The <span data-x="concept-form-reset-control">reset algorithm</span> for <code>output</code>
  elements is to run these steps:

  <ol>
   <li><p><span>String replace all</span> with this element's <span
   data-x="concept-output-default-value">default value</span> within this element.</p></li>

   <li><p>Set this element's <span data-x="concept-output-default-value-override">default value
   override</span> to null.</p></li>
  </ol>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>output</var>.<span subdfn data-x="dom-output-value">value</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the element's current value.</p>

    <p>Can be set, to change the value.</p>
   </dd>

   <dt><code data-x=""><var>output</var>.<span subdfn data-x="dom-output-defaultValue">defaultValue</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the element's current default value.</p>

    <p>Can be set, to change the default value.</p>
   </dd>

   <dt><code data-x=""><var>output</var>.<span subdfn data-x="dom-output-type">type</span></code></dt>
   <dd><p>Returns the string "<code data-x="">output</code>".</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLOutputElement"><code data-x="dom-output-value">value</code></dfn>
  getter steps are to return <span>this</span>'s <span>descendant text content</span>.</p>

  <p>The <code data-x="dom-output-value">value</code> setter steps are:</p>

  <ol>
   <li><p>Set <span>this</span>'s <span data-x="concept-output-default-value-override">default value
   override</span> to its <span data-x="concept-output-default-value">default value</span>.</p></li>

   <li><p><span>String replace all</span> with the given value within <span>this</span>.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLOutputElement"><code
  data-x="dom-output-defaultValue">defaultValue</code></dfn> getter steps are to return the result
  of running <span>this</span>'s <span data-x="concept-output-default-value">default
  value</span>.</p>

  <p>The <code data-x="dom-output-defaultValue">defaultValue</code> setter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="concept-output-default-value-override">default value
   override</span> is null, then <span>string replace all</span> with the given value within
   <span>this</span> and return.</p></li>

   <li><p>Set <span>this</span>'s <span data-x="concept-output-default-value-override">default value
   override</span> to the given value.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLOutputElement"><code data-x="dom-output-type">type</code></dfn>
  getter steps are to return "<code data-x="">output</code>".</p>

  <p>The <dfn attribute for="HTMLOutputElement"><code
  data-x="dom-output-htmlFor">htmlFor</code></dfn> IDL attribute must <span>reflect</span> the <code
  data-x="attr-output-for">for</code> content attribute.</p>

  <p>The <code data-x="dom-cva-willValidate">willValidate</code>, <code
  data-x="dom-cva-validity">validity</code>, and <code
  data-x="dom-cva-validationMessage">validationMessage</code> IDL attributes, and the <code
  data-x="dom-cva-checkValidity">checkValidity()</code>, <code
  data-x="dom-cva-reportValidity">reportValidity()</code>, and <code
  data-x="dom-cva-setCustomValidity">setCustomValidity()</code> methods, are part of the
  <span>constraint validation API</span>. The <code data-x="dom-lfe-labels">labels</code> IDL
  attribute provides a list of the element's <code>label</code>s. The <code
  data-x="dom-fae-form">form</code> and <code data-x="dom-fe-name">name</code> IDL attributes are
  part of the element's forms API.</p>

  </div>

  <div class="example">

   <p>A simple calculator could use <code>output</code> for its display of calculated results:</p>

   <pre><code class="html">&lt;form onsubmit="return false" oninput="o.value = a.valueAsNumber + b.valueAsNumber">
 &lt;input id=a type=number step=any> +
 &lt;input id=b type=number step=any> =
 &lt;output id=o for="a b">&lt;/output>
&lt;/form></code></pre>

  </div>

  <div class="example">

   <p>In this example, an <code>output</code> element is used to report the results of a calculation performed by a remote
   server, as they come in:</p>

   <pre><code class="html">&lt;output id="result">&lt;/output>
&lt;script>
 var primeSource = new WebSocket('ws://primes.example.net/');
 primeSource.onmessage = function (event) {
   document.getElementById('result').value = event.data;
 }
&lt;/script></code></pre>

  </div>



  <h4>The <dfn element><code>progress</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span data-x="category-label">Labelable element</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>, but there must be no <code>progress</code> element descendants.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-progress-value">value</code></dd>
   <dd><code data-x="attr-progress-max">max</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-progress">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-progress">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLProgressElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute double <span data-x="dom-progress-value">value</span>;
  [<span>CEReactions</span>] attribute double <span data-x="dom-progress-max">max</span>;
  readonly attribute double <span data-x="dom-progress-position">position</span>;
  readonly attribute <span>NodeList</span> <span data-x="dom-lfe-labels">labels</span>;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLProgressElement</code>.</dd>
  </dl>

  <p>The <code>progress</code> element <span>represents</span> the completion progress of a task.
  The progress is either indeterminate, indicating that progress is being made but that it is not
  clear how much more work remains to be done before the task is complete (e.g. because the task is
  waiting for a remote host to respond), or the progress is a number in the range zero to a maximum,
  giving the fraction of work that has so far been completed.</p>

  <p>There are two attributes that determine the current task completion represented by the element.
  The <dfn element-attr for="progress"><code data-x="attr-progress-value">value</code></dfn>
  attribute specifies how much of the task has been completed, and the <dfn element-attr
  for="progress"><code data-x="attr-progress-max">max</code></dfn> attribute specifies how much work
  the task requires in total. The units are arbitrary and not specified.</p>

  <p class="note">To make a determinate progress bar, add a <code
  data-x="attr-progress-value">value</code> attribute with the current progress (either a number from
  0.0 to 1.0, or, if the <code data-x="attr-progress-max">max</code> attribute is specified, a number
  from 0 to the value of the <code data-x="attr-progress-max">max</code> attribute). To make an
  indeterminate progress bar, remove the <code data-x="attr-progress-value">value</code>
  attribute.</p>

  <p>Authors are encouraged to also include the current value and the maximum value inline as text
  inside the element, so that the progress is made available to users of legacy user agents.</p>

  <div class="example">

   <p>Here is a snippet of a web application that shows the progress of some automated task:</p>

   <pre><code class="html">&lt;section>
 &lt;h2>Task Progress&lt;/h2>
 &lt;p>Progress: &lt;progress id=p max=100>&lt;span>0&lt;/span>%&lt;/progress>&lt;/p>
 &lt;script>
  var progressBar = document.getElementById('p');
  function updateProgress(newValue) {
    progressBar.value = newValue;
    progressBar.getElementsByTagName('span')[0].textContent = newValue;
  }
 &lt;/script>
&lt;/section></code></pre>

   <p>(The <code data-x="">updateProgress()</code> method in this example would be called by some
   other code on the page to update the actual progress bar as the task progressed.)</p>

  </div>

  <p>The <code data-x="attr-progress-value">value</code> and <code
  data-x="attr-progress-max">max</code> attributes, when present, must have values that are <span
  data-x="valid floating-point number">valid floating-point numbers</span>. The <code
  data-x="attr-progress-value">value</code> attribute, if present, must have a value greater than or
  equal to zero, and less than or equal to the value of the <code
  data-x="attr-progress-max">max</code> attribute, if present, or 1.0, otherwise. The <code
  data-x="attr-progress-max">max</code> attribute, if present, must have a value greater than
  zero.</p>

  <p class="note">The <code>progress</code> element is the wrong element to use for something that
  is just a gauge, as opposed to task progress. For instance, indicating disk space usage using
  <code>progress</code> would be inappropriate. Instead, the <code>meter</code> element is available
  for such use cases.</p>

  <div w-nodev>

  <p><strong>User agent requirements</strong>: If the <code data-x="attr-progress-value">value</code>
  attribute is omitted, then the progress bar is an indeterminate progress bar. Otherwise, it is a
  determinate progress bar.</p>

  <p>If the progress bar is a determinate progress bar and the element has a <code
  data-x="attr-progress-max">max</code> attribute, the user agent must parse the <code
  data-x="attr-progress-max">max</code> attribute's value according to the <span>rules for parsing
  floating-point number values</span>. If this does not result in an error, and if the parsed value
  is greater than zero, then the <dfn data-x="concept-progress-maximum">maximum value</dfn> of the
  progress bar is that value. Otherwise, if the element has no <code
  data-x="attr-progress-max">max</code> attribute, or if it has one but parsing it resulted in an
  error, or if the parsed value was less than or equal to zero, then the <span
  data-x="concept-progress-maximum">maximum value</span> of the progress bar is 1.0.</p>

  <p>If the progress bar is a determinate progress bar, user agents must parse the <code
  data-x="attr-progress-value">value</code> attribute's value according to the <span>rules for
  parsing floating-point number values</span>. If this does not result in an error and the parsed
  value is greater than zero, then the <dfn data-x="concept-progress-value">value</dfn> of the
  progress bar is that parsed value. Otherwise, if parsing the <code
  data-x="attr-progress-value">value</code> attribute's value resulted in an error or a number less
  than or equal to zero, then the <span data-x="concept-progress-value">value</span> of the progress
  bar is zero.</p>

  <p>If the progress bar is a determinate progress bar, then the <dfn
  data-x="concept-progress-current-value">current value</dfn> is the <span
  data-x="concept-progress-maximum">maximum value</span>, if <span
  data-x="concept-progress-value">value</span> is greater than the <span
  data-x="concept-progress-maximum">maximum value</span>, and <span
  data-x="concept-progress-value">value</span> otherwise.</p>

  <p><strong>UA requirements for showing the progress bar</strong>: When representing a
  <code>progress</code> element to the user, the UA should indicate whether it is a determinate or
  indeterminate progress bar, and in the former case, should indicate the relative position of the
  <span data-x="concept-progress-current-value">current value</span> relative to the <span
  data-x="concept-progress-maximum">maximum value</span>.</p>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>progress</var>.<span subdfn data-x="dom-progress-position">position</span></code></dt>

   <dd>
    <p>For a determinate progress bar (one with known current and maximum values), returns the
    result of dividing the current value by the maximum value.</p>

    <p>For an indeterminate progress bar, returns &#x2212;1.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>If the progress bar is an indeterminate progress bar, then the <dfn attribute
  for="HTMLProgressElement"><code data-x="dom-progress-position">position</code></dfn> IDL attribute
  must return &#x2212;1.  Otherwise, it must return the result of dividing the <span
  data-x="concept-progress-current-value">current value</span> by the <span
  data-x="concept-progress-maximum">maximum value</span>.</p>

  <p>If the progress bar is an indeterminate progress bar, then the <dfn attribute
  for="HTMLProgressElement"><code data-x="dom-progress-value">value</code></dfn> IDL attribute, on
  getting, must return 0.  Otherwise, it must return the <span
  data-x="concept-progress-value">current value</span>. On setting, the given value must be
  converted to the <span>best representation of the number as a floating-point number</span> and
  then the <code data-x="dom-progress-value">value</code> content attribute must be set to that
  string.</p>

  <p class="note">Setting the <code data-x="dom-progress-value">value</code> IDL attribute to itself
  when the corresponding content attribute is absent would change the progress bar from an
  indeterminate progress bar to a determinate progress bar with no progress.</p>

  <p>The <dfn attribute for="HTMLProgressElement"><code data-x="dom-progress-max">max</code></dfn>
  IDL attribute must <span>reflect</span> the content attribute of the same name, <span>limited to
  only positive numbers</span>. The <span>default value</span> for <code
  data-x="dom-progress-max">max</code> is 1.0.</p>

  <p>The <code data-x="dom-lfe-labels">labels</code> IDL attribute provides a list of the element's
  <code>label</code>s.</p>

  </div>



  <h4>The <dfn element><code>meter</code></dfn> element</h4>
  <!-- Keep this after <progress> and NOT close to <time> -->

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span data-x="category-label">Labelable element</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>, but there must be no <code>meter</code> element descendants.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-meter-value">value</code></dd>
   <dd><code data-x="attr-meter-min">min</code></dd>
   <dd><code data-x="attr-meter-max">max</code></dd>
   <dd><code data-x="attr-meter-low">low</code></dd>
   <dd><code data-x="attr-meter-high">high</code></dd>
   <dd><code data-x="attr-meter-optimum">optimum</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-meter">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-meter">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLMeterElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute double <span data-x="dom-meter-value">value</span>;
  [<span>CEReactions</span>] attribute double <span data-x="dom-meter-min">min</span>;
  [<span>CEReactions</span>] attribute double <span data-x="dom-meter-max">max</span>;
  [<span>CEReactions</span>] attribute double <span data-x="dom-meter-low">low</span>;
  [<span>CEReactions</span>] attribute double <span data-x="dom-meter-high">high</span>;
  [<span>CEReactions</span>] attribute double <span data-x="dom-meter-optimum">optimum</span>;
  readonly attribute <span>NodeList</span> <span data-x="dom-lfe-labels">labels</span>;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLMeterElement</code>.</dd>
  </dl>

  <p>The <code>meter</code> element <span>represents</span> a scalar measurement within a known
  range, or a fractional value; for example disk usage, the relevance of a query result, or the
  fraction of a voting population to have selected a particular candidate.</p>

  <p>This is also known as a gauge.</p>

  <p>The <code>meter</code> element should not be used to indicate progress (as in a progress bar).
  For that role, HTML provides a separate <code>progress</code> element.</p>

  <p class="note">The <code>meter</code> element also does not represent a scalar value of arbitrary
  range &mdash; for example, it would be wrong to use this to report a weight, or height, unless
  there is a known maximum value.</p>

  <p>There are six attributes that determine the semantics of the gauge represented by the
  element.</p>

  <p>The <dfn element-attr for="meter"><code data-x="attr-meter-min">min</code></dfn> attribute
  specifies the lower bound of the range, and the <dfn element-attr for="meter"><code
  data-x="attr-meter-max">max</code></dfn> attribute specifies the upper bound. The <dfn
  element-attr for="meter"><code data-x="attr-meter-value">value</code></dfn> attribute specifies
  the value to have the gauge indicate as the "measured" value.</p>

  <p>The other three attributes can be used to segment the gauge's range into "low", "medium", and
  "high" parts, and to indicate which part of the gauge is the "optimum" part. The <dfn element-attr
  for="meter"><code data-x="attr-meter-low">low</code></dfn> attribute specifies the range that is
  considered to be the "low" part, and the <dfn element-attr for="meter"><code
  data-x="attr-meter-high">high</code></dfn> attribute specifies the range that is considered to be
  the "high" part. The <dfn element-attr for="meter"><code
  data-x="attr-meter-optimum">optimum</code></dfn> attribute gives the position that is "optimum";
  if that is higher than the "high" value then this indicates that the higher the value, the better;
  if it's lower than the "low" mark then it indicates that lower values are better, and naturally if
  it is in between then it indicates that neither high nor low values are good.</p>

  <p><span w-nodev><strong>Authoring requirements</strong>:</span> The <code
  data-x="attr-meter-value">value</code> attribute must be specified. The <code
  data-x="attr-meter-value">value</code>, <code data-x="attr-meter-min">min</code>, <code
  data-x="attr-meter-low">low</code>, <code data-x="attr-meter-high">high</code>, <code
  data-x="attr-meter-max">max</code>, and <code data-x="attr-meter-optimum">optimum</code> attributes,
  when present, must have values that are <span data-x="valid floating-point number">valid
  floating-point numbers</span>.</p>

  <p>In addition, the attributes' values are further constrained:</p>

  <p>Let <var>value</var> be the <code data-x="attr-meter-value">value</code> attribute's
  number.</p>

  <p>If the <code data-x="attr-meter-min">min</code> attribute is specified, then let <var>minimum</var> be that attribute's value; otherwise, let it be zero.</p>

  <p>If the <code data-x="attr-meter-max">max</code> attribute is specified, then let <var>maximum</var> be that attribute's value; otherwise, let it be 1.0.</p>

  <p>The following inequalities must hold, as applicable:</p>

  <ul>
   <li><p><var>minimum</var> &le; <var>value</var> &le; <var>maximum</var></p></li>

   <li><p><var>minimum</var> &le; <code data-x="attr-meter-low">low</code> &le; <var>maximum</var>
   (if <code data-x="attr-meter-low">low</code> is specified)</p></li>

   <li><p><var>minimum</var> &le; <code data-x="attr-meter-high">high</code> &le;
   <var>maximum</var> (if <code data-x="attr-meter-high">high</code> is specified)</p></li>

   <li><p><var>minimum</var> &le; <code data-x="attr-meter-optimum">optimum</code> &le;
   <var>maximum</var> (if <code data-x="attr-meter-optimum">optimum</code> is specified)</p></li>

   <li><p><code data-x="attr-meter-low">low</code> &le; <code data-x="attr-meter-high">high</code>
   (if both <code data-x="attr-meter-low">low</code> and <code data-x="attr-meter-high">high</code>
   are specified)</p></li>
  </ul>

  <p class="note">If no minimum or maximum is specified, then the range is assumed to be 0..1, and
  the value thus has to be within that range.</p>

  <p>Authors are encouraged to include a textual representation of the gauge's state in the
  element's contents, for users of user agents that do not support the <code>meter</code>
  element.</p>

  <p>When used with <span>microdata</span>, the <code>meter</code> element's <code
  data-x="attr-meter-value">value</code> attribute provides the element's machine-readable value.</p>

  <div class="example">

   <p>The following examples show three gauges that would all be three-quarters full:</p>

   <pre><code class="html">Storage space usage: &lt;meter value=6 max=8>6 blocks used (out of 8 total)&lt;/meter></code></pre>

   <pre><code class="html">Voter turnout: &lt;meter value=0.75>&lt;img alt="75%" src="graph75.png">&lt;/meter></code></pre>

   <pre><code class="html">Tickets sold: &lt;meter min="0" max="100" value="75">&lt;/meter></code></pre>

   <p>The following example is incorrect use of the element, because it doesn't give a range (and
   since the default maximum is 1, both of the gauges would end up looking maxed out):</p>

   <pre class="bad"><code class="html">&lt;p>The grapefruit pie had a radius of &lt;meter value=12>12cm&lt;/meter>
and a height of &lt;meter value=2>2cm&lt;/meter>.&lt;/p> &lt;!-- <strong>BAD!</strong> --></code></pre>

   <p>Instead, one would either not include the meter element, or use the meter element with a
   defined range to give the dimensions in context compared to other pies:</p>

   <pre><code class="html">&lt;p>The grapefruit pie had a radius of 12cm and a height of
2cm.&lt;/p>
&lt;dl>
 &lt;dt>Radius: &lt;dd> &lt;meter min=0 max=20 value=12>12cm&lt;/meter>
 &lt;dt>Height: &lt;dd> &lt;meter min=0 max=10 value=2>2cm&lt;/meter>
&lt;/dl></code></pre>

  </div>

  <p>There is no explicit way to specify units in the <code>meter</code> element, but the units may
  be specified in the <code data-x="attr-title">title</code> attribute in free-form text.</p>

  <div class="example">

   <p>The example above could be extended to mention the units:</p>

   <pre><code class="html">&lt;dl>
 &lt;dt>Radius: &lt;dd> &lt;meter min=0 max=20 value=12 title="centimeters">12cm&lt;/meter>
 &lt;dt>Height: &lt;dd> &lt;meter min=0 max=10 value=2 title="centimeters">2cm&lt;/meter>
&lt;/dl></code></pre>

  </div>

  <div w-nodev>

  <p><strong>User agent requirements</strong>: User agents must parse the <code
  data-x="attr-meter-min">min</code>, <code data-x="attr-meter-max">max</code>, <code
  data-x="attr-meter-value">value</code>, <code data-x="attr-meter-low">low</code>, <code
  data-x="attr-meter-high">high</code>, and <code data-x="attr-meter-optimum">optimum</code>
  attributes using the <span>rules for parsing floating-point number values</span>.</p>

  <p>User agents must then use all these numbers to obtain values for six points on the gauge, as
  follows. (The order in which these are evaluated is important, as some of the values refer to
  earlier ones.)</p>

  <dl>
   <dt>The <dfn data-x="concept-meter-minimum">minimum value</dfn></dt>

   <dd>
    <p>If the <code data-x="attr-meter-min">min</code> attribute is specified and a value could be
    parsed out of it, then the minimum value is that value. Otherwise, the minimum value is
    zero.</p>
   </dd>

   <dt>The <dfn data-x="concept-meter-maximum">maximum value</dfn></dt>

   <dd>
    <p>If the <code data-x="attr-meter-max">max</code> attribute is specified and a value could be
    parsed out of it, then the candidate maximum value is that value. Otherwise, the candidate
    maximum value is 1.0.</p>

    <p>If the candidate maximum value is greater than or equal to the minimum value, then the
    maximum value is the candidate maximum value. Otherwise, the maximum value is the same as the
    minimum value.</p>
   </dd>

   <dt>The <dfn data-x="concept-meter-actual">actual value</dfn></dt>

   <dd>
    <p>If the <code data-x="attr-meter-value">value</code> attribute is specified and a value could
    be parsed out of it, then that value is the candidate actual value. Otherwise, the candidate
    actual value is zero.</p>

    <p>If the candidate actual value is less than the minimum value, then the actual value is the
    minimum value.</p>

    <p>Otherwise, if the candidate actual value is greater than the maximum value, then the actual
    value is the maximum value.</p>

    <p>Otherwise, the actual value is the candidate actual value.</p>
   </dd>

   <dt>The <dfn data-x="concept-meter-low">low boundary</dfn></dt>

   <dd>
    <p>If the <code data-x="attr-meter-low">low</code> attribute is specified and a value could be
    parsed out of it, then the candidate low boundary is that value. Otherwise, the candidate low
    boundary is the same as the minimum value.</p>

    <p>If the candidate low boundary is less than the minimum value, then the low boundary is the
    minimum value.</p>

    <p>Otherwise, if the candidate low boundary is greater than the maximum value, then the low
    boundary is the maximum value.</p>

    <p>Otherwise, the low boundary is the candidate low boundary.</p>
   </dd>

   <dt>The <dfn data-x="concept-meter-high">high boundary</dfn></dt>

   <dd>
    <p>If the <code data-x="attr-meter-high">high</code> attribute is specified and a value could be
    parsed out of it, then the candidate high boundary is that value. Otherwise, the candidate high
    boundary is the same as the maximum value.</p>

    <p>If the candidate high boundary is less than the low boundary, then the high boundary is the
    low boundary.</p>

    <p>Otherwise, if the candidate high boundary is greater than the maximum value, then the high
    boundary is the maximum value.</p>

    <p>Otherwise, the high boundary is the candidate high boundary.</p>
   </dd>

   <dt>The <dfn data-x="concept-meter-optimum">optimum point</dfn></dt>

   <dd>
    <p>If the <code data-x="attr-meter-optimum">optimum</code> attribute is specified and a value
    could be parsed out of it, then the candidate optimum point is that value. Otherwise, the
    candidate optimum point is the midpoint between the minimum value and the maximum value.</p>

    <p>If the candidate optimum point is less than the minimum value, then the optimum point is the
    minimum value.</p>

    <p>Otherwise, if the candidate optimum point is greater than the maximum value, then the optimum
    point is the maximum value.</p>

    <p>Otherwise, the optimum point is the candidate optimum point.</p>
   </dd>
  </dl>

  <p>All of which will result in the following inequalities all being true:</p>

  <ul>
   <li><p>minimum value &le; actual value &le; maximum value</p></li>

   <li><p>minimum value &le; low boundary &le; high boundary &le; maximum value</p></li>

   <li><p>minimum value &le; optimum point &le; maximum value</p></li>
  </ul>

  <p><strong>UA requirements for regions of the gauge</strong>: If the optimum point is equal to the
  low boundary or the high boundary, or anywhere in between them, then the region between the low
  and high boundaries of the gauge must be treated as the optimum region, and the low and high
  parts, if any, must be treated as suboptimal. Otherwise, if the optimum point is less than the low
  boundary, then the region between the minimum value and the low boundary must be treated as the
  optimum region, the region from the low boundary up to the high boundary must be treated as a
  suboptimal region, and the remaining region must be treated as an even less good region. Finally,
  if the optimum point is higher than the high boundary, then the situation is reversed; the region
  between the high boundary and the maximum value must be treated as the optimum region, the region
  from the high boundary down to the low boundary must be treated as a suboptimal region, and the
  remaining region must be treated as an even less good region.</p>

  <p><strong>UA requirements for showing the gauge</strong>: When representing a <code>meter</code>
  element to the user, the UA should indicate the relative position of the actual value to the
  minimum and maximum values, and the relationship between the actual value and the three regions of
  the gauge.</p>

  </div>

  <div class="example">
   <p>The following markup:</p>
   <pre><code class="html">&lt;h3>Suggested groups&lt;/h3>
&lt;menu>
 &lt;li>&lt;a href="?cmd=hsg" onclick="hideSuggestedGroups()">Hide suggested groups&lt;/a>&lt;/li>
&lt;/menu>
&lt;ul>
 &lt;li>
  &lt;p>&lt;a href="/group/comp.infosystems.www.authoring.stylesheets/view">comp.infosystems.www.authoring.stylesheets&lt;/a> -
     &lt;a href="/group/comp.infosystems.www.authoring.stylesheets/subscribe">join&lt;/a>&lt;/p>
  &lt;p>Group description: &lt;strong>Layout/presentation on the WWW.&lt;/strong>&lt;/p>
  &lt;p><strong>&lt;meter value="0.5">Moderate activity,&lt;/meter></strong> Usenet, 618 subscribers&lt;/p>
 &lt;/li>
 &lt;li>
  &lt;p>&lt;a href="/group/netscape.public.mozilla.xpinstall/view">netscape.public.mozilla.xpinstall&lt;/a> -
     &lt;a href="/group/netscape.public.mozilla.xpinstall/subscribe">join&lt;/a>&lt;/p>
  &lt;p>Group description: &lt;strong>Mozilla XPInstall discussion.&lt;/strong>&lt;/p>
  &lt;p><strong>&lt;meter value="0.25">Low activity,&lt;/meter></strong> Usenet, 22 subscribers&lt;/p>
 &lt;/li>
 &lt;li>
  &lt;p>&lt;a href="/group/mozilla.dev.general/view">mozilla.dev.general&lt;/a> -
     &lt;a href="/group/mozilla.dev.general/subscribe">join&lt;/a>&lt;/p>
  &lt;p><strong>&lt;meter value="0.25">Low activity,&lt;/meter></strong> Usenet, 66 subscribers&lt;/p>
 &lt;/li>
&lt;/ul></code></pre>
   <p>Might be rendered as follows:</p>
   <p><img src="/images/sample-meter.png" width="332" height="178" alt="With the &lt;meter> elements rendered as inline green bars of varying lengths."></p>
  </div>

  <p>User agents <span w-nodev>may</span> combine the value of the <code
  data-x="attr-title">title</code> attribute and the other attributes to provide context-sensitive
  help or inline text detailing the actual values.</p>

  <div class="example">
   <p>For example, the following snippet:</p>
   <pre><code class="html">&lt;meter min=0 max=60 value=23.2 title=seconds>&lt;/meter></code></pre>
   <p>...might cause the user agent to display a gauge with a tooltip
   saying "Value: 23.2 out of 60." on one line and "seconds" on a
   second line.</p>
  </div>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLMeterElement"><code data-x="dom-meter-value">value</code></dfn> IDL
  attribute, on getting, must return the <span data-x="concept-meter-actual">actual value</span>. On
  setting, the given value must be converted to the <span>best representation of the number as a
  floating-point number</span> and then the <code data-x="attr-meter-value">value</code> content
  attribute must be set to that string.</p>

  <p>The <dfn attribute for="HTMLMeterElement"><code data-x="dom-meter-min">min</code></dfn> IDL
  attribute, on getting, must return the <span data-x="concept-meter-minimum">minimum value</span>.
  On setting, the given value must be converted to the <span>best representation of the number as a
  floating-point number</span> and then the <code data-x="attr-meter-min">min</code> content
  attribute must be set to that string.</p>

  <p>The <dfn attribute for="HTMLMeterElement"><code data-x="dom-meter-max">max</code></dfn> IDL
  attribute, on getting, must return the <span data-x="concept-meter-maximum">maximum value</span>.
  On setting, the given value must be converted to the <span>best representation of the number as a
  floating-point number</span> and then the <code data-x="attr-meter-max">max</code> content
  attribute must be set to that string.</p>

  <p>The <dfn attribute for="HTMLMeterElement"><code data-x="dom-meter-low">low</code></dfn> IDL
  attribute, on getting, must return the <span data-x="concept-meter-low">low boundary</span>. On
  setting, the given value must be converted to the <span>best representation of the number as a
  floating-point number</span> and then the <code data-x="attr-meter-low">low</code> content
  attribute must be set to that string.</p>

  <p>The <dfn attribute for="HTMLMeterElement"><code data-x="dom-meter-high">high</code></dfn> IDL
  attribute, on getting, must return the <span data-x="concept-meter-high">high boundary</span>. On
  setting, the given value must be converted to the <span>best representation of the number as a
  floating-point number</span> and then the <code data-x="attr-meter-high">high</code> content
  attribute must be set to that string.</p>

  <p>The <dfn attribute for="HTMLMeterElement"><code data-x="dom-meter-optimum">optimum</code></dfn>
  IDL attribute, on getting, must return the <span data-x="concept-meter-optimum">optimum
  value</span>. On setting, the given value must be converted to the <span>best representation of
  the number as a floating-point number</span> and then the <code
  data-x="attr-meter-optimum">optimum</code> content attribute must be set to that string.</p>

  <p>The <code data-x="dom-lfe-labels">labels</code> IDL attribute provides a list of the element's
  <code>label</code>s.</p>

  </div>

  <div class="example">

   <p>The following example shows how a gauge could fall back to localized or pretty-printed
   text.</p>

   <pre><code class="html">&lt;p>Disk usage: &lt;meter min=0 value=170261928 max=233257824>170&#x2009;261&#x2009;928 bytes used
out of 233&#x2009;257&#x2009;824 bytes available&lt;/meter>&lt;/p></code></pre>

  </div>




  <h4>The <dfn element><code>fieldset</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span data-x="category-listed">Listed</span> and <span data-x="category-autocapitalize-and-autocorrect">autocapitalize-and-autocorrect inheriting</span> <span>form-associated element</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>Optionally a <code>legend</code> element, followed by <span>flow content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-fieldset-disabled">disabled</code></dd>
   <dd><code data-x="attr-fae-form">form</code></dd>
   <dd><code data-x="attr-fe-name">name</code> <!-- variant --> </dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-fieldset">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-fieldset">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLFieldSetElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute boolean <span data-x="dom-fieldset-disabled">disabled</span>;
  readonly attribute <span>HTMLFormElement</span>? <span data-x="dom-fae-form">form</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-fe-name">name</span>;

  readonly attribute DOMString <span data-x="dom-fieldset-type">type</span>;

  [SameObject] readonly attribute <span>HTMLCollection</span> <span data-x="dom-fieldset-elements">elements</span>;

  readonly attribute boolean <span data-x="dom-cva-willValidate">willValidate</span>;
  [SameObject] readonly attribute <span>ValidityState</span> <span data-x="dom-cva-validity">validity</span>;
  readonly attribute DOMString <span data-x="dom-cva-validationMessage">validationMessage</span>;
  boolean <span data-x="dom-cva-checkValidity">checkValidity</span>();
  boolean <span data-x="dom-cva-reportValidity">reportValidity</span>();
  undefined <span data-x="dom-cva-setCustomValidity">setCustomValidity</span>(DOMString error);
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLFieldSetElement</code>.</dd>
  </dl>

  <p>The <code>fieldset</code> element <span>represents</span> a set of form controls (or other
  content) grouped together, optionally with a caption. The caption is given by the first
  <code>legend</code> element that is a child of the <code>fieldset</code> element, if any. The
  remainder of the descendants form the group.</p>

  <p>The <dfn element-attr for="fieldset"><code
  data-x="attr-fieldset-disabled">disabled</code></dfn> attribute, when specified, causes all the
  form control descendants of the <code>fieldset</code> element, excluding those that are
  descendants of the <code>fieldset</code> element's first <code>legend</code> element child, if
  any, to be <span data-x="concept-fe-disabled">disabled</span>.</p>

  <p>A <code>fieldset</code> element is a <dfn data-x="concept-fieldset-disabled">disabled
  fieldset</dfn> if it matches any of the following conditions:</p>

  <ul>
   <li>Its <code data-x="attr-fieldset-disabled">disabled</code> attribute is specified

   <li>It is a descendant of another <code>fieldset</code> element whose <code
   data-x="attr-fieldset-disabled">disabled</code> attribute is specified, and is <em>not</em> a
   descendant of that <code>fieldset</code> element's first <code>legend</code> element child, if
   any.</li>
  </ul>

  <p>The <code data-x="attr-fae-form">form</code> attribute is used to explicitly associate the
  <code>fieldset</code> element with its <span>form owner</span>. The <code
  data-x="attr-fe-name">name</code> attribute represents the element's name.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>fieldset</var>.<span subdfn data-x="dom-fieldset-type">type</span></code></dt>

   <dd><p>Returns the string "fieldset".</p></dd>

   <dt><code data-x=""><var>fieldset</var>.<span subdfn data-x="dom-fieldset-elements">elements</span></code></dt>

   <dd><p>Returns an <code>HTMLCollection</code> of the form controls in the element.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLFieldSetElement"><code
  data-x="dom-fieldset-disabled">disabled</code></dfn> IDL attribute must <span>reflect</span> the
  content attribute of the same name.</p>

  <p>The <dfn attribute for="HTMLFieldSetElement"><code data-x="dom-fieldset-type">type</code></dfn>
  IDL attribute must return the string "<code data-x="">fieldset</code>".</p>

  <p>The <dfn attribute for="HTMLFieldSetElement"><code
  data-x="dom-fieldset-elements">elements</code></dfn> IDL attribute must return an
  <code>HTMLCollection</code> rooted at the <code>fieldset</code> element, whose filter matches
  <span data-x="category-listed">listed elements</span>.</p>

  <p>The <code data-x="dom-cva-willValidate">willValidate</code>, <code
  data-x="dom-cva-validity">validity</code>, and <code
  data-x="dom-cva-validationMessage">validationMessage</code> attributes, and the <code
  data-x="dom-cva-checkValidity">checkValidity()</code>, <code
  data-x="dom-cva-reportValidity">reportValidity()</code>, and <code
  data-x="dom-cva-setCustomValidity">setCustomValidity()</code> methods, are part of the
  <span>constraint validation API</span>. The <code data-x="dom-fae-form">form</code> and <code
  data-x="dom-fe-name">name</code> IDL attributes are part of the element's forms API.</p>

  </div>

  <div class="example">

   <p>This example shows a <code>fieldset</code> element being used to group a set of related
   controls:</p>

   <pre><code class="html">&lt;fieldset>
 &lt;legend>Display&lt;/legend>
 &lt;p>&lt;label>&lt;input type=radio name=c value=0 checked> Black on White&lt;/label>
 &lt;p>&lt;label>&lt;input type=radio name=c value=1> White on Black&lt;/label>
 &lt;p>&lt;label>&lt;input type=checkbox name=g> Use grayscale&lt;/label>
 &lt;p>&lt;label>Enhance contrast &lt;input type=range name=e list=contrast min=0 max=100 value=0 step=1>&lt;/label>
 &lt;datalist id=contrast>
  &lt;option label=Normal value=0>
  &lt;option label=Maximum value=100>
 &lt;/datalist>
&lt;/fieldset></code></pre>

  </div>

  <div class="example">

   <p>The following snippet shows a fieldset with a checkbox in the legend that controls whether or
   not the fieldset is enabled. The contents of the fieldset consist of two required text controls
   and an optional year/month control.</p>

   <pre><code class="html">&lt;fieldset name="clubfields" disabled>
 &lt;legend> &lt;label>
  &lt;input type=checkbox name=club onchange="form.clubfields.disabled = !checked">
  Use Club Card
 &lt;/label> &lt;/legend>
 &lt;p>&lt;label>Name on card: &lt;input name=clubname required>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Card number: &lt;input name=clubnum required pattern="[-0-9]+">&lt;/label>&lt;/p>
 &lt;p>&lt;label>Expiry date: &lt;input name=clubexp type=month>&lt;/label>&lt;/p>
&lt;/fieldset></code></pre>

  </div>

  <div class="example">

   <p>You can also nest <code>fieldset</code> elements. Here is an example expanding on the previous
   one that does so:</p>

   <pre><code class="html">&lt;fieldset name="clubfields" disabled>
 &lt;legend> &lt;label>
  &lt;input type=checkbox name=club onchange="form.clubfields.disabled = !checked">
  Use Club Card
 &lt;/label> &lt;/legend>
 &lt;p>&lt;label>Name on card: &lt;input name=clubname required>&lt;/label>&lt;/p>
 &lt;fieldset name="numfields">
  &lt;legend> &lt;label>
   &lt;input type=radio checked name=clubtype onchange="form.numfields.disabled = !checked">
   My card has numbers on it
  &lt;/label> &lt;/legend>
  &lt;p>&lt;label>Card number: &lt;input name=clubnum required pattern="[-0-9]+">&lt;/label>&lt;/p>
 &lt;/fieldset>
 &lt;fieldset name="letfields" disabled>
  &lt;legend> &lt;label>
   &lt;input type=radio name=clubtype onchange="form.letfields.disabled = !checked">
   My card has letters on it
  &lt;/label> &lt;/legend>
  &lt;p>&lt;label>Card code: &lt;input name=clublet required pattern="[A-Za-z]+">&lt;/label>&lt;/p>
 &lt;/fieldset>
&lt;/fieldset></code></pre>

   <p>In this example, if the outer "Use Club Card" checkbox is not checked, everything inside the
   outer <code>fieldset</code>, including the two radio buttons in the legends of the two nested
   <code>fieldset</code>s, will be disabled. However, if the checkbox is checked, then the radio
   buttons will both be enabled and will let you select which of the two inner
   <code>fieldset</code>s is to be enabled.</p>

  </div>

  <div class="example">

   <p>This example shows a grouping of controls where the <code>legend</code> element both labels
   the grouping, and the nested heading element surfaces the grouping in the document outline:</p>

   <pre><code class="html">&lt;fieldset>
 &lt;legend> &lt;h2>
  How can we best reach you?
 &lt;/h2> &lt;/legend>
 &lt;p> &lt;label>
 &lt;input type=radio checked name=contact_pref>
  Phone
 &lt;/label> &lt;/p>
 &lt;p> &lt;label>
  &lt;input type=radio name=contact_pref>
  Text
 &lt;/label> &lt;/p>
 &lt;p> &lt;label>
  &lt;input type=radio name=contact_pref>
  Email
 &lt;/label> &lt;/p>
&lt;/fieldset></code></pre>

  </div>


  <h4>The <dfn element><code>legend</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As the <span>first child</span> of a <code>fieldset</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>, optionally intermixed with <span>heading content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-legend">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-legend">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLLegendElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  readonly attribute <span>HTMLFormElement</span>? <span data-x="dom-legend-form">form</span>;

  // <a href="#HTMLLegendElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLLegendElement</code>.</dd>
  </dl>

  <p>The <code>legend</code> element <span>represents</span> a caption for the rest of the contents
  of the <code>legend</code> element's parent <code>fieldset</code> element<span w-nodev>, if
  any</span>.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>legend</var>.<span subdfn data-x="dom-legend-form">form</span></code></dt>

   <dd><p>Returns the element's <code>form</code> element, if any, or null otherwise.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLLegendElement"><code data-x="dom-legend-form">form</code></dfn> IDL
  attribute's behavior depends on whether the <code>legend</code> element is in a
  <code>fieldset</code> element or not. If the <code>legend</code> has a <code>fieldset</code>
  element as its parent, then the <code data-x="dom-legend-form">form</code> IDL attribute must
  return the same value as the <code data-x="dom-fae-form">form</code> IDL attribute on that
  <code>fieldset</code> element. Otherwise, it must return null.</p>

  </div>



  <h4 split-filename="form-control-infrastructure">Form control infrastructure</h4>

  <h5>A form control's value</h5>

  <p>Most form controls have a <dfn data-x="concept-fe-value">value</dfn> and a <dfn
  data-x="concept-fe-checked">checkedness</dfn>. (The latter is only used by <code>input</code>
  elements.) These are used to describe how the user interacts with the control.</p>

  <p>A control's <span data-x="concept-fe-value">value</span> is its internal state. As such, it
  might not match the user's current input.</p>

  <p class="example">For instance, if a user enters the word "<kbd>three</kbd>" into <span
  data-x="attr-input-type-number">a numeric field</span> that expects digits, the user's input would
  be the string "three" but the control's <span data-x="concept-fe-value">value</span> would remain
  unchanged. Or, if a user enters the email address "<kbd>&nbsp;&nbsp;awesome@example.com</kbd>"
  (with leading whitespace) into <span data-x="attr-input-type-email">an email field</span>, the
  user's input would be the string "&nbsp;&nbsp;awesome@example.com" but the browser's UI for
  email fields might translate that into a <span data-x="concept-fe-value">value</span> of "<code
  data-x="">awesome@example.com</code>" (without the leading whitespace).</p>

  <p id="concept-input-value-dirty-flag"><span id="concept-textarea-dirty"></span><code>input</code>
  and <code>textarea</code> elements have a <dfn data-x="concept-fe-dirty">dirty value flag</dfn>.
  This is used to track the interaction between the <span data-x="concept-fe-value">value</span> and
  default value. If it is false, <span data-x="concept-fe-value">value</span> mirrors the default
  value. If it is true, the default value is ignored.</p>

  <p><code>input</code>, <code>textarea</code>, and <code>select</code> elements have a
  <dfn>user validity</dfn> boolean. It is initially set to false.</p>

  <p>To define the behavior of constraint validation in the face of the <code>input</code>
  element's <code data-x="attr-input-multiple">multiple</code> attribute, <code>input</code> elements
  can also have separately defined <dfn data-x="concept-fe-values">value<em>s</em></dfn>.</p>

  <p>To define the behavior of the <code data-x="attr-fe-maxlength">maxlength</code> and <code
  data-x="attr-fe-minlength">minlength</code> attributes, as well as other APIs specific to the
  <code>textarea</code> element, all form control with a <span
  data-x="concept-fe-value">value</span> also have an algorithm for obtaining an <span
  id="concept-textarea-api-value"></span><dfn data-x="concept-fe-api-value">API value</dfn>. By
  default this algorithm is to simply return the control's <span
  data-x="concept-fe-value">value</span>.</p>

  <p>The <code>select</code> element does not have a <span data-x="concept-fe-value">value</span>;
  the <span data-x="concept-option-selectedness">selectedness</span> of its <code>option</code>
  elements is what is used instead.</p>


  <h5>Mutability</h5>

  <p>A form control can be designated as <dfn data-x="concept-fe-mutable"><i>mutable</i></dfn>.</p>

  <p class="note">This determines (by means of definitions and requirements in this specification
  that rely on whether an element is so designated) whether or not the user can modify the <span
  data-x="concept-fe-value">value</span> or <span data-x="concept-fe-checked">checkedness</span> of a
  form control, or whether or not a control can be automatically prefilled.</p>
  <!-- or the concept-option-selectedness, or the concept-fe-values... -->


  <h5>Association of controls and forms</h5>

  <p>A <span>form-associated element</span> can have a relationship with a <code>form</code>
  element, which is called the element's <dfn export
  for="button,fieldset,input,object,output,select,textarea,img,form-associated custom element">form
  owner</dfn>. If a <span>form-associated element</span> is not associated with a <code>form</code>
  element, its <span>form owner</span> is said to be null.</p>

  <p>A <span>form-associated element</span> has an associated <dfn>parser inserted flag</dfn>.</p>

  <p>A <span>form-associated element</span> is, by default, associated with its <span
  w-nodev>nearest</span> ancestor <code>form</code> element<span w-nodev> (as described
  below)</span>, but, if it is <span data-x="category-listed">listed</span>, may have a <dfn
  element-attr for="button,fieldset,input,object,output,select,textarea"><code
  data-x="attr-fae-form">form</code></dfn> attribute specified to override this.</p>

  <p class="note">This feature allows authors to work around the lack of support for nested
  <code>form</code> elements.</p>

  <p>If a <span data-x="category-listed">listed</span> <span>form-associated element</span> has a
  <code data-x="attr-fae-form">form</code> attribute specified, then that attribute's value must be
  the <span data-x="concept-id">ID</span> of a <code>form</code> element in the element's
  <span>tree</span>.</p>

  <div w-nodev>

  <p class="note">The rules in this section are complicated by the fact that although conforming
  documents or <span data-x="tree">trees</span> will never contain nested <code>form</code>
  elements, it is quite possible (e.g., using a script that performs DOM manipulation) to generate
  <span data-x="tree">trees</span> that have such nested elements. They are also complicated by
  rules in the HTML parser that, for historical reasons, can result in a <span>form-associated
  element</span> being associated with a <code>form</code> element that is not its ancestor.</p>

  <p>When a <span>form-associated element</span> is created, its <span>form owner</span> must be
  initialized to null (no owner).</p>

  <p>When a <span>form-associated element</span> is to be <dfn
  data-x="concept-form-association">associated</dfn> with a form, its <span>form owner</span> must
  be set to that form.</p>

  <p>When a <span data-x="category-listed">listed</span> <span>form-associated element</span>'s
  <code data-x="attr-fae-form">form</code> attribute is set, changed, or removed, then the user
  agent must <span>reset the form owner</span> of that element.</p>

  <p>When a <span data-x="category-listed">listed</span> <span>form-associated element</span> has a
  <code data-x="attr-fae-form">form</code> attribute and the <span data-x="concept-id">ID</span> of
  any of the elements in the <span>tree</span> changes, then the user agent must <span>reset the
  form owner</span> of that <span>form-associated element</span>.</p>

  <p>When a <span data-x="category-listed">listed</span> <span>form-associated element</span> has a
  <code data-x="attr-fae-form">form</code> attribute and an element with an <span
  data-x="concept-id">ID</span> is <span data-x="node is inserted into a document">inserted
  into</span> or <span data-x="node is removed from a document">removed from</span> the
  <code>Document</code>, or its <span>HTML element moving steps</span> are run, then the user agent
  must <span>reset the form owner</span> of that <span>form-associated element</span>.</p>

  <p class="note">The form owner is also reset by the <span>HTML element insertion steps</span>,
  <span>HTML element removing steps</span>, and <span>HTML element moving steps</span>.</p>

  <p>To <dfn>reset the form owner</dfn> of a <span>form-associated element</span>
  <var>element</var>:</p>

  <ol>
   <li><p>Unset <var>element</var>'s <span>parser inserted flag</span>.</p></li>

   <li>
    <p>If all of the following are true:</p>

    <ul>
     <li><p><var>element</var>'s <span>form owner</span> is not null;</p></li>

     <li><p><var>element</var> is not <span data-x="category-listed">listed</span> or its <code
     data-x="attr-fae-form">form</code> content attribute is not present; and</p></li>

     <li><p><var>element</var>'s <span>form owner</span> is its nearest <code>form</code> element
     ancestor after the change to the ancestor chain,</p></li>
    </ul>

    <p>then return.</p>
   </li>

   <li><p>Set <var>element</var>'s <span>form owner</span> to null.</p></li>

   <li>
    <p>If <var>element</var> is <span data-x="category-listed">listed</span>, has a <code
    data-x="attr-fae-form">form</code> content attribute, and is <span>connected</span>, then:</p>

    <ol>
     <li><p>If the first element in <var>element</var>'s <span>tree</span>, in <span>tree
     order</span>, to have an <span data-x="concept-id">ID</span> that is <span>identical to</span>
     <var>element</var>'s <code data-x="attr-fae-form">form</code> content attribute's value, is a
     <code>form</code> element, then <span data-x="concept-form-association">associate</span> the
     <var>element</var> with that <code>form</code> element.</p></li>
     <!-- note that this ignores the name="" attribute and is unaffected by quirks mode (it's always
          case sensitive) -->
    </ol>
   </li>

   <li><p>Otherwise, if <var>element</var> has an ancestor <code>form</code> element, then <span
   data-x="concept-form-association">associate</span> <var>element</var> with the nearest such
   ancestor <code>form</code> element.</p></li>
  </ol>

  <div class="example">

   <p>In the following non-conforming snippet</p>

   <pre class="bad"><code class="html">...
 &lt;form id="a">
  &lt;div id="b">&lt;/div>
 &lt;/form>
 &lt;script>
  document.getElementById('b').innerHTML =
     '&lt;table>&lt;tr>&lt;td>&lt;/form>&lt;form id="c">&lt;input id="d">&lt;/table>' +
     '&lt;input id="e">';
 &lt;/script>
...</code></pre>

   <p>the <span>form owner</span> of "d" would be the inner nested form "c", while the <span>form
   owner</span> of "e" would be the outer form "a".</p>

   <p>This happens as follows: First, the "e" node gets associated with "c" in the <span>HTML
   parser</span>. Then, the <code data-x="dom-element-innerHTML">innerHTML</code> algorithm moves the nodes
   from the temporary document to the "b" element. At this point, the nodes see their ancestor chain
   change, and thus all the "magic" associations done by the parser are reset to normal ancestor
   associations.</p>

   <p>This example is a non-conforming document, though, as it is a violation of the content models
   to nest <code>form</code> elements, and there is a <span>parse error</span> for the <code
   data-x="">&lt;/form></code> tag.</p>

  </div>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-fae-form">form</span></code></dt>

   <dd>
    <p>Returns the element's <span>form owner</span>.</p>

    <p>Returns null if there isn't one.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p><span data-x="category-listed">Listed</span> <span data-x="form-associated
  element">form-associated elements</span> except for <span data-x="form-associated custom
  element">form-associated custom elements</span> have a <dfn attribute
  for="HTMLObjectElement,HTMLInputElement,HTMLButtonElement,HTMLSelectElement,HTMLTextAreaElement,HTMLOutputElement,HTMLFieldSetElement"><code
  data-x="dom-fae-form">form</code></dfn> IDL attribute, which, on getting, must return the
  element's <span>form owner</span>, or null if there isn't one.</p>

  <p><span data-x="form-associated custom element">Form-associated custom elements</span> don't have
  <code data-x="dom-fae-form">form</code> IDL attribute. Instead, their
  <code>ElementInternals</code> object has a <dfn attribute for="ElementInternals"><code
  data-x="dom-ElementInternals-form">form</code></dfn> IDL attribute. On getting, it must throw a
  <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code> if the <span
  data-x="internals-target">target element</span> is not a <span>form-associated custom
  element</span>. Otherwise, it must return the element's <span>form owner</span>, or null if there
  isn't one.</p>

  </div>



  <h4>Attributes common to form controls</h4>

  <h5>Naming form controls: the <code data-x="attr-fe-name">name</code> attribute</h5>

  <p>The <dfn element-attr for="button,fieldset,input,output,select,textarea"><code
  data-x="attr-fe-name">name</code></dfn> content attribute gives the name of the form control, as
  used in <span>form submission</span> and in the <code>form</code> element's <code
  data-x="dom-form-elements">elements</code> object. If the attribute is specified, its value must
  not be the empty string or <code data-x="">isindex</code>.</p>

  <p class="note">A number of user agents historically implemented special support for first-in-form
  text controls with the name <code data-x="">isindex</code>, and this specification previously
  defined related user agent requirements for it. However, some user agents subsequently dropped
  that special support, and the related requirements were removed from this specification. So, to
  avoid problematic reinterpretations in legacy user agents, the name <code data-x="">isindex</code>
  is no longer allowed.</p>

  <p>Other than <code data-x="">isindex</code>, any non-empty value for <code
  data-x="attr-form-name">name</code> is allowed. An <span>ASCII case-insensitive</span> match for
  the name <dfn><code data-x="attr-fe-name-charset">_charset_</code></dfn> is special: if used as
  the name of a <span data-x="attr-input-type-hidden">Hidden</span> control with no <code
  data-x="attr-input-value">value</code> attribute, then during submission the <code
  data-x="attr-input-value">value</code> attribute is automatically given a value consisting of the
  submission character encoding.</p>

  <div w-nodev>

  <p>The <dfn attribute
  for="HTMLInputElement,HTMLButtonElement,HTMLSelectElement,HTMLTextAreaElement,HTMLOutputElement,HTMLFieldSetElement"><code
  data-x="dom-fe-name">name</code></dfn> IDL attribute must <span>reflect</span> the <code
  data-x="attr-fe-name">name</code> content attribute.</p>

  </div>

  <div class="note">
   <p>DOM clobbering is a common cause of security issues. Avoid using the names of
   built-in form properties with the <code data-x="attr-fe-name">name</code> content attribute.</p>

   <p>In this example, the <code>input</code> element overrides the built-in <code
   data-x="attr-fs-method">method</code> property:</p>

   <pre><code class="js">let form = document.createElement("form");
let input = document.createElement("input");
form.appendChild(input);

form.method;           // => "get"
input.name = "method"; // DOM clobbering occurs here
form.method === input; // => true</code></pre>

   <p>Since the input name takes precedence over built-in form properties, the JavaScript reference
   <code data-x="">form.method</code> will point to the <code>input</code> element named "method"
   instead of the built-in <code data-x="attr-fs-method">method</code> property.</p>
  </div>


  <h5>Submitting element directionality: the <code data-x="attr-fe-dirname">dirname</code> attribute</h5>

  <p>The <dfn element-attr for="input,textarea"><code data-x="attr-fe-dirname">dirname</code></dfn>
  attribute on a form control element enables the submission of <span>the directionality</span> of
  the element, and gives the name of the control that contains this value during <span>form
  submission</span>. If such an attribute is specified, its value must not be the empty string.</p>

  <div class="example">

   <p>In this example, a form contains a text control and a submission button:</p>

   <pre><code class="html">&lt;form action="addcomment.cgi" method=post>
 &lt;p>&lt;label>Comment: &lt;input type=text name="comment" dirname="comment.dir" required>&lt;/label>&lt;/p>
 &lt;p>&lt;button name="mode" type=submit value="add">Post Comment&lt;/button>&lt;/p>
&lt;/form></code></pre>

   <p>When the user submits the form, the user agent includes three fields, one called "comment",
   one called "comment.dir", and one called "mode"; so if the user types "Hello", the submission
   body might be something like:</p>

   <pre>comment=Hello&amp;<strong>comment.dir=ltr</strong>&amp;mode=add</pre>

   <p>If the user manually switches to a right-to-left writing direction and enters "<span data-x=""
   dir=rtl lang=ar>&#x645;&#x631;&#x62d;&#x628;&#x627;</span>", the submission body might be
   something like:</p>

   <pre>comment=%D9%85%D8%B1%D8%AD%D8%A8%D8%A7&amp;<strong>comment.dir=rtl</strong>&amp;mode=add</pre>

  </div>


  <h5>Limiting user input length: the <code data-x="attr-fe-maxlength">maxlength</code> attribute</h5>

  <p>A <dfn data-lt="maxlength" element-attr for="input,textarea" data-x="attr-fe-maxlength">form
  control <code data-x="">maxlength</code> attribute</dfn>, controlled by the <span
  data-x="concept-fe-dirty">dirty value flag</span>, declares a limit on the number of characters a
  user can input. The number of characters is measured using <span>length</span> and, in the case
  of <code>textarea</code> elements, with all newlines normalized to a single character (as opposed
  to CRLF pairs).</p>

  <p>If an element has its <span data-x="attr-fe-maxlength">form control <code
  data-x="">maxlength</code> attribute</span> specified, the attribute's value must be a <span>valid
  non-negative integer</span>. If the attribute is specified and applying the <span>rules for
  parsing non-negative integers</span> to its value results in a number, then that number is the
  element's <dfn>maximum allowed value length</dfn>. If the attribute is omitted or parsing its
  value results in an error, then there is no <span>maximum allowed value length</span>.</p>

  <div w-nodev>

  <p><strong>Constraint validation</strong>: If an element has a <span>maximum allowed value
  length</span>, its <span data-x="concept-fe-dirty">dirty value flag</span> is true, its <span
  data-x="concept-fe-value">value</span> was last changed by a user edit (as opposed to a change
  made by a script), and the <span>length</span> of the element's <span
  data-x="concept-fe-api-value">API value</span> is greater than the element's <span>maximum allowed
  value length</span>, then the element is <span>suffering from being too long</span>.</p>

  <p>User agents may prevent the user from causing the element's <span
  data-x="concept-fe-api-value">API value</span> to be set to a value whose <span>length</span> is
  greater than the element's <span>maximum allowed value length</span>.</p>

  <p class="note">In the case of <code>textarea</code> elements, the <span
  data-x="concept-fe-api-value">API value</span> and <span data-x="concept-fe-value">value</span>
  differ. In particular, <span data-x="normalize newlines">newline normalization</span> is applied
  before the <span>maximum allowed value length</span> is checked (whereas the <span>textarea
  wrapping transformation</span> is not applied).</p>

  </div>


  <h5>Setting minimum input length requirements: the <code data-x="attr-fe-minlength">minlength</code> attribute</h5>

  <p>A <dfn data-lt="minlength" element-attr for="input,textarea" data-x="attr-fe-minlength">form
  control <code data-x="">minlength</code> attribute</dfn>, controlled by the <span
  data-x="concept-fe-dirty">dirty value flag</span>, declares a lower bound on the number of
  characters a user can input. The "number of characters" is measured using <span>length</span> and,
  in the case of <code>textarea</code> elements, with all newlines normalized to a single character
  (as opposed to CRLF pairs).</p>

  <p class="note">The <code data-x="attr-fe-minlength">minlength</code> attribute does not imply the
  <code data-x="">required</code> attribute. If the form control has no <code
  data-x="">required</code> attribute, then the value can still be omitted; the <code
  data-x="attr-fe-minlength">minlength</code> attribute only kicks in once the user has entered a
  value at all. If the empty string is not allowed, then the <code data-x="">required</code>
  attribute also needs to be set.</p>

  <p>If an element has its <span data-x="attr-fe-minlength">form control <code
  data-x="">minlength</code> attribute</span> specified, the attribute's value must be a <span>valid
  non-negative integer</span>. If the attribute is specified and applying the <span>rules for
  parsing non-negative integers</span> to its value results in a number, then that number is the
  element's <dfn>minimum allowed value length</dfn>. If the attribute is omitted or parsing its
  value results in an error, then there is no <span>minimum allowed value length</span>.</p>

  <p>If an element has both a <span>maximum allowed value length</span> and a <span>minimum allowed
  value length</span>, the <span>minimum allowed value length</span> must be smaller than or equal
  to the <span>maximum allowed value length</span>.</p>

  <div w-nodev>

  <p><strong>Constraint validation</strong>: If an element has a <span>minimum allowed value
  length</span>, its <span data-x="concept-fe-dirty">dirty value flag</span> is true, its <span
  data-x="concept-fe-value">value</span> was last changed by a user edit (as opposed to a change
  made by a script), its <span data-x="concept-fe-value">value</span> is not the empty string, and
  the <span>length</span> of the element's <span data-x="concept-fe-api-value">API value</span> is
  less than the element's <span>minimum allowed value length</span>, then the element is
  <span>suffering from being too short</span>.</p>

  </div>

  <div class="example">

   <p>In this example, there are four text controls. The first is required, and has to be at least 5
   characters long. The other three are optional, but if the user fills one in, the user has to
   enter at least 10 characters.</p>

   <pre><code class="html">&lt;form action="/events/menu.cgi" method="post">
 &lt;p>&lt;label>Name of Event: &lt;input required minlength=5 maxlength=50 name=event>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Describe what you would like for breakfast, if anything:
    &lt;textarea name="breakfast" minlength="10">&lt;/textarea>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Describe what you would like for lunch, if anything:
    &lt;textarea name="lunch" minlength="10">&lt;/textarea>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Describe what you would like for dinner, if anything:
    &lt;textarea name="dinner" minlength="10">&lt;/textarea>&lt;/label>&lt;/p>
 &lt;p>&lt;input type=submit value="Submit Request">&lt;/p>
&lt;/form></code></pre>


  </div>


  <h5>Enabling and disabling form controls: the <code data-x="attr-fe-disabled">disabled</code> attribute</h5>

  <p>The <dfn element-attr for="button,fieldset,input,object,output,select,textarea"><code
  data-x="attr-fe-disabled">disabled</code></dfn> content attribute is a <span>boolean
  attribute</span>.</p>

  <p class="note">The <code data-x="attr-option-disabled">disabled</code> attribute for
  <code>option</code> elements and the <code data-x="attr-optgroup-disabled">disabled</code>
  attribute for <code>optgroup</code> elements are defined separately.</p>

  <p>A form control is <dfn data-x="concept-fe-disabled">disabled</dfn> if any of the following are
  true:</p>

  <ul>
   <li><p>the element is a <code>button</code>, <code>input</code>, <code>select</code>,
   <code>textarea</code>, or <span>form-associated custom element</span>, and the <code
   data-x="attr-fe-disabled">disabled</code> attribute is specified on this element (regardless of
   its value); or</p></li>

   <li><p>the element is a descendant of a <code>fieldset</code> element whose <code
   data-x="attr-fieldset-disabled">disabled</code> attribute is specified, and is <em>not</em> a
   descendant of that <code>fieldset</code> element's first <code>legend</code> element child, if
   any.</p></li>
  </ul>

  <div w-nodev>

  <p>A form control that is <span data-x="concept-fe-disabled">disabled</span> must prevent any <code
  data-x="event-click">click</code> events that are <span data-x="queue a task">queued</span> on the
  <span>user interaction task source</span> from being dispatched on the element.</p>

  <p><strong>Constraint validation</strong>: If an element is <span
  data-x="concept-fe-disabled">disabled</span>, it is <span>barred from constraint
  validation</span>.</p>

  <p>The <dfn attribute
  for="HTMLInputElement,HTMLButtonElement,HTMLSelectElement,HTMLTextAreaElement"><code
  data-x="dom-fe-disabled">disabled</code></dfn> IDL attribute must <span>reflect</span> the <code
  data-x="attr-fe-disabled">disabled</code> content attribute.</p>

  </div>


  <h5>Form submission attributes</h5>

  <p><dfn>Attributes for form submission</dfn> can be specified both on <code>form</code> elements
  and on <span data-x="concept-submit-button">submit buttons</span> (elements that represent buttons
  that submit forms, e.g. an <code>input</code> element whose <code
  data-x="attr-input-type">type</code> attribute is in the <span
  data-x="attr-input-type-submit">Submit Button</span> state).

  <p>The <span>attributes for form submission</span> that may be specified on <code>form</code>
  elements are <code data-x="attr-fs-action">action</code>, <code
  data-x="attr-fs-enctype">enctype</code>, <code data-x="attr-fs-method">method</code>, <code
  data-x="attr-fs-novalidate">novalidate</code>, and <code data-x="attr-fs-target">target</code>.</p>

  <p>The corresponding <span>attributes for form submission</span> that may be specified on <span
  data-x="concept-submit-button">submit buttons</span> are <code
  data-x="attr-fs-formaction">formaction</code>, <code
  data-x="attr-fs-formenctype">formenctype</code>, <code
  data-x="attr-fs-formmethod">formmethod</code>, <code
  data-x="attr-fs-formnovalidate">formnovalidate</code>, and <code
  data-x="attr-fs-formtarget">formtarget</code>. When omitted, they default to the values given on
  the corresponding attributes on the <code>form</code> element.</p>

  <hr>

  <p>The <dfn element-attr for="form,button"><code data-x="attr-fs-action">action</code></dfn> and
  <dfn element-attr for="form,button"><code data-x="attr-fs-formaction">formaction</code></dfn>
  content attributes, if specified, must have a value that is a <span>valid non-empty URL
  potentially surrounded by spaces</span>.</p>

  <p>The <dfn data-x="concept-fs-action">action</dfn> of an element is the value of the element's
  <code data-x="attr-fs-formaction">formaction</code> attribute, if the element is a <span
  data-x="concept-submit-button">submit button</span> and has such an attribute, or the value of its
  <span>form owner</span>'s <code data-x="attr-fs-action">action</code> attribute, if <em>it</em> has
  one, or else the empty string.</p>

  <hr>

  <p>The <dfn element-attr for="form,button"><code data-x="attr-fs-method">method</code></dfn> and
  <dfn element-attr for="form,button"><code data-x="attr-fs-formmethod">formmethod</code></dfn>
  content attributes are <span data-x="enumerated attribute">enumerated attributes</span> with the
  following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="form/method,button/formmethod,input/formmethod"><code
     data-x="attr-fs-method-GET-keyword">get</code></dfn>
     <td><dfn data-x="attr-fs-method-GET">GET</dfn>
     <td>Indicates the <code>form</code> will use the HTTP GET method.
    <tr>
     <td><dfn attr-value for="form/method,button/formmethod,input/formmethod"><code
     data-x="attr-fs-method-POST-keyword">post</code></dfn>
     <td><dfn data-x="attr-fs-method-POST">POST</dfn>
     <td>Indicates the <code>form</code> will use the HTTP POST method.
    <tr>
     <td><dfn attr-value for="form/method,button/formmethod,input/formmethod"><code
     data-x="attr-fs-method-dialog-keyword">dialog</code></dfn>
     <td><dfn data-x="attr-fs-method-dialog">Dialog</dfn>
     <td>Indicates the <code>form</code> is intended to close the <code>dialog</code> box in which
     the form finds itself, if any, and otherwise not submit.
  </table>

  <p>The <code data-x="attr-fs-method">method</code> attribute's <i data-x="missing value
  default">missing value default</i> and <i data-x="invalid value default">invalid value default</i>
  are both the <span data-x="attr-fs-method-GET">GET</span> state.</p>

  <p>The <code data-x="attr-fs-formmethod">formmethod</code> attribute has no <i data-x="missing
  value default">missing value default</i>, and its <i data-x="invalid value default">invalid value
  default</i> is the <span data-x="attr-fs-method-GET">GET</span> state.</p>

  <p>The <dfn data-x="concept-fs-method">method</dfn> of an element is one of those states. If the
  element is a <span data-x="concept-submit-button">submit button</span> and has a <code
  data-x="attr-fs-formmethod">formmethod</code> attribute, then the element's <span
  data-x="concept-fs-method">method</span> is that attribute's state; otherwise, it is the <span>form
  owner</span>'s <code data-x="attr-fs-method">method</code> attribute's state.</p>

  <div class="example">

   <p>Here the <code data-x="attr-fs-method">method</code> attribute is used to explicitly specify
   the default value, "<code data-x="attr-fs-method-GET-keyword">get</code>", so that the search
   query is submitted in the URL:</p>

   <pre><code class="html">&lt;form method="get" action="/search.cgi">
 &lt;p>&lt;label>Search terms: &lt;input type=search name=q>&lt;/label>&lt;/p>
 &lt;p>&lt;input type=submit>&lt;/p>
&lt;/form></code></pre>

  </div>

  <div class="example">

   <p>On the other hand, here the <code data-x="attr-fs-method">method</code> attribute is used to
   specify the value "<code data-x="attr-fs-method-POST-keyword">post</code>", so that the user's
   message is submitted in the HTTP request's body:</p>

   <pre><code class="html">&lt;form method="post" action="/post-message.cgi">
 &lt;p>&lt;label>Message: &lt;input type=text name=m>&lt;/label>&lt;/p>
 &lt;p>&lt;input type=submit value="Submit message">&lt;/p>
&lt;/form></code></pre>

  </div>

  <div class="example">

   <p>In this example, a <code>form</code> is used with a <code>dialog</code>. The <code
   data-x="attr-fs-method">method</code> attribute's "<code
   data-x="attr-fs-method-dialog-keyword">dialog</code>" keyword is used to have the dialog
   automatically close when the form is submitted.</p>

   <pre lang="en-GB"><code class="html">&lt;dialog id="ship">
 &lt;form method=dialog>
  &lt;p>A ship has arrived in the harbour.&lt;/p>
  &lt;button type=submit value="board">Board the ship&lt;/button>
  &lt;button type=submit value="call">Call to the captain&lt;/button>
 &lt;/form>
&lt;/dialog>
&lt;script>
 var ship = document.getElementById('ship');
 ship.showModal();
 ship.onclose = function (event) {
   if (ship.returnValue == 'board') {
     // ...
   } else {
     // ...
   }
 };
&lt;/script></code></pre>

  </div>

  <hr>

  <p>The <dfn element-attr for="form,button"><code data-x="attr-fs-enctype">enctype</code></dfn> and
  <dfn element-attr for="form,button"><code data-x="attr-fs-formenctype">formenctype</code></dfn>
  content attributes are <span data-x="enumerated attribute">enumerated attributes</span> with the
  following keywords and states:</p>

  <ul>
   <li>The "<dfn attr-value for="form/enctype,button/enctype"><code
   data-x="attr-fs-enctype-urlencoded">application/x-www-form-urlencoded</code></dfn>" keyword and
   corresponding state.</li>
   <li>The "<dfn attr-value for="form/enctype,button/enctype"><code
   data-x="attr-fs-enctype-formdata">multipart/form-data</code></dfn>" keyword and corresponding
   state.</li>
   <li>The "<dfn attr-value for="form/enctype,button/enctype"><code
   data-x="attr-fs-enctype-text">text/plain</code></dfn>" keyword and corresponding state.</li>
  </ul>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <code
  data-x="attr-fs-enctype-urlencoded">application/x-www-form-urlencoded</code> state.</p>

  <p>The <code data-x="attr-fs-formenctype">formenctype</code> attribute has no <i data-x="missing
  value default">missing value default</i>, and its <i data-x="invalid value default">invalid value
  default</i> is the <code
  data-x="attr-fs-enctype-urlencoded">application/x-www-form-urlencoded</code> state.</p>

  <p>The <dfn data-x="concept-fs-enctype">enctype</dfn> of an element is one of those three states.
  If the element is a <span data-x="concept-submit-button">submit button</span> and has a <code
  data-x="attr-fs-formenctype">formenctype</code> attribute, then the element's <span
  data-x="concept-fs-enctype">enctype</span> is that attribute's state; otherwise, it is the
  <span>form owner</span>'s <code data-x="attr-fs-enctype">enctype</code> attribute's state.</p>

  <hr>

  <p>The <dfn element-attr for="form,button"><code data-x="attr-fs-target">target</code></dfn> and
  <dfn element-attr for="form,button"><code data-x="attr-fs-formtarget">formtarget</code></dfn>
  content attributes, if specified, must have values that are <span data-x="valid navigable target
  name or keyword">valid navigable target names or keywords</span>.</p>

  <hr>

  <p>The <dfn element-attr for="form,button"><code
  data-x="attr-fs-novalidate">novalidate</code></dfn> and <dfn element-attr for="form,button"><code
  data-x="attr-fs-formnovalidate">formnovalidate</code></dfn> content attributes are <span
  data-x="boolean attribute">boolean attributes</span>. If present, they indicate that the form is
  not to be validated during submission.</p>

  <p>The <dfn data-x="concept-fs-novalidate">no-validate state</dfn> of an element is true if the
  element is a <span data-x="concept-submit-button">submit button</span> and the element's <code
  data-x="attr-fs-formnovalidate">formnovalidate</code> attribute is present, or if the element's
  <span>form owner</span>'s <code data-x="attr-fs-novalidate">novalidate</code> attribute is present,
  and false otherwise.</p>

  <div class="example">

   <p>This attribute is useful to include "save" buttons on forms that have validation constraints,
   to allow users to save their progress even though they haven't fully entered the data in the
   form. The following example shows a simple form that has two required fields. There are three
   buttons: one to submit the form, which requires both fields to be filled in; one to save the form
   so that the user can come back and fill it in later; and one to cancel the form altogether.</p>

   <pre><code class="html">&lt;form action="editor.cgi" method="post">
 &lt;p>&lt;label>Name: &lt;input required name=fn>&lt;/label>&lt;/p>
 &lt;p>&lt;label>Essay: &lt;textarea required name=essay>&lt;/textarea>&lt;/label>&lt;/p>
 &lt;p>&lt;input type=submit name=submit value="Submit essay">&lt;/p>
 &lt;p>&lt;input type=submit formnovalidate name=save value="Save essay">&lt;/p>
 &lt;p>&lt;input type=submit formnovalidate name=cancel value="Cancel">&lt;/p>
&lt;/form></code></pre>

  </div>

  <div w-nodev>

  <hr>

  <p>The <dfn attribute for="HTMLFormElement"><code data-x="dom-fs-action">action</code></dfn> IDL
  attribute must <span>reflect</span> the content attribute of the same name, except that on
  getting, when the content attribute is missing or its value is the empty string, the element's
  <span>node document</span>'s <span data-x="concept-document-url">URL</span> must be returned
  instead. The <dfn attribute for="HTMLFormElement"><code data-x="dom-fs-target">target</code></dfn>
  IDL attribute must <span>reflect</span> the content attribute of the same name. The <dfn attribute
  for="HTMLFormElement"><code data-x="dom-fs-method">method</code></dfn> and <dfn attribute
  for="HTMLFormElement"><code data-x="dom-fs-enctype">enctype</code></dfn> IDL attributes must
  <span>reflect</span> the respective content attributes of the same name, <span>limited to only
  known values</span>. The <dfn attribute for="HTMLFormElement"><code
  data-x="dom-fs-encoding">encoding</code></dfn> IDL attribute must <span>reflect</span> the <code
  data-x="attr-fs-enctype">enctype</code> content attribute, <span>limited to only known
  values</span>. The <dfn attribute for="HTMLFormElement"><code
  data-x="dom-fs-noValidate">noValidate</code></dfn> IDL attribute must <span>reflect</span> the
  <code data-x="attr-fs-novalidate">novalidate</code> content attribute. The <dfn attribute
  for="HTMLButtonElement,HTMLInputElement"><code data-x="dom-fs-formAction">formAction</code></dfn>
  IDL attribute must <span>reflect</span> the <code data-x="attr-fs-formaction">formaction</code>
  content attribute, except that on getting, when the content attribute is missing or its value is
  the empty string, the element's <span>node document</span>'s <span
  data-x="concept-document-url">URL</span> must be returned instead. The <dfn attribute
  for="HTMLButtonElement,HTMLInputElement"><code
  data-x="dom-fs-formEnctype">formEnctype</code></dfn> IDL attribute must <span>reflect</span> the
  <code data-x="attr-fs-formenctype">formenctype</code> content attribute, <span>limited to only
  known values</span>. The <dfn attribute for="HTMLButtonElement,HTMLInputElement"><code
  data-x="dom-fs-formMethod">formMethod</code></dfn> IDL attribute must <span>reflect</span> the
  <code data-x="attr-fs-formmethod">formmethod</code> content attribute, <span>limited to only known
  values</span>. The <dfn attribute for="HTMLButtonElement,HTMLInputElement"><code
  data-x="dom-fs-formNoValidate">formNoValidate</code></dfn> IDL attribute must <span>reflect</span>
  the <code data-x="attr-fs-formnovalidate">formnovalidate</code> content attribute. The <dfn
  attribute for="HTMLButtonElement,HTMLInputElement"><code data-x="dom-fs-formTarget">formTarget</code></dfn> IDL
  attribute must <span>reflect</span> the <code data-x="attr-fs-formtarget">formtarget</code>
  content attribute.

  </div>


  <h5>Autofill</h5>

  <h6>Autofilling form controls: the <code data-x="attr-fe-autocomplete">autocomplete</code> attribute</h6>

  <p>User agents sometimes have features for helping users fill forms in, for example prefilling the
  user's address based on earlier user input. The <dfn element-attr
  for="button,fieldset,input,object,output,select,textarea"><code
  data-x="attr-fe-autocomplete">autocomplete</code></dfn> content attribute can be used to hint to
  the user agent how to, or indeed whether to, provide such a feature.</p>

  <!-- "mode" is already used by enumerated attributes... "role" is used by role=""... "mantle" it is. -->

  <p>There are two ways this attribute is used. When wearing the <dfn>autofill expectation
  mantle</dfn>, the <code data-x="attr-fe-autocomplete">autocomplete</code> attribute describes what
  input is expected from users. When wearing the <dfn>autofill anchor mantle</dfn>, the <code
  data-x="attr-fe-autocomplete">autocomplete</code> attribute describes the meaning of the given
  value.</p>

  <p>On an <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is
  in the <span data-x="attr-input-type-hidden">Hidden</span> state, the <code
  data-x="attr-fe-autocomplete">autocomplete</code> attribute wears the <span>autofill anchor
  mantle</span>. In all other cases, it wears the <span>autofill expectation mantle</span>.</p>

  <p>When wearing the <span>autofill expectation mantle</span>, the <code
  data-x="attr-fe-autocomplete">autocomplete</code> attribute, if specified, must have a value that
  is an ordered <span>set of space-separated tokens</span> consisting of either a single token that
  is an <span>ASCII case-insensitive</span> match for the string "<code
  data-x="attr-fe-autocomplete-off">off</code>", or a single token that is an <span>ASCII
  case-insensitive</span> match for the string "<code data-x="attr-fe-autocomplete-on">on</code>",
  or <span>autofill detail tokens</span>.</p>

  <p>When wearing the <span>autofill anchor
  mantle</span>, the <code
  data-x="attr-fe-autocomplete">autocomplete</code> attribute, if specified, must have a value that is an ordered <span>set of
  space-separated tokens</span> consisting of just <span>autofill detail tokens</span> (i.e. the
  "<code data-x="attr-fe-autocomplete-on">on</code>" and "<code
  data-x="attr-fe-autocomplete-off">off</code>" keywords are not allowed).</p>

  <p><dfn>Autofill detail tokens</dfn> are the following, in the order given below:</p>

  <ol>
   <li>

    <p>Optionally, a token whose first eight characters are an <span>ASCII case-insensitive</span>
    match for the string "<dfn attr-value
    for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
    data-x="attr-fe-autocomplete-section">section-</code></dfn>", meaning that the field belongs to
    the named group.</p>

    <div class="example">

     <p>For example, if there are two shipping addresses in the form, then they could be marked up
     as:</p>

     <pre><code class="html">&lt;fieldset>
 &lt;legend>Ship the blue gift to...&lt;/legend>
 &lt;p> &lt;label> Address:     &lt;textarea name=ba autocomplete="section-blue shipping street-address">&lt;/textarea> &lt;/label>
 &lt;p> &lt;label> City:        &lt;input name=bc autocomplete="section-blue shipping address-level2"> &lt;/label>
 &lt;p> &lt;label> Postal Code: &lt;input name=bp autocomplete="section-blue shipping postal-code"> &lt;/label>
&lt;/fieldset>
&lt;fieldset>
 &lt;legend>Ship the red gift to...&lt;/legend>
 &lt;p> &lt;label> Address:     &lt;textarea name=ra autocomplete="section-red shipping street-address">&lt;/textarea> &lt;/label>
 &lt;p> &lt;label> City:        &lt;input name=rc autocomplete="section-red shipping address-level2"> &lt;/label>
 &lt;p> &lt;label> Postal Code: &lt;input name=rp autocomplete="section-red shipping postal-code"> &lt;/label>
&lt;/fieldset></code></pre>

    </div>
   </li>

   <li>
    <p>Optionally, a token that is an <span>ASCII case-insensitive</span> match for one of the
    following strings:</p>

    <ul class="brief">

     <li>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-shipping">shipping</code></dfn>", meaning the field is part of the
     shipping address or contact information

     <li>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-billing">billing</code></dfn>", meaning the field is part of the
     billing address or contact information
    </ul>
   </li>

   <li>
    <p>Either of the following two options:</p>

    <ul>
     <li>
      <p>A token that is an <span>ASCII case-insensitive</span> match for one of the following
      <span>autofill field</span> names, excluding those that are <span>inappropriate for the
      control</span>:</p>

      <ul class="brief">
       <li>"<code data-x="attr-fe-autocomplete-name">name</code>"
       <li>"<code data-x="attr-fe-autocomplete-honorific-prefix">honorific-prefix</code>"
       <li>"<code data-x="attr-fe-autocomplete-given-name">given-name</code>"
       <li>"<code data-x="attr-fe-autocomplete-additional-name">additional-name</code>"
       <li>"<code data-x="attr-fe-autocomplete-family-name">family-name</code>"
       <li>"<code data-x="attr-fe-autocomplete-honorific-suffix">honorific-suffix</code>"
       <li>"<code data-x="attr-fe-autocomplete-nickname">nickname</code>"
       <li>"<code data-x="attr-fe-autocomplete-username">username</code>"
       <li>"<code data-x="attr-fe-autocomplete-new-password">new-password</code>"
       <li>"<code data-x="attr-fe-autocomplete-current-password">current-password</code>"
       <li>"<code data-x="attr-fe-autocomplete-one-time-code">one-time-code</code>"
       <li>"<code data-x="attr-fe-autocomplete-organization-title">organization-title</code>"
       <li>"<code data-x="attr-fe-autocomplete-organization">organization</code>"
       <li>"<code data-x="attr-fe-autocomplete-street-address">street-address</code>"
       <li>"<code data-x="attr-fe-autocomplete-address-line1">address-line1</code>"
       <li>"<code data-x="attr-fe-autocomplete-address-line2">address-line2</code>"
       <li>"<code data-x="attr-fe-autocomplete-address-line3">address-line3</code>"
       <li>"<code data-x="attr-fe-autocomplete-address-level4">address-level4</code>"
       <li>"<code data-x="attr-fe-autocomplete-address-level3">address-level3</code>"
       <li>"<code data-x="attr-fe-autocomplete-address-level2">address-level2</code>"
       <li>"<code data-x="attr-fe-autocomplete-address-level1">address-level1</code>"
       <li>"<code data-x="attr-fe-autocomplete-country">country</code>"
       <li>"<code data-x="attr-fe-autocomplete-country-name">country-name</code>"
       <li>"<code data-x="attr-fe-autocomplete-postal-code">postal-code</code>"
       <li>"<code data-x="attr-fe-autocomplete-cc-name">cc-name</code>"
       <li>"<code data-x="attr-fe-autocomplete-cc-given-name">cc-given-name</code>"
       <li>"<code data-x="attr-fe-autocomplete-cc-additional-name">cc-additional-name</code>"
       <li>"<code data-x="attr-fe-autocomplete-cc-family-name">cc-family-name</code>"
       <li>"<code data-x="attr-fe-autocomplete-cc-number">cc-number</code>"
       <li>"<code data-x="attr-fe-autocomplete-cc-exp">cc-exp</code>"
       <li>"<code data-x="attr-fe-autocomplete-cc-exp-month">cc-exp-month</code>"
       <li>"<code data-x="attr-fe-autocomplete-cc-exp-year">cc-exp-year</code>"
       <li>"<code data-x="attr-fe-autocomplete-cc-csc">cc-csc</code>"
       <li>"<code data-x="attr-fe-autocomplete-cc-type">cc-type</code>"
       <li>"<code data-x="attr-fe-autocomplete-transaction-currency">transaction-currency</code>"
       <li>"<code data-x="attr-fe-autocomplete-transaction-amount">transaction-amount</code>"
       <li>"<code data-x="attr-fe-autocomplete-language">language</code>"
       <li>"<code data-x="attr-fe-autocomplete-bday">bday</code>"
       <li>"<code data-x="attr-fe-autocomplete-bday-day">bday-day</code>"
       <li>"<code data-x="attr-fe-autocomplete-bday-month">bday-month</code>"
       <li>"<code data-x="attr-fe-autocomplete-bday-year">bday-year</code>"
       <li>"<code data-x="attr-fe-autocomplete-sex">sex</code>"
       <li>"<code data-x="attr-fe-autocomplete-url">url</code>"
       <li>"<code data-x="attr-fe-autocomplete-photo">photo</code>"
      </ul>

      <p>(See the table below for descriptions of these values.)</p>
     </li>

     <li>
      <p>The following, in the given order:</p>

      <ol>
       <li>
        <p>Optionally, a token that is an <span>ASCII case-insensitive</span> match for one of the
        following strings:</p>

        <ul class="brief">

         <li>"<dfn attr-value
         for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
         data-x="attr-fe-autocomplete-home">home</code></dfn>", meaning the field is for contacting
         someone at their residence

         <li>"<dfn attr-value
         for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
         data-x="attr-fe-autocomplete-work">work</code></dfn>", meaning the field is for contacting
         someone at their workplace

         <li>"<dfn attr-value
         for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
         data-x="attr-fe-autocomplete-mobile">mobile</code></dfn>"<!-- or "cell"?  -->, meaning the
         field is for contacting someone regardless of location

         <li>"<dfn attr-value
         for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
         data-x="attr-fe-autocomplete-fax">fax</code></dfn>", meaning the field describes a fax
         machine's contact details

         <li>"<dfn attr-value
         for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
         data-x="attr-fe-autocomplete-pager">pager</code></dfn>", meaning the field describes a
         pager's or beeper's contact details
        </ul>
       </li>

       <li>
        <p>A token that is an <span>ASCII case-insensitive</span> match for one of the following
        <span>autofill field</span> names, excluding those that are <span>inappropriate for the
        control</span>:</p>

        <ul class="brief">
         <li>"<code data-x="attr-fe-autocomplete-tel">tel</code>"
         <li>"<code data-x="attr-fe-autocomplete-tel-country-code">tel-country-code</code>"
         <li>"<code data-x="attr-fe-autocomplete-tel-national">tel-national</code>"
         <li>"<code data-x="attr-fe-autocomplete-tel-area-code">tel-area-code</code>"
         <li>"<code data-x="attr-fe-autocomplete-tel-local">tel-local</code>"
         <li>"<code data-x="attr-fe-autocomplete-tel-local-prefix">tel-local-prefix</code>"
         <li>"<code data-x="attr-fe-autocomplete-tel-local-suffix">tel-local-suffix</code>"
         <li>"<code data-x="attr-fe-autocomplete-tel-extension">tel-extension</code>"
         <li>"<code data-x="attr-fe-autocomplete-email">email</code>"
         <li>"<code data-x="attr-fe-autocomplete-impp">impp</code>"
        </ul>

        <p>(See the table below for descriptions of these values.)</p>
       </li>
      </ol>
     </li>
    </ul>
   </li>

   <li><p>Optionally, a token that is an <span>ASCII case-insensitive</span> match for the string
   "<dfn data-x="attr-fe-autocomplete-webauthn"><code>webauthn</code></dfn>", meaning the user agent
   should show <span data-x="public key credential">public key credentials</span> available via
   <code data-x="conditional mediation">conditional</code> mediation when the user interacts with the
   form control. <code data-x="attr-fe-autocomplete-webauthn">webauthn</code> is only valid for
   <code>input</code> and <code>textarea</code> elements.</p></li>
  </ol>

  <p>As noted earlier, the meaning of the attribute and its keywords depends on the mantle that the
  attribute is wearing.</p>

  <dl class="switch">

   <dt>When wearing the <span>autofill expectation mantle</span>...

   <dd>
    <p>The "<dfn attr-value
    for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
    data-x="attr-fe-autocomplete-off">off</code></dfn>" keyword indicates either that the control's
    input data is particularly sensitive (for example the activation code for a nuclear weapon); or
    that it is a value that will never be reused (for example a one-time-key for a bank login) and
    the user will therefore have to explicitly enter the data each time, instead of being able to
    rely on the UA to prefill the value for them; or that the document provides its own autocomplete
    mechanism and does not want the user agent to provide autocompletion values.</p>

    <p>The "<dfn attr-value
    for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
    data-x="attr-fe-autocomplete-on">on</code></dfn>" keyword indicates that the user agent is
    allowed to provide the user with autocompletion values, but does not provide any further
    information about what kind of data the user might be expected to enter. User agents would have
    to use heuristics to decide what autocompletion values to suggest.</p>

    <p>The <span>autofill field</span> listed above indicate that the user agent is allowed to
    provide the user with autocompletion values, and specifies what kind of value is expected. The
    meaning of each such keyword is described in the table below.</p>

    <p>If the <code data-x="attr-fe-autocomplete">autocomplete</code> attribute is omitted, the default
    value corresponding to the state of the element's <span>form owner</span>'s <code
    data-x="attr-form-autocomplete">autocomplete</code> attribute is used instead (either "<code
    data-x="attr-fe-autocomplete-on">on</code>" or "<code
    data-x="attr-fe-autocomplete-off">off</code>"). If there is no <span>form owner</span>, then the
    value "<code data-x="attr-fe-autocomplete-on">on</code>" is used.</p>
   </dd>

   <dt>When wearing the <span>autofill anchor mantle</span>...

   <dd>
    <p>The <span>autofill field</span> listed above indicate that the value of the particular kind
    of value specified is that value provided for this element. The meaning of each such keyword is
    described in the table below.</p>

    <div class="example">

     <p>In this example the page has explicitly specified the currency and amount of the
     transaction. The form requests a credit card and other billing details. The user agent could
     use this information to suggest a credit card that it knows has sufficient balance and that
     supports the relevant currency.</p>

     <pre><code class="html">&lt;form method=post action="step2.cgi">
 &lt;input type=hidden autocomplete=transaction-currency value="CHF">
 &lt;input type=hidden autocomplete=transaction-amount value="15.00">
 &lt;p>&lt;label>Credit card number: &lt;input type=text inputmode=numeric autocomplete=cc-number>&lt;/label>
 &lt;p>&lt;label>Expiry Date: &lt;input type=month autocomplete=cc-exp>&lt;/label>
 &lt;p>&lt;input type=submit value="Continue...">
&lt;/form></code></pre>

    </div>
   </dd>
  </dl>

  <p>The <dfn>autofill field</dfn> keywords relate to each other as described in the table below. Each field name
  listed on a row of this table corresponds to the meaning given in the cell for that row in the
  column labeled "Meaning". Some fields correspond to subparts of other fields; for example, a
  credit card expiry date can be expressed as one field giving both the month and year of expiry
  ("<code data-x="attr-fe-autocomplete-cc-exp">cc-exp</code>"), or as two fields, one giving the
  month ("<code data-x="attr-fe-autocomplete-cc-exp-month">cc-exp-month</code>") and one the year
  ("<code data-x="attr-fe-autocomplete-cc-exp-year">cc-exp-year</code>"). In such cases, the names of
  the broader fields cover multiple rows, in which the narrower fields are defined.</p>

  <p class="note">Generally, authors are encouraged to use the broader fields rather than the
  narrower fields, as the narrower fields tend to expose Western biases. For example, while it is
  common in some Western cultures to have a given name and a family name, in that order (and thus
  often referred to as a <i>first name</i> and a <i>surname</i>), many cultures put the family name
  first and the given name second, and many others simply have one name (a <i>mononym</i>). Having a
  single field is therefore more flexible.</p>

  <p>Some fields are only appropriate for certain form controls. An <span>autofill field</span> name
  is <dfn data-x="inappropriate for the control">inappropriate for a control</dfn> if the control
  does not belong to the group listed for that <span>autofill field</span> in the fifth column of
  the first row describing that <span>autofill field</span> in the table below. What controls fall
  into each group is described below the table.</p>

  <table>
   <thead>
    <tr>
     <th colspan=4> Field name
     <th> Meaning
     <th> Canonical Format
     <th> Canonical Format Example
     <th> Control group
   <tbody>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-name">name</code></dfn>"
     <td>Full name
     <td>Free-form text, no newlines
     <td>Sir Timothy John Berners-Lee, OM, KBE, FRS, FREng, FRSA
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td rowspan=5 class="non-rectangular-cell-indentation">
      <td colspan=3>"<dfn attr-value for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code data-x="attr-fe-autocomplete-honorific-prefix">honorific-prefix</code></dfn>"
      <td>Prefix or title (e.g. "Mr.", "Ms.", "Dr.", "<span lang="fr" data-x="">M<sup>lle</sup></span>")
      <td>Free-form text, no newlines
      <td>Sir
     <td><span data-x="Control Group Text">Text</span>
    <tr>
      <td colspan=3>"<dfn attr-value
      for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
      data-x="attr-fe-autocomplete-given-name">given-name</code></dfn>"
      <td>Given name (in some Western cultures, also known as the <i>first name</i>)
      <td>Free-form text, no newlines
      <td>Timothy
     <td><span data-x="Control Group Text">Text</span>
    <tr>
      <td colspan=3>"<dfn attr-value
      for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
      data-x="attr-fe-autocomplete-additional-name">additional-name</code></dfn>"
      <td>Additional names (in some Western cultures, also known as <i>middle names</i>, forenames other than the first name)
      <td>Free-form text, no newlines
      <td>John
     <td><span data-x="Control Group Text">Text</span>
    <tr>
      <td colspan=3>"<dfn attr-value
      for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
      data-x="attr-fe-autocomplete-family-name">family-name</code></dfn>"
      <td>Family name (in some Western cultures, also known as the <i>last name</i> or <i>surname</i>)
      <td>Free-form text, no newlines
      <td>Berners-Lee
     <td><span data-x="Control Group Text">Text</span>
    <tr>
      <td colspan=3>"<dfn attr-value
      for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
      data-x="attr-fe-autocomplete-honorific-suffix">honorific-suffix</code></dfn>"
      <td>Suffix (e.g. "Jr.", "B.Sc.", "MBASW", "II")
      <td>Free-form text, no newlines
      <td>OM, KBE, FRS, FREng, FRSA
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-nickname">nickname</code></dfn>"
     <td>Nickname, screen name, handle: a typically short name used instead of the full name
     <td>Free-form text, no newlines
     <td>Tim
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-organization-title">organization-title</code></dfn>"
     <td>Job title (e.g. "Software Engineer", "Senior Vice President", "Deputy Managing Director")
     <td>Free-form text, no newlines
     <td>Professor
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-username">username</code></dfn>"
     <td>A username
     <td>Free-form text, no newlines
     <td>timbl
     <td><span data-x="Control Group Username">Username</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-new-password">new-password</code></dfn>"
     <td>A new password (e.g. when creating an account or changing a password)
     <td>Free-form text, no newlines
     <td>GUMFXbadyrS3
     <td><span data-x="Control Group Password">Password</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-current-password">current-password</code></dfn>"
     <td>The current password for the account identified by the <code data-x="attr-fe-autocomplete-username">username</code> field (e.g. when logging in)
     <td>Free-form text, no newlines
     <td>qwerty <!-- http://splashdata.com/press/worstpasswords2013.htm -->
     <td><span data-x="Control Group Password">Password</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-one-time-code">one-time-code</code></dfn>"
     <td>One-time code used for verifying user identity
     <td>Free-form text, no newlines
     <td>123456
     <td><span data-x="Control Group Password">Password</span>
   <tbody>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-organization">organization</code></dfn>"
     <td>Company name corresponding to the person, address, or contact information in the other fields associated with this field
     <td>Free-form text, no newlines
     <td>World Wide Web Consortium
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-street-address">street-address</code></dfn>"
     <td>Street address (multiple lines, newlines preserved)
     <td>Free-form text
     <td>32 Vassar Street<br>
MIT Room 32-G524
     <td><span data-x="Control Group Multiline">Multiline</span>
    <tr>
     <td rowspan=3 class="non-rectangular-cell-indentation">
      <td colspan=3>"<dfn attr-value
      for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
      data-x="attr-fe-autocomplete-address-line1">address-line1</code></dfn>"
      <td rowspan=3>Street address (one line per field)
      <td>Free-form text, no newlines
      <td>32 Vassar Street
     <td><span data-x="Control Group Text">Text</span>
    <tr>
      <td colspan=3>"<dfn attr-value
      for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
      data-x="attr-fe-autocomplete-address-line2">address-line2</code></dfn>"
      <td>Free-form text, no newlines
      <td>MIT Room 32-G524
     <td><span data-x="Control Group Text">Text</span>
    <tr>
      <td colspan=3>"<dfn attr-value
      for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
      data-x="attr-fe-autocomplete-address-line3">address-line3</code></dfn>"
      <td>Free-form text, no newlines
      <td>
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-address-level4">address-level4</code></dfn>"
     <td>The most fine-grained <a href="#more-on-address-levels">administrative level</a>, in
     addresses with four administrative levels
     <td>Free-form text, no newlines
     <td>
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-address-level3">address-level3</code></dfn>"
     <td>The <a href="#more-on-address-levels">third administrative level</a>, in addresses with
     three or more administrative levels
     <td>Free-form text, no newlines
     <td>
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-address-level2">address-level2</code></dfn>"
     <td>The <a href="#more-on-address-levels">second administrative level</a>, in addresses with
     two or more administrative levels; in the countries with two administrative levels, this would
     typically be the city, town, village, or other locality within which the relevant street
     address is found
     <td>Free-form text, no newlines
     <td>Cambridge
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-address-level1">address-level1</code></dfn>"
     <td>The broadest <a href="#more-on-address-levels">administrative level</a> in the address,
     i.e. the province within which the locality is found; for example, in the US, this would be the
     state; in Switzerland it would be the canton; in the UK, the post town
     <td>Free-form text, no newlines
     <td>MA
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-country">country</code></dfn>"
     <td>Country code
     <td>Valid <a href="https://www.iso.org/iso-3166-country-codes.html">ISO 3166-1-alpha-2 country code</a> <ref>ISO3166</ref>
     <td>US
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-country-name">country-name</code></dfn>"
     <td>Country name
     <td>Free-form text, no newlines; <a href="#autofill-country">derived from <code data-x="attr-fe-autocomplete-country">country</code> in some cases</a>
     <td>US
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-postal-code">postal-code</code></dfn>"
     <td>Postal code, post code, ZIP code, CEDEX code (if CEDEX, append "CEDEX", and the <i lang="fr">arrondissement</i>, if relevant, to the <code data-x="attr-fe-autocomplete-address-level2">address-level2</code> field)
     <td>Free-form text, no newlines
     <td>02139
     <td><span data-x="Control Group Text">Text</span>
   <tbody>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-cc-name">cc-name</code></dfn>"
     <td>Full name as given on the payment instrument
     <td>Free-form text, no newlines
     <td>Tim Berners-Lee
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td rowspan=3 class="non-rectangular-cell-indentation">
      <td colspan=3>"<dfn attr-value
      for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
      data-x="attr-fe-autocomplete-cc-given-name">cc-given-name</code></dfn>"
      <td>Given name as given on the payment instrument (in some Western cultures, also known as the <i>first name</i>)
      <td>Free-form text, no newlines
      <td>Tim
     <td><span data-x="Control Group Text">Text</span>
    <tr>
      <td colspan=3>"<dfn attr-value
      for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
      data-x="attr-fe-autocomplete-cc-additional-name">cc-additional-name</code></dfn>"
      <td>Additional names given on the payment instrument (in some Western cultures, also known as <i>middle names</i>, forenames other than the first name)
      <td>Free-form text, no newlines
      <td>
     <td><span data-x="Control Group Text">Text</span>
    <tr>
      <td colspan=3>"<dfn attr-value
      for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
      data-x="attr-fe-autocomplete-cc-family-name">cc-family-name</code></dfn>"
      <td>Family name given on the payment instrument (in some Western cultures, also known as the <i>last name</i> or <i>surname</i>)
      <td>Free-form text, no newlines
      <td>Berners-Lee
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-cc-number">cc-number</code></dfn>"
     <td>Code identifying the payment instrument (e.g. the credit card number)
     <td><span>ASCII digits</span>
     <td>4114360123456785 <!-- http://www.auricsystems.com/support-center/sample-credit-card-numbers -->
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-cc-exp">cc-exp</code></dfn>"
     <td>Expiration date of the payment instrument
     <td><span>Valid month string</span>
     <td>2014-12
     <td><span data-x="Control Group Month">Month</span>
    <tr>
     <td rowspan=2 class="non-rectangular-cell-indentation">
      <td colspan=3>"<dfn attr-value
      for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
      data-x="attr-fe-autocomplete-cc-exp-month">cc-exp-month</code></dfn>"
      <td>Month component of the expiration date of the payment instrument
      <td><span>Valid integer</span> in the range 1..12
      <td>12
      <td><span data-x="Control Group Numeric">Numeric</span>
    <tr>
      <td colspan=3>"<dfn attr-value
      for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
      data-x="attr-fe-autocomplete-cc-exp-year">cc-exp-year</code></dfn>"
      <td>Year component of the expiration date of the payment instrument
      <td><span>Valid integer</span> greater than zero
      <td>2014
      <td><span data-x="Control Group Numeric">Numeric</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-cc-csc">cc-csc</code></dfn>"
     <td>Security code for the payment instrument (also known as the card security code (CSC), card validation code (CVC), card verification value (CVV), signature panel code (SPC), credit card ID (CCID), etc.)
     <td><span>ASCII digits</span>
     <td>419
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-cc-type">cc-type</code></dfn>"
     <td>Type of payment instrument
     <td>Free-form text, no newlines
     <td>Visa
     <td><span data-x="Control Group Text">Text</span>
   <tbody>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-transaction-currency">transaction-currency</code></dfn>"
     <td>The currency that the user would prefer the transaction to use
     <td>ISO 4217 currency code <ref>ISO4217</ref>
     <td>GBP
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-transaction-amount">transaction-amount</code></dfn>"
     <td>The amount that the user would like for the transaction (e.g. when entering a bid or sale price)
     <td><span>Valid floating-point number</span>
     <td>401.00
     <td><span data-x="Control Group Numeric">Numeric</span>
   <tbody>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-language">language</code></dfn>"
     <td>Preferred language
     <td>Valid BCP 47 language tag <ref>BCP47</ref>
     <td>en
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-bday">bday</code></dfn>"
     <td>Birthday
     <td><span>Valid date string</span>
     <td>1955-06-08
     <td><span data-x="Control Group Date">Date</span>
    <tr>
     <td rowspan=3 class="non-rectangular-cell-indentation">
      <td colspan=3>"<dfn attr-value
      for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
      data-x="attr-fe-autocomplete-bday-day">bday-day</code></dfn>"
      <td>Day component of birthday
      <td><span>Valid integer</span> in the range 1..31
      <td>8
      <td><span data-x="Control Group Numeric">Numeric</span>
    <tr>
      <td colspan=3>"<dfn attr-value
      for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
      data-x="attr-fe-autocomplete-bday-month">bday-month</code></dfn>"
      <td>Month component of birthday
      <td><span>Valid integer</span> in the range 1..12
      <td>6
      <td><span data-x="Control Group Numeric">Numeric</span>
    <tr>
      <td colspan=3>"<dfn attr-value
      for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
      data-x="attr-fe-autocomplete-bday-year">bday-year</code></dfn>"
      <td>Year component of birthday
      <td><span>Valid integer</span> greater than zero
      <td>1955
      <td><span data-x="Control Group Numeric">Numeric</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-sex">sex</code></dfn>"
     <td>Gender identity (e.g. Female, Fa'afafine)
     <td>Free-form text, no newlines
     <td>Male
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-url">url</code></dfn>"
     <td>Home page or other web page corresponding to the company, person, address, or contact information in the other fields associated with this field
     <td><span>Valid URL string</span>
     <td>https://www.w3.org/People/Berners-Lee/
     <td><span data-x="Control Group URL">URL</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-photo">photo</code></dfn>"
     <td>Photograph, icon, or other image corresponding to the company, person, address, or contact information in the other fields associated with this field
     <td><!--File or--> <span>Valid URL string</span>
     <td>https://www.w3.org/Press/Stock/Berners-Lee/2001-europaeum-eighth.jpg
     <td><!--<span data-x="Control Group File">File</span>--> <span data-x="Control Group URL">URL</span>
   <tbody>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-tel">tel</code></dfn>"
     <td>Full telephone number, including country code
     <td><span>ASCII digits</span> and U+0020 SPACE characters, prefixed by a U+002B PLUS SIGN character (+)
     <td>+1 617 253 5702
     <td><span data-x="Control Group Tel">Tel</span>
    <tr>
     <td rowspan=6 class="non-rectangular-cell-indentation">
      <td colspan=3>"<dfn attr-value
      for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
      data-x="attr-fe-autocomplete-tel-country-code">tel-country-code</code></dfn>"
      <td>Country code component of the telephone number
      <td><span>ASCII digits</span> prefixed by a U+002B PLUS SIGN character (+)
      <td>+1
      <td><span data-x="Control Group Text">Text</span>
    <tr>
      <td colspan=3>"<dfn attr-value
      for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
      data-x="attr-fe-autocomplete-tel-national">tel-national</code></dfn>"
      <td>Telephone number without the county code component, with a country-internal prefix applied if applicable
      <td><span>ASCII digits</span> and U+0020 SPACE characters
      <td>617 253 5702
      <td><span data-x="Control Group Text">Text</span>
    <tr>
      <td rowspan=4 class="non-rectangular-cell-indentation">
       <td colspan=2>"<dfn attr-value
       for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
       data-x="attr-fe-autocomplete-tel-area-code">tel-area-code</code></dfn>"
       <td>Area code component of the telephone number, with a country-internal prefix applied if applicable
       <td><span>ASCII digits</span>
       <td>617
       <td><span data-x="Control Group Text">Text</span>
    <tr>
       <td colspan=2>"<dfn attr-value
       for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
       data-x="attr-fe-autocomplete-tel-local">tel-local</code></dfn>"
       <td>Telephone number without the country code and area code components
       <td><span>ASCII digits</span>
       <td>2535702
       <td><span data-x="Control Group Text">Text</span>
    <tr>
       <td rowspan=2 class="non-rectangular-cell-indentation">
        <td>"<dfn attr-value
        for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
        data-x="attr-fe-autocomplete-tel-local-prefix">tel-local-prefix</code></dfn>"
        <td>First part of the component of the telephone number that follows the area code, when that component is split into two components
        <td><span>ASCII digits</span>
        <td>253
        <td><span data-x="Control Group Text">Text</span>
    <tr>
        <td>"<dfn attr-value
        for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
        data-x="attr-fe-autocomplete-tel-local-suffix">tel-local-suffix</code></dfn>"
        <td>Second part of the component of the telephone number that follows the area code, when that component is split into two components
        <td><span>ASCII digits</span>
        <td>5702
        <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-tel-extension">tel-extension</code></dfn>"
     <td>Telephone number internal extension code
     <td><span>ASCII digits</span>
     <td>1000
     <td><span data-x="Control Group Text">Text</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-email">email</code></dfn>"
     <td>Email address
     <td><span>Valid email address</span>
     <td>timbl@w3.org
     <td><span data-x="Control Group Username">Username</span>
    <tr>
     <td colspan=4>"<dfn attr-value
     for="button/autocomplete,fieldset/autocomplete,input/autocomplete,object/autocomplete,output/autocomplete,select/autocomplete,textarea/autocomplete"><code
     data-x="attr-fe-autocomplete-impp">impp</code></dfn>"
     <td>URL representing an instant messaging protocol endpoint (for example, "<code data-x="">aim:goim?screenname=example</code>" or "<code data-x="">xmpp:fred@example.net</code>")
     <td><span>Valid URL string</span>
     <td>irc://example.org/timbl,isuser
     <td><span data-x="Control Group URL">URL</span>
  </table>

  <p>The groups correspond to controls as follows:</p>

  <dl>
   <dt><dfn data-x="Control Group Text">Text</dfn>

   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-hidden">Hidden</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-text">Text</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-search">Search</span> state
   <dd><code>textarea</code> elements
   <dd><code>select</code> elements

   <dt><dfn data-x="Control Group Multiline">Multiline</dfn>

   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-hidden">Hidden</span> state
   <dd><code>textarea</code> elements
   <dd><code>select</code> elements

   <dt><dfn data-x="Control Group Password">Password</dfn>

   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-hidden">Hidden</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-text">Text</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-search">Search</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-password">Password</span> state
   <dd><code>textarea</code> elements
   <dd><code>select</code> elements

   <dt><dfn data-x="Control Group URL">URL</dfn>

   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-hidden">Hidden</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-text">Text</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-search">Search</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-url">URL</span> state
   <dd><code>textarea</code> elements
   <dd><code>select</code> elements

   <dt><dfn data-x="Control Group Username">Username</dfn><span id="control-group-e-mail"></span>

   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-hidden">Hidden</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-text">Text</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-search">Search</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-email">Email</span> state
   <dd><code>textarea</code> elements
   <dd><code>select</code> elements

   <dt><dfn data-x="Control Group Tel">Tel</dfn>

   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-hidden">Hidden</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-text">Text</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-search">Search</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-tel">Telephone</span> state
   <dd><code>textarea</code> elements
   <dd><code>select</code> elements

   <dt><dfn data-x="Control Group Numeric">Numeric</dfn>

   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-hidden">Hidden</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-text">Text</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-search">Search</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-number">Number</span> state
   <dd><code>textarea</code> elements
   <dd><code>select</code> elements

   <dt><dfn data-x="Control Group Month">Month</dfn>

   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-hidden">Hidden</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-text">Text</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-search">Search</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-month">Month</span> state
   <dd><code>textarea</code> elements
   <dd><code>select</code> elements

   <dt><dfn data-x="Control Group Date">Date</dfn>

   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-hidden">Hidden</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-text">Text</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-search">Search</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-date">Date</span> state
   <dd><code>textarea</code> elements
   <dd><code>select</code> elements

<!--
   <dt><dfn data-x="Control Group File">File</dfn>

   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-hidden">Hidden</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-text">Text</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-search">Search</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-url">URL</span> state
   <dd><code>input</code> elements with a <code data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-file">File</span> state
   <dd><code>textarea</code> elements
   <dd><code>select</code> elements
-->
  </dl>

  <p id="more-on-address-levels"><strong>Address levels</strong>: The "<code
  data-x="attr-fe-autocomplete-address-level1">address-level1</code>" &ndash; "<code
  data-x="attr-fe-autocomplete-address-level4">address-level4</code>" fields are used to describe
  the locality of the street address. Different locales have different numbers of levels. For
  example, the US uses two levels (state and town), the UK uses one or two depending on the address
  (the post town, and in some cases the locality), and China can use three (province, city,
  district). The "<code data-x="attr-fe-autocomplete-address-level1">address-level1</code>" field
  represents the widest administrative division. Different locales order the fields in different
  ways; for example, in the US the town (level 2) precedes the state (level 1); while in Japan the
  prefecture (level 1) precedes the city (level 2) which precedes the district (level 3). Authors
  are encouraged to provide forms that are presented in a way that matches the country's conventions
  (hiding, showing, and rearranging fields accordingly as the user changes the country).</p>


  <div w-nodev>

  <h6 id="autofill-processing-model"><span id="processing-model-3"></span>Processing model</h6>

  <p>Each <code>input</code> element to which the <code data-x="attr-fe-autocomplete">autocomplete</code> attribute <span data-x="concept-input-apply">applies</span>, each <code>select</code> element, and each <code>textarea</code> element, has an
  <dfn>autofill hint set</dfn>, an <dfn>autofill scope</dfn>, an <dfn>autofill field name</dfn>, a
  <dfn>non-autofill credential type</dfn>, and an <dfn>IDL-exposed autofill value</dfn>.</p>

  <p>The <span>autofill field name</span> specifies the specific kind of data expected in the field,
  e.g. "<code data-x="attr-fe-autocomplete-street-address">street-address</code>" or "<code
  data-x="attr-fe-autocomplete-cc-exp">cc-exp</code>".</p>

  <p>The <span>autofill hint set</span> identifies what address or contact information type the user
  agent is to look at, e.g. "<code data-x="attr-fe-autocomplete-shipping">shipping</code> <code
  data-x="attr-fe-autocomplete-fax">fax</code>" or "<code
  data-x="attr-fe-autocomplete-billing">billing</code>".</p>

  <p>The <span>non-autofill credential type</span> identifies a type of
  <span data-x="credman-credential">credential</span> that may be offered by the user agent when the
  user interacts with the field alongside other <span>autofill field</span> values. If this value is
  "<code data-x="">webauthn</code>" instead of null, selecting a credential of that type will resolve
  a pending <code data-x="conditional mediation">conditional</code> mediation
  <code>navigator.credentials.get()</code> request, instead of autofilling the field.</p>

  <div class="example">
   <p>For example, a sign-in page could instruct the user agent to either autofill a saved password,
   or show a <span>public key credential</span> that will resolve a pending
   <code>navigator.credentials.get()</code> request. A user can select either to sign-in.</p>

   <pre><code class="html">&lt;input name=password type=password autocomplete="current-password webauthn"></code></pre>
  </div>

  <p>The <span>autofill scope</span> identifies the group of fields whose information concerns the
  same subject, and consists of the <span>autofill hint set</span> with, if
  applicable, the "<code data-x="">section-*</code>" prefix, e.g. "<code data-x="">billing</code>",
  "<code data-x="">section-parent shipping</code>", or "<code data-x="">section-child shipping
  home</code>".</p>

  <p>These values are defined as the result of running the following algorithm:</p>

  <ol>
   <li><p>If the element has no <code data-x="attr-fe-autocomplete">autocomplete</code> attribute,
   then jump to the step labeled <i>default</i>.</p></li>

   <li><p>Let <var>tokens</var> be the result of <span data-x="split a string on ASCII
   whitespace">splitting the attribute's value on ASCII whitespace</span>.</p></li>

   <li><p>If <var>tokens</var> is empty, then jump to the step labeled <i>default</i>.</p></li>

   <li><p>Let <var>index</var> be the index of the last token in <var>tokens</var>.</p></li>

   <li><p>Let <var>field</var> be the <var>index</var>th token in <var>tokens</var>.</p></li>

   <li><p>Set the <var>category</var>, <var>maximum tokens</var> pair to the result of <span
   data-x="determine a field's category">determining a field's category</span> given
   <var>field</var>.</p></li>

   <li><p>If <var>category</var> is null, then jump to the step labeled <i>default</i>.</p></li>

   <li><p>If the number of tokens in <var>tokens</var> is greater than <var>maximum tokens</var>,
   then jump to the step labeled <i>default</i>.</p></li>

   <li><p>If <var>category</var> is Off or Automatic but the element's <code
   data-x="attr-fe-autocomplete">autocomplete</code> attribute is wearing the <span>autofill anchor
   mantle</span>, then jump to the step labeled <i>default</i>.</p></li>

   <li><p>If <var>category</var> is Off, let the element's <span>autofill field name</span>
   be the string "<code data-x="">off</code>", let its <span>autofill hint set</span> be empty, and
   let its <span>IDL-exposed autofill value</span> be the string "<code data-x="">off</code>". Then,
   return.</p>

   <li><p>If <var>category</var> is Automatic, let the element's <span>autofill field
   name</span> be the string "<code data-x="">on</code>", let its <span>autofill hint set</span> be
   empty, and let its <span>IDL-exposed autofill value</span> be the string "<code
   data-x="">on</code>". Then, return.</p>

   <li><p>Let <var>scope tokens</var> be an empty list.</p></li>

   <li><p>Let <var>hint tokens</var> be an empty set.</p></li>

   <li><p>Let <var>credential type</var> be null.</p></li>

   <li><p>Let <var>IDL value</var> have the same value as <var>field</var>.</p></li>

   <li>
    <p>If <var>category</var> is Credential and the <var>index</var>th token in <var>tokens</var> is
    an <span>ASCII case-insensitive</span> match for "<code
    data-x="attr-fe-autocomplete-webauthn">webauthn</code>", then run the substeps that follow:</p>
    <ol>
     <li><p>Set <var>credential type</var> to "<code data-x="">webauthn</code>".</p></li>

     <li><p>If the <var>index</var>th token in <var>tokens</var> is the first entry,
     then skip to the step labeled <i>done</i>.</p></li>

     <li><p>Decrement <var>index</var> by one.</p></li>

     <li><p>Set the <var>category</var>, <var>maximum tokens</var> pair to the result of <span
     data-x="determine a field's category">determining a field's category</span> given
     the <var>index</var>th token in <var>tokens</var>.</p></li>

     <li><p>If <var>category</var> is not Normal and <var>category</var> is not Contact, then jump to
     the step labeled <i>default</i>.</p></li>

     <li><p>If <var>index</var> is greater than <var>maximum tokens</var> minus one (i.e. if the
     number of remaining tokens is greater than <var>maximum tokens</var>), then jump to the step
     labeled <i>default</i>.</p></li>

     <li><p>Set <var>IDL value</var> to the concatenation of the <var>index</var>th token in
     <var>tokens</var>, a U+0020 SPACE character, and the previous value of <var>IDL
     value</var>.</p></li>
    </ol>
   </li>

   <li><p>If the <var>index</var>th token in <var>tokens</var> is the first entry,
   then skip to the step labeled <i>done</i>.</p></li>

   <li><p>Decrement <var>index</var> by one.</p></li>

   <li>
    <p>If <var>category</var> is Contact and the <var>index</var>th token in <var>tokens</var> is an <span>ASCII case-insensitive</span> match for one of the strings in
    the following list, then run the substeps that follow:</p>

    <ul class="brief">
     <li>"<code data-x="attr-fe-autocomplete-home">home</code>"
     <li>"<code data-x="attr-fe-autocomplete-work">work</code>"
     <li>"<code data-x="attr-fe-autocomplete-mobile">mobile</code>"
     <li>"<code data-x="attr-fe-autocomplete-fax">fax</code>"
     <li>"<code data-x="attr-fe-autocomplete-pager">pager</code>"
    </ul>

    <p>The substeps are:</p>

    <ol>
     <li><p>Let <var>contact</var> be the matching string from the list above.</p></li>

     <li><p>Insert <var>contact</var> at the start of <var>scope
     tokens</var>.</p></li>

     <li><p>Add <var>contact</var> to <var>hint tokens</var>.</p></li>

     <li><p>Let <var>IDL value</var> be the concatenation of <var>contact</var>, a
     U+0020 SPACE character, and the previous value of <var>IDL value</var>.</p></li>

     <li><p>If the <var>index</var>th entry in <var>tokens</var> is the first
     entry, then skip to the step labeled <i>done</i>.</p></li>

     <li><p>Decrement <var>index</var> by one.</p></li>
    </ol>
   </li>

   <li>
    <p>If the <var>index</var>th token in <var>tokens</var> is an <span>ASCII
    case-insensitive</span> match for one of the strings in the following list, then run the
    substeps that follow:</p>

    <ul class="brief">
     <li>"<code data-x="attr-fe-autocomplete-shipping">shipping</code>"
     <li>"<code data-x="attr-fe-autocomplete-billing">billing</code>"
    </ul>

    <p>The substeps are:</p>

    <ol>
     <li><p>Let <var>mode</var> be the matching string from the list above.</p></li>

     <li><p>Insert <var>mode</var> at the start of <var>scope
     tokens</var>.</p></li>

     <li><p>Add <var>mode</var> to <var>hint tokens</var>.</p></li>

     <li><p>Let <var>IDL value</var> be the concatenation of <var>mode</var>, a
     U+0020 SPACE character, and the previous value of <var>IDL value</var>.</p></li>

     <li><p>If the <var>index</var>th entry in <var>tokens</var> is the first
     entry, then skip to the step labeled <i>done</i>.</p></li>

     <li><p>Decrement <var>index</var> by one.</p></li>
    </ol>
   </li>

   <li><p>If the <var>index</var>th entry in <var>tokens</var> is not the first entry, then jump to
   the step labeled <i>default</i>.</p></li>

   <li><p>If the first eight characters of the <var>index</var>th token in <var>tokens</var> are not
   an <span>ASCII case-insensitive</span> match for the string "<code
   data-x="attr-fe-autocomplete-section">section-</code>", then jump to the step labeled
   <i>default</i>.</p></li>

   <li><p>Let <var>section</var> be the <var>index</var>th token in <var>tokens</var>,
   <span>converted to ASCII lowercase</span>.</p></li>

   <li><p>Insert <var>section</var> at the start of <var>scope tokens</var>.</p></li>

   <li><p>Let <var>IDL value</var> be the concatenation of <var>section</var>, a U+0020 SPACE
   character, and the previous value of <var>IDL value</var>.</p></li>

   <li><p><i>Done</i>: Let the element's <span>autofill hint set</span> be <var>hint
   tokens</var>.</p>

   <li><p>Let the element's <span>non-autofill credential type</span> be <var>credential
   type</var>.</p></li>

   <li><p>Let the element's <span>autofill scope</span> be <var>scope tokens</var>.</p>

   <li><p>Let the element's <span>autofill field name</span> be <var>field</var>.</p>

   <li><p>Let the element's <span>IDL-exposed autofill value</span> be <var>IDL value</var>.</p>

   <li><p>Return.</p></li>

   <li><p><i>Default</i>: Let the element's <span>IDL-exposed autofill value</span> be the empty
   string, and its <span>autofill hint set</span> and <span>autofill scope</span> be empty.</p></li>

   <li><p>If the element's <code data-x="attr-fe-autocomplete">autocomplete</code> attribute is
   wearing the <span>autofill anchor mantle</span>, then let the element's <span>autofill field
   name</span> be the empty string and return.</p></li>

   <li><p>Let <var>form</var> be the element's <span>form owner</span>, if any, or null
   otherwise.</p></li>

   <li>
    <p>If <var>form</var> is not null and <var>form</var>'s <code
    data-x="attr-form-autocomplete">autocomplete</code> attribute is in the <span
    data-x="attr-form-autocomplete-off-state">off</span> state, then let the element's
    <span>autofill field name</span> be "<code data-x="attr-fe-autocomplete-off">off</code>".</p>

    <p>Otherwise, let the element's <span>autofill field name</span> be "<code
    data-x="attr-fe-autocomplete-on">on</code>".</p>
   </li>
  </ol>

  <p>To <dfn>determine a field's category</dfn>, given <var>field</var>:</p>

  <ol>
   <li>
    <p>If the <var>field</var> is not an <span>ASCII case-insensitive</span> match for one of the
    tokens given in the first column of the following table, return the pair (null, null).</p>

    <table>
     <thead>
      <tr>
       <th>Token
       <th>Maximum number of tokens
       <th>Category
     <tbody>
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-off">off</code>"
       <td>1
       <td>Off
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-on">on</code>"
       <td>1
       <td>Automatic
     <tbody>
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-name">name</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-honorific-prefix">honorific-prefix</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-given-name">given-name</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-additional-name">additional-name</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-family-name">family-name</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-honorific-suffix">honorific-suffix</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-nickname">nickname</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-organization-title">organization-title</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-username">username</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-new-password">new-password</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-current-password">current-password</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-one-time-code">one-time-code</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-organization">organization</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-street-address">street-address</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-address-line1">address-line1</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-address-line2">address-line2</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-address-line3">address-line3</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-address-level4">address-level4</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-address-level3">address-level3</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-address-level2">address-level2</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-address-level1">address-level1</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-country">country</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-country-name">country-name</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-postal-code">postal-code</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-cc-name">cc-name</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-cc-given-name">cc-given-name</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-cc-additional-name">cc-additional-name</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-cc-family-name">cc-family-name</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-cc-number">cc-number</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-cc-exp">cc-exp</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-cc-exp-month">cc-exp-month</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-cc-exp-year">cc-exp-year</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-cc-csc">cc-csc</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-cc-type">cc-type</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-transaction-currency">transaction-currency</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-transaction-amount">transaction-amount</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-language">language</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-bday">bday</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-bday-day">bday-day</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-bday-month">bday-month</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-bday-year">bday-year</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-sex">sex</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-url">url</code>"
       <td>3
       <td>Normal
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-photo">photo</code>"
       <td>3
       <td>Normal
     <tbody>
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-tel">tel</code>"
       <td>4
       <td>Contact
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-tel-country-code">tel-country-code</code>"
       <td>4
       <td>Contact
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-tel-national">tel-national</code>"
       <td>4
       <td>Contact
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-tel-area-code">tel-area-code</code>"
       <td>4
       <td>Contact
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-tel-local">tel-local</code>"
       <td>4
       <td>Contact
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-tel-local-prefix">tel-local-prefix</code>"
       <td>4
       <td>Contact
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-tel-local-suffix">tel-local-suffix</code>"
       <td>4
       <td>Contact
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-tel-extension">tel-extension</code>"
       <td>4
       <td>Contact
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-email">email</code>"
       <td>4
       <td>Contact
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-impp">impp</code>"
       <td>4
       <td>Contact
     <tbody>
      <tr>
       <td>"<code data-x="attr-fe-autocomplete-webauthn">webauthn</code>"
       <td>5
       <td>Credential
    </table>
   </li>

   <li><p>Otherwise, let <var>maximum tokens</var> and <var>category</var> be the values of the
   cells in the second and third columns of that row respectively.</p></li>

   <li><p>Return the pair (<var>category</var>, <var>maximum tokens</var>).</p></li>
  </ol>

  <hr>

  <p>For the purposes of autofill, a <dfn>control's data</dfn> depends on the kind of control:</p>

  <dl>
   <dt>An <code>input</code> element with its <code data-x="attr-input-type">type</code> attribute
   in the <span data-x="attr-input-type-email">Email</span> state and with the <code
   data-x="attr-input-multiple">multiple</code> attribute specified</dt>

   <dd>The element's <span data-x="concept-fe-values">value<em>s</em></span>.</dd>

<!--
   <dt>An <code>input</code> element with its <code data-x="attr-input-type">type</code> attribute
   in the <span data-x="attr-input-type-range">Range</span> state and with the <code
   data-x="attr-input-multiple">multiple</code> attribute specified</dt>
-->
   <dt>Any other <code>input</code> element</dt>
   <dt>A <code>textarea</code> element</dt>

   <dd>The element's <span data-x="concept-fe-value">value</span>.</dd>

   <dt>A <code>select</code> element with its <code data-x="attr-select-multiple">multiple</code>
   attribute specified</dt>

   <dd>The <code>option</code> elements in the <code>select</code> element's <span
   data-x="concept-select-option-list">list of options</span> that have their <span
   data-x="concept-option-selectedness">selectedness</span> set to true.</dd>

   <dt>Any other <code>select</code> element</dt>

   <dd>The <code>option</code> element in the <code>select</code> element's <span
   data-x="concept-select-option-list">list of options</span> that has its <span
   data-x="concept-option-selectedness">selectedness</span> set to true.</dd>
  </dl>

  <hr>

  <p>How to process the <span>autofill hint set</span>, <span>autofill scope</span>, and
  <span>autofill field name</span> depends on the mantle that the <code
  data-x="attr-fe-autocomplete">autocomplete</code> attribute is wearing.</p>

  <dl class="switch">

   <dt>When wearing the <span>autofill expectation mantle</span>...

   <dd>
    <p>When an element's <span>autofill field name</span> is "<code
    data-x="attr-fe-autocomplete-off">off</code>", the user agent should not remember the <span>control's
    data</span>, and should not offer past values to the user.</p>

    <p class="note">In addition, when an element's <span>autofill field name</span> is "<code
    data-x="attr-fe-autocomplete-off">off</code>", <a href="#history-autocomplete">values are
    reset</a> when <span data-x="reactivate a document">reactivating a document</span>.</p>

    <div class="example">

     <p>Banks frequently do not want UAs to prefill login information:</p>

     <pre><code class="html">&lt;p>&lt;label>Account: &lt;input type="text" name="ac" autocomplete="off">&lt;/label>&lt;/p>
&lt;p>&lt;label>PIN: &lt;input type="password" name="pin" autocomplete="off">&lt;/label>&lt;/p></code></pre>

    </div>

    <p>When an element's <span>autofill field name</span> is <em>not</em> "<code
    data-x="attr-fe-autocomplete-off">off</code>", the user agent may store the <span>control's
    data</span>, and may offer previously stored values to the user.</p>

    <div class="example">

     <p>For example, suppose a user visits a page with this control:</p>

     <pre><code class="html">&lt;select name="country">
 &lt;option>Afghanistan
 &lt;option>Albania
 &lt;option>Algeria
 &lt;option>Andorra
 &lt;option>Angola
 &lt;option>Antigua and Barbuda
 &lt;option>Argentina
 &lt;option>Armenia
 &lt;!-- <em>...</em> -->
 &lt;option>Yemen
 &lt;option>Zambia
 &lt;option>Zimbabwe
&lt;/select></code></pre>

     <p>This might render as follows:</p>

     <p><img src="/images/select-country-1.png" alt="A drop-down control with a long alphabetical list of countries."></p>

     <p>Suppose that on the first visit to this page, the user selects "Zambia". On the second visit,
     the user agent could duplicate the entry for Zambia at the top of the list, so that the interface
     instead looks like this:</p>

     <p><img src="/images/select-country-2.png" alt="The same drop-down control with the alphabetical list of countries, but with Zambia as an entry at the top."></p>

    </div>

    <p>When the <span>autofill field name</span> is "<code
    data-x="attr-fe-autocomplete-on">on</code>", the user agent should attempt to use heuristics to
    determine the most appropriate values to offer the user, e.g. based on the element's <code
    data-x="attr-fe-name">name</code> value, the position of the element in its <span>tree</span>,
    what other fields exist in the form, and so forth.</p>

    <p>When the <span>autofill field name</span> is one of the names of the <span data-x="autofill
    field">autofill fields</span> described above, the user agent should provide suggestions that
    match the meaning of the field name as given in the table earlier in this section. The
    <span>autofill hint set</span> should be used to select amongst multiple possible suggestions.</p>

    <p class="example">For example, if a user once entered one address into fields that used the
    "<code data-x="attr-fe-autocomplete-shipping">shipping</code>" keyword, and another address into
    fields that used the "<code data-x="attr-fe-autocomplete-billing">billing</code>" keyword, then in
    subsequent forms only the first address would be suggested for form controls whose <span>autofill
    hint set</span> contains the keyword "<code
    data-x="attr-fe-autocomplete-shipping">shipping</code>". Both addresses might be suggested,
    however, for address-related form controls whose <span>autofill hint set</span> does not contain
    either keyword.</p>
   </dd>

   <dt>When wearing the <span>autofill anchor mantle</span>...

   <dd>
    <p>When the <span>autofill field name</span> is not the empty string, then the user agent must
    act as if the user had specified the <span>control's data</span> for the given <span>autofill
    hint set</span>, <span>autofill scope</span>, and <span>autofill field name</span>
    combination.</p>
   </dd>
  </dl>

  <p>When the user agent <dfn data-x="concept-fe-autofill">autofills form controls</dfn>, elements
  with the same <span>form owner</span> and the same <span>autofill scope</span> must use data
  relating to the same person, address, payment instrument, and contact details. <span
  id="autofill-country" data-x="">When a user agent autofills "<code
  data-x="attr-fe-autocomplete-country">country</code>" and "<code
  data-x="attr-fe-autocomplete-country-name">country-name</code>" fields with the same <span>form
  owner</span> and <span>autofill scope</span>, and the user agent has a value for the <code
  data-x="attr-fe-autocomplete-country">country</code>" field(s), then the "<code
  data-x="attr-fe-autocomplete-country-name">country-name</code>" field(s) must be filled using a
  human-readable name for the same country.</span> When a user agent fills in multiple fields at
  once, all fields with the same <span>autofill field name</span>, <span>form owner</span>, and
  <span>autofill scope</span> must be filled with the same value.</p>

  <p class="example">Suppose a user agent knows of two phone numbers, +1 555 123 1234 and +1 555 666
  7777. It would not be conforming for the user agent to fill a field with <code
  data-x="">autocomplete="shipping tel-local-prefix"</code> with the value "123" and another field
  in the same form with <code data-x="">autocomplete="shipping tel-local-suffix"</code> with the
  value "7777". The only valid prefilled values given the aforementioned information would be "123"
  and "1234", or "666" and "7777", respectively.</p>

  <p class="example">Similarly, if a form for some reason contained both a "<code
  data-x="attr-fe-autocomplete-cc-exp">cc-exp</code>" field and a "<code
  data-x="attr-fe-autocomplete-cc-exp-month">cc-exp-month</code>" field, and the user agent
  prefilled the form, then the month component of the former would have to match the latter.</p>

  <div class="example">

   <p>This requirement interacts with the <span>autofill anchor mantle</span> also. Consider the
   following markup snippet:</p>

   <pre><code class="html">&lt;form>
 &lt;input type=hidden autocomplete="nickname" value="TreePlate">
 &lt;input type=text autocomplete="nickname">
&lt;/form></code></pre>

   <p>The only value that a conforming user agent could suggest in the text control is "TreePlate",
   the value given by the hidden <code>input</code> element.</p>

  </div>

  <p>The "<code data-x="">section-*</code>" tokens in the <span>autofill scope</span> are opaque;
  user agents must not attempt to derive meaning from the precise values of these tokens.</p>

  <p class="example">For example, it would not be conforming if the user agent decided that it
  <!--non-normative-->should offer the address it knows to be the user's daughter's address for
  "<code data-x="">section-child</code>" and the addresses it knows to be the user's spouses'
  addresses for "<code data-x="">section-spouse</code>".</p>

  <p>The autocompletion mechanism must be implemented by the user agent acting as if the user had
  modified the <span>control's data</span>, and must be done at a time where the element is <i
  data-x="concept-fe-mutable">mutable</i> (e.g. just after the element has been inserted into the
  document, or when the user agent <span data-x="stop parsing">stops parsing</span>). User agents
  must only prefill controls using values that the user could have entered.</p>

  <p class="example">For example, if a <code>select</code> element only has <code>option</code>
  elements with values "Steve" and "Rebecca", "Jay", and "Bob", and has an <span>autofill field
  name</span> "<code data-x="attr-fe-autocomplete-given-name">given-name</code>", but the user
  agent's only idea for what to prefill the field with is "Evan", then the user agent cannot prefill
  the field. It would not be conforming to somehow set the <code>select</code> element to the value
  "Evan", since the user could not have done so themselves.</p>

  <p>A user agent prefilling a form control must not discriminate between form controls that are
  <span>in a document tree</span> and those that are <span>connected</span>; that is, it is not
  conforming to make the decision on whether or not to autofill based on whether the element's
  <span>root</span> is a <span>shadow root</span> versus a <code>Document</code>.</p>

  <p>A user agent prefilling a form control's <span data-x="concept-fe-value">value</span> must not
  cause that control to <span data-x="suffering from a type mismatch">suffer from a type
  mismatch</span>, <span data-x="suffering from being too long">suffer from being too long</span>,
  <span data-x="suffering from being too short">suffer from being too short</span>, <span
  data-x="suffering from an underflow">suffer from an underflow</span>, <span data-x="suffering from
  an overflow">suffer from an overflow</span>, or <span data-x="suffering from a step
  mismatch">suffer from a step mismatch</span>. A user agent prefilling a form control's <span
  data-x="concept-fe-value">value</span> must not cause that control to <span data-x="suffering from
  a pattern mismatch">suffer from a pattern mismatch</span> either. Where possible given the
  control's constraints, user agents must use the format given as canonical in the aforementioned
  table. Where it's not possible for the canonical format to be used, user agents should use
  heuristics to attempt to convert values so that they can be used.</p>

  <div class="example">

   <p>For example, if the user agent knows that the user's middle name is "Ines", and attempts to
   prefill a form control that looks like this:</p>

   <pre><code class="html">&lt;input name=middle-initial maxlength=1 autocomplete="additional-name"></code></pre>

   <p>...then the user agent could convert "Ines" to "I" and prefill it that way.</p>

  </div>

  <div class="example">

   <p>A more elaborate example would be with month values. If the user agent knows that the user's
   birthday is the 27th of July 2012, then it might try to prefill all of the following controls
   with slightly different values, all driven from this information:</p>

   <table>
    <tr>
     <td>
      <pre><code class="html">&lt;input name=b type=month autocomplete="bday"></code></pre>
     <td>
      2012-07
     <td>
      The day is dropped since the <span
      data-x="attr-input-type-month">Month</span> state only accepts a
      month/year combination. (Note that this example is non-conforming, because the <span>autofill
      field name</span> <code data-x="attr-fe-autocomplete-bday">bday</code> is not allowed with the
      <span data-x="attr-input-type-month">Month</span> state.)
    <tr>
     <td>
      <pre><code class="html">&lt;select name=c autocomplete="bday">
 &lt;option>Jan
 &lt;option>Feb
 <em>...</em>
 &lt;option>Jul
 &lt;option>Aug
 <em>...</em>
&lt;/select></code></pre>
     <td>
      July
     <td>
      The user agent picks the month from the listed options, either by noticing there are twelve
      options and picking the 7th, or by recognizing that one of the strings (three characters "Jul"
      followed by a newline and a space) is a close match for the name of the month (July) in one of
      the user agent's supported languages, or through some other similar mechanism.
    <tr>
     <td>
      <pre><code class="html">&lt;input name=a type=number min=1 max=12 autocomplete="bday-month"></code></pre>
     <td>
      7
     <td>
      User agent converts "July" to a month number in the range 1..12, like the field.
    <tr>
     <td>
      <pre><code class="html">&lt;input name=a type=number min=0 max=11 autocomplete="bday-month"></code></pre>
     <td>
      6
     <td>
      User agent converts "July" to a month number in the range 0..11, like the field.
    <tr>
     <td>
      <pre><code class="html">&lt;input name=a type=number min=1 max=11 autocomplete="bday-month"></code></pre>
     <td>
     <td>
      User agent doesn't fill in the field, since it can't make a good guess as to what the form expects.
   </table>

  </div>

  <p>A user agent may allow the user to override an element's <span>autofill field name</span>, e.g.
  to change it from "<code data-x="attr-fe-autocomplete-off">off</code>" to "<code
  data-x="attr-fe-autocomplete-on">on</code>" to allow values to be remembered and prefilled despite
  the page author's objections, or to always "<code data-x="attr-fe-autocomplete-off">off</code>",
  never remembering values.</p>

  <p>More specifically, user agents may in particular consider replacing the <span>autofill field
  name</span> of form controls that match the description given in the first column of the following
  table, when their <span>autofill field name</span> is either "<code
  data-x="attr-fe-autocomplete-on">on</code>" or "<code
  data-x="attr-fe-autocomplete-off">off</code>", with the value given in the second cell of that
  row. If this table is used, the replacements must be done in <span>tree order</span>, since all
  but the first row references the <span>autofill field name</span> of earlier elements. When the
  descriptions below refer to form controls being preceded or followed by others, they mean in the
  list of <span data-x="category-listed">listed elements</span> that share the same <span>form
  owner</span>.</p>

  <table>

   <thead>
    <tr>
     <th>Form control
     <th>New <span>autofill field name</span>

   <tbody>


    <tr>

     <td>

      an <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
      the <span data-x="attr-input-type-text">Text</span> state that is followed by an
      <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
      the <span data-x="attr-input-type-password">Password</span> state

     <td>

      "<code data-x="attr-fe-autocomplete-username">username</code>"


    <tr>

     <td>

      an <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
      the <span data-x="attr-input-type-password">Password</span> state that is preceded by an
      <code>input</code> element whose <span>autofill field name</span> is "<code
      data-x="attr-fe-autocomplete-username">username</code>"

     <td>

      "<code data-x="attr-fe-autocomplete-current-password">current-password</code>"


    <tr>

     <td>

      an <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
      the <span data-x="attr-input-type-password">Password</span> state that is preceded by an
      <code>input</code> element whose <span>autofill field name</span> is "<code
      data-x="attr-fe-autocomplete-current-password">current-password</code>"

     <td>

      "<code data-x="attr-fe-autocomplete-new-password">new-password</code>"


    <tr>

     <td>

      an <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
      the <span data-x="attr-input-type-password">Password</span> state that is preceded by an
      <code>input</code> element whose <span>autofill field name</span> is "<code
      data-x="attr-fe-autocomplete-new-password">new-password</code>"

     <td>

      "<code data-x="attr-fe-autocomplete-new-password">new-password</code>"


  </table>

  <p>The <dfn attribute for="HTMLInputElement,HTMLSelectElement,HTMLTextAreaElement"><code
  data-x="dom-fe-autocomplete">autocomplete</code></dfn> IDL attribute, on getting, must return the
  element's <span>IDL-exposed autofill value</span>, and on setting, must <span>reflect</span> the
  content attribute of the same name.</p>

  </div>


  <h4 id="textFieldSelection">APIs for the text control selections</h4>

  <!-- v2 idea: DOM Range APIs to expose the textarea/input edited value (ack martijnw) -->

  <p>The <code>input</code> and <code>textarea</code> elements define several attributes and methods
  for handling their selection. Their shared algorithms are defined here.</p>

  <!-- v2: also add textLength? it seems to be widely used -->

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-textarea/input-select">select</span>()</code></dt>

   <dd><p>Selects everything in the text control.</p></dd>

   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-textarea/input-selectionStart">selectionStart</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the offset to the start of the selection.</p>

    <p>Can be set, to change the start of the selection.</p>
   </dd>

   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-textarea/input-selectionEnd">selectionEnd</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the offset to the end of the selection.</p>

    <p>Can be set, to change the end of the selection.</p>
   </dd>

   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-textarea/input-selectionDirection">selectionDirection</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current direction of the selection.</p>

    <p>Can be set, to change the direction of the selection.</p>

    <p>The possible values are "<code data-x="">forward</code>", "<code data-x="">backward</code>",
    and "<code data-x="">none</code>".</p>
   </dd>

   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-textarea/input-setSelectionRange">setSelectionRange</span>(<var>start</var>, <var>end</var> [, <var>direction</var>])</code></dt>

   <dd>
    <p>Changes the selection to cover the given substring in the given direction. If the direction
    is omitted, it will be reset to be the platform default (none or forward).</p>
   </dd>

   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-textarea/input-setRangeText">setRangeText</span>(<var>replacement</var> [, <var>start</var>, <var>end</var> [, <var>selectionMode</var> ] ])</code></dt>

   <dd>
    <p>Replaces a range of text with the new text. If the <var>start</var> and <var>end</var>
    arguments are not provided, the range is assumed to be the selection.</p>

    <p>The final argument determines how the selection will be set after the text has been replaced.
    The possible values are:</p>

    <dl>

     <dt>"<code subdfn data-x="dom-SelectionMode-select">select</code>"</dt>

     <dd>Selects the newly inserted text.</dd>


     <dt>"<code subdfn data-x="dom-SelectionMode-start">start</code>"</dt>

     <dd>Moves the selection to just before the inserted text.</dd>


     <dt>"<code subdfn data-x="dom-SelectionMode-end">end</code>"</dt>

     <dd>Moves the selection to just after the selected text.</dd>


     <dt>"<code subdfn data-x="dom-SelectionMode-preserve">preserve</code>"</dt>

     <dd>Attempts to preserve the selection. This is the default.</dd>

    </dl>
   </dd>
  </dl>

  <div w-nodev>

  <p>All <code>input</code> elements to which these APIs <span
  data-x="concept-input-apply">apply</span>, and all <code>textarea</code> elements, have either a
  <dfn data-x="concept-textarea/input-selection">selection</dfn> or a <dfn
  data-x="concept-textarea/input-cursor">text entry cursor position</dfn> at all times (even for
  elements that are not <span>being rendered</span>), measured in offsets into the <span
  data-x="code unit">code units</span> of the control's <span
  data-x="concept-textarea/input-relevant-value">relevant value</span>. The initial state must
  consist of a <span data-x="concept-textarea/input-cursor">text entry cursor</span> at the
  beginning of the control.</p>
  <!-- Previously the spec said to let platform conventions decide, but the web depends on it being
  at the beginning; see https://bugzilla.mozilla.org/show_bug.cgi?id=1337392 -->

  <p>For <code>input</code> elements, these APIs must operate on the element's <span
  data-x="concept-fe-value">value</span>. For <code>textarea</code> elements, these APIs must
  operate on the element's <span data-x="concept-fe-api-value">API value</span>. In the below
  algorithms, we call the value string being operated on the <dfn
  data-x="concept-textarea/input-relevant-value">relevant value</dfn>.</p>

  <div class="example">
   <p>The use of <span data-x="concept-fe-api-value">API value</span> instead of <span
   data-x="concept-textarea-raw-value">raw value</span> for <code>textarea</code> elements means
   that U+000D (CR) characters are normalized away. For example,

   <pre><code class="html">&lt;textarea id="demo">&lt;/textarea>
&lt;script>
 demo.value = "A\r\nB";
 demo.setRangeText("replaced", 0, 2);
 assert(demo.value === "replacedB");
&lt;/script></code></pre>

   <p>If we had operated on the <span data-x="concept-textarea-raw-value">raw value</span> of "<code
   data-x="">A\r\nB</code>", then we would have replaced the characters "<code
   data-x="">A\r</code>", ending up with a result of "<code data-x="">replaced\nB</code>". But since
   we used the <span data-x="concept-fe-api-value">API value</span> of "<code
   data-x="">A\nB</code>", we replaced the characters "<code data-x="">A\n</code>", giving "<code
   data-x="">replacedB</code>".</p>
  </div>

  <p class="note">Characters with no visible rendering, such as U+200D ZERO WIDTH JOINER, still
  count as characters. Thus, for instance, the selection can include just an invisible character,
  and the text insertion cursor can be placed to one side or another of such a character.</p>

  <p>Whenever the <span data-x="concept-textarea/input-relevant-value">relevant value</span> changes
  for an element to which these APIs apply, run these steps:</p>

  <ol>
   <li>
    <p>If the element has a <span data-x="concept-textarea/input-selection">selection</span>:</p>

    <ol>
     <li><p>If the start of the selection is now past the end of the <span
     data-x="concept-textarea/input-relevant-value">relevant value</span>, set it to the end of the
     <span data-x="concept-textarea/input-relevant-value">relevant value</span>.</p></li>

     <li><p>If the end of the selection is now past the end of the <span
     data-x="concept-textarea/input-relevant-value">relevant value</span>, set it to the end of the
     <span data-x="concept-textarea/input-relevant-value">relevant value</span>.</p></li>

     <li><p>If the user agent does not support empty selection, and both the start and end of the
     selection are now pointing to the end of the <span
     data-x="concept-textarea/input-relevant-value">relevant value</span>, then instead set the
     element's <span data-x="concept-textarea/input-cursor">text entry cursor position</span> to the
     end of the <span data-x="concept-textarea/input-relevant-value">relevant value</span>, removing
     any selection.</p></li>
    </ol>
   </li>

   <li><p>Otherwise, the element must have a <span data-x="concept-textarea/input-cursor">text entry
   cursor position</span> position. If it is now past the end of the <span
   data-x="concept-textarea/input-relevant-value">relevant value</span>, set it to the end of the
   <span data-x="concept-textarea/input-relevant-value">relevant value</span>.</p></li>
  </ol>

  <p class="note">In some cases where the <span
  data-x="concept-textarea/input-relevant-value">relevant value</span> changes, other parts of the
  specification will also modify the <span data-x="concept-textarea/input-cursor">text entry cursor
  position</span>, beyond just the clamping steps above. For example, see the <code
  data-x="dom-textarea-value">value</code> setter for <code>textarea</code>.</p>

  <p>Where possible, user interface features for changing the <span
  data-x="concept-textarea/input-selection">text selection</span> in <code>input</code> and
  <code>textarea</code> elements must be implemented using the <span>set the selection range</span>
  algorithm so that, e.g., all the same events fire.</p>

  <p>The <span data-x="concept-textarea/input-selection">selections</span> of <code>input</code> and
  <code>textarea</code> elements have a <dfn>selection direction</dfn>, which is either "<code
  data-x="">forward</code>", "<code data-x="">backward</code>", or "<code data-x="">none</code>".
  The exact meaning of the selection direction depends on the platform. This direction is set when
  the user manipulates the selection.  The initial <span>selection direction</span> must be "<code
  data-x="">none</code>" if the platform supports that direction, or "<code
  data-x="">forward</code>" otherwise.</p>

  <p>To <dfn>set the selection direction</dfn> of an element to a given direction, update the
  element's <span>selection direction</span> to the given direction, unless the direction is "<code
  data-x="">none</code>" and the platform does not support that direction; in that case, update the
  element's <span>selection direction</span> to "<code data-x="">forward</code>".</p>

  <div class="note">
   <p>On Windows, the direction indicates the position of the caret relative to
   the selection: a "<code data-x="">forward</code>" selection has the caret at the end of the
   selection and a "<code data-x="">backward</code>" selection has the caret at the start of the
   selection. Windows has no "<code data-x="">none</code>" direction.</p>

   <p>On Mac, the direction indicates which end of the selection is affected when the user adjusts
   the size of the selection using the arrow keys with the Shift modifier: the "<code
   data-x="">forward</code>" direction means the end of the selection is modified, and the "<code
   data-x="">backward</code>" direction means the start of the selection is modified. The "<code
   data-x="">none</code>" direction is the default on Mac, it indicates that no particular direction
   has yet been selected. The user sets the direction implicitly when first adjusting the selection,
   based on which directional arrow key was used.</p>
  </div>


  <p>The <dfn method for="HTMLInputElement,HTMLTextAreaElement"><code
  data-x="dom-textarea/input-select">select()</code></dfn> method, when invoked, must run the
  following steps:</p>

  <ol>
   <li>
    <p>If this element is an <code>input</code> element, and either <code
    data-x="dom-textarea/input-select">select()</code> <span data-x="do not apply">does not
    apply</span> to this element or the corresponding control has no selectable text, return.</p>

    <p class="example">For instance, in a user agent where <code
    data-x="attr-input-type-color">&lt;input type=color></code> is rendered as a color well with a
    picker, as opposed to a text control accepting a hexadecimal color code, there would be no
    selectable text, and thus calls to the method are ignored.</p>
   </li>

   <li><p><span>Set the selection range</span> with 0 and infinity.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLInputElement,HTMLTextAreaElement"><code
  data-x="dom-textarea/input-selectionStart">selectionStart</code></dfn> attribute's getter must run
  the following steps:</p>

  <ol>
   <li><p>If this element is an <code>input</code> element, and <code
   data-x="dom-textarea/input-selectionStart">selectionStart</code> <span data-x="do not apply">does
   not apply</span> to this element, return null.</p></li>

   <li><p>If there is no <span data-x="concept-textarea/input-selection">selection</span>, return
   the <span>code unit</span> offset within the <span
   data-x="concept-textarea/input-relevant-value">relevant value</span> to the character that
   immediately follows the <span data-x="concept-textarea/input-cursor">text entry
   cursor</span>.</p></li>

   <li><p>Return the <span>code unit</span> offset within the <span
   data-x="concept-textarea/input-relevant-value">relevant value</span> to the character that
   immediately follows the start of the <span
   data-x="concept-textarea/input-selection">selection</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-textarea/input-selectionStart">selectionStart</code> attribute's setter
  must run the following steps:</p>

  <ol>
   <li><p>If this element is an <code>input</code> element, and <code
   data-x="dom-textarea/input-selectionStart">selectionStart</code> <span data-x="do not apply">does
   not apply</span> to this element, throw an <span>"<code>InvalidStateError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Let <var>end</var> be the value of this element's <code
   data-x="dom-textarea/input-selectionEnd">selectionEnd</code> attribute.</p></li>

   <li><p>If <var>end</var> is less than the given value, set <var>end</var> to the given
   value.</p></li>

   <li><p><span>Set the selection range</span> with the given value, <var>end</var>, and the value
   of this element's <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code>
   attribute.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLInputElement,HTMLTextAreaElement"><code
  data-x="dom-textarea/input-selectionEnd">selectionEnd</code></dfn> attribute's getter must run the
  following steps:</p>

  <ol>
   <li><p>If this element is an <code>input</code> element, and <code
   data-x="dom-textarea/input-selectionEnd">selectionEnd</code> <span data-x="do not apply">does
   not apply</span> to this element, return null.</p></li>

   <li><p>If there is no <span data-x="concept-textarea/input-selection">selection</span>, return
   the <span>code unit</span> offset within the <span
   data-x="concept-textarea/input-relevant-value">relevant value</span> to the character that
   immediately follows the <span data-x="concept-textarea/input-cursor">text entry
   cursor</span>.</p></li>

   <li><p>Return the <span>code unit</span> offset within the <span
   data-x="concept-textarea/input-relevant-value">relevant value</span> to the character that
   immediately follows the end of the <span
   data-x="concept-textarea/input-selection">selection</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code> attribute's setter must
  run the following steps:</p>

  <ol>
   <li><p>If this element is an <code>input</code> element, and <code
   data-x="dom-textarea/input-selectionEnd">selectionEnd</code> <span data-x="do not apply">does not
   apply</span> to this element, throw an <span>"<code>InvalidStateError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p><span>Set the selection range</span> with the value of this element's <code
   data-x="dom-textarea/input-selectionStart">selectionStart</code> attribute, the given value, and
   the value of this element's <code
   data-x="dom-textarea/input-selectionDirection">selectionDirection</code> attribute.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLInputElement,HTMLTextAreaElement"><code
  data-x="dom-textarea/input-selectionDirection">selectionDirection</code></dfn> attribute's getter
  must run the following steps:</p>

  <ol>
   <li><p>If this element is an <code>input</code> element, and <code
   data-x="dom-textarea/input-selectionDirection">selectionDirection</code> <span data-x="do not
   apply">does not apply</span> to this element, return null.</p></li>

   <li><p>Return this element's <span>selection direction</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-textarea/input-selectionDirection">selectionDirection</code> attribute's
  setter must run the following steps:</p>

  <ol>
   <li><p>If this element is an <code>input</code> element, and <code
   data-x="dom-textarea/input-selectionDirection">selectionDirection</code> <span data-x="do not
   apply">does not apply</span> to this element, throw an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p><span>Set the selection range</span> with the value of this element's <code
   data-x="dom-textarea/input-selectionStart">selectionStart</code> attribute, the value of this
   element's <code data-x="dom-textarea/input-selectionEnd">selectionEnd</code> attribute, and the
   given value.</p></li>
  </ol>

  <p>The <dfn method for="HTMLInputElement,HTMLTextAreaElement"><code
  data-x="dom-textarea/input-setSelectionRange">setSelectionRange(<var>start</var>, <var>end</var>,
  <var>direction</var>)</code></dfn> method, when invoked, must run the following steps:</p>

  <ol>
   <li><p>If this element is an <code>input</code> element, and <code
   data-x="dom-textarea/input-setSelectionRange">setSelectionRange()</code> <span data-x="do not
   apply">does not apply</span> to this element, throw an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p><span>Set the selection range</span> with <var>start</var>, <var>end</var>, and
   <var>direction</var>.</p></li>
  </ol>

  <p>To <dfn>set the selection range</dfn> with an integer or null <var>start</var>, an integer or
  null or the special value infinity <var>end</var>, and optionally a string <var>direction</var>,
  run the following steps:</p>

  <ol>
   <li><p>If <var>start</var> is null, let <var>start</var> be zero.</p></li>

   <li><p>If <var>end</var> is null, let <var>end</var> be zero.</p></li>

   <li><p>Set the <span data-x="concept-textarea/input-selection">selection</span> of the text
   control to the sequence of <span data-x="code unit">code units</span> within the <span
   data-x="concept-textarea/input-relevant-value">relevant value</span> starting with the code unit
   at the <var>start</var>th position (in logical order) and ending with the code unit at the <span
   data-x="">(<var>end</var>-1)</span>th position. Arguments greater than the <span>length</span> of
   the <span data-x="concept-textarea/input-relevant-value">relevant value</span> of the text control
   (including the special value infinity) must be treated as pointing at the end of the text
   control. If <var>end</var> is less than or equal to <var>start</var> then the start of the
   selection and the end of the selection must both be placed immediately before the character with
   offset <var>end</var>. In UAs where there is no concept of an empty selection, this must set the
   cursor to be just before the character with offset <var>end</var>.</p></li>

   <li><p>If <var>direction</var> is not <span>identical to</span> either "<code
   data-x="">backward</code>" or "<code data-x="">forward</code>", or if the <var>direction</var>
   argument was not given, set <var>direction</var> to "<code data-x="">none</code>".</p></li>

   <li><p><span>Set the selection direction</span> of the text control to
   <var>direction</var>.</p></li>

   <li><p>If the previous steps caused the <span
   data-x="concept-textarea/input-selection">selection</span> of the text control to be modified (in
   either extent or <span data-x="selection direction">direction</span>), then <span>queue an
   element task</span> on the <span>user interaction task source</span> given the element to <span
   data-x="concept-event-fire">fire an event</span> named <code data-x="event-select">select</code>
   at the element, with the <code data-x="dom-Event-bubbles">bubbles</code> attribute initialized to
   true.</p></li>
  </ol>


  <p>The <dfn method for="HTMLInputElement,HTMLTextAreaElement"><code
  data-x="dom-textarea/input-setRangeText">setRangeText(<var>replacement</var>, <var>start</var>,
  <var>end</var>, <var>selectMode</var>)</code></dfn> method, when invoked, must run the following
  steps:</p>

  <ol>
   <li><p>If this element is an <code>input</code> element, and <code
   data-x="dom-textarea/input-setRangeText">setRangeText()</code> <span data-x="do not apply">does
   not apply</span> to this element, throw an <span>"<code>InvalidStateError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Set this element's <span data-x="concept-fe-dirty">dirty value flag</span> to
   true.</p></li>

   <li>
    <p>If the method has only one argument, then let <var>start</var> and <var>end</var> have the values of the <code
    data-x="dom-textarea/input-selectionStart">selectionStart</code> attribute and the <code
    data-x="dom-textarea/input-selectionEnd">selectionEnd</code> attribute respectively.</p>

    <p>Otherwise, let <var>start</var>, <var>end</var> have the values of the
    second and third arguments respectively.</p>
   </li>

   <li><p>If <var>start</var> is greater than <var>end</var>, then throw an
   <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <var>start</var> is greater than the <span>length</span> of the <span
   data-x="concept-textarea/input-relevant-value">relevant value</span> of the text control, then
   set it to the <span>length</span> of the <span
   data-x="concept-textarea/input-relevant-value">relevant value</span> of the text control.</p></li>

   <li><p>If <var>end</var> is greater than the <span>length</span> of the <span
   data-x="concept-textarea/input-relevant-value">relevant value</span> of the text control, then
   set it to the <span>length</span> of the <span
   data-x="concept-textarea/input-relevant-value">relevant value</span> of the text control.</p></li>

   <li><p>Let <var>selection start</var> be the current value of the <code
   data-x="dom-textarea/input-selectionStart">selectionStart</code> attribute.</p></li>

   <li><p>Let <var>selection end</var> be the current value of the <code
   data-x="dom-textarea/input-selectionEnd">selectionEnd</code> attribute.</p></li>

   <li><p>If <var>start</var> is less than <var>end</var>, delete the sequence of <span data-x="code
   unit">code units</span> within the element's <span
   data-x="concept-textarea/input-relevant-value">relevant value</span> starting with the code unit
   at the <var>start</var>th position and ending with the code unit at the <span
   data-x="">(<var>end</var>-1)</span>th position.</p></li>

   <li><p>Insert the value of the first argument into the text of the <span
   data-x="concept-textarea/input-relevant-value">relevant value</span> of the text control,
   immediately before the <var>start</var>th <span>code unit</span>.</p></li>

   <li><p>Let <var>new length</var> be the <span>length</span> of the value of the first
   argument.</p></li>

   <li><p>Let <var>new end</var> be the sum of <var>start</var> and <var>new length</var>.</p></li>

   <li>
    <p>Run the appropriate set of substeps from the following list:</p>

    <dl class="switch">

     <dt>If the fourth argument's value is "<dfn enum-value for="SelectionMode"><code data-x="dom-SelectionMode-select">select</code></dfn>"</dt>

     <dd>
      <p>Let <var>selection start</var> be <var>start</var>.</p>

      <p>Let <var>selection end</var> be <var>new end</var>.</p>
     </dd>

     <dt>If the fourth argument's value is "<dfn enum-value for="SelectionMode"><code data-x="dom-SelectionMode-start">start</code></dfn>"</dt>

     <dd>
      <p>Let <var>selection start</var> and <var>selection end</var> be <var>start</var>.</p>
     </dd>

     <dt>If the fourth argument's value is "<dfn enum-value for="SelectionMode"><code data-x="dom-SelectionMode-end">end</code></dfn>"</dt>

     <dd>
      <p>Let <var>selection start</var> and <var>selection end</var> be <var>new end</var>.</p>
     </dd>

     <dt>If the fourth argument's value is "<dfn enum-value for="SelectionMode"><code data-x="dom-SelectionMode-preserve">preserve</code></dfn>"</dt>

     <dt>If the method has only one argument</dt>

     <dd>
      <ol>
       <li><p>Let <var>old length</var> be <var>end</var> minus <var>start</var>.</p>

       <li><p>Let <var>delta</var> be <var>new length</var> minus <var>old length</var>.</p>

       <li>
        <p>If <var>selection start</var> is greater than <var>end</var>, then
        increment it by <var>delta</var>. (If <var>delta</var> is negative, i.e.
        the new text is shorter than the old text, then this will <em>decrease</em> the value of
        <var>selection start</var>.)</p>

        <p>Otherwise: if <var>selection start</var> is greater than <var>start</var>, then set it to <var>start</var>. (This snaps the start of the
        selection to the start of the new text if it was in the middle of the text that it
        replaced.)</p>
       </li>

       <li>
        <p>If <var>selection end</var> is greater than <var>end</var>, then
        increment it by <var>delta</var> in the same way.</p>

        <p>Otherwise: if <var>selection end</var> is greater than <var>start</var>, then set it to <var>new end</var>. (This snaps the end of the
        selection to the end of the new text if it was in the middle of the text that it
        replaced.)</p>
       </li>
      </ol>
     </dd>
    </dl>
   </li>

   <li><p><span>Set the selection range</span> with <var>selection start</var> and <var>selection
   end</var>.</p></li>
  </ol>

  <p>The <code data-x="dom-textarea/input-setRangeText">setRangeText()</code> method uses the
  following enumeration:<span w-dev> "<span data-x="dom-SelectionMode-select">select</span>", "<span
  data-x="dom-SelectionMode-start">start</span>", "<span data-x="dom-SelectionMode-end">end</span>",
  "<span data-x="dom-SelectionMode-preserve">preserve</span>" (default).</span></p>

  <pre><code class="idl">enum <dfn enum>SelectionMode</dfn> {
  "<span data-x="dom-SelectionMode-select">select</span>",
  "<span data-x="dom-SelectionMode-start">start</span>",
  "<span data-x="dom-SelectionMode-end">end</span>",
  "<span data-x="dom-SelectionMode-preserve">preserve</span>" // default
};</code></pre>

  <hr>

  </div>

  <div class="example">

   <p>To obtain the currently selected text, the following JavaScript suffices:</p>

   <pre><code class="js">var selectionText = control.value.substring(control.selectionStart, control.selectionEnd);</code></pre>

   <p>...where <var>control</var> is the <code>input</code> or <code>textarea</code>
   element.</p>

  </div>

  <div class="example">

   <p>To add some text at the start of a text control, while maintaining the text selection, the
   three attributes must be preserved:</p>

   <pre><code class="js">var oldStart = control.selectionStart;
var oldEnd = control.selectionEnd;
var oldDirection = control.selectionDirection;
var prefix = "http://";
control.value = prefix + control.value;
control.setSelectionRange(oldStart + prefix.length, oldEnd + prefix.length, oldDirection);</code></pre>

   <p>...where <var>control</var> is the <code>input</code> or <code>textarea</code>
   element.</p>

  </div>



  <h4>Constraints</h4>

  <div w-nodev>

  <h5>Definitions</h5>

  <p>A <span data-x="category-submit">submittable element</span> is a <dfn>candidate for constraint
  validation</dfn> except when a condition has <dfn data-x="barred from constraint validation">barred
  the element from constraint validation</dfn>. (For example, an element is <span>barred from
  constraint validation</span> if it has a <code>datalist</code> element ancestor.)</p>

  <p>An element can have a <dfn>custom validity error message</dfn> defined. Initially, an element
  must have its <span>custom validity error message</span> set to the empty string. When its value
  is not the empty string, the element is <span>suffering from a custom error</span>. It can be set
  using the <code data-x="dom-cva-setCustomValidity">setCustomValidity()</code> method, except for
  <span data-x="form-associated custom element">form-associated custom elements</span>. <span
  data-x="form-associated custom element">Form-associated custom elements</span> can have a
  <span>custom validity error message</span> set via their <code>ElementInternals</code> object's
  <code data-x="dom-ElementInternals-setValidity">setValidity()</code> method. The user agent should use the
  <span>custom validity error message</span> when alerting the user to the problem with the
  control.</p>

  <p>An element can be constrained in various ways. The following is the list of <dfn>validity
  states</dfn> that a form control can be in, making the control invalid for the purposes of
  constraint validation. (The definitions below are non-normative; other parts of this specification
  define more precisely when each state applies or does not.)</p>

  <dl>
   <dt> <dfn>Suffering from being missing</dfn> </dt>

   <dd>
    <p>When a control has no <span data-x="concept-fe-value">value</span> but has a <code
    data-x="">required</code> attribute (<code>input</code> <code
    data-x="attr-input-required">required</code>, <code>textarea</code> <code
    data-x="attr-textarea-required">required</code>); or, more complicated rules for
    <code>select</code> elements and controls in <span data-x="radio button group">radio button
    groups</span>, as specified in their sections.</p>

    <p>When the <code data-x="dom-ElementInternals-setValidity">setValidity()</code> method sets
    <code data-x="">valueMissing</code> flag to true for a
    <span>form-associated custom element</span>.</p>
   </dd>

   <dt> <dfn>Suffering from a type mismatch</dfn> </dt>

   <dd>
    <p>When a control that allows arbitrary user input has a <span
    data-x="concept-fe-value">value</span> that is not in the correct syntax (<span
    data-x="attr-input-type-email">Email</span>, <span data-x="attr-input-type-url">URL</span>).
    </p>

    <p>When the <code data-x="dom-ElementInternals-setValidity">setValidity()</code> method sets
    <code data-x="">typeMismatch</code> flag to true for a
    <span>form-associated custom element</span>.</p>
   </dd>

   <dt> <dfn>Suffering from a pattern mismatch</dfn> </dt>

   <dd>
    <p>When a control has a <span data-x="concept-fe-value">value</span> that doesn't satisfy the
    <code data-x="attr-input-pattern">pattern</code> attribute.</p>

    <p>When the <code data-x="dom-ElementInternals-setValidity">setValidity()</code> method sets
    <code data-x="">patternMismatch</code> flag to true for a
    <span>form-associated custom element</span>.</p>
   </dd>

   <dt> <dfn>Suffering from being too long</dfn> </dt>

   <dd>
    <p>When a control has a <span data-x="concept-fe-value">value</span> that is too long for the
    <span data-x="attr-fe-maxlength">form control <code data-x="">maxlength</code> attribute</span>
    (<code>input</code> <code data-x="attr-input-maxlength">maxlength</code>, <code>textarea</code>
    <code data-x="attr-textarea-maxlength">maxlength</code>). </p>

    <p>When the <code data-x="dom-ElementInternals-setValidity">setValidity()</code> method sets
    <code data-x="">tooLong</code> flag to true for a
    <span>form-associated custom element</span>.</p>
   </dd>

   <dt> <dfn>Suffering from being too short</dfn> </dt>

   <dd>
    <p>When a control has a <span data-x="concept-fe-value">value</span> that is too short for the
    <span data-x="attr-fe-minlength">form control <code data-x="">minlength</code> attribute</span>
    (<code>input</code> <code data-x="attr-input-minlength">minlength</code>, <code>textarea</code>
    <code data-x="attr-textarea-minlength">minlength</code>). </p>

    <p>When the <code data-x="dom-ElementInternals-setValidity">setValidity()</code> method sets
    <code data-x="">tooShort</code> flag to true for a
    <span>form-associated custom element</span>.</p>
   </dd>

   <dt> <dfn>Suffering from an underflow</dfn> </dt>

   <dd>
    <p>When a control has a <span data-x="concept-fe-value">value</span> that is not the empty
    string and is too low for the <code data-x="attr-input-min">min</code> attribute.</p>

    <p>When the <code data-x="dom-ElementInternals-setValidity">setValidity()</code> method sets
    <code data-x="">rangeUnderflow</code> flag to true for a
    <span>form-associated custom element</span>.</p>
  </dd>

   <dt> <dfn>Suffering from an overflow</dfn> </dt>

   <dd>
    <p>When a control has a <span data-x="concept-fe-value">value</span> that is not the empty
    string and is too high for the <code data-x="attr-input-max">max</code> attribute.</p>

    <p>When the <code data-x="dom-ElementInternals-setValidity">setValidity()</code> method sets
    <code data-x="">rangeOverflow</code> flag to true for a
    <span>form-associated custom element</span>.</p>
  </dd>

   <dt> <dfn>Suffering from a step mismatch</dfn> </dt>

   <dd>
    <p>When a control has a <span data-x="concept-fe-value">value</span> that doesn't fit the
    rules given by the <code data-x="attr-input-step">step</code> attribute.</p>

    <p>When the <code data-x="dom-ElementInternals-setValidity">setValidity()</code> method sets
    <code data-x="">stepMismatch</code> flag to true for a
    <span>form-associated custom element</span>.</p>
  </dd>

   <dt> <dfn>Suffering from bad input</dfn> </dt>

   <dd>
    <p>When a control has incomplete input and the user agent does not think the user ought to
    be able to submit the form in its current state.</p>

    <p>When the <code data-x="dom-ElementInternals-setValidity">setValidity()</code> method sets
    <code data-x="">badInput</code> flag to true for a <span>form-associated custom element</span>.
    </p>
   </dd>

   <dt> <dfn>Suffering from a custom error</dfn> </dt>

   <dd> <p>When a control's <span>custom validity error message</span> (as set by the element's
   <code data-x="dom-cva-setCustomValidity">setCustomValidity()</code> method or
   <code>ElementInternals</code>'s <code data-x="dom-ElementInternals-setValidity">setValidity()</code> method) is
   not the empty string.</p> </dd>
  </dl>

  <p class="note">An element can still suffer from these states even when the element is <span
  data-x="concept-fe-disabled">disabled</span>; thus these states can be represented in the DOM even
  if validating the form during submission wouldn't indicate a problem to the user.</p>

  <p>An element <dfn data-x="concept-fv-valid">satisfies its constraints</dfn> if it is not suffering
  from any of the above <span>validity states</span>.</p>

  </div>



  <div w-nodev>

  <h5>Constraint validation</h5>

  <p>When the user agent is required to <dfn>statically validate the constraints</dfn> of
  <code>form</code> element <var>form</var>, it must run the following steps, which return
  either a <i>positive</i> result (all the controls in the form are valid) or a <i>negative</i>
  result (there are invalid controls) along with a (possibly empty) list of elements that are
  invalid and for which no script has claimed responsibility:</p>

  <ol>
   <li><p>Let <var>controls</var> be a list of all the <span
   data-x="category-submit">submittable elements</span> whose <span>form owner</span> is <var>form</var>, in <span>tree order</span>.</p></li>

   <li><p>Let <var>invalid controls</var> be an initially empty list of elements.</p></li>

   <li>
    <p>For each element <var>field</var> in <var>controls</var>, in <span>tree order</span>:</p>

    <ol>
     <li><p>If <var>field</var> is not a <span>candidate for constraint validation</span>,
     then move on to the next element.</p></li>

     <li><p>Otherwise, if <var>field</var> <span data-x="concept-fv-valid">satisfies its
     constraints</span>, then move on to the next element.</p></li>

     <li><p>Otherwise, add <var>field</var> to <var>invalid
     controls</var>.</p></li>
    </ol>
   </li>

   <li><p>If <var>invalid controls</var> is empty, then return a <i>positive</i> result.</p></li>

   <li><p>Let <var>unhandled invalid controls</var> be an initially empty list of
   elements.</p></li>

   <li>
    <p>For each element <var>field</var> in <var>invalid controls</var>, if any, in <span>tree
    order</span>:</p>

    <ol>
     <li><p>Let <var>notCanceled</var> be the result of <span data-x="concept-event-fire">firing an
     event</span> named <code data-x="event-invalid">invalid</code> at <var>field</var>, with the
     <code data-x="dom-Event-cancelable">cancelable</code> attribute initialized to true.</p></li>

     <li><p>If <var>notCanceled</var> is true, then add <var>field</var> to <var>unhandled invalid
     controls</var>.</p></li>
    </ol>
   </li>

   <li><p>Return a <i>negative</i> result with the list of elements in the <var>unhandled
   invalid controls</var> list.</p></li>
  </ol>

  <p>If a user agent is to <dfn>interactively validate the constraints</dfn> of <code>form</code>
  element <var>form</var>, then the user agent must run the following steps:</p>

  <ol>
   <li><p><span>Statically validate the constraints</span> of <var>form</var>, and let
   <var>unhandled invalid controls</var> be the list of elements returned if the result was
   <i>negative</i>.</p></li>

   <li><p>If the result was <i>positive</i>, then return that result.</p></li>

   <li>
    <p>Report the problems with the constraints of at least one of the elements given in
    <var>unhandled invalid controls</var> to the user.</p>

    <ul>
     <li><p>User agents may focus one of those elements in the process, by running the
     <span>focusing steps</span> for that element, and may change the scrolling position of the
     document, or perform some other action that brings the element to the user's attention. For
     elements that are <span data-x="form-associated custom element">form-associated custom
     elements</span>, user agents should use their <span data-x="face-validation-anchor">validation
     anchor</span> instead, for the purposes of these actions.</p></li>

     <li><p>User agents may report more than one constraint violation.</p></li>

     <li><p>User agents may coalesce related constraint violation reports if appropriate (e.g. if
     multiple radio buttons in a <span data-x="radio button group">group</span> are marked as
     required, only one error need be reported).</p></li>

     <li><p>If one of the controls is not <span>being rendered</span> (e.g. it has the <code
     data-x="attr-hidden">hidden</code> attribute set) then user agents may report a script
     error.</p></li>
    </ul>
   </li>

   <li><p>Return a <i>negative</i> result.</p></li>
  </ol>

  </div>



  <h5>The <dfn>constraint validation API</dfn></h5>

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-cva-willValidate">willValidate</span></code></dt>

   <dd>
    <p>Returns true if the element will be validated when the form is submitted; false
    otherwise.</p>
   </dd>

   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-cva-setCustomValidity">setCustomValidity</span>(<var>message</var>)</code></dt>

   <dd>
    <p>Sets a custom error, so that the element would fail to validate. The given message is the
    message to be shown to the user when reporting the problem to the user.</p>

    <p>If the argument is the empty string, clears the custom error.</p>
   </dd>

   <dt><code data-x=""><var>element</var>.<span data-x="dom-cva-validity">validity</span>.<span subdfn data-x="dom-validitystate-valueMissing">valueMissing</span></code></dt>

   <dd><p>Returns true if the element has no value but is a required field; false
   otherwise.</p></dd>

   <dt><code data-x=""><var>element</var>.<span data-x="dom-cva-validity">validity</span>.<span subdfn data-x="dom-validitystate-typeMismatch">typeMismatch</span></code></dt>

   <dd><p>Returns true if the element's value is not in the correct syntax; false
   otherwise.</p></dd>

   <dt><code data-x=""><var>element</var>.<span data-x="dom-cva-validity">validity</span>.<span subdfn data-x="dom-validitystate-patternMismatch">patternMismatch</span></code></dt>

   <dd><p>Returns true if the element's value doesn't match the provided pattern; false
   otherwise.</p></dd>

   <dt><code data-x=""><var>element</var>.<span data-x="dom-cva-validity">validity</span>.<span subdfn data-x="dom-validitystate-tooLong">tooLong</span></code></dt>

   <dd>
    <p>Returns true if the element's value is longer than the provided maximum length; false
    otherwise.</p>
   </dd>

   <dt><code data-x=""><var>element</var>.<span data-x="dom-cva-validity">validity</span>.<span subdfn data-x="dom-validitystate-tooShort">tooShort</span></code></dt>

   <dd>
    <p>Returns true if the element's value, if it is not the empty string, is shorter than the
    provided minimum length; false otherwise.</p>
   </dd>

   <dt><code data-x=""><var>element</var>.<span data-x="dom-cva-validity">validity</span>.<span subdfn data-x="dom-validitystate-rangeUnderflow">rangeUnderflow</span></code></dt>

   <dd><p>Returns true if the element's value is lower than the provided minimum; false
   otherwise.</p></dd>

   <dt><code data-x=""><var>element</var>.<span data-x="dom-cva-validity">validity</span>.<span subdfn data-x="dom-validitystate-rangeOverflow">rangeOverflow</span></code></dt>

   <dd><p>Returns true if the element's value is higher than the provided maximum; false
   otherwise.</p></dd>

   <dt><code data-x=""><var>element</var>.<span data-x="dom-cva-validity">validity</span>.<span subdfn data-x="dom-validitystate-stepMismatch">stepMismatch</span></code></dt>

   <dd><p>Returns true if the element's value doesn't fit the rules given by the <code
   data-x="attr-input-step">step</code> attribute; false otherwise.</p></dd>

   <dt><code data-x=""><var>element</var>.<span data-x="dom-cva-validity">validity</span>.<span subdfn data-x="dom-validitystate-badInput">badInput</span></code></dt>

   <dd>
    <p>Returns true if the user has provided input in the user interface that the user agent is
    unable to convert to a value; false otherwise.</p>
   </dd>

   <dt><code data-x=""><var>element</var>.<span data-x="dom-cva-validity">validity</span>.<span subdfn data-x="dom-validitystate-customError">customError</span></code></dt>

   <dd><p>Returns true if the element has a custom error; false otherwise.</p></dd>

   <dt><code data-x=""><var>element</var>.<span data-x="dom-cva-validity">validity</span>.<span subdfn data-x="dom-validitystate-valid">valid</span></code></dt>

   <dd><p>Returns true if the element's value has no validity problems; false otherwise.</p></dd>

   <dt><code data-x=""><var>valid</var> = <var>element</var>.<span subdfn data-x="dom-cva-checkValidity">checkValidity</span>()</code></dt>

   <dd><p>Returns true if the element's value has no validity problems; false otherwise. Fires an
   <code data-x="event-invalid">invalid</code> event at the element in the latter case.</p></dd>

   <dt><code data-x=""><var>valid</var> = <var>element</var>.<span subdfn data-x="dom-cva-reportValidity">reportValidity</span>()</code></dt>

   <dd>
    <p>Returns true if the element's value has no validity problems; otherwise, returns false, fires
    an <code data-x="event-invalid">invalid</code> event at the element, and (if the event isn't
    canceled) reports the problem to the user.</p>
   </dd>

   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-cva-validationMessage">validationMessage</span></code></dt>

   <dd>
    <p>Returns the error message that would be shown to the user if the element was to be checked
    for validity.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute
  for="HTMLObjectElement,HTMLInputElement,HTMLButtonElement,HTMLSelectElement,HTMLTextAreaElement,HTMLOutputElement,HTMLFieldSetElement"><code
  data-x="dom-cva-willValidate">willValidate</code></dfn> attribute's getter must return true, if
  this element is a <span>candidate for constraint validation</span>, and false otherwise (i.e.,
  false if any conditions are <span data-x="barred from constraint validation">barring it from
  constraint validation</span>).</p>

  <p>The <dfn attribute for="ElementInternals"><code
  data-x="dom-ElementInternals-willValidate">willValidate</code></dfn> attribute of
  <code>ElementInternals</code> interface, on getting, must throw a
  <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code> if the <span
  data-x="internals-target">target element</span> is not a <span>form-associated custom
  element</span>. Otherwise, it must return true if the <span data-x="internals-target">target
  element</span> is a <span>candidate for constraint validation</span>, and false otherwise.</p>

  <p>The <dfn method
  for="HTMLObjectElement,HTMLInputElement,HTMLButtonElement,HTMLSelectElement,HTMLTextAreaElement,HTMLOutputElement,HTMLFieldSetElement"><code
  data-x="dom-cva-setCustomValidity">setCustomValidity(<var>error</var>)</code></dfn> method steps
  are:</p>

  <ol>
   <li><p>Set <var>error</var> to the result of <span data-x="normalize newlines">normalizing
   newlines</span> given <var>error</var>.</p></li>

   <li><p>Set the <span>custom validity error message</span> to <var>error</var>.</p></li>
  </ol>

  </div>

  <div class="example">

   <p>In the following example, a script checks the value of a form control each time it is edited,
   and whenever it is not a valid value, uses the <code
   data-x="dom-cva-setCustomValidity">setCustomValidity()</code> method to set an appropriate
   message.</p>

   <pre><code class="html">&lt;label>Feeling: &lt;input name=f type="text" oninput="check(this)">&lt;/label>
&lt;script>
 function check(input) {
   if (input.value == "good" ||
       input.value == "fine" ||
       input.value == "tired") {
     input.setCustomValidity('"' + input.value + '" is not a feeling.');
<!--   } else if (input.value == "...") {
     input.setCustomValidity('...');
-->   } else {
     // input is fine -- reset the error message
     input.setCustomValidity('');
   }
 }
&lt;/script></code></pre>

  </div>

  <div w-nodev>

  <p>The <dfn attribute
  for="HTMLObjectElement,HTMLInputElement,HTMLButtonElement,HTMLSelectElement,HTMLTextAreaElement,HTMLOutputElement,HTMLFieldSetElement"><code
  data-x="dom-cva-validity">validity</code></dfn> attribute's getter must return a
  <code>ValidityState</code> object that represents the <span>validity states</span> of this
  element. This object is <span>live</span>.</p>

  <p>The <dfn attribute for="ElementInternals"><code
  data-x="dom-ElementInternals-validity">validity</code></dfn> attribute of
  <code>ElementInternals</code> interface, on getting, must throw a
  <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code> if the <span
  data-x="internals-target">target element</span> is not a <span>form-associated custom
  element</span>. Otherwise, it must return a <code>ValidityState</code> object that represents the
  <span>validity states</span> of the <span data-x="internals-target">target element</span>. This
  object is <span>live</span>.</p>

<pre><code class="idl">[Exposed=Window]
interface <dfn interface>ValidityState</dfn> {
  readonly attribute boolean <span data-x="dom-ValidityState-valueMissing">valueMissing</span>;
  readonly attribute boolean <span data-x="dom-ValidityState-typeMismatch">typeMismatch</span>;
  readonly attribute boolean <span data-x="dom-ValidityState-patternMismatch">patternMismatch</span>;
  readonly attribute boolean <span data-x="dom-ValidityState-tooLong">tooLong</span>;
  readonly attribute boolean <span data-x="dom-ValidityState-tooShort">tooShort</span>;
  readonly attribute boolean <span data-x="dom-ValidityState-rangeUnderflow">rangeUnderflow</span>;
  readonly attribute boolean <span data-x="dom-ValidityState-rangeOverflow">rangeOverflow</span>;
  readonly attribute boolean <span data-x="dom-ValidityState-stepMismatch">stepMismatch</span>;
  readonly attribute boolean <span data-x="dom-ValidityState-badInput">badInput</span>;
  readonly attribute boolean <span data-x="dom-ValidityState-customError">customError</span>;
  readonly attribute boolean <span data-x="dom-ValidityState-valid">valid</span>;
};</code></pre>

  <p>A <code>ValidityState</code> object has the following attributes. On getting, they must return
  true if the corresponding condition given in the following list is true, and false otherwise.</p>

  <dl>
   <dt><dfn attribute for="ValidityState"><code data-x="dom-ValidityState-valueMissing">valueMissing</code></dfn></dt>
   <dd> <p>The control is <span>suffering from being missing</span>.</p> </dd>

   <dt><dfn attribute for="ValidityState"><code data-x="dom-ValidityState-typeMismatch">typeMismatch</code></dfn></dt>
   <dd> <p>The control is <span>suffering from a type mismatch</span>.</p> </dd>

   <dt><dfn attribute for="ValidityState"><code data-x="dom-ValidityState-patternMismatch">patternMismatch</code></dfn></dt>
   <dd> <p>The control is <span>suffering from a pattern mismatch</span>.</p> </dd>

   <dt><dfn attribute for="ValidityState"><code data-x="dom-ValidityState-tooLong">tooLong</code></dfn></dt>
   <dd> <p>The control is <span>suffering from being too long</span>.</p> </dd>

   <dt><dfn attribute for="ValidityState"><code data-x="dom-ValidityState-tooShort">tooShort</code></dfn></dt>
   <dd> <p>The control is <span>suffering from being too short</span>.</p> </dd>

   <dt><dfn attribute for="ValidityState"><code data-x="dom-ValidityState-rangeUnderflow">rangeUnderflow</code></dfn></dt>
   <dd> <p>The control is <span>suffering from an underflow</span>.</p> </dd>

   <dt><dfn attribute for="ValidityState"><code data-x="dom-ValidityState-rangeOverflow">rangeOverflow</code></dfn></dt>
   <dd> <p>The control is <span>suffering from an overflow</span>.</p> </dd>

   <dt><dfn attribute for="ValidityState"><code data-x="dom-ValidityState-stepMismatch">stepMismatch</code></dfn></dt>
   <dd> <p>The control is <span>suffering from a step mismatch</span>.</p> </dd>

   <dt><dfn attribute for="ValidityState"><code data-x="dom-ValidityState-badInput">badInput</code></dfn></dt>
   <dd> <p>The control is <span>suffering from bad input</span>.</p> </dd>

   <dt><dfn attribute for="ValidityState"><code data-x="dom-ValidityState-customError">customError</code></dfn></dt>
   <dd> <p>The control is <span>suffering from a custom error</span>.</p> </dd>

   <dt><dfn attribute for="ValidityState"><code data-x="dom-ValidityState-valid">valid</code></dfn></dt>
   <dd> <p>None of the other conditions are true.</p> </dd>
  </dl>

  <p>The <dfn>check validity steps</dfn> for an element <var>element</var> are:</p>

  <ol>
   <li>
    <p>If <var>element</var> is a <span>candidate for constraint validation</span> and does not
    <span data-x="concept-fv-valid">satisfy its constraints</span>, then:</p>

    <ol>
     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-invalid">invalid</code> at <var>element</var>, with the <code
     data-x="dom-Event-cancelable">cancelable</code> attribute initialized to true (though canceling
     has no effect).</p></li>

     <li><p>Return false.</p></li>
    </ol>

   <li><p>Return true.</p></li>
  </ol>

  <p>The <dfn method
  for="HTMLObjectElement,HTMLInputElement,HTMLButtonElement,HTMLSelectElement,HTMLTextAreaElement,HTMLOutputElement,HTMLFieldSetElement"><code
  data-x="dom-cva-checkValidity">checkValidity()</code></dfn> method, when invoked, must run the
  <span>check validity steps</span> on this element.</p>

  <p>The <dfn method for="ElementInternals"><code
  data-x="dom-ElementInternals-checkValidity">checkValidity()</code></dfn> method of the
  <code>ElementInternals</code> interface must run these steps:</p>

  <ol>
   <li><p>Let <var>element</var> be this <code>ElementInternals</code>'s <span
   data-x="internals-target">target element</span>.</p></li>

   <li><p>If <var>element</var> is not a <span>form-associated custom element</span>, then throw a
   <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Run the <span>check validity steps</span> on <var>element</var>.</p></li>
  </ol>

  <p>The <dfn>report validity steps</dfn> for an element <var>element</var> are:</p>

  <ol>
   <li>
    <p>If <var>element</var> is a <span>candidate for constraint validation</span> and does not
    <span data-x="concept-fv-valid">satisfy its constraints</span>, then:

    <ol>
     <li><p>Let <var>report</var> be the result of <span data-x="concept-event-fire">firing an
     event</span> named <code data-x="event-invalid">invalid</code> at <var>element</var>, with the
     <code data-x="dom-Event-cancelable">cancelable</code> attribute initialized to true.</p></li>

     <li><p>If <var>report</var> is true, then report the problems with the constraints of this
     element to the user. When reporting the problem with the constraints to the user, the user
     agent may run the <span>focusing steps</span> for <var>element</var>, and may change the
     scrolling position of the document, or perform some other action that brings <var>element</var>
     to the user's attention. User agents may report more than one constraint violation, if
     <var>element</var> suffers from multiple problems at once.</p>

     <li><p>Return false.</p></li>
    </ol>

   <li><p>Return true.</p></li>
  </ol>

  <p>The <dfn method
  for="HTMLObjectElement,HTMLInputElement,HTMLButtonElement,HTMLSelectElement,HTMLTextAreaElement,HTMLOutputElement,HTMLFieldSetElement"><code
  data-x="dom-cva-reportValidity">reportValidity()</code></dfn> method, when invoked, must run the
  <span>report validity steps</span> on this element.</p>

  <p>The <dfn method for="ElementInternals"><code
  data-x="dom-ElementInternals-reportValidity">reportValidity()</code></dfn> method of the
  <code>ElementInternals</code> interface must run these steps:</p>

  <ol>
   <li><p>Let <var>element</var> be this <code>ElementInternals</code>'s <span
   data-x="internals-target">target element</span>.</p></li>

   <li><p>If <var>element</var> is not a <span>form-associated custom element</span>, then throw a
   <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Run the <span>report validity steps</span> on <var>element</var>.</p></li>
  </ol>

  <p>The <dfn attribute
  for="HTMLObjectElement,HTMLInputElement,HTMLButtonElement,HTMLSelectElement,HTMLTextAreaElement,HTMLOutputElement,HTMLFieldSetElement"><code
  data-x="dom-cva-validationMessage">validationMessage</code></dfn> attribute's getter must run
  these steps:</p>

  <ol>
   <li><p>If this element is not a <span>candidate for constraint validation</span> or if this
   element <span data-x="concept-fv-valid">satisfies its constraints</span>, then return the empty
   string.</p></li>

   <li><p>Return a suitably localized message that the user agent would show the user if this were
   the only form control with a validity constraint problem. If the user agent would not actually
   show a textual message in such a situation (e.g., it would show a graphical cue instead), then
   return a suitably localized message that expresses (one or more of) the validity constraint(s)
   that the control does not satisfy. If the element is a <span>candidate for constraint
   validation</span> and is <span>suffering from a custom error</span>, then the
   <span>custom validity error message</span> should be present in the return value.</p></li>
  </ol>

  </div>


  <h5 id="security-forms">Security</h5>

  <p id="security-0">Servers should not rely on client-side validation. Client-side validation can
  be intentionally bypassed by hostile users, and unintentionally bypassed by users of older user
  agents or automated tools that do not implement these features. The constraint validation features
  are only intended to improve the user experience, not to provide any kind of security
  mechanism.</p>




  <h4 id="form-submission-2"><dfn id="form-submission">Form submission</dfn></h4>

  <div w-nodev>

  <h5>Introduction</h5>

  </div>

  <!-- NON-NORMATIVE SECTION -->

  <p>When a form is submitted, the data in the form is converted into the structure specified by the
  <span data-x="concept-fs-enctype">enctype</span>, and then sent to the destination specified by the
  <span data-x="concept-fs-action">action</span> using the given <span
  data-x="concept-fs-method">method</span>.</p>

  <p>For example, take the following form:</p>

  <pre><code class="html">&lt;form action="/find.cgi" method=get>
 &lt;input type=text name=t>
 &lt;input type=search name=q>
 &lt;input type=submit>
&lt;/form></code></pre>

  <p>If the user types in "cats" in the first field and "fur" in the second, and then hits the
  submit button, then the user agent will load <code data-x="">/find.cgi?t=cats&amp;q=fur</code>.</p>

  <p>On the other hand, consider this form:</p>

  <pre><code class="html">&lt;form action="/find.cgi" method=post enctype="multipart/form-data">
 &lt;input type=text name=t>
 &lt;input type=search name=q>
 &lt;input type=submit>
&lt;/form></code></pre>

  <p>Given the same user input, the result on submission is quite different: the user agent instead
  does an HTTP POST to the given URL, with as the entity body something like the following text:</p>

  <pre>------kYFrd4jNJEgCervE
Content-Disposition: form-data; name="t"

cats
------kYFrd4jNJEgCervE
Content-Disposition: form-data; name="q"

fur
------kYFrd4jNJEgCervE--</pre>



  <div w-nodev>

  <h5>Implicit submission</h5>

  <p>A <code>form</code> element's <dfn>default button</dfn> is the first <span
  data-x="concept-submit-button">submit button</span> in <span>tree order</span> whose <span>form
  owner</span> is that <code>form</code> element.</p>

  <p>If the user agent supports letting the user submit a form implicitly (for example, on some
  platforms hitting the "enter" key while a text control is <span>focused</span> implicitly submits
  the form), then doing so for a form, whose <span>default button</span> has <span>activation
  behavior</span> and is not <span data-x="concept-fe-disabled">disabled</span>, must cause the user
  agent to <span>fire a <code data-x="event-click">click</code> event</span> at that <span>default
  button</span>.</p>

  <p class="note">There are pages on the web that are only usable if there is a way to implicitly
  submit forms, so user agents are strongly encouraged to support this.</p>

  <p><!-- For web compatibility reasons caused by obscure historical accidents, -->If the form has
  no <span data-x="concept-submit-button">submit button</span>, then the implicit submission
  mechanism must perform the following steps:

  <ol>
   <li><p>If the form has more than one <span>field that blocks implicit submission</span>, then
   return.</p></li>

   <li><p><span data-x="concept-form-submit">Submit</span> the <code>form</code> element from the
   <code>form</code> element itself with <i data-x="submit-user-involvement">userInvolvement</i> set
   to "<code data-x="uni-activation">activation</code>".</p></li>
  </ol>

  <p>For the purpose of the previous paragraph, an element is a <dfn>field that blocks implicit
  submission</dfn> of a <code>form</code> element if it is an <code>input</code> element whose
  <span>form owner</span> is that <code>form</code> element and whose <code
  data-x="attr-input-type">type</code> attribute is in one of the following states:
  <span data-x="attr-input-type-text">Text</span>,
  <span data-x="attr-input-type-search">Search</span>,
  <span data-x="attr-input-type-tel">Telephone</span>,
  <span data-x="attr-input-type-url">URL</span>,
  <span data-x="attr-input-type-email">Email</span>,
  <span data-x="attr-input-type-password">Password</span>,
  <span data-x="attr-input-type-date">Date</span>,
  <span data-x="attr-input-type-month">Month</span>,
  <span data-x="attr-input-type-week">Week</span>,
  <span data-x="attr-input-type-time">Time</span>,
  <span data-x="attr-input-type-datetime-local">Local Date and Time</span>,
  <span data-x="attr-input-type-number">Number</span>
  </p>

  </div>


  <div w-nodev>

  <h5>Form submission algorithm</h5>

  <p>Each <code>form</code> element has a <dfn>constructing entry list</dfn> boolean, initially
  false.</p>

  <p>Each <code>form</code> element has a <dfn>firing submission events</dfn> boolean, initially
  false.</p>

  <p>To <dfn data-x="concept-form-submit">submit</dfn> a <code>form</code> element <var>form</var>
  from an element <var>submitter</var> (typically a button), given an optional boolean <dfn
  data-x="submit-subbmitted-from-method"><var>submitted from <code
  data-x="dom-form-submit">submit()</code> method</var></dfn> (default false) and an optional
  <span>user navigation involvement</span> <dfn
  data-x="submit-user-involvement"><var>userInvolvement</var></dfn> (default "<code
  data-x="uni-none">none</code>"):</p>

  <ol>
   <li><p>If <var>form</var> <span>cannot navigate</span>, then return.</p></li>

   <li><p>If <var>form</var>'s <span>constructing entry list</span> is true, then return.</p></li>

   <li><p>Let <var>form document</var> be <var>form</var>'s <span>node document</span>.</p></li>

   <li id="sandboxSubmitBlocked"><p>If <var>form document</var>'s <span>active sandboxing flag
   set</span> has its <span>sandboxed forms browsing context flag</span> set, then return.</p></li>

   <li>
    <p>If <var>submitted from <code data-x="dom-form-submit">submit()</code> method</var> is false,
    then:</p>

    <ol>
     <li><p>If <var>form</var>'s <span>firing submission events</span> is true, then
     return.</p></li>

     <li><p>Set <var>form</var>'s <span>firing submission events</span> to true.</p></li>

     <li><p>For each element <var>field</var> in the list of
     <span data-x="category-submit">submittable elements</span> whose <span>form owner</span> is
     <var>form</var>, set <var>field</var>'s <span>user validity</span> to true.

     <li>
      <p>If the <var>submitter</var> element's <span data-x="concept-fs-novalidate">no-validate
      state</span> is false, then <span>interactively validate the constraints</span> of
      <var>form</var> and examine the result. If the result is negative (i.e., the constraint
      validation concluded that there were invalid fields and probably informed the user of this),
      then:</p>

      <ol>
       <li><p>Set <var>form</var>'s <span>firing submission events</span> to false.</p></li>

       <li><p>Return.</p></li>
      </ol>
     </li>

     <li><p>Let <var>submitterButton</var> be null if <var>submitter</var> is <var>form</var>.
     Otherwise, let <var>submitterButton</var> be <var>submitter</var>.</p></li>

     <li><p>Let <var>shouldContinue</var> be the result of <span data-x="concept-event-fire">firing an
     event</span> named <code data-x="event-submit">submit</code> at <var>form</var> using
     <code>SubmitEvent</code>, with the <code data-x="dom-SubmitEvent-submitter">submitter</code>
     attribute initialized to <var>submitterButton</var>, the <code
     data-x="dom-Event-bubbles">bubbles</code> attribute initialized to true, and the <code
     data-x="dom-Event-cancelable">cancelable</code> attribute initialized to true.</p></li>

     <li><p>Set <var>form</var>'s <span>firing submission events</span> to false.</p></li>

     <li><p>If <var>shouldContinue</var> is false, then return.</p></li>

     <li>
      <p>If <var>form</var> <span>cannot navigate</span>, then return.</p>

      <p class="note"><span>Cannot navigate</span> is run again as dispatching the <code
      data-x="event-submit">submit</code> event could have changed the outcome.</p>
     </li>
    </ol>
   </li>

   <li><p>Let <var>encoding</var> be the result of <span>picking an encoding for the
   form</span>.</p>

   <li><p>Let <var>entry list</var> be the result of <span>constructing the entry list</span> with
   <var>form</var>, <var>submitter</var>, and <var>encoding</var>.</p></li>

   <li><p><span>Assert</span>: <var>entry list</var> is not null.</p></li>

   <li>
    <p>If <var>form</var> <span>cannot navigate</span>, then return.</p>

    <p class="note"><span>Cannot navigate</span> is run again as dispatching the <code
    data-x="event-formdata">formdata</code> event in <span>constructing the entry list</span>
    could have changed the outcome.</p>
   </li>

   <li><p>Let <var>method</var> be the <var>submitter</var> element's <span
   data-x="concept-fs-method">method</span>.</p></li>

   <li>
    <p>If <var>method</var> is <span data-x="attr-fs-method-dialog">dialog</span>, then:

    <ol>
     <li><p>If <var>form</var> does not have an ancestor <code>dialog</code> element, then
     return.</p></li>

     <li><p>Let <var>subject</var> be <var>form</var>'s nearest ancestor <code>dialog</code>
     element.</p></li>

     <li><p>Let <var>result</var> be null.</p></li>

     <li>
      <p>If <var>submitter</var> is an <code>input</code> element whose <code
      data-x="attr-input-type">type</code> attribute is in the <span
      data-x="attr-input-type-image">Image Button</span> state, then:</p>

      <ol>
       <li><p>Let (<var>x</var>, <var>y</var>) be the <span
       data-x="concept-input-type-image-coordinate">selected coordinate</span>.</p></li>

       <li><p>Set <var>result</var> to the concatenation of <var>x</var>, "<code
       data-x="">,</code>", and <var>y</var>.</p></li>
      </ol>
     </li>

     <li><p>Otherwise, if <var>submitter</var> has a <span data-x="concept-fe-value">value</span>,
     then set <var>result</var> to that <span data-x="concept-fe-value">value</span>.</p></li>

     <li><p><span>Close the dialog</span> <var>subject</var> with <var>result</var>.</p></li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li><p>Let <var>action</var> be the <var>submitter</var> element's <span
   data-x="concept-fs-action">action</span>.</p></li>

   <li>
    <p>If <var>action</var> is the empty string, let <var>action</var> be the <span
    data-x="concept-document-url">URL</span> of the <var>form document</var>.</p>

    <!-- Don't ask me why. But that's what IE does. It even treats action="" differently from
    action=" " or action="#" (the latter two resolve to the base URL, the first one resolves to the
    doc URL). And other browsers concur. It is even required, see e.g.
      https://bugs.webkit.org/show_bug.cgi?id=7763
      https://bugzilla.mozilla.org/show_bug.cgi?id=297761
    -->
   </li>

   <li><p>Let <var>parsed action</var> be the result of <span>encoding-parsing a URL</span> given
   <var>action</var>, relative to <var>submitter</var>'s <span>node document</span>.</p></li>

   <li><p>If <var>parsed action</var> is failure, then return.</p></li>

   <li><p>Let <var>scheme</var> be the <span data-x="concept-url-scheme">scheme</span> of
   <var>parsed action</var>.</p></li>

   <li><p>Let <var>enctype</var> be the <var>submitter</var> element's <span
   data-x="concept-fs-enctype">enctype</span>.</p></li>

   <li><p>Let <var>formTarget</var> be null.</p></li>

   <li><p>If the <var>submitter</var> element is a <span data-x="concept-submit-button">submit
   button</span> and it has a <code data-x="attr-fs-formtarget">formtarget</code> attribute, then
   set <var>formTarget</var> to the <code data-x="attr-fs-formtarget">formtarget</code> attribute
   value.</p></li>

   <li><p>Let <var>target</var> be the result of <span data-x="get an element's target">getting an
   element's target</span> given <var>submitter</var>'s <span>form owner</span> and
   <var>formTarget</var>.</p></li>

   <li><p>Let <var>noopener</var> be the result of <span data-x="get an element's noopener">getting
   an element's noopener</span> with <var>form</var>, <var>parsed action</var>, and
   <var>target</var>.</p></li>

   <li><p>Let <var>targetNavigable</var> be the first return value of applying <span>the rules for
   choosing a navigable</span> given <var>target</var>, <var>form</var>'s <span>node
   navigable</span>, and <var>noopener</var>.</p></li>

   <li><p>If <var>targetNavigable</var> is null, then return.</p></li>

   <li><p>Let <var>historyHandling</var> be "<code
   data-x="NavigationHistoryBehavior-auto">auto</code>".</p></li>

   <li><p>If <var>form document</var> equals <var>targetNavigable</var>'s <span
   data-x="nav-document">active document</span>, and <var>form document</var> has not yet
   <span>completely loaded</span>, then set <var>historyHandling</var> to "<code
   data-x="NavigationHistoryBehavior-replace">replace</code>".</p></li>

   <li>
    <p>Select the appropriate row in the table below based on <var>scheme</var> as given by the
    first cell of each row. Then, select the appropriate cell on that row based on <var>method</var>
    as given in the first cell of each column. Then, jump to the steps named in that cell and
    defined below the table.</p>

    <table>
     <thead>
     <tr>
      <td>
      <th> <span data-x="attr-fs-method-GET">GET</span>
      <th> <span data-x="attr-fs-method-POST">POST</span>
     <tbody>
     <tr>
      <th> <code data-x="">http</code>
      <td> <span data-x="submit-mutate-action">Mutate action URL</span>
      <td> <span data-x="submit-body">Submit as entity body</span>
     <tr>
      <th> <code data-x="">https</code>
      <td> <span data-x="submit-mutate-action">Mutate action URL</span>
      <td> <span data-x="submit-body">Submit as entity body</span>
     <tr>
      <th> <code data-x="">ftp</code>
      <td> <span data-x="submit-get-action">Get action URL</span>
      <td> <span data-x="submit-get-action">Get action URL</span>
     <tr>
      <th> <code data-x="">javascript</code>
      <td> <span data-x="submit-get-action">Get action URL</span>
      <td> <span data-x="submit-get-action">Get action URL</span>
     <tr>
      <th> <code data-x="">data</code>
      <td> <span data-x="submit-mutate-action">Mutate action URL</span>
      <td> <span data-x="submit-get-action">Get action URL</span>
     <tr>
      <th> <code data-x="">mailto</code>
      <td> <span data-x="submit-mailto-headers">Mail with headers</span>
      <td> <span data-x="submit-mailto-body">Mail as body</span>
    </table>

    <p>If <var>scheme</var> is not one of those listed in this table, then the behavior is
    not defined by this specification. User agents should, in the absence of another specification
    defining this, act in a manner analogous to that defined in this specification for similar
    schemes.</p>

    <p>Each <code>form</code> element has a <dfn>planned navigation</dfn>, which is either null or a
    <span data-x="concept-task">task</span>; when the <code>form</code> is first created, its
    <span>planned navigation</span> must be set to null. In the behaviors described below, when the
    user agent is required to <dfn>plan to navigate</dfn> to a <span>URL</span> <var>url</var> given
    an optional <span>POST resource</span>-or-null <var>postResource</var> (default null), it must
    run the following steps:</p>

    <ol>
     <li><p>Let <var>referrerPolicy</var> be the empty string.</p></li>

     <li><p>If the <code>form</code> element's <a href="#linkTypes">link types</a> include the <code
     data-x="rel-noreferrer">noreferrer</code> keyword, then set <var>referrerPolicy</var> to "<code
     data-x="">no-referrer</code>".</p></li>

     <li><p>If the <code>form</code> has a non-null <span>planned navigation</span>, remove it from
     its <span>task queue</span>.</p></li>

     <li>
      <p><span>Queue an element task</span> on the <span>DOM manipulation task source</span> given
      the <code>form</code> element and the following steps:</p>

      <ol>
       <li><p>Set the <code>form</code>'s <span>planned navigation</span> to null.</p></li>

       <li><p><span>Navigate</span><!--DONAV form--> <var>targetNavigable</var> to <var>url</var>
       using the <code>form</code> element's <span>node document</span>, with <var
       data-x="navigation-hh">historyHandling</var> set to <var>historyHandling</var>, <i
       data-x="navigation-user-involvement">userInvolvement</i> set to <var>userInvolvement</var>,
       <i data-x="navigation-source-element">sourceElement</i> set to <var>submitter</var>,
       <i data-x="navigation-referrer-policy">referrerPolicy</i> set to <var>referrerPolicy</var>,
       <i data-x="navigation-resource">documentResource</i> set to <var>postResource</var>, and <i
       data-x="navigation-form-data-entry-list">formDataEntryList</i> set to <var>entry
       list</var>.</p>
      </ol>
     </li>

     <li><p>Set the <code>form</code>'s <span>planned navigation</span> to the just-queued <span
     data-x="concept-task">task</span>.</p>
    </ol>

    <p>The behaviors are as follows:</p>

    <dl>
     <dt><dfn data-x="submit-mutate-action">Mutate action URL</dfn>
     <dd>
      <p>Let <var>pairs</var> be the result of <span data-x="convert to a list of name-value
      pairs">converting to a list of name-value pairs</span> with <var>entry list</var>.</p>

      <p>Let <var>query</var> be the result of running the
      <span><code>application/x-www-form-urlencoded</code> serializer</span> with <var>pairs</var>
      and <var>encoding</var>.</p>

      <p>Set <var>parsed action</var>'s <span data-x="concept-url-query">query</span>
      component to <var>query</var>.</p>

      <p><span>Plan to navigate</span> to <var>parsed action</var>.</p>
     </dd>

     <dt><dfn data-x="submit-body">Submit as entity body</dfn>
     <dd>
      <p><span>Assert</span>: <var>method</var> is <span
      data-x="attr-fs-method-post">POST</span>.</p>

      <p>Switch on <var>enctype</var>:

      <dl class="switch">
       <dt><code data-x="attr-fs-enctype-urlencoded">application/x-www-form-urlencoded</code></dt>

       <dd>
        <p>Let <var>pairs</var> be the result of <span data-x="convert to a list of name-value
        pairs">converting to a list of name-value pairs</span> with <var>entry list</var>.</p>

        <p>Let <var>body</var> be the result of running the
        <span><code>application/x-www-form-urlencoded</code> serializer</span> with <var>pairs</var>
        and <var>encoding</var>.</p>

        <p>Set <var>body</var> to the result of <span data-x="UTF-8 encode">encoding</span>
        <var>body</var>.</p>

        <p>Let <var>mimeType</var> be `<code>application/x-www-form-urlencoded</code>`.</p>
       </dd>

       <dt><code data-x="attr-fs-enctype-formdata">multipart/form-data</code></dt>

       <dd>
        <p>Let <var>body</var> be the result of running the <span><code
        data-x="">multipart/form-data</code> encoding algorithm</span> with <var>entry list</var>
        and <var>encoding</var>.</p>

        <p>Let <var>mimeType</var> be the <span data-x="isomorphic encode">isomorphic
        encoding</span> of the concatenation of "<code
        data-x="">multipart/form-data; boundary=</code>" and the <span><code
        data-x="">multipart/form-data</code> boundary string</span> generated by the <span><code
        data-x="">multipart/form-data</code> encoding algorithm</span>.</p>
       </dd>

       <dt><code data-x="attr-fs-enctype-text">text/plain</code></dt>

       <dd>
        <p>Let <var>pairs</var> be the result of <span data-x="convert to a list of name-value
        pairs">converting to a list of name-value pairs</span> with <var>entry list</var>.</p>

        <p>Let <var>body</var> be the result of running the <span><code data-x="">text/plain</code>
        encoding algorithm</span> with <var>pairs</var>.</p>

        <p>Set <var>body</var> to the result of <span data-x="encode">encoding</span>
        <var>body</var> using <var>encoding</var>.</p>

        <p>Let <var>mimeType</var> be `<code>text/plain</code>`.</p>
       </dd>
      </dl>

      <p><span>Plan to navigate</span> to <var>parsed action</var> given a <span>POST
      resource</span> whose <span data-x="post-resource-request-body">request body</span> is
      <var>body</var> and <span data-x="post-resource-request-content-type">request
      content-type</span> is <var>mimeType</var>.</p>
     </dd>

     <dt><dfn data-x="submit-get-action">Get action URL</dfn>
     <dd>
      <p><span>Plan to navigate</span> to <var>parsed action</var>.</p>

      <p class="note"><var>entry list</var> is discarded.</p>
     </dd>

     <dt><dfn data-x="submit-mailto-headers">Mail with headers</dfn>
     <dd>
      <p>Let <var>pairs</var> be the result of <span data-x="convert to a list of name-value
      pairs">converting to a list of name-value pairs</span> with <var>entry list</var>.</p>

      <p>Let <var>headers</var> be the result of running the
      <span><code>application/x-www-form-urlencoded</code> serializer</span> with <var>pairs</var>
      and <var>encoding</var>.</p>

      <p>Replace occurrences of U+002B PLUS SIGN characters (+) in <var>headers</var> with
      the string "<code data-x="">%20</code>".</p>

      <p>Set <var>parsed action</var>'s <span data-x="concept-url-query">query</span> to
      <var>headers</var>.</p>

      <p><span>Plan to navigate</span> to <var>parsed action</var>.</p>
     </dd>

     <dt><dfn data-x="submit-mailto-body">Mail as body</dfn>
     <dd>
      <p>Let <var>pairs</var> be the result of <span data-x="convert to a list of name-value
      pairs">converting to a list of name-value pairs</span> with <var>entry list</var>.</p>

      <p>Switch on <var>enctype</var>:

      <dl class="switch">
       <dt><code data-x="attr-fs-enctype-text">text/plain</code></dt>

       <dd>
        <p>Let <var>body</var> be the result of running the <span><code data-x="">text/plain</code>
        encoding algorithm</span> with <var>pairs</var>.</p>

        <p>Set <var>body</var> to the result of running <span>UTF-8 percent-encode</span> on
        <var>body</var> using the <span>default encode set</span>. <ref>URL</ref></p>
       </dd>

       <dt>Otherwise</dt>

       <dd><p>Let <var>body</var> be the result of running the
       <span><code>application/x-www-form-urlencoded</code> serializer</span> with <var>pairs</var>
       and <var>encoding</var>.</p></dd>
      </dl>

      <p>If <var>parsed action</var>'s <span data-x="concept-url-query">query</span> is null, then
      set it to the empty string.

      <p>If <var>parsed action</var>'s <span data-x="concept-url-query">query</span> is not the
      empty string, then append a single U+0026 AMPERSAND character (&amp;) to it.

      <p>Append "<code data-x="">body=</code>" to <var>parsed action</var>'s <span
      data-x="concept-url-query">query</span>.</p>

      <p>Append <var>body</var> to <var>parsed action</var>'s <span
      data-x="concept-url-query">query</span>.</p>

      <p><span>Plan to navigate</span> to <var>parsed action</var>.</p>
     </dd>
    </dl>
   </li>
  </ol>


  <h5 id="constructing-form-data-set">Constructing the entry list</h5>

  <p>An <dfn export>entry list</dfn> is a <span>list</span> of <span data-x="form
  entry">entries</span>, typically representing the contents of a form. An <dfn data-x="form entry"
  for="entry list" export>entry</dfn> is a tuple consisting of a <dfn data-x="form entry name"
  for="entry list/entry" export>name</dfn> (a <span>scalar value string</span>) and a <dfn
  data-x="form entry value" for="entry list/entry" export>value</dfn> (either a <span>scalar value
  string</span> or a <code>File</code> object).</p>

  <p id="append-an-entry">To <dfn export for="entry list">create an entry</dfn> given a string
  <var>name</var>, a string or <code>Blob</code> object <var>value</var>, and optionally a
  <span>scalar value string</span> <var>filename</var>:</p>

  <ol>
   <li><p>Set <var>name</var> to the result of <span data-x="convert">converting</span>
   <var>name</var> into a <span>scalar value string</span>.</p></li>

   <li><p>If <var>value</var> is a string, then set <var>value</var> to the result of <span
   data-x="convert">converting</span> <var>value</var> into a <span>scalar value
   string</span>.</p></li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p>If <var>value</var> is not a <code>File</code> object, then set <var>value</var> to a
     new <code>File</code> object, representing the same bytes, whose <code
     data-x="dom-File-name">name</code> attribute value is "<code data-x="">blob</code>".</p></li>
     <!-- XXX at some point File API should get internal slots for this -->

     <li><p>If <var>filename</var> is given, then set <var>value</var> to a new <code>File</code>
     object, representing the same bytes, whose <code data-x="dom-File-name">name</code> attribute
     is <var>filename</var>.</p></li>
    </ol>

    <p class="note">These operations will create a new <code>File</code> object if either
    <var>filename</var> is given or the passed <code>Blob</code> is not a <code>File</code> object.
    In those cases, the identity of the passed <code>Blob</code> object is not kept.</p>
   </li>

   <li><p>Return an <span data-x="form entry">entry</span> whose <span data-x="form entry
   name">name</span> is <var>name</var> and whose <span data-x="form entry value">value</span> is
   <var>value</var>.</p></li>
  </ol>

  <p>To <dfn id="constructing-the-form-data-set" export data-lt="constructing the entry list"
  data-x="constructing the entry list">construct the entry list</dfn> given a <var>form</var>, an
  optional <var>submitter</var> (default null), and an optional <var>encoding</var> (default
  <span>UTF-8</span>):

  <ol>
   <li><p>If <var>form</var>'s <span>constructing entry list</span> is true, then return
   null.</p></li>

   <li><p>Set <var>form</var>'s <span>constructing entry list</span> to true.</p></li>

   <li><p>Let <var>controls</var> be a list of all the <span
   data-x="category-submit">submittable elements</span> whose <span>form owner</span> is <var>form</var>, in <span>tree order</span>.</p></li>

   <li><p>Let <var>entry list</var> be a new empty <span>entry list</span>.</p></li>

   <li>
    <p>For each element <var>field</var> in <var>controls</var>, in <span>tree order</span>:</p>

    <ol>
     <li>
      <p>If any of the following are true:</p>

      <ul>
       <li><p><var>field</var> has a <code>datalist</code> element ancestor;</p></li>

       <li><p><var>field</var> is <span data-x="concept-fe-disabled">disabled</span>;</p></li>

       <li><p><var>field</var> is a <span data-x="concept-button">button</span> but it is not
       <var>submitter</var>;</p></li>

       <li><p><var>field</var> is an <code>input</code> element whose <code
       data-x="attr-input-type">type</code> attribute is in the <span
       data-x="attr-input-type-checkbox">Checkbox</span> state and whose <span
       data-x="concept-fe-checked">checkedness</span> is false; or</p></li>

       <li><p><var>field</var> is an <code>input</code> element whose <code
       data-x="attr-input-type">type</code> attribute is in the <span
       data-x="attr-input-type-radio">Radio Button</span> state and whose <span
       data-x="concept-fe-checked">checkedness</span> is false,</p></li>
      </ul>

      <p>then <span>continue</span>.</p>
     </li>

     <li>
      <p>If the <var>field</var> element is an <code>input</code> element whose <code
      data-x="attr-input-type">type</code> attribute is in the <span
      data-x="attr-input-type-image">Image Button</span> state, then:</p>

      <ol>
       <li><p>If the <var>field</var> element is not <var>submitter</var>, then
       <span>continue</span>.</p></li>

       <li><p>If the <var>field</var> element has a <code data-x="attr-fe-name">name</code>
       attribute specified and its value is not the empty string, let <var>name</var> be that value
       followed by U+002E (.). Otherwise, let <var>name</var> be the empty string.</p></li>

       <li><p>Let <var>name<sub>x</sub></var> be the concatenation of <var>name</var> and
       U+0078 (x).</p></li>

       <li><p>Let <var>name<sub>y</sub></var> be the concatenation of <var>name</var> and
       U+0079 (y).</p></li>

       <li><p>Let (<var>x</var>, <var>y</var>) be the <span
       data-x="concept-input-type-image-coordinate">selected coordinate</span>.</p></li>

       <li><p><span>Create an entry</span> with <var>name<sub>x</sub></var> and <var>x</var>, and
       <span data-x="list append">append</span> it to <var>entry list</var>.</p></li>

       <li><p><span>Create an entry</span> with <var>name<sub>y</sub></var> and <var>y</var>, and
       <span data-x="list append">append</span> it to <var>entry list</var>.</p></li>

       <li><p><span>Continue</span>.</p></li>
      </ol>
     </li>

     <li><p>If the <var>field</var> is a <span>form-associated custom element</span>, then perform
     the <span data-x="face-entry-construction">entry construction algorithm</span> given
     <var>field</var> and <var>entry list</var>, then <span>continue</span>.</p></li>

     <li><p>If either the <var>field</var> element does not have a
     <code data-x="attr-fe-name">name</code> attribute specified, or its
     <code data-x="attr-fe-name">name</code> attribute's value is the empty string, then
     <span>continue</span>.</p></li>

     <li><p>Let <var>name</var> be the value of the <var>field</var> element's
     <code data-x="attr-fe-name">name</code> attribute.</p></li>

     <li><p>If the <var>field</var> element is a <code>select</code> element, then for each
     <code>option</code> element in the <code>select</code> element's <span
     data-x="concept-select-option-list">list of options</span> whose <span
     data-x="concept-option-selectedness">selectedness</span> is true and that is not <span
     data-x="concept-option-disabled">disabled</span>, <span>create an entry</span> with
     <var>name</var> and the <span data-x="concept-option-value">value</span> of the
     <code>option</code> element, and <span data-x="list append">append</span> it to <var>entry
     list</var>.</p></li>

     <li>
      <p>Otherwise, if the <var>field</var> element is an <code>input</code> element whose
      <code data-x="attr-input-type">type</code> attribute is in the <span
      data-x="attr-input-type-checkbox">Checkbox</span> state or the <span
      data-x="attr-input-type-radio">Radio Button</span> state, then:</p>

      <ol>
       <li><p>If the <var>field</var> element has a <code
       data-x="attr-input-value">value</code> attribute specified, then let <var>value</var>
       be the value of that attribute; otherwise, let <var>value</var> be the string "<code
       data-x="">on</code>".</p></li>

       <li><p><span>Create an entry</span> with <var>name</var> and <var>value</var>, and <span
       data-x="list append">append</span> it to <var>entry list</var>.</p></li>
      </ol>
     </li>

     <li>
      <p>Otherwise, if the <var>field</var> element is an <code>input</code> element whose <code
      data-x="attr-input-type">type</code> attribute is in the <span
      data-x="attr-input-type-file">File Upload</span> state, then:</p>

      <ol>
       <li><p>If there are no <span data-x="concept-input-type-file-selected">selected files</span>,
       then <span>create an entry</span> with <var>name</var> and a new <code>File</code> object
       with an empty name, <code>application/octet-stream</code> as type, and an empty body, and
       <span data-x="list append">append</span> it to <var>entry list</var>.</p></li>
       <!-- https://bugzilla.mozilla.org/show_bug.cgi?id=529859 -->

       <li><p>Otherwise, for each file in <span data-x="concept-input-type-file-selected">selected
       files</span>, <span>create an entry</span> with <var>name</var> and a <code>File</code>
       object representing the file, and <span data-x="list append">append</span> it to <var>entry
       list</var>.</p></li>
      </ol>
     </li>

     <li>
      <p>Otherwise, if the <var>field</var> element is an <code>input</code> element whose <code
      data-x="attr-input-type">type</code> attribute is in the <span
      data-x="attr-input-type-hidden">Hidden</span> state and <var>name</var> is an <span>ASCII
      case-insensitive</span> match for "<code data-x="attr-fe-name-charset">_charset_</code>":</p>

      <ol>
       <li><p>Let <var>charset</var> be the <span data-x="encoding name">name</span> of
       <var>encoding</var>.</p></li>

       <li><p><span>Create an entry</span> with <var>name</var> and <var>charset</var>, and <span
       data-x="list append">append</span> it to <var>entry list</var>.</p></li>
      </ol>
     </li>

     <li><p>Otherwise, <span>create an entry</span> with <var>name</var> and the <span
     data-x="concept-fe-value">value</span> of the <var>field</var> element, and <span
     data-x="list append">append</span> it to <var>entry list</var>.</p></li>

     <li>
      <p>If the element has a <code data-x="attr-fe-dirname">dirname</code> attribute, that
      attribute's value is not the empty string, and the element is an <span
      data-x="auto-directionality form-associated elements">auto-directionality form-associated
      element</span>:

      <ol>
       <li><p>Let <var>dirname</var> be the value of the element's <code
       data-x="attr-fe-dirname">dirname</code> attribute.</p></li>

       <li><p>Let <var>dir</var> be the string "<code data-x="">ltr</code>" if <span>the
       directionality</span> of the element is '<span data-x="concept-ltr">ltr</span>', and "<code
       data-x="">rtl</code>" otherwise (i.e., when <span>the directionality</span> of the element is
       '<span data-x="concept-rtl">rtl</span>').</p></li>

       <li><p><span>Create an entry</span> with <var>dirname</var> and <var>dir</var>, and <span
       data-x="list append">append</span> it to <var>entry list</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Let <var>form data</var> be a new <code>FormData</code> object associated with
   <var>entry list</var>.</p></li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named
   <code data-x="event-formdata">formdata</code> at <var>form</var> using
   <code>FormDataEvent</code>, with the <code data-x="dom-FormDataEvent-formData">formData</code>
   attribute initialized to <var>form data</var> and the
   <code data-x="dom-Event-bubbles">bubbles</code> attribute initialized to true.</p></li>

   <li><p>Set <var>form</var>'s <span>constructing entry list</span> to false.</p></li>

   <li><p>Return a <span data-x="list clone">clone</span> of <var>entry list</var>.</p></li>
  </ol>

  </div>


  <div w-nodev>

  <h5>Selecting a form submission encoding</h5>

  <p>If the user agent is to <dfn data-x="picking an encoding for the form">pick an encoding for a
  form</dfn>, it must run the following steps:</p>

  <ol>
   <li><p>Let <var>encoding</var> be the <span>document's character encoding</span>.</p></li>

   <li>
    <p>If the <code>form</code> element has an <code
    data-x="attr-form-accept-charset">accept-charset</code> attribute, set <var>encoding</var> to
    the return value of running these substeps:</p>

    <ol>
     <li><p>Let <var>input</var> be the value of the <code>form</code> element's <code
     data-x="attr-form-accept-charset">accept-charset</code> attribute.</p></li>

     <li><p>Let <var>candidate encoding labels</var> be the result of <span data-x="split a string
     on ASCII whitespace">splitting <var>input</var> on ASCII whitespace</span>.</p></li>

     <li><p>Let <var>candidate encodings</var> be an empty list of <span data-x="encoding">character
     encodings</span>.</p></li>

     <li><p>For each token in <var>candidate encoding labels</var> in turn (in the order in which
     they were found in <var>input</var>), <span data-x="getting an encoding">get an encoding</span>
     for the token and, if this does not result in failure, append the <span>encoding</span> to
     <var>candidate encodings</var>.</p></li>

     <li><p>If <var>candidate encodings</var> is empty, return <span>UTF-8</span>.</p></li>

     <li><p>Return the first encoding in <var>candidate encodings</var>.</p></li>
    </ol>
   </li>

   <li><p>Return the result of <span data-x="get an output encoding">getting an output
   encoding</span> from <var>encoding</var>.</p></li>
  </ol>

  </div>

  <div w-nodev>

  <h5>Converting an entry list to a list of name-value pairs</h5>

  <p>The <code>application/x-www-form-urlencoded</code> and <code data-x="text/plain encoding
  algorithm">text/plain</code> encoding algorithms take a list of name-value pairs, where the values
  must be strings, rather than an <span>entry list</span> where the value can be a
  <code>File</code>. The following algorithm performs the conversion.</p>

  <p>To <dfn>convert to a list of name-value pairs</dfn> an <span>entry list</span> <var>entry
  list</var>, run these steps:</p>

  <ol>
   <li><p>Let <var>list</var> be an empty <span>list</span> of name-value pairs.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>entry</var> of <var>entry list</var>:</p>

    <ol>
     <li><p>Let <var>name</var> be <var>entry</var>'s <span data-x="form entry name">name</span>,
     with every occurrence of U+000D (CR) not followed by U+000A (LF), and every occurrence of
     U+000A (LF) not preceded by U+000D (CR), replaced by a string consisting of U+000D (CR) and
     U+000A (LF).</p></li>

     <li><p>If <var>entry</var>'s <span data-x="form entry value">value</span> is a
     <code>File</code> object, then let <var>value</var> be <var>entry</var>'s <span data-x="form
     entry value">value</span>'s <code data-x="dom-file-name">name</code>. Otherwise, let
     <var>value</var> be <var>entry</var>'s <span data-x="form entry value">value</span>.</p></li>

     <li><p>Replace every occurrence of U+000D (CR) not followed by U+000A (LF), and every occurrence
     of U+000A (LF) not preceded by U+000D (CR), in <var>value</var>, by a string consisting of
     U+000D (CR) and U+000A (LF).</p></li>

     <li><p><span data-x="list append">Append</span> to <var>list</var> a new name-value pair whose
     name is <var>name</var> and whose value is <var>value</var>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>list</var>.</p></li>
  </ol>

  </div>


  <h5>URL-encoded form data</h5>

  <p id="application-x-www-form-urlencoded-encoding-algorithm"><span
  id="application/x-www-form-urlencoded-encoding-algorithm"></span>See <cite>URL</cite> for details
  on <code>application/x-www-form-urlencoded</code>. <ref>URL</ref></p>


  <h5>Multipart form data</h5>

  <div w-nodev>

  <!-- https://hixie.ch/tests/adhoc/html/forms/submission/multipart_form-data/ -->

  <!-- NOTE: This algorithm is also used by
  https://fetch.spec.whatwg.org/#concept-bodyinit-extract -->

  <p>The <dfn export><code>multipart/form-data</code> encoding algorithm</dfn>, given an <span>entry
  list</span> <var>entry list</var> and an <span>encoding</span> <var>encoding</var>, is as
  follows:</p>

  <ol>
   <li>
    <p><span data-x="list iterate">For each</span> <var>entry</var> of <var>entry list</var>:</p>

    <ol>
     <li><p>Replace every occurrence of U+000D (CR) not followed by U+000A (LF), and every
     occurrence of U+000A (LF) not preceded by U+000D (CR), in <var>entry</var>'s <span data-x="form
     entry name">name</span>, by a string consisting of a U+000D (CR) and U+000A (LF).</p></li>

     <li><p>If <var>entry</var>'s <span data-x="form entry value">value</span> is not a
     <code>File</code> object, then replace every occurrence of U+000D (CR) not followed by U+000A
     (LF), and every occurrence of U+000A (LF) not preceded by U+000D (CR), in <var>entry</var>'s
     <span data-x="form entry value">value</span>, by a string consisting of a U+000D (CR) and
     U+000A (LF).</p></li>
    </ol>
   </li>

   <li>
    <p>Return the byte sequence resulting from encoding the <var>entry list</var> using the rules
    described by RFC 7578, <cite>Returning Values from Forms: <code
    data-x="">multipart/form-data</code></cite>, given the following conditions:
    <ref>RFC7578</ref></p>

    <ul>
     <li><p>Each <span data-x="form entry">entry</span> in <var>entry list</var> is a <i>field</i>,
     the <span data-x="form entry name">name</span> of the entry is the <i>field name</i> and the
     <span data-x="form entry value">value</span> of the entry is the <i>field value</i>.</p></li>

     <li><p>The order of parts must be the same as the order of fields in <var>entry list</var>.
     Multiple entries with the same name must be treated as distinct fields.</p></li>

     <li><p>Field names, field values for non-file fields, and filenames for file fields, in the
     generated <code>multipart/form-data</code> resource must be set to the result of <span
     data-x="encode">encoding</span> the corresponding entry's name or value with
     <var>encoding</var>, converted to a byte sequence.</p></li>

     <li><p>For field names and filenames for file fields, the result of the encoding in the
     previous bullet point must be escaped by replacing any 0x0A (LF) bytes with the byte sequence
     `<code data-x="">%0A</code>`, 0x0D (CR) with `<code data-x="">%0D</code>` and 0x22 (") with
     `<code data-x="">%22</code>`. The user agent must not perform any other escapes.</p></li>

     <li><p>The parts of the generated <code>multipart/form-data</code> resource that correspond to
     non-file fields must not have a `<code>Content-Type</code>` header specified.</p></li>

     <li><p>The boundary used by the user agent in generating the return value of this algorithm is
     the <dfn export><code>multipart/form-data</code> boundary string</dfn>. (This value is used to
     generate the MIME type of the form submission payload generated by this algorithm.)</p></li>
    </ul>
   </li>
  </ol>

  </div>

  <p>For details on how to interpret <code>multipart/form-data</code> payloads, see RFC 7578.
  <ref>RFC7578</ref></p>


  <h5>Plain text form data</h5>

  <div w-nodev>

  <p>The <dfn><code data-x="">text/plain</code> encoding algorithm</dfn>, given a list of name-value
  pairs <var>pairs</var>, is as follows:</p>

  <ol>
   <li><p>Let <var>result</var> be the empty string.</p></li>

   <li>
    <p>For each <var>pair</var> in <var>pairs</var>:</p>

    <ol>
     <li><p>Append <var>pair</var>'s name to <var>result</var>.</p></li>

     <li><p>Append a single U+003D EQUALS SIGN character (=) to <var>result</var>.</p></li>

     <li><p>Append <var>pair</var>'s value to <var>result</var>.</p></li>

     <li><p>Append a U+000D CARRIAGE RETURN (CR) U+000A LINE FEED (LF) character pair to <var>result</var>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>result</var>.</p></li>
  </ol>

  </div>

  <p>Payloads using the <code>text/plain</code> format are intended to be human readable. They are
  not reliably interpretable by computer, as the format is ambiguous (for example, there is no way
  to distinguish a literal newline in a value from the newline at the end of the value).</p>

  <h5>The <code>SubmitEvent</code> interface</h5>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>SubmitEvent</dfn> : <span>Event</span> {
  constructor(DOMString type, optional <span>SubmitEventInit</span> eventInitDict = {});

  readonly attribute <span>HTMLElement</span>? <span data-x="dom-SubmitEvent-submitter">submitter</span>;
};

dictionary <dfn dictionary>SubmitEventInit</dfn> : <span>EventInit</span> {
  <span>HTMLElement</span>? submitter = null;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>event</var>.<span data-x="dom-SubmitEvent-submitter">submitter</span></code></dt>
   <dd><p>Returns the element representing the <span data-x="concept-submit-button">submit
   button</span> that triggered the <span>form submission</span>, or null if the submission was not
   triggered by a button.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="SubmitEvent"><code
  data-x="dom-SubmitEvent-submitter">submitter</code></dfn> attribute must return the value it was
  initialized to.</p>

  </div>

  <h5>The <code>FormDataEvent</code> interface</h5>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>FormDataEvent</dfn> : <span>Event</span> {
  constructor(DOMString type, <span>FormDataEventInit</span> eventInitDict);

  readonly attribute <span>FormData</span> <span data-x="dom-FormDataEvent-formData">formData</span>;
};

dictionary <dfn dictionary>FormDataEventInit</dfn> : <span>EventInit</span> {
  required <span>FormData</span> formData;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>event</var>.<span data-x="dom-FormDataEvent-formData">formData</span></code></dt>

   <dd>
    <p>Returns a <code>FormData</code> object representing names and values of elements associated
    to the target <code>form</code>. Operations on the <code>FormData</code> object will affect form
    data to be submitted.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="FormDataEvent"><code
  data-x="dom-FormDataEvent-formData">formData</code></dfn> attribute must return the value it was
  initialized to. It represents a <code>FormData</code> object associated to the <span>entry
  list</span> that is <span data-x="constructing the entry list">constructed</span> when the
  <code>form</code> is submitted.</p>

  <h4>Resetting a form</h4>

  <p>When a <code>form</code> element <var>form</var> is <dfn
  data-x="concept-form-reset">reset</dfn>, run these steps:</p>

  <ol>
   <li><p>Let <var>reset</var> be the result of <span data-x="concept-event-fire">firing an
   event</span> named <code data-x="event-reset">reset</code> at <var>form</var>, with the <code
   data-x="dom-Event-bubbles">bubbles</code> and <code
   data-x="dom-Event-cancelable">cancelable</code> attributes initialized to true.</p></li>

   <li><p>If <var>reset</var> is true, then invoke the <span
   data-x="concept-form-reset-control">reset algorithm</span> of each <span
   data-x="category-reset">resettable element</span> whose <span>form owner</span> is
   <var>form</var>.</p></li>
  </ol>

  <p>Each <span data-x="category-reset">resettable element</span> defines its own <dfn
  data-x="concept-form-reset-control">reset algorithm</dfn>. Changes made to form controls as part of
  these algorithms do not count as changes caused by the user (and thus, e.g., do not cause <code
  data-x="event-input">input</code> events to fire).</p>

  </div>



  <h3 split-filename="interactive-elements" id="interactive-elements">Interactive elements</h3>

  <h4>The <dfn element><code>details</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Interactive content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>One <code>summary</code> element followed by <span>flow content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-details-name">name</code></dd>
   <dd><code data-x="attr-details-open">open</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-details">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-details">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLDetailsElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-details-name">name</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-details-open">open</span>;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLDetailsElement</code>.</dd>
  </dl>

  <p>The <code>details</code> element <span>represents</span> a disclosure widget from which the
  user can obtain additional information or controls.</p>

  <p class="note">As with all HTML elements, it is not conforming to use the <code>details</code>
  element when attempting to represent another type of control. For example, tab widgets and
  menu widgets are not disclosure widgets, so abusing the <code>details</code> element to
  implement these patterns is incorrect.</p>

  <p class="note">The <code>details</code> element is not appropriate for footnotes. Please see <a
  href="#footnotes">the section on footnotes</a> for details on how to mark up footnotes.</p>

  <p>The <span w-nodev>first</span> <code>summary</code> element child of the element, if any,
  <span>represents</span> the summary or legend of the details. <span w-nodev>If there is no
  child <code>summary</code> element, the user agent should provide its own legend (e.g.
  "Details").</span></p>

  <p>The rest of the element's contents <span>represents</span> the additional information or
  controls.</p>

  <p>The <dfn element-attr for="details"><code data-x="attr-details-name">name</code></dfn> content
  attribute gives the name of the group of related <code>details</code> elements that the element is
  a member of. Opening one member of this group causes other members of the group to close. If the
  attribute is specified, its value must not be the empty string.</p>

  <p>Before using this feature, authors should consider whether this grouping of related
  <code>details</code> elements into an exclusive accordion is helpful or harmful to users.
  While using an exclusive accordion can reduce the maximum amount of space that a set of content
  can occupy, it can also frustrate users who have to open many items to find what they want or
  users who want to look at the contents of multiple items at the same time.</p>

  <p>A document must not contain more than one <code>details</code> element in the same
  <span>details name group</span> that has the <code data-x="attr-details-open">open</code>
  attribute present.  Authors must not use script to add <code>details</code> elements to a document
  in a way that would cause a <span>details name group</span> to have more than one
  <code>details</code> element with the <code data-x="attr-details-open">open</code> attribute
  present.</p>

  <p class="note">The group of elements that is created by a common <code
  data-x="attr-details-name">name</code> attribute is exclusive, meaning that at most one of the
  <code>details</code> elements can be open at once.  While this exclusivity is enforced by user
  agents, the resulting enforcement immediately changes the <code
  data-x="attr-details-open">open</code> attributes in the markup.  This requirement on authors
  forbids such misleading markup.</p>

  <p>A document must not contain a <code>details</code> element that is a descendant of another
  <code>details</code> element in the same <span>details name group</span>.</p>

  <p>Documents that use the <code data-x="attr-details-name">name</code> attribute to group multiple
  related <code>details</code> elements should keep those related elements together in a containing
  element (such as a <code>section</code> element or <code>article</code> element). When it makes
  sense for the group to be introduced with a heading, authors should put that heading in a <span
  data-x="concept-heading">heading</span> element at the start of the containing element.</p>

  <p class="note">Visually and programmatically grouping related elements together can be important
  for accessible user experiences. This can help users understand the relationship between such
  elements. When related elements are in disparate sections of a web page rather than being grouped,
  the elements' relationships to each other can be less discoverable or understandable.</p>

  <p>The <dfn element-attr for="details"><code data-x="attr-details-open">open</code></dfn> content
  attribute is a <span>boolean attribute</span>. If present, it indicates that both the summary and
  the additional information is to be shown to the user. If the attribute is absent, only the
  summary is to be shown.</p>

  <div w-nodev>

  <p>When the element is created, if the attribute is absent, the additional information should be
  hidden; if the attribute is present, that information should be shown. Subsequently, if the
  attribute is removed, then the information should be hidden; if the attribute is added, the
  information should be shown.</p>

  <p>The user agent should allow the user to request that the additional information be shown or
  hidden. To honor a request for the details to be shown, the user agent must <span
  data-x="concept-element-attributes-set-value">set</span> the <code
  data-x="attr-details-open">open</code> attribute on the element to the empty string. To honor a
  request for the information to be hidden, the user agent must <span
  data-x="concept-element-attributes-remove">remove</span> the <code
  data-x="attr-details-open">open</code> attribute from the element.</p>

  <p class="note">This ability to request that additional information be shown or hidden
  <!--non-normative-->may simply be the <span>activation behavior</span> of the appropriate
  <code>summary</code> element, in the case such an element exists. However, if no such element
  exists, user agents can still provide this ability through some other user interface
  affordance.</p>

  <p>The <dfn>details name group</dfn> that contains a <code>details</code> element <var>a</var>
  also contains all the other <code>details</code> elements <var>b</var> that fulfill all of the
  following conditions:</p>

  <ul>
   <li>Both <var>a</var> and <var>b</var> are in the same <span>tree</span>.</li>

   <li>They both have a <code data-x="attr-details-name">name</code> attribute, their <code
   data-x="attr-details-name">name</code> attributes are not the empty string, and the value of
   <var>a</var>'s <code data-x="attr-details-name">name</code> attribute equals the value of
   <var>b</var>'s <code data-x="attr-details-name">name</code> attribute.</li>
  </ul>

  <p>Every <code>details</code> element has a <dfn>details toggle task tracker</dfn>, which is a
  <span>toggle task tracker</span> or null, initially null.</p>

  <p>The following <span data-x="concept-element-attributes-change-ext">attribute change
  steps</span>, given <var>element</var>, <var>localName</var>, <var>oldValue</var>,
  <var>value</var>, and <var>namespace</var>, are used for all <code>details</code> elements:</p>

  <ol>
   <li><p>If <var>namespace</var> is not null, then return.</p></li>

   <li><p>If <var>localName</var> is <code data-x="attr-details-name">name</code>, then <span>ensure
   details exclusivity by closing the given element if needed</span> given
   <var>element</var>.</p></li>

   <li><p>If <var>localName</var> is <code data-x="attr-details-open">open</code>, then:
    <ol>
     <li>
      <p>If one of <var>oldValue</var> or <var>value</var> is null and the other is not null,
      run the following steps, which are known as the <dfn>details notification task steps</dfn>, for
      this <code>details</code> element:</p>

      <p class="note">When the <code data-x="attr-details-open">open</code> attribute is toggled
      several times in succession, the resulting tasks essentially get coalesced so that only one
      event is fired.</p>

      <ol>
        <li><p>If <var>oldValue</var> is null, <span>queue a details toggle event task</span> given
        the <code>details</code> element, "<code data-x="">closed</code>", and "<code
        data-x="">open</code>".</p></li>

        <li><p>Otherwise, <span>queue a details toggle event task</span> given the
        <code>details</code> element, "<code data-x="">open</code>", and "<code
        data-x="">closed</code>".</p></li>
      </ol>
     </li>

     <li><p>If <var>oldValue</var> is null and <var>value</var> is not null, then <span>ensure
     details exclusivity by closing other elements if needed</span> given
     <var>element</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>The <code>details</code> <span data-x="html element insertion steps">HTML element insertion
  steps</span>, given <var>insertedNode</var>, are:</p>

  <ol>
   <li><p><span>Ensure details exclusivity by closing the given element if needed</span> given
   <var>insertedNode</var>.</p></li>
  </ol>

  <p class="note">To be clear, these attribute change and insertion steps also run when an attribute
  or element is inserted via the parser.</p>

  <p>To <dfn>queue a details toggle event task</dfn> given a <code>details</code> element
  <var>element</var>, a string <var>oldState</var>, and a string <var>newState</var>:

  <ol>
   <li>
    <p>If <var>element</var>'s <span>details toggle task tracker</span> is not null, then:</p>

    <ol>
     <li><p>Set <var>oldState</var> to <var>element</var>'s <span>details toggle task
     tracker</span>'s <span data-x="toggle-task-old-state">old state</span>.</p></li>

     <li><p>Remove <var>element</var>'s <span>details toggle task tracker</span>'s <span
     data-x="toggle-task-task">task</span> from its <span>task queue</span>.</p></li>

     <li><p>Set <var>element</var>'s <span>details toggle task tracker</span> to null.</p></li>
    </ol>
   </li>

   <li>
    <p><span>Queue an element task</span> given the <span>DOM manipulation task source</span> and
    <var>element</var> to run the following steps:</p>

    <ol>
     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-toggle">toggle</code> at <var>element</var>, using <code>ToggleEvent</code>, with
     the <code data-x="dom-ToggleEvent-oldState">oldState</code> attribute initialized to
     <var>oldState</var> and the <code data-x="dom-ToggleEvent-newState">newState</code> attribute
     initialized to <var>newState</var>.</p></li>

     <li><p>Set <var>element</var>'s <span>details toggle task tracker</span> to null.</p></li>
    </ol>
   </li>

   <li><p>Set <var>element</var>'s <span>details toggle task tracker</span> to a struct with <span
   data-x="toggle-task-task">task</span> set to the just-queued <span
   data-x="concept-task">task</span> and <span data-x="toggle-task-old-state">old state</span> set
   to <var>oldState</var>.</p></li>
  </ol>

  <p>To <dfn>ensure details exclusivity by closing other elements if needed</dfn> given a
  <code>details</code> element <var>element</var>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>element</var> has an <code
   data-x="attr-details-open">open</code> attribute.</p></li>

   <!-- This step is an optimization, but it may also make things clearer. -->
   <li><p>If <var>element</var> does not have a <code data-x="attr-details-name">name</code>
   attribute, or its <code data-x="attr-details-name">name</code> attribute is the empty string,
   then return.</p></li>

   <li><p>Let <var>groupMembers</var> be a list of elements, containing all elements in
   <var>element</var>'s <span>details name group</span> except for <var>element</var>, in <span>tree
   order</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> element <var>otherElement</var> of
    <var>groupMembers</var>:</p>
    <ol>
     <li>
      <p>If the <code data-x="attr-details-open">open</code> attribute is set on
      <var>otherElement</var>, then:</p>

      <ol>
       <li><p><span>Assert</span>: <var>otherElement</var> is the only element in
       <var>groupMembers</var> that has the <code data-x="attr-details-open">open</code> attribute
       set.</p></li>

       <li><p><span data-x="concept-element-attributes-remove">Remove</span> the <code
       data-x="attr-details-open">open</code> attribute on <var>otherElement</var>.</p></li>

       <li><p><span>Break</span>.</p></li>
      </ol>
    </ol>
   </li>
  </ol>

  <p>To <dfn>ensure details exclusivity by closing the given element if needed</dfn> given a
  <code>details</code> element <var>element</var>:</p>

  <ol>
   <li><p>If <var>element</var> does not have an <code data-x="attr-details-open">open</code>
   attribute, then return.</p></li>

   <!-- This step is an optimization, but it may also make things clearer. -->
   <li><p>If <var>element</var> does not have a <code data-x="attr-details-name">name</code>
   attribute, or its <code data-x="attr-details-name">name</code> attribute is the empty string,
   then return.</p></li>

   <li><p>Let <var>groupMembers</var> be a list of elements, containing all elements in
   <var>element</var>'s <span>details name group</span> except for <var>element</var>, in <span>tree
   order</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> element <var>otherElement</var> of
    <var>groupMembers</var>:</p>

    <ol>
     <li>
      <p>If the <code data-x="attr-details-open">open</code> attribute is set on
      <var>otherElement</var>, then:</p>

      <ol>
       <li><p><span data-x="concept-element-attributes-remove">Remove</span> the <code
       data-x="attr-details-open">open</code> attribute on <var>element</var>.</p></li>

       <li><p><span>Break</span>.</p></li>
      </ol>
     </li>
    </ol>
   </li>
  </ol>

  <p>The <dfn attribute for="HTMLDetailsElement"><code data-x="dom-details-name">name</code></dfn>
  and <dfn attribute for="HTMLDetailsElement"><code data-x="dom-details-open">open</code></dfn>
  IDL attributes must <span>reflect</span> the respective content attributes of the same name.</p>

  </div>

  <p>The <dfn>ancestor details revealing algorithm</dfn> is to run the following steps on
  <var>currentNode</var>:</p>

  <ol>
   <li>
    <p>While <var>currentNode</var> has a parent node within the <span>flat tree</span>:</p>

    <ol>
     <li>
      <p>If <var>currentNode</var> is slotted into the second slot of a <code>details</code>
      element:</p>

      <ol>
       <li><p>Set <var>currentNode</var> to the <code>details</code> element which
       <var>currentNode</var> is slotted into.</p></li>

       <li><p>If the <code data-x="attr-details-open">open</code> attribute is not set on
       <var>currentNode</var>, then <span data-x="concept-element-attributes-set-value">set</span>
       the <code data-x="attr-details-open">open</code> attribute on <var>currentNode</var> to the
       empty string.</p></li>
      </ol>
     </li>

     <li><p>Otherwise, set <var>currentNode</var> to the parent node of
     <var>currentNode</var> within the <span>flat tree</span>.</p></li>
    </ol>
   </li>
  </ol>

  <div class="example">

   <p>The following example shows the <code>details</code> element being used to hide technical
   details in a progress report.</p>

   <pre><code class="html">&lt;section class="progress window">
 &lt;h1>Copying "Really Achieving Your Childhood Dreams"&lt;/h1>
 &lt;details>
  &lt;summary>Copying... &lt;progress max="375505392" value="97543282">&lt;/progress> 25%&lt;/summary>
  &lt;dl>
   &lt;dt>Transfer rate:&lt;/dt> &lt;dd>452KB/s&lt;/dd>
   &lt;dt>Local filename:&lt;/dt> &lt;dd>/home/rpausch/raycd.m4v&lt;/dd>
   &lt;dt>Remote filename:&lt;/dt> &lt;dd>/var/www/lectures/raycd.m4v&lt;/dd>
   &lt;dt>Duration:&lt;/dt> &lt;dd>01:16:27&lt;/dd>
   &lt;dt>Color profile:&lt;/dt> &lt;dd>SD (6-1-6)&lt;/dd>
   &lt;dt>Dimensions:&lt;/dt> &lt;dd>320&times;240&lt;/dd>
  &lt;/dl>
 &lt;/details>
&lt;/section></code></pre>

  </div>

  <div class="example">

   <p>The following shows how a <code>details</code> element can be used to hide some controls by
   default:</p>

   <pre><code class="html">&lt;details>
 &lt;summary>&lt;label for=fn>Name &amp; Extension:&lt;/label>&lt;/summary>
 &lt;p>&lt;input type=text id=fn name=fn value="Pillar Magazine.pdf">
 &lt;p>&lt;label>&lt;input type=checkbox name=ext checked> Hide extension&lt;/label>
&lt;/details></code></pre>

   <p>One could use this in conjunction with other <code>details</code> in a list to allow the user
   to collapse a set of fields down to a small set of headings, with the ability to open each
   one.</p>

   <p class="details-example"><img src="/images/sample-details-1.png" width="345" height="611" alt=""><img src="/images/sample-details-2.png" width="345" height="666" alt=""></p>

   <p>In these examples, the summary really just summarizes what the controls can change, and not
   the actual values, which is less than ideal.</p>

  </div>

  <div class="example" id="example-details-exclusive-accordion">
   <p>The following example shows the <code data-x="attr-details-name">name</code> attribute of the
   <code>details</code> element being used to create an exclusive accordion, a set of
   <code>details</code> elements where a user action to open one <code>details</code> element causes
   any open <code>details</code> to close.</p>

   <pre><code class="html">&lt;section class="characteristics">
 &lt;details name="frame-characteristics">
  &lt;summary>Material&lt;/summary>
  The picture frame is made of solid oak wood.
 &lt;/details>
 &lt;details name="frame-characteristics">
  &lt;summary>Size&lt;/summary>
  The picture frame fits a photo 40cm tall and 30cm wide.
  The frame is 45cm tall, 35cm wide, and 2cm thick.
 &lt;/details>
 &lt;details name="frame-characteristics">
  &lt;summary>Color&lt;/summary>
  The picture frame is available in its natural wood
  color, or with black stain.
 &lt;/details>
&lt;/section></code></pre>
  </div>

  <div class="example" id="example-details-exclusive-accordion-setting-open">
   <p>The following example shows what happens when the <code data-x="attr-details-open">open</code>
   attribute is set on a <code>details</code> element that is part of a set of elements using the
   <code data-x="attr-details-name">name</code> attribute to create an exclusive accordion.</p>

   <p>Given the initial markup:</p>

   <pre><code class="html">&lt;section class="characteristics">
 &lt;details name="frame-characteristics" id="d1" open>...&lt;/details>
 &lt;details name="frame-characteristics" id="d2">...&lt;/details>
 &lt;details name="frame-characteristics" id="d3">...&lt;/details>
&lt;/section></code></pre>

   <p>and the script:</p>

   <pre><code class="js">document.getElementById("d2").setAttribute("open", "");</code></pre>

   <p>then the resulting tree after the script executes will be equivalent to the markup:</p>

   <pre><code class="html">&lt;section class="characteristics">
 &lt;details name="frame-characteristics" id="d1">...&lt;/details>
 &lt;details name="frame-characteristics" id="d2" open>...&lt;/details>
 &lt;details name="frame-characteristics" id="d3">...&lt;/details>
&lt;/section></code></pre>

   <p>because setting the <code data-x="attr-details-open">open</code> attribute on <code
   data-x="">d2</code> removes it from <code data-x="">d1</code>.</p>

   <p>The same happens when the user activates the <code>summary</code> element inside of <code
   data-x="">d2</code>.</p>
  </div>

  <div class="example">

   <p>Because the <code data-x="attr-details-open">open</code> attribute is added and removed
   automatically as the user interacts with the control, it can be used in CSS to style the element
   differently based on its state. Here, a style sheet is used to animate the color of the summary
   when the element is opened or closed:</p>

   <pre><code class="html">&lt;style>
 details > summary { transition: color 1s; color: black; }
 details[open] > summary { color: red; }
&lt;/style>
&lt;details>
 &lt;summary>Automated Status: Operational&lt;/summary>
 &lt;p>Velocity: 12m/s&lt;/p>
 &lt;p>Direction: North&lt;/p>
&lt;/details></code></pre>

  </div>


  <h4>The <dfn element><code>summary</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd>None.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>As the <span>first child</span> of a <code>details</code> element.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Phrasing content</span>, optionally intermixed with <span>heading content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-summary">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-summary">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>summary</code> element <span>represents</span> a summary, caption, or legend for the
  rest of the contents of the <code>summary</code> element's parent <code>details</code>
  element<span w-nodev>, if any</span>.</p>

  <div w-nodev>

  <p>A <code>summary</code> element is a <dfn>summary for its parent details</dfn> if the following
  algorithm returns true:</p>

  <ol>
   <li><p>If this <code>summary</code> element has no parent, then return false.</p></li>

   <li><p>Let <var>parent</var> be this <code>summary</code> element's parent.</p></li>

   <li><p>If <var>parent</var> is not a <code>details</code> element, then return false.</p></li>

   <li><p>If <var>parent</var>'s first <code>summary</code> element child is not this
   <code>summary</code> element, then return false.</p></li>

   <li><p>Return true.</p></li>
  </ol>

  <p>The <span>activation behavior</span> of <code>summary</code> elements is to run the following
  steps:</p>

  <ol>
   <li><p>If this <code>summary</code> element is not the <span>summary for its parent
   details</span>, then return.</p></li>

   <li><p>Let <var>parent</var> be this <code>summary</code> element's parent.</p></li>

   <li>
    <p>If the <code data-x="attr-details-open">open</code> attribute is present on
    <var>parent</var>, then <span data-x="concept-element-attributes-remove">remove</span> it.
    Otherwise, <span data-x="concept-element-attributes-set-value">set</span> <var>parent</var>'s
    <code data-x="attr-details-open">open</code> attribute to the empty string.</p>

    <p class="note">This will then run the <span>details notification task steps</span>.</p>
   </li>
  </ol>

  </div>




  <h4 id="commands">Commands</h4>

  <h5>Facets</h5>

  <p>A <dfn data-x="concept-command">command</dfn> is the abstraction behind menu items, buttons, and
  links. Once a command is defined, other parts of the interface can refer to the same command,
  allowing many access points to a single feature to share facets such as the <span
  data-x="command-facet-DisabledState">Disabled State</span>.</p>

  <p id="facets">Commands are defined to have the following <dfn
  data-x="concept-facet">facets</dfn>:</p>

  <dl>
   <dt><dfn data-x="command-facet-Label">Label</dfn></dt>

   <dd>The name of the command as seen by the user.</dd>

   <dt><dfn data-x="command-facet-AccessKey">Access Key</dfn></dt>

   <dd>A key combination selected by the user agent that triggers the command. A command might not
   have an Access Key.</dd>

   <dt><dfn data-x="command-facet-HiddenState">Hidden State</dfn></dt>

   <dd>Whether the command is hidden or not (basically, whether it should be shown in menus).</dd>

   <dt><dfn data-x="command-facet-DisabledState">Disabled State</dfn></dt>

   <dd>Whether the command is relevant and can be triggered or not.</dd>

   <dt><dfn data-x="command-facet-Action">Action</dfn></dt>

   <dd>The actual effect that triggering the command will have. This could be a scripted event
   handler, a <span>URL</span> to which to <span>navigate</span>, or a form submission.</dd>
  </dl>

  <p id="expose-commands-in-ui">User agents may expose the <span
  data-x="concept-command">commands</span> that match the following criteria:</p>

  <ul>
   <li><p>The <span data-x="command-facet-HiddenState">Hidden State</span> facet is false
   (visible)</p></li>

   <li><p>The element is <span>in a document</span> with a non-null <span
   data-x="concept-document-bc">browsing context</span>.</p></li>

   <li><p>Neither the element nor any of its ancestors has a <code
   data-x="attr-hidden">hidden</code> attribute specified.</p></li>
  </ul>

  <p>User agents are encouraged to do this especially for commands that have <span
  data-x="command-facet-AccessKey">Access Keys</span>, as a way to advertise those keys to the
  user.</p>

  <p class="example">For example, such commands could be listed in the user agent's menu bar.</p>


  <div w-nodev>

  <h5><dfn data-x="a-command">Using the <code>a</code> element to define a command</dfn></h5>

  <p>An <code>a</code> element with an <code data-x="attr-hyperlink-href">href</code> attribute <span
  data-x="concept-command">defines a command</span>.</p>

  <p>The <span data-x="command-facet-Label">Label</span> of the command is the element's
  <span>descendant text content</span>.</p>

  <p>The <span data-x="command-facet-AccessKey">Access Key</span> of the command is the element's
  <span>assigned access key</span>, if any.</p>

  <p>The <span data-x="command-facet-HiddenState">Hidden State</span> of the command is true (hidden)
  if the element has a <code data-x="attr-hidden">hidden</code> attribute, and false otherwise.</p>

  <p>The <span data-x="command-facet-DisabledState">Disabled State</span> facet of the command is
  true if the element or one of its ancestors is <span>inert</span>, and false otherwise.</p>

  <p>The <span data-x="command-facet-Action">Action</span> of the command is to <span>fire a <code
  data-x="event-click">click</code> event</span> at the element.</p>


  <h5><dfn data-x="button-command">Using the <code>button</code> element to define a command</dfn></h5>

  <p>A <code>button</code> element always <span data-x="concept-command">defines a
  command</span>.</p>

  <p>The <span data-x="command-facet-Label">Label</span>, <span
  data-x="command-facet-AccessKey">Access Key</span>, <span
  data-x="command-facet-HiddenState">Hidden State</span>, and <span
  data-x="command-facet-Action">Action</span> facets of the command are determined <span
  data-x="a-command">as for <code>a</code> elements</span> (see the previous section).</p>

  <p>The <span data-x="command-facet-DisabledState">Disabled State</span> of the command is true if
  the element or one of its ancestors is <span>inert</span>, or if the element's <span
  data-x="concept-fe-disabled">disabled</span> state is set, and false otherwise.</p>


  <h5><dfn data-x="input-command">Using the <code>input</code> element to define a command</dfn></h5>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  one of the <span data-x="attr-input-type-submit">Submit Button</span>, <span
  data-x="attr-input-type-reset">Reset Button</span>, <span data-x="attr-input-type-image">Image
  Button</span>, <span data-x="attr-input-type-button">Button</span>, <span
  data-x="attr-input-type-radio">Radio Button</span>, or <span
  data-x="attr-input-type-checkbox">Checkbox</span> states <span data-x="concept-command">defines a
  command</span>.</p>

  <p>The <span data-x="command-facet-Label">Label</span> of the command is determined as
  follows:</p>

  <ul>
    <li><p>If the <code data-x="attr-input-type">type</code> attribute is in one of the
    <span data-x="attr-input-type-submit">Submit Button</span>, <span
    data-x="attr-input-type-reset">Reset Button</span>, <span data-x="attr-input-type-image">Image
    Button</span>, or <span data-x="attr-input-type-button">Button</span> states, then the
    <span data-x="command-facet-Label">Label</span> is the string given by the
    <code data-x="attr-input-value">value</code> attribute, if any, and a UA-dependent,
    locale-dependent value that the UA uses to label the button itself if the attribute is
    absent.</p></li>

    <li><p>Otherwise, if the element is a <span>labeled control</span>, then the <span
    data-x="command-facet-Label">Label</span> is the <span>descendant text content</span> of the
    first <code>label</code> element in <span>tree order</span> whose <span>labeled control</span>
    is the element in question. (In JavaScript terms, this is given by <code
    data-x=""><var>element</var>.labels[0].textContent</code>.)</p></li>

    <li><p>Otherwise, if the <code data-x="attr-input-value">value</code> attribute is present, then
    the <span data-x="command-facet-Label">Label</span> is the value of that attribute.</p></li>

    <li><p>Otherwise, the <span data-x="command-facet-Label">Label</span> is the empty string.</p></li>
  </ul>

  <div class="note">
    <p>Even though the <code data-x="attr-input-value">value</code> attribute on
    <code>input</code> elements in the <span data-x="attr-input-type-image">Image Button</span>
    state is non-conformant, the attribute can still contribute to the <span
    data-x="command-facet-Label">Label</span> determination, if it is present and the Image Button's
    <code data-x="attr-input-alt">alt</code> attribute is missing.</p>
  </div>

  <p>The <span data-x="command-facet-AccessKey">Access Key</span> of the command is the element's
  <span>assigned access key</span>, if any.</p>

  <p>The <span data-x="command-facet-HiddenState">Hidden State</span> of the command is true (hidden)
  if the element has a <code data-x="attr-hidden">hidden</code> attribute, and false otherwise.</p>

  <p>The <span data-x="command-facet-DisabledState">Disabled State</span> of the command is true if
  the element or one of its ancestors is <span>inert</span>, or if the element's <span
  data-x="concept-fe-disabled">disabled</span> state is set, and false otherwise.</p>

  <p>The <span data-x="command-facet-Action">Action</span> of the command is to <span>fire a <code
  data-x="event-click">click</code> event</span> at the element.</p>


  <h5><dfn data-x="option-command">Using the <code>option</code> element to define a command</dfn></h5>

  <p>An <code>option</code> element with an ancestor <code>select</code> element and either no <code
  data-x="attr-option-value">value</code> attribute or a <code data-x="attr-option-value">value</code>
  attribute that is not the empty string <span data-x="concept-command">defines a command</span>.</p>

  <p>The <span data-x="command-facet-Label">Label</span> of the command is the value of the
  <code>option</code> element's <code data-x="attr-option-label">label</code> attribute, if there is
  one, or else the <code>option</code> element's <span>descendant text content</span>, with <span
  data-x="strip and collapse ASCII whitespace">ASCII whitespace stripped and collapsed</span>.</p>

  <p>The <span data-x="command-facet-AccessKey">Access Key</span> of the command is the element's
  <span>assigned access key</span>, if any.</p>

  <p>The <span data-x="command-facet-HiddenState">Hidden State</span> of the command is true (hidden)
  if the element has a <code data-x="attr-hidden">hidden</code> attribute, and false otherwise.</p>

  <p>The <span data-x="command-facet-DisabledState">Disabled State</span> of the command is true if
  the element is <span data-x="concept-option-disabled">disabled</span>, or if its nearest ancestor
  <code>select</code> element is <span data-x="concept-fe-disabled">disabled</span>, or if it or one
  of its ancestors is <span>inert</span>, and false otherwise.</p>

  <p>If the <code>option</code>'s nearest ancestor <code>select</code> element has a <code
  data-x="attr-select-multiple">multiple</code> attribute, the <span
  data-x="command-facet-Action">Action</span> of the command is to <span
  data-x="concept-select-toggle">toggle</span> the <code>option</code> element. Otherwise, the <span
  data-x="command-facet-Action">Action</span> is to <span
  data-x="concept-select-pick">pick</span> the <code>option</code> element.</p>


  <h5><dfn data-x="legend-command">Using the <code data-x="attr-accesskey">accesskey</code> attribute
  on a <code>legend</code> element to define a command</dfn></h5>

  <p>A <code>legend</code> element <span data-x="concept-command">defines a command</span> if all of
  the following are true:</p>

  <ul>
   <li><p>It has an <span>assigned access key</span>.</p></li>

   <li><p>It is a child of a <code>fieldset</code> element.</p></li>

   <li><p>Its parent has a descendant that <span data-x="concept-command">defines a command</span>
   that is neither a <code>label</code> element nor a <code>legend</code> element. This element,
   if it exists, is <dfn>the <code>legend</code> element's <code>accesskey</code>
   delegatee</dfn>.</p></li>
  </ul>

  <p>The <span data-x="command-facet-Label">Label</span> of the command is the element's
  <span>descendant text content</span>.</p>

  <p>The <span data-x="command-facet-AccessKey">Access Key</span> of the command is the element's
  <span>assigned access key</span>.</p>

  <p>The <span data-x="command-facet-HiddenState">Hidden State</span>, <span
  data-x="command-facet-DisabledState">Disabled State</span>, and <span
  data-x="command-facet-Action">Action</span> facets of the command are the same as the respective
  facets of <span>the <code>legend</code> element's <code>accesskey</code> delegatee</span>.</p>

  <div class="example">
   <p>In this example, the <code>legend</code> element specifies an <code
   data-x="attr-accesskey">accesskey</code>, which, when activated, will delegate to the
   <code>input</code> element inside the <code>legend</code> element.</p>

   <pre><code class="html">&lt;fieldset>
 &lt;legend accesskey=p>
  &lt;label>I want &lt;input name=pizza type=number step=1 value=1 min=0>
   pizza(s) with these toppings&lt;/label>
 &lt;/legend>
 &lt;label>&lt;input name=pizza-cheese type=checkbox checked> Cheese&lt;/label>
 &lt;label>&lt;input name=pizza-ham type=checkbox checked> Ham&lt;/label>
 &lt;label>&lt;input name=pizza-pineapple type=checkbox> Pineapple&lt;/label>
&lt;/fieldset></code></pre>
  </div>


  <h5><dfn data-x="accesskey-command">Using the <code data-x="attr-accesskey">accesskey</code>
  attribute to define a command on other elements</dfn></h5>

  <p>An element that has an <span>assigned access key</span> <span data-x="concept-command">defines a
  command</span>.</p>

  <p>If one of the earlier sections that define elements that <span data-x="concept-command">define
  commands</span> define that this element <span data-x="concept-command">defines a command</span>,
  then that section applies to this element, and this section does not. Otherwise, this section
  applies to that element.</p>

  <p>The <span data-x="command-facet-Label">Label</span> of the command depends on the element. If
  the element is a <span>labeled control</span>, the <span>descendant text content</span> of the
  first <code>label</code> element in <span>tree order</span> whose <span>labeled control</span> is
  the element in question is the <span data-x="command-facet-Label">Label</span> (in JavaScript
  terms, this is given by <code data-x=""><var>element</var>.labels[0].textContent</code>).
  Otherwise, the <span data-x="command-facet-Label">Label</span> is the element's <span>descendant
  text content</span>.</p>

  <p>The <span data-x="command-facet-AccessKey">Access Key</span> of the command is the element's
  <span>assigned access key</span>.</p>

  <p>The <span data-x="command-facet-HiddenState">Hidden State</span> of the command is true (hidden)
  if the element has a <code data-x="attr-hidden">hidden</code> attribute, and false otherwise.</p>

  <p>The <span data-x="command-facet-DisabledState">Disabled State</span> of the command is true if
  the element or one of its ancestors is <span>inert</span>, and false otherwise.</p>

  <p>The <span data-x="command-facet-Action">Action</span> of the command is to run the following
  steps:</p>

  <ol>
   <li>Run the <span>focusing steps</span> for the element.</li>

   <li><span>Fire a <code data-x="event-click">click</code> event</span> at the element.</li>
  </ol>

  </div>


  <h4>The <dfn element><code>dialog</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>flow content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-dialog-closedby">closedby</code></dd>
   <dd><code data-x="attr-dialog-open">open</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-dialog">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-dialog">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLDialogElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute boolean <span data-x="dom-dialog-open">open</span>;
  attribute DOMString <span data-x="dom-dialog-returnValue">returnValue</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-dialog-closedBy">closedBy</span>;
  [<span>CEReactions</span>] undefined <span data-x="dom-dialog-show">show</span>();
  [<span>CEReactions</span>] undefined <span data-x="dom-dialog-showModal">showModal</span>();
  [<span>CEReactions</span>] undefined <span data-x="dom-dialog-close">close</span>(optional DOMString returnValue);
  [<span>CEReactions</span>] undefined <span data-x="dom-dialog-requestClose">requestClose</span>(optional DOMString returnValue);
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLDialogElement</code>.</dd>
  </dl>

  <p>The <code>dialog</code> element represents a transitory part of an application, in the form of
  a small window ("dialog box"), which the user interacts with to perform a task or gather
  information. Once the user is done, the dialog can be automatically closed by the application, or
  manually closed by the user.</p>

  <p>Especially for modal dialogs, which are a familiar pattern across all types of applications,
  authors should work to ensure that dialogs in their web applications behave in a way that is
  familiar to users of non-web applications.</p>

  <p class="note">As with all HTML elements, it is not conforming to use the <code>dialog</code>
  element when attempting to represent another type of control. For example, context menus,
  tooltips, and popup listboxes are not dialog boxes, so abusing the <code>dialog</code> element to
  implement these patterns is incorrect.</p>

  <p>An important part of user-facing dialog behavior is the placement of initial focus. The
  <span>dialog focusing steps</span> attempt to pick a good candidate for initial focus when a
  dialog is shown, but might not be a substitute for authors carefully thinking through the correct
  choice to match user expectations for a specific dialog. As such, authors should use the <code
  data-x="attr-fe-autofocus">autofocus</code> attribute on the descendant element of the dialog that
  the user is expected to immediately interact with after the dialog opens. If there is no such
  element, then authors should use the <code data-x="attr-fe-autofocus">autofocus</code> attribute
  on the <code>dialog</code> element itself.</p>

  <div class="example">
   <p>In the following example, a dialog is used for editing the details of a product in an
   inventory management web application.</p>

   <pre><code class="html">&lt;dialog>
  &lt;label>Product Number &lt;input type="text" readonly>&lt;/label>
  &lt;label>Product Name &lt;input type="text" autofocus>&lt;/label>
&lt;/dialog></code></pre>

   <p>If the <code data-x="attr-fe-autofocus">autofocus</code> attribute was not present, the
   Product Number field would have been focused by the dialog focusing steps. Although that is
   reasonable behavior, the author determined that the more relevant field to focus was the Product
   Name field, as the Product Number field is readonly and expects no user input. So, the author
   used autofocus to override the default.</p>

   <p>Even if the author wants to focus the Product Number field by default, they are best off
   explicitly specifying that by using autofocus on that <code>input</code> element. This makes the
   intent obvious to future readers of the code, and ensures the code stays robust in the face of
   future updates. (For example, if another developer added a close button, and positioned it in the
   node tree before the Product Number field).</p>
  </div>

  <p>Another important aspect of user behavior is whether dialogs are scrollable or not. In some
  cases, overflow (and thus scrollability) cannot be avoided, e.g., when it is caused by the user's
  high text zoom settings. But in general, scrollable dialogs are not expected by users. Adding
  large text nodes directly to dialog elements is particularly bad as this is likely to cause the
  dialog element itself to overflow. Authors are best off avoiding them.</p>

  <div class="example">
   <p>The following terms of service dialog respects the above suggestions.</p>

   <pre><code class="html">&lt;dialog style="height: 80vh;">
  &lt;div style="overflow: auto; height: 60vh;" autofocus>
    &lt;p>By placing an order via this Web site on the first day of the fourth month of the year
    2010 Anno Domini, you agree to grant Us a non-transferable option to claim, for now and for
    ever more, your immortal soul.&lt;/p>
    &lt;p>Should We wish to exercise this option, you agree to surrender your immortal soul,
    and any claim you may have on it, within 5 (five) working days of receiving written
    notification from  this site or one of its duly authorized minions.&lt;/p>
    &lt;!-- ... etc., with many more &lt;p> elements ... -->
  &lt;/div>
  &lt;form method="dialog">
    &lt;button type="submit" value="agree">Agree&lt;/button>
    &lt;button type="submit" value="disagree">Disagree&lt;/button>
  &lt;/form>
&lt;/dialog></code></pre>

   <p>Note how the <span>dialog focusing steps</span> would have picked the scrollable
   <code>div</code> element by default, but similarly to the previous example, we have placed <code
   data-x="attr-fe-autofocus">autofocus</code> on the <code>div</code> so as to be more explicit and
   robust against future changes.</p>

   <p>In contrast, if the <code>p</code> elements expressing the terms of service did not have such
   a wrapper <code>div</code> element, then the <code>dialog</code> itself would become scrollable,
   violating the above advice. Furthermore, in the absence of any <code
   data-x="attr-fe-autofocus">autofocus</code> attribute, such a markup pattern would have violated
   the above advice and tripped up the <span>dialog focusing steps</span>'s default behavior, and
   caused focus to jump to the Agree <code>button</code>, which is a bad user experience.</p>
  </div>

  <div class="example">
   <p>This dialog box has some small print. The <code>strong</code> element is used to draw the
   user's attention to the more important part.</p>

   <pre><code class="html">&lt;dialog>
 &lt;h1>Add to Wallet&lt;/h1>
 &lt;p>&lt;strong>&lt;label for=amt>How many gold coins do you want to add to your wallet?&lt;/label>&lt;/strong>&lt;/p>
 &lt;p>&lt;input id=amt name=amt type=number min=0 step=0.01 value=100>&lt;/p>
 &lt;p>&lt;small>You add coins at your own risk.&lt;/small>&lt;/p>
 &lt;p>&lt;label>&lt;input name=round type=checkbox> Only add perfectly round coins&lt;/label>&lt;/p>
 &lt;p>&lt;input type=button onclick="submit()" value="Add Coins">&lt;/p>
&lt;/dialog></code></pre>
  </div>

  <hr>

  <p>The <dfn element-attr for="dialog"><code data-x="attr-dialog-open">open</code></dfn> attribute
  is a <span>boolean attribute</span>. When specified, it indicates that the <code>dialog</code>
  element is active and that the user can interact with it.</p>

  <p>The <dfn element-attr for="dialog"><code data-x="attr-dialog-closedby">closedby</code></dfn>
  content attribute is an <span>enumerated attribute</span> with the following keywords and
  states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="dialog/closedby"><code data-x="attr-dialog-closedby-any">any</code></dfn>
     <td><dfn data-x="attr-dialog-closedby-any-state">Any</dfn>
     <td><span data-x="close request">Close requests</span> or clicks outside close the dialog.
    <tr>
     <td><dfn attr-value for="dialog/closedby"><code data-x="attr-dialog-closedby-closerequest">closerequest</code></dfn>
     <td><dfn data-x="attr-dialog-closedby-closerequest-state">Close Request</dfn>
     <td><span data-x="close request">Close requests</span> close the dialog.
    <tr>
     <td><dfn attr-value for="dialog/closedby"><code data-x="attr-dialog-closedby-none">none</code></dfn>
     <td><dfn data-x="attr-dialog-closedby-none-state">None</dfn>
     <td>No user actions automatically close the dialog.
  </table>

  <p>The <code data-x="attr-dialog-closedby">closedby</code> attribute's <i data-x="invalid value
  default">invalid value default</i> and <i data-x="missing value default">missing value
  default</i> are both the <dfn data-x="attr-dialog-closedby-auto-state">Auto</dfn> state.</p>

  <p class="note">The <span data-x="attr-dialog-closedby-auto-state">Auto</span> state behaves as
  <span data-x="attr-dialog-closedby-closerequest-state">Close Request</span> state when the
  <code>dialog</code> was shown using its <code data-x="dom-dialog-showModal">showModal()</code>
  method; otherwise the <span data-x="attr-dialog-closedby-none-state">None</span> state.</p>

  <div w-nodev>

  <p>A <code>dialog</code> element without an <code data-x="attr-dialog-open">open</code> attribute
  specified should not be shown to the user. This requirement may be implemented indirectly through
  the style layer. For example, user agents that <a href="#renderingUA">support the suggested
  default rendering</a> implement this requirement using the CSS rules described in the <a
  href="#rendering">Rendering section</a>.</p>

  </div>

  <div class="note" id="note-dialog-remove-open-attribute">
   <p>Removing the <code data-x="attr-dialog-open">open</code> attribute will usually hide the
   dialog. However, doing so has a number of strange additional consequences:

   <ul>
    <li><p>The <code data-x="event-close">close</code> event will not be fired.</p></li>

    <li><p>The <code data-x="dom-dialog-close">close()</code> method, and any <span data-x="close
    request">close requests</span>, will no longer be able to close the dialog.</p></li>

    <li><p>If the dialog was shown using its <code data-x="dom-dialog-showModal">showModal()</code>
    method, the <code>Document</code> will still be <span data-x="blocked by a modal
    dialog">blocked</span>.</p></li>
   </ul>

   <p>For these reasons, it is generally better to never remove the <code
   data-x="attr-dialog-open">open</code> attribute manually. Instead, use the <code
   data-x="dom-dialog-requestclose">requestClose()</code> or <code
   data-x="dom-dialog-close">close()</code> methods to close the dialog, or the <code
   data-x="attr-hidden">hidden</code> attribute to hide it.</p>
  </div>

  <p>The <code data-x="attr-tabindex">tabindex</code> attribute must not be specified on
  <code>dialog</code> elements.</p>

  <hr>

  <dl class="domintro">
   <dt><code data-x=""><var>dialog</var>.<span subdfn data-x="dom-dialog-show">show</span>()</code></dt>
   <dd><p>Displays the <code>dialog</code> element.</p></dd>

   <dt><code data-x=""><var>dialog</var>.<span subdfn data-x="dom-dialog-showModal">showModal</span>()</code></dt>
   <dd>
    <p>Displays the <code>dialog</code> element and makes it the top-most modal dialog.</p>

    <p>This method honors the <code data-x="attr-fe-autofocus">autofocus</code> attribute.</p>
   </dd>

   <dt><code data-x=""><var>dialog</var>.<span subdfn data-x="dom-dialog-close">close</span>([ <var>result</var> ])</code></dt>
   <dd>
    <p>Closes the <code>dialog</code> element.</p>

    <p>The argument, if provided, provides a return value.</p>
   </dd>

   <dt><code data-x=""><var>dialog</var>.<span subdfn data-x="dom-dialog-requestClose">requestClose</span>([ <var>result</var> ])</code></dt>
   <dd>
    <p>Acts as if a <span data-x="close request">close request</span> was sent targeting
    <var>dialog</var>, by first firing a <code data-x="event-cancel">cancel</code> event, and if
    that event is not canceled with <code data-x="dom-Event-preventDefault">preventDefault()</code>,
    proceeding to close the <var>dialog</var> in the same way as the <code
    data-x="dom-dialog-close">close()</code> method (including firing a <code
    data-x="event-close">close</code> event).</p>

    <p>This is a helper utility that can be used to consolidate cancelation and closing logic into
    the <code data-x="event-cancel">cancel</code> and <code data-x="event-close">close</code> event
    handlers, by having all non-<span data-x="close request">close request</span> closing
    affordances call this method.</p>

    <p>Note that this method ignores the <code data-x="attr-dialog-closedby">closedby</code>
    attribute: that is, even if <code data-x="attr-dialog-closedby">closedby</code> is set to
    "<code data-x="attr-dialog-closedby-none">none</code>", the same behavior will apply.</p>

    <p>The argument, if provided, provides a return value.</p>
   </dd>

   <dt><code data-x=""><var>dialog</var>.<span subdfn data-x="dom-dialog-returnValue">returnValue</span> [ = <var>result</var> ]</code></dt>
   <dd>
    <p>Returns the <code>dialog</code>'s return value.</p>

    <p>Can be set, to update the return value.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="HTMLDialogElement"><code data-x="dom-dialog-show">show()</code></dfn>
  method steps are:</p>

  <ol>
   <li><p>If <span>this</span> has an <code data-x="attr-dialog-open">open</code> attribute and
   <span>is modal</span> of <span>this</span> is false, then return.</p></li>

   <li><p>If <span>this</span> has an <code data-x="attr-dialog-open">open</code> attribute, then
   throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If the result of <span data-x="concept-event-fire">firing an event</span> named <code
   data-x="event-beforetoggle">beforetoggle</code>, using <code>ToggleEvent</code>, with the <code
   data-x="dom-Event-cancelable">cancelable</code> attribute initialized to true, the <code
   data-x="dom-ToggleEvent-oldState">oldState</code> attribute initialized to "<code
   data-x="">closed</code>", and the <code data-x="dom-ToggleEvent-newState">newState</code>
   attribute initialized to "<code data-x="">open</code>" at <span>this</span> is false, then
   return.</p></li>

   <li><p>If <span>this</span> has an <code data-x="attr-dialog-open">open</code> attribute, then
   return.</p></li>

   <li><p><span>Queue a dialog toggle event task</span> given <span>this</span>, "<code
   data-x="">closed</code>", and "<code data-x="">open</code>".</p></li>

   <li><p>Add an <code data-x="attr-dialog-open">open</code> attribute to <span>this</span>, whose
   value is the empty string.</p></li>

   <li><p><span>Assert</span>: <span>this</span>'s <span>node document</span>'s <span>open
   dialogs list</span> does not <span data-x="list contains">contain</span>
   <span>this</span>.</p></li>

   <li><p>Add <span>this</span> to <span>this</span>'s <span>node document</span>'s <span>open
   dialogs list</span>.</p></li>

   <li><p><span>Set the dialog close watcher</span> with <span>this</span>.</p></li>

   <li><p>Set <span>this</span>'s <span>previously focused element</span> to the
   <span>focused</span> element.</p></li>

   <li><p>Let <var>document</var> be <span>this</span>'s <span>node document</span>.</p></li>

   <li><p>Let <var>hideUntil</var> be the result of running <span>topmost popover ancestor</span>
   given <span>this</span>, <var>document</var>'s <span>showing hint popover list</span>, null, and
   false.</p></li>

   <li><p>If <var>hideUntil</var> is null, then set <var>hideUntil</var> to the result of running
   <span>topmost popover ancestor</span> given <span>this</span>, <var>document</var>'s
   <span>showing auto popover list</span>, null, and false.</p></li>

   <li><p>If <var>hideUntil</var> is null, then set <var>hideUntil</var> to
   <var>document</var>.</p></li>

   <li><p>Run <span data-x="hide-all-popovers-until">hide all popovers until</span> given
   <var>hideUntil</var>, false, and true.</p></li>

   <li><p>Run the <span>dialog focusing steps</span> given <span>this</span>.</p></li>
  </ol>

  <p>The <dfn method for="HTMLDialogElement"><code
  data-x="dom-dialog-showModal">showModal()</code></dfn> method steps are to <span>show a modal
  dialog</span> given <span>this</span>.</p>

  <p>The <dfn method for="HTMLDialogElement"><code
  data-x="dom-dialog-close">close(<var>returnValue</var>)</code></dfn> method steps are:

  <ol>
   <li><p>If <var>returnValue</var> is not given, then set it to null.</p></li>

   <li><p><span>Close the dialog</span> <span>this</span> with <var>returnValue</var>.</p></li>
  </ol>

  <p>The <dfn method for="HTMLDialogElement"><code
  data-x="dom-dialog-requestClose">requestClose(<var>returnValue</var>)</code></dfn> method steps
  are:</p>

  <ol>
   <li><p>If <span>this</span> does not have an <code data-x="attr-dialog-open">open</code>
   attribute, then return.</p></li>

   <li><p><span>Assert</span>: <span>this</span>'s <span data-x="dialog-close-watcher">close
   watcher</span> is not null.</p></li>

   <li><p>Set <var>dialog</var>'s <span>enable close watcher for <code
   data-x="dom-dialog-requestClose">requestClose()</code></span> to true.</p></li>

   <li><p>If <var>returnValue</var> is not given, then set it to null.</p></li>

   <li><p>Set <span>this</span>'s <span>request close return value</span> to
   <var>returnValue</var>.</p></li>

   <li><p><span data-x="close-watcher-request-close">Request to close</span> <var>dialog</var>'s
   <span data-x="dialog-close-watcher">close watcher</span> with false.</p></li>

   <li><p>Set <var>dialog</var>'s <span>enable close watcher for <code
   data-x="dom-dialog-requestClose">requestClose()</code></span> to false.</p></li>
  </ol>

  <div class="note" id="note-dialog-method-names">
   <p>We use show/close as the verbs for <code>dialog</code> elements, as opposed to verb pairs that
   are more commonly thought of as antonyms such as show/hide or open/close, due to the following
   constraints:</p>

   <ul>
    <li><p>Hiding a dialog is different from closing one. Closing a dialog gives it a return value,
    fires an event, unblocks the page for other dialogs, and so on. Whereas hiding a dialog is a
    purely visual property, and is something you can already do with the <code
    data-x="attr-hidden">hidden</code> attribute or by removing the <code
    data-x="attr-dialog-open">open</code> attribute. (See also the <a
    href="#note-dialog-remove-open-attribute">note above</a> about removing the <code
    data-x="attr-dialog-open">open</code> attribute, and how hiding the dialog in that way is
    generally not desired.)</p></li>

    <li><p>Showing a dialog is different from opening one. Opening a dialog consists of creating
    and showing that dialog (similar to how <code data-x="dom-open">window.open()</code> both
    creates and shows a new window). Whereas showing the dialog is the process of taking a
    <code>dialog</code> element that is already in the DOM, and making it interactive and visible
    to the user.</p></li>

    <li><p>If we were to have a <code data-x="">dialog.open()</code> method despite the above, it
    would conflict with the <code data-x="dom-dialog-open">dialog.open</code> property.</p></li>
   </ul>

   <p>Furthermore, a <a
   href="https://lists.whatwg.org/pipermail/whatwg-whatwg.org/2013-December/041799.html">survey</a>
   of many other UI frameworks contemporary to the original design of the <code>dialog</code>
   element made it clear that the show/close verb pair was reasonably common.</p>

   <p>In summary, it turns out that the implications of certain verbs, and how they are used in
   technology contexts, mean that paired actions such as showing and closing a dialog are not
   always expressible as antonyms.</p>
  </div>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLDialogElement"><code
  data-x="dom-dialog-returnValue">returnValue</code></dfn> IDL attribute, on getting, must return
  the last value to which it was set. On setting, it must be set to the new value. When the element
  is created, it must be set to the empty string.</p>

  <p>The <dfn attribute for="HTMLDialogElement"><code
  data-x="dom-dialog-closedBy">closedBy</code></dfn> getter steps are to return the keyword
  corresponding to the <span>computed closed-by state</span> given <span>this</span>.</p>

  <p>The <code data-x="dom-dialog-closedBy">closedBy</code> setter steps are to set the <code
  data-x="attr-dialog-closedby">closedby</code> content attribute to the given value.</p>

  <p>The <dfn attribute for="HTMLDialogElement"><code data-x="dom-dialog-open">open</code></dfn> IDL
  attribute must <span>reflect</span> the <code data-x="attr-dialog-open">open</code> content
  attribute.</p>

  <hr>

  <p>Each <code>Document</code> has a <dfn>dialog pointerdown target</dfn>, which is an <span
  data-x="HTMLDialogElement">HTML dialog element</span> or null, initially null.</p>

  <p>Each <span data-x="html elements">HTML element</span> has a <dfn>previously focused
  element</dfn> which is null or an element, and it is initially null. When <code
  data-x="dom-dialog-showModal">showModal()</code> and <code data-x="dom-dialog-show">show()</code>
  are called, this element is set to the currently <span>focused</span> element before running the
  <span>dialog focusing steps</span>. Elements with the <code data-x="attr-popover">popover</code>
  attribute set this element to the currently <span>focused</span> element during the <span
  data-x="show popover">show popover algorithm</span>.</p>

  <p>Each <code>dialog</code> element has a <dfn>dialog toggle task tracker</dfn>, which is a
  <span>toggle task tracker</span> or null, initially null.</p>

  <p>Each <code>dialog</code> element has a <dfn data-x="dialog-close-watcher">close watcher</dfn>,
  which is a <span>close watcher</span> or null, initially null.</p>

  <p>Each <code>dialog</code> element has a <dfn>request close return value</dfn>, which is a
  string, initially null.</p>

  <p>Each <code>dialog</code> element has an <dfn>enable close watcher for <code
  data-x="dom-dialog-requestClose">requestClose()</code></dfn> boolean, initially false.</p>

  <p>Each <code>dialog</code> element has an <dfn>is modal</dfn> boolean, initially false.</p>

  <hr>

  </div>

  <p>The <code>dialog</code> <span>HTML element removing steps</span>, given <var>removedNode</var>
  and <var>oldParent</var>, are:</p>

  <ol>
   <li>
    <p>If <var>removedNode</var>'s <span data-x="dialog-close-watcher">close watcher</span> is not
    null, then:</p>

    <ol>
     <li><p><span data-x="close-watcher-destroy">Destroy</span> <var>removedNode</var>'s <span
     data-x="dialog-close-watcher">close watcher</span>.</p></li>

     <li><p>Set <var>removedNode</var>'s <span data-x="dialog-close-watcher">close watcher</span> to
     null.</p></li>
    </ol>
   </li>

   <li><p>If <var>removedNode</var>'s <span>node document</span>'s <span>top layer</span> <span
   data-x="list contains">contains</span> <var>removedNode</var>, then <span>remove an element from
   the top layer immediately</span> given <var>removedNode</var>.</p></li>

   <li><p>Set <span>is modal</span> of <var>removedNode</var> to false.</p></li>

   <li><p><span data-x="list remove">Remove</span> <var>removedNode</var> from
   <var>removedNode</var>'s <span>node document</span>'s <span>open dialogs list</span>.</p></li>
  </ol>

  <p>To <dfn>show a modal dialog</dfn> given a <code>dialog</code> element <var>subject</var>:</p>

  <ol>
   <li><p>If <var>subject</var> has an <code data-x="attr-dialog-open">open</code> attribute and
   <span>is modal</span> of <var>subject</var> is true, then return.</p></li>

   <li><p>If <var>subject</var> has an <code data-x="attr-dialog-open">open</code> attribute, then
   throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <var>subject</var>'s <span>node document</span> is not <span>fully active</span>, then
   throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <var>subject</var> is not <span>connected</span>, then throw an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <var>subject</var> is in the <span data-x="popover-showing-state">popover showing
   state</span>, then throw an <span>"<code>InvalidStateError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>If the result of <span data-x="concept-event-fire">firing an event</span> named <code
   data-x="event-beforetoggle">beforetoggle</code>, using <code>ToggleEvent</code>, with the <code
   data-x="dom-Event-cancelable">cancelable</code> attribute initialized to true, the <code
   data-x="dom-ToggleEvent-oldState">oldState</code> attribute initialized to "<code
   data-x="">closed</code>", and the <code data-x="dom-ToggleEvent-newState">newState</code>
   attribute initialized to "<code data-x="">open</code>" at <var>subject</var> is false, then
   return.</p></li>

   <li><p>If <var>subject</var> has an <code data-x="attr-dialog-open">open</code> attribute,
   then return.</p></li>

   <li><p>If <var>subject</var> is not <span>connected</span>, then return.</p></li>

   <li><p>If <var>subject</var> is in the <span data-x="popover-showing-state">popover showing
   state</span>, then return.</p></li>

   <li><p><span>Queue a dialog toggle event task</span> given <var>subject</var>, "<code
   data-x="">closed</code>", and "<code data-x="">open</code>".</p></li>

   <li><p>Add an <code data-x="attr-dialog-open">open</code> attribute to <var>subject</var>, whose
   value is the empty string.</p></li>

   <li><p>Set <span>is modal</span> of <var>subject</var> to true.</p></li>

   <li><p><span>Assert</span>: <var>subject</var>'s <span>node document</span>'s <span>open
   dialogs list</span> does not <span data-x="list contains">contain</span>
   <var>subject</var>.</p></li>

   <li><p>Add <var>subject</var> to <var>subject</var>'s <span>node document</span>'s <span>open
   dialogs list</span>.</p></li>

   <li>
    <p>Let <var>subject</var>'s <span>node document</span> be <span data-x="blocked by a modal
    dialog">blocked by the modal dialog</span> <var>subject</var>.</p>

    <p class="note" id="note-dialog-plus-focus-fixup">This will cause the <span>focused area of the
    document</span> to become <span>inert</span> (unless that currently focused area is a
    <span>shadow-including descendant</span> of <var>subject</var>). In such cases, the
    <span>focused area of the document</span> will soon be <a href="#focus-fixup-rule">reset</a> to
    the <span>viewport</span>. In a couple steps we will attempt to find a better candidate to
    focus.</p>
   </li>

   <li><p>If <var>subject</var>'s <span>node document</span>'s <span>top layer</span> does not
   already <span data-x="list contains">contain</span> <var>subject</var>, then <span>add an element
   to the top layer</span> given <var>subject</var>.</p></li>

   <li><p><span>Set the dialog close watcher</span> with <var>subject</var>.</p></li>

   <li><p>Set <var>subject</var>'s <span>previously focused element</span> to the
   <span>focused</span> element.</p></li>

   <li><p>Let <var>document</var> be <var>subject</var>'s <span>node document</span>.</p></li>

   <li><p>Let <var>hideUntil</var> be the result of running <span>topmost popover ancestor</span>
   given <var>subject</var>, <var>document</var>'s <span>showing hint popover list</span>, null, and
   false.</p></li>

   <li><p>If <var>hideUntil</var> is null, then set <var>hideUntil</var> to the result of running
   <span>topmost popover ancestor</span> given <var>subject</var>, <var>document</var>'s
   <span>showing auto popover list</span>, null, and false.</p></li>

   <li><p>If <var>hideUntil</var> is null, then set <var>hideUntil</var> to
   <var>document</var>.</p></li>

   <li><p>Run <span data-x="hide-all-popovers-until">hide all popovers until</span> given
   <var>hideUntil</var>, false, and true.</p></li>

   <li><p>Run the <span>dialog focusing steps</span> given <var>subject</var>.</p></li>
  </ol>

  <p id="canceling-dialogs">To <dfn>set the dialog close watcher</dfn>, given a <code>dialog</code>
  element <var>dialog</var>:</p>

  <ol>
   <li>
    <p>Set <var>dialog</var>'s <span data-x="dialog-close-watcher">close watcher</span> to the
    result of <span data-x="establish a close watcher">establishing a close watcher</span> given
    <var>dialog</var>'s <span>relevant global object</span>, with:</p>

    <ul>
     <li><p><i data-x="create-close-watcher-cancelAction">cancelAction</i> given
     <var>canPreventClose</var> being to return the result of <span
     data-x="concept-event-fire">firing an event</span> named <code
     data-x="event-cancel">cancel</code> at <var>dialog</var>, with the <code
     data-x="dom-Event-cancelable">cancelable</code> attribute initialized to
     <var>canPreventClose</var>.</p></li>

     <li><p><i data-x="create-close-watcher-closeAction">closeAction</i> being to <span>close the
     dialog</span> given <var>dialog</var> and <var>dialog</var>'s <span>request close return
     value</span>.</p></li>

     <li><p><i data-x="create-close-watcher-getEnabledState">getEnabledState</i> being to return
     true if <var>dialog</var>'s <span>enable close watcher for <code
     data-x="dom-dialog-requestClose">requestClose()</code></span> is true or <var>dialog</var>'s
     <span>computed closed-by state</span> is not <span
     data-x="attr-dialog-closedby-none-state">None</span>; otherwise false.</p></li>
    </ul>
   </li>
  </ol>

  <p>The <span>is valid invoker command steps</span> for <code>dialog</code> elements, given
   a <code data-x="attr-button-command">command</code> attribute <var>command</var>, are:</p>

  <ol>
   <li><p>If <var>command</var> is in the <span
   data-x="attr-button-command-close-state">Close</span> state or in the <span
   data-x="attr-button-command-show-modal-state">Show Modal</span> state, then
   return true.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  <p>The <span>invoker command steps</span> for <code>dialog</code> elements, given an element
  <var>element</var>, an element <var>invoker</var>, and a <code
  data-x="attr-button-command">command</code> attribute <var>command</var>, are:</p>

  <ol>
   <li><p>If <var>element</var> is in the <span
   data-x="popover-showing-state">popover showing</span> state, then return.</p></li>

   <li>
    <p>If <var>command</var> is in the <span data-x="attr-button-command-close-state">Close</span>
    state and <var>element</var> has an <code data-x="attr-dialog-open">open</code> attribute:</p>

    <ol>
     <li><p>Let <var>value</var> be <var>invoker</var>'s <span
     data-x="concept-fe-value">value</span>.</p></li>

     <li><p><span>Close the dialog</span> <var>element</var> with <var>value</var>.</p></li>
    </ol>
   </li>

   <li><p>If <var>command</var> is the <span
   data-x="attr-button-command-show-modal-state">Show Modal</span> state and <var>element</var> does
   not have an <code data-x="attr-dialog-open">open</code> attribute, then
   <span>show a modal dialog</span> given <var>element</var>.</p></li>
  </ol>

  <div class="example">
   <p>The following buttons use <code data-x="attr-button-commandfor">commandfor</code> to open and
   close a "confirm" <code>dialog</code> as modal when activated:</p>

   <pre><code class="html">&lt;button type=button
        commandfor="the-dialog"
        command="show-modal">
 Delete
&lt;/button>
&lt;dialog id="the-dialog">
 &lt;form action="/delete" method="POST">
  &lt;button type="submit">
   Delete
  &lt;/button>
  &lt;button commandfor="the-dialog"
          command="close">
   Cancel
  &lt;/button>
 &lt;/form>
&lt;/dialog>
    </code></pre>
  </div>

  <p>When a <code>dialog</code> element <var>subject</var> is to be <dfn data-x="close the
  dialog">closed</dfn>, with null or a string <var>result</var>, run these steps:</p>

  <ol>
   <li><p>If <var>subject</var> does not have an <code data-x="attr-dialog-open">open</code>
   attribute, then return.</p></li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
   data-x="event-beforetoggle">beforetoggle</code>, using <code>ToggleEvent</code>, with the <code
   data-x="dom-ToggleEvent-oldState">oldState</code> attribute initialized to "<code
   data-x="">open</code>" and the <code data-x="dom-ToggleEvent-newState">newState</code> attribute
   initialized to "<code data-x="">closed</code>" at <var>subject</var>.</p></li>

   <li><p>If <var>subject</var> does not have an <code data-x="attr-dialog-open">open</code>
   attribute, then return.</p></li>

   <li><p><span>Queue a dialog toggle event task</span> given <var>subject</var>, "<code
   data-x="">open</code>", and "<code data-x="">closed</code>".</p></li>

   <li><p>Remove <var>subject</var>'s <code data-x="attr-dialog-open">open</code>
   attribute.</p></li>

   <li><p>If <span>is modal</span> of <var>subject</var> is true, then <span>request an
   element to be removed from the top layer</span> given <var>subject</var>.</p></li>

   <li><p>Let <var>wasModal</var> be the value of <var>subject</var>'s <span>is
   modal</span> flag.</p></li>

   <li><p>Set <span>is modal</span> of <var>subject</var> to false.</p></li>

   <li><p><span data-x="list remove">Remove</span> <var>subject</var> from <var>subject</var>'s
   <span>node document</span>'s <span>open dialogs list</span>.</p></li>

   <li><p>If <var>result</var> is not null, then set the <code
   data-x="dom-dialog-returnValue">returnValue</code> attribute to <var>result</var>.</p></li>

   <li><p>Set the <span>request close return value</span> to null.</p></li>

   <li>
    <p>If <var>subject</var>'s <span>previously focused element</span> is not null, then:</p>

    <ol>
     <li><p>Let <var>element</var> be <var>subject</var>'s <span>previously focused
     element</span>.</p></li>

     <li><p>Set <var>subject</var>'s <span>previously focused element</span> to null.</p></li>

     <li><p>If <var>subject</var>'s <span>node document</span>'s <span>focused area of the
     document</span>'s <span>DOM anchor</span> is a <span>shadow-including inclusive
     descendant</span> of <var>subject</var>, or <var>wasModal</var> is true, then run the
     <span>focusing steps</span> for <var>element</var>; the viewport should not be scrolled by
     doing this step.</p></li>
    </ol>
   </li>

   <li><p><span>Queue an element task</span> on the <span>user interaction task source</span> given the
   <var>subject</var> element to <span data-x="concept-event-fire">fire an event</span> named
   <code data-x="event-close">close</code> at <var>subject</var>.</p></li>

   <li>
    <p>If <var>subject</var>'s <span data-x="dialog-close-watcher">close watcher</span> is not null,
    then:</p>

    <ol>
     <li><p><span data-x="close-watcher-destroy">Destroy</span> <var>subject</var>'s <span
     data-x="dialog-close-watcher">close watcher</span>.</p></li>

     <li><p>Set <var>subject</var>'s <span data-x="dialog-close-watcher">close watcher</span> to
     null.</p></li>
    </ol>
   </li>
  </ol>

  </div>

  <p>To <dfn>queue a dialog toggle event task</dfn> given a <code>dialog</code> element
  <var>element</var>, a string <var>oldState</var>, and a string <var>newState</var>:

  <ol>
   <li>
    <p>If <var>element</var>'s <span>dialog toggle task tracker</span> is not null, then:</p>

    <ol>
     <li><p>Set <var>oldState</var> to <var>element</var>'s <span>dialog toggle task
     tracker</span>'s <span data-x="toggle-task-old-state">old state</span>.</p></li>

     <li><p>Remove <var>element</var>'s <span>dialog toggle task tracker</span>'s <span
     data-x="toggle-task-task">task</span> from its <span>task queue</span>.</p></li>

     <li><p>Set <var>element</var>'s <span>dialog toggle task tracker</span> to null.</p></li>
    </ol>
   </li>

   <li>
    <p><span>Queue an element task</span> given the <span>DOM manipulation task source</span> and
    <var>element</var> to run the following steps:</p>

    <ol>
     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-toggle">toggle</code> at <var>element</var>, using <code>ToggleEvent</code>, with
     the <code data-x="dom-ToggleEvent-oldState">oldState</code> attribute initialized to
     <var>oldState</var> and the <code data-x="dom-ToggleEvent-newState">newState</code> attribute
     initialized to <var>newState</var>.</p></li>

     <li><p>Set <var>element</var>'s <span>dialog toggle task tracker</span> to null.</p></li>
    </ol>
   </li>

   <li><p>Set <var>element</var>'s <span>dialog toggle task tracker</span> to a struct with <span
   data-x="toggle-task-task">task</span> set to the just-queued <span
   data-x="concept-task">task</span> and <span data-x="toggle-task-old-state">old state</span> set
   to <var>oldState</var>.</p></li>
  </ol>

  <p>To retrieve a dialog's <dfn>computed closed-by state</dfn>, given a <code>dialog</code>
  <var>dialog</var>:

  <ol>
   <li>
    <p>If the state of <var>dialog</var>'s <code data-x="attr-dialog-closedby">closedby</code>
    attribute is <span data-x="attr-dialog-closedby-auto-state">Auto</span>:</p>

    <ol>
     <li><p>If <var>dialog</var>'s <span>is modal</span> is true, then return <span
     data-x="attr-dialog-closedby-closerequest-state">Close Request</span>.</p></li>

     <li><p>Return <span data-x="attr-dialog-closedby-none-state">None</span>.</p></li>
    </ol>
   </li>

   <li><p>Return the state of <var>dialog</var>'s <code
   data-x="attr-dialog-closedby">closedby</code> attribute.</p></li>
  </ol>

  <p>The <dfn>dialog focusing steps</dfn>, given a <code>dialog</code> element <var>subject</var>,
  are as follows:</p>

  <ol>
   <li><p>If the <span>allow focus steps</span> given <var>subject</var>'s <span>node
   document</span> return false, then return.</p></li>

   <li><p>Let <var>control</var> be null.</p></li>

   <li><p>If <var>subject</var> has the <code data-x="attr-fe-autofocus">autofocus</code>
   attribute, then set <var>control</var> to <var>subject</var>.</p></li>

   <li><p>If <var>control</var> is null, then set <var>control</var> to the <span>focus
   delegate</span> of <var>subject</var>.</p></li>

   <li><p>If <var>control</var> is null, then set <var>control</var> to <var>subject</var>.</p></li>

   <li>
    <p>Run the <span>focusing steps</span> for <var>control</var>.</p>

    <p class="note">If <var>control</var> is not <span>focusable</span>, this will do nothing. This
    would only happen if subject had no focus delegate, and the user agent decided that
    <code>dialog</code> elements were not generally focusable. In that case, any <a
    href="#note-dialog-plus-focus-fixup">earlier modifications</a> to the <span>focused area of the
    document</span> will apply.</p>
   </li>

   <li><p>Let <var>topDocument</var> be <var>control</var>'s <span>node navigable</span>'s <span
   data-x="nav-top">top-level traversable</span>'s <span data-x="nav-document">active
   document</span>.</p></li>

   <li><p>If <var>control</var>'s <span>node document</span>'s <span
   data-x="concept-document-origin">origin</span> is not the <span data-x="same origin">same</span>
   as the <span data-x="concept-document-origin">origin</span> of <var>topDocument</var>, then
   return.</p></li>

   <li><p><span data-x="list empty">Empty</span> <var>topDocument</var>'s
   <span>autofocus candidates</span>.</p></li>

   <li><p>Set <var>topDocument</var>'s <span>autofocus processed flag</span> to true.</p></li>
  </ol>

  <h4><dfn>Dialog light dismiss</dfn></h4>

  <p class="note">"Light dismiss" means that clicking outside of a <code>dialog</code> element whose <code
  data-x="attr-dialog-closedby">closedby</code> attribute is in the <span
  data-x="attr-dialog-closedby-any-state">Any</span> state will close the <code>dialog</code> element. This
  is in addition to how such <code>dialog</code>s respond to <span data-x="close request">close requests</span>.</p>

  <p>To <dfn>light dismiss open dialogs</dfn>, given a <code>PointerEvent</code> <var>event</var>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>event</var>'s <code
   data-x="dom-Event-isTrusted">isTrusted</code> attribute is true.</p></li>

   <li><p>Let <var>document</var> be <var>event</var>'s <span
   data-x="concept-event-target">target</span>'s <span>node document</span>.</p></li>

   <li><p>If <var>document</var>'s <span>open dialogs list</span> is <span
   data-x="list is empty">empty</span>, then return.</p></li>

   <li><p>Let <var>ancestor</var> be the result of running <span>nearest clicked dialog</span>
   given <var>event</var>.</p></li>

   <li><p>If <var>event</var>'s <code data-x="dom-Event-type">type</code> is
   "<code data-x="event-pointerdown">pointerdown</code>", then set <var>document</var>'s
   <span>dialog pointerdown target</span> to <var>ancestor</var>.</p></li>

   <li>
    <p>If <var>event</var>'s <code data-x="dom-Event-type">type</code> is
    "<code data-x="event-pointerup">pointerup</code>", then:</p>

    <ol>
     <li><p>Let <var>sameTarget</var> be true if <var>ancestor</var> is <var>document</var>'s
     <span>dialog pointerdown target</span>.</p></li>

     <li><p>Set <var>document</var>'s <span>dialog pointerdown target</span> to null.</p></li>

     <li><p>If <var>sameTarget</var> is false, then return.</p></li>

     <li><p>Let <var>topmostDialog</var> be the last element of <var>document</var>'s <span>open
     dialogs list</span>.</p></li>

     <li><p>If <var>ancestor</var> is <var>topmostDialog</var>, then return.</p></li>

     <li><p>If <var>topmostDialog</var>'s <span>computed closed-by state</span> is not <span
     data-x="attr-dialog-closedby-any-state">Any</span>, then return.</p></li>

     <li><p><span>Assert</span>: <var>topmostDialog</var>'s <span
     data-x="dialog-close-watcher">close watcher</span> is not null.</p></li>

     <li><p><span data-x="close-watcher-request-close">Request to close</span>
     <var>topmostDialog</var>'s <span data-x="dialog-close-watcher">close watcher</span> with
     false.</p></li>
    </ol>
   </li>
  </ol>

  <p>To <dfn>run light dismiss activities</dfn>, given a <code>PointerEvent</code>
  <var>event</var>:</p>

  <ol>
   <li><p>Run <span>light dismiss open popovers</span> with <var>event</var>.</p></li>

   <li><p>Run <span>light dismiss open dialogs</span> with <var>event</var>.</p></li>
  </ol>

  <p class="XXX"><span>Run light dismiss activities</span> will be called by the <a
  href="https://github.com/w3c/pointerevents/pull/460">Pointer Events spec</a> when the user clicks
  or touches anywhere on the page.</p>

  <p>To find the <dfn>nearest clicked dialog</dfn>, given a <code>PointerEvent</code>
  <var>event</var>:</p>

  <ol>
   <li><p>Let <var>target</var> be <var>event</var>'s <span
   data-x="concept-event-target">target</span>.</p></li>

   <li><p>If <var>target</var> is a <code>dialog</code> element, <var>target</var> has an <code
   data-x="attr-dialog-open">open</code> attribute, <var>target</var>'s <span>is modal</span> is
   true, and <var>event</var>'s <code data-x="mouseevent-clientx">clientX</code> and
   <code data-x="mouseevent-clienty">clientY</code> are outside the bounds of <var>target</var>,
   then return null.

   <p class="note">The check for <code data-x="mouseevent-clientx">clientX</code> and <code
   data-x="mouseevent-clienty">clientY</code> is because a pointer event that hits the <code
   data-x="">::backdrop</code> pseudo element of a dialog will result in <var>event</var> having a
   target of the dialog element itself.</p>

   <li><p>Let <var>currentNode</var> be <var>target</var>.</p></li>

   <li>
    <p>While <var>currentNode</var> is not null:</p>

    <ol>
     <li><p>If <var>currentNode</var> is a <code>dialog</code> element and <var>currentNode</var>
     has an <code data-x="attr-dialog-open">open</code> attribute, then return
     <var>currentNode</var>.</p></li>

     <li><p>Set <var>currentNode</var> to <var>currentNode</var>'s parent in the <span>flat
     tree</span>.</p></li>
    </ol>
   </li>

   <li><p>Return null.</p></li>
  </ol>

  <h3 split-filename="scripting">Scripting</h3>

  <p>Scripts allow authors to add interactivity to their documents.</p>

  <p>Authors are encouraged to use declarative alternatives to scripting where possible, as
  declarative mechanisms are often more maintainable, and many users disable scripting.</p>

  <div class="example">

   <p>For example, instead of using a script to show or hide a section to show more details, the
   <code>details</code> element could be used.</p>

  </div>

  <p>Authors are also encouraged to make their applications degrade gracefully in the absence of
  scripting support.</p>

  <div class="example">

   <p>For example, if an author provides a link in a table header to dynamically resort the table,
   the link could also be made to function without scripts by requesting the sorted table from the
   server.</p>

  </div>


  <h4 id="the-script-element">The <dfn
  id="script" element><code>script</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Metadata content</span>.</dd>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span data-x="script-supporting elements">Script-supporting element</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>metadata content</span> is expected.</dd>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dd>Where <span>script-supporting elements</span> are expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>If there is no <code data-x="attr-script-src">src</code>
   attribute, depends on the value of the <code
   data-x="attr-script-type">type</code> attribute, but must match
   <span>script content restrictions</span>.</dd>
   <dd>If there <em>is</em> a <code data-x="attr-script-src">src</code>
   attribute, the element must be either empty or contain only
   <span>script documentation</span> that also matches <span>script
   content restrictions</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-script-src">src</code></dd>
   <dd><code data-x="attr-script-type">type</code></dd>
   <dd><code data-x="attr-script-nomodule">nomodule</code></dd>
   <dd><code data-x="attr-script-async">async</code></dd>
   <dd><code data-x="attr-script-defer">defer</code></dd>
   <dd><code data-x="attr-script-crossorigin">crossorigin</code></dd>
   <dd><code data-x="attr-script-integrity">integrity</code></dd>
   <dd><code data-x="attr-script-referrerpolicy">referrerpolicy</code></dd>
   <dd><code data-x="attr-script-blocking">blocking</code></dd>
   <dd><code data-x="attr-script-fetchpriority">fetchpriority</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-script">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-script">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLScriptElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute USVString <span data-x="dom-script-src">src</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-script-type">type</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-script-noModule">noModule</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-script-async">async</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-script-defer">defer</span>;
  [<span>CEReactions</span>] attribute DOMString? <span data-x="dom-script-crossOrigin">crossOrigin</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-script-text">text</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-script-integrity">integrity</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-script-referrerPolicy">referrerPolicy</span>;
  [SameObject, PutForwards=<span data-x="dom-DOMTokenList-value">value</span>] readonly attribute <span>DOMTokenList</span> <span data-x="dom-script-blocking">blocking</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-script-fetchPriority">fetchPriority</span>;

  static boolean <span data-x="dom-script-supports">supports</span>(DOMString type);

  // <a href="#HTMLScriptElement-partial">also has obsolete members</a>
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLScriptElement</code>.</dd>
  </dl>

  <p>The <code>script</code> element allows authors to include dynamic script and data blocks in
  their documents. The element does not <span data-x="represents">represent</span> content for the
  user.</p>

  <p>The <dfn element-attr for="script"><code data-x="attr-script-type">type</code></dfn> attribute
  allows customization of the type of script represented:</p>

  <ul>
    <li><p>Omitting the attribute, setting it to the empty string, or setting it to a
    <span>JavaScript MIME type essence match</span>, means that the script is a <span>classic
    script</span>, to be interpreted according to the JavaScript <i
    data-x="js-prod-Script">Script</i> top-level production. Classic scripts are affected by the
    <code data-x="attr-script-async">async</code> and <code data-x="attr-script-defer">defer</code>
    attributes, but only when the <code data-x="attr-script-src">src</code> attribute is set.
    Authors should omit the <code data-x="attr-script-type">type</code> attribute instead of
    redundantly setting it.</p></li>

    <li><p>Setting the attribute to an <span>ASCII case-insensitive</span> match for "<code
    data-x="">module</code>" means that the script is a <span>JavaScript module script</span>, to
    be interpreted according to the JavaScript <i data-x="js-prod-Module">Module</i> top-level
    production. Module scripts are not affected by the <code
    data-x="attr-script-defer">defer</code> attribute, but are affected by the <code
    data-x="attr-script-async">async</code> attribute (regardless of the state of the <code
    data-x="attr-script-src">src</code> attribute).</p></li>

    <li><p>Setting the attribute to an <span>ASCII case-insensitive</span> match for "<code
    data-x="">importmap</code>" means that the script is an <span>import map</span>, containing
    JSON that will be used to control the behavior of <span data-x="resolve a module
    specifier">module specifier resolution</span>. Import maps can only be inline, i.e., the <code
    data-x="attr-script-src">src</code> attribute and most other attributes are meaningless and not
    to be used with them.</p></li>

    <li><p>Setting the attribute to any other value means that the script is a <dfn>data
    block</dfn>, which is not processed. None of the <code>script</code> attributes (except <code
    data-x="attr-script-type">type</code> itself) have any effect on data blocks. Authors must use
    a <span>valid MIME type string</span> that is not a <span>JavaScript MIME type essence
    match</span> to denote data blocks.</p></li>
  </ul>

  <p class="note">The requirement that <span data-x="data block">data blocks</span>
  <!--non-normative-->must be denoted using a <span>valid MIME type string</span> is in place to
  avoid potential future collisions. If this specification ever adds additional types of
  <span data-x="concept-script">script</span>, they will be triggered by setting the <code
  data-x="attr-script-type">type</code> attribute to something which is not a MIME type, like how
  the "<code data-x="">module</code>" value denotes <span data-x="module script">module
  scripts</span>. By using a valid MIME type string now, you ensure that your data block will not
 ever be reinterpreted as a different script type, even in future user agents.</p>

  <p><span data-x="classic script">Classic scripts</span> and <span
  data-x="JavaScript module script">JavaScript module scripts</span> can be embedded inline, or be
  imported from an external file using the <dfn for="script" element-attr><code
  data-x="attr-script-src">src</code></dfn> attribute, which if specified gives the <span>URL</span>
  of the external script resource to use. If <code data-x="attr-script-src">src</code> is specified,
  it must be a <span>valid non-empty URL potentially surrounded by spaces</span>.</p>

  <p>The contents of inline <code>script</code> elements, or the external script resource, must
  conform with the requirements of the JavaScript specification's <i
  data-x="js-prod-Script">Script</i> or <i data-x="js-prod-Module">Module</i> productions, for <span
  data-x="classic script">classic scripts</span> and <span data-x="JavaScript module
  script">JavaScript module scripts</span> respectively. <ref>JAVASCRIPT</ref></p>

  <p>The contents of the external script resource for <span data-x="CSS module script">CSS module
  scripts</span> must conform to the requirements of the CSS specification. <ref>CSS</ref></p>

  <p>The contents of the external script resource for <span data-x="JSON module script">JSON module
  scripts</span> must conform to the requirements of the JSON specification. <ref>JSON</ref></p>

  <p>The contents of inline <code>script</code> elements for <span data-x="import map">import
  maps</span> must conform with the <span>import map authoring requirements</span>.</p>

  <p>For <span>import map</span> <code>script</code> elements, the <code
  data-x="attr-script-src">src</code>, <code data-x="attr-script-async">async</code>, <code
  data-x="attr-script-nomodule">nomodule</code>, <code data-x="attr-script-defer">defer</code>,
  <code data-x="attr-script-crossorigin">crossorigin</code>, <code
  data-x="attr-script-integrity">integrity</code>, and <code
  data-x="attr-script-referrerpolicy">referrerpolicy</code> attributes must not be specified.</p>

  <p>A document must not have more than one <span>import map</span> <code>script</code> element.</p>

  <p>When used to include <span data-x="data block">data blocks</span>, the data must be embedded
  inline, the format of the data must be given using the <code data-x="attr-script-type">type</code>
  attribute, and the contents of the <code>script</code> element must conform to the requirements
  defined for the format used. The <code data-x="attr-script-src">src</code>, <code
  data-x="attr-script-async">async</code>, <code data-x="attr-script-nomodule">nomodule</code>,
  <code data-x="attr-script-defer">defer</code>, <code
  data-x="attr-script-crossorigin">crossorigin</code>, <code
  data-x="attr-script-integrity">integrity</code>, <code
  data-x="attr-script-referrerpolicy">referrerpolicy</code>, and <code
  data-x="attr-script-fetchpriority">fetchpriority</code> attributes must not be specified.</p>

  <p>The <dfn element-attr for="script"><code data-x="attr-script-nomodule">nomodule</code></dfn>
  attribute is a <span>boolean attribute</span> that prevents a script from being executed in user
  agents that support <span data-x="module script">module scripts</span>. This allows selective
  execution of <span data-x="module script">module scripts</span> in modern user agents and <span
  data-x="classic script">classic scripts</span> in older user agents, <a
  href="#script-nomodule-example">as shown below</a>. The <code
  data-x="attr-script-nomodule">nomodule</code> attribute must not be specified on <span
  data-x="module script">module scripts</span> (and will be ignored if it is).</p>

  <p>The <dfn element-attr for="script"><code data-x="attr-script-async">async</code></dfn> and
  <dfn element-attr for="script"><code data-x="attr-script-defer">defer</code></dfn> attributes are
  <span data-x="boolean attribute">boolean attributes</span> that indicate how the script should be
  evaluated. <span data-x="classic script">Classic scripts</span> may specify <code
  data-x="attr-script-defer">defer</code> or <code data-x="attr-script-async">async</code>, but must
  not specify either unless the <code data-x="attr-script-src">src</code> attribute is present.
  <span data-x="module script">Module scripts</span> may specify the <code
  data-x="attr-script-async">async</code> attribute, but must not specify the <code
  data-x="attr-script-defer">defer</code> attribute.</p>

  <p>There are several possible modes that can be selected using these attributes, and depending on
  the script's type.</p>

  <p>For <span data-x="classic script">classic scripts</span>, if the <code
  data-x="attr-script-async">async</code> attribute is present, then the classic script will be
  fetched <span>in parallel</span> to parsing and evaluated as soon as it is available (potentially
  before parsing completes). If the <code data-x="attr-script-async">async</code> attribute is not
  present but the <code data-x="attr-script-defer">defer</code> attribute is present, then the
  classic script will be fetched <span>in parallel</span> and evaluated when the page has finished
  parsing. If neither attribute is present, then the script is fetched and evaluated immediately,
  blocking parsing until these are both complete.</p>

  <p>For <span data-x="module script">module scripts</span>, if the <code
  data-x="attr-script-async">async</code> attribute is present, then the module script and all its
  dependencies will be fetched <span>in parallel</span> to parsing, and the module script will
  be evaluated as soon as it is available (potentially before parsing completes). Otherwise, the
  module script and its dependencies will be fetched <span>in parallel</span> to parsing and
  evaluated when the page has finished parsing. (The <code data-x="attr-script-defer">defer</code>
  attribute has no effect on module scripts.)</p>

  <p>This is all summarized in the following schematic diagram:</p>

  <p><img src="/images/asyncdefer.svg" style="width: 80%; min-width: 820px;" alt="With &lt;script&gt;, parsing is interrupted by fetching and execution. With &lt;script defer&gt;, fetching is parallel to parsing and execution takes place after all parsing has finished. And with &lt;script async&gt;, fetching is parallel to parsing but once it finishes parsing is interrupted to execute the script. The story for &lt;script type=&quot;module&quot;&gt; is similar to &lt;script defer&gt;, but the dependencies will be fetched as well, and the story for &lt;script type=&quot;module&quot; async&gt; is similar to &lt;script async&gt; with the extra dependency fetching." class="darkmode-aware"></p>

  <p class="note">The exact processing details for these attributes are, for mostly historical
  reasons, somewhat non-trivial, involving a number of aspects of HTML. The implementation
  requirements are therefore by necessity scattered throughout the specification. The algorithms
  below (in this section) describe the core of this processing, but these algorithms reference and
  are referenced by the parsing rules for <code>script</code> <a href="#scriptTag">start</a> and <a
  href="#scriptEndTag">end</a> tags in HTML, <a href="#scriptForeignEndTag">in foreign content</a>,
  and <a href="#scriptTagXML">in XML</a>, the rules for the <code
  data-x="dom-document-write">document.write()</code> method, the handling of <a
  href="#scripting">scripting</a>, etc.</p>

  <p class="note">When inserted using the <code data-x="dom-document-write">document.write()</code>
  method, <code>script</code> elements <a href="#document-written-scripts-intervention">usually</a>
  execute (typically blocking further script execution or HTML parsing). When inserted using the
  <code data-x="dom-element-innerHTML">innerHTML</code> and <code
  data-x="dom-element-outerHTML">outerHTML</code> attributes, they do not execute at all.</p>

  <p>The <code data-x="attr-script-defer">defer</code> attribute may be specified even if the <code
  data-x="attr-script-async">async</code> attribute is specified, to cause legacy web browsers that
  only support <code data-x="attr-script-defer">defer</code> (and not <code
  data-x="attr-script-async">async</code>) to fall back to the <code
  data-x="attr-script-defer">defer</code> behavior instead of the blocking behavior that
  is the default.</p>

  <p>The <dfn element-attr for="script"><code
  data-x="attr-script-crossorigin">crossorigin</code></dfn> attribute is a <span>CORS settings
  attribute</span>. For <span data-x="classic script">classic scripts</span>, it controls whether
  error information will be exposed, when the script is obtained from other <span
  data-x="origin">origins</span>. For <span data-x="module script">module scripts</span>, it
  controls the <span data-x="concept-request-credentials-mode">credentials mode</span> used for
  cross-origin requests.</p>

  <p class="note">Unlike <span data-x="classic script">classic scripts</span>, <span
  data-x="module script">module scripts</span> require the use of the <span
  data-x="CORS protocol">CORS protocol</span> for cross-origin fetching.</p>

  <p>The <dfn for="script" element-attr><code data-x="attr-script-integrity">integrity</code></dfn>
  attribute represents the <span data-x="concept-request-integrity-metadata">integrity
  metadata</span> for requests which this element is responsible for. The value is text. The <code
  data-x="attr-script-integrity">integrity</code> attribute must not be specified when the <code
  data-x="attr-script-src">src</code> attribute is not specified. <ref>SRI</ref></p>

  <p>The <dfn element-attr for="script"><code
  data-x="attr-script-referrerpolicy">referrerpolicy</code></dfn> attribute is a <span>referrer
  policy attribute</span>. Its purpose is to set the <span>referrer policy</span> used when <span
  data-x="concept-fetch">fetching</span> the script, as well as any scripts imported from it.
  <ref>REFERRERPOLICY</ref></p>

  <div class="example">
   <p>An example of a <code>script</code> element's referrer policy being used when fetching
   imported scripts but not other subresources:</p>

   <pre><code class="html">&lt;script referrerpolicy="origin">
  fetch('/api/data');    // not fetched with &lt;script>'s referrer policy
  import('./utils.mjs'); // is fetched with &lt;script>'s referrer policy ("origin" in this case)
&lt;/script></code></pre>
  </div>

  <p>The <dfn element-attr for="script"><code data-x="attr-script-blocking">blocking</code></dfn>
  attribute is a <span>blocking attribute</span>.</p>

  <p>The <dfn element-attr for="script"><code
  data-x="attr-script-fetchpriority">fetchpriority</code></dfn> attribute is a
  <span>fetch priority attribute</span>. Its purpose is to set the <span
  data-x="concept-request-priority">priority</span> used when <span
  data-x="concept-fetch">fetching</span> the script.</p>

  <p>Changing the <code data-x="attr-script-src">src</code>, <code
  data-x="attr-script-type">type</code>, <code data-x="attr-script-nomodule">nomodule</code>, <code
  data-x="attr-script-async">async</code>, <code data-x="attr-script-defer">defer</code>, <code
  data-x="attr-script-crossorigin">crossorigin</code>, <code
  data-x="attr-script-integrity">integrity</code>, <code
  data-x="attr-script-referrerpolicy">referrerpolicy</code>, and <code
  data-x="attr-script-fetchpriority">fetchpriority</code> attributes dynamically has no direct
  effect; these attributes are only used at specific times described below.</p> <!-- by implication,
  changes to the base URL also have no effect -->

  <div w-nodev>

  <p>The IDL attributes <dfn attribute for="HTMLScriptElement"><code
  data-x="dom-script-src">src</code></dfn>, <dfn attribute for="HTMLScriptElement"><code
  data-x="dom-script-type">type</code></dfn>, <dfn attribute for="HTMLScriptElement"><code
  data-x="dom-script-defer">defer</code></dfn>, <dfn attribute for="HTMLScriptElement"><code
  data-x="dom-script-integrity">integrity</code></dfn>, and <dfn attribute
  for="HTMLScriptElement"><code data-x="dom-script-blocking">blocking</code></dfn>, must each
  <span>reflect</span> the respective content attributes of the same name.</p>

  <p>The <dfn attribute for="HTMLScriptElement"><code
  data-x="dom-script-referrerPolicy">referrerPolicy</code></dfn> IDL attribute must
  <span>reflect</span> the <code data-x="attr-script-referrerpolicy">referrerpolicy</code> content
  attribute, <span>limited to only known values</span>.</p>

  <p>The <dfn attribute for="HTMLScriptElement"><code
  data-x="dom-script-fetchPriority">fetchPriority</code></dfn> IDL attribute must
  <span>reflect</span> the <code data-x="attr-script-fetchpriority">fetchpriority</code>
  content attribute, <span>limited to only known values</span>.</p>

  <p>The <dfn attribute for="HTMLScriptElement"><code
  data-x="dom-script-crossOrigin">crossOrigin</code></dfn> IDL attribute must <span>reflect</span>
  the <code data-x="attr-script-crossorigin">crossorigin</code> content attribute, <span>limited to
  only known values</span>.</p>

  <p>The <dfn attribute for="HTMLScriptElement"><code
  data-x="dom-script-noModule">noModule</code></dfn> IDL attribute must <span>reflect</span> the
  <code data-x="attr-script-nomodule">nomodule</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLScriptElement"><code data-x="dom-script-async">async</code></dfn>
  getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="script-force-async">force async</span> is true, then
   return true.</p></li>

   <li><p>If <span>this</span>'s <code data-x="attr-script-async">async</code> content attribute is
   present, then return true.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  <p>The <code data-x="dom-script-async">async</code> setter steps are:</p>

  <ol>
   <li><p>Set <span>this</span>'s <span data-x="script-force-async">force async</span> to
   false.</p></li>

   <li><p>If the given value is true, then set <span>this</span>'s <code
   data-x="attr-script-async">async</code> content attribute to the empty string.</p></li>

   <li><p>Otherwise, remove <span>this</span>'s <code data-x="attr-script-async">async</code>
   content attribute.</p></li>
  </ol>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>script</var>.<span subdfn data-x="dom-script-text">text</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the <span>child text content</span> of the element.</p>

    <p>Can be set, to replace the element's children with the given value.</p>
   </dd>

   <dt><code data-x=""><code>HTMLScriptElement</code>.<span subdfn data-x="dom-script-supports">supports</span>(<var>type</var>)</code></dt>

   <dd>
    <p>Returns true if the given <var>type</var> is a script type supported by the user agent. The
    possible script types in this specification are "<code data-x="">classic</code>", "<code
    data-x="">module</code>", and "<code data-x="">importmap</code>", but others might be added in
    the future.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLScriptElement"><code data-x="dom-script-text">text</code></dfn>
  attribute's getter must return this <code>script</code> element's <span>child text
  content</span>.</p>

  <p>The <code data-x="dom-script-text">text</code> attribute's setter must <span>string replace
  all</span> with the given value within this <code>script</code> element.</p>

  </div>

  <div w-nodev>

  <p>The <dfn method for="HTMLScriptElement"><code
  data-x="dom-script-supports">supports(<var>type</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>If <var>type</var> <span>is</span> "<code data-x="">classic</code>", then return
   true.</p></li>

   <li><p>If <var>type</var> <span>is</span> "<code data-x="">module</code>", then return
   true.</p></li>

   <li><p>If <var>type</var> <span>is</span> "<code data-x="">importmap</code>", then return
   true.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  </div>

  <p class="note">The <var>type</var> argument has to exactly match these values; we do not
  perform an <span>ASCII case-insensitive</span> match. This is different from how <code
  data-x="attr-script-type">type</code> content attribute values are treated, and how
  <code>DOMTokenList</code>'s <code data-x="dom-DOMTokenList-supports">supports()</code> method
  works, but it aligns with the <code>WorkerType</code> enumeration used in the <code
  data-x="dom-Worker">Worker()</code> constructor.</p>

  <div class="example">

   <p>In this example, two <code>script</code> elements are used. One embeds an external
   <span>classic script</span>, and the other includes some data as a <span>data block</span>.</p>

   <pre><code class="html">&lt;script src="game-engine.js">&lt;/script>
&lt;script type="text/x-game-map">
........U.........e
o............A....e
.....A.....AAA....e
.A..AAA...AAAAA...e
&lt;/script></code></pre>

   <p>The data in this case might be used by the script to generate the map of a video game. The
   data doesn't have to be used that way, though; maybe the map data is actually embedded in other
   parts of the page's markup, and the data block here is just used by the site's search engine to
   help users who are looking for particular features in their game maps.</p>

  </div>

  <div class="example">

   <p>The following sample shows how a <code>script</code> element can be used to define a function
   that is then used by other parts of the document, as part of a <span>classic script</span>. It
   also shows how a <code>script</code> element can be used to invoke script while the document is
   being parsed, in this case to initialize the form's output.</p>

   <pre><code class="html">&lt;script>
 function calculate(form) {
   var price = 52000;
   if (form.elements.brakes.checked)
     price += 1000;
   if (form.elements.radio.checked)
     price += 2500;
   if (form.elements.turbo.checked)
     price += 5000;
   if (form.elements.sticker.checked)
     price += 250;
   form.elements.result.value = price;
 }
&lt;/script>
&lt;form name="pricecalc" onsubmit="return false" onchange="calculate(this)">
 &lt;fieldset>
  &lt;legend>Work out the price of your car&lt;/legend>
  &lt;p>Base cost: &pound;52000.&lt;/p>
  &lt;p>Select additional options:&lt;/p>
  &lt;ul>
   &lt;li>&lt;label>&lt;input type=checkbox name=brakes> Ceramic brakes (&pound;1000)&lt;/label>&lt;/li>
   &lt;li>&lt;label>&lt;input type=checkbox name=radio> Satellite radio (&pound;2500)&lt;/label>&lt;/li>
   &lt;li>&lt;label>&lt;input type=checkbox name=turbo> Turbo charger (&pound;5000)&lt;/label>&lt;/li>
   &lt;li>&lt;label>&lt;input type=checkbox name=sticker> "XZ" sticker (&pound;250)&lt;/label>&lt;/li>
  &lt;/ul>
  &lt;p>Total: &pound;&lt;output name=result>&lt;/output>&lt;/p>
 &lt;/fieldset>
 &lt;script>
  calculate(document.forms.pricecalc);
 &lt;/script>
&lt;/form></code></pre>

  </div>

  <div class="example" id="script-type-module-example-1">

   <p>The following sample shows how a <code>script</code> element can be used to include an
   external <span>JavaScript module script</span>.

   <pre><code class="html">&lt;script type="module" src="app.mjs">&lt;/script></code></pre>

   <p>This module, and all its dependencies (expressed through JavaScript <code
   data-x="">import</code> statements in the source file), will be fetched. Once the entire
   resulting module graph has been imported, and the document has finished parsing, the contents of
   <code data-x="">app.mjs</code> will be evaluated.</p>

   <p>Additionally, if code from another <code>script</code> element in the same <code>Window</code>
   imports the module from <code data-x="">app.mjs</code> (e.g. via <code data-x="">import
   "./app.mjs";</code>), then the same <span>JavaScript module script</span> created by the
   former <code>script</code> element will be imported.</p>

  </div>

  <div class="example" id="script-nomodule-example">

  <p>This example shows how to include a <span>JavaScript module script</span> for modern user
  agents, and a <span>classic script</span> for older user agents:</p>

  <pre><code class="html">&lt;script type="module" src="app.mjs">&lt;/script>
&lt;script nomodule defer src="classic-app-bundle.js">&lt;/script></code></pre>

  <p>In modern user agents that support <span data-x="JavaScript module script">JavaScript module
  scripts</span>, the <code>script</code> element with the <code
  data-x="attr-script-nomodule">nomodule</code> attribute will be ignored, and the
  <code>script</code> element with a <code data-x="attr-script-type">type</code> of "<code
  data-x="">module</code>" will be fetched and evaluated (as a <span>JavaScript module
  script</span>). Conversely, older user agents will ignore the <code>script</code> element with a
  <code data-x="attr-script-type">type</code> of "<code data-x="">module</code>", as that is an
  unknown script type for them — but they will have no problem fetching and evaluating the other
  <code>script</code> element (as a <span>classic script</span>), since they do not implement the
  <code data-x="attr-script-nomodule">nomodule</code> attribute.</p>

  </div>

  <div class="example" id="script-type-module-example-2">

   <p>The following sample shows how a <code>script</code> element can be used to write an inline
   <span>JavaScript module script</span> that performs a number of substitutions on the document's
   text, in order to make for a more interesting reading experience (e.g. on a news site):
   <ref>XKCD1288</ref></p>

   <pre><code class="html">&lt;script type="module">
 import { walkAllTextNodeDescendants } from "./dom-utils.mjs";

 const substitutions = new Map([
   ["witnesses", "these dudes I know"]
   ["allegedly", "kinda probably"]
   ["new study", "Tumblr post"]
   ["rebuild", "avenge"]
   ["space", "spaaace"]
   ["Google glass", "Virtual Boy"]
   ["smartphone", "Pokédex"]
   ["electric", "atomic"]
   ["Senator", "Elf-Lord"]
   ["car", "cat"]
   ["election", "eating contest"]
   ["Congressional leaders", "river spirits"]
   ["homeland security", "Homestar Runner"]
   ["could not be reached for comment", "is guilty and everyone knows it"]
 ]);

 function substitute(textNode) {
   for (const [before, after] of substitutions.entries()) {
     textNode.data = textNode.data.replace(new RegExp(`\\b${before}\\b`, "ig"), after);
   }
 }

 walkAllTextNodeDescendants(document.body, substitute);
&lt;/script></code></pre>

   <p>Some notable features gained by using a JavaScript module script include the ability to import
   functions from other JavaScript modules, strict mode by default, and how top-level declarations
   do not introduce new properties onto the <span>global object</span>. Also note that no matter
   where this <code>script</code> element appears in the document, it will not be evaluated until
   both document parsing has complete and its dependency (<code data-x="">dom-utils.mjs</code>) has
   been fetched and evaluated.</p>
  </div>

  <div class="example" id="json-module-script-example">
   <p>The following sample shows how a <span>JSON module script</span> can be imported from inside
   a <span>JavaScript module script</span>:</p>

   <pre><code class="html" data-x="">&lt;script type="module">
 import peopleInSpace from "http://api.open-notify.org/astros.json" with { type: "json" };

 const list = document.querySelector("#people-in-space");
 for (const { craft, name } of peopleInSpace.people) {
   const li = document.createElement("li");
   li.textContent = `${name} / ${craft}`;
   list.append(li);
 }
&lt;/script></code></pre>

   <p>MIME type checking for module scripts is strict. In order for the fetch of the <span>JSON
   module script</span> to succeed, the HTTP response must have a <span>JSON MIME type</span>, for
   example <code data-x="">Content-Type: text/json</code>. On the other hand, if the <code
   data-x="">with { type: "json" }</code> part of the statement is omitted, it is assumed that the
   intent is to import a <span>JavaScript module script</span>, and the fetch will fail if the HTTP
   response has a MIME type that is not a <span>JavaScript MIME type</span>.</p>
  </div>

  <div w-nodev>

  <h5 id="script-processing-model">Processing model</h5>

  <p>A <code>script</code> element has several associated pieces of state.</p>

  <p>A <code>script</code> element has a <dfn>parser document</dfn>, which is either null or a
  <code>Document</code>, initially null. It is set by the <span>HTML parser</span> and the
  <span>XML parser</span> on <code>script</code> elements they insert, and affects the processing
  of those elements. <code>script</code> elements with non-null <span data-x="parser
  document">parser documents</span> are known as <dfn export
  for="script">parser-inserted</dfn>.</p>

  <p>A <code>script</code> element has a <dfn>preparation-time document</dfn>, which is either null
  or a <code>Document</code>, initially null. It is used to prevent scripts that move between
  documents during <span data-x="prepare the script element">preparation</span> from <span
  data-x="execute the script element">executing</span>.</p>

  <p id="non-blocking">A <code>script</code> element has a <dfn data-x="script-force-async">force
  async</dfn> boolean, initially true. It is set to false by the <span>HTML parser</span> and the
  <span>XML parser</span> on <code>script</code> elements they insert, and when the element gets an
  <code data-x="attr-script-async">async</code> content attribute added.</p>

  <p>A <code>script</code> element has a <dfn data-x="concept-script-external">from an external
  file</dfn> boolean, initially false. It is determined when the script is <span data-x="prepare
  the script element">prepared</span>, based on the <code data-x="attr-script-src">src</code>
  attribute of the element at that time.</p>

  <p>A <code>script</code> element has a <dfn>ready to be parser-executed</dfn> boolean, initially
  false. This is used only used for elements that are also <span>parser-inserted</span>, to let the
  parser know when to execute the script.</p>

  <p>A <code>script</code> element has an <dfn>already started</dfn> boolean, initially false.</p>

  <p>A <code>script</code> element has a <dfn data-x="concept-script-delay-load">delaying the
  load event</dfn> boolean, initially false.</p>

  <p>A <code>script</code> element has a <dfn data-x="concept-script-type">type</dfn>, which is
  either null, "<code data-x="">classic</code>", "<code data-x="">module</code>", or "<code
  data-x="">importmap</code>", initially null. It is determined when the element is <span
  data-x="prepare the script element">prepared</span>, based on the <code
  data-x="attr-script-type">type</code> attribute of the element at that time.</p>

  <p id="concept-script-script">A <code>script</code> element has a <dfn
  data-x="concept-script-result">result</dfn>, which is either "<code
  data-x="">uninitialized</code>", null (representing an error), a <span
  data-x="concept-script">script</span>, or an <span>import map parse result</span>. It is
  initially "<code data-x="">uninitialized</code>".</p>

  <p id="the-script-is-ready">A <code>script</code> element has <dfn>steps to run when the result
  is ready</dfn>, which are a series of steps or null, initially null. To <dfn>mark as ready</dfn> a
  <code>script</code> element <var>el</var> given a <span
  data-x="concept-script">script</span>, <span>import map parse result</span>, or null
  <var>result</var>:</p>

  <ol>
   <li><p>Set <var>el</var>'s <span data-x="concept-script-result">result</span> to
   <var>result</var>.</p></li>

   <li><p>If <var>el</var>'s <span>steps to run when the result is ready</span> are not null, then
   run them.</p></li>

   <!-- This is not strictly needed but seems like good hygiene. -->
   <li><p>Set <var>el</var>'s <span>steps to run when the result is ready</span> to null.</p></li>

   <li><p>Set <var>el</var>'s <span data-x="concept-script-delay-load">delaying the load
   event</span> to false.</p></li>
  </ol>

  <hr>

  <p>A <code>script</code> element <var>el</var> is <span>implicitly potentially
  render-blocking</span> if <var>el</var>'s <span data-x="concept-script-type">type</span> is
  "<code data-x="">classic</code>", <var>el</var> is <span>parser-inserted</span>, and
  <var>el</var> does not have an <code data-x="attr-script-async">async</code> or <code
  data-x="attr-script-defer">defer</code> attribute.</p>

  <p>The <span data-x="concept-node-clone-ext">cloning steps</span> for <code>script</code>
  elements given <var>node</var>, <var>copy</var>, and <var>subtree</var> are to set
  <var>copy</var>'s <span>already started</span> to <var>node</var>'s <span>already
  started</span>.</p>

  <p>When an <code data-x="attr-script-async">async</code> attribute is added to a
  <code>script</code> element <var>el</var>, the user agent must set <var>el</var>'s
  <span data-x="script-force-async">force async</span> to false.</p>
  <!-- There's no need to set it to true it when the attribute is removed since you can't have the
  attribute present and force async = true at the same time. -->

  <p>Whenever a <code>script</code> element <var>el</var>'s <span
  data-x="concept-script-delay-load">delaying the load event</span> is true, the user agent must
  <span>delay the load event</span> of <var>el</var>'s <span>preparation-time document</span>.</p>

  <hr>

  <p>The <code>script</code> <span>HTML element post-connection steps</span>, given
  <var>insertedNode</var>, are:</p>

  <ol>
   <li>
    <p>If <var>insertedNode</var> is not <span>connected</span>, then return.</p>

    <div class="example">
     <p>This can happen in the case where an earlier-inserted <code>script</code> removes a
     later-inserted <code>script</code>. For instance:</p>

     <pre><code class="html">&lt;script>
const script1 = document.createElement('script');
script1.innerText = `
  document.querySelector('#script2').remove();
`;

const script2 = document.createElement('script');
script2.id = 'script2';
script2.textContent = `console.log('script#2 running')`;

document.body.append(script1, script2);
&lt;/script></code></pre>

     <p>Nothing is printed to the console in this example. By the time the <span>HTML element
     post-connection steps</span> run for the first <code>script</code> that was atomically inserted
     by <code data-x="dom-Node-append">append()</code>, it can observe that the second
     <code>script</code> is already <span>connected</span> to the DOM. It removes the second
     <code>script</code>, so that by the time <em>its</em> <span>HTML element post-connection
     steps</span> run, it is no longer <span>connected</span>, and does not get <span
     data-x="prepare the script element">prepared</span>.</p>
    </div>
   </li>

   <li><p>If <var>insertedNode</var> is <span>parser-inserted</span>, then return.</p></li>

   <li><p><span>Prepare the script element</span> given <var>insertedNode</var>.</p></li>
  </ol>

  <p>The <code>script</code> <span>children changed steps</span> are:</p>

  <ol>
   <li><p>Run the <code>script</code> <span>HTML element post-connection steps</span>, given the
   <code>script</code> element.</p></li>
  </ol>

  <div class="example">
   <p>This has an interesting implication on the execution order of a <code>script</code> element
   and any newly-inserted child <code>script</code> elements. Consider the following snippet:</p>

   <pre><code class="html">&lt;script id=outer-script>&lt;/script>

&lt;script>
  const outerScript = document.querySelector('#outer-script');

  const start = new Text('console.log(1);');
  const innerScript = document.createElement('script');
  innerScript.textContent = `console.log('inner script executing')`;
  const end = new Text('console.log(2);');

  outerScript.append(start, innerScript, end);

  // Logs:
  // 1
  // 2
  // inner script executing
&lt;/script></code></pre>

   <p>By the time the second script block executes, the <code data-x="">outer-script</code> has
   already been <span data-x="prepare the script element">prepared</span>, but because it is empty,
   it did not execute and therefore is not marked as <span>already started</span>. The atomic
   insertion of the <code>Text</code> nodes and nested <code>script</code> element have the
   following effects:</p>

   <ol>
    <li><p>All three child nodes get atomically inserted as children of <code
    data-x="">outer-script</code>; all of their <span data-x="concept-node-insert-ext">insertion
    steps</span> run, which have no observable consequences in this case.</p></li>

    <li><p>The <code data-x="">outer-script</code>'s <span>children changed steps</span> run, which
    <span data-x="prepare the script element">prepares</span> that script; because its body is now
    non-empty, this executes the contents of the two <code>Text</code> nodes, in order.</p></li>

    <li><p>The <code>script</code> <span>HTML element post-connection steps</span> finally run for
    <code data-x="">innerScript</code>, causing its body to execute.</p></li>
   </ol>
  </div>

  <p>The following <span data-x="concept-element-attributes-change-ext">attribute change
  steps</span>, given <var>element</var>, <var>localName</var>, <var>oldValue</var>,
  <var>value</var>, and <var>namespace</var>, are used for all <code>script</code> elements:</p>

  <ol>
   <li><p>If <var>namespace</var> is not null, then return.</p></li>

   <li><p>If <var>localName</var> is <code data-x="attr-script-src">src</code>, then run the
   <code>script</code> <span>HTML element post-connection steps</span>, given
   <var>element</var>.</p></li>
  </ol>

  <p id="prepare-a-script">To <dfn>prepare the script element</dfn> given a <code>script</code>
  element <var>el</var>:</p>

  <ol>
   <li><p>If <var>el</var>'s <span>already started</span> is true, then return.</p></li>

   <li><p>Let <var>parser document</var> be <var>el</var>'s <span>parser document</span>.</p></li>

   <li>
    <p>Set <var>el</var>'s <span>parser document</span> to null.</p>

    <p class="note">This is done so that if parser-inserted <code>script</code> elements fail to run
    when the parser tries to run them, e.g. because they are empty or specify an unsupported
    scripting language, another script can later mutate them and cause them to run again.</p>
   </li>

   <li>
    <p>If <var>parser document</var> is non-null and <var>el</var> does not have an <code
    data-x="attr-script-async">async</code> attribute, then set <var>el</var>'s <span
    data-x="script-force-async">force async</span> to true.</p>

    <p class="note">This is done so that if a parser-inserted <code>script</code> element fails to
    run when the parser tries to run it, but it is later executed after a script dynamically
    updates it, it will execute in an async fashion even if the <code
    data-x="attr-script-async">async</code> attribute isn't set.</p>
   </li>

   <li><p>Let <var>source text</var> be <var>el</var>'s <span>child text content</span>.</p></li>

   <li id="script-processing-empty"><p>If <var>el</var> has no <code
   data-x="attr-script-src">src</code> attribute, and <var>source text</var> is the empty string,
   then return.</p></li>

   <li><p>If <var>el</var> is not <span>connected</span>, then return.</p></li>

   <li id="script-processing-prepare">
    <p>If any of the following are true:</p>

    <ul>
     <li><p><var>el</var> has a <code data-x="attr-script-type">type</code> attribute whose value is
     the empty string;</p></li>

     <li><p><var>el</var> has no <code data-x="attr-script-type">type</code> attribute but it has a
     <code data-x="attr-script-language">language</code> attribute and <em>that</em> attribute's
     value is the empty string; or</p></li>

     <li><p><var>el</var> has neither a <code data-x="attr-script-type">type</code> attribute nor a
     <code data-x="attr-script-language">language</code> attribute,</p></li>
    </ul>

    <p>then let <var>the script block's type string</var> for this <code>script</code> element be
    "<code data-x="">text/javascript</code>".</p>

    <p>Otherwise, if <var>el</var> has a <code data-x="attr-script-type">type</code> attribute, then
    let <var>the script block's type string</var> be the value of that attribute with <span
    data-x="strip leading and trailing ASCII whitespace">leading and trailing ASCII whitespace
    stripped</span>.</p>

    <p>Otherwise, <var>el</var> has a non-empty <code data-x="attr-script-language">language</code>
    attribute; let <var>the script block's type string</var> be the concatenation of "<code
    data-x="">text/</code>" and the value of <var>el</var>'s <code
    data-x="attr-script-language">language</code> attribute.</p>
    <!-- user agents already support, e.g., type="text/javascript1.3", so we don't have to support
    that separately. -->

    <p class="note">The <code data-x="attr-script-language">language</code> attribute is never
    conforming, and is always ignored if there is a <code data-x="attr-script-type">type</code>
    attribute present.</p>
   </li>

   <li><p>If <var>the script block's type string</var> is a <span>JavaScript MIME type essence
   match</span>, then set <var>el</var>'s <span data-x="concept-script-type">type</span> to "<code
   data-x="">classic</code>".</p></li>

   <li><p>Otherwise, if <var>the script block's type string</var> is an <span>ASCII
   case-insensitive</span> match for the string "<code data-x="">module</code>", then set
   <var>el</var>'s <span data-x="concept-script-type">type</span> to "<code
   data-x="">module</code>".</p></li>

   <li><p>Otherwise, if <var>the script block's type string</var> is an <span>ASCII
   case-insensitive</span> match for the string "<code data-x="">importmap</code>", then set
   <var>el</var>'s <span data-x="concept-script-type">type</span> to "<code
   data-x="">importmap</code>".</p></li>

   <li><p>Otherwise, return. (No script is executed, and <var>el</var>'s <span
   data-x="concept-script-type">type</span> is left as null.)</p></li>

   <li><p>If <var>parser document</var> is non-null, then set <var>el</var>'s <span>parser
   document</span> back to <var>parser document</var> and set <var>el</var>'s <span
   data-x="script-force-async">force async</span> to false.</p></li>

   <li id="script-processing-start"><p>Set <var>el</var>'s <span>already started</span> to
   true.</p></li>

   <li><p>Set <var>el</var>'s <span>preparation-time document</span> to its <span>node
   document</span>.</p>

   <li><p>If <var>parser document</var> is non-null, and <var>parser document</var> is not equal to
   <var>el</var>'s <span>preparation-time document</span>, then return.</p></li>

   <li id="script-processing-noscript"> <p>If <span data-x="concept-n-noscript">scripting is
   disabled</span> for <var>el</var>, then return.</p>

    <p class="note">The definition of <span data-x="concept-n-noscript">scripting is
    disabled</span> means that, amongst others, the following scripts will not execute: scripts in
    <code>XMLHttpRequest</code>'s <code data-x="dom-XMLHttpRequest-responseXML">responseXML</code>
    documents, scripts in <code>DOMParser</code>-created documents, scripts in documents created by
    <code>XSLTProcessor</code>'s <code
    data-x="dom-XSLTProcessor-transformToDocument">transformToDocument</code> feature, and scripts
    that are first inserted by a script into a <code>Document</code> that was created using the
    <code data-x="dom-DOMImplementation-createDocument">createDocument()</code> API. <ref>XHR</ref>
    <ref>DOMPARSING</ref> <ref>XSLTP</ref> <ref>DOM</ref></p>
   </li>

   <li>
    <p>If <var>el</var> has a <code data-x="attr-script-nomodule">nomodule</code> content attribute
    and its <span data-x="concept-script-type">type</span> is "<code data-x="">classic</code>",
    then return.</p>

    <p class="note">This means specifying <code data-x="attr-script-nomodule">nomodule</code> on a
    <span>module script</span> has no effect; the algorithm continues onward.</p>
   </li>

   <li id="script-processing-csp"><p>If <var>el</var> does not have a <code
   data-x="attr-script-src">src</code> content attribute, and the <span>Should element's inline
   behavior be blocked by Content Security Policy?</span> algorithm returns "<code
   data-x="">Blocked</code>" when given <var>el</var>, "<code data-x="">script</code>", and
   <var>source text</var>, then return. <ref>CSP</ref></p></li>

   <li id="script-processing-for">
    <p>If <var>el</var> has an <code data-x="attr-script-event">event</code> attribute and a <code
    data-x="attr-script-for">for</code> attribute, and <var>el</var>'s <span
    data-x="concept-script-type">type</span> is "<code data-x="">classic</code>", then:</p>

    <ol>
     <li><p>Let <var>for</var> be the value of <var>el</var>'s <code
     data-x="attr-script-for">for</code> attribute.</p></li>

     <li><p>Let <var>event</var> be the value of <var>el</var>'s <code
     data-x="attr-script-event">event</code> attribute.</p></li>

     <li><p><span>Strip leading and trailing ASCII whitespace</span> from <var>event</var> and
     <var>for</var>.</p></li>

     <li><p>If <var>for</var> is not an <span>ASCII case-insensitive</span> match for the string
     "<code data-x="">window</code>", then return.</p></li>

     <li><p>If <var>event</var> is not an <span>ASCII case-insensitive</span> match for either the
     string "<code data-x="">onload</code>" or the string "<code data-x="">onload()</code>", then
     return.</p></li>
    </ol>
   </li>

   <li id="script-processing-encoding">
    <p>If <var>el</var> has a <code data-x="attr-script-charset">charset</code> attribute, then let
    <var>encoding</var> be the result of <span>getting an encoding</span> from the value of the
    <code data-x="attr-script-charset">charset</code> attribute.</p>

    <p>If <var>el</var> does not have a <code data-x="attr-script-charset">charset</code>
    attribute, or if <span>getting an encoding</span> failed, then let <var>encoding</var> be
    <var>el</var>'s <span>node document</span>'s <span data-x="document's character encoding">the
    encoding</span>.</p>

    <p class="note">If <var>el</var>'s <span data-x="concept-script-type">type</span> is "<code
    data-x="">module</code>", this encoding will be ignored.</p>
   </li>

   <li><p>Let <var>classic script CORS setting</var> be the current state of <var>el</var>'s <code
   data-x="attr-script-crossorigin">crossorigin</code> content attribute.</p></li>

   <li><p>Let <var>module script credentials mode</var> be the <span>CORS settings attribute
   credentials mode</span> for <var>el</var>'s <code
   data-x="attr-script-crossorigin">crossorigin</code> content attribute.</p>

   <li><p>Let <var>cryptographic nonce</var> be <var>el</var>'s <span>[[CryptographicNonce]]</span>
   internal slot's value.</p></li>

   <li>
    <p>If <var>el</var> has an <code data-x="attr-script-integrity">integrity</code> attribute,
    then let <var>integrity metadata</var> be that attribute's value.</p>

    <p>Otherwise, let <var>integrity metadata</var> be the empty string.</p>
   </li>

   <li><p>Let <var>referrer policy</var> be the current state of <var>el</var>'s <code
   data-x="attr-script-referrerpolicy">referrerpolicy</code> content attribute.</p></li>

   <li><p>Let <var>fetch priority</var> be the current state of <var>el</var>'s <code
   data-x="attr-script-fetchpriority">fetchpriority</code> content attribute.</p></li>

   <li><p>Let <var>parser metadata</var> be "<code data-x="">parser-inserted</code>" if
   <var>el</var> is <span>parser-inserted</span>, and "<code data-x="">not-parser-inserted</code>"
   otherwise.</p></li>

   <li><p>Let <var>options</var> be a <span>script fetch options</span> whose <span
   data-x="concept-script-fetch-options-nonce">cryptographic nonce</span> is <var>cryptographic
   nonce</var>, <span data-x="concept-script-fetch-options-integrity">integrity metadata</span> is
   <var>integrity metadata</var>, <span data-x="concept-script-fetch-options-parser">parser
   metadata</span> is <var>parser metadata</var>, <span
   data-x="concept-script-fetch-options-credentials">credentials mode</span> is <var>module script
   credentials mode</var>, <span data-x="concept-script-fetch-options-referrer-policy">referrer
   policy</span> is <var>referrer policy</var>, and <span
   data-x="concept-script-fetch-options-fetch-priority">fetch priority</span> is <var>fetch
   priority</var>.</p></li>

   <li><p>Let <var>settings object</var> be <var>el</var>'s <span>node document</span>'s
   <span>relevant settings object</span>.</p></li>

   <li id="script-processing-src-prepare">
    <p>If <var>el</var> has a <code data-x="attr-script-src">src</code> content attribute, then:</p>

    <ol>
     <li>
      <p>If <var>el</var>'s <span data-x="concept-script-type">type</span> is "<code
      data-x="">importmap</code>", then <span>queue an element task</span> on the <span>DOM
      manipulation task source</span> given <var>el</var> to <span data-x="concept-event-fire">fire
      an event</span> named <code data-x="event-error">error</code> at <var>el</var>, and
      return.</p>

      <p class="note">External import map scripts are not currently supported. See <a
      href="https://github.com/WICG/import-maps/issues/235">WICG/import-maps issue #235</a> for
      discussions on adding support.</p>
     </li>

     <li><p>Let <var>src</var> be the value of <var>el</var>'s <code
     data-x="attr-script-src">src</code> attribute.</p></li>

     <li><p>If <var>src</var> is the empty string, then <span>queue an element task</span> on the
     <span>DOM manipulation task source</span> given <var>el</var> to <span
     data-x="concept-event-fire">fire an event</span> named <code data-x="event-error">error</code>
     at <var>el</var>, and return.</p></li>

     <li><p>Set <var>el</var>'s <span data-x="concept-script-external">from an external file</span>
     to true.</p></li>

     <li><p>Let <var>url</var> be the result of <span>encoding-parsing a URL</span> given
     <var>src</var>, relative to <var>el</var>'s <span>node document</span>.</p></li>

     <li><p>If <var>url</var> is failure, then <span>queue an element task</span> on the <span>DOM
     manipulation task source</span> given <var>el</var> to <span data-x="concept-event-fire">fire
     an event</span> named <code data-x="event-error">error</code> at <var>el</var>, and
     return.</p></li>

     <li><p>If <var>el</var> is <span>potentially render-blocking</span>, then <span>block
     rendering</span> on <var>el</var>.</p></li>

     <li><p>Set <var>el</var>'s <span data-x="concept-script-delay-load">delaying the load
     event</span> to true.</p></li>

     <li><p>If <var>el</var> is currently <span>render-blocking</span>, then set
     <var>options</var>'s <span data-x="concept-script-fetch-options-render-blocking">
     render-blocking</span> to true.</p></li>

     <li>
      <p>Let <var>onComplete</var> given <var>result</var> be the following steps:</p>

      <ol>
       <li><p><span>Mark as ready</span> <var>el</var> given <var>result</var>.</p></li>
      </ol>
     </li>

     <li>
      <p>Switch on <var>el</var>'s <span data-x="concept-script-type">type</span>:</p>

      <dl class="switch">
       <dt>"<code data-x="">classic</code>"</dt>
       <dd>
        <p><span>Fetch a classic script</span> given <var>url</var>, <var>settings object</var>,
        <var>options</var>, <var>classic script CORS setting</var>, <var>encoding</var>, and
        <var>onComplete</var>.</p>
       </dd>

       <dt>"<code data-x="">module</code>"</dt>
       <dd>
        <p>If <var>el</var> does not have an <code data-x="attr-script-integrity">integrity</code>
        attribute, then set <var>options</var>'s <span
        data-x="concept-script-fetch-options-integrity">integrity metadata</span> to the result of
        <span>resolving a module integrity metadata</span> with <var>url</var> and <var>settings
        object</var>.</p>

        <p><span>Fetch an external module script graph</span> given <var>url</var>, <var>settings
        object</var>, <var>options</var>, and <var>onComplete</var>.</p>
       </dd>
      </dl>

      <p>For performance reasons, user agents may start fetching the classic script or module graph
      (as defined above) as soon as the <code data-x="attr-script-src">src</code> attribute is set,
      instead, in the hope that <var>el</var> will become connected (and that the <code
      data-x="attr-script-crossorigin">crossorigin</code> attribute won't change value in the
      meantime). Either way, once <var>el</var> <span>becomes connected</span>, the load must have
      started as described in this step. If the UA performs such prefetching, but <var>el</var>
      never becomes connected, or the <code data-x="attr-script-src">src</code> attribute is
      dynamically changed, <!-- or the base URL is dynamically changed,--> or the <code
      data-x="attr-script-crossorigin">crossorigin</code> attribute is dynamically changed, then the
      user agent will not execute the script so obtained, and the fetching process will have been
      effectively wasted.</p>
     </li>
    </ol>
   </li>

   <li id="establish-script-block-source">
    <p>If <var>el</var> does not have a <code data-x="attr-script-src">src</code> content
    attribute:</p>

    <ol>
     <li><p>Let <var>base URL</var> be <var>el</var>'s <span>node document</span>'s <span>document
     base URL</span>.</p></li>

     <li>
      <p>Switch on <var>el</var>'s <span data-x="concept-script-type">type</span>:</p>

      <dl class="switch">
       <dt>"<code data-x="">classic</code>"</dt>
       <dd>
        <ol>
         <li><p>Let <var>script</var> be the result of <span>creating a classic script</span> using
         <var>source text</var>, <var>settings object</var>, <var>base URL</var>, and
         <var>options</var>.</p></li>

         <li><p><span>Mark as ready</span> <var>el</var> given <var>script</var>.</p></li>
        </ol>
       </dd>

       <dt>"<code data-x="">module</code>"</dt>
       <dd>
        <ol>
         <li><p>Set <var>el</var>'s <span data-x="concept-script-delay-load">delaying the load
         event</span> to true.</p></li>

         <li>
          <p>If <var>el</var> is <span>potentially render-blocking</span>, then:</p>

          <ol>
           <li><p><span>Block rendering</span> on <var>el</var>.</p></li>

           <li><p>Set <var>options</var>'s <span
           data-x="concept-script-fetch-options-render-blocking">render-blocking</span> to
           true.</p></li>
          </ol>
         <li>
          <p><span>Fetch an inline module script graph</span>, given <var>source text</var>,
          <var>base URL</var>, <var>settings object</var>, <var>options</var>, and with the
          following steps given <var>result</var>:</p>

          <ol>
           <li>
            <p><span>Queue an element task</span> on the <span>networking task source</span>
            given <var>el</var> to perform the following steps:</p>

            <ol>
             <li><p><span>Mark as ready</span> <var>el</var> given <var>result</var>.</p></li>
            </ol>

            <p class="note">Queueing a task here means that, even if the inline module script has
            no dependencies or synchronously results in a parse error, we won't proceed to
            <span>execute the script element</span> synchronously.</p>
           </li>
          </ol>
         </li>
        </ol>
       </dd>

       <dt>"<code data-x="">importmap</code>"</dt>
       <dd>
        <ol>
         <li><p>Let <var>result</var> be the result of <span data-x="create an import map parse
         result">creating an import map parse result</span> given <var>source text</var> and
         <var>base URL</var>.</p></li>

         <li><p><span>Mark as ready</span> <var>el</var> given <var>result</var>.</p></li>
        </ol>
       </dd>
      </dl>
     </li>
    </ol>
   </li>

   <li>
    <p>If <var>el</var>'s <span data-x="concept-script-type">type</span> is "<code
    data-x="">classic</code>" and <var>el</var> has a <code data-x="attr-script-src">src</code>
    attribute, or <var>el</var>'s <span data-x="concept-script-type">type</span> is "<code
    data-x="">module</code>":</p>

    <ol>
     <li><p><span>Assert</span>: <var>el</var>'s <span data-x="concept-script-result">result</span>
     is "<code data-x="">uninitialized</code>".</p></li>

     <li id="script-processing-src">
      <p id="script-processing-module-async">If <var>el</var> has an <code
      data-x="attr-script-async">async</code> attribute or <var>el</var>'s <span
      data-x="script-force-async">force async</span> is true:</p>

      <ol>
       <li><p>Let <var>scripts</var> be <var>el</var>'s <span>preparation-time document</span>'s
       <span>set of scripts that will execute as soon as possible</span>.</p></li>

       <li><p><span data-x="set append">Append</span> <var>el</var> to <var>scripts</var>.</p></li>

       <li>
        <p>Set <var>el</var>'s <span>steps to run when the result is ready</span> to the
        following:</p>

        <ol>
         <li><p><span>Execute the script element</span> <var>el</var>.</p></li>

         <li><p><span data-x="list remove">Remove</span> <var>el</var> from
         <var>scripts</var>.</p></li>
        </ol>
       </li>
      </ol>
     </li>

     <li id="script-processing-src-sync">
      <p id="script-processing-module-noasync">Otherwise, if <var>el</var> is not
      <span>parser-inserted</span>:</p>

      <ol>
       <li><p>Let <var>scripts</var> be <var>el</var>'s <span>preparation-time document</span>'s
       <span>list of scripts that will execute in order as soon as possible</span>.</p></li>

       <li><p><span data-x="list append">Append</span> <var>el</var> to <var>scripts</var>.</p></li>

       <li>
        <p>Set <var>el</var>'s <span>steps to run when the result is ready</span> to the
        following:</p>

        <ol>
         <li><p>If <var>scripts</var>[0] is not <var>el</var>, then abort these steps.</p></li>

         <li>
          <p>While <var>scripts</var> is not empty, and <var>scripts</var>[0]'s <span
          data-x="concept-script-result">result</span> is not "<code
          data-x="">uninitialized</code>":</p>

          <ol>
           <li><p><span>Execute the script element</span> <var>scripts</var>[0].</p></li>

           <li><p><span data-x="list remove">Remove</span> <var>scripts</var>[0].</p></li>
          </ol>
         </li>
        </ol>
       </li>
      </ol>
     </li>

     <li id="script-processing-defer">
      <p id="script-processing-module-noasync-parser-inserted">Otherwise, if <var>el</var> has a
      <code data-x="attr-script-defer">defer</code> attribute or <var>el</var>'s <span
      data-x="concept-script-type">type</span> is "<code data-x="">module</code>":</p>


      <ol>
       <li><p><span data-x="list append">Append</span> <var>el</var> to its <span>parser
       document</span>'s <span>list of scripts that will execute when the document has finished
       parsing</span>.</p></li>

       <li><p>Set <var>el</var>'s <span>steps to run when the result is ready</span> to the
       following: set <var>el</var>'s <span>ready to be parser-executed</span> to true. (The parser
       will handle executing the script.)</p></li>
      </ol>
     </li>

     <li id="script-processing-parser-inserted">
      <p>Otherwise:</p>

      <ol>
       <li><p>Set <var>el</var>'s <span>parser document</span>'s <span>pending parsing-blocking
       script</span> to <var>el</var>.</p></li>

       <li><p><span>Block rendering</span> on <var>el</var>.</p></li>

       <li><p>Set <var>el</var>'s <span>steps to run when the result is ready</span> to the
       following: set <var>el</var>'s <span>ready to be parser-executed</span> to true. (The parser
       will handle executing the script.)</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p><span>Assert</span>: <var>el</var>'s <span data-x="concept-script-result">result</span>
     is <em>not</em> "<code data-x="">uninitialized</code>".</p></li>

     <li id="script-processing-style-delayed">
      <p>If all of the following are true:</p>

      <ul>
       <li><var>el</var>'s <span data-x="concept-script-type">type</span> is "<code
       data-x="">classic</code>";</li>

       <li><var>el</var> is <span>parser-inserted</span>;</li>

       <li><var>el</var>'s <span>parser document</span> <span>has a style sheet that is blocking
       scripts</span>; and</li>

       <li>either the parser that created <var>el</var> is an <span>XML parser</span>, or it's an
       <span>HTML parser</span> whose <span>script nesting level</span> is not greater than
       one,</li>
      </ul>

      <p>then:</p>

      <ol>
       <li><p>Set <var>el</var>'s <span>parser document</span>'s <span>pending parsing-blocking
       script</span> to <var>el</var>.</p></li>

       <li><p>Set <var>el</var>'s <span>ready to be parser-executed</span> to true. (The parser
       will handle executing the script.)</p></li>
      </ol>
     </li>

     <li id="script-processing-inline"><p>Otherwise, <span>immediately</span> <span>execute the
     script element</span> <var>el</var>, even if other scripts are already executing.</p></li>
    </ol>
   </li>
  </ol>

  <p>Each <code>Document</code> has a <dfn>pending parsing-blocking script</dfn>, which is a
  <code>script</code> element or null, initially null.</p>

  <p>Each <code>Document</code> has a <dfn>set of scripts that will execute as soon as
  possible</dfn>, which is a <span>set</span> of <code>script</code> elements, initially empty.</p>

  <p>Each <code>Document</code> has a <dfn>list of scripts that will execute in order as soon as
  possible</dfn>, which is a <span>list</span> of <code>script</code> elements, initially
  empty.</p>

  <p>Each <code>Document</code> has a <dfn>list of scripts that will execute when the document has
  finished parsing</dfn>, which is a <span>list</span> of <code>script</code> elements, initially
  empty.</p>

  <p class="note">If a <code>script</code> element that blocks a parser gets moved to another
  <code>Document</code> before it would normally have stopped blocking that parser, it nonetheless
  continues blocking that parser until the condition that causes it to be blocking the parser no
  longer applies (e.g., if the script is a <span>pending parsing-blocking script</span> because the
  original <code>Document</code> <span>has a style sheet that is blocking scripts</span> when it was
  parsed, but then the script is moved to another <code>Document</code> before the blocking style
  sheet(s) loaded, the script still blocks the parser until the style sheets are all loaded, at
  which time the script executes and the parser is unblocked).</p>

  <!-- also (and this would be worth testing): the way the spec is written, if you do not have a
  style sheet that is blocking scripts, then you parse a <script src>, then while waiting for the
  script to load you insert an external style sheet, the script will delay until the sheet is
  loaded, because there's just a binary "are style sheets blocking scripts" state, things aren't
  defined in terms of which style sheets are blocking which scripts -->

  <p id="execute-the-script-block">To <dfn>execute the script element</dfn> given a
  <code>script</code> element <var>el</var>:</p>

  <ol>
   <li><p>Let <var>document</var> be <var>el</var>'s <span>node document</span>.</p></li>

   <li><p>If <var>el</var>'s <span>preparation-time document</span> is not equal to
   <var>document</var>, then return.</p></li>

   <li><p><span>Unblock rendering</span> on <var>el</var>.</p></li>

   <li><p>If <var>el</var>'s <span data-x="concept-script-result">result</span> is null, then <span
   data-x="concept-event-fire">fire an event</span> named <code data-x="event-error">error</code>
   at <var>el</var>, and return.</p></li>

   <li><p>If <var>el</var>'s <span data-x="concept-script-external">from an external file</span> is
   true, or <var>el</var>'s <span data-x="concept-script-type">type</span> is "<code
   data-x="">module</code>", then increment <var>document</var>'s <span>ignore-destructive-writes
   counter</span>.</p></li>

   <li>
    <p>Switch on <var>el</var>'s <span data-x="concept-script-type">type</span>:</p>

    <dl class="switch">
     <dt>"<code data-x="">classic</code>"</dt>
     <dd>
      <ol>
       <li><p>Let <var>oldCurrentScript</var> be the value to which <var>document</var>'s <code
       data-x="dom-document-currentScript">currentScript</code> object was most recently
       set.</p></li>

       <li>
        <p>If <var>el</var>'s <span>root</span> is <em>not</em> a <span>shadow root</span>, then
        set <var>document</var>'s <code data-x="dom-document-currentScript">currentScript</code>
        attribute to <var>el</var>. Otherwise, set it to null.</p>

        <p class="note">This does not use the <span>in a document tree</span> check, as
        <var>el</var> could have been removed from the document prior to execution, and in that
        scenario <code data-x="dom-document-currentScript">currentScript</code> still needs to
        point to it.</p>
       </li>

       <li><p><span data-x="run a classic script">Run the classic script</span> given by
       <var>el</var>'s <span data-x="concept-script-result">result</span>.</p></li>

       <li><p>Set <var>document</var>'s <code
       data-x="dom-document-currentScript">currentScript</code> attribute to
       <var>oldCurrentScript</var>.</p></li>
      </ol>
     </dd>

     <dt>"<code data-x="">module</code>"</dt>
     <dd>
      <ol>
       <li><p><span>Assert</span>: <var>document</var>'s <code
       data-x="dom-document-currentScript">currentScript</code> attribute is null.</p></li>

       <li><p><span data-x="run a module script">Run the module script</span> given by
       <var>el</var>'s <span data-x="concept-script-result">result</span>.</p></li>
      </ol>
     </dd>

     <dt>"<code data-x="">importmap</code>"</dt>
     <dd>
      <ol>
       <li><p><span>Register an import map</span> given <var>el</var>'s <span>relevant global
       object</span> and <var>el</var>'s <span
       data-x="concept-script-result">result</span>.</p></li>
      </ol>
     </dd>
    </dl>
   </li>

   <li><p>Decrement the <span>ignore-destructive-writes counter</span> of <var>document</var>, if
   it was incremented in the earlier step.</p></li>

   <li><p>If <var>el</var>'s <span data-x="concept-script-external">from an external file</span> is
   true, then <span data-x="concept-event-fire">fire an event</span> named <code
   data-x="event-load">load</code> at <var>el</var>.</p></li>
  </ol>

  </div>


  <h5 id="scriptingLanguages">Scripting languages</h5>

  <p w-nodev>User agents are not required to support JavaScript. This standard needs to be updated
  if a language other than JavaScript comes along and gets similar wide adoption by web browsers.
  Until such a time, implementing other languages is in conflict with this standard, given the
  processing model defined for the <code>script</code> element.</p>

  <p>Servers should use <code>text/javascript</code> for JavaScript resources, in accordance with
  <cite>Updates to ECMAScript Media Types</cite>. Servers should not use other
  <span data-x="JavaScript MIME type">JavaScript MIME types</span> for JavaScript resources, and
  must not use non-<span data-x="JavaScript MIME type">JavaScript MIME types</span>.
  <ref>RFC9239</ref></p>

  <div w-nodev>

  <p>For external JavaScript resources, MIME type parameters in `<code>Content-Type</code>` headers
  are generally ignored. (In some cases the `<code data-x="">charset</code>` parameter has an
  effect.) However, for the <code>script</code> element's <code
  data-x="attr-script-type">type</code> attribute they are significant; it uses the <span>JavaScript
  MIME type essence match</span> concept.</p>

  <p class="note">For example, scripts with their <code data-x="attr-script-type">type</code>
  attribute set to "<code data-x="">text/javascript; charset=utf-8</code>" will not be
  evaluated, even though that is a valid <span>JavaScript MIME type</span> when parsed.</p>

  <p>Furthermore, again for external JavaScript resources, special considerations apply around
  `<code>Content-Type</code>` header processing as detailed in the <span>prepare the script
  element</span> algorithm and <cite>Fetch</cite>. <ref>FETCH</ref>

  </div>


  <h5><dfn data-x="script content restrictions">Restrictions for contents of <code>script</code> elements</dfn></h5>

  <p class="note">The easiest and safest way to avoid the rather strange restrictions described in
  this section is to always escape an ASCII case-insensitive match for "<code
  data-x="">&lt;!--</code>" as "<code data-x="">\x3C!--</code>", "<code
  data-x="">&lt;script</code>" as "<code data-x="">\x3Cscript</code>", and "<code
  data-x="">&lt;/script</code>" as "<code data-x="">\x3C/script</code>" when these sequences appear
  in literals in scripts (e.g. in strings, regular expressions, or comments), and to avoid writing
  code that uses such constructs in expressions. Doing so avoids the pitfalls that the restrictions
  in this section are prone to triggering: namely, that, for historical reasons, parsing of
  <code>script</code> blocks in HTML is a strange and exotic practice that acts unintuitively in the
  face of these sequences.</p>

  <p>The <code>script</code> element's <span>descendant text content</span> must match the <code
  data-x="">script</code> production in the following ABNF, the character set for which is Unicode.
  <ref>ABNF</ref></p>

  <pre><code data-x="" class="abnf">script        = outer *( comment-open inner comment-close outer )

outer         = &lt; any string that doesn't contain a substring that matches not-in-outer >
not-in-outer  = comment-open
inner         = &lt; any string that doesn't contain a substring that matches not-in-inner >
not-in-inner  = comment-close / script-open

comment-open  = "&lt;!--"
comment-close = "-->"
script-open   = "&lt;" s c r i p t tag-end

s             =  %x0053 ; U+0053 LATIN CAPITAL LETTER S
s             =/ %x0073 ; U+0073 LATIN SMALL LETTER S
c             =  %x0043 ; U+0043 LATIN CAPITAL LETTER C
c             =/ %x0063 ; U+0063 LATIN SMALL LETTER C
r             =  %x0052 ; U+0052 LATIN CAPITAL LETTER R
r             =/ %x0072 ; U+0072 LATIN SMALL LETTER R
i             =  %x0049 ; U+0049 LATIN CAPITAL LETTER I
i             =/ %x0069 ; U+0069 LATIN SMALL LETTER I
p             =  %x0050 ; U+0050 LATIN CAPITAL LETTER P
p             =/ %x0070 ; U+0070 LATIN SMALL LETTER P
t             =  %x0054 ; U+0054 LATIN CAPITAL LETTER T
t             =/ %x0074 ; U+0074 LATIN SMALL LETTER T

tag-end       =  %x0009 ; U+0009 CHARACTER TABULATION (tab)
tag-end       =/ %x000A ; U+000A LINE FEED (LF)
tag-end       =/ %x000C ; U+000C FORM FEED (FF)
tag-end       =/ %x0020 ; U+0020 SPACE
tag-end       =/ %x002F ; U+002F SOLIDUS (/)
tag-end       =/ %x003E ; U+003E GREATER-THAN SIGN (&gt;)</code></pre>

  <p>When a <code>script</code> element contains <span>script documentation</span>, there are
  further restrictions on the contents of the element, as described in the section below.</p>

  <div class="example">

   <p>The following script illustrates this issue. Suppose you have a script that contains a string,
   as in:</p>

   <pre><code class="js">const example = 'Consider this string: &lt;!-- &lt;script>';
console.log(example);</code></pre>

   <p>If one were to put this string directly in a <code>script</code> block, it would violate the
   restrictions above:</p>

   <pre><code class="html">&lt;script>
  const example = 'Consider this string: &lt;!-- &lt;script>';
  console.log(example);
&lt;/script></code></pre>

   <p>The bigger problem, though, and the reason why it would violate those restrictions, is that
   actually the script would get parsed weirdly: <em>the script block above is not terminated</em>.
   That is, what looks like a "<code data-x="">&lt;/script></code>" end tag in this snippet is
   actually still part of the <code>script</code> block. The script doesn't execute (since it's not
   terminated); if it somehow were to execute, as it might if the markup looked as follows, it would
   fail because the script (highlighted here) is not valid JavaScript:</p>

   <pre><code class="html">&lt;script><mark>
  const example = 'Consider this string: &lt;!-- &lt;script>';
  console.log(example);
&lt;/script>
&lt;!-- despite appearances, this is actually part of the script still! -->
&lt;script>
 ... // this is the same script block still...
</mark>&lt;/script></code></pre>

   <p>What is going on here is that for legacy reasons, "<code data-x="">&lt;!--</code>" and "<code
   data-x="">&lt;script</code>" strings in <code>script</code> elements in HTML need to be balanced
   in order for the parser to consider closing the block.</p>

   <p>By escaping the problematic strings as mentioned at the top of this section, the problem is
   avoided entirely:</p>

   <pre><code class="html">&lt;script><mark>
  // Note: `\x3C` is an escape sequence for `&lt;`.
  const example = 'Consider this string: \x3C!-- \x3Cscript>';
  console.log(example);
</mark>&lt;/script>
&lt;!-- this is just a comment between script blocks -->
&lt;script><mark>
 ... // this is a new script block
</mark>&lt;/script></code></pre>

   <p>It is possible for these sequences to naturally occur in script expressions, as in the
   following examples:</p>

   <pre><code class="js">if (x&lt;!--y) { ... }
if ( player&lt;script ) { ... }</code></pre>

   <p>In such cases the characters cannot be escaped, but the expressions can be rewritten so that
   the sequences don't occur, as in:</p>

   <pre><code class="js">if (x &lt; !--y) { ... }
if (!--y > x) { ... }
if (!(--y) > x) { ... }
if (player &lt; script) { ... }
if (script > player) { ... }</code></pre>

   <p>Doing this also avoids a different pitfall as well: for related historical reasons, the string
   "&lt;!--" in <span data-x="classic script">classic scripts</span> is actually treated as a line
   comment start, just like "//".</p>

  </div>


  <h5><dfn data-x="script documentation">Inline documentation for external scripts</dfn></h5>

  <p>If a <code>script</code> element's <code data-x="attr-script-src">src</code> attribute is
  specified, then the contents of the <code>script</code> element, if any, must be such that the
  value of the <code data-x="dom-script-text">text</code> IDL attribute, which is derived from the
  element's contents, matches the <code data-x="">documentation</code> production in the following
  ABNF, the character set for which is Unicode. <ref>ABNF</ref></p>

  <pre><code data-x="" class="abnf">documentation = *( *( space / tab / comment ) [ line-comment ] newline )
comment       = slash star *( not-star / star not-slash ) 1*star slash
line-comment  = slash slash *not-newline

; characters
tab           = %x0009 ; U+0009 CHARACTER TABULATION (tab)
newline       = %x000A ; U+000A LINE FEED (LF)
space         = %x0020 ; U+0020 SPACE
star          = %x002A ; U+002A ASTERISK (*)
slash         = %x002F ; U+002F SOLIDUS (/)
not-newline   = %x0000-0009 / %x000B-10FFFF
                ; a <span>scalar value</span> other than U+000A LINE FEED (LF)
not-star      = %x0000-0029 / %x002B-10FFFF
                ; a <span>scalar value</span> other than U+002A ASTERISK (*)
not-slash     = %x0000-002E / %x0030-10FFFF
                ; a <span>scalar value</span> other than U+002F SOLIDUS (/)</code></pre>

  <p class="note">This corresponds to putting the contents of the element in JavaScript
  comments.</p>

  <p class="note">This requirement is in addition to the earlier restrictions on the syntax of
  contents of <code>script</code> elements.</p>

  <div class="example">

   <p>This allows authors to include documentation, such as license information or API information,
   inside their documents while still referring to external script files. The syntax is constrained
   so that authors don't accidentally include what looks like valid script while also providing a
   <code data-x="attr-script-src">src</code> attribute.</p>

   <pre><code class="html">&lt;script src="cool-effects.js">
 // create new instances using:
 //    var e = new Effect();
 // start the effect using .play, stop using .stop:
 //    e.play();
 //    e.stop();
&lt;/script></code></pre>

  </div>


  <div w-nodev>

  <h5 id="scriptTagXSLT">Interaction of <code>script</code> elements and XSLT</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>This specification does not define how XSLT interacts with the <code>script</code> element.
  However, in the absence of another specification actually defining this, here are some guidelines
  for implementers, based on existing implementations:</p>

  <ul>
   <li><p>When an XSLT transformation program is triggered by an <code
   data-x="">&lt;?xml-stylesheet?></code> processing instruction and the browser implements a
   direct-to-DOM transformation, <code>script</code> elements created by the XSLT processor need to
   have its <span>parser document</span> set correctly, and run in document order (modulo scripts
   marked <code data-x="attr-script-defer">defer</code> or <code
   data-x="attr-script-async">async</code>), <span>immediately</span>, as the transformation is
   occurring.</p></li>

   <li><p>The <code>XSLTProcessor</code> <code
   data-x="dom-XSLTProcessor-transformToDocument">transformToDocument()</code> method adds elements
   to a <code>Document</code> object with a null <span data-x="concept-document-bc">browsing
   context</span>, and, accordingly, any <code>script</code> elements they create need to have
   their <span>already started</span> set to true in the <span>prepare the script element</span>
   algorithm and never get executed (<span data-x="concept-environment-noscript">scripting is
   disabled</span>). Such <code>script</code> elements still need to have their <span>parser
   document</span> set, though, such that their <code data-x="dom-script-async">async</code> IDL
   attribute will return false in the absence of an <code data-x="attr-script-async">async</code>
   content attribute.</p></li>

   <li><p>The <code>XSLTProcessor</code> <code
   data-x="dom-XSLTProcessor-transformToFragment">transformToFragment()</code> method needs to
   create a fragment that is equivalent to one built manually by creating the elements using <code
   data-x="dom-document-createElementNS">document.createElementNS()</code>. For instance, it needs
   to create <code>script</code> elements with null <span>parser document</span> and with their
   <span>already started</span> set to false, so that they will execute when the fragment is
   inserted into a document.</p></li>
  </ul>

  <p>The main distinction between the first two cases and the last case is that the first two
  operate on <code>Document</code>s and the last operates on a fragment.</p>

  </div>



  <h4>The <dfn element><code>noscript</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Metadata content</span>.</dd>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>In a <code>head</code> element of an <span data-x="HTML documents">HTML document</span>, if there are no ancestor <code>noscript</code> elements.</dd>
   <dd>Where <span>phrasing content</span> is expected in <span>HTML documents</span>, if there are no ancestor <code>noscript</code> elements.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd>When <span data-x="concept-n-noscript">scripting is disabled</span>, in a <code>head</code> element: in any order, zero or more <code>link</code> elements, zero or more <code>style</code> elements, and zero or more <code>meta</code> elements.</dd>
   <dd>When <span data-x="concept-n-noscript">scripting is disabled</span>, not in a <code>head</code> element: <span>transparent</span>, but there must be no <code>noscript</code> element descendants.</dd>
   <dd>Otherwise: text that conforms to the requirements given in the prose.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-noscript">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-noscript">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Uses <code>HTMLElement</code>.</dd>
  </dl>

  <p>The <code>noscript</code> element <span>represents</span> nothing if <span
  data-x="concept-n-script">scripting is enabled</span>, and <span>represents</span> its children if
  <span data-x="concept-n-noscript">scripting is disabled</span>. It is used to present different
  markup to user agents that support scripting and those that don't support scripting, by affecting
  how the document is parsed.</p>

  <p>When used in <span>HTML documents</span>, the allowed content model is as follows:</p>

  <dl>
   <dt>In a <code>head</code> element, if <span data-x="concept-n-noscript">scripting is
   disabled</span> for the <code>noscript</code> element</dt>

   <dd><p>The <code>noscript</code> element must contain only <code>link</code>, <code>style</code>,
   and <code>meta</code> elements.</p></dd>

   <dt>In a <code>head</code> element, if <span data-x="concept-n-script">scripting is enabled</span>
   for the <code>noscript</code> element</dt>

   <dd><p>The <code>noscript</code> element must contain only text, except that invoking the
   <span>HTML fragment parsing algorithm</span> <!-- (which disables <script> execution) --> with
   the <code>noscript</code> element as the <var data-x="concept-frag-parse-context">context</var>
   element and the text contents as the <var>input</var> must result in a list of nodes
   that consists only of <code>link</code>, <code>style</code>, and <code>meta</code> elements that
   would be conforming if they were children of the <code>noscript</code> element, and no <span
   data-x="parse error">parse errors</span>.</p></dd>

   <dt>Outside of <code>head</code> elements, if <span data-x="concept-n-noscript">scripting is
   disabled</span> for the <code>noscript</code> element</dt>

   <dd><p>The <code>noscript</code> element's content model is <span>transparent</span>, with the
   additional restriction that a <code>noscript</code> element must not have a <code>noscript</code>
   element as an ancestor (that is, <code>noscript</code> can't be nested).</p></dd>

   <dt>Outside of <code>head</code> elements, if <span data-x="concept-n-script">scripting is
   enabled</span> for the <code>noscript</code> element</dt>

   <dd>
    <p>The <code>noscript</code> element must contain only text, except that the text must be such
    that running the following algorithm results in a conforming document with no
    <code>noscript</code> elements and no <code>script</code> elements, and such that no step in the
    algorithm throws an exception or causes an <span>HTML parser</span> to flag a <span>parse
    error</span>:</p>
    <!-- the possible exception is from the outerHTML setter -->

    <ol>
     <li>Remove every <code>script</code> element from the document.</li>

     <li>Make a list of every <code>noscript</code> element in the document. For every
     <code>noscript</code> element in that list, perform the following steps:

      <ol>
       <li>Let <var>s</var> be the <span>child text content</span> of the <code>noscript</code>
       element.</li>

       <li>Set the <code data-x="dom-element-outerHTML">outerHTML</code> attribute of the
       <code>noscript</code> element to the value of <var>s</var>. (This, as a side-effect, causes
       the <code>noscript</code> element to be removed from the document.)</li>
      </ol>
     </li>
    </ol>
   </dd>
  </dl>

  <p class="note">All these contortions are required because, for historical reasons, the
  <code>noscript</code> element is handled differently by the <span>HTML parser</span> based on
  whether <span data-x="scripting flag">scripting was enabled or not</span> when the parser was
  invoked.</p>

  <p>The <code>noscript</code> element must not be used in <span>XML documents</span>.</p>

  <p class="note">The <code>noscript</code> element is only effective in <span>the HTML
  syntax</span>, it has no effect in <span>the XML syntax</span>. This is because the way it works
  is by essentially "turning off" the parser when scripts are enabled, so that the contents of the
  element are treated as pure text and not as real elements. XML does not define a mechanism by
  which to do this.</p>

  <div w-nodev>

  <p>The <code>noscript</code> element has no other requirements. In particular, children of the
  <code>noscript</code> element are not exempt from <span>form submission</span>, scripting, and so
  forth, even when <span data-x="concept-n-script">scripting is enabled</span> for the element.</p>

  </div>

  <div class="example">

   <p>In the following example, a <code>noscript</code> element is
   used to provide fallback for a script.</p>

   <pre><code class="html">&lt;form action="calcSquare.php">
 &lt;p>
  &lt;label for=x>Number&lt;/label>:
  &lt;input id="x" name="x" type="number">
 &lt;/p>
 &lt;script>
  var x = document.getElementById('x');
  var output = document.createElement('p');
  output.textContent = 'Type a number; it will be squared right then!';
  x.form.appendChild(output);
  x.form.onsubmit = function () { return false; }
  x.oninput = function () {
    var v = x.valueAsNumber;
    output.textContent = v + ' squared is ' + v * v;
  };
 &lt;/script>
 &lt;noscript>
  &lt;input type=submit value="Calculate Square">
 &lt;/noscript>
&lt;/form></code></pre>

   <p>When script is disabled, a button appears to do the calculation on the server side. When
   script is enabled, the value is computed on-the-fly instead.</p>

   <p>The <code>noscript</code> element is a blunt instrument. Sometimes, scripts might be enabled,
   but for some reason the page's script might fail. For this reason, it's generally better to avoid
   using <code>noscript</code>, and to instead design the script to change the page from being a
   scriptless page to a scripted page on the fly, as in the next example:</p>

   <pre><code class="html">&lt;form action="calcSquare.php">
 &lt;p>
  &lt;label for=x>Number&lt;/label>:
  &lt;input id="x" name="x" type="number">
 &lt;/p>
 <strong>&lt;input id="submit" type=submit value="Calculate Square"></strong>
 &lt;script>
  var x = document.getElementById('x');
  var output = document.createElement('p');
  output.textContent = 'Type a number; it will be squared right then!';
  x.form.appendChild(output);
  x.form.onsubmit = function () { return false; }
  x.oninput = function () {
    var v = x.valueAsNumber;
    output.textContent = v + ' squared is ' + v * v;
  };
<strong>  var submit = document.getElementById('submit');
  submit.parentNode.removeChild(submit);</strong>
 &lt;/script>
&lt;/form></code></pre>

   <p>The above technique is also useful in <span>XML documents</span>, since <code>noscript</code>
   is not allowed there.</p>

  </div>


  <h4 id="the-template-element">The <dfn element><code>template</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Metadata content</span>.</dd>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span data-x="script-supporting elements">Script-supporting element</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>metadata content</span> is expected.</dd>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dd>Where <span>script-supporting elements</span> are expected.</dd>
   <dd>As a child of a <code>colgroup</code> element that doesn't have a <code data-x="attr-colgroup-span">span</code> attribute.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span data-x="concept-content-nothing">Nothing</span> (for clarification, <a href="#template-example">see example</a>).</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-template-shadowrootmode">shadowrootmode</code></dd>
   <dd><code data-x="attr-template-shadowrootdelegatesfocus">shadowrootdelegatesfocus</code></dd>
   <dd><code data-x="attr-template-shadowrootclonable">shadowrootclonable</code></dd>
   <dd><code data-x="attr-template-shadowrootserializable">shadowrootserializable</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-template">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-template">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLTemplateElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  readonly attribute <span>DocumentFragment</span> <span data-x="dom-template-content">content</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-template-shadowrootmode">shadowRootMode</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-template-shadowrootdelegatesfocus">shadowRootDelegatesFocus</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-template-shadowrootclonable">shadowRootClonable</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-template-shadowRootSerializable">shadowRootSerializable</span>;
};</code></pre>
</dd>
   <dd w-dev>Uses <code>HTMLTemplateElement</code>.</dd>
  </dl>

  <p>The <code>template</code> element is used to declare fragments of HTML that can be cloned and
  inserted in the document by script.</p>

  <p>In a rendering, the <code>template</code> element <span>represents</span> nothing.</p>

  <p>The <dfn element-attr for="template"><code
  data-x="attr-template-shadowrootmode">shadowrootmode</code></dfn> content attribute is an
  <span>enumerated attribute</span> with the following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="template/shadowrootmode"><code data-x="attr-shadowrootmode-open">open</code></dfn>
     <td><dfn data-x="attr-shadowrootmode-open-state">open</dfn>
     <td>The template element represents an open declarative shadow root.
    <tr>
     <td><dfn attr-value for="template/shadowrootmode"><code data-x="attr-shadowrootmode-closed">closed</code></dfn>
     <td><dfn data-x="attr-shadowrootmode-closed-state">closed</dfn>
     <td>The template element represents a closed declarative shadow root.
  </table>

  <p>The <code data-x="attr-template-shadowrootmode">shadowrootmode</code> attribute's <i
  data-x="invalid value default">invalid value default</i> and <i data-x="missing value
  default">missing value default</i> are both the <dfn
  data-x="attr-shadowrootmode-none-state">none</dfn> state.</p>

  <p>The <dfn element-attr for="template"><code
  data-x="attr-template-shadowrootdelegatesfocus">shadowrootdelegatesfocus</code></dfn> content
  attribute is a <span>boolean attribute</span>.</p>

  <p>The <dfn element-attr for="template"><code
  data-x="attr-template-shadowrootclonable">shadowrootclonable</code></dfn> content attribute is a
  <span>boolean attribute</span>.</p>

  <p>The <dfn element-attr for="template"><code
  data-x="attr-template-shadowrootserializable">shadowrootserializable</code></dfn> content
  attribute is a <span>boolean attribute</span>.</p>

  <p>The <span>template contents</span> of a <code>template</code> element <a
  href="#template-syntax">are not children of the element itself</a>.</p>

  <p class="note">It is also possible, as a result of DOM manipulation, for a <code>template</code>
  element to contain <code>Text</code> nodes and element nodes; however, having any is a violation
  of the <code>template</code> element's content model, since its content model is defined as <span
  data-x="concept-content-nothing">nothing</span>.</p>

  <div class="example" id="template-example">

   <p>For example, consider the following document:</p>

   <!-- https://www.quora.com/Homework-Question-How-do-I-write-a-program-that-produces-the-following-output-1 -->

   <pre><code class="html">&lt;!doctype html>
&lt;html lang="en">
 &lt;head>
  &lt;title>Homework&lt;/title>
 &lt;body>
  &lt;template id="template">&lt;p>Smile!&lt;/p>&lt;/template>
  &lt;script>
   let num = 3;
   const fragment = document.getElementById('template').content.cloneNode(true);
   while (num-- > 1) {
     fragment.firstChild.before(fragment.firstChild.cloneNode(true));
     fragment.firstChild.textContent += fragment.lastChild.textContent;
   }
   document.body.appendChild(fragment);
  &lt;/script>
&lt;/html></code></pre>

   <p>The <code>p</code> element in the <code>template</code> is <em>not</em> a child of the
   <code>template</code> in the DOM; it is a child of the <code>DocumentFragment</code> returned by
   the <code>template</code> element's <code data-x="dom-template-content">content</code> IDL
   attribute.</p>

   <p>If the script were to call <code data-x="dom-node-appendchild">appendChild()</code> on the
   <code>template</code> element, that would add a child to the <code>template</code> element (as
   for any other element); however, doing so is a violation of the <code>template</code> element's
   content model.</p>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>template</var>.<span subdfn data-x="dom-template-content">content</span></code></dt>

   <dd><p>Returns the <span>template contents</span> (a <code>DocumentFragment</code>).</p></dd>
  </dl>

  <div w-nodev>

  <p>Each <code>template</code> element has an associated <code>DocumentFragment</code> object that
  is its <dfn>template contents</dfn>. The <span>template contents</span> have <a
  href="#no-browsing-context">no conformance requirements</a>. When a <code>template</code> element
  is created, the user agent must run the following steps to establish the <span>template
  contents</span>:</p>

  <ol>
   <li><p>Let <var>doc</var> be the <code>template</code> element's <span>node document</span>'s <span>appropriate template contents owner
   document</span>.</p></li>

   <li><p>Create a <code>DocumentFragment</code> object whose <span>node document</span> is
   <var>doc</var> and <span data-x="concept-DocumentFragment-host">host</span> is the
   <code>template</code> element.</p></li>

   <li><p>Set the <code>template</code> element's <span>template contents</span> to the newly
   created <code>DocumentFragment</code> object.</p></li>
  </ol>

  <p>A <code>Document</code> <var>doc</var>'s <dfn>appropriate template contents owner
  document</dfn> is the <code>Document</code> returned by the following algorithm:</p>

  <ol>
   <li>
    <p>If <var>doc</var> is not a <code>Document</code> created by this algorithm, then:</p>

    <ol>
     <li>
      <p>If <var>doc</var> does not yet have an <dfn>associated inert template document</dfn>,
      then:</p>

      <ol>
       <li><p>Let <var>new doc</var> be a new <code>Document</code> (whose <span
       data-x="concept-document-bc">browsing context</span> is null). This is "a
       <code>Document</code> created by this algorithm" for the purposes of the step above.</p></li>

       <li><p>If <var>doc</var> is an <span data-x="HTML documents">HTML document</span>, mark
       <var>new doc</var> as an <span data-x="HTML documents">HTML document</span>
       also.</p></li>

       <li><p>Let <var>doc</var>'s <span>associated inert template document</span> be <var>new doc</var>.</p></li>
      </ol>
     </li>

     <li><p>Set <var>doc</var> to <var>doc</var>'s <span>associated inert
     template document</span>.</p></li>
    </ol>

    <p class="note">Each <code>Document</code> not created by this algorithm thus gets a single
    <code>Document</code> to act as its proxy for owning the <span>template contents</span> of all
    its <code>template</code> elements, so that they aren't in a <span>browsing context</span> and
    thus remain inert (e.g. scripts do not run). Meanwhile, <code>template</code> elements inside
    <code>Document</code> objects that <em>are</em> created by this algorithm just reuse the same
    <code>Document</code> owner for their contents.</p>
   </li>

   <li><p>Return <var>doc</var>.</p></li>
  </ol>

  <p id="template-adopting-steps">The <span data-x="concept-node-adopt-ext">adopting steps</span>
  (with <var>node</var> and <var>oldDocument</var> as parameters) for <code>template</code> elements
  are the following:</p>

  <ol>
   <li>
    <p>Let <var>doc</var> be <var>node</var>'s <span>node document</span>'s
    <span>appropriate template contents owner document</span>.</p>

    <p class="note"><var>node</var>'s <span>node document</span> is the <code>Document</code> object
    that <var>node</var> was just adopted <em>into</em>.</p>
   </li>

   <li><p><span data-x="concept-node-adopt">Adopt</span> <var>node</var>'s
   <span>template contents</span> (a <code>DocumentFragment</code> object) into <var>doc</var>.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLTemplateElement"><code
  data-x="dom-template-content">content</code></dfn> getter steps are to return
  <code>template</code>'s <span>template contents</span>, if the <span>template contents</span> is
  not a <code>ShadowRoot</code> node; otherwise null.</p>

  <p>The <dfn attribute for="HTMLTemplateElement"><code
  data-x="dom-template-shadowrootmode">shadowRootMode</code></dfn> IDL attribute must
  <span>reflect</span> the <code data-x="attr-template-shadowrootmode">shadowrootmode</code> content
  attribute, <span>limited to only known values</span>.</p>

  <p>The <dfn attribute for="HTMLTemplateElement"><code
  data-x="dom-template-shadowrootdelegatesfocus">shadowRootDelegatesFocus</code></dfn> IDL attribute
  must <span>reflect</span> the <code
  data-x="attr-template-shadowrootdelegatesfocus">shadowrootdelegatesfocus</code> content
  attribute.</p>

  <p>The <dfn attribute for="HTMLTemplateElement"><code
  data-x="dom-template-shadowrootclonable">shadowRootClonable</code></dfn> IDL attribute must
  <span>reflect</span> the <code data-x="attr-template-shadowrootclonable">shadowrootclonable</code>
  content attribute.</p>

  <p>The <dfn attribute for="HTMLTemplateElement"><code
  data-x="dom-template-shadowRootSerializable">shadowRootSerializable</code></dfn> IDL attribute
  must <span>reflect</span> the <code
  data-x="attr-template-shadowrootserializable">shadowrootserializable</code> content attribute.</p>

  <hr>

  <p>The <span data-x="concept-node-clone-ext">cloning steps</span> for <code>template</code>
  elements given <var>node</var>, <var>copy</var>, and <var>subtree</var> are:</p>

  <ol>
   <li><p>If <var>subtree</var> is false, then return.</p></li>

   <li><p>For each <var>child</var> of <var>node</var>'s <span>template contents</span>'s <span
   data-x="concept-tree-child">children</span>, in <span>tree order</span>: <span
   data-x="concept-node-clone">clone a node</span> given <var>child</var> with <span
   data-x="concept-node-clone-document"><i>document</i></span> set to <var>copy</var>'s
   <span>template contents</span>'s <span>node document</span>, <span
   data-x="concept-node-clone-subtree"><i>subtree</i></span> set to true, and <span
   data-x="concept-node-clone-parent"><i>parent</i></span> set to <var>copy</var>'s <span>template
   contents</span>.</p></li>
  </ol>

  </div>

  <div class="example">

   <p>In this example, a script populates a table four-column with data from a data structure, using
   a <code>template</code> to provide the element structure instead of manually generating the
   structure from markup.</p>

   <pre><code class="html">&lt;!DOCTYPE html>
&lt;html lang='en'>
&lt;title>Cat data&lt;/title>
&lt;script>
 // Data is hard-coded here, but could come from the server
 var data = [
   { name: 'Pillar', color: 'Ticked Tabby', sex: 'Female (neutered)', legs: 3 },
   { name: 'Hedral', color: 'Tuxedo', sex: 'Male (neutered)', legs: 4 },
 ];
&lt;/script>
&lt;table>
 &lt;thead>
  &lt;tr>
   &lt;th>Name &lt;th>Color &lt;th>Sex &lt;th>Legs
 &lt;tbody>
  &lt;template id="row">
   &lt;tr>&lt;td>&lt;td>&lt;td>&lt;td>
  &lt;/template>
&lt;/table>
&lt;script>
 var template = document.querySelector('#row');
 for (var i = 0; i &lt; data.length; i += 1) {
   var cat = data[i];
   var clone = template.content.cloneNode(true);
   var cells = clone.querySelectorAll('td');
   cells[0].textContent = cat.name;
   cells[1].textContent = cat.color;
   cells[2].textContent = cat.sex;
   cells[3].textContent = cat.legs;
   template.parentNode.appendChild(clone);
 }
&lt;/script></code></pre>

   <p>This example uses <code data-x="dom-node-cloneNode">cloneNode()</code> on the
   <code>template</code>'s contents; it could equivalently have used <code
   data-x="dom-Document-importNode">document.importNode()</code>, which does the same thing. The
   only difference between these two APIs is when the <span>node document</span> is updated: with
   <code data-x="dom-node-cloneNode">cloneNode()</code> it is updated when the nodes are appended
   with <code data-x="dom-node-appendChild">appendChild()</code>, with <code
   data-x="dom-Document-importNode">document.importNode()</code> it is updated when the nodes are
   cloned.</p>

  </div>



  <div w-nodev>

  <h5 id="template-XSLT-XPath">Interaction of <code>template</code> elements with XSLT and XPath</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>This specification does not define how XSLT and XPath interact with the <code>template</code>
  element. However, in the absence of another specification actually defining this, here are some
  guidelines for implementers, which are intended to be consistent with other processing described
  in this specification:</p>

  <ul>
   <li><p>An XSLT processor based on an XML parser that acts <span data-x="xml parser">as described
   in this specification</span> needs to act as if <code>template</code> elements contain as
   descendants their <span>template contents</span> for the purposes of the transform.</p></li>

   <li><p>An XSLT processor that outputs a DOM needs to ensure that nodes that would go into a
   <code>template</code> element are instead placed into the element's <span>template
   contents</span>.</p></li>

   <li><p>XPath evaluation using the XPath DOM API when applied to a <code>Document</code> parsed
   using the <span>HTML parser</span> or the <span>XML parser</span> described in this specification
   needs to ignore <span>template contents</span>.</p>
  </ul>

  </div>



  <h4 id="the-slot-element">The <dfn element><code>slot</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Transparent</span></dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-slot-name">name</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-slot">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-slot">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLSlotElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-slot-name">name</span>;
  sequence&lt;Node> <span data-x="dom-slot-assignedNodes">assignedNodes</span>(optional <span>AssignedNodesOptions</span> options = {});
  sequence&lt;Element> <span data-x="dom-slot-assignedElements">assignedElements</span>(optional <span>AssignedNodesOptions</span> options = {});
  undefined <span data-x="dom-slot-assign">assign</span>((<span>Element</span> or <span>Text</span>)... nodes);
};

dictionary <dfn dictionary>AssignedNodesOptions</dfn> {
  boolean <dfn dict-member for="AssignedNodeOptions" data-x="dom-AssignedNodesOptions-flatten">flatten</dfn> = false;
};</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLSlotElement</code>.</dd>
  </dl>

  <p>The <code>slot</code> element defines a <span data-x="concept-slot">slot</span>. It is
  typically used in a <span>shadow tree</span>. A <code>slot</code> element <span>represents</span>
  its <span>assigned nodes</span>, if any, and its contents otherwise.</p>

  <p>The <dfn element-attr for="slot" data-x="attr-slot-name"><code>name</code></dfn> content
  attribute may contain any string value. It represents a <span data-x="concept-slot">slot</span>'s
  <span data-x="slot-name">name</span>.</p>

  <p class="note">The <code data-x="attr-slot-name">name</code> attribute is used to <span
  data-x="assign a slot">assign slots</span> to other elements: a <code>slot</code> element with a
  <code data-x="attr-slot-name">name</code> attribute creates a named <span
  data-x="concept-slot">slot</span> to which any element is <span data-x="assign a
  slot">assigned</span> if that element has a <code data-x="attr-slot">slot</code> attribute whose
  value matches that <code data-x="attr-slot-name">name</code> attribute's value, and the
  <code>slot</code> element is a child of the <span>shadow tree</span> whose <span>root</span>'s
  <span data-x="concept-DocumentFragment-host">host</span> has that corresponding <code
  data-x="attr-slot">slot</code> attribute value.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>slot</var>.<span subdfn data-x="dom-slot-name">name</span></code></dt>
   <dd>Can be used to get and set <var>slot</var>'s <span data-x="slot-name">name</span>.</dd>

   <dt><code data-x=""><var>slot</var>.<span subdfn data-x="dom-slot-assignedNodes">assignedNodes</span>()</code></dt>
   <dd>Returns <var>slot</var>'s <span>assigned nodes</span>.</dd>

   <dt><code data-x=""><var>slot</var>.<span data-x="dom-slot-assignedNodes">assignedNodes</span>({ flatten: true })</code></dt>
   <dd>Returns <var>slot</var>'s <span>assigned nodes</span>, if any, and <var>slot</var>'s children
   otherwise, and does the same for any <code>slot</code> elements encountered therein, recursively,
   until there are no <code>slot</code> elements left.</dd>

   <dt><code data-x=""><var>slot</var>.<span subdfn data-x="dom-slot-assignedElements">assignedElements</span>()</code></dt>
   <dd>Returns <var>slot</var>'s <span>assigned nodes</span>, limited to elements.</dd>

   <dt><code data-x=""><var>slot</var>.<span data-x="dom-slot-assignedElements">assignedElements</span>({ flatten: true })</code></dt>
   <dd>Returns the same as <code data-x="dom-slot-assignedNodes">assignedNodes({ flatten: true
   })</code>, limited to elements.</dd>

   <dt><code data-x=""><var>slot</var>.<span data-x="dom-slot-assign">assign</span>(...<var>nodes</var>)</code></dt>
   <dd><p>Sets <var>slot</var>'s <span>manually assigned nodes</span> to the given
   <var>nodes</var>.</p></dd>
  </dl>

  <p>The <dfn attribute for="HTMLSlotElement" data-x="dom-slot-name"><code>name</code></dfn> IDL
  attribute must <span>reflect</span> the content attribute of the same name.</p>

  <p>The <code>slot</code> element has <dfn export for="HTMLSlotElement">manually assigned
  nodes</dfn>, which is an <span data-x="set">ordered set</span> of <span
  data-x="slottable">slottables</span> set by <code data-x="dom-slot-assign">assign()</code>.
  This set is initially empty.</p>

  <p class="note">The <span>manually assigned nodes</span> set can be implemented using weak
  references to the <span data-x="slottable">slottables</span>, because this set is not
  directly accessible from script.</p>

  <p>The <dfn method for="HTMLSlotElement"
  data-x="dom-slot-assignedNodes"><code>assignedNodes(<var>options</var>)</code></dfn> method steps
  are:</p>

  <ol>
   <li><p>If <var>options</var>["<code data-x="dom-AssignedNodesOptions-flatten">flatten</code>"] is
   false, then return <span>this</span>'s <span>assigned nodes</span>.</p></li>

   <li><p>Return the result of <span>finding flattened slottables</span> with
   <span>this</span>.</p></li>
  </ol>

  <p>The <dfn method for="HTMLSlotElement"
  data-x="dom-slot-assignedElements"><code>assignedElements(<var>options</var>)</code></dfn> method
  steps are:</p>

  <ol>
   <li><p>If <var>options</var>["<code data-x="dom-AssignedNodesOptions-flatten">flatten</code>"] is
   false, then return <span>this</span>'s <span>assigned nodes</span>, filtered to contain only
   <code>Element</code> nodes.</p></li>

   <li><p>Return the result of <span>finding flattened slottables</span> with <span>this</span>,
   filtered to contain only <code>Element</code> nodes.</p></li>
  </ol>

  <p>The <dfn method for="HTMLSlotElement"
  data-x="dom-slot-assign"><code>assign(...<var>nodes</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p><span data-x="list iterate">For each</span> <var>node</var> of <span>this</span>'s
   <span>manually assigned nodes</span>, set <var>node</var>'s <span>manual slot assignment</span>
   to null.</p></li>

   <li><p>Let <var>nodesSet</var> be a new <span data-x="set">ordered set</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>node</var> of <var>nodes</var>:</p>

    <ol>
     <li><p>If <var>node</var>'s <span>manual slot assignment</span> refers to a <span>slot</span>,
     then remove <var>node</var> from that <span>slot</span>'s
     <span>manually assigned nodes</span>.</p></li>

     <li><p>Set <var>node</var>'s <span>manual slot assignment</span> to <span>this</span>.</p></li>

     <li><p><span data-x="set append">Append</span> <var>node</var> to <var>nodesSet</var>.</p></li>
    </ol>
   </li>

   <li><p>Set <span>this</span>'s <span>manually assigned nodes</span> to
   <var>nodesSet</var>.</p></li>

   <li><p>Run <span>assign slottables for a tree</span> for <span>this</span>'s
   <span>root</span>.</p></li>
  </ol>


  <h4 split-filename="canvas">The <dfn id="canvas" element><code>canvas</code></dfn> element</h4>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Embedded content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>embedded content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Transparent</span>, but with no <span>interactive content</span> descendants except
   for <code>a</code> elements, <code>img</code> elements with
   <code data-x="attr-hyperlink-usemap">usemap</code> attributes, <code>button</code> elements,
   <code>input</code> elements whose <code data-x="attr-input-type">type</code> attribute are in
   the <span data-x="attr-input-type-checkbox">Checkbox</span> or <span
   data-x="attr-input-type-radio">Radio Button</span> states, <code>input</code> elements that are
   <span data-x="concept-button">buttons</span>, and <code>select</code> elements with a <code
   data-x="attr-select-multiple">multiple</code> attribute or a <span
   data-x="concept-select-size">display size</span> greater than 1.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span></dd>
   <dd><code data-x="attr-canvas-width">width</code></dd>
   <dd><code data-x="attr-canvas-height">height</code></dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd><a href="https://w3c.github.io/html-aria/#el-canvas">For authors</a>.</dd>
   <dd><a href="https://w3c.github.io/html-aam/#el-canvas">For implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd w-nodev>
    <pre><code class="idl">typedef (<span>CanvasRenderingContext2D</span> or <span>ImageBitmapRenderingContext</span> or <span>WebGLRenderingContext</span> or <span>WebGL2RenderingContext</span> or <span>GPUCanvasContext</span>) <dfn typedef>RenderingContext</dfn>;

[Exposed=Window]
interface <dfn interface>HTMLCanvasElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-canvas-width">width</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-canvas-height">height</span>;

  <span>RenderingContext</span>? <span data-x="dom-canvas-getContext">getContext</span>(DOMString contextId, optional any options = null);

  USVString <span data-x="dom-canvas-toDataURL">toDataURL</span>(optional DOMString type = "image/png", optional any quality);
  undefined <span data-x="dom-canvas-toBlob">toBlob</span>(<span>BlobCallback</span> _callback, optional DOMString type = "image/png", optional any quality);
  <span>OffscreenCanvas</span> <span data-x="dom-canvas-transferControlToOffscreen">transferControlToOffscreen</span>();
};

callback <dfn callback>BlobCallback</dfn> = undefined (<span>Blob</span>? blob);</code></pre>
   </dd>
   <dd w-dev>Uses <code>HTMLCanvasElement</code>.</dd>
  </dl>

  <p>The <code>canvas</code> element provides scripts with a resolution-dependent bitmap canvas,
  which can be used for rendering graphs, game graphics, art, or other visual images on the fly.</p>

  <p>Authors should not use the <code>canvas</code> element in a document when a more suitable
  element is available. For example, it is inappropriate to use a <code>canvas</code> element to
  render a page heading: if the desired presentation of the heading is graphically intense, it
  should be marked up using appropriate elements (typically <code>h1</code>) and then styled using
  CSS and supporting technologies such as <span data-x="shadow tree">shadow trees</span>.</p>

  <p>When authors use the <code>canvas</code> element, they must also provide content that, when
  presented to the user, conveys essentially the same function or purpose as the
  <code>canvas</code>'s bitmap. This content may be placed as content of the <code>canvas</code>
  element. The contents of the <code>canvas</code> element, if any, are the element's <span>fallback
  content</span>.</p>

  <hr>

  <p>In interactive visual media, if <span data-x="concept-n-script">scripting is enabled</span> for
  the <code>canvas</code> element, and if support for <code>canvas</code> elements has been enabled,
  then the <code>canvas</code> element <span>represents</span> <span>embedded content</span>
  consisting of a dynamically created image, the element's bitmap.</p>

  <p>In non-interactive, static, visual media, if the <code>canvas</code> element has been
  previously associated with a rendering context (e.g. if the page was viewed in an interactive
  visual medium and is now being printed, or if some script that ran during the page layout process
  painted on the element), then the <code>canvas</code> element <span>represents</span>
  <span>embedded content</span> with the element's current bitmap and size. Otherwise, the element
  represents its <span>fallback content</span> instead.</p>

  <p>In non-visual media, and in visual media if <span data-x="concept-n-noscript">scripting is
  disabled</span> for the <code>canvas</code> element or if support for <code>canvas</code> elements
  has been disabled, the <code>canvas</code> element <span>represents</span> its <span>fallback
  content</span> instead.</p>

  <!-- CANVAS-FOCUS-FALLBACK -->
  <p>When a <code>canvas</code> element <span>represents</span> <span>embedded content</span>, the
  user can still focus descendants of the <code>canvas</code> element (in the <span>fallback
  content</span>). When an element is <span>focused</span>, it is the target of keyboard interaction
  events (even though the element itself is not visible). This allows authors to make an interactive
  canvas keyboard-accessible: authors should have a one-to-one mapping of interactive regions to <i
  data-x="focusable area">focusable areas</i> in the <span>fallback content</span>. (Focus has no
  effect on mouse interaction events.) <ref>UIEVENTS</ref></p> <!-- user
  interaction spec integration point -->

  <p>An element whose nearest <code>canvas</code> element ancestor is <span>being rendered</span>
  and <span>represents</span> <span>embedded content</span> is an element that is <dfn>being used as
  relevant canvas fallback content</dfn>.</p>

  <hr>

  <p>The <code>canvas</code> element has two attributes to control the size of the element's bitmap:
  <dfn element-attr for="canvas"><code data-x="attr-canvas-width">width</code></dfn> and <dfn
  element-attr for="canvas"><code data-x="attr-canvas-height">height</code></dfn>. These attributes,
  when specified, must have values that are <span data-x="valid non-negative integer">valid
  non-negative integers</span>.  <span w-nodev>The <span>rules for parsing non-negative
  integers</span> must be used to <dfn data-x="obtain-numeric-values">obtain their numeric
  values</dfn>. If an attribute is missing, or if parsing its value returns an error, then the
  default value must be used instead.</span> The <code data-x="attr-canvas-width">width</code>
  attribute defaults to 300, and the <code data-x="attr-canvas-height">height</code> attribute
  defaults to 150.</p>

  <p>When setting the value of the <code data-x="attr-canvas-width">width</code> or <code
  data-x="attr-canvas-height">height</code> attribute, if the <span
  data-x="concept-canvas-context-mode">context mode</span> of the <code>canvas</code>
  element is set to <span data-x="concept-canvas-placeholder">placeholder</span>, the
  user agent must throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>
  and leave the attribute's value unchanged.</p>

  <p>The <span>natural dimensions</span> of the <code>canvas</code> element when it
  <span>represents</span> <span>embedded content</span> are equal to the dimensions of the
  element's bitmap.</p>

  <p>The user agent must use a square pixel density consisting of one pixel of image data per
  coordinate space unit for the bitmaps of a <code>canvas</code> and its rendering contexts.</p>

  <p class="note">A <code>canvas</code> element can be sized arbitrarily by a style sheet, its
  bitmap is then subject to the <span>'object-fit'</span> CSS property.</p>

  <div w-nodev>

  <hr>

  <p>The bitmaps of <code>canvas</code> elements, the bitmaps of <code>ImageBitmap</code> objects,
  as well as some of the bitmaps of rendering contexts, such as those described in the sections on
  the <code>CanvasRenderingContext2D</code>, <code>OffscreenCanvasRenderingContext2D</code>, and
  <code>ImageBitmapRenderingContext</code> objects below, have an <dfn
  data-x="concept-canvas-origin-clean">origin-clean</dfn> flag, which can be set to true or false.
  Initially, when the <code>canvas</code> element or <code>ImageBitmap</code> object is created, its
  bitmap's <span data-x="concept-canvas-origin-clean">origin-clean</span> flag must be set to
  true.</p>

  <p>A <code>canvas</code> element can have a rendering context bound to it. Initially, it does not
  have a bound rendering context. To keep track of whether it has a rendering context or not, and
  what kind of rendering context it is, a <code>canvas</code> also has a <dfn
  data-x="concept-canvas-context-mode">canvas context mode</dfn>, which is initially <dfn
  data-x="concept-canvas-none">none</dfn> but can be changed to either <dfn
  data-x="concept-canvas-placeholder">placeholder</dfn>, <dfn
  data-x="concept-canvas-2d">2d</dfn>, <dfn
  data-x="concept-canvas-bitmaprenderer">bitmaprenderer</dfn>, <dfn
  data-x="concept-canvas-webgl">webgl</dfn>, <dfn
  data-x="concept-canvas-webgl2">webgl2</dfn>, or <dfn
  data-x="concept-canvas-webgpu">webgpu</dfn> by algorithms defined in this specification.</p>

  <p>When its <span data-x="concept-canvas-context-mode">canvas context mode</span> is <span
  data-x="concept-canvas-none">none</span>, a <code>canvas</code> element has no rendering context,
  and its bitmap must be <span>transparent black</span> with a <span>natural width</span> equal
  to <span data-x="obtain-numeric-values">the numeric value</span> of the element's <code
  data-x="attr-canvas-width">width</code> attribute and a <span>natural height</span> equal to
  <span data-x="obtain-numeric-values">the numeric value</span> of the element's <code
  data-x="attr-canvas-height">height</code> attribute, those values being interpreted in <span
  data-x="'px'">CSS pixels</span>, and being updated as the attributes are set, changed, or
  removed.</p>

  <p>When its <span data-x="concept-canvas-context-mode">canvas context mode</span> is <span
  data-x="concept-canvas-placeholder">placeholder</span>, a <code>canvas</code> element has no
  rendering context. It serves as a placeholder for an <code>OffscreenCanvas</code> object, and
  the content of the <code>canvas</code> element is updated by the <code>OffscreenCanvas</code>
  object's rendering context.</p>

  <p>When a <code>canvas</code> element represents <span>embedded content</span>, it provides a
  <span>paint source</span> whose width is the element's <span>natural width</span>, whose height
  is the element's <span>natural height</span>, and whose appearance is the element's bitmap.</p>

  <p>Whenever the <code data-x="attr-canvas-width">width</code> and <code
  data-x="attr-canvas-height">height</code> content attributes are set, removed, changed, or
  redundantly set to the value they already have, then the user agent must perform the action
  from the row of the following table that corresponds to the <code>canvas</code> element's <span
  data-x="concept-canvas-context-mode">context mode</span>.</p>

  <table>
   <thead>
    <tr>
     <th>
      <p><span data-x="concept-canvas-context-mode">Context Mode</span></p>
     <th>
      <p>Action</p>

   <tbody>
    <tr>
     <th>
      <p><span data-x="concept-canvas-2d">2d</span></p>
     <td>
      <p>Follow the steps to <span data-x="concept-canvas-set-bitmap-dimensions">set bitmap
      dimensions</span> to <span data-x="obtain-numeric-values">the numeric values</span>
      of the <code data-x="attr-canvas-width">width</code> and <code
      data-x="attr-canvas-height">height</code> content attributes.</p>

    <tr>
     <th>
      <p><span data-x="concept-canvas-webgl">webgl</span> or <span
      data-x="concept-canvas-webgl2">webgl2</span></p>
     <td>
      <p>Follow the behavior defined in the WebGL specifications. <ref>WEBGL</ref></p>

    <tr>
     <th>
      <p><span data-x="concept-canvas-webgpu">webgpu</span></p>
     <td>
      <p>Follow the behavior defined in <cite>WebGPU</cite>. <ref>WEBGPU</ref></p>

    <tr>
     <th>
      <p><span data-x="concept-canvas-bitmaprenderer">bitmaprenderer</span></p>
     <td>
      <p>If the context's <span data-x="concept-ImageBitmapRenderingContext-bitmap-mode">bitmap
      mode</span> is set to <span data-x="concept-ImageBitmapRenderingContext-blank">blank</span>,
      run the steps to <span>set an <code>ImageBitmapRenderingContext</code>'s output bitmap</span>,
      passing the <code>canvas</code> element's rendering context.</p>

    <tr>
     <th>
      <p><span data-x="concept-canvas-placeholder">placeholder</span></p>
     <td>
      <p>Do nothing.</p>

    <tr>
     <th>
      <p><span data-x="concept-canvas-none">none</span></p>
     <td>
      <p>Do nothing.</p>
  </table>

  <p>The <dfn attribute for="HTMLCanvasElement"><code data-x="dom-canvas-width">width</code></dfn>
  and <dfn attribute for="HTMLCanvasElement"><code data-x="dom-canvas-height">height</code></dfn>
  IDL attributes must <span>reflect</span> the respective content attributes of the same name, with
  the same defaults.</p>

  </div>

  <hr>

  <dl class="domintro">
   <dt><code data-x=""><var>context</var> = <var>canvas</var>.<span subdfn data-x="dom-canvas-getContext">getContext</span>(<var>contextId</var> [, <var>options</var> ])</code></dt>
   <dd>
    <p>Returns an object that exposes an API for drawing on the canvas. <var>contextId</var>
    specifies the desired API: "<code data-x="canvas-context-2d">2d</code>", "<code
    data-x="canvas-context-bitmaprenderer">bitmaprenderer</code>", "<code
    data-x="canvas-context-webgl">webgl</code>", "<code
    data-x="canvas-context-webgl2">webgl2</code>", or "<code
    data-x="canvas-context-webgpu">webgpu</code>". <var>options</var> is handled by that API.</p>

    <p>This specification defines the "<code data-x="canvas-context-2d">2d</code>" and "<code
    data-x="canvas-context-bitmaprenderer">bitmaprenderer</code>" contexts below. The WebGL
    specifications define the "<code data-x="canvas-context-webgl">webgl</code>" and "<code
    data-x="canvas-context-webgl2">webgl2</code>" contexts. <cite>WebGPU</cite> defines the "<code
    data-x="canvas-context-webgpu">webgpu</code>" context. <ref>WEBGL</ref> <ref>WEBGPU</ref></p>

    <p>Returns null if <var>contextId</var> is not supported, or if the canvas has already been
    initialized with another context type (e.g., trying to get a "<code
    data-x="canvas-context-2d">2d</code>" context after getting a "<code
    data-x="canvas-context-webgl">webgl</code>" context).</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="HTMLCanvasElement"><code
  data-x="dom-canvas-getContext">getContext(<var>contextId</var>, <var>options</var>)</code></dfn>
  method of the <code>canvas</code> element, when invoked, must run these steps:

  <ol>
   <li><p>If <var>options</var> is not an <span data-x="idl-object">object</span>, then set
   <var>options</var> to null.</p></li>

   <li><p>Set <var>options</var> to the result of <span
   data-x="concept-idl-convert">converting</span> <var>options</var> to a JavaScript value.</p></li>

   <li>
    <p>Run the steps in the cell of the following table whose column header matches this
    <code>canvas</code> element's <span data-x="concept-canvas-context-mode">canvas context
    mode</span> and whose row header matches <var>contextId</var>:</p>

    <table>
     <thead>
      <tr>
       <td>
       <th><span data-x="concept-canvas-none">none</span>
       <th><span data-x="concept-canvas-2d">2d</span>
       <th><span data-x="concept-canvas-bitmaprenderer">bitmaprenderer</span>
       <th><span data-x="concept-canvas-webgl">webgl</span> or <span
       data-x="concept-canvas-webgl2">webgl2</span>
       <th><span data-x="concept-canvas-webgpu">webgpu</span>
       <th><span data-x="concept-canvas-placeholder">placeholder</span>
     <tbody>
      <tr>
       <th>"<dfn><code data-x="canvas-context-2d">2d</code></dfn>"
       <td>
        <ol>
         <li><p>Let <var>context</var> be the result of running the <span>2D context
         creation algorithm</span> given <span>this</span> and <var>options</var>.</p></li>

         <li><p>Set <span>this</span>'s <span data-x="concept-canvas-context-mode">context
         mode</span> to <span data-x="concept-canvas-2d">2d</span>.</p></li>

         <li><p>Return <var>context</var>.</p></li>
        </ol>
       <td>
        Return the same object as was returned the last time the method was invoked with this same
        first argument.
       <td>
        Return null.
       <td>
        Return null.
       <td>
        Return null.
       <td>
        Throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.
      <tr>
       <th>"<dfn><code data-x="canvas-context-bitmaprenderer">bitmaprenderer</code></dfn>"
       <td>
        <ol>
         <li><p>Let <var>context</var> be the result of running the
         <span><code>ImageBitmapRenderingContext</code> creation algorithm</span> given
         <span>this</span> and <var>options</var>.</p></li>

         <li><p>Set <span>this</span>'s <span data-x="concept-canvas-context-mode">context
         mode</span> to <span
         data-x="concept-canvas-bitmaprenderer">bitmaprenderer</span>.</p></li>

         <li><p>Return <var>context</var>.</p></li>
        </ol>
       <td>
        Return null.
       <td>
        Return the same object as was returned the last time the method was invoked with this same
        first argument.
       <td>
        Return null.
       <td>
        Return null.
       <td>
        Throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.
      <tr>
       <th>"<dfn><code data-x="canvas-context-webgl">webgl</code></dfn>" or "<dfn><code
       data-x="canvas-context-webgl2">webgl2</code></dfn>", if the user agent supports the WebGL
       feature in its current configuration
       <td>
        <ol>
         <li><p>Let <var>context</var> be the result of following the instructions given in
         the WebGL specifications' <i>Context Creation</i> sections. <ref>WEBGL</ref></p></li>

         <li><p>If <var>context</var> is null, then return null; otherwise set <span>this</span>'s
         <span data-x="concept-canvas-context-mode">context mode</span> to <span
         data-x="concept-canvas-webgl">webgl</span> or <span
         data-x="concept-canvas-webgl2">webgl2</span>.</p></li>

         <li><p>Return <var>context</var>.</p></li>
        </ol>
       <td>
        Return null.
       <td>
        Return null.
       <td>
        Return the same object as was returned the last time the method was invoked with this same
        first argument.
       <td>
        Return null.
       <td>
        Throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.
      <tr>
       <th>"<dfn><code data-x="canvas-context-webgpu">webgpu</code></dfn>", if the user
       agent supports the WebGPU feature in its current configuration
       <td>
        <ol>
         <li><p>Let <var>context</var> be the result of following the instructions given
         in <cite>WebGPU</cite>'s <a
         href="https://gpuweb.github.io/gpuweb/#canvas-rendering">Canvas Rendering</a>
         section. <ref>WEBGPU</ref></p></li>

         <li><p>If <var>context</var> is null, then return null; otherwise set
         <span>this</span>'s <span data-x="concept-canvas-context-mode">context mode</span> to
         <span data-x="concept-canvas-webgpu">webgpu</span>.</p></li>

         <li><p>Return <var>context</var>.</p></li>
        </ol>
       <td>
        Return null.
       <td>
        Return null.
       <td>
        Return null.
       <td>
        Return the same object as was returned the last time the method was invoked with this same
        first argument.
       <td>
        Throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.
      <tr>
       <th>An unsupported value*
       <td>
        Return null.
       <td>
        Return null.
       <td>
        Return null.
       <td>
        Return null.
       <td>
        Return null.
       <td>
        Throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.
    </table>

    <p class="tablenote"><small>* For example, the "<code
    data-x="canvas-context-webgl">webgl</code>" or "<code
    data-x="canvas-context-webgl2">webgl2</code>" value in the case of a user agent having exhausted
    the graphics hardware's abilities and having no software fallback implementation.</small></p>
   </li>
  </ol>

  <hr>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>url</var> = <var>canvas</var>.<span subdfn data-x="dom-canvas-toDataURL">toDataURL</span>([ <var>type</var> [, <var>quality</var> ] ])</code></dt>

   <dd>
    <p>Returns a <span data-x="data protocol"><code>data:</code> URL</span> for the image in the
    canvas.</p>

    <p>The first argument, if provided, controls the type of the image to be returned (e.g. PNG or
    JPEG). The default is "<code>image/png</code>"; that type is also used if the given type isn't
    supported. The second argument applies if the type is an image format that supports variable
    quality (such as "<code>image/jpeg</code>"), and is a number in the range 0.0 to 1.0 inclusive
    indicating the desired quality level for the resulting image.</p>

    <p>When trying to use types other than "<code>image/png</code>", authors can check if the image
    was really returned in the requested format by checking to see if the returned string starts
    with one of the exact strings "<code data-x="">data:image/png,</code>" or "<code
    data-x="">data:image/png;</code>". If it does, the image is PNG, and thus the requested type was
    not supported. (The one exception to this is if the canvas has either no height or no width, in
    which case the result might simply be "<code data-x="">data:,</code>".)</p>
   </dd>

   <dt><code data-x=""><var>canvas</var>.<span subdfn data-x="dom-canvas-toBlob">toBlob</span>(<var>callback</var> [, <var>type</var> [, quality ] ])</code></dt>

   <dd>
    <p>Creates a <code>Blob</code> object representing a file containing the image in the canvas,
    and invokes a callback with a handle to that object.</p>

    <p>The second argument, if provided, controls the type of the image to be returned (e.g. PNG or
    JPEG). The default is "<code>image/png</code>"; that type is also used if the given type isn't
    supported. The third argument applies if the type is an image format that supports variable
    quality (such as "<code>image/jpeg</code>"), and is a number in the range 0.0 to 1.0 inclusive
    indicating the desired quality level for the resulting image.</p>
   </dd>

   <dt><code data-x=""><var>canvas</var>.<span subdfn data-x="dom-canvas-transferControlToOffscreen">transferControlToOffscreen</span>()</code></dt>

   <dd>
    <p>Returns a newly created <code>OffscreenCanvas</code> object that uses the
    <code>canvas</code> element as a placeholder. Once the <code>canvas</code> element has become a
    placeholder for an <code>OffscreenCanvas</code> object, its natural size can no longer be
    changed, and it cannot have a rendering context. The content of the placeholder canvas is
    updated on the <code>OffscreenCanvas</code>'s <span>relevant agent</span>'s <span
    data-x="concept-agent-event-loop">event loop</span>'s <span>update the rendering</span>
    steps.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="HTMLCanvasElement"><code
  data-x="dom-canvas-toDataURL">toDataURL(<var>type</var>, <var>quality</var>)</code></dfn> method,
  when invoked, must run these steps:</p>

  <ol>
   <li><p>If this <code>canvas</code> element's bitmap's <span
   data-x="concept-canvas-origin-clean">origin-clean</span> flag is set to false, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p>

   <li><p>If this <code>canvas</code> element's bitmap has no pixels (i.e. either its horizontal
   dimension or its vertical dimension is zero) then return the string "<code
   data-x="">data:,</code>". (This is the shortest <span data-x="data
   protocol"><code>data:</code> URL</span>; it represents the empty string in a <code
   data-x="">text/plain</code> resource.)</p></li>

   <li><p>Let <var>file</var> be <span data-x="a serialization of the bitmap as a file">a
   serialization of this <code>canvas</code> element's bitmap as a file</span>, passing
   <var>type</var> and <var>quality</var> if given.</p></li>

   <li><p>If <var>file</var> is null then return "<code data-x="">data:,</code>".</p></li>

   <li><p>Return a <span data-x="data protocol"><code>data:</code> URL</span> representing
   <var>file</var>. <ref>RFC2397</ref></p>

   <!-- should we explicitly require the URL to be base64-encoded and not have any parameters, to
   ensure the same exact URL is generated in each browser? -->
  </ol>

  <p>The <dfn method for="HTMLCanvasElement"><code
  data-x="dom-canvas-toBlob">toBlob(<var>callback</var>, <var>type</var>,
  <var>quality</var>)</code></dfn> method, when invoked, must run these steps:</p>

  <ol>
   <li><p>If this <code>canvas</code> element's bitmap's <span
   data-x="concept-canvas-origin-clean">origin-clean</span> flag is set to false, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>result</var> be null.</p></li>

   <li><p>If this <code>canvas</code> element's bitmap has pixels (i.e., neither its horizontal
   dimension nor its vertical dimension is zero), then set <var>result</var> to a copy of this
   <code>canvas</code> element's bitmap.

   <li>
    <p>Run these steps <span>in parallel</span>:</p>

    <ol>
     <li><p>If <var>result</var> is non-null, then set <var>result</var> to <span data-x="a
     serialization of the bitmap as a file">a serialization of <var>result</var> as a file</span>
     with <var>type</var> and <var>quality</var> if given.</p></li>

     <li>
      <p><span>Queue an element task</span> on the
      <!--en-GB--><dfn id="canvas-blob-serialisation-task-source">canvas blob serialization task
      source</dfn> given the <code>canvas</code> element to run these steps:</p>

      <ol>
       <li><p>If <var>result</var> is non-null, then set <var>result</var> to a new
       <code>Blob</code> object, created in the <span data-x="concept-relevant-realm">relevant
       realm</span> of this <code>canvas</code> element, representing <var>result</var>.
       <ref>FILEAPI</ref></p></li>

       <li><p><span data-x="es-invoking-callback-functions">Invoke</span> <var>callback</var> with
       « <var>result</var> » and "<code data-x="">report</code>".</p></li>
      </ol>
     </li>
    </ol>
   </li>
  </ol>

  <p>The <dfn method for="HTMLCanvasElement"><code
  data-x="dom-canvas-transferControlToOffscreen">transferControlToOffscreen()</code></dfn> method,
  when invoked, must run these steps:</p>

  <ol>
   <li><p>If this <code>canvas</code> element's <span data-x="concept-canvas-context-mode">context
   mode</span> is not set to <span data-x="concept-canvas-none">none</span>, throw an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>offscreenCanvas</var> be a new <code>OffscreenCanvas</code> object with its
   width and height equal to the values of the <code data-x="attr-canvas-width">width</code>
   and <code data-x="attr-canvas-height">height</code> content attributes of this
   <code>canvas</code> element.</p></li>

   <li><p>Set the <var>offscreenCanvas</var>'s <span
   data-x="offscreencanvas-placeholder">placeholder <code>canvas</code> element</span>
   to a weak reference to this <code>canvas</code> element.</p></li>

   <li><p>Set this <code>canvas</code> element's <span data-x="concept-canvas-context-mode">context
   mode</span> to <span data-x="concept-canvas-placeholder">placeholder</span>.</p></li>

   <li><p>Set the <var>offscreenCanvas</var>'s <span
   data-x="offscreencanvas-inherited-lang">inherited language</span> to the
   <span>language</span> of this <code>canvas</code> element.</p></li>

   <li><p>Set the <var>offscreenCanvas</var>'s <span
   data-x="offscreencanvas-inherited-direction">inherited direction</span> to the <span
   data-x="the directionality">directionality</span> of this <code>canvas</code> element.</p></li>

   <li><p>Return <var>offscreenCanvas</var>.</p></li>
  </ol>

  </div>


  <h5 id="2dcontext">The 2D rendering context</h5>

  <!-- v2: we're on v5. suggestions for subsequent versions are marked v6, v7, v8. -->

  <pre><code class="idl">typedef (<span>HTMLImageElement</span> or
         <span>SVGImageElement</span>) <dfn typedef>HTMLOrSVGImageElement</dfn>;

typedef (<span>HTMLOrSVGImageElement</span> or
         <span>HTMLVideoElement</span> or
         <span>HTMLCanvasElement</span> or
         <span>ImageBitmap</span> or
         <span>OffscreenCanvas</span> or
         <span>VideoFrame</span>) <dfn typedef>CanvasImageSource</dfn>;

enum <dfn enum>PredefinedColorSpace</dfn> { "<span data-x="dom-PredefinedColorSpace-srgb">srgb</span>", "<span data-x="dom-PredefinedColorSpace-display-p3">display-p3</span>" };

enum <dfn enum>CanvasColorType</dfn> { "<span data-x="dom-CanvasColorType-unorm8">unorm8</span>", "<span data-x="dom-CanvasColorType-float16">float16</span>" };

enum <dfn enum>CanvasFillRule</dfn> { "<span data-x="dom-context-2d-fillRule-nonzero">nonzero</span>", "<span data-x="dom-context-2d-fillRule-evenodd">evenodd</span>" };

dictionary <dfn dictionary>CanvasRenderingContext2DSettings</dfn> {
  boolean <span data-x="dom-CanvasRenderingContext2DSettings-alpha">alpha</span> = true;
  boolean <span data-x="dom-CanvasRenderingContext2DSettings-desynchronized">desynchronized</span> = false;
  <span>PredefinedColorSpace</span> <span data-x="dom-CanvasRenderingContext2DSettings-colorSpace">colorSpace</span> = "<span data-x="dom-PredefinedColorSpace-srgb">srgb</span>";
  <span>CanvasColorType</span> <span data-x="dom-CanvasRenderingContext2DSettings-colorType">colorType</span> = "<span data-x="dom-CanvasColorType-unorm8">unorm8</span>";
  boolean <span data-x="dom-CanvasRenderingContext2DSettings-willReadFrequently">willReadFrequently</span> = false;
};

enum <dfn enum>ImageSmoothingQuality</dfn> { "<span data-x="dom-context-2d-imageSmoothingQuality-low">low</span>", "<span data-x="dom-context-2d-imageSmoothingQuality-medium">medium</span>", "<span data-x="dom-context-2d-imageSmoothingQuality-high">high</span>" };

[Exposed=Window]
interface <dfn interface>CanvasRenderingContext2D</dfn> {
  // back-reference to the canvas
  readonly attribute <span>HTMLCanvasElement</span> <span data-x="dom-context-2d-canvas">canvas</span>;
};
<span>CanvasRenderingContext2D</span> includes <span>CanvasSettings</span>;
<span>CanvasRenderingContext2D</span> includes <span>CanvasState</span>;
<span>CanvasRenderingContext2D</span> includes <span>CanvasTransform</span>;
<span>CanvasRenderingContext2D</span> includes <span>CanvasCompositing</span>;
<span>CanvasRenderingContext2D</span> includes <span>CanvasImageSmoothing</span>;
<span>CanvasRenderingContext2D</span> includes <span>CanvasFillStrokeStyles</span>;
<span>CanvasRenderingContext2D</span> includes <span>CanvasShadowStyles</span>;
<span>CanvasRenderingContext2D</span> includes <span>CanvasFilters</span>;
<span>CanvasRenderingContext2D</span> includes <span>CanvasRect</span>;
<span>CanvasRenderingContext2D</span> includes <span>CanvasDrawPath</span>;
<span>CanvasRenderingContext2D</span> includes <span>CanvasUserInterface</span>;
<span>CanvasRenderingContext2D</span> includes <span>CanvasText</span>;
<span>CanvasRenderingContext2D</span> includes <span>CanvasDrawImage</span>;
<span>CanvasRenderingContext2D</span> includes <span>CanvasImageData</span>;
<span>CanvasRenderingContext2D</span> includes <span>CanvasPathDrawingStyles</span>;
<span>CanvasRenderingContext2D</span> includes <span>CanvasTextDrawingStyles</span>;
<span>CanvasRenderingContext2D</span> includes <span>CanvasPath</span>;

interface mixin <dfn interface>CanvasSettings</dfn> {
  // settings
  <span>CanvasRenderingContext2DSettings</span> <span data-x="dom-context-2d-canvas-getcontextattributes">getContextAttributes</span>();
};

interface mixin <dfn interface>CanvasState</dfn> {
  // state
  undefined <span data-x="dom-context-2d-save">save</span>(); // push state on state stack
  undefined <span data-x="dom-context-2d-restore">restore</span>(); // pop state stack and restore state
  undefined <span data-x="dom-context-2d-reset">reset</span>(); // <span>reset the rendering context to its default state</span>
  boolean <span data-x="dom-context-2d-isContextLost">isContextLost</span>(); // return whether context is lost
};

interface mixin <dfn interface>CanvasTransform</dfn> {
  // transformations (default transform is the identity matrix)
  undefined <span data-x="dom-context-2d-scale">scale</span>(unrestricted double x, unrestricted double y);
  undefined <span data-x="dom-context-2d-rotate">rotate</span>(unrestricted double angle);
  undefined <span data-x="dom-context-2d-translate">translate</span>(unrestricted double x, unrestricted double y);
  undefined <span data-x="dom-context-2d-transform">transform</span>(unrestricted double a, unrestricted double b, unrestricted double c, unrestricted double d, unrestricted double e, unrestricted double f);

  [NewObject] <span>DOMMatrix</span> <span data-x="dom-context-2d-getTransform">getTransform</span>();
  undefined <span data-x="dom-context-2d-setTransform">setTransform</span>(unrestricted double a, unrestricted double b, unrestricted double c, unrestricted double d, unrestricted double e, unrestricted double f);
  undefined <span data-x="dom-context-2d-setTransform-matrix">setTransform</span>(optional <span>DOMMatrix2DInit</span> transform = {});
  undefined <span data-x="dom-context-2d-resetTransform">resetTransform</span>();
<!--
  // v7 we've also received requests (though not many so far) for:
  undefined skew(...);                             // is this common enough that one can't just use setTransform()?
  undefined reflect(...); and undefined mirror(...);  // aren't negative values in scale() sufficient for these?
-->
};

interface mixin <dfn interface>CanvasCompositing</dfn> {
  // compositing
  attribute unrestricted double <span data-x="dom-context-2d-globalAlpha">globalAlpha</span>; // (default 1.0)
  attribute DOMString <span data-x="dom-context-2d-globalCompositeOperation">globalCompositeOperation</span>; // (default "source-over")
};

interface mixin <dfn interface>CanvasImageSmoothing</dfn> {
  // image smoothing
  attribute boolean <span data-x="dom-context-2d-imageSmoothingEnabled">imageSmoothingEnabled</span>; // (default true)
  attribute <span>ImageSmoothingQuality</span> <span data-x="dom-context-2d-imageSmoothingQuality">imageSmoothingQuality</span>; // (default low)
<!--
  // v7 we've also received requests for:
  - turning off antialiasing to avoid seams when patterns are painted next to each other
    - might be better to overdraw?
    - might be better to just draw at a higher res then downsample, like for 3d?
  - nested layers
    - the ability to composite an entire set of drawing operations with one shadow all at once
      https://lists.w3.org/Archives/Public/public-whatwg-archive/2008Aug/0024.html
-->
};

interface mixin <dfn interface>CanvasFillStrokeStyles</dfn> {
  // colors and styles (see also the <span>CanvasPathDrawingStyles</span> and <span>CanvasTextDrawingStyles</span> interfaces)
  attribute (DOMString or CanvasGradient or CanvasPattern) <span data-x="dom-context-2d-strokeStyle">strokeStyle</span>; // (default black)
  attribute (DOMString or CanvasGradient or CanvasPattern) <span data-x="dom-context-2d-fillStyle">fillStyle</span>; // (default black)
  <span>CanvasGradient</span> <span data-x="dom-context-2d-createLinearGradient">createLinearGradient</span>(double x0, double y0, double x1, double y1);
  <span>CanvasGradient</span> <span data-x="dom-context-2d-createRadialGradient">createRadialGradient</span>(double x0, double y0, double r0, double x1, double y1, double r1);
  <span>CanvasGradient</span> <span data-x="dom-context-2d-createConicGradient">createConicGradient</span>(double startAngle, double x, double y);
  <span>CanvasPattern</span>? <span data-x="dom-context-2d-createPattern">createPattern</span>(<span>CanvasImageSource</span> image, [<span>LegacyNullToEmptyString</span>] DOMString repetition);
<!--
  // v8 we received one request from Ralf Richard G&oml;bel for a new kind of pattern: a hatch.
  // basically it would be a series of dash styles, angles, line widths, and offsets
-->
};

interface mixin <dfn interface>CanvasShadowStyles</dfn> {
  // shadows
  attribute unrestricted double <span data-x="dom-context-2d-shadowOffsetX">shadowOffsetX</span>; // (default 0)
  attribute unrestricted double <span data-x="dom-context-2d-shadowOffsetY">shadowOffsetY</span>; // (default 0)
  attribute unrestricted double <span data-x="dom-context-2d-shadowBlur">shadowBlur</span>; // (default 0)
  attribute DOMString <span data-x="dom-context-2d-shadowColor">shadowColor</span>; // (default <span>transparent black</span>)
};

interface mixin <dfn interface>CanvasFilters</dfn> {
  // filters
  attribute DOMString <span data-x="dom-context-2d-filter">filter</span>; // (default "none")
};

interface mixin <dfn interface>CanvasRect</dfn> {
  // rects
  undefined <span data-x="dom-context-2d-clearRect">clearRect</span>(unrestricted double x, unrestricted double y, unrestricted double w, unrestricted double h);
  undefined <span data-x="dom-context-2d-fillRect">fillRect</span>(unrestricted double x, unrestricted double y, unrestricted double w, unrestricted double h);
  undefined <span data-x="dom-context-2d-strokeRect">strokeRect</span>(unrestricted double x, unrestricted double y, unrestricted double w, unrestricted double h);
};

interface mixin <dfn interface>CanvasDrawPath</dfn> {
  // path API (see also <span>CanvasPath</span>)
  undefined <span data-x="dom-context-2d-beginPath">beginPath</span>();
  undefined <span data-x="dom-context-2d-fill">fill</span>(optional <span>CanvasFillRule</span> fillRule = "<span data-x="dom-context-2d-fillRule-nonzero">nonzero</span>");
  undefined <span data-x="dom-context-2d-fill-path">fill</span>(<span>Path2D</span> path, optional <span>CanvasFillRule</span> fillRule = "<span data-x="dom-context-2d-fillRule-nonzero">nonzero</span>");
  undefined <span data-x="dom-context-2d-stroke">stroke</span>();
  undefined <span data-x="dom-context-2d-stroke-path">stroke</span>(<span>Path2D</span> path);
  undefined <span data-x="dom-context-2d-clip">clip</span>(optional <span>CanvasFillRule</span> fillRule = "<span data-x="dom-context-2d-fillRule-nonzero">nonzero</span>");
  undefined <span data-x="dom-context-2d-clip-path">clip</span>(<span>Path2D</span> path, optional <span>CanvasFillRule</span> fillRule = "<span data-x="dom-context-2d-fillRule-nonzero">nonzero</span>");
  boolean <span data-x="dom-context-2d-isPointInPath">isPointInPath</span>(unrestricted double x, unrestricted double y, optional <span>CanvasFillRule</span> fillRule = "<span data-x="dom-context-2d-fillRule-nonzero">nonzero</span>");
  boolean <span data-x="dom-context-2d-isPointInPath-path">isPointInPath</span>(<span>Path2D</span> path, unrestricted double x, unrestricted double y, optional <span>CanvasFillRule</span> fillRule = "<span data-x="dom-context-2d-fillRule-nonzero">nonzero</span>");
  boolean <span data-x="dom-context-2d-isPointInStroke">isPointInStroke</span>(unrestricted double x, unrestricted double y);
  boolean <span data-x="dom-context-2d-isPointInStroke-path">isPointInStroke</span>(<span>Path2D</span> path, unrestricted double x, unrestricted double y);
};

interface mixin <dfn interface>CanvasUserInterface</dfn> {
  undefined <span data-x="dom-context-2d-drawFocusIfNeeded">drawFocusIfNeeded</span>(<span>Element</span> element);
  undefined <span data-x="dom-context-2d-drawFocusIfNeeded-path-element">drawFocusIfNeeded</span>(<span>Path2D</span> path, <span>Element</span> element);
};

interface mixin <dfn interface>CanvasText</dfn> {
  // text (see also the <span>CanvasPathDrawingStyles</span> and <span>CanvasTextDrawingStyles</span> interfaces)
  undefined <span data-x="dom-context-2d-fillText">fillText</span>(DOMString text, unrestricted double x, unrestricted double y, optional unrestricted double maxWidth);
  undefined <span data-x="dom-context-2d-strokeText">strokeText</span>(DOMString text, unrestricted double x, unrestricted double y, optional unrestricted double maxWidth);
  <span>TextMetrics</span> <span data-x="dom-context-2d-measureText">measureText</span>(DOMString text);
};

interface mixin <dfn interface>CanvasDrawImage</dfn> {
  // drawing images
  undefined <span data-x="dom-context-2d-drawImage">drawImage</span>(<span>CanvasImageSource</span> image, unrestricted double dx, unrestricted double dy);
  undefined <span data-x="dom-context-2d-drawImage">drawImage</span>(<span>CanvasImageSource</span> image, unrestricted double dx, unrestricted double dy, unrestricted double dw, unrestricted double dh);
  undefined <span data-x="dom-context-2d-drawImage">drawImage</span>(<span>CanvasImageSource</span> image, unrestricted double sx, unrestricted double sy, unrestricted double sw, unrestricted double sh, unrestricted double dx, unrestricted double dy, unrestricted double dw, unrestricted double dh);
};

interface mixin <dfn interface>CanvasImageData</dfn> {
  // <span>pixel manipulation</span>
  <span>ImageData</span> <span data-x="dom-context-2d-createImageData">createImageData</span>([EnforceRange] long sw, [EnforceRange] long sh, optional <span>ImageDataSettings</span> settings = {});
  <span>ImageData</span> <span data-x="dom-context-2d-createImageData-imagedata">createImageData</span>(<span>ImageData</span> imageData);
  <span>ImageData</span> <span data-x="dom-context-2d-getImageData">getImageData</span>([EnforceRange] long sx, [EnforceRange] long sy, [EnforceRange] long sw, [EnforceRange] long sh, optional <span>ImageDataSettings</span> settings = {});
  undefined <span data-x="dom-context-2d-putImageData-short">putImageData</span>(<span>ImageData</span> imageData, [EnforceRange] long dx, [EnforceRange] long dy);
  undefined <span data-x="dom-context-2d-putImageData">putImageData</span>(<span>ImageData</span> imageData, [EnforceRange] long dx, [EnforceRange] long dy, [EnforceRange] long dirtyX, [EnforceRange] long dirtyY, [EnforceRange] long dirtyWidth, [EnforceRange] long dirtyHeight);
};

enum <dfn enum>CanvasLineCap</dfn> { "butt", "round", "square" };
enum <dfn enum>CanvasLineJoin</dfn> { "round", "bevel", "miter" };
enum <dfn enum>CanvasTextAlign</dfn> { "<span data-x="dom-context-2d-textalign-start">start</span>", "<span data-x="dom-context-2d-textalign-end">end</span>", "<span data-x="dom-context-2d-textalign-left">left</span>", "<span data-x="dom-context-2d-textalign-right">right</span>", "<span data-x="dom-context-2d-textalign-center">center</span>" };
enum <dfn enum>CanvasTextBaseline</dfn> { "<span data-x="dom-context-2d-textbaseline-top">top</span>", "<span data-x="dom-context-2d-textbaseline-hanging">hanging</span>", "<span data-x="dom-context-2d-textbaseline-middle">middle</span>", "<span data-x="dom-context-2d-textbaseline-alphabetic">alphabetic</span>", "<span data-x="dom-context-2d-textbaseline-ideographic">ideographic</span>", "<span data-x="dom-context-2d-textbaseline-bottom">bottom</span>" };
enum <dfn enum>CanvasDirection</dfn> { "<span data-x="dom-context-2d-direction-ltr">ltr</span>", "<span data-x="dom-context-2d-direction-rtl">rtl</span>", "<span data-x="dom-context-2d-direction-inherit">inherit</span>" };
enum <dfn enum>CanvasFontKerning</dfn> { "<span data-x="dom-context-2d-fontkerning-auto">auto</span>", "<span data-x="dom-context-2d-fontkerning-normal">normal</span>", "<span data-x="dom-context-2d-fontkerning-none">none</span>" };
enum <dfn enum>CanvasFontStretch</dfn> { "<span data-x="dom-context-2d-fontstretch-ultra-condensed">ultra-condensed</span>", "<span data-x="dom-context-2d-fontstretch-extra-condensed">extra-condensed</span>", "<span data-x="dom-context-2d-fontstretch-condensed">condensed</span>", "<span data-x="dom-context-2d-fontstretch-semi-condensed">semi-condensed</span>", "<span data-x="dom-context-2d-fontstretch-normal">normal</span>", "<span data-x="dom-context-2d-fontstretch-semi-expanded">semi-expanded</span>", "<span data-x="dom-context-2d-fontstretch-expanded">expanded</span>", "<span data-x="dom-context-2d-fontstretch-extra-expanded">extra-expanded</span>", "<span data-x="dom-context-2d-fontstretch-ultra-expanded">ultra-expanded</span>" };
enum <dfn enum>CanvasFontVariantCaps</dfn> { "<span data-x="dom-context-2d-fontvariantcaps-normal">normal</span>", "<span data-x="dom-context-2d-fontvariantcaps-small-caps">small-caps</span>", "<span data-x="dom-context-2d-fontvariantcaps-all-small-caps">all-small-caps</span>", "<span data-x="dom-context-2d-fontvariantcaps-petite-caps">petite-caps</span>", "<span data-x="dom-context-2d-fontvariantcaps-all-petite-caps">all-petite-caps</span>", "<span data-x="dom-context-2d-fontvariantcaps-unicase">unicase</span>", "<span data-x="dom-context-2d-fontvariantcaps-titling-caps">titling-caps</span>" };
enum <dfn enum>CanvasTextRendering</dfn> { "<span data-x="dom-context-2d-textRendering-auto">auto</span>", "<span data-x="dom-context-2d-textRendering-optimizeSpeed">optimizeSpeed</span>", "<span data-x="dom-context-2d-textRendering-optimizeLegibility">optimizeLegibility</span>", "<span data-x="dom-context-2d-textRendering-geometricPrecision">geometricPrecision</span>" };

interface mixin <dfn interface>CanvasPathDrawingStyles</dfn> {
  // line caps/joins
  attribute unrestricted double <span data-x="dom-context-2d-lineWidth">lineWidth</span>; // (default 1)
  attribute <span>CanvasLineCap</span> <span data-x="dom-context-2d-lineCap">lineCap</span>; // (default "butt")
  attribute <span>CanvasLineJoin</span> <span data-x="dom-context-2d-lineJoin">lineJoin</span>; // (default "miter")
  attribute unrestricted double <span data-x="dom-context-2d-miterLimit">miterLimit</span>; // (default 10)

  // dashed lines
  undefined <span data-x="dom-context-2d-setLineDash">setLineDash</span>(sequence&lt;unrestricted double> segments); // default empty
  sequence&lt;unrestricted double> <span data-x="dom-context-2d-getLineDash">getLineDash</span>();
  attribute unrestricted double <span data-x="dom-context-2d-lineDashOffset">lineDashOffset</span>;
};

interface mixin <dfn interface>CanvasTextDrawingStyles</dfn> {
  // text
  attribute DOMString <span data-x="dom-context-2d-lang">lang</span>; // (default: "inherit")
  attribute DOMString <span data-x="dom-context-2d-font">font</span>; // (default 10px sans-serif)
  attribute <span>CanvasTextAlign</span> <span data-x="dom-context-2d-textAlign">textAlign</span>; // (default: "start")
  attribute <span>CanvasTextBaseline</span> <span data-x="dom-context-2d-textBaseline">textBaseline</span>; // (default: "alphabetic")
  attribute <span>CanvasDirection</span> <span data-x="dom-context-2d-direction">direction</span>; // (default: "inherit")
  attribute DOMString <span data-x="dom-context-2d-letterSpacing">letterSpacing</span>; // (default: "0px")
  attribute <span>CanvasFontKerning</span> <span data-x="dom-context-2d-fontKerning">fontKerning</span>; // (default: "auto")
  attribute <span>CanvasFontStretch</span> <span data-x="dom-context-2d-fontStretch">fontStretch</span>; // (default: "normal")
  attribute <span>CanvasFontVariantCaps</span> <span data-x="dom-context-2d-fontVariantCaps">fontVariantCaps</span>; // (default: "normal")
  attribute <span>CanvasTextRendering</span> <span data-x="dom-context-2d-textRendering">textRendering</span>; // (default: "auto")
  attribute DOMString <span data-x="dom-context-2d-wordSpacing">wordSpacing</span>; // (default: "0px")
};

interface mixin <dfn interface>CanvasPath</dfn> {
  // shared path API methods
  undefined <span data-x="dom-context-2d-closePath">closePath</span>();
  undefined <span data-x="dom-context-2d-moveTo">moveTo</span>(unrestricted double x, unrestricted double y);
  undefined <span data-x="dom-context-2d-lineTo">lineTo</span>(unrestricted double x, unrestricted double y);
  undefined <span data-x="dom-context-2d-quadraticCurveTo">quadraticCurveTo</span>(unrestricted double cpx, unrestricted double cpy, unrestricted double x, unrestricted double y);
  undefined <span data-x="dom-context-2d-bezierCurveTo">bezierCurveTo</span>(unrestricted double cp1x, unrestricted double cp1y, unrestricted double cp2x, unrestricted double cp2y, unrestricted double x, unrestricted double y);
  undefined <span data-x="dom-context-2d-arcTo">arcTo</span>(unrestricted double x1, unrestricted double y1, unrestricted double x2, unrestricted double y2, unrestricted double radius); <!-- see ARC-ORDER note below -->
  undefined <span data-x="dom-context-2d-rect">rect</span>(unrestricted double x, unrestricted double y, unrestricted double w, unrestricted double h);
  undefined <span data-x="dom-context-2d-roundRect">roundRect</span>(unrestricted double x, unrestricted double y, unrestricted double w, unrestricted double h, optional (unrestricted double or <span>DOMPointInit</span> or sequence&lt;(unrestricted double or <span>DOMPointInit</span>)>) radii = 0);
  undefined <span data-x="dom-context-2d-arc">arc</span>(unrestricted double x, unrestricted double y, unrestricted double radius, unrestricted double startAngle, unrestricted double endAngle, optional boolean counterclockwise = false); <!-- see ARC-ORDER note below -->
  undefined <span data-x="dom-context-2d-ellipse">ellipse</span>(unrestricted double x, unrestricted double y, unrestricted double radiusX, unrestricted double radiusY, unrestricted double rotation, unrestricted double startAngle, unrestricted double endAngle, optional boolean counterclockwise = false); <!-- see ARC-ORDER note below -->
};

[Exposed=(Window,Worker)]
interface <dfn interface>CanvasGradient</dfn> {
  // opaque object
  undefined <span data-x="dom-canvasgradient-addColorStop">addColorStop</span>(double offset, DOMString color);
};

[Exposed=(Window,Worker)]
interface <dfn interface>CanvasPattern</dfn> {
  // opaque object
  undefined <span data-x="dom-canvaspattern-setTransform">setTransform</span>(optional <span>DOMMatrix2DInit</span> transform = {});
};

[Exposed=(Window,Worker)]
interface <dfn interface>TextMetrics</dfn> {
  // x-direction
  readonly attribute double <span data-x="dom-textmetrics-width">width</span>; // advance width
  readonly attribute double <span data-x="dom-textmetrics-actualBoundingBoxLeft">actualBoundingBoxLeft</span>;
  readonly attribute double <span data-x="dom-textmetrics-actualBoundingBoxRight">actualBoundingBoxRight</span>;

  // y-direction
  readonly attribute double <span data-x="dom-textmetrics-fontBoundingBoxAscent">fontBoundingBoxAscent</span>;
  readonly attribute double <span data-x="dom-textmetrics-fontBoundingBoxDescent">fontBoundingBoxDescent</span>;
  readonly attribute double <span data-x="dom-textmetrics-actualBoundingBoxAscent">actualBoundingBoxAscent</span>;
  readonly attribute double <span data-x="dom-textmetrics-actualBoundingBoxDescent">actualBoundingBoxDescent</span>;
  readonly attribute double <span data-x="dom-textmetrics-emHeightAscent">emHeightAscent</span>;
  readonly attribute double <span data-x="dom-textmetrics-emHeightDescent">emHeightDescent</span>;
  readonly attribute double <span data-x="dom-textmetrics-hangingBaseline">hangingBaseline</span>;
  readonly attribute double <span data-x="dom-textmetrics-alphabeticBaseline">alphabeticBaseline</span>;
  readonly attribute double <span data-x="dom-textmetrics-ideographicBaseline">ideographicBaseline</span>;
};

dictionary <dfn dictionary>ImageDataSettings</dfn> {
  <span>PredefinedColorSpace</span> <span data-x="dom-PredefinedColorSpace-srgb">colorSpace</span>;
};

[Exposed=(Window,Worker),
 <span>Serializable</span>]
interface <dfn interface>ImageData</dfn> {
  <span data-x="dom-imagedata">constructor</span>(unsigned long sw, unsigned long sh, optional <span>ImageDataSettings</span> settings = {});
  <span data-x="dom-imagedata-with-data">constructor</span>(<span data-x="idl-Uint8ClampedArray">Uint8ClampedArray</span> data, unsigned long sw, optional unsigned long sh, optional <span>ImageDataSettings</span> settings = {});

  readonly attribute unsigned long <span data-x="dom-imagedata-width">width</span>;
  readonly attribute unsigned long <span data-x="dom-imagedata-height">height</span>;
  readonly attribute <span data-x="idl-Uint8ClampedArray">Uint8ClampedArray</span> <span data-x="dom-imagedata-data">data</span>;
  readonly attribute <span>PredefinedColorSpace</span> <span data-x="dom-imagedata-colorSpace">colorSpace</span>;
};

[Exposed=(Window,Worker)]
interface <dfn interface>Path2D</dfn> {
  <span data-x="dom-Path2D">constructor</span>(optional (<span>Path2D</span> or DOMString) path);

  undefined <span data-x="dom-Path2D-addPath">addPath</span>(<span>Path2D</span> path, optional <span>DOMMatrix2DInit</span> transform = {});
};
<span>Path2D</span> includes <span>CanvasPath</span>;</code></pre>

 <!-- ARC-ORDER note (see above):
      some demos rely on the precise order of the arc() and arcTo()
      methods, see https://bugzilla.mozilla.org/show_bug.cgi?id=623437
      for an example, and its duplicates for more -->

  <p class="note" w-nodev>To maintain compatibility with existing web content, user agents need to
  enumerate methods defined in <code>CanvasUserInterface</code> immediately after the <code
  data-x="dom-context-2d-stroke">stroke()</code> method on <code>CanvasRenderingContext2D</code>
  objects.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>context</var> = <var>canvas</var>.<span data-x="dom-canvas-getContext">getContext</span>('2d' [, { [ <span data-x="dom-CanvasRenderingContext2DSettings-alpha">alpha</span>: true ] [, <span data-x="dom-CanvasRenderingContext2DSettings-desynchronized">desynchronized</span>: false ] [, <span data-x="dom-CanvasRenderingContext2DSettings-colorSpace">colorSpace</span>: 'srgb'] [, <span data-x="dom-CanvasRenderingContext2DSettings-willReadFrequently">willReadFrequently</span>: false ]} ])</code></dt>
   <dd>
    <p>Returns a <code>CanvasRenderingContext2D</code> object that is permanently bound to a
    particular <code>canvas</code> element.</p>

    <p>If the <code data-x="dom-CanvasRenderingContext2DSettings-alpha">alpha</code> member is
    false, then the context is forced to always be opaque.</p>

    <p>If the <code
    data-x="dom-CanvasRenderingContext2DSettings-desynchronized">desynchronized</code> member is
    true, then the context might be <span
    data-x="concept-canvas-desynchronized">desynchronized</span>.</p>

    <p>The <code data-x="dom-CanvasRenderingContext2DSettings-colorSpace">colorSpace</code> member
    specifies the <span data-x="concept-canvas-color-space">color space</span> of the rendering
    context.</p>

    <p>The <code data-x="dom-CanvasRenderingContext2DSettings-colorType">colorType</code> member
    specifies the <span data-x="concept-canvas-color-type">color type</span> of the rendering
    context.</p>

    <p>If the <code
    data-x="dom-CanvasRenderingContext2DSettings-willReadFrequently">willReadFrequently</code>
    member is true, then the context is marked for <span
    data-x="concept-canvas-will-read-frequently">readback optimization</span>.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-canvas">canvas</span></code></dt>
   <dd><p>Returns the <code>canvas</code> element.</p></dd>

   <dt><code data-x=""><var>attributes</var> = <var>context</var>.<span data-x="dom-context-2d-canvas-getcontextattributes">getContextAttributes</span>()</code></dt>
   <dd>
    <p>Returns an object whose:</p>

    <ul>
     <li><code data-x="concept-canvas-alpha">alpha</code> member is true if the context has an alpha
     component that is not 1.0; otherwise false.</li>

     <li><code data-x="dom-CanvasRenderingContext2DSettings-desynchronized">desynchronized</code>
     member is true if the context can be <span data-x="concept-canvas-desynchronized">desynchronized</span>.</li>

     <li><code data-x="dom-CanvasRenderingContext2DSettings-colorSpace">colorSpace</code> member is
     a string indicating the context's <span data-x="concept-canvas-color-space">color
     space</span>.</li>

     <li><code data-x="dom-CanvasRenderingContext2DSettings-colorType">colorType</code> member is
     a string indicating the context's <span data-x="concept-canvas-color-type">color
     type</span>.</li>

     <li><code data-x="dom-CanvasRenderingContext2DSettings-willReadFrequently">willReadFrequently</code>
     member is true if the context is marked for <span data-x="concept-canvas-will-read-frequently">readback
     optimization</span>.</li>
    </ul>
   </dd>
  </dl>

  <div w-nodev>

  <hr>

  <p>The <code>CanvasRenderingContext2D</code> 2D rendering context represents a flat linear
  Cartesian surface whose origin (0,0) is at the top left corner, with the coordinate space having
  <var>x</var> values increasing when going right, and <var>y</var> values increasing when going
  down. The <var>x</var>-coordinate of the right-most edge is equal to the width of the rendering
  context's <span>output bitmap</span> in <span data-x="'px'">CSS pixels</span>; similarly, the
  <var>y</var>-coordinate of the bottom-most edge is equal to the height of the rendering context's
  <span>output bitmap</span> in <span data-x="'px'">CSS pixels</span>.</p>

  <p>The size of the coordinate space does not necessarily represent the size of the actual bitmaps
  that the user agent will use internally or during rendering. On high-definition displays, for
  instance, the user agent may internally use bitmaps with four device pixels per unit in the
  coordinate space, so that the rendering remains at high quality throughout. Anti-aliasing can
  similarly be implemented using oversampling with bitmaps of a higher resolution than the final
  image on the display.</p>

  <div class="example">
   <p>Using <span data-x="'px'">CSS pixels</span> to describe the size of a rendering context's
   <span>output bitmap</span> does not mean that when rendered the canvas will cover an equivalent
   area in <span data-x="'px'">CSS pixels</span>. <span data-x="'px'">CSS pixels</span> are reused
   for ease of integration with CSS features, such as text layout.</p>

   <p>In other words, the <code>canvas</code> element below's rendering context has a 200x200
   <span>output bitmap</span> (which internally uses <span data-x="'px'">CSS pixels</span> as a
   unit for ease of integration with CSS) and is rendered as 100x100 <span data-x="'px'">CSS
   pixels</span>:

   <pre><code class="html">&lt;canvas width=200 height=200 style=width:100px;height:100px></code></pre>
  </div>

  <hr>

  <p>The <dfn>2D context creation algorithm</dfn>, which is passed a <var>target</var> (a
  <code>canvas</code> element) and <var>options</var>, consists of running these steps:</p>

  <ol>
   <li><p>Let <var>settings</var> be the result of <span
   data-x="concept-idl-convert">converting</span> <var>options</var> to the dictionary type
   <code>CanvasRenderingContext2DSettings</code>. (This can throw an exception.).</p></li>

   <li><p>Let <var>context</var> be a new <code>CanvasRenderingContext2D</code> object.</p></li>

   <li><p>Initialize <var>context</var>'s <code data-x="dom-context-2d-canvas">canvas</code>
   attribute to point to <var>target</var>.</p></li>

   <li><p>Set <var>context</var>'s <span>output bitmap</span> to the same bitmap as
   <var>target</var>'s bitmap (so that they are shared).</p></li>

   <li><p><span data-x="concept-canvas-set-bitmap-dimensions">Set bitmap dimensions</span> to
   <span data-x="obtain-numeric-values">the numeric values</span> of <var>target</var>'s <code
   data-x="attr-canvas-width">width</code> and <code data-x="attr-canvas-height">height</code>
   content attributes.</p></li>

   <li><p>Run the <span data-x="canvas-setting-init-bitmap">canvas settings output bitmap
   initialization algorithm</span>, given <var>context</var> and <var>settings</var>.</li>

   <li><p>Return <var>context</var>.</p></li>
  </ol>

  <hr>

  <p>When the user agent is to <dfn data-x="concept-canvas-set-bitmap-dimensions">set bitmap
  dimensions</dfn> to <var>width</var> and <var>height</var>, it must run these steps:</p>

  <ol>
   <li><p><span>Reset the rendering context to its default state</span>.</p></li>

   <li><p>Resize the <span>output bitmap</span> to the new <var>width</var> and
   <var>height</var>.</p></li>

   <li><p>Let <var>canvas</var> be the <code>canvas</code> element to which the rendering
   context's <code data-x="dom-context-2d-canvas">canvas</code> attribute was initialized.</p></li>

   <li><p>If <span data-x="obtain-numeric-values">the numeric value</span> of
   <var>canvas</var>'s <code data-x="attr-canvas-width">width</code> content
   attribute differs from <var>width</var>, then set <var>canvas</var>'s <code
   data-x="attr-canvas-width">width</code> content attribute to the shortest possible string
   representing <var>width</var> as a <span>valid non-negative integer</span>.</p></li>

   <li><p>If <span data-x="obtain-numeric-values">the numeric value</span> of
   <var>canvas</var>'s <code data-x="attr-canvas-height">height</code> content
   attribute differs from <var>height</var>, then set <var>canvas</var>'s <code
   data-x="attr-canvas-height">height</code> content attribute to the shortest possible string
   representing <var>height</var> as a <span>valid non-negative integer</span>.</p></li>
  </ol>

  <div class="example">
   <p>Only one square appears to be drawn in the following example:</p>
   <pre><code class="js">// canvas is a reference to a &lt;canvas> element
var context = canvas.getContext('2d');
context.fillRect(0,0,50,50);
canvas.setAttribute('width', '300'); // clears the canvas
context.fillRect(0,100,50,50);
canvas.width = canvas.width; // clears the canvas
context.fillRect(100,0,50,50); // only this square remains</code></pre>
  </div>

  <hr>

  <p>The <dfn attribute for="CanvasRenderingContext2D"><code
  data-x="dom-context-2d-canvas">canvas</code></dfn> attribute must return the value it was
  initialized to when the object was created.</p>

  </div>

  <hr>

  <p>The <code>PredefinedColorSpace</code> enumeration is used to specify the <span
  data-x="concept-canvas-color-space">color space</span> of the canvas's backing store.</p>

  <p>The "<dfn enum-value for="PredefinedColorSpace"><code
  data-x="dom-PredefinedColorSpace-srgb">srgb</code></dfn>" value indicates the <span>'srgb'</span>
  color space.</p>

  <p>The "<dfn enum-value for="PredefinedColorSpace"><code
  data-x="dom-PredefinedColorSpace-display-p3">display-p3</code></dfn>" value indicates the
  <span>'display-p3'</span> color space.</p>

  <p class="note">The algorithm for converting between color spaces can be found in the
  <span>Converting Colors</span> section of <cite>CSS Color</cite>. <ref>CSSCOLOR</ref></p>

  <hr>

  <p>The <code>CanvasColorType</code> enumeration is used to specify the <span
  data-x="concept-canvas-color-type">color type</span> of the canvas's backing store.</p>

  <p>The "<dfn enum-value for="CanvasColorType"><code
  data-x="dom-CanvasColorType-unorm8">unorm8</code></dfn>" value indicates that the type
  for all color components is 8-bit unsigned normalized.</p>

  <p>The "<dfn enum-value for="CanvasColorType"><code
  data-x="dom-CanvasColorType-float16">float16</code></dfn>" value indicates that the type
  for all color components is 16-bit floating point.</p>

  <hr>

  <p>The <code>CanvasFillRule</code> enumeration is used to select the <dfn>fill rule</dfn>
  algorithm by which to determine if a point is inside or outside a path.</p>

  <p>The "<dfn enum-value for="CanvasFillRule"><code
  data-x="dom-context-2d-fillRule-nonzero">nonzero</code></dfn>" value indicates the nonzero winding
  rule, wherein

     a point is considered to be outside a shape if the number of times a half-infinite straight
     line drawn from that point crosses the shape's path going in one direction is equal to the
     number of times it crosses the path going in the other direction.

  </p>


  <p>The "<dfn enum-value for="CanvasFillRule"><code
  data-x="dom-context-2d-fillRule-evenodd">evenodd</code></dfn>" value indicates the even-odd rule,
  wherein

     a point is considered to be outside a shape if the number of times a half-infinite straight
     line drawn from that point crosses the shape's path is even.

  </p>

  <p>If a point is not outside a shape, it is inside the shape.</p>


  <hr>

  <p>The <code>ImageSmoothingQuality</code> enumeration is used to express a preference for the
  interpolation quality to use when smoothing images.</p>

  <p>The "<dfn enum-value for="ImageSmoothingQuality"><code
  data-x="dom-context-2d-imageSmoothingQuality-low">low</code></dfn>" value indicates a preference
  for a low level of image interpolation quality. Low-quality image interpolation may be more
  computationally efficient than higher settings.</p>

  <p>The "<dfn enum-value for="ImageSmoothingQuality"><code
  data-x="dom-context-2d-imageSmoothingQuality-medium">medium</code></dfn>" value indicates a
  preference for a medium level of image interpolation quality.</p>

  <p>The "<dfn enum-value for="ImageSmoothingQuality"><code
  data-x="dom-context-2d-imageSmoothingQuality-high">high</code></dfn>" value indicates a preference
  for a high level of image interpolation quality. High-quality image interpolation may be more
  computationally expensive than lower settings.</p>

  <p class="note">Bilinear scaling is an example of a relatively fast, lower-quality image-smoothing
  algorithm. Bicubic or Lanczos scaling are examples of image-smoothing algorithms that produce
  higher-quality output. This specification does not mandate that specific interpolation algorithms
  be used.</p>


  <h6>Implementation notes</h6>

  <!-- NON-NORMATIVE SECTION -->

  <p>The <span>output bitmap</span>, when it is not directly displayed by the user agent,
  implementations can, instead of updating this bitmap, merely remember the sequence of drawing
  operations that have been applied to it until such time as the bitmap's actual data is needed
  (for example because of a call to <code data-x="dom-context-2d-drawImage">drawImage()</code>, or
  the <code data-x="dom-createImageBitmap">createImageBitmap()</code> factory method). In many
  cases, this will be more memory efficient.</p>

  <p>The bitmap of a <code>canvas</code> element is the one bitmap that's pretty much always going
  to be needed in practice. The <span>output bitmap</span> of a rendering context, when it has one,
  is always just an alias to a <code>canvas</code> element's bitmap.</p>

  <p>Additional bitmaps are sometimes needed, e.g. to enable fast drawing when the canvas is being
  painted at a different size than its <span data-x="natural dimensions">natural size</span>,
  or to enable double buffering so that graphics updates, like page scrolling for example, can be
  processed concurrently while canvas draw commands are being executed.</p>

  <h6>The canvas settings</h6>

  <p>A <code>CanvasSettings</code> object has an <dfn>output bitmap</dfn> that is initialized when
  the object is created.</p>

  <p>The <span>output bitmap</span> has an <span
  data-x="concept-canvas-origin-clean">origin-clean</span> flag, which can be set to true or false.
  Initially, when one of these bitmaps is created, its <span
  data-x="concept-canvas-origin-clean">origin-clean</span> flag must be set to true.</p>

  <p>The <code>CanvasSettings</code> object also has an <dfn
  data-x="concept-canvas-alpha">alpha</dfn> boolean. When a <code>CanvasSettings</code> object's
  <span data-x="concept-canvas-alpha">alpha</span> is false, then its alpha component must be fixed
  to 1.0 (fully opaque) for all pixels, and attempts to change the alpha component of any pixel must
  be silently ignored.</p>

  <p class="note">Thus, the bitmap of such a context starts off as <span>opaque black</span> instead
  of <span>transparent black</span>; <code data-x="dom-context-2d-clearRect">clearRect()</code>
  always results in <span>opaque black</span> pixels, every fourth byte from <code
  data-x="dom-context-2d-getImageData">getImageData()</code> is always 255, the <code
  data-x="dom-context-2d-putImageData">putImageData()</code> method effectively ignores every fourth
  byte in its input, and so on. However, the alpha component of styles and images drawn onto the
  canvas are still honoured up to the point where they would impact the <span>output bitmap</span>'s
  alpha component; for instance, drawing a 50% transparent white square on a freshly created
  <span>output bitmap</span> with its <span data-x="concept-canvas-alpha">alpha</span> set to false
  will result in a fully-opaque gray square.</p>

  <p>The <code>CanvasSettings</code> object also has a <dfn
  data-x="concept-canvas-desynchronized">desynchronized</dfn> boolean. When a
  <code>CanvasSettings</code> object's <span
  data-x="concept-canvas-desynchronized">desynchronized</span> is true, then the user agent may
  optimize the rendering of the canvas to reduce the latency, as measured from input events to
  rasterization, by desynchronizing the canvas paint cycle from the event loop, bypassing the
  ordinary user agent rendering algorithm, or both. Insofar as this mode involves bypassing the
  usual paint mechanisms, rasterization, or both, it might introduce visible tearing artifacts.</p>

  <p class="note">The user agent usually renders on a buffer which is not being displayed, quickly
  swapping it and the one being scanned out for presentation; the former buffer is called back
  buffer and the latter <i>front buffer</i>. A popular technique for reducing latency is called
  front buffer rendering, also known as <i>single buffer</i> rendering, where rendering happens in
  parallel and racily with the scanning out process. This technique reduces the latency at the price
  of potentially introducing tearing artifacts and can be used to implement in total or part of the
  <span data-x="concept-canvas-desynchronized">desynchronized</span> boolean.
  <ref>MULTIPLEBUFFERING</ref></p>

  <p class="note">The <span data-x="concept-canvas-desynchronized">desynchronized</span> boolean can
  be useful when implementing certain kinds of applications, such as drawing applications, where the
  latency between input and rasterization is critical.</p>

  <p>The <code>CanvasSettings</code> object also has a <dfn
  data-x="concept-canvas-will-read-frequently">will read frequently</dfn> boolean. When a
  <code>CanvasSettings</code> object's <span data-x="concept-canvas-will-read-frequently">will read
  frequently</span> is true, the user agent may optimize the canvas for readback operations.</p>

  <p class="note">On most devices the user agent needs to decide whether to store the canvas's
  <span>output bitmap</span> on the GPU (this is also called "hardware accelerated"), or on the CPU
  (also called "software"). Most rendering operations are more performant for accelerated canvases,
  with the major exception being readback with <code
  data-x="dom-context-2d-getImageData">getImageData()</code>, <code
  data-x="dom-canvas-toDataURL">toDataURL()</code>, or <code
  data-x="dom-canvas-toBlob">toBlob()</code>. <code>CanvasSettings</code> objects with <span
  data-x="concept-canvas-will-read-frequently">will read frequently</span> equal to true tell the
  user agent that the webpage is likely to perform many readback operations and that it is
  advantageous to use a software canvas.</p>

  <p>The <code>CanvasSettings</code> object also has a <dfn
  data-x="concept-canvas-color-space">color space</dfn> setting of type
  <code>PredefinedColorSpace</code>. The <code>CanvasSettings</code> object's <span
  data-x="concept-canvas-color-space">color space</span> indicates the color space for the
  <span>output bitmap</span>.</p>

  <p>The <code>CanvasSettings</code> object also has a <dfn
  data-x="concept-canvas-color-type">color type</dfn> setting of type
  <code>CanvasColorType</code>. The <code>CanvasSettings</code> object's <span
  data-x="concept-canvas-color-type">color type</span> indicates the data type of the
  color and alpha components of the pixels of the <span>output bitmap</span>.</p>

  <p>To <dfn data-x="canvas-setting-init-bitmap">initialize a <code>CanvasSettings</code> output
  bitmap</dfn>, given a <code>CanvasSettings</code> <var>context</var> and a
  <code>CanvasRenderingContext2DSettings</code> <var>settings</var>:</p>

  <ol>
   <li><p>Set <var>context</var>'s <span data-x="concept-canvas-alpha">alpha</span> to
   <var>settings</var>["<dfn dict-member for="CanvasRenderingContext2DSettings"><code
   data-x="dom-CanvasRenderingContext2DSettings-alpha">alpha</code></dfn>"].</p></li>

   <li><p>Set <var>context</var>'s <span
   data-x="concept-canvas-desynchronized">desynchronized</span> to <var>settings</var>["<dfn
   dict-member for="CanvasRenderingContext2DSettings"><code
   data-x="dom-CanvasRenderingContext2DSettings-desynchronized">desynchronized</code></dfn>"].</p></li>

   <li><p>Set <var>context</var>'s <span data-x="concept-canvas-color-space">color space</span> to
   <var>settings</var>["<dfn dict-member for="CanvasRenderingContext2DSettings"><code
   data-x="dom-CanvasRenderingContext2DSettings-colorSpace">colorSpace</code></dfn>"].</p></li>

   <li><p>Set <var>context</var>'s <span data-x="concept-canvas-color-type">color type</span> to
   <var>settings</var>["<dfn dict-member for="CanvasRenderingContext2DSettings"><code
   data-x="dom-CanvasRenderingContext2DSettings-colorType">colorType</code></dfn>"].</p></li>

   <li><p>Set <var>context</var>'s <span data-x="concept-canvas-will-read-frequently">will read
   frequently</span> to <var>settings</var>["<dfn dict-member
   for="CanvasRenderingContext2DSettings"><code
   data-x="dom-CanvasRenderingContext2DSettings-willReadFrequently">willReadFrequently</code></dfn>"].</p></li>
  </ol>

  <p>The <dfn method for="CanvasSettings"><code
  data-x="dom-context-2d-canvas-getcontextattributes">getContextAttributes()</code></dfn> method
  steps are to return «[ "<code data-x="dom-CanvasRenderingContext2DSettings-alpha">alpha</code>" →
  <span>this</span>'s <span data-x="concept-canvas-alpha">alpha</span>, "<code
  data-x="dom-CanvasRenderingContext2DSettings-desynchronized">desynchronized</code>" →
  <span>this</span>'s <span data-x="concept-canvas-desynchronized">desynchronized</span>, "<code
  data-x="dom-CanvasRenderingContext2DSettings-colorSpace">colorSpace</code>" → <span>this</span>'s
  <span data-x="concept-canvas-color-space">color space</span>, "<code
  data-x="dom-CanvasRenderingContext2DSettings-colorType">colorType</code>" → <span>this</span>'s
  <span data-x="concept-canvas-color-type">color type</span>, "<code
  data-x="dom-CanvasRenderingContext2DSettings-willReadFrequently">willReadFrequently</code>" →
  <span>this</span>'s <span data-x="concept-canvas-will-read-frequently">will read frequently</span>
  ]».</p>

  <h6>The canvas state</h6>

  <p>Objects that implement the <code>CanvasState</code> interface maintain a stack of drawing
  states. <dfn data-x="drawing state">Drawing states</dfn> consist of:</p>

  <ul>
   <li><p>The current <span data-x="dom-context-2d-transformation">transformation
   matrix</span>.</p></li>

   <li><p>The current <span>clipping region</span>.</p></li>

   <li><p>The current <span data-x="concept-CanvasTextDrawingStyles-letter-spacing">letter
   spacing</span>, <span data-x="concept-CanvasTextDrawingStyles-word-spacing">word spacing</span>,
   <span data-x="concept-CanvasFillStrokeStyles-fill-style">fill style</span>, <span
   data-x="concept-CanvasFillStrokeStyles-stroke-style">stroke style</span>, <span
   data-x="concept-canvas-current-filter">filter</span>, <span
   data-x="concept-canvas-global-alpha">global alpha</span>, <span data-x="current compositing and
   blending operator">compositing and blending operator</span>, and <span
   data-x="concept-CanvasShadowStyles-shadow-color">shadow color</span>.</p></li>

   <li><p>The current values of the following attributes: <code
   data-x="dom-context-2d-lineWidth">lineWidth</code>, <code
   data-x="dom-context-2d-lineCap">lineCap</code>, <code
   data-x="dom-context-2d-lineJoin">lineJoin</code>, <code
   data-x="dom-context-2d-miterLimit">miterLimit</code>, <code
   data-x="dom-context-2d-lineDashOffset">lineDashOffset</code>, <code
   data-x="dom-context-2d-shadowOffsetX">shadowOffsetX</code>, <code
   data-x="dom-context-2d-shadowOffsetY">shadowOffsetY</code>, <code
   data-x="dom-context-2d-shadowBlur">shadowBlur</code>, <code
   data-x="dom-context-2d-lang">lang</code>, <code
   data-x="dom-context-2d-font">font</code>, <code
   data-x="dom-context-2d-textAlign">textAlign</code>, <code
   data-x="dom-context-2d-textBaseline">textBaseline</code>, <code
   data-x="dom-context-2d-direction">direction</code>, <code
   data-x="dom-context-2d-fontKerning">fontKerning</code>, <code
   data-x="dom-context-2d-fontStretch">fontStretch</code>, <code
   data-x="dom-context-2d-fontVariantCaps">fontVariantCaps</code>, <code
   data-x="dom-context-2d-textRendering">textRendering</code>, <code
   data-x="dom-context-2d-imageSmoothingEnabled">imageSmoothingEnabled</code>, <code
   data-x="dom-context-2d-imageSmoothingQuality">imageSmoothingQuality</code>.</p></li>

   <li><p>The current <span>dash list</span>.</p></li>
  </ul>

  <p class="note">The rendering context's bitmaps are not part of the drawing state, as they
  depend on whether and how the rendering context is bound to a <code>canvas</code> element.</p>

  <p>Objects that implement the <code>CanvasState</code> mixin have a <dfn
  data-x="concept-canvas-context-lost">context lost</dfn> boolean, that is initialized to false
  when the object is created. The <span data-x="concept-canvas-context-lost">context lost</span>
  value is updated in the <span>context lost steps</span>.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-save">save</span>()</code></dt>
   <dd><p>Pushes the current state onto the stack.</p></dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-restore">restore</span>()</code></dt>
   <dd><p>Pops the top state on the stack, restoring the context to that state.</p></dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-reset">reset</span>()</code></dt>
   <dd><p>Resets the rendering context, which includes the backing buffer, the drawing state stack,
   path, and styles.</p></dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-isContextLost">isContextLost</span>()</code></dt>
   <dd><p>Returns true if the rendering context was lost. Context loss can occur due to driver
   crashes, running out of memory, etc. In these cases, the canvas loses its backing storage and
   takes steps to <span>reset the rendering context to its default state</span>.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="CanvasState"><code data-x="dom-context-2d-save">save()</code></dfn> method
  steps are to push a copy of the current drawing state onto the drawing state stack.</p>

  <p>The <dfn method for="CanvasState"><code data-x="dom-context-2d-restore">restore()</code></dfn>
  method steps are to pop the top entry in the drawing state stack, and reset the drawing state it
  describes. If there is no saved state, then the method must do nothing.</p>

  <p>The <dfn method for="CanvasState"><code data-x="dom-context-2d-reset">reset()</code></dfn>
  method steps are to <span>reset the rendering context to its default state</span>.</p>

  <p>To <dfn>reset the rendering context to its default state</dfn>:</p>

  <ol>
   <li><p>Clear canvas's bitmap to <span>transparent black</span>.</p></li>

   <li><p>Empty the list of subpaths in context's <span>current default path</span>.</p></li>

   <li><p>Clear the context's drawing state stack.</p></li>

   <li><p>Reset everything that <span>drawing state</span> consists of to their initial
   values.</p></li>
  </ol>

  <p>The <dfn method for="CanvasState"><code
  data-x="dom-context-2d-isContextLost">isContextLost()</code></dfn> method steps are to return
  <span>this</span>'s <span data-x="concept-canvas-context-lost">context lost</span>.</p>

  </div>

  <h6>Line styles</h6>

  <dl class="domintro">
   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-lineWidth">lineWidth</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>styles</var>.<span data-x="dom-context-2d-lineWidth">lineWidth</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current line width.</p>

    <p>Can be set, to change the line width. Values that are not finite values greater than zero are
    ignored.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-lineCap">lineCap</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>styles</var>.<span data-x="dom-context-2d-lineCap">lineCap</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current line cap style.</p>

    <p>Can be set, to change the line cap style.</p>

    <p>The possible line cap styles are "<code data-x="">butt</code>", "<code
    data-x="">round</code>", and "<code data-x="">square</code>". Other values are ignored.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-lineJoin">lineJoin</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>styles</var>.<span data-x="dom-context-2d-lineJoin">lineJoin</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current line join style.</p>

    <p>Can be set, to change the line join style.</p>

    <p>The possible line join styles are "<code data-x="">bevel</code>", "<code
    data-x="">round</code>", and "<code data-x="">miter</code>". Other values are ignored.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-miterLimit">miterLimit</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>styles</var>.<span data-x="dom-context-2d-miterLimit">miterLimit</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current miter limit ratio.</p>

    <p>Can be set, to change the miter limit ratio. Values that are not finite values greater than
    zero are ignored.</p>
   </dd>


   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-setLineDash">setLineDash</span>(<var>segments</var>)</code></dt>
   <dt><code data-x=""><var>styles</var>.<span data-x="dom-context-2d-setLineDash">setLineDash</span>(<var>segments</var>)</code></dt>

   <dd>
    <p>Sets the current line dash pattern (as used when stroking). The argument is a list of
    distances for which to alternately have the line on and the line off.</p>
   </dd>


   <dt><code data-x=""><var>segments</var> = <var>context</var>.<span subdfn data-x="dom-context-2d-getLineDash">getLineDash</span>()</code></dt>
   <dt><code data-x=""><var>segments</var> = <var>styles</var>.<span data-x="dom-context-2d-getLineDash">getLineDash</span>()</code></dt>

   <dd>
    <p>Returns a copy of the current line dash pattern. The array returned will always have an even
    number of entries (i.e. the pattern is normalized).</p>
   </dd>


   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-lineDashOffset">lineDashOffset</span></code></dt>
   <dt><code data-x=""><var>styles</var>.<span data-x="dom-context-2d-lineDashOffset">lineDashOffset</span></code></dt>

   <dd>
    <p>Returns the phase offset (in the same units as the line dash pattern).</p>

    <p>Can be set, to change the phase offset. Values that are not finite values are ignored.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>Objects that implement the <code>CanvasPathDrawingStyles</code> interface have attributes and
  methods (defined in this section) that control how lines are treated by the object.</p>

  <p>The <dfn attribute for="CanvasPathDrawingStyles"><code
  data-x="dom-context-2d-lineWidth">lineWidth</code></dfn> attribute gives the width of lines, in
  coordinate space units. On getting, it must return the current value. On setting, zero, negative,
  infinite, and NaN values must be ignored, leaving the value unchanged; other values must change
  the current value to the new value.</p>

  <p>When the object implementing the <code>CanvasPathDrawingStyles</code> interface is created, the
  <code data-x="dom-context-2d-lineWidth">lineWidth</code> attribute must initially have the value
  1.0.</p>

  <hr>

  <p>The <dfn attribute for="CanvasPathDrawingStyles"><code
  data-x="dom-context-2d-lineCap">lineCap</code></dfn> attribute defines the type of endings that
  UAs will place on the end of lines. The three valid values are "<code data-x="">butt</code>",
  "<code data-x="">round</code>", and "<code data-x="">square</code>".</p>

  <p>On getting, it must return the current value. On setting, the current value must be changed
  to the new value.</p>

  <p>When the object implementing the <code>CanvasPathDrawingStyles</code> interface is created, the
  <code data-x="dom-context-2d-lineCap">lineCap</code> attribute must initially have the value
  "<code data-x="">butt</code>".</p>

  <hr>

  <p>The <dfn attribute for="CanvasPathDrawingStyles"><code
  data-x="dom-context-2d-lineJoin">lineJoin</code></dfn> attribute defines the type of corners that
  UAs will place where two lines meet. The three valid values are "<code data-x="">bevel</code>",
  "<code data-x="">round</code>", and "<code data-x="">miter</code>".</p>

  <p>On getting, it must return the current value. On setting, the current value must be changed
  to the new value.</p>

  <p>When the object implementing the <code>CanvasPathDrawingStyles</code> interface is created, the
  <code data-x="dom-context-2d-lineJoin">lineJoin</code> attribute must initially have the value
  "<code data-x="">miter</code>".</p>

  <hr>

  <p>When the <code data-x="dom-context-2d-lineJoin">lineJoin</code> attribute has the value "<code
  data-x="">miter</code>", strokes use the miter limit ratio to decide how to render joins. The
  miter limit ratio can be explicitly set using the <dfn attribute
  for="CanvasPathDrawingStyles"><code data-x="dom-context-2d-miterLimit">miterLimit</code></dfn>
  attribute. On getting, it must return the current value. On setting, zero, negative, infinite, and
  NaN values must be ignored, leaving the value unchanged; other values must change the current
  value to the new value.</p>
  <!-- values less than 1.0 are equivalent to 1.0, since it's compared against a ratio that can't be
       less than 1.0 -->

  <p>When the object implementing the <code>CanvasPathDrawingStyles</code> interface is created, the
  <code data-x="dom-context-2d-miterLimit">miterLimit</code> attribute must initially have the value
  10.0.</p>

  <hr>

  <p>Each <code>CanvasPathDrawingStyles</code> object has a <dfn>dash list</dfn>, which is either
  empty or consists of an even number of non-negative numbers. Initially, the <span>dash list</span>
  must be empty.</p>

  <p>The <dfn method for="CanvasPathDrawingStyles"><code
  data-x="dom-context-2d-setLineDash">setLineDash(<var>segments</var>)</code></dfn> method, when invoked, must run
  these steps:</p>

  <ol>
   <li><p>If any value in <var>segments</var> is not finite (e.g. an Infinity or a NaN value), or
   if any value is negative (less than zero), then return (without throwing an exception;
   user agents could show a message on a developer console, though, as that would be helpful for
   debugging).</p></li>

   <li><p>If the number of elements in <var>segments</var> is odd, then let <var>segments</var>
   be the concatenation of two copies of <var>segments</var>.</p></li>

   <li><p>Let the object's <span>dash list</span> be <var>segments</var>.</p></li>
  </ol>

  <p>When the <dfn method for="CanvasPathDrawingStyles"><code
  data-x="dom-context-2d-getLineDash">getLineDash()</code></dfn> method is invoked, it must return a
  sequence whose values are the values of the object's <span>dash list</span>, in the same
  order.</p>

  <p>It is sometimes useful to change the "phase" of the dash pattern, e.g. to achieve a "marching
  ants" effect. The phase can be set using the <dfn attribute for="CanvasPathDrawingStyles"><code
  data-x="dom-context-2d-lineDashOffset">lineDashOffset</code></dfn> attribute. On getting, it must
  return the current value. On setting, infinite and NaN values must be ignored, leaving the value
  unchanged; other values must change the current value to the new value.</p>

  <p>When the object implementing the <code>CanvasPathDrawingStyles</code> interface is created, the
  <code data-x="dom-context-2d-lineDashOffset">lineDashOffset</code> attribute must initially have
  the value 0.0.</p>

  <hr>

  <p>When a user agent is to <dfn>trace a path</dfn>, given an object <var>style</var>
  that implements the <code>CanvasPathDrawingStyles</code> interface, it must run the following
  algorithm. This algorithm returns a new <span data-x="concept-path">path</span>.</p>

  <ol>
   <li><p>Let <var>path</var> be a copy of the path being traced.</p></li>

   <!-- PATH SANITIZATION -->

   <li><p>Prune all zero-length <span>line segments</span> from <var>path</var>.</p></li>

   <li><p>Remove from <var>path</var> any subpaths containing no lines (i.e. subpaths with
   just one point).</p></li>

   <li><p>Replace each point in each subpath of <var>path</var> other than the first point
   and the last point of each subpath by a <i>join</i> that joins the line leading to that point to
   the line leading out of that point, such that the subpaths all consist of two points (a starting
   point with a line leading out of it, and an ending point with a line leading into it), one or
   more lines (connecting the points and the joins), and zero or more joins (each connecting one
   line to another), connected together such that each subpath is a series of one or more lines with
   a join between each one and a point on each end.</p></li>

   <li><p>Add a straight closing line to each closed subpath in <var>path</var> connecting
   the last point and the first point of that subpath; change the last point to a join (from the
   previously last line to the newly added closing line), and change the first point to a join (from
   the newly added closing line to the first line).</p>

   <li><p>If <var>style</var>'s <span>dash list</span> is empty, then jump to the step
   labeled <i>convert</i>.</p></li>

   <!-- DASHING -->

   <li><p>Let <var>pattern width</var> be the concatenation of all the entries of
   <var>style</var>'s <span>dash list</span>, in coordinate space units.</p>

   <li><p>For each subpath <var>subpath</var> in <var>path</var>, run the
   following substeps. These substeps mutate the subpaths in <var>path</var> <i>in
   vivo</i>.</p>

    <ol>
     <li><p>Let <var>subpath width</var> be the length of all the lines of <var>subpath</var>, in coordinate space units.</p>

     <li><p>Let <var>offset</var> be the value of <var>style</var>'s <code
     data-x="dom-context-2d-lineDashOffset">lineDashOffset</code>, in coordinate space
     units.</p></li>

     <li>
      <p>While <var>offset</var> is greater than <var>pattern width</var>,
      decrement it by <var>pattern width</var>.</p>

      <p>While <var>offset</var> is less than zero, increment it by <var>pattern
      width</var>.</p>
     </li>

     <li><p>Define <var>L</var> to be a linear coordinate line defined along all lines in
     <var>subpath</var>, such that the start of the first line in the subpath is defined
     as coordinate 0, and the end of the last line in the subpath is defined as coordinate <var>subpath width</var>.</p></li>

     <li><p>Let <var>position</var> be zero minus <var>offset</var>.</p></li>

     <li><p>Let <var>index</var> be 0.</p></li>

     <li><p>Let <var>current state</var> be <i>off</i> (the other states being <i>on</i>
     and <i>zero-on</i>).</p></li>

     <li><p><i>Dash on</i>: Let <var>segment length</var> be
     the value of <var>style</var>'s <span>dash
     list</span>'s <var>index</var>th entry.</p></li>

     <li><p>Increment <var>position</var> by <var>segment length</var>.</p></li>

     <li><p>If <var>position</var> is greater than <var>subpath width</var>,
     then end these substeps for this subpath and start them again for the next subpath; if there
     are no more subpaths, then jump to the step labeled <i>convert</i> instead.</p></li>

     <li><p>If <var>segment length</var> is nonzero, then let <var>current state</var> be
     <i>on</i>.</p></li>

     <li><p>Increment <var>index</var> by one.</p></li>

     <li><p><i>Dash off<!-- labeled for consistency only --></i>: Let <var>segment
     length</var> be the value of <var>style</var>'s <span>dash list</span>'s <var>index</var>th entry.</p></li>

     <li><p>Let <var>start</var> be the offset <var>position</var> on <var>L</var>.</p></li>

     <li><p>Increment <var>position</var> by <var>segment length</var>.</p></li>

     <li><p>If <var>position</var> is less than zero, then jump to the step labeled
     <i>post-cut</i>.</p></li> <!-- start and end both before zero -->

     <li><p>If <var>start</var> is less than zero, then let <var>start</var> be
     zero.</p></li> <!-- end will be (at or) after zero -->

     <li><p>If <var>position</var> is greater than <var>subpath width</var>,
     then let <var>end</var> be the offset <var>subpath width</var> on <var>L</var>. Otherwise, let <var>end</var> be the offset <var>position</var> on <var>L</var>.</p></li>

     <li>
      <p>Jump to the first appropriate step:</p>

      <dl class="switch">

       <dt>If <var>segment length</var> is zero and <var>current state</var> is
       <i>off</i></dt>

       <dd>
        <p>Do nothing, just continue to the next step.</p>
       </dd>


       <dt>If <var>current state</var> is <i>off</i></dt>

       <dd>
        <p>Cut the line on which <var>end</var> finds itself short at <var>end</var> and place a point there, cutting in two the subpath that it was in;
        remove all line segments, joins, points, and subpaths that are between <var>start</var> and <var>end</var>; and finally place a single point at <var>start</var> with no lines connecting to it.</p>

        <p>The point has a <i>directionality</i> for the purposes of drawing line caps (see below).
        The directionality is the direction that the original line had at that point (i.e. when <var>L</var> was defined above).</p>
       </dd>


       <dt>Otherwise</dt>

       <dd>
        <p>Cut the line on which <var>start</var> finds itself into two at <var>start</var> and place a point there, cutting in two the subpath that it was in, and
        similarly cut the line on which <var>end</var> finds itself short at <var>end</var> and place a point there, cutting in two the subpath that <em>it</em> was in,
        and then remove all line segments, joins, points, and subpaths that are between <var>start</var> and <var>end</var>.</p>

        <p>If <var>start</var> and <var>end</var> are the same point, then this
        results in just the line being cut in two and two points being inserted there, with nothing
        being removed, unless a join also happens to be at that point, in which case the join must
        be removed.</p>
       </dd>
      </dl>
     </li>

     <li><p><i>Post-cut</i>: If <var>position</var> is greater than <var>subpath width</var>, then jump to the step labeled <i>convert</i>.</p></li>

     <li><p>Increment <var>index</var> by one. If it is equal to the number of entries in
     <var>style</var>'s <span>dash list</span>, then let <var>index</var> be
     0.</p></li>

     <li><p>Return to the step labeled <i>dash on</i>.</p></li>
    </ol>
   </li>

   <!-- STROKING -->

   <li>
    <p><i>Convert</i>: This is the step that converts the path to a new path that represents its
    stroke.</p>

    <p>Create a new <span data-x="concept-path">path</span> that describes the edge of the areas
    that would be covered if a straight line of length equal to <var>style</var>'s
    <code data-x="dom-context-2d-lineWidth">lineWidth</code> was swept along each subpath in <var>path</var> while being kept at an angle such that the line is orthogonal to the path
    being swept, replacing each point with the end cap necessary to satisfy <var>style</var>'s <code data-x="dom-context-2d-lineCap">lineCap</code> attribute as
    described previously and elaborated below, and replacing each join with the join necessary to
    satisfy <var>style</var>'s <code data-x="dom-context-2d-lineJoin">lineJoin</code>
    type, as defined below.</p>

    <p><strong>Caps</strong>: Each point has a flat edge perpendicular to the direction of the line
    coming out of it. This is then augmented according to the value of <var>style</var>'s <code data-x="dom-context-2d-lineCap">lineCap</code>. The "<code
    data-x="">butt</code>" value means that no additional line cap is added. The "<code
    data-x="">round</code>" value means that a semi-circle with the diameter equal to
    <var>style</var>'s <code data-x="dom-context-2d-lineWidth">lineWidth</code> width must
    additionally be placed on to the line coming out of each point. The "<code
    data-x="">square</code>" value means that a rectangle with the length of <var>style</var>'s <code data-x="dom-context-2d-lineWidth">lineWidth</code> width and the
    width of half <var>style</var>'s <code
    data-x="dom-context-2d-lineWidth">lineWidth</code> width, placed flat against the edge
    perpendicular to the direction of the line coming out of the point, must be added at each
    point.</p>

    <p>Points with no lines coming out of them must have two caps placed back-to-back as if it was
    really two points connected to each other by an infinitesimally short straight line in the
    direction of the point's <i>directionality</i> (as defined above).</p>

    <p><strong>Joins</strong>: In addition to the point where a join occurs, two additional points
    are relevant to each join, one for each line: the two corners found half the line width away
    from the join point, one perpendicular to each line, each on the side furthest from the other
    line.</p>

    <p>A triangle connecting these two opposite corners with a straight line, with the third point
    of the triangle being the join point, must be added at all joins. The <code
    data-x="dom-context-2d-lineJoin">lineJoin</code> attribute controls whether anything else is
    rendered. The three aforementioned values have the following meanings:</p>

    <p>The "<code data-x="">bevel</code>" value means that this is all that is rendered at
    joins.</p>

    <p>The "<code data-x="">round</code>" value means that an arc connecting the two aforementioned
    corners of the join, abutting (and not overlapping) the aforementioned triangle, with the
    diameter equal to the line width and the origin at the point of the join, must be added at
    joins.</p>

    <p>The "<code data-x="">miter</code>" value means that a second triangle must (if it can given
    the miter length) be added at the join, with one line being the line between the two
    aforementioned corners, abutting the first triangle, and the other two being continuations of
    the outside edges of the two joining lines, as long as required to intersect without going over
    the miter length.</p>

    <p>The miter length is the distance from the point where the join occurs to the intersection of
    the line edges on the outside of the join. The miter limit ratio is the maximum allowed ratio of
    the miter length to half the line width. If the miter length would cause the miter limit ratio
    (as set by <var>style</var>'s <code
    data-x="dom-context-2d-miterLimit">miterLimit</code> attribute) to be exceeded, then this second
    triangle must not be added.</p> <!-- the actual ratio can't possibly be less than 1.0 -->

<!-- here's a nice demo of the miter:
  c.clearRect(0, 0, 640, 480);
  c.beginPath();

  c.moveTo((new Date()/100) % 600, 100);
  c.lineTo(300, 300);
  c.lineTo(500, 300);

  c.lineWidth = 50;
  c.miterLimit = 10000;
  c.strokeStyle = 'silver'
  c.stroke();

  c.miterLimit = 0.0001;
  c.strokeStyle = 'black'
  c.stroke();

  c.lineWidth = 1;
  c.strokeStyle = 'red'
  c.stroke();
-->

    <p>The subpaths in the newly created path must be oriented such that for any point, the number
    of times a half-infinite straight line drawn from that point crosses a subpath is even if and
    only if the number of times a half-infinite straight line drawn from that same point crosses a
    subpath going in one direction is equal to the number of times it crosses a subpath going in the
    other direction.</p>
   </li>

   <li><p>Return the newly created path.</p></li>
  </ol>

<!--v6: Another request has been for hairline width lines, that remain hairline width with
transform. ack Shaun Morris. -->

  </div>


  <h6>Text styles</h6>

  <dl class="domintro">
   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-lang">lang</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>styles</var>.<span data-x="dom-context-2d-lang">lang</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the current language setting.</p>

    <p>Can be set to a BCP 47 language tag, the empty string, or "<code
    data-x="dom-context-2d-lang-inherit">inherit</code>", to change the language used when
    resolving fonts. "<code data-x="dom-context-2d-lang-inherit">inherit</code>" takes the language
    from the <code>canvas</code> element's language, or the associated <span>document
    element</span> when there is no <code>canvas</code> element.</p>

    <p>The default is "<code data-x="dom-context-2d-lang-inherit">inherit</code>".</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-font">font</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>styles</var>.<span data-x="dom-context-2d-font">font</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the current font settings.</p>

    <p>Can be set, to change the font. The syntax is the same as for the CSS <span>'font'</span>
    property; values that cannot be parsed as CSS font values are ignored. The default is "10px
    sans-serif".</p>

    <p>Relative keywords and lengths are computed relative to the font of the <code>canvas</code>
    element.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-textAlign">textAlign</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>styles</var>.<span data-x="dom-context-2d-textAlign">textAlign</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the current text alignment settings.</p>

    <p>Can be set, to change the alignment. The possible values are and their meanings are given
    below. Other values are ignored. The default is "<code data-x="">start</code>".</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-textBaseline">textBaseline</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>styles</var>.<span data-x="dom-context-2d-textBaseline">textBaseline</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the current baseline alignment settings.</p>

    <p>Can be set, to change the baseline alignment. The possible values and their meanings are
    given below. Other values are ignored. The default is "<code
    data-x="dom-context-2d-textBaseline-alphabetic">alphabetic</code>".</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-direction">direction</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>styles</var>.<span data-x="dom-context-2d-direction">direction</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the current directionality.</p>

    <p>Can be set, to change the directionality. The possible values and their meanings are given
    below. Other values are ignored. The default is "<code
    data-x="dom-context-2d-direction-inherit">inherit</code>".</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-letterSpacing">letterSpacing</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>styles</var>.<span data-x="dom-context-2d-letterSpacing">letterSpacing</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the current spacing between characters in the text.</p>

    <p>Can be set, to change spacing between characters. Values that cannot be parsed as a CSS
    <span>&lt;length></span> are ignored. The default is "<code data-x="">0px</code>".</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-fontKerning">fontKerning</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>styles</var>.<span data-x="dom-context-2d-fontKerning">fontKerning</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the current font kerning settings.</p>

    <p>Can be set, to change the font kerning. The possible values and their meanings are given
    below. Other values are ignored. The default is "<code
    data-x="dom-context-2d-fontKerning-auto">auto</code>".</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-fontStretch">fontStretch</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>styles</var>.<span data-x="dom-context-2d-fontStretch">fontStretch</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the current font stretch settings.</p>

    <p>Can be set, to change the font stretch. The possible values and their meanings are given
    below. Other values are ignored. The default is "<code
    data-x="dom-context-2d-fontStretch-normal">normal</code>".</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-fontVariantCaps">fontVariantCaps</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>styles</var>.<span data-x="dom-context-2d-fontVariantCaps">fontVariantCaps</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the current font variant caps settings.</p>

    <p>Can be set, to change the font variant caps. The possible values and their meanings are given
    below. Other values are ignored. The default is "<code
    data-x="dom-context-2d-fontVariantCaps-normal">normal</code>".</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-textRendering">textRendering</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>styles</var>.<span data-x="dom-context-2d-textRendering">textRendering</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the current text rendering settings.</p>

    <p>Can be set, to change the text rendering. The possible values and their meanings are given
    below. Other values are ignored. The default is "<code
    data-x="dom-context-2d-textRendering-auto">auto</code>".</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-wordSpacing">wordSpacing</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>styles</var>.<span data-x="dom-context-2d-wordSpacing">wordSpacing</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the current spacing between words in the text.</p>

    <p>Can be set, to change spacing between words. Values that cannot be parsed as a CSS
    <span>&lt;length></span> are ignored. The default is "<code data-x="">0px</code>".</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>Objects that implement the <code>CanvasTextDrawingStyles</code> interface have attributes
  (defined in this section) that control how text is laid out (rasterized or outlined) by the
  object. Such objects can also have a <dfn>font style source object</dfn>. For
  <code>CanvasRenderingContext2D</code> objects, this is the <code>canvas</code> element given by
  the value of the context's <code data-x="dom-context-2d-canvas">canvas</code> attribute. For
  <code>OffscreenCanvasRenderingContext2D</code> objects, this is the <span>associated
  <code>OffscreenCanvas</code> object</span>.</p>

  <p>Font resolution for the <span>font style source object</span> requires a <span>font
  source</span>. This is determined for a given <var>object</var> implementing
  <code>CanvasTextDrawingStyles</code> by the following steps: <ref>CSSFONTLOAD</ref></p>

  <ol>
   <li><p>If <var>object</var>'s <span>font style source object</span> is a <code>canvas</code>
   element, return the element's <span>node document</span>.</p></li>

   <li>
    <p>Otherwise, <var>object</var>'s <span>font style source object</span> is an
    <code>OffscreenCanvas</code> object:</p>

    <ol>
     <li><p>Let <var>global</var> be <var>object</var>'s <span>relevant global
     object</span>.</p></li>

     <li><p>If <var>global</var> is a <code>Window</code> object, then return <var>global</var>'s
     <span data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

     <li><p><span>Assert</span>: <var>global</var> implements
     <code>WorkerGlobalScope</code>.</p></li>

     <li><p>Return <var>global</var>.</p></li>
    </ol>
   </li>
  </ol>

  <div class="example">
   <p>This is an example of font resolution with a regular <code>canvas</code> element with ID <code
   data-x="">c1</code>.</p>

   <pre><code class="js">const font = new FontFace("MyCanvasFont", "url(mycanvasfont.ttf)");
documents.fonts.add(font);

const context = document.getElementById("c1").getContext("2d");
document.fonts.ready.then(function() {
  context.font = "64px MyCanvasFont";
  context.fillText("hello", 0, 0);
});</code></pre>

   <p>In this example, the canvas will display text using <code data-x="">mycanvasfont.ttf</code> as
   its font.</p>
  </div>

  <div class="example">
   <p>This is an example of how font resolution can happen using <code>OffscreenCanvas</code>.
   Assuming a <code>canvas</code> element with ID <code data-x="">c2</code> which is transferred to
   a worker like so:</p>

   <pre><code class="js">const offscreenCanvas = document.getElementById("c2").transferControlToOffscreen();
worker.postMessage(offscreenCanvas, [offscreenCanvas]);</code></pre>

   <p>Then, in the worker:</p>
   <pre><code class="js">self.onmessage = function(ev) {
  const transferredCanvas = ev.data;
  const context = transferredCanvas.getContext("2d");
  const font = new FontFace("MyFont", "url(myfont.ttf)");
  self.fonts.add(font);
  self.fonts.ready.then(function() {
    context.font = "64px MyFont";
    context.fillText("hello", 0, 0);
  });
};</code></pre>

   <p>In this example, the canvas will display a text using <code data-x="">myfont.ttf</code>.
   Notice that the font is only loaded inside the worker, and not in the document context.</p>
  </div>

  <p>The <dfn attribute for="CanvasTextDrawingStyles"><code
  data-x="dom-context-2d-font">font</code></dfn> IDL attribute, on setting, must be <span
  data-x="parse something according to a CSS grammar">parsed as a CSS &lt;'font'> value</span> (but
  without supporting property-independent style sheet syntax like 'inherit'), and the resulting font
  must be assigned to the context, with the <span>'line-height'</span> component forced to 'normal',
  with the <span>'font-size'</span> component converted to <span data-x="'px'">CSS pixels</span>,
  and with system fonts being computed to explicit values. If the new value is syntactically
  incorrect (including using property-independent style sheet syntax like 'inherit' or 'initial'),
  then it must be ignored, without assigning a new font value. <ref>CSS</ref></p>

  <p>Font family names must be interpreted in the context of the <span>font style source
  object</span> when the font is to be used; any fonts embedded using <code
  data-x="">@font-face</code> or loaded using <code>FontFace</code> objects that are visible to the
  <span>font style source object</span> must therefore be available once they are loaded. (Each <span>font style source
  object</span> has a <span>font source</span>, which determines what fonts are available.) If a font
  is used before it is fully loaded, or if the <span>font style source object</span> does not have
  that font in scope at the time the font is to be used, then it must be treated as if it was an
  unknown font, falling back to another as described by the relevant CSS specifications.
  <ref>CSSFONTS</ref> <ref>CSSFONTLOAD</ref></p>

  <p>On getting, the <code data-x="dom-context-2d-font">font</code> attribute must return the <span
  data-x="serializing a CSS value">serialized form</span> of the current font of the context (with
  no <span>'line-height'</span> component). <ref>CSSOM</ref></p>

  <div class="example">

   <p>For example, after the following statement:</p>

   <pre><code class="js">context.font = 'italic 400 12px/2 Unknown Font, sans-serif';</code></pre>

   <p>...the expression <code data-x="">context.font</code> would evaluate to the string "<code
   data-x="">italic&nbsp;12px&nbsp;&quot;Unknown&nbsp;Font&quot;,&nbsp;sans-serif</code>". The "400"
   font-weight doesn't appear because that is the default value. The line-height doesn't appear
   because it is forced to "normal", the default value.</p>

  </div>

  <p>When the object implementing the <code>CanvasTextDrawingStyles</code> interface is created, the
  font of the context must be set to 10px sans-serif. When the <span>'font-size'</span> component is
  set to lengths using percentages, <span>'em'</span> or <span>'ex'</span> units, or the 'larger' or
  'smaller' keywords, these must be interpreted relative to the <span>computed value</span> of the
  <span>'font-size'</span> property of the <span>font style source object</span> at the time that
  the attribute is set, if it is an element. When the <span>'font-weight'</span> component is set to
  the relative values 'bolder' and 'lighter', these must be interpreted relative to the
  <span>computed value</span> of the <span>'font-weight'</span> property of the <span>font style
  source object</span> at the time that the attribute is set, if it is an element. If the <span
  data-x="computed value">computed values</span> are undefined for a particular case (e.g. because
  the <span>font style source object</span> is not an element or is not <span>being
  rendered</span>), then the relative keywords must be interpreted relative to the normal-weight
  10px sans-serif default.</p>

  <p>The <dfn attribute for="CanvasTextDrawingStyles"><code
  data-x="dom-context-2d-textAlign">textAlign</code></dfn> IDL attribute, on getting, must return
  the current value. On setting, the current value must be changed to the new value. When the object
  implementing the <code>CanvasTextDrawingStyles</code> interface is created, the <code
  data-x="dom-context-2d-textAlign">textAlign</code> attribute must initially have the value <code
  data-x="dom-context-2d-textAlign-start">start</code>.</p>

  <p>The <dfn attribute for="CanvasTextDrawingStyles"><code
  data-x="dom-context-2d-textBaseline">textBaseline</code></dfn> IDL attribute, on getting, must
  return the current value. On setting, the current value must be changed to the new value. When the
  object implementing the <code>CanvasTextDrawingStyles</code> interface is created, the <code
  data-x="dom-context-2d-textBaseline">textBaseline</code> attribute must initially have the value
  <code data-x="dom-context-2d-textBaseline-alphabetic">alphabetic</code>.</p>

  <p>Objects that implement the <code>CanvasTextDrawingStyles</code> interface have an associated
  <dfn data-x="concept-CanvasTextDrawingStyles-language">language</dfn> value used to localize font
  rendering. Valid values are a BCP 47 language tag, the empty string, or "<dfn
  data-x="dom-context-2d-lang-inherit">inherit</dfn>" where the language comes from the
  <code>canvas</code> element's language, or the associated <span>document element</span> when there
  is no <code>canvas</code> element. Initially, the <span
  data-x="concept-CanvasTextDrawingStyles-language">language</span> must be "<code
  data-x="dom-context-2d-lang-inherit">inherit</code>".</p>

  <p>The <dfn attribute for="CanvasTextDrawingStyles" data-x="dom-context-2d-lang"><code
  data-x="dom-context-2d-lang">lang</code></dfn> getter steps are to return
  <span>this</span>'s <span data-x="concept-CanvasTextDrawingStyles-language">language</span>.</p>

  <p>The <code data-x="dom-context-2d-lang">lang</code> setter steps are to set <span>this</span>'s
  <span data-x="concept-CanvasTextDrawingStyles-language">language</span> to the given value.</p>

  <p>The <dfn attribute for="CanvasTextDrawingStyles"><code
  data-x="dom-context-2d-direction">direction</code></dfn> IDL attribute, on getting, must return
  the current value. On setting, the current value must be changed to the new value. When the object
  implementing the <code>CanvasTextDrawingStyles</code> interface is created, the <code
  data-x="dom-context-2d-direction">direction</code> attribute must initially have the value "<code
  data-x="dom-context-2d-direction-inherit">inherit</code>".</p>

  <p>Objects that implement the <code>CanvasTextDrawingStyles</code> interface have attributes that
  control the spacing between letters and words. Such objects have associated <dfn
  data-x="concept-CanvasTextDrawingStyles-letter-spacing">letter spacing</dfn> and <dfn
  data-x="concept-CanvasTextDrawingStyles-word-spacing">word spacing</dfn> values, which are CSS
  <span>&lt;length></span> values. Initially, both must be the result of <span data-x="parse
  something according to a CSS grammar">parsing</span> "<code data-x="">0px</code>" as a CSS
  <span>&lt;length></span>.</p>

  <p>The <dfn attribute for="CanvasTextDrawingStyles"
  data-x="dom-context-2d-letterSpacing"><code>letterSpacing</code></dfn> getter steps are to return
  the <span data-x="serializing a CSS value">serialized form</span> of <span>this</span>'s <span
  data-x="concept-CanvasTextDrawingStyles-letter-spacing">letter spacing</span>.</p>

  <p>The <code data-x="dom-context-2d-letterSpacing">letterSpacing</code> setter steps are:</p>

  <ol>
   <li><p>Let <var>parsed</var> be the result of <span data-x="parse something according to a CSS
   grammar">parsing</span> the given value as a CSS <span>&lt;length></span>.</p></li>

   <li><p>If <var>parsed</var> is failure, then return.</p></li>

   <li><p>Set <span>this</span>'s <span
   data-x="concept-CanvasTextDrawingStyles-letter-spacing">letter spacing</span> to
   <var>parsed</var>.</p></li>
  </ol>

  <p>The <dfn attribute for="CanvasTextDrawingStyles"
  data-x="dom-context-2d-wordSpacing"><code>wordSpacing</code></dfn> getter steps are to return the
  <span data-x="serializing a CSS value">serialized form</span> of <span>this</span>'s <span
  data-x="concept-CanvasTextDrawingStyles-word-spacing">word spacing</span>.</p>

  <p>The <code data-x="dom-context-2d-wordSpacing">wordSpacing</code> setter steps are:</p>

  <ol>
   <li><p>Let <var>parsed</var> be the result of <span data-x="parse something according to a CSS
   grammar">parsing</span> the given value as a CSS <span>&lt;length></span>.</p></li>

   <li><p>If <var>parsed</var> is failure, then return.</p></li>

   <li><p>Set <span>this</span>'s <span data-x="concept-CanvasTextDrawingStyles-word-spacing">word
   spacing</span> to <var>parsed</var>.</p></li>
  </ol>

  <p>The <dfn attribute for="CanvasTextDrawingStyles"><code
  data-x="dom-context-2d-fontKerning">fontKerning</code></dfn> IDL attribute, on
  getting, must return the current value. On setting, the current value must be changed to the new
  value. When the object implementing the <code>CanvasTextDrawingStyles</code> interface is created,
  the <code data-x="dom-context-2d-fontKerning">fontKerning</code> attribute must
  initially have the value "<code data-x="dom-context-2d-fontkerning-auto">auto</code>".</p>

  <p>The <dfn attribute for="CanvasTextDrawingStyles"><code
  data-x="dom-context-2d-fontStretch">fontStretch</code></dfn> IDL attribute, on
  getting, must return the current value. On setting, the current value must be changed to the new
  value. When the object implementing the <code>CanvasTextDrawingStyles</code> interface is created,
  the <code data-x="dom-context-2d-fontStretch">fontStretch</code> attribute must
  initially have the value "<code data-x="dom-context-2d-fontStretch-normal">normal</code>".</p>

  <p>The <dfn attribute for="CanvasTextDrawingStyles"><code
  data-x="dom-context-2d-fontVariantCaps">fontVariantCaps</code></dfn> IDL attribute, on
  getting, must return the current value. On setting, the current value must be changed to the new
  value. When the object implementing the <code>CanvasTextDrawingStyles</code> interface is created,
  the <code data-x="dom-context-2d-fontVariantCaps">fontVariantCaps</code> attribute must
  initially have the value "<code data-x="dom-context-2d-fontVariantCaps-normal">normal</code>".</p>

  <p>The <dfn attribute for="CanvasTextDrawingStyles"><code
  data-x="dom-context-2d-textRendering">textRendering</code></dfn> IDL attribute, on
  getting, must return the current value. On setting, the current value must be changed to the new
  value. When the object implementing the <code>CanvasTextDrawingStyles</code> interface is created,
  the <code data-x="dom-context-2d-textRendering">textRendering</code> attribute must
  initially have the value "<code data-x="dom-context-2d-textRendering-auto">auto</code>".</p>
  </div>

  <p>The <code data-x="dom-context-2d-textAlign">textAlign</code> attribute's allowed keywords are
  as follows:</p>

  <dl>
   <dt><dfn enum-value for="CanvasTextAlign"><code data-x="dom-context-2d-textAlign-start">start</code></dfn>
   <dd><p>Align to the start edge of the text (left side in left-to-right text, right side in
   right-to-left text).</p></dd>

   <dt><dfn enum-value for="CanvasTextAlign"><code data-x="dom-context-2d-textAlign-end">end</code></dfn>
   <dd><p>Align to the end edge of the text (right side in left-to-right text, left side in
   right-to-left text).</p></dd>

   <dt><dfn enum-value for="CanvasTextAlign"><code data-x="dom-context-2d-textAlign-left">left</code></dfn>
   <dd><p>Align to the left.</p></dd>

   <dt><dfn enum-value for="CanvasTextAlign"><code data-x="dom-context-2d-textAlign-right">right</code></dfn>
   <dd><p>Align to the right.</p></dd>

   <dt><dfn enum-value for="CanvasTextAlign"><code data-x="dom-context-2d-textAlign-center">center</code></dfn>
   <dd><p>Align to the center.</p></dd>
  </dl>

  <p>The <code data-x="dom-context-2d-textBaseline">textBaseline</code>
  attribute's allowed keywords correspond to alignment points in the
  font:</p>

  <p><img src="/images/baselines.png" width="738" height="300" alt="The em-over baseline is roughly at the top of the glyphs in a font, the hanging baseline is where some glyphs like &#x0906; are anchored, the middle is half-way between the em-over and em-under baselines, the alphabetic baseline is where characters like &#x00C1;, &#x00FF;, &#x0066;, and &#x03A9; are anchored, the ideographic-under baseline is where glyphs like &#x79C1; and &#x9054; are anchored, and the em-under baseline is roughly at the bottom of the glyphs in a font. The top and bottom of the bounding box can be far from these baselines, due to glyphs extending far outside em-over and em-under baselines."></p>

  <p>The keywords map to these alignment points as follows:</p>

  <dl>
   <dt><dfn enum-value for="CanvasTextBaseline"><code data-x="dom-context-2d-textBaseline-top">top</code></dfn>
   <dd>The <span>em-over baseline</span></dd>

   <dt><dfn enum-value for="CanvasTextBaseline"><code data-x="dom-context-2d-textBaseline-hanging">hanging</code></dfn>
   <dd>The <span>hanging baseline</span></dd>

   <dt><dfn enum-value for="CanvasTextBaseline"><code data-x="dom-context-2d-textBaseline-middle">middle</code></dfn>
   <dd>Halfway between the <span>em-over baseline</span> and the <span>em-under baseline</span></dd>

   <dt><dfn enum-value for="CanvasTextBaseline"><code data-x="dom-context-2d-textBaseline-alphabetic">alphabetic</code></dfn>
   <dd>The <span>alphabetic baseline</span></dd>

   <dt><dfn enum-value for="CanvasTextBaseline"><code data-x="dom-context-2d-textBaseline-ideographic">ideographic</code></dfn>
   <dd>The <span>ideographic-under baseline</span></dd>

   <dt><dfn enum-value for="CanvasTextBaseline"><code data-x="dom-context-2d-textBaseline-bottom">bottom</code></dfn>
   <dd>The <span>em-under baseline</span></dd>
  </dl>

  <p>The <code data-x="dom-context-2d-direction">direction</code> attribute's allowed keywords are
  as follows:</p>

  <dl>
   <dt><dfn enum-value for="CanvasDirection"><code data-x="dom-context-2d-direction-ltr">ltr</code></dfn>

   <dd><p>Treat input to the <span>text preparation algorithm</span> as left-to-right text.</p></dd>

   <dt><dfn enum-value for="CanvasDirection"><code data-x="dom-context-2d-direction-rtl">rtl</code></dfn>

   <dd><p>Treat input to the <span>text preparation algorithm</span> as right-to-left text.</p></dd>

   <dt><dfn enum-value for="CanvasDirection"><code data-x="dom-context-2d-direction-inherit">inherit</code></dfn>

   <dd><p>Use the process in the <span>text preparation algorithm</span> to obtain the text
   direction from the <code>canvas</code> element, <span
   data-x="offscreencanvas-placeholder">placeholder <code>canvas</code> element</span>, or
   <span>document element</span>.</p></dd>
  </dl>

  <p>The <code data-x="dom-context-2d-fontKerning">fontKerning</code> attribute's allowed keywords
  are as follows:</p>

  <dl>
   <dt><dfn enum-value for="CanvasFontKerning"><code data-x="dom-context-2d-fontkerning-auto">auto</code></dfn>
   <dd><p>Kerning is applied at the discretion of the user agent.</p></dd>

   <dt><dfn enum-value for="CanvasFontKerning"><code data-x="dom-context-2d-fontkerning-normal">normal</code></dfn>
   <dd><p>Kerning is applied.</p></dd>

   <dt><dfn enum-value for="CanvasFontKerning"><code data-x="dom-context-2d-fontkerning-none">none</code></dfn>
   <dd><p>Kerning is not applied.</p></dd>
  </dl>

  <p>The <code data-x="dom-context-2d-fontStretch">fontStretch</code> attribute's allowed keywords
  are as follows:</p>

  <dl>
   <dt><dfn enum-value for="CanvasFontStretch"><code data-x="dom-context-2d-fontstretch-ultra-condensed">ultra-condensed</code></dfn>
   <dd><p>Same as CSS <span>'font-stretch'</span> <span>'ultra-condensed'</span> setting.</p></dd>

   <dt><dfn enum-value for="CanvasFontStretch"><code data-x="dom-context-2d-fontstretch-extra-condensed">extra-condensed</code></dfn>
   <dd><p>Same as CSS <span>'font-stretch'</span> <span>'extra-condensed'</span> setting.</p></dd>

   <dt><dfn enum-value for="CanvasFontStretch"><code data-x="dom-context-2d-fontstretch-condensed">condensed</code></dfn>
   <dd><p>Same as CSS <span>'font-stretch'</span> <span>'condensed'</span> setting.</p></dd>

   <dt><dfn enum-value for="CanvasFontStretch"><code data-x="dom-context-2d-fontstretch-semi-condensed">semi-condensed</code></dfn>
   <dd><p>Same as CSS <span>'font-stretch'</span> <span>'semi-condensed'</span> setting.</p></dd>

   <dt><dfn enum-value for="CanvasFontStretch"><code data-x="dom-context-2d-fontstretch-normal">normal</code></dfn>
   <dd><p>The default setting, where width of the glyphs is at 100%.</p></dd>

   <dt><dfn enum-value for="CanvasFontStretch"><code data-x="dom-context-2d-fontstretch-semi-expanded">semi-expanded</code></dfn>
   <dd><p>Same as CSS <span>'font-stretch'</span> <span>'semi-expanded'</span> setting.</p></dd>

   <dt><dfn enum-value for="CanvasFontStretch"><code data-x="dom-context-2d-fontstretch-expanded">expanded</code></dfn>
   <dd><p>Same as CSS <span>'font-stretch'</span> <span>'expanded'</span> setting.</p></dd>

   <dt><dfn enum-value for="CanvasFontStretch"><code data-x="dom-context-2d-fontstretch-extra-expanded">extra-expanded</code></dfn>
   <dd><p>Same as CSS <span>'font-stretch'</span> <span>'extra-expanded'</span> setting.</p></dd>

   <dt><dfn enum-value for="CanvasFontStretch"><code data-x="dom-context-2d-fontstretch-ultra-expanded">ultra-expanded</code></dfn>
   <dd><p>Same as CSS <span>'font-stretch'</span> <span>'ultra-expanded'</span> setting.</p></dd>
  </dl>

  <p>The <code data-x="dom-context-2d-fontVariantCaps">fontVariantCaps</code> attribute's allowed
  keywords are as follows:</p>

  <dl>
   <dt><dfn enum-value for="CanvasFontVariantCaps"><code data-x="dom-context-2d-fontvariantcaps-normal">normal</code></dfn>
   <dd><p>None of the features listed below are enabled.</p></dd>

   <dt><dfn enum-value for="CanvasFontVariantCaps"><code data-x="dom-context-2d-fontvariantcaps-small-caps">small-caps</code></dfn>
   <dd><p>Same as CSS <span>'font-variant-caps'</span> <span>'small-caps'</span> setting.</p></dd>

   <dt><dfn enum-value for="CanvasFontVariantCaps"><code data-x="dom-context-2d-fontvariantcaps-all-small-caps">all-small-caps</code></dfn>
   <dd><p>Same as CSS <span>'font-variant-caps'</span> <span>'all-small-caps'</span> setting.</p></dd>

   <dt><dfn enum-value for="CanvasFontVariantCaps"><code data-x="dom-context-2d-fontvariantcaps-petite-caps">petite-caps</code></dfn>
   <dd><p>Same as CSS <span>'font-variant-caps'</span> <span>'petite-caps'</span> setting.</p></dd>

   <dt><dfn enum-value for="CanvasFontVariantCaps"><code data-x="dom-context-2d-fontvariantcaps-all-petite-caps">all-petite-caps</code></dfn>
   <dd><p>Same as CSS <span>'font-variant-caps'</span> <span>'all-petite-caps'</span> setting.</p></dd>

   <dt><dfn enum-value for="CanvasFontVariantCaps"><code data-x="dom-context-2d-fontvariantcaps-unicase">unicase</code></dfn>
   <dd><p>Same as CSS <span>'font-variant-caps'</span> <span>'unicase'</span> setting.</p></dd>

   <dt><dfn enum-value for="CanvasFontVariantCaps"><code data-x="dom-context-2d-fontvariantcaps-titling-caps">titling-caps</code></dfn>
   <dd><p>Same as CSS <span>'font-variant-caps'</span> <span>'titling-caps'</span> setting.</p></dd>
  </dl>

  <p>The <code data-x="dom-context-2d-textRendering">textRendering</code> attribute's allowed
  keywords are as follows:</p>

  <dl>
    <dt><dfn enum-value for="CanvasTextRendering"><code data-x="dom-context-2d-textRendering-auto">auto</code></dfn>
    <dd><p>Same as 'auto' in <span>SVG text-rendering</span> property.</p></dd>

    <dt><dfn enum-value for="CanvasTextRendering"><code data-x="dom-context-2d-textRendering-optimizeSpeed">optimizeSpeed</code></dfn>
    <dd><p>Same as 'optimizeSpeed' in <span>SVG text-rendering</span> property.</p></dd>

    <dt><dfn enum-value for="CanvasTextRendering"><code data-x="dom-context-2d-textRendering-optimizeLegibility">optimizeLegibility</code></dfn>
    <dd><p>Same as 'optimizeLegibility' in <span>SVG text-rendering</span> property.</p></dd>

    <dt><dfn enum-value for="CanvasTextRendering"><code data-x="dom-context-2d-textRendering-geometricPrecision">geometricPrecision</code></dfn>
    <dd><p>Same as 'geometricPrecision' in <span>SVG text-rendering</span> property.</p></dd>
  </dl>

  <p>The <dfn>text preparation algorithm</dfn> is as follows. It takes as input a string <var>text</var>,
  a <code>CanvasTextDrawingStyles</code> object <var>target</var>, and an optional length
  <var>maxWidth</var>. It returns an array of glyph shapes, each positioned on a common coordinate
  space, a <var>physical alignment</var> whose value is one of <i>left</i>, <i>right</i>, and
  <i>center</i>, and an <span>inline box</span>. (Most callers of this algorithm ignore the
  <var>physical alignment</var> and the <span>inline box</span>.)</p>

  <ol>
   <li><p>If <var>maxWidth</var> was provided but is less than or equal to zero or equal to NaN,
   then return an empty array.</p></li>

   <li><p>Replace all <span>ASCII whitespace</span> in <var>text</var> with U+0020 SPACE
   characters.</p></li>

   <li><p>Let <var>font</var> be the current font of <var>target</var>, as given
   by that object's <code data-x="dom-context-2d-font">font</code> attribute.</p></li>

   <li><p>Let <var>language</var> be the <var>target</var>'s <span
   data-x="concept-CanvasTextDrawingStyles-language">language</span>.</p></li>

   <li>
    <p>If <var>language</var> is "<code data-x="dom-context-2d-lang-inherit">inherit</code>":</p>

    <ol>
     <li><p>Let <var>sourceObject</var> be <var>object</var>'s <span>font style source
     object</span></p></li>

     <li><p>If <var>sourceObject</var> is a <code>canvas</code>
     element, then set <var>language</var> to the <var>sourceObject</var>'s
     <span data-x="language">language</span>.</p></li>

     <li>
      <p>Otherwise:</p>

      <ol>
       <li><p>Assert: <var>sourceObject</var> is an <code>OffscreenCanvas</code> object.</p></li>

       <li><p>Set <var>language</var> to the <var>sourceObject</var>'s
       <span data-x="offscreencanvas-inherited-lang">inherited language</span>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>If <var>language</var> is the empty string, then set <var>language</var> to
   <span data-x="concept-explicitly-unknown">explicitly unknown</span>.</p></li>

   <li>
    <p>Apply the appropriate step from the following list to determine the value of
    <var>direction</var>:</p>

    <dl class="switch">

     <dt>If the <var>target</var> object's <code
     data-x="dom-context-2d-direction">direction</code> attribute has the value "<code
     data-x="dom-context-2d-direction-ltr">ltr</code>"</dt>
     <dd>Let <var>direction</var> be '<span data-x="concept-ltr">ltr</span>'.</dd>

     <dt>If the <var>target</var> object's <code
     data-x="dom-context-2d-direction">direction</code> attribute has the value "<code
     data-x="dom-context-2d-direction-rtl">rtl</code>"</dt>
     <dd>Let <var>direction</var> be '<span data-x="concept-rtl">rtl</span>'.</dd>

     <dt>If the <var>target</var> object's <code
     data-x="dom-context-2d-direction">direction</code> attribute has the value "<code
     data-x="dom-context-2d-direction-inherit">inherit</code>"</dt>
     <dd>
      <ol>
       <li><p>Let <var>sourceObject</var> be <var>object</var>'s <span>font style source
       object</span></p></li>

       <li><p>If <var>sourceObject</var> is a <code>canvas</code>
       element, then let <var>direction</var> be <var>sourceObject</var>'s
       <span data-x="the directionality">directionality</span>.</p></li>

       <li>
        <p>Otherwise:</p>
        <ol>
         <li><p>Assert: <var>sourceObject</var> is an <code>OffscreenCanvas</code> object.</p></li>

         <li><p>Let <var>direction</var> be <var>sourceObject</var>'s
         <span data-x="offscreencanvas-inherited-direction">inherited direction</span>.</p></li>
        </ol>
       </li>
      </ol>
     </dd>
    </dl>
   </li>

   <li>
    <p>Form a hypothetical infinitely-wide CSS <span>line box</span> containing a single
    <span>inline box</span> containing the text <var>text</var>, with the CSS
    <span>content language</span> set to <var>language</var>, and with its CSS properties set as
    follows:</p>

    <table>
     <thead>
      <tr>
       <th>Property</th>
       <th>Source</th>
      </tr>
     </thead>
     <tbody>
      <tr>
       <td><span>'direction'</span></td>
       <td><var>direction</var></td>
      </tr>
      <tr>
        <td><span>'font'</span></td>
        <td><var>font</var></td>
       </tr>
       <tr>
        <td><span>'font-kerning'</span></td>
        <td><var>target</var>'s <code data-x="dom-context-2d-fontKerning">fontKerning</code></td>
       </tr>
       <tr>
        <td><span>'font-stretch'</span></td>
        <td><var>target</var>'s <code data-x="dom-context-2d-fontStretch">fontStretch</code></td>
       </tr>
       <tr>
        <td><span>'font-variant-caps'</span></td>
        <td><var>target</var>'s <code data-x="dom-context-2d-fontVariantCaps">fontVariantCaps</code></td>
       </tr>
       <tr>
        <td><span>'letter-spacing'</span></td>
        <td><var>target</var>'s <span data-x="concept-CanvasTextDrawingStyles-letter-spacing">letter spacing</span></td>
       </tr>
       <tr>
        <td><span>SVG text-rendering</span></td>
        <td><var>target</var>'s <code data-x="dom-context-2d-textRendering">textRendering</code></td>
       </tr>
       <tr>
        <td><span>'white-space'</span></td>
        <td>'pre'</td>
       </tr>
       <tr>
        <td><span>'word-spacing'</span></td>
        <td><var>target</var>'s <span data-x="concept-CanvasTextDrawingStyles-word-spacing">word spacing</span></td>
       </tr>
     </tbody>
    </table>

    <p>and with all other properties set to their initial values.</p>
   </li>

   <!-- if you insert a step here, make sure to adjust the next step's final words -->

   <li><p>If <var>maxWidth</var> was provided and the hypothetical width of the
   <span>inline box</span> in the hypothetical <span>line box</span> is greater than
   <var>maxWidth</var> <span data-x="'px'">CSS pixels</span>, then change <var>font</var> to have a
   more condensed font (if one is available or if a reasonably readable one can be synthesized by
   applying a horizontal scale factor to the font) or a smaller font, and return to the previous
   step.</p></li>

   <li>
    <p>The <var>anchor point</var> is a point on the <span>inline box</span>, and the <var>physical
    alignment</var> is one of the values <i>left</i>, <i>right</i>, and <i>center</i>. These
    variables are determined by the <code data-x="dom-context-2d-textAlign">textAlign</code> and
    <code data-x="dom-context-2d-textBaseline">textBaseline</code> values as follows:</p>

    <p>Horizontal position:</p>

    <dl>
     <dt>If <code data-x="dom-context-2d-textAlign">textAlign</code> is <code
     data-x="dom-context-2d-textAlign-left">left</code></dt>

     <dt>If <code data-x="dom-context-2d-textAlign">textAlign</code> is <code
     data-x="dom-context-2d-textAlign-start">start</code> and <var>direction</var> is
     'ltr'</dt>

     <dt>If <code data-x="dom-context-2d-textAlign">textAlign</code> is <code
     data-x="dom-context-2d-textAlign-end">end</code> and <var>direction</var> is 'rtl'</dt>

     <dd>Let the <var>anchor point</var>'s horizontal position be the left edge of the
     <span>inline box</span>, and let <var>physical alignment</var> be <i>left</i>.</dd>


     <dt>If <code data-x="dom-context-2d-textAlign">textAlign</code> is <code
     data-x="dom-context-2d-textAlign-right">right</code></dt>

     <dt>If <code data-x="dom-context-2d-textAlign">textAlign</code> is <code
     data-x="dom-context-2d-textAlign-end">end</code> and <var>direction</var> is 'ltr'</dt>

     <dt>If <code data-x="dom-context-2d-textAlign">textAlign</code> is <code
     data-x="dom-context-2d-textAlign-start">start</code> and <var>direction</var> is
     'rtl'</dt>

     <dd>Let the <var>anchor point</var>'s horizontal position be the right edge of the
     <span>inline box</span>, and let <var>physical alignment</var> be <i>right</i>.</dd>


     <dt>If <code data-x="dom-context-2d-textAlign">textAlign</code> is <code
     data-x="dom-context-2d-textAlign-center">center</code></dt>

     <dd>Let the <var>anchor point</var>'s horizontal position be half way between the left
     and right edges of the <span>inline box</span>, and let <var>physical alignment</var> be
     <i>center</i>.</dd>
    </dl>

    <p>Vertical position:</p>

    <dl>
     <dt>If <code data-x="dom-context-2d-textBaseline">textBaseline</code> is <code
     data-x="dom-context-2d-textBaseline-top">top</code></dt>
     <dd>Let the <var>anchor point</var>'s vertical position be the top of the em box of
     the <span>first available font</span> of the <span>inline box</span>.</dd>

     <dt>If <code data-x="dom-context-2d-textBaseline">textBaseline</code> is <code
     data-x="dom-context-2d-textBaseline-hanging">hanging</code></dt>
     <dd>Let the <var>anchor point</var>'s vertical position be the <span>hanging baseline</span> of
     the <span>first available font</span> of the <span>inline box</span>.</dd>

     <dt>If <code data-x="dom-context-2d-textBaseline">textBaseline</code> is <code
     data-x="dom-context-2d-textBaseline-middle">middle</code></dt>
     <dd>Let the <var>anchor point</var>'s vertical position be half way between the bottom
     and the top of the em box of the <span>first available font</span> of the <span>inline
     box</span>.</dd>

     <dt>If <code data-x="dom-context-2d-textBaseline">textBaseline</code> is <code
     data-x="dom-context-2d-textBaseline-alphabetic">alphabetic</code></dt>
     <dd>Let the <var>anchor point</var>'s vertical position be the <span>alphabetic baseline</span>
     of the <span>first available font</span> of the <span>inline box</span>.</dd>

     <dt>If <code data-x="dom-context-2d-textBaseline">textBaseline</code> is <code
     data-x="dom-context-2d-textBaseline-ideographic">ideographic</code></dt>
     <dd>Let the <var>anchor point</var>'s vertical position be the <span>ideographic-under
     baseline</span> of the <span>first available font</span> of the <span>inline box</span>.</dd>

     <dt>If <code data-x="dom-context-2d-textBaseline">textBaseline</code> is <code
     data-x="dom-context-2d-textBaseline-bottom">bottom</code></dt>
     <dd>Let the <var>anchor point</var>'s vertical position be the bottom of the em box of
     the <span>first available font</span> of the <span>inline box</span>.</dd>
    </dl>
   </li>

   <li>
    <p>Let <var>result</var> be an array constructed by iterating over each glyph in the
    <span>inline box</span> from left to right (if any), adding to the array, for each glyph, the
    shape of the glyph as it is in the <span>inline box</span>, positioned on a coordinate space
    using <span data-x="'px'">CSS pixels</span> with its origin at the <var>anchor
    point</var>.</p>
   </li>

   <li><p>Return <var>result</var>, <var>physical alignment</var>, and the inline
   box.</p></li>
  </ol>


  <h6>Building paths</h6>

  <p>Objects that implement the <code>CanvasPath</code> interface have a <span
  data-x="concept-path">path</span>. A <dfn data-x="concept-path">path</dfn> has a list of zero or
  more subpaths. Each subpath consists of a list of one or more points, connected by straight or
  curved <dfn>line segments</dfn>, and a flag indicating whether the subpath is closed or not. A
  closed subpath is one where the last point of the subpath is connected to the first point of the
  subpath by a straight line. Subpaths with only one point are ignored when painting the path.</p>

  <p><span data-x="concept-path">Paths</span> have a <dfn>need new subpath</dfn> flag. When this
  flag is set, certain APIs create a new subpath rather than extending the previous one. When a
  <span data-x="concept-path">path</span> is created, its <span>need new subpath</span> flag must be
  set.</p>

  <p>When an object implementing the <code>CanvasPath</code> interface is created, its <span
  data-x="concept-path">path</span> must be initialized to zero subpaths.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-moveTo">moveTo</span>(<var>x</var>, <var>y</var>)</code></dt>
   <dt><code data-x=""><var>path</var>.<span data-x="dom-context-2d-moveTo">moveTo</span>(<var>x</var>, <var>y</var>)</code></dt>

   <dd><p>Creates a new subpath with the given point.</p></dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-closePath">closePath</span>()</code></dt>
   <dt><code data-x=""><var>path</var>.<span data-x="dom-context-2d-closePath">closePath</span>()</code></dt>

   <dd>
    <p>Marks the current subpath as closed, and starts a new subpath with a point the same as the
    start and end of the newly closed subpath.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-lineTo">lineTo</span>(<var>x</var>, <var>y</var>)</code></dt>
   <dt><code data-x=""><var>path</var>.<span data-x="dom-context-2d-lineTo">lineTo</span>(<var>x</var>, <var>y</var>)</code></dt>

   <dd>
    <p>Adds the given point to the current subpath, connected to the previous one by a straight
    line.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-quadraticCurveTo">quadraticCurveTo</span>(<var>cpx</var>, <var>cpy</var>, <var>x</var>, <var>y</var>)</code></dt>
   <dt><code data-x=""><var>path</var>.<span data-x="dom-context-2d-quadraticCurveTo">quadraticCurveTo</span>(<var>cpx</var>, <var>cpy</var>, <var>x</var>, <var>y</var>)</code></dt>

   <dd>
    <p>Adds the given point to the current subpath, connected to the previous one by a quadratic
    Bézier curve with the given control point.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-bezierCurveTo">bezierCurveTo</span>(<var>cp1x</var>, <var>cp1y</var>, <var>cp2x</var>, <var>cp2y</var>, <var>x</var>, <var>y</var>)</code></dt>
   <dt><code data-x=""><var>path</var>.<span data-x="dom-context-2d-bezierCurveTo">bezierCurveTo</span>(<var>cp1x</var>, <var>cp1y</var>, <var>cp2x</var>, <var>cp2y</var>, <var>x</var>, <var>y</var>)</code></dt>

   <dd>
    <p>Adds the given point to the current subpath, connected to the previous one by a cubic Bézier
    curve with the given control points.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-arcTo">arcTo</span>(<var>x1</var>, <var>y1</var>, <var>x2</var>, <var>y2</var>, <var>radius</var>)</code></dt>
   <dt><code data-x=""><var>path</var>.<span data-x="dom-context-2d-arcTo">arcTo</span>(<var>x1</var>, <var>y1</var>, <var>x2</var>, <var>y2</var>, <var>radius</var>)</code></dt>

   <dd>
    <p>Adds an arc with the given control points and radius to the current subpath, connected to the
    previous point by a straight line.</p>

    <p>Throws an <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code> if the given
    radius is negative.</p>

    <figure class="diagrams">
     <!-- if anyone wants to try writing alternative text for these,
     be my guest. I can't work out how to describe them. -->
     <img src="/images/arcTo1.png" alt="" width="357" height="254">
     <img src="/images/arcTo2.png" alt="" width="468" height="310">
     <img src="/images/arcTo3.png" alt="" width="513" height="233">
    </figure>

<!--

c.clearRect(0, 0, 640, 480);
c.save();
try {
  var x0 = 150;
  var y0 = 100;
  var x1 = 375;
  var y1 = 100;
  var x2 = x1;
  var y2 = 250;
  var r = 50;

  var bottom = 210;
  var center = 100;

  var fontSize = 16;
  var margin = 50;
  var dotHalfSize = 5;

  // (also update the label positions at the bottom)


  // ========

  function extendLine(xa,ya,xb,yb) {
    var dx = xb-xa;
    var dy = yb-ya;
    var theta = Math.atan2(dx,dy);
    var extX = margin * Math.sin(theta);
    var extY = margin * Math.cos(theta);
    c.beginPath();
    c.moveTo(xa-extX,ya-extY);
    c.lineTo(xb+extX,yb+extY);
    c.stroke();
  }

  function dot(x,y,s) {
    c.beginPath();
    c.arc(x,y,dotHalfSize,0,Math.PI*2);
    c.fill();
    c.textAlign = 'left';
    c.textBaseline = 'top';
    c.fillText(s,x+dotHalfSize,y);
  }

  c.font = fontSize + 'px Gill Sans';
  c.fillStyle = 'black';
  c.lineWidth = 1;
  c.strokeStyle = 'red';
  extendLine(x0,y0,x1,y1);
  extendLine(x1,y1,x2,y2);

  c.lineWidth = 1;
  c.strokeStyle = 'silver';
  c.beginPath();
  c.moveTo(x0,y0);

  // based on https://lists.freedesktop.org/archives/cairo/2005-August/004801.html
  var angle0, angle1, angle2, angled;
  var xc, yc, dc;
  angle0 = Math.atan2(y0 - y1, x0 - x1);
  angle2 = Math.atan2(y2 - y1, x2 - x1);
  angle1 = (angle0 + angle2) / 2;
  angled = angle2 - angle0;
  if (angled > Math.PI || (angled < 0 && angled > -Math.PI)) {
    angle1 += Math.PI;
    angled = 2 * Math.PI - angled;
    angle0 += Math.PI*2;
    angle2 -= Math.PI*2;
  } else {
    angle0 -= Math.PI*2;
    angle2 += Math.PI*2;
  }
  dc = r / Math.sin (angled / 2);
  yc = Math.sin(angle1);
  xc = Math.cos(angle1);
  xc = x1 + xc * dc;
  yc = y1 + yc * dc;

  c.lineWidth = 1;
  c.strokeStyle = 'blue';
  c.beginPath();
  c.moveTo(xc,yc);
  c.arc(xc,yc,r, Math.PI*3/4, Math.PI*(3/4));
  c.stroke();
  c.beginPath();
  c.strokeStyle = 'silver';
  c.arc(xc,yc,r, Math.PI*3/4,Math.PI*(2+3/4));
  c.stroke();
  c.save();
  try {
    c.translate(xc, yc);
    c.rotate(-Math.PI/4);
    c.textAlign = 'center';
    c.textBaseline = 'top';
    c.fillText("radius", -r/2, 0);
  } finally {
    c.restore();
  }

  c.lineWidth = 15;
  c.lineCap = 'round';
  c.strokeStyle = 'rgba(20,200,40,0.8)';
  c.beginPath();
  c.moveTo(x0,y0);
  c.arcTo(x1,y1,x2,y2,r);
  c.stroke();

  dot(x0,y0, "x\u2080,y\u2080");
  dot(x1,y1, "x\u2081,y\u2081");
  dot(x2,y2, "x\u2082,y\u2082");
  dot(xc,yc, '');

  c.textAlign = 'left';
  c.textBaseline = 'alphabetic';
  var L = 0;

  function print(s) {
    c.fillText(s, center, bottom+L*fontSize*1.3);
    L += 1;
  }
  print('// the thick line corresponds to:');
  print('context.moveTo(x\u2080, y\u2080)');
  print('context.arcTo(x\u2081, y\u2081, x\u2082, y\u2082, radius)');
  print('context.stroke()');

  c.fillStyle = 'green';
  c.fillText("\u25be initial line", 160, 70);
  c.textAlign = 'right';
  c.fillText("arc \u25b8", x1-25, 115);

} finally {
  c.restore();
}

-->
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-arc">arc</span>(<var>x</var>, <var>y</var>, <var>radius</var>, <var>startAngle</var>, <var>endAngle</var> [, <var>counterclockwise</var> ])</code></dt>
   <dt><code data-x=""><var>path</var>.<span data-x="dom-context-2d-arc">arc</span>(<var>x</var>, <var>y</var>, <var>radius</var>, <var>startAngle</var>, <var>endAngle</var> [, <var>counterclockwise</var> ])</code></dt>

   <dd>
    <p>Adds points to the subpath such that the arc described by the circumference of the circle
    described by the arguments, starting at the given start angle and ending at the given end angle,
    going in the given direction (defaulting to clockwise), is added to the path, connected to the
    previous point by a straight line.</p>

    <p>Throws an <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code> if the given
    radius is negative.</p>

    <figure class="diagrams">
     <!-- if anyone wants to try writing alternative text for this, be my guest. I can't work out how to describe it. -->
     <img src="/images/arc1.png" alt="" width="590" height="255">
    </figure>

<!--

c.clearRect(0, 0, 640, 480);
c.save();
try {
  var x0 = 50;
  var y0 = 200;
  var x = 255;
  var y = 150;
  var radius = 100;
  var startAngle = 2.0;
  var endAngle = 5.7;
  var direction = false;

  var bottom = 110;
  var center = 380;

  var fontSize = 16;
  var margin = 50;
  var dotHalfSize = 5;
  var arrowHead = 5;

  // (also update the label positions at the bottom)

  // ========

  function dot(x,y,s) {
    c.beginPath();
    c.arc(x,y,dotHalfSize,0,Math.PI*2);
    c.fill();
    c.textAlign = 'center';
    c.textBaseline = 'top';
    c.fillText(s,x,y+dotHalfSize);
  }

  function arc(x, y, r, sA, eA, d, fr, s) {
    c.lineWidth = 1;
    c.strokeStyle = 'red';
    c.beginPath();
    c.arc(x, y, r, sA, eA, d);
    c.stroke();

    c.save();
    c.translate(x, y);
    c.rotate(eA);
    c.beginPath();
    c.moveTo(r-arrowHead, d?arrowHead:-arrowHead);
    c.lineTo(r, 0);
    c.lineTo(r+arrowHead, d?arrowHead:-arrowHead);
    c.stroke();
    c.strokeStyle = 'silver';
    c.beginPath();
    c.moveTo(0, 0);
    c.lineTo(fr, 0);
    c.stroke();
    c.restore();

    c.fillStyle = 'black';
    if (eA<Math.PI) {
      c.save();
      c.translate(x, y);
      c.rotate(sA+(eA-sA)/2-Math.PI/2);
      c.translate(0, r);
      c.rotate(-(sA+(eA-sA)/2-Math.PI/2));
      c.beginPath();
      c.textAlign = 'center';
      c.textBaseline = 'bottom';
      c.fillText(s + '  ',0,0);
      c.restore();
    } else {
      c.save();
      c.beginPath();
      c.textAlign = 'left';
      c.textBaseline = 'middle';
      c.fillText(' ' + s,x-r,y);
      c.restore();
    }
  }

  function radial(x, y, length, rotation, s) {
    c.lineWidth = 1;
    c.strokeStyle = 'blue';
    c.save();
    c.translate(x, y);
    c.rotate(rotation);
    c.beginPath();
    c.moveTo(0, 0);
    c.lineTo(length, 0);
    c.stroke();
    c.textAlign = 'center';
    c.textBaseline = 'bottom';
    c.fillStyle = 'black';
    c.fillText(s,length/2,0);
    c.restore();
  }

  c.font = fontSize + 'px Gill Sans';

  // construction lines
  c.lineWidth = 1;
  c.strokeStyle = 'silver';
  // circle
  c.beginPath();
  c.arc(x, y, radius, 0, 2*Math.PI);
  c.stroke();
  // angles
  arc(x, y, radius/10+2*radius/3, 0, startAngle, false, radius, '\u03B1');
  arc(x, y, radius/10+radius/3, 0, endAngle, false, radius, '\u03B2');
  // radii
  radial(x, y, radius, 0, 'radius');

  // the line
  c.lineWidth = 15;
  c.lineCap = 'round';
  c.strokeStyle = 'rgba(20,200,40,0.8)';
  c.beginPath();
  c.moveTo(x0,y0);
  c.arc(x,y,radius,startAngle,endAngle,direction);
  c.stroke();

  dot(x0,y0, "x\u2080,y\u2080");
  dot(x,y, "x,y");

  c.textAlign = 'left';
  c.textBaseline = 'alphabetic';
  var L = 0;

  function print(s) {
    c.fillText(s, center, bottom+L*fontSize*1.3);
    L += 1;
  }
  print('// the thick line corresponds to:');
  print('context.moveTo(x\u2080, y\u2080)');
  print('context.arc(x, y, radius, \u03B1, \u03B2)');
  print('context.stroke()');

  c.fillStyle = 'green';

  c.textBaseline = 'top';
  c.fillText("initial line \u25b4", x0+20, y0+fontSize*2);

  c.textAlign = 'right';
  c.textBaseline = 'middle';
  c.fillText("arc \u25b8", x-radius-15, y);

} finally {
  c.restore();
}
-->
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-ellipse">ellipse</span>(<var>x</var>, <var>y</var>, <var>radiusX</var>, <var>radiusY</var>, <var>rotation</var>, <var>startAngle</var>, <var>endAngle</var> [, <var>counterclockwise</var>])</code></dt>
   <dt><code data-x=""><var>path</var>.<span data-x="dom-context-2d-ellipse">ellipse</span>(<var>x</var>, <var>y</var>, <var>radiusX</var>, <var>radiusY</var>, <var>rotation</var>, <var>startAngle</var>, <var>endAngle</var> [, <var>counterclockwise</var>])</code></dt>

   <dd>
    <p>Adds points to the subpath such that the arc described by the circumference of the ellipse
    described by the arguments, starting at the given start angle and ending at the given end angle,
    going in the given direction (defaulting to clockwise), is added to the path, connected to the
    previous point by a straight line.</p>

    <p>Throws an <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code> if the given
    radius is negative.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-rect">rect</span>(<var>x</var>, <var>y</var>, <var>w</var>, <var>h</var>)</code></dt>
   <dt><code data-x=""><var>path</var>.<span data-x="dom-context-2d-rect">rect</span>(<var>x</var>, <var>y</var>, <var>w</var>, <var>h</var>)</code></dt>

   <dd><p>Adds a new closed subpath to the path, representing the given rectangle.</p></dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-roundRect">roundRect</span>(<var>x</var>, <var>y</var>, <var>w</var>, <var>h</var>, <var>radii</var>)</code></dt>
   <dt><code data-x=""><var>path</var>.<span data-x="dom-context-2d-roundRect">roundRect</span>(<var>x</var>, <var>y</var>, <var>w</var>, <var>h</var>, <var>radii</var>)</code></dt>

   <dd>
    <p>Adds a new closed subpath to the path representing the given rounded rectangle.
    <var>radii</var> is either a list of radii or a single radius representing the corners of the
    rectangle in pixels. If a list is provided, the number and order of these radii function in the
    same way as the CSS <span>'border-radius'</span> property. A single radius behaves the same way
    as a list with a single element.</p>

    <p>If <var>w</var> and <var>h</var> are both greater than or equal to 0, or if both are smaller
    than 0, then the path is drawn clockwise. Otherwise, it is drawn counterclockwise.</p>

    <p>When <var>w</var> is negative, the rounded rectangle is flipped horizontally, which means
    that the radius values that normally apply to the left corners are used on the right and vice
    versa. Similarly, when <var>h</var> is negative, the rounded rect is flipped vertically.</p>

    <p>When a value <var>r</var> in <var>radii</var> is a number, the corresponding corner(s) are
    drawn as circular arcs of radius <var>r</var>.</p>

    <p>When a value <var>r</var> in <var>radii</var> is an object with <code data-x="">{ x, y
    }</code> properties, the corresponding corner(s) are drawn as elliptical arcs whose <var>x</var>
    and <var>y</var> radii are equal to <var>r.x</var> and <var>r.y</var>, respectively.</p>

    <p>When the sum of the <var>radii</var> of two corners of the same edge is greater than the
    length of the edge, all the <var>radii</var> of the rounded rectangle are scaled by a factor of
    length / (<var>r1</var> + <var>r2</var>). If multiple edges have this property, the scale factor
    of the edge with the smallest scale factor is used. This is consistent with CSS behavior.</p>

    <p>Throws a <code data-x="js-RangeError">RangeError</code> if <var>radii</var> is a list whose
    size is not one, two, three, or four.</p>

    <p>Throws a <code data-x="js-RangeError">RangeError</code> if a value in <var>radii</var> is a
    negative number, or is an <code data-x="">{ x, y }</code> object whose <code data-x="">x</code>
    or <code data-x="">y</code> properties are negative numbers.</p>

    <!-- Consider adding this useful information to all the methods which behave this way: -->
    <!--
    <p>If any of <var>x</var>, <var>y</var>, <var>width</var> or <var>height</var> are non-finite
    numbers, or if a value in <var>radii</var> is a non-finite number, or if a value in
    <var>radii</var> is an <code data-x="">{ x, y }</code>  object whose <code data-x="">x</code> or
    <code data-x="">y</code> properties are non-finite numbers, then the method aborts without
    throwing an exception and without adding anything to the current path.</p>
    -->
   </dd>
  </dl>

  <div w-nodev>

  <p>The following methods allow authors to manipulate the <span data-x="concept-path">paths</span>
  of objects implementing the <code>CanvasPath</code> interface.</p>

  <p>For objects implementing the <code>CanvasDrawPath</code> and <code>CanvasTransform</code>
  interfaces, the points passed to the methods, and the resulting lines added to <span>current
  default path</span> by these methods, must be transformed according to the <span
  data-x="dom-context-2d-transformation">current transformation matrix</span> before being added to
  the path.</p>

  <p>The <dfn method for="CanvasPath"><code data-x="dom-context-2d-moveTo">moveTo(<var>x</var>,
  <var>y</var>)</code></dfn> method, when invoked, must run these steps:</p>

  <ol>
   <li><p>If either of the arguments are infinite or NaN, then return.</p></li>

   <li><p>Create a new subpath with the specified point as its first (and only) point.</p></li>
  </ol>

  <p>When the user agent is to <dfn>ensure there is a subpath</dfn> for a coordinate (<var>x</var>,
  <var>y</var>) on a <span data-x="concept-path">path</span>, the user agent must check to see if
  the <span data-x="concept-path">path</span> has its <span>need new subpath</span> flag set. If it
  does, then the user agent must create a new subpath with the point (<var>x</var>, <var>y</var>) as
  its first (and only) point, as if the <code data-x="dom-context-2d-moveTo">moveTo()</code> method
  had been called, and must then unset the <span data-x="concept-path">path</span>'s <span>need new
  subpath</span> flag.</p>


  <p>The <dfn method for="CanvasPath"><code
  data-x="dom-context-2d-closePath">closePath()</code></dfn> method, when invoked, must do nothing
  if the object's path has no subpaths. Otherwise, it must mark the last subpath as closed, create a
  new subpath whose first point is the same as the previous subpath's first point, and finally add
  this new subpath to the path.</p>

  <p class="note">If the last subpath had more than one point in its list of points, then this is
  equivalent to adding a straight line connecting the last point back to the first point of the last
  subpath, thus "closing" the subpath.</p>

  <hr>

  <p>New points and the lines connecting them are added to subpaths using the methods described
  below. In all cases, the methods only modify the last subpath in the object's path.</p>

  <p>The <dfn method for="CanvasPath"><code data-x="dom-context-2d-lineTo">lineTo(<var>x</var>,
  <var>y</var>)</code></dfn> method, when invoked, must run these steps:</p>

  <ol>
   <li><p>If either of the arguments are infinite or NaN, then return.</p></li>

   <li><p>If the object's path has no subpaths, then <span>ensure there is a subpath</span> for
   <span data-x="">(<var>x</var>, <var>y</var>)</span>.</p></li>

   <li><p>Otherwise, connect the last point in the subpath to the given point (<var>x</var>,
   <var>y</var>) using a straight line, and then add the given point (<var>x</var>, <var>y</var>) to
   the subpath.</p></li>
  </ol>

  <p>The <dfn method for="CanvasPath"><code
  data-x="dom-context-2d-quadraticCurveTo">quadraticCurveTo(<var>cpx</var>, <var>cpy</var>,
  <var>x</var>, <var>y</var>)</code></dfn> method, when invoked, must run these steps:</p>

  <ol>
   <li><p>If any of the arguments are infinite or NaN, then return.</p></li>

   <li><p><span>Ensure there is a subpath</span> for <span data-x="">(<var>cpx</var>,
   <var>cpy</var>)</span></p></li>

   <li><p>Connect the last point in the subpath to the given point (<var>x</var>, <var>y</var>)
   using a quadratic B&eacute;zier curve with control point (<var>cpx</var>, <var>cpy</var>).
   <ref>BEZIER</ref></p></li>

   <li><p>Add the given point (<var>x</var>, <var>y</var>) to the subpath.</p></li>
  </ol>

  <p>The <dfn method for="CanvasPath"><code
  data-x="dom-context-2d-bezierCurveTo">bezierCurveTo(<var>cp1x</var>, <var>cp1y</var>,
  <var>cp2x</var>, <var>cp2y</var>, <var>x</var>, <var>y</var>)</code></dfn> method, when invoked,
  must run these steps:</p>

  <ol>
   <li><p>If any of the arguments are infinite or NaN, then return.</p></li>

   <li><p><span>Ensure there is a subpath</span> for <span data-x="">(<var>cp1x</var>,
   <var>cp1y</var>)</span>.</p></li>

   <li><p>Connect the last point in the subpath to the given point (<var>x</var>, <var>y</var>)
   using a cubic B&eacute;zier curve with control points (<var>cp1x</var>, <var>cp1y</var>) and
   (<var>cp2x</var>, <var>cp2y</var>). <ref>BEZIER</ref></p></li>

   <li><p>Add the point (<var>x</var>, <var>y</var>) to the subpath.</p></li>
  </ol>

  <hr>

  <p>The <dfn method for="CanvasPath"><code data-x="dom-context-2d-arcTo">arcTo(<var>x1</var>,
  <var>y1</var>, <var>x2</var>, <var>y2</var>, <var>radius</var>)</code></dfn> method, when invoked,
  must run these steps:</p>

  <ol>
   <li><p>If any of the arguments are infinite or NaN, then return.</p></li>

   <li><p><span>Ensure there is a subpath</span> for <span data-x="">(<var>x1</var>,
   <var>y1</var>)</span>.</p></li>

   <li><p>If <var>radius</var> is negative, then throw an <span>"<code>IndexSizeError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Let the point (<var>x0</var>, <var>y0</var>) be the last point in the subpath, transformed
   by the inverse of the <span data-x="dom-context-2d-transformation">current transformation
   matrix</span> (so that it is in the same coordinate system as the points passed to the
   method).</p></li>

   <li><p>If the point (<var>x0</var>, <var>y0</var>) is equal to the point (<var>x1</var>,
   <var>y1</var>), or if the point (<var>x1</var>, <var>y1</var>) is equal to the point
   (<var>x2</var>, <var>y2</var>), or if <var>radius</var> is zero, then add the point
   (<var>x1</var>, <var>y1</var>) to the subpath, and connect that point to the previous point
   (<var>x0</var>, <var>y0</var>) by a straight line.</p></li>

   <li><p>Otherwise, if the points (<var>x0</var>, <var>y0</var>), (<var>x1</var>, <var>y1</var>),
   and (<var>x2</var>, <var>y2</var>) all lie on a single straight line, then add the point
   (<var>x1</var>, <var>y1</var>) to the subpath, and connect that point to the previous point
   (<var>x0</var>, <var>y0</var>) by a straight line.</p></li>

   <li><p>Otherwise, let <var>The Arc</var> be the shortest arc given by circumference of the circle
   that has radius <var>radius</var>, and that has one point tangent to the half-infinite line that
   crosses the point (<var>x0</var>, <var>y0</var>) and ends at the point (<var>x1</var>,
   <var>y1</var>), and that has a different point tangent to the half-infinite line that ends at the
   point (<var>x1</var>, <var>y1</var>) and crosses the point (<var>x2</var>, <var>y2</var>). The
   points at which this circle touches these two lines are called the start and end tangent points
   respectively. Connect the point (<var>x0</var>, <var>y0</var>) to the start tangent point by a
   straight line, adding the start tangent point to the subpath, and then connect the start tangent
   point to the end tangent point by <var>The Arc</var>, adding the end tangent point to the
   subpath.</p></li>
  </ol>

  <hr>

  <p>The <dfn method for="CanvasPath"><code data-x="dom-context-2d-arc">arc(<var>x</var>,
  <var>y</var>, <var>radius</var>, <var>startAngle</var>, <var>endAngle</var>,
  <var>counterclockwise</var>)</code></dfn> method, when invoked, must run the <span>ellipse method
  steps</span> with this, <var>x</var>, <var>y</var>, <var>radius</var>, <var>radius</var>, 0,
  <var>startAngle</var>, <var>endAngle</var>, and <var>counterclockwise</var>.</p>

  <p class="note">This makes it equivalent to <code data-x="dom-context-2d-ellipse">ellipse()</code>
  except that both radii are equal and <var>rotation</var> is 0.</p>

  <p>The <dfn method for="CanvasPath"><code data-x="dom-context-2d-ellipse">ellipse(<var>x</var>,
  <var>y</var>, <var>radiusX</var>, <var>radiusY</var>, <var>rotation</var>, <var>startAngle</var>,
  <var>endAngle</var>, <var>counterclockwise</var>)</code></dfn> method, when invoked, must run the
  <span>ellipse method steps</span> with this, <var>x</var>, <var>y</var>, <var>radiusX</var>,
  <var>radiusY</var>, <var>rotation</var>, <var>startAngle</var>, <var>endAngle</var>, and
  <var>counterclockwise</var>.</p>

  <p>The <dfn>determine the point on an ellipse steps</dfn>, given <var>ellipse</var>, and
  <var>angle</var>, are:</p>

  <ol>
   <li><p>Let <var>eccentricCircle</var> be the circle that shares its origin with
   <var>ellipse</var>, with a radius equal to the semi-major axis of <var>ellipse</var>.</p></li>

   <li><p>Let <var>outerPoint</var> be the point on <var>eccentricCircle</var>'s circumference at
   <var>angle</var> measured in radians clockwise from <var>ellipse</var>'s semi-major
   axis.</p></li>

   <li><p>Let <var>chord</var> be the line perpendicular to <var>ellipse</var>'s major axis between
   this axis and <var>outerPoint</var>.</p></li>

   <li><p>Return the point on <var>chord</var> that crosses <var>ellipse</var>'s
   circumference.</p></li>
  </ol>

  <p>The <dfn>ellipse method steps</dfn>, given <var>canvasPath</var>, <var>x</var>, <var>y</var>,
  <var>radiusX</var>, <var>radiusY</var>, <var>rotation</var>, <var>startAngle</var>,
  <var>endAngle</var>, and <var>counterclockwise</var>, are:</p>

  <ol>
   <li><p>If any of the arguments are infinite or NaN, then return.</p></li>

   <li><p>If either <var>radiusX</var> or <var>radiusY</var> are negative, then throw an
   <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <var>canvasPath</var>'s path has any subpaths, then add a straight line from the last
   point in the subpath to the start point of the arc.</p></li>

   <li>
    <p>Add the start and end points of the arc to the subpath, and connect them with an arc. The
    arc and its start and end points are defined as follows:</p>

    <p>Consider an ellipse that has its origin at (<var>x</var>, <var>y</var>), that has a
    major-axis radius <var>radiusX</var> and a minor-axis radius <var>radiusY</var>, and that is
    rotated about its origin such that its semi-major axis is inclined <var>rotation</var> radians
    clockwise from the x-axis.</p>

    <p>If <var>counterclockwise</var> is false and <span data-x=""><var>endAngle</var> &minus;
    <var>startAngle</var></span> is greater than or equal to <span data-x="">2&pi;</span>, or, if
    <var>counterclockwise</var> is <em>true</em> and <span data-x=""><var>startAngle</var> &minus;
    <var>endAngle</var></span> is greater than or equal to <span data-x="">2&pi;</span>, then the
    arc is the whole circumference of this ellipse, and both the start point and the end point are
    the result of running the <span>determine the point on an ellipse steps</span> given this
    ellipse and <var>startAngle</var>.</p>

    <p>Otherwise, the start point is the result of running the <span>determine the point on an
    ellipse steps</span> given this ellipse and <var>startAngle</var>, the end point is the result
    of running the <span>determine the point on an ellipse steps</span> given this ellipse and
    <var>endAngle</var>, and the arc is the path along the circumference of this ellipse from the
    start point to the end point, going counterclockwise if <var>counterclockwise</var> is true,
    and clockwise otherwise. Since the points are on the ellipse, as opposed to being simply angles
    from zero, the arc can never cover an angle greater than <span data-x="">2&pi;</span>
    radians.</p>

    <p class="note">Even if the arc covers the entire circumference of the ellipse and there are no
    other points in the subpath, the path is not closed unless the <code
    data-x="dom-context-2d-closePath">closePath()</code> method is appropriately invoked.</p>
   </li>
  </ol>

  <hr>

  <p>The <dfn method for="CanvasPath"><code data-x="dom-context-2d-rect">rect(<var>x</var>,
  <var>y</var>, <var>w</var>, <var>h</var>)</code></dfn> method, when invoked, must run these
  steps:</p>

  <ol>
   <li><p>If any of the arguments are infinite or NaN, then return.</p></li>

   <li><p>Create a new subpath containing just the four points (<var>x</var>, <var>y</var>),
   (<var>x</var>+<var>w</var>, <var>y</var>), (<var>x</var>+<var>w</var>,
   <var>y</var>+<var>h</var>), (<var>x</var>, <var>y</var>+<var>h</var>), in that order, with those
   four points connected by straight lines.</p></li>

   <li><p>Mark the subpath as closed.</p></li>

   <li><p>Create a new subpath with the point (<var>x</var>, <var>y</var>) as the only point in the
   subpath.</p></li>
  </ol>

  <p>The <dfn method for="CanvasPath"><code
  data-x="dom-context-2d-roundRect">roundRect(<var>x</var>, <var>y</var>, <var>w</var>,
  <var>h</var>, <var>radii</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>If any of <var>x</var>, <var>y</var>, <var>w</var>, or <var>h</var> are infinite or NaN,
   then return.</p></li>

   <li><p>If <var>radii</var> is an <code data-x="idl-unrestricted-double">unrestricted
   double</code> or <code>DOMPointInit</code>, then set <var>radii</var> to « <var>radii</var>
   ».</p></li>

   <li><p>If <var>radii</var> is not a list of size one, two, three, or four, then throw a <code
   data-x="js-RangeError">RangeError</code>.</p></li>

   <li><p>Let <var>normalizedRadii</var> be an empty list.</p></li>

   <li>
    <p>For each <var>radius</var> of <var>radii</var>:</p>

    <ol>
     <li>
      <p>If <var>radius</var> is a <code>DOMPointInit</code>:</p>

      <ol>
       <li><p>If <var>radius</var>["<code data-x="DOMPointInit-x">x</code>"] or
       <var>radius</var>["<code data-x="DOMPointInit-y">y</code>"] is infinite or NaN, then
       return.</p></li>

       <li><p>If <var>radius</var>["<code data-x="DOMPointInit-x">x</code>"] or
       <var>radius</var>["<code data-x="DOMPointInit-y">y</code>"] is negative, then throw a <code
       data-x="js-RangeError">RangeError</code>.</p></li>

       <li><p>Otherwise, append <var>radius</var> to <var>normalizedRadii</var>.</p></li>
      </ol>
     </li>

     <li>
      <p>If <var>radius</var> is a <code data-x="idl-unrestricted-double">unrestricted
      double</code>:</p>

      <ol>
       <li><p>If <var>radius</var> is infinite or NaN, then return.</p></li>

       <li><p>If <var>radius</var> is negative, then throw a <code
       data-x="js-RangeError">RangeError</code>.</p></li>

       <li><p>Otherwise, append «[ "<code data-x="DOMPointInit-x">x</code>" → <var>radius</var>,
       "<code data-x="DOMPointInit-x">y</code>" → <var>radius</var> ]» to
       <var>normalizedRadii</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Let <var>upperLeft</var>, <var>upperRight</var>, <var>lowerRight</var>, and
   <var>lowerLeft</var> be null.</p></li>

   <li><p>If <var>normalizedRadii</var>'s size is 4, then set <var>upperLeft</var> to
   <var>normalizedRadii</var>[0], set <var>upperRight</var> to <var>normalizedRadii</var>[1], set
   <var>lowerRight</var> to <var>normalizedRadii</var>[2], and set <var>lowerLeft</var> to
   <var>normalizedRadii</var>[3].</p></li>

   <li><p>If <var>normalizedRadii</var>'s size is 3, then set <var>upperLeft</var> to
   <var>normalizedRadii</var>[0], set <var>upperRight</var> and <var>lowerLeft</var> to
   <var>normalizedRadii</var>[1], and set <var>lowerRight</var> to
   <var>normalizedRadii</var>[2].</p></li>

   <li><p>If <var>normalizedRadii</var>'s size is 2, then set <var>upperLeft</var> and
   <var>lowerRight</var> to <var>normalizedRadii</var>[0] and set <var>upperRight</var> and
   <var>lowerLeft</var> to <var>normalizedRadii</var>[1].</p></li>

   <li><p>If <var>normalizedRadii</var>'s size is 1, then set <var>upperLeft</var>,
   <var>upperRight</var>, <var>lowerRight</var>, and <var>lowerLeft</var> to
   <var>normalizedRadii</var>[0].</p></li>

   <li>
    <p>Corner curves must not overlap. Scale all radii to prevent this:</p>

    <ol>
     <li><p>Let <var>top</var> be <var>upperLeft</var>["<code data-x="DOMPointInit-x">x</code>"] +
     <var>upperRight</var>["<code data-x="DOMPointInit-x">x</code>"].</p></li>

     <li><p>Let <var>right</var> be <var>upperRight</var>["<code
     data-x="DOMPointInit-y">y</code>"] + <var>lowerRight</var>["<code
     data-x="DOMPointInit-y">y</code>"].</p></li>

     <li><p>Let <var>bottom</var> be <var>lowerRight</var>["<code
     data-x="DOMPointInit-x">x</code>"] + <var>lowerLeft</var>["<code
     data-x="DOMPointInit-x">x</code>"].</p></li>

     <li><p>Let <var>left</var> be <var>upperLeft</var>["<code data-x="DOMPointInit-y">y</code>"]
     + <var>lowerLeft</var>["<code data-x="DOMPointInit-y">y</code>"].</p></li>

     <li><p>Let <var>scale</var> be the minimum value of the ratios <var>w</var> / <var>top</var>,
     <var>h</var> / <var>right</var>, <var>w</var> / <var>bottom</var>,  <var>h</var> /
     <var>left</var>.</p></li>

     <li><p>If <var>scale</var> is less than 1, then set the <code data-x="DOMPointInit-x">x</code>
     and <code data-x="DOMPointInit-y">y</code> members of <var>upperLeft</var>,
     <var>upperRight</var>, <var>lowerLeft</var>, and <var>lowerRight</var> to their current values
     multiplied by <var>scale</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>Create a new subpath:</p>

    <ol>
     <li><p>Move to the point (<var>x</var> + <var>upperLeft</var>["<code
     data-x="DOMPointInit-x">x</code>"], <var>y</var>).</p></li>

     <li><p>Draw a straight line to the point (<var>x</var> + <var>w</var> &minus;
     <var>upperRight</var>["<code data-x="DOMPointInit-x">x</code>"], <var>y</var>).</p></li>

     <li><p>Draw an arc to the point (<var>x</var> + <var>w</var>, <var>y</var> +
     <var>upperRight</var>["<code data-x="DOMPointInit-y">y</code>"]).</p></li>

     <li><p>Draw a straight line to the point (<var>x</var> + <var>w</var>, <var>y</var> +
     <var>h</var> &minus; <var>lowerRight</var>["<code data-x="DOMPointInit-y">y</code>"]).</p></li>

     <li><p>Draw an arc to the point (<var>x</var> + <var>w</var> &minus;
     <var>lowerRight</var>["<code data-x="DOMPointInit-x">x</code>"], <var>y</var> +
     <var>h</var>).</p></li>

     <li><p>Draw a straight line to the point (<var>x</var> + <var>lowerLeft</var>["<code
     data-x="DOMPointInit-x">x</code>"], <var>y</var> + <var>h</var>).</p></li>

     <li><p>Draw an arc to the point (<var>x</var>, <var>y</var> + <var>h</var> &minus;
     <var>lowerLeft</var>["<code data-x="DOMPointInit-y">y</code>"]).</p></li>

     <li><p>Draw a straight line to the point (<var>x</var>, <var>y</var> +
     <var>upperLeft</var>["<code data-x="DOMPointInit-y">y</code>"]).</p></li>

     <li><p>Draw an arc to the point (<var>x</var> + <var>upperLeft</var>["<code
     data-x="DOMPointInit-x">x</code>"], <var>y</var>).</p></li>
    </ol>
   </li>

   <li><p>Mark the subpath as closed.</p></li>

   <li><p>Create a new subpath with the point (<var>x</var>, <var>y</var>) as the only point in the
   subpath.</p></li>
  </ol>

  <p class="note">This is designed to behave similarly to the CSS <span
  data-x="'border-radius'">'border-radius'</span> property.</p>

  <!-- v6 feature request:
        * points as a primitive shape
          http://home.comcast.net/~urbanjost/canvas/vogle4.html
  -->

  </div>



  <h6><code>Path2D</code> objects</h6>

  <p><code>Path2D</code> objects can be used to declare paths that are then later used on
  objects implementing the <code>CanvasDrawPath</code> interface. In addition to many of the APIs
  described in earlier sections, <code>Path2D</code> objects have methods to combine paths, and to
  add text to paths.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>path</var> = new <span subdfn data-x="dom-Path2D">Path2D</span>()</code></dt>
   <dd><p>Creates a new empty <code>Path2D</code> object.</p></dd>

   <dt><code data-x=""><var>path</var> = new <span data-x="dom-Path2D">Path2D</span>(<var>path</var>)</code></dt>
   <dd>
    <p>When <var>path</var> is a <code>Path2D</code> object, returns a copy.</p>

    <p>When <var>path</var> is a string, creates the path described by the argument, interpreted as
    SVG path data. <ref>SVG</ref>
   </p></dd>

   <dt><code data-x=""><var>path</var>.<span subdfn data-x="dom-Path2D-addPath">addPath</span>(<var>path</var> [, <var>transform</var> ])</code></dt>
   <dd><p>Adds to the path the path given by the argument.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn constructor for="Path2D"><code
  data-x="dom-Path2D">Path2D(<var>path</var>)</code></dfn> constructor, when invoked, must run these
  steps:</p>

  <ol>
   <li><p>Let <var>output</var> be a new <code>Path2D</code> object.</p></li>

   <li><p>If <var>path</var> is not given, then return <var>output</var>.</p></li>

   <li><p>If <var>path</var> is a <code>Path2D</code> object, then add all subpaths of
   <var>path</var> to <var>output</var> and return <var>output</var>. (In other words, it returns a
   copy of the argument.)</p></li>

   <li>
    <p>Let <var>svgPath</var> be the result of parsing and interpreting <var>path</var> according to
    <cite>SVG 2</cite>'s rules for path data. <ref>SVG</ref></p>

    <p class="note">The resulting path could be empty. SVG defines error handling rules for parsing
    and applying path data.</p>
   </li>

   <li><p>Let (<var>x</var>, <var>y</var>) be the last point in <var>svgPath</var>.</p></li>

   <li><p>Add all the subpaths, if any, from <var>svgPath</var> to <var>output</var>.</p></li>

   <li><p>Create a new subpath in <var>output</var> with (<var>x</var>, <var>y</var>) as the only
   point in the subpath.</p></li>

   <li><p>Return <var>output</var>.</p></li>
  </ol>

  <hr>

  <p>The <dfn method for="Path2D"><code data-x="dom-Path2D-addPath">addPath(<var>path</var>,
  <var>transform</var>)</code></dfn> method, when invoked on a <code>Path2D</code> object
  <var>a</var>, must run these steps:</p>

  <ol>
   <li><p>If the <code>Path2D</code> object <var>path</var> has no subpaths, then return.</p></li>

   <li><p>Let <var>matrix</var> be the result of <span data-x="create a DOMMatrix from a 2D
   dictionary">creating a <code>DOMMatrix</code> from the 2D dictionary</span>
   <var>transform</var>.</p></li>

   <li><p>If one or more of <var>matrix</var>'s <span>m11 element</span>, <span>m12 element</span>,
   <span>m21 element</span>, <span>m22 element</span>, <span>m41 element</span>, or <span>m42
   element</span> are infinite or NaN, then return.</p></li>

   <li><p>Create a copy of all the subpaths in <var>path</var>. Let this copy be known as
   <var>c</var>.</p></li>

   <li><p>Transform all the coordinates and lines in <var>c</var> by the transform matrix
   <var>matrix</var>.</p></li>

   <li><p>Let (<var>x</var>, <var>y</var>) be the last point in the last subpath
   of <var>c</var>.</p></li>

   <li><p>Add all the subpaths in <var>c</var> to <var>a</var>.</p></li>

   <li><p>Create a new subpath in <var>a</var> with (<var>x</var>, <var>y</var>) as the only point
   in the subpath.</p></li>
  </ol>

  </div>


  <h6><dfn data-x="dom-context-2d-transformation">Transformations</dfn></h6>

  <p>Objects that implement the <code>CanvasTransform</code> interface have a <dfn>current
  transformation matrix</dfn>, as well as methods (described in this section) to manipulate it. When
  an object implementing the <code>CanvasTransform</code> interface is created, its transformation
  matrix must be initialized to the identity matrix.</p>

  <p>The <span>current transformation matrix</span> is applied to coordinates when creating the
  <span>current default path</span>, and when painting text, shapes, and <code>Path2D</code>
  objects, on objects implementing the <code>CanvasTransform</code> interface.</p>
  <!-- conformance criteria for actual drawing are described in the various sections below -->

  <div w-nodev>

  <p>The transformations must be performed in reverse order.</p>

  <p class="note">For instance, if a scale transformation that doubles the width is applied to the
  canvas, followed by a rotation transformation that rotates drawing operations by a quarter turn,
  and a rectangle twice as wide as it is tall is then drawn on the canvas, the actual result will be
  a square.</p> <!-- q.v. https://goo.gl/5RLrN -->

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-scale">scale</span>(<var>x</var>, <var>y</var>)</code></dt>

   <dd>
    <p>Changes the <span>current transformation matrix</span> to apply a scaling transformation with
    the given characteristics.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-rotate">rotate</span>(<var>angle</var>)</code></dt>

   <dd>
    <p>Changes the <span>current transformation matrix</span> to apply a rotation transformation
    with the given characteristics. The angle is in radians.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-translate">translate</span>(<var>x</var>, <var>y</var>)</code></dt>

   <dd>
    <p>Changes the <span>current transformation matrix</span> to apply a translation transformation
    with the given characteristics.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-transform">transform</span>(<var>a</var>, <var>b</var>, <var>c</var>, <var>d</var>, <var>e</var>, <var>f</var>)</code></dt>

   <dd>
    <p>Changes the <span>current transformation matrix</span> to apply the matrix given by the
    arguments as described below.</p>
   </dd>

   <dt><code data-x=""><var>matrix</var> = <var>context</var>.<span subdfn data-x="dom-context-2d-getTransform">getTransform</span>()</code></dt>

   <dd>
    <p>Returns a copy of the <span>current transformation matrix</span>, as a newly created
    <code>DOMMatrix</code> object.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-setTransform">setTransform</span>(<var>a</var>, <var>b</var>, <var>c</var>, <var>d</var>, <var>e</var>, <var>f</var>)</code></dt>

   <dd>
    <p>Changes the <span>current transformation matrix</span> <em>to</em> the matrix given by the
    arguments as described below.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-setTransform-matrix">setTransform</span>(<var>transform</var>)</code></dt>

   <dd>
    <p>Changes the <span>current transformation matrix</span> <em>to</em> the matrix represented by
    the passed <code>DOMMatrix2DInit</code> dictionary.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-resetTransform">resetTransform</span>()</code></dt>

   <dd><p>Changes the <span>current transformation matrix</span> to the identity matrix.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="CanvasTransform"><code data-x="dom-context-2d-scale">scale(<var>x</var>,
  <var>y</var>)</code></dfn> method, when invoked, must run these steps:</p>

  <ol>
   <li><p>If either of the arguments are infinite or NaN, then return.</p></li>

   <li><p>Add the scaling transformation described by the arguments to the <span>current
   transformation matrix</span>. The <var>x</var> argument represents the scale factor in the
   horizontal direction and the <var>y</var> argument represents the scale factor in the vertical
   direction. The factors are multiples.</p></li>
  </ol>

  <p>The <dfn method for="CanvasTransform"><code
  data-x="dom-context-2d-rotate">rotate(<var>angle</var>)</code></dfn> method, when invoked, must
  run these steps:</p>

  <ol>
   <li><p>If <var>angle</var> is infinite or NaN, then return.</p></li>

   <li><p>Add the rotation transformation described by the argument to the <span>current
   transformation matrix</span>. The <var>angle</var> argument represents a clockwise rotation angle
   expressed in radians.</p></li>
  </ol>

  <p>The <dfn method for="CanvasTransform"><code
  data-x="dom-context-2d-translate">translate(<var>x</var>, <var>y</var>)</code></dfn> method, when
  invoked, must run these steps:</p>

  <ol>
   <li><p>If either of the arguments are infinite or NaN, then return.</p></li>

   <li><p>Add the translation transformation described by the arguments to the <span>current
   transformation matrix</span>. The <var>x</var> argument represents the translation distance in
   the horizontal direction and the <var>y</var> argument represents the translation distance in the
   vertical direction. The arguments are in coordinate space units.</p></li>
  </ol>

  <p>The <dfn method for="CanvasTransform"><code
  data-x="dom-context-2d-transform">transform(<var>a</var>, <var>b</var>, <var>c</var>,
  <var>d</var>, <var>e</var>, <var>f</var>)</code></dfn> method, when invoked, must run these
  steps:</p>

  <ol>
   <li><p>If any of the arguments are infinite or NaN, then return.</p></li>

   <li>
    <p>Replace the <span>current transformation matrix</span> with the result of <span
    data-x="matrix multiplication">multiplying</span> the <span>current transformation matrix</span>
    with the matrix described by:</p>

    <table class="matrix">
     <tr>
      <td><var>a</var></td>
      <td><var>c</var></td>
      <td><var>e</var></td>
     </tr>
     <tr>
      <td><var>b</var></td>
      <td><var>d</var></td>
      <td><var>f</var></td>
     </tr>
     <tr>
      <td>0</td>
      <td>0</td>
      <td>1</td>
     </tr>
    </table>
   </li>
  </ol>

  </div>

  <p class="note">The arguments <var>a</var>, <var>b</var>, <var>c</var>, <var>d</var>,
  <var>e</var>, and <var>f</var> are sometimes called <var>m11</var>, <var>m12</var>,
  <var>m21</var>, <var>m22</var>, <var>dx</var>, and <var>dy</var> or <var>m11</var>,
  <var>m21</var>, <var>m12</var>, <var>m22</var>, <var>dx</var>, and <var>dy</var>. Care ought to be
  taken in particular with the order of the second and third arguments (<var>b</var> and
  <var>c</var>) as their order varies from API to API and APIs sometimes use the notation
  <var>m12</var>/<var>m21</var> and sometimes <var>m21</var>/<var>m12</var> for those positions.</p>

  <div w-nodev>

  <p>The <dfn method for="CanvasTransform"><code
  data-x="dom-context-2d-getTransform">getTransform()</code></dfn> method, when invoked, must return
  a newly created <code>DOMMatrix</code> representing a copy of the <span>current transformation
  matrix</span> matrix of the context.</p>

  <p class="note">This returned object is not live, so updating it will not affect the
  <span>current transformation matrix</span>, and updating the <span>current transformation
  matrix</span> will not affect an already returned <code>DOMMatrix</code>.</p>

  <p>The <dfn method for="CanvasTransform"><code
  data-x="dom-context-2d-setTransform">setTransform(<var>a</var>, <var>b</var>, <var>c</var>,
  <var>d</var>, <var>e</var>, <var>f</var>)</code></dfn> method, when invoked, must run these
  steps:</p>

  <ol>
   <li><p>If any of the arguments are infinite or NaN, then return.</p></li>

   <li>
    <p>Reset the <span>current transformation matrix</span> to the matrix described by:</p>

    <table class="matrix">
     <tr>
      <td><var>a</var></td>
      <td><var>c</var></td>
      <td><var>e</var></td>
     </tr>
     <tr>
      <td><var>b</var></td>
      <td><var>d</var></td>
      <td><var>f</var></td>
     </tr>
     <tr>
      <td>0</td>
      <td>0</td>
      <td>1</td>
     </tr>
    </table>
   </li>
  </ol>

  <p>The <dfn method for="CanvasTransform"><code
  data-x="dom-context-2d-setTransform-matrix">setTransform(<var>transform</var>)</code></dfn>
  method, when invoked, must run these steps:</p>

  <ol>
   <li><p>Let <var>matrix</var> be the result of <span data-x="create a DOMMatrix from a 2D
   dictionary">creating a <code>DOMMatrix</code> from the 2D dictionary</span>
   <var>transform</var>.</p>

   <li><p>If one or more of <var>matrix</var>'s <span>m11 element</span>, <span>m12 element</span>,
   <span>m21 element</span>, <span>m22 element</span>, <span>m41 element</span>, or <span>m42
   element</span> are infinite or NaN, then return.</p></li>

   <li><p>Reset the <span>current transformation matrix</span> to <var>matrix</var>.</p></li>
  </ol>

  <p>The <dfn method for="CanvasTransform"><code
  data-x="dom-context-2d-resetTransform">resetTransform()</code></dfn> method, when invoked, must
  reset the <span>current transformation matrix</span> to the identity matrix.</p>

  </div>

  <div class="note">
   <p>Given a matrix of the form created by the <code
   data-x="dom-context-2d-transform">transform()</code> and <code
   data-x="dom-context-2d-setTransform">setTransform()</code> methods, i.e.,</p>

   <table class="matrix">
    <tr>
     <td><var>a</var></td>
     <td><var>c</var></td>
     <td><var>e</var></td>
    </tr>
    <tr>
     <td><var>b</var></td>
     <td><var>d</var></td>
     <td><var>f</var></td>
    </tr>
    <tr>
     <td>0</td>
     <td>0</td>
     <td>1</td>
    </tr>
   </table>

   <p>the resulting transformed coordinates after transform matrix multiplication will be</p>

   <p class="equations">
    <var>x</var><sub>new</sub> = <var>a</var> <var>x</var> + <var>c</var> <var>y</var> + <var>e</var><br>
    <var>y</var><sub>new</sub> = <var>b</var> <var>x</var> + <var>d</var> <var>y</var> + <var>f</var>
   </p>
  </div>



  <h6>Image sources for 2D rendering contexts</h6>

  <p>Some methods on the <code>CanvasDrawImage</code> and <code>CanvasFillStrokeStyles</code>
  interfaces take the union type <code>CanvasImageSource</code> as an argument.</p>

  <p>This union type allows objects implementing any of the following interfaces to be used as image
  sources:</p>

  <ul>
   <li><code>HTMLOrSVGImageElement</code> (<code>img</code> or <span>SVG <code>image</code></span>
   elements)</li>

   <li><code>HTMLVideoElement</code> (<code>video</code> elements)</li>

   <li><code>HTMLCanvasElement</code> (<code>canvas</code> elements)</li>

   <li><code>OffscreenCanvas</code></li>

   <li><code>ImageBitmap</code></li>

   <li><code>VideoFrame</code></li>
  </ul>

  <p class="note">Although not formally specified as such, <span>SVG <code>image</code></span>
  elements are expected to be implemented nearly identical to <code>img</code> elements. That is,
  <span>SVG <code>image</code></span> elements share the fundamental concepts and features of
  <code>img</code> elements.</p>

  <p class="note">The <code>ImageBitmap</code> interface can be created from a number of other
  image-representing types, including <code>ImageData</code>.</p>

  <p>To <dfn>check the usability of the <var>image</var> argument</dfn>, where <var>image</var>
  is a <code>CanvasImageSource</code> object, run these steps:</p>

  <ol>
   <li>
    <p>Switch on <var>image</var>:</p>

    <dl class="switch">
     <dt><code>HTMLOrSVGImageElement</code></dt>
     <dd>
      <p>If <var>image</var>'s <span>current request</span>'s <span
      data-x="img-req-state">state</span> is <span data-x="img-error">broken</span>, then
      throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p>

      <p>If <var>image</var> is not <span data-x="img-good">fully decodable</span>,
      then return <i>bad</i>.</p>

      <p>If <var>image</var> has a <span>natural width</span> or <span>natural height</span>
      (or both) equal to zero, then return <i>bad</i>.</p> <!--
      https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2567 -->
     </dd>

     <dt><code>HTMLVideoElement</code></dt>
     <dd><p>If <var>image</var>'s <code data-x="dom-media-readyState">readyState</code>
     attribute is either <code data-x="dom-media-HAVE_NOTHING">HAVE_NOTHING</code> or <code
     data-x="dom-media-HAVE_METADATA">HAVE_METADATA</code>, then return <i>bad</i>.</p></dd>

     <dt><code>HTMLCanvasElement</code></dt>
     <dt><code>OffscreenCanvas</code></dt>
     <dd><p>If <var>image</var> has either a horizontal dimension or a vertical dimension
     equal to zero, then throw an <span>"<code>InvalidStateError</code>"</span>
     <code>DOMException</code>.</p></dd>

     <dt><code>ImageBitmap</code></dt>
     <dt><code>VideoFrame</code></dt>
     <dd><p>If <var>image</var>'s <span>[[Detached]]</span> internal slot value
     is set to true, then throw an <span>"<code>InvalidStateError</code>"</span>
     <code>DOMException</code>.</p></dd>
    </dl>
   </li>

   <li>
    <p>Return <i>good</i>.</p>
   </li>
  </ol>

  <p>When a <code>CanvasImageSource</code> object represents an <code>HTMLOrSVGImageElement</code>,
  the element's image must be used as the source image.</p>

  <p>Specifically, when a <code>CanvasImageSource</code> object represents an animated image in an
  <code>HTMLOrSVGImageElement</code>, the user agent must use the default image of the animation
  (the one that the format defines is to be used when animation is not supported or is disabled),
  or, if there is no such image, the first frame of the animation, when rendering the image for
  <code>CanvasRenderingContext2D</code> APIs.</p>

  <p>When a <code>CanvasImageSource</code> object represents an <code>HTMLVideoElement</code>, then
  the frame at the <span>current playback position</span> when the method with the argument is
  invoked must be used as the source image when rendering the image for
  <code>CanvasRenderingContext2D</code> APIs, and the source image's dimensions must be the <span
  data-x="concept-video-natural-width">natural width</span> and <span
  data-x="concept-video-natural-height">natural height</span> of the <span>media resource</span>
  (i.e., after any aspect-ratio correction has been applied).</p>

  <p>When a <code>CanvasImageSource</code> object represents an <code>HTMLCanvasElement</code>, the
  element's bitmap must be used as the source image.</p>

  <p>When a <code>CanvasImageSource</code> object represents an element that is <span>being
  rendered</span> and that element has been resized, the original image data of the source image
  must be used, not the image as it is rendered (e.g. <code data-x="attr-dim-width">width</code> and
  <code data-x="attr-dim-height">height</code> attributes on the source element have no effect on how
  the object is interpreted when rendering the image for <code>CanvasRenderingContext2D</code>
  APIs).</p>

  <p>When a <code>CanvasImageSource</code> object represents an <code>ImageBitmap</code>, the
  object's bitmap image data must be used as the source image.</p>

  <p>When a <code>CanvasImageSource</code> object represents an <code>OffscreenCanvas</code>, the
  object's bitmap must be used as the source image.</p>

  <p>When a <code>CanvasImageSource</code> object represents a <code>VideoFrame</code>, the object's
  pixel data must be used as the source image, and the source image's dimensions must be the
  object's <span>[[display width]]</span> and <span>[[display height]]</span>.</p>

  <p>An object <var>image</var> <dfn id=the-image-argument-is-not-origin-clean>is not
  origin-clean</dfn> if, switching on <var>image</var>'s type:</p>

  <dl class=switch>
   <dt><code>HTMLOrSVGImageElement</code>
   <dd><p><var>image</var>'s <span>current request</span>'s <span data-x="img-req-data">image
   data</span> is <span>CORS-cross-origin</span>.</p></dd>

   <dt><code>HTMLVideoElement</code>
   <dd><p><var>image</var>'s <span>media data</span> is <span>CORS-cross-origin</span>.</p></dd>

   <dt><code>HTMLCanvasElement</code>
   <dt><code>ImageBitmap</code>
   <dt><code>OffscreenCanvas</code>
   <dd><p><var>image</var>'s bitmap's <span data-x="concept-canvas-origin-clean">origin-clean</span>
   flag is false.</p></dd>
  </dl>



  <h6>Fill and stroke styles</h6>

  <dl class="domintro">
   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-fillStyle">fillStyle</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current style used for filling shapes.</p>

    <p>Can be set, to change the <span data-x="concept-CanvasFillStrokeStyles-fill-style">fill
    style</span>.</p>

    <p>The style can be either a string containing a CSS color, or a <code>CanvasGradient</code> or
    <code>CanvasPattern</code> object. Invalid values are ignored.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-strokeStyle">strokeStyle</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current style used for stroking shapes.</p>

    <p>Can be set, to change the <span data-x="concept-CanvasFillStrokeStyles-stroke-style">stroke
    style.</span></p>

    <p>The style can be either a string containing a CSS color, or a <code>CanvasGradient</code> or
    <code>CanvasPattern</code> object. Invalid values are ignored.</p>
   </dd>
  </dl>

  <div w-nodev>

  <!-- v6 feature requests:

   * Getting and setting colors by component to bypass the CSS value parsing.

     Either:
        context.fillStyle.red += 1;

     Or:
        var array = context.fillStyle;
        array[1] += 1;
        context.fillStyle = array;

   * A more performant way of setting colors in general, e.g.:

       context.setFillColor(r,g,b,a) // already supported by webkit

     Or:

       context.fillStyle = 0xRRGGBBAA; // set a 32bit int directly

  -->

  <p>Objects that implement the <code>CanvasFillStrokeStyles</code> interface have attributes and
  methods (defined in this section) that control how shapes are treated by the object.</p>

  <p>Such objects have associated <dfn data-x="concept-CanvasFillStrokeStyles-fill-style">fill
  style</dfn> and <dfn data-x="concept-CanvasFillStrokeStyles-stroke-style">stroke style</dfn>
  values, which are either CSS colors, <code>CanvasPattern</code>s, or <code>CanvasGradient</code>s.
  Initially, both must be the result of <span data-x="parse a CSS &lt;color> value">parsing</span>
  the string "<code data-x="">#000000</code>".</p>

  <p>When the value is a CSS color, it must not be affected by the transformation matrix when used
  to draw on bitmaps.</p> <!-- so singular matrices don't affect solid color fillStyles -->

  <p class="note">When set to a <code>CanvasPattern</code> or <code>CanvasGradient</code> object,
  changes made to the object after the assignment do affect subsequent stroking or filling of
  shapes.</p>

  <p>The <dfn attribute for="CanvasFillStrokeStyles"><code
  data-x="dom-context-2d-fillStyle">fillStyle</code></dfn> getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="concept-CanvasFillStrokeStyles-fill-style">fill
   style</span> is a CSS color, then return the <span data-x="serialize a CSS &lt;color>
   value">serialization</span> of that color with <span data-x="HTML-compatible serialization is
   requested">HTML-compatible serialization requested</span>.</p></li>

   <li><p>Return <span>this</span>'s <span data-x="concept-CanvasFillStrokeStyles-fill-style">fill
   style</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-context-2d-fillStyle">fillStyle</code> setter steps are:</p>

  <ol>
   <li>
    <p>If the given value is a string, then:</p>

    <ol>
     <li><p>Let <var>context</var> be <span>this</span>'s <code
     data-x="dom-context-2d-canvas">canvas</code> attribute's value, if that is an element;
     otherwise null.</p></li>

     <li><p>Let <var>parsedValue</var> be the result of <span data-x="parse a CSS &lt;color>
     value">parsing</span> the given value with <var>context</var> if non-null.</p></li>

     <li><p>If <var>parsedValue</var> is failure, then return.</p></li>

     <li><p>Set <span>this</span>'s <span data-x="concept-CanvasFillStrokeStyles-fill-style">fill
     style</span> to <var>parsedValue</var>.</p></li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li><p>If the given value is a <code>CanvasPattern</code> object that is marked as <span
   data-x="concept-canvas-pattern-not-origin-clean">not origin-clean</span>, then set
   <span>this</span>'s <span data-x="concept-canvas-origin-clean">origin-clean</span> flag to
   false.</p></li>

   <li><p>Set <span>this</span>'s <span data-x="concept-CanvasFillStrokeStyles-fill-style">fill
   style</span> to the given value.</p></li>
  </ol>

  <p>The <dfn attribute for="CanvasFillStrokeStyles"><code
  data-x="dom-context-2d-strokeStyle">strokeStyle</code></dfn> getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="concept-CanvasFillStrokeStyles-stroke-style">stroke
   style</span> is a CSS color, then return the <span data-x="serialize a CSS &lt;color>
   value">serialization</span> of that color with <span data-x="HTML-compatible serialization is
   requested">HTML-compatible serialization requested</span>.</p></li>

   <li><p>Return <span>this</span>'s <span data-x="concept-CanvasFillStrokeStyles-stroke-style">stroke
   style</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-context-2d-strokeStyle">strokeStyle</code> setter steps are:</p>

  <ol>
   <li>
    <p>If the given value is a string, then:</p>

    <ol>
     <li><p>Let <var>context</var> be <span>this</span>'s <code
     data-x="dom-context-2d-canvas">canvas</code> attribute's value, if that is an element;
     otherwise null.</p></li>

     <li><p>Let <var>parsedValue</var> be the result of <span data-x="parse a CSS &lt;color>
     value">parsing</span> the given value with <var>context</var> if non-null.</p></li>

     <li><p>If <var>parsedValue</var> is failure, then return.</p></li>

     <li><p>Set <span>this</span>'s <span
     data-x="concept-CanvasFillStrokeStyles-stroke-style">stroke style</span> to
     <var>parsedValue</var>.</p></li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li><p>If the given value is a <code>CanvasPattern</code> object that is marked as <span
   data-x="concept-canvas-pattern-not-origin-clean">not origin-clean</span>, then set
   <span>this</span>'s <span data-x="concept-canvas-origin-clean">origin-clean</span> flag to
   false.</p></li>

   <li><p>Set <span>this</span>'s <span data-x="concept-CanvasFillStrokeStyles-stroke-style">stroke
   style</span> to the given value.</p></li>
  </ol>

  </div>

  <hr>

  <p>There are three types of gradients, linear gradients, radial gradients, and conic gradients,
  represented by objects implementing the opaque <code>CanvasGradient</code> interface.</p>

  <p id="interpolation">Once a gradient has been created (see below), stops are placed along it to
  define how the colors are distributed along the gradient. <span w-nodev>The color of the
  gradient at each stop is the color specified for that stop. Between each such stop, the colors and
  the alpha component must be linearly interpolated over the RGBA space without premultiplying the
  alpha value to find the color to use at that offset. Before the first stop, the color must be the
  color of the first stop. After the last stop, the color must be the color of the last stop. When
  there are no stops, the gradient is <span>transparent black</span>.</span></p>

  <dl class="domintro">
   <dt><code data-x=""><var>gradient</var>.<span subdfn data-x="dom-canvasgradient-addColorStop">addColorStop</span>(<var>offset</var>, <var>color</var>)</code></dt>

   <dd>
    <p>Adds a color stop with the given color to the gradient at the given offset. 0.0 is the offset
    at one end of the gradient, 1.0 is the offset at the other end.</p>

    <p>Throws an <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code> if the offset
    is out of range. Throws a <span>"<code>SyntaxError</code>"</span> <code>DOMException</code> if
    the color cannot be parsed.</p>
   </dd>

   <dt><code data-x=""><var>gradient</var> = <var>context</var>.<span subdfn data-x="dom-context-2d-createLinearGradient">createLinearGradient</span>(<var>x0</var>, <var>y0</var>, <var>x1</var>, <var>y1</var>)</code></dt>

   <dd>
    <p>Returns a <code>CanvasGradient</code> object that represents a linear gradient that paints
    along the line given by the coordinates represented by the arguments.</p>
   </dd>

   <dt><code data-x=""><var>gradient</var> = <var>context</var>.<span subdfn data-x="dom-context-2d-createRadialGradient">createRadialGradient</span>(<var>x0</var>, <var>y0</var>, <var>r0</var>, <var>x1</var>, <var>y1</var>, <var>r1</var>)</code></dt>

   <dd>
    <p>Returns a <code>CanvasGradient</code> object that represents a radial gradient that paints
    along the cone given by the circles represented by the arguments.</p>

    <p>If either of the radii are negative, throws an <span>"<code>IndexSizeError</code>"</span>
    <code>DOMException</code> exception.</p>
   </dd>

   <dt><code data-x=""><var>gradient</var> = <var>context</var>.<span subdfn data-x="dom-context-2d-createConicGradient">createConicGradient</span>(<var>startAngle</var>, <var>x</var>, <var>y</var>)</code></dt>

   <dd><p>Returns a <code>CanvasGradient</code> object that represents a conic gradient that paints
   clockwise along the rotation around the center represented by the arguments.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="CanvasGradient"><code
  data-x="dom-canvasgradient-addColorStop">addColorStop(<var>offset</var>,
  <var>color</var>)</code></dfn> method on the <code>CanvasGradient</code>, when invoked, must run
  these steps:</p>

  <ol>
   <li><p>If the <var>offset</var> is less than 0 or greater than 1, then throw an
   <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p></li>

   <li>
    <p>Let <var>parsed color</var> be the result of <span data-x="parse a CSS &lt;color>
    value">parsing</span> <var>color</var>.</p>

    <p class="note">No element is passed to the parser because <code>CanvasGradient</code> objects
    are <code>canvas</code>-neutral &mdash; a <code>CanvasGradient</code> object created by one
    <code>canvas</code> can be used by another, and there is therefore no way to know which is the
    "element in question" at the time that the color is specified.</p>
   </li>

   <li><p>If <var>parsed color</var> is failure, throw a <span>"<code>SyntaxError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li>
    <p>Place a new stop on the gradient, at offset <var>offset</var> relative to the whole gradient,
    and with the color <var>parsed color</var>.</p>

    <p>If multiple stops are added at the same offset on a gradient, then they must be placed in the
    order added, with the first one closest to the start of the gradient, and each subsequent one
    infinitesimally further along towards the end point (in effect causing all but the first and
    last stop added at each point to be ignored).</p>
   </li>
  </ol>

  <p>The <dfn method for="CanvasFillStrokeStyles"><code
  data-x="dom-context-2d-createLinearGradient">createLinearGradient(<var>x0</var>, <var>y0</var>,
  <var>x1</var>, <var>y1</var>)</code></dfn> method takes four arguments that represent the start
  point (<var>x0</var>, <var>y0</var>) and end point (<var>x1</var>, <var>y1</var>) of the gradient.
  The method, when invoked, must return a linear <code>CanvasGradient</code> initialized with the
  specified line.</p>

  <p>Linear gradients must be rendered such that all points on a line perpendicular to the line that
  crosses the start and end points have the color at the point where those two lines cross (with the
  colors coming from the <a href="#interpolation">interpolation and extrapolation</a> described
  above). The points in the linear gradient must be transformed as described by the <span
  data-x="dom-context-2d-transformation">current transformation matrix</span> when rendering.</p>

  <p>If <var>x0</var>&nbsp;=&nbsp;<var>x1</var> and <span
  data-x=""><var>y0</var>&nbsp;=&nbsp;<var>y1</var></span>, then the linear
  gradient must paint nothing.</p>

  <p>The <dfn method for="CanvasFillStrokeStyles"><code
  data-x="dom-context-2d-createRadialGradient">createRadialGradient(<var>x0</var>, <var>y0</var>,
  <var>r0</var>, <var>x1</var>, <var>y1</var>, <var>r1</var>)</code></dfn> method takes six
  arguments, the first three representing the start circle with origin (<var>x0</var>,
  <var>y0</var>) and radius <var>r0</var>, and the last three representing the end circle with
  origin (<var>x1</var>, <var>y1</var>) and radius <var>r1</var>. The values are in coordinate space
  units. If either of <var>r0</var> or <var>r1</var> are negative, then an
  <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code> must be thrown. Otherwise,
  the method, when invoked, must return a radial <code>CanvasGradient</code> initialized with the
  two specified circles.</p>

  <p>Radial gradients must be rendered by following these steps:</p>

  <ol>
   <li><p>If <var>x<sub>0</sub></var>&nbsp;=&nbsp;<var>x<sub>1</sub></var> and <var>y<sub>0</sub></var>&nbsp;=&nbsp;<var>y<sub>1</sub></var> and <var>r<sub>0</sub></var>&nbsp;=&nbsp;<var>r<sub>1</sub></var>, then the radial gradient must
   paint nothing. Return.</p></li>

   <li>
    <p>Let <span data-x="">x(<var>&omega;</var>)&nbsp;=&nbsp;(<var>x<sub>1</sub></var>-<var>x<sub>0</sub></var>)<var>&omega;</var>&nbsp;+&nbsp;<var>x<sub>0</sub></var></span></p>

    <p>Let <span data-x="">y(<var>&omega;</var>)&nbsp;=&nbsp;(<var>y<sub>1</sub></var>-<var>y<sub>0</sub></var>)<var>&omega;</var>&nbsp;+&nbsp;<var>y<sub>0</sub></var></span></p>

    <p>Let <span data-x="">r(<var>&omega;</var>)&nbsp;=&nbsp;(<var>r<sub>1</sub></var>-<var>r<sub>0</sub></var>)<var>&omega;</var>&nbsp;+&nbsp;<var>r<sub>0</sub></var></span></p>

    <p>Let the color at <var>&omega;</var> be the color at that position on the gradient
    (with the colors coming from the <a href="#interpolation">interpolation and extrapolation</a>
    described above).</p>
   </li>

   <li><p>For all values of <var>&omega;</var> where <span data-x="">r(<var>&omega;</var>)&nbsp;&gt;&nbsp;0</span>, starting with the value of <var>&omega;</var> nearest to positive infinity and ending with the value of <var>&omega;</var> nearest to negative infinity, draw the circumference of the circle with
   radius <span data-x="">r(<var>&omega;</var>)</span> at position (<span data-x="">x(<var>&omega;</var>)</span>, <span data-x="">y(<var>&omega;</var>)</span>), with the
   color at <var>&omega;</var>, but only painting on the parts of the bitmap that have not
   yet been painted on by earlier circles in this step for this rendering of the gradient.</p></li>
  </ol>

  <p class="note">This effectively creates a cone, touched by the two circles defined in the
  creation of the gradient, with the part of the cone before the start circle (0.0) using the color
  of the first offset, the part of the cone after the end circle (1.0) using the color of the last
  offset, and areas outside the cone untouched by the gradient (<span>transparent black</span>).</p>

  <p>The resulting radial gradient must then be transformed as described by the <span
  data-x="dom-context-2d-transformation">current transformation matrix</span> when rendering.</p>

  <p>The <dfn method for="CanvasFillStrokeStyles"><code
  data-x="dom-context-2d-createConicGradient">createConicGradient(<var>startAngle</var>,
  <var>x</var>, <var>y</var>)</code></dfn> method takes three arguments, the first argument,
  <var>startAngle</var>, represents the angle in radians at which the gradient begins, and the last
  two arguments, (<var>x</var>, <var>y</var>), represent the center of the gradient in <span
  data-x="'px'">CSS pixels</span>. The method, when invoked, must return a conic
  <code>CanvasGradient</code> initialized with the specified center and angle.</p>

  <p>It follows the same rendering rule as CSS <span>'conic-gradient'</span> and it is
  equivalent to CSS 'conic-gradient(from <var>adjustedStartAngle</var>rad at
  <var>x</var>px <var>y</var>px, <var>angularColorStopList</var>)'. Here:</p>

  <ul>
   <li><p><var>adjustedStartAngle</var> is given by <var>startAngle</var> + π/2;</p></li>

   <li><p><var>angularColorStopList</var> is given by the color stops that have been added to the
   <code>CanvasGradient</code> using <code
   data-x="dom-canvasgradient-addColorStop">addColorStop()</code>, with the color stop offsets
   interpreted as percentages.</p></li>
  </ul>

  <p>Gradients must be painted only where the relevant stroking or filling effects requires that
  they be drawn.</p>

  </div>

  <hr>

  <p>Patterns are represented by objects implementing the opaque <code>CanvasPattern</code>
  interface.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>pattern</var> = <var>context</var>.<span subdfn data-x="dom-context-2d-createPattern">createPattern</span>(<var>image</var>, <var>repetition</var>)</code></dt>

   <dd>
    <p>Returns a <code>CanvasPattern</code> object that uses the given image and repeats in the
    direction(s) given by the <var>repetition</var> argument.</p>

    <p>The allowed values for <var>repetition</var> are <code data-x="">repeat</code> (both
    directions), <code data-x="">repeat-x</code> (horizontal only), <code data-x="">repeat-y</code>
    (vertical only), and <code data-x="">no-repeat</code> (neither). If the <var>repetition</var>
    argument is empty, the value <code data-x="">repeat</code> is used.</p>

    <p>If the image isn't yet fully decoded, then nothing is drawn. If the image is a canvas with no
    data, throws an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p>
   </dd>

   <dt><code data-x=""><var>pattern</var>.<span subdfn data-x="dom-canvaspattern-setTransform">setTransform</span>(<var>transform</var>)</code></dt>

   <dd>
    <p>Sets the transformation matrix that will be used when rendering the pattern during a fill or
    stroke painting operation.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="CanvasFillStrokeStyles"><code
  data-x="dom-context-2d-createPattern">createPattern(<var>image</var>,
  <var>repetition</var>)</code></dfn> method, when invoked, must run these steps:</p>

  <ol>
   <li><p>Let <var>usability</var> be the result of <span data-x="check the usability of the image
   argument">checking the usability of</span> <var>image</var>.</p></li>

   <li><p>If <var>usability</var> is <i>bad</i>, then return null.</p></li>

   <li><p><span>Assert</span>: <var>usability</var> is <i>good</i>.</p></li>

   <li><p>If <var>repetition</var> is the empty string, then set it to "<code
   data-x="">repeat</code>".</p>

   <li><p>If <var>repetition</var> is not <span>identical to</span> one of "<code
   data-x="">repeat</code>", "<code data-x="">repeat-x</code>", "<code data-x="">repeat-y</code>",
   or "<code data-x="">no-repeat</code>", then throw a
   <span>"<code>SyntaxError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>pattern</var> be a new <code>CanvasPattern</code> object with the image
   <var>image</var> and the repetition behavior given by <var>repetition</var>.</p></li>

   <li><p>If <var>image</var> <span>is not origin-clean</span>, then mark <var>pattern</var> as <dfn
   data-x="concept-canvas-pattern-not-origin-clean">not origin-clean</dfn>.</p></li>

   <li><p>Return <var>pattern</var>.</p></li>
  </ol>

  <p>Modifying the <var>image</var> used when creating a <code>CanvasPattern</code> object
  after calling the <code data-x="dom-context-2d-createPattern">createPattern()</code> method must
  not affect the pattern(s) rendered by the <code>CanvasPattern</code> object.</p>

  <p>Patterns have a transformation matrix, which controls how the pattern is used when it is
  painted. Initially, a pattern's transformation matrix must be the identity matrix.</p>

  <p>The <dfn method for="CanvasPattern"><code
  data-x="dom-canvaspattern-setTransform">setTransform(<var>transform</var>)</code></dfn> method,
  when invoked, must run these steps:</p>

  <ol>
   <li><p>Let <var>matrix</var> be the result of <span data-x="create a DOMMatrix from a 2D
   dictionary">creating a <code>DOMMatrix</code> from the 2D dictionary</span>
   <var>transform</var>.</p>

   <li><p>If one or more of <var>matrix</var>'s <span>m11 element</span>, <span>m12 element</span>,
   <span>m21 element</span>, <span>m22 element</span>, <span>m41 element</span>, or <span>m42
   element</span> are infinite or NaN, then return.</p></li>

   <li><p>Reset the pattern's transformation matrix to <var>matrix</var>.</p></li>
  </ol>

  <p>When a pattern is to be rendered within an area, the user agent must run the following steps to
  determine what is rendered:</p>

  <ol>
   <li><p>Create an infinite <span>transparent black</span> bitmap.</p></li>

   <li>
    <p>Place a copy of the image on the bitmap, anchored such that its top left corner is at the
    origin of the coordinate space, with one coordinate space unit per <span data-x="'px'">CSS
    pixel</span> of the image, then place repeated copies of this image horizontally to the left and
    right, if the repetition behavior is "<code data-x="">repeat-x</code>", or vertically up and
    down, if the repetition behavior is "<code data-x="">repeat-y</code>", or in all four
    directions all over the bitmap, if the repetition behavior is "<code
    data-x="">repeat</code>".</p>

    <p>If the original image data is a bitmap image, then the value painted at a point in the area
    of the repetitions is computed by filtering the original image data. When scaling up, if the
    <code data-x="dom-context-2d-imageSmoothingEnabled">imageSmoothingEnabled</code> attribute is
    set to false, then the image must be rendered using nearest-neighbor interpolation. Otherwise,
    the user agent may use any filtering algorithm (for example bilinear interpolation or
    nearest-neighbor). User agents which support multiple filtering algorithms may use the value of
    the <code data-x="dom-context-2d-imageSmoothingQuality">imageSmoothingQuality</code> attribute
    to guide the choice of filtering algorithm. When such a filtering algorithm requires a pixel
    value from outside the original image data, it must instead use the value from wrapping the
    pixel's coordinates to the original image's dimensions. (That is, the filter uses 'repeat'
    behavior, regardless of the value of the pattern's repetition behavior.)</p> <!-- drawImage()
    has similar paragraphs with different rules -->
   </li>

   <li><p>Transform the resulting bitmap according to the pattern's transformation matrix.</p></li>

   <li><p>Transform the resulting bitmap again, this time according to the <span
   data-x="dom-context-2d-transformation">current transformation matrix</span>.</p></li>

   <li><p>Replace any part of the image outside the area in which the pattern is to be rendered with
   <span>transparent black</span>.</p></li>

   <li><p>The resulting bitmap is what is to be rendered, with the same origin and same
   scale.</p></li>
  </ol>

  <hr>

  <p>If a radial gradient or repeated pattern is used when the transformation matrix is singular,
  then the resulting style must be <span>transparent black</span> (otherwise the gradient or pattern
  would be collapsed to a point or line, leaving the other pixels undefined). Linear gradients and
  solid colors always define all points even with singular transformation matrices.</p>

  </div>




  <h6>Drawing rectangles to the bitmap</h6>

  <p>Objects that implement the <code>CanvasRect</code> interface provide the following methods for
  immediately drawing rectangles to the bitmap. The methods each take four arguments; the first two
  give the <var>x</var> and <var>y</var> coordinates of the top left of the rectangle, and the
  second two give the width <var>w</var> and height <var>h</var> of the rectangle, respectively.</p>

  <div w-nodev>

  <p>The <span data-x="dom-context-2d-transformation">current transformation matrix</span> must be
  applied to the following four coordinates, which form the path that must then be closed to get the
  specified rectangle: <span data-x="">(<var>x</var>, <var>y</var>)</span>, <span
  data-x="">(<var>x</var>+<var>w</var>, <var>y</var>)</span>, <span data-x="">(<var>x</var>+<var>w</var>, <var>y</var>+<var>h</var>)</span>, <span data-x="">(<var>x</var>, <var>y</var>+<var>h</var>)</span>.</p>

  <p>Shapes are painted without affecting the <span>current default path</span>, and are subject to
  the <span>clipping region</span>, and, with the exception of <code
  data-x="dom-context-2d-clearRect">clearRect()</code>, also <span data-x="shadows">shadow
  effects</span>, <span data-x="concept-canvas-global-alpha">global alpha</span>, and the
  <span>current compositing and blending operator</span>.</p>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-clearRect">clearRect</span>(<var>x</var>, <var>y</var>, <var>w</var>, <var>h</var>)</code></dt>

   <dd><p>Clears all pixels on the bitmap in the given rectangle to <span>transparent
   black</span>.</p></dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-fillRect">fillRect</span>(<var>x</var>, <var>y</var>, <var>w</var>, <var>h</var>)</code></dt>

   <dd><p>Paints the given rectangle onto the bitmap, using the current fill style.</p></dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-strokeRect">strokeRect</span>(<var>x</var>, <var>y</var>, <var>w</var>, <var>h</var>)</code></dt>

   <dd>
    <p>Paints the box that outlines the given rectangle onto the bitmap, using the current stroke
    style.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="CanvasRect"><code
  data-x="dom-context-2d-clearRect">clearRect(<var>x</var>, <var>y</var>, <var>w</var>,
  <var>h</var>)</code></dfn> method, when invoked, must run these steps:</p>

  <ol>
   <li><p>If any of the arguments are infinite or NaN, then return.</p></li>

   <li><p>Let <var>pixels</var> be the set of pixels in the specified rectangle that also
   intersect the current <span>clipping region</span>.</p></li>

   <li><p>Clear the pixels in <var>pixels</var> to a <span>transparent black</span>, erasing any
   previous image.</p></li>
  </ol>

  <p class="note">If either height or width are zero, this method has no effect, since the set of
  pixels would be empty.</p>

  <p>The <dfn method for="CanvasRect"><code data-x="dom-context-2d-fillRect">fillRect(<var>x</var>,
  <var>y</var>, <var>w</var>, <var>h</var>)</code></dfn> method, when invoked, must run these
  steps:</p>

  <ol>
   <li><p>If any of the arguments are infinite or NaN, then return.</p></li>

   <li><p>If either <var>w</var> or <var>h</var> are zero, then return.</p></li>

   <li><p>Paint the specified rectangular area using <span>this</span>'s <span
   data-x="concept-CanvasFillStrokeStyles-fill-style">fill style</span>.</p></li>
  </ol>

  <p>The <dfn method for="CanvasRect"><code
  data-x="dom-context-2d-strokeRect">strokeRect(<var>x</var>, <var>y</var>, <var>w</var>,
  <var>h</var>)</code></dfn> method, when invoked, must run these steps:</p>

  <ol>
   <li><p>If any of the arguments are infinite or NaN, then return.</p></li>

   <li><p>Take the result of <span data-x="trace a path">tracing the path</span> described below,
   using the <code>CanvasPathDrawingStyles</code> interface's line styles, and fill it with
   <span>this</span>'s <span data-x="concept-CanvasFillStrokeStyles-stroke-style">stroke
   style</span>.</p></li>
  </ol>

  <p>If both <var>w</var> and <var>h</var> are zero, the path has a single subpath
  with just one point (<var>x</var>, <var>y</var>), and no lines, and this method
  thus has no effect (the <span>trace a path</span> algorithm returns an empty path in that
  case).</p>

  <p>If just one of either <var>w</var> or <var>h</var> is zero, then the path has
  a single subpath consisting of two points, with coordinates (<var>x</var>, <var>y</var>) and (<var>x</var>+<var>w</var>, <var>y</var>+<var>h</var>), in that order, connected by a single straight line.</p>

  <p>Otherwise, the path has a single subpath consisting of four points, with coordinates (<span
  data-x=""><var>x</var></span>, <var>y</var>), (<span
  data-x=""><var>x</var>+<var>w</var></span>, <var>y</var>), (<var>x</var>+<var>w</var>,
  <var>y</var>+<var>h</var>), and (<var>x</var>, <var>y</var>+<var>h</var>),
  connected to each other in that order by straight lines.</p>

  </div>



  <h6>Drawing text to the bitmap</h6>

  <dl class="domintro" id="text-0">

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-fillText">fillText</span>(<var>text</var>, <var>x</var>, <var>y</var> [, <var>maxWidth</var> ])</code></dt>
   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-strokeText">strokeText</span>(<var>text</var>, <var>x</var>, <var>y</var> [, <var>maxWidth</var> ])</code></dt>

   <dd>
    <p>Fills or strokes (respectively) the given text at the given position. If a maximum width is
    provided, the text will be scaled to fit that width if necessary.</p>
   </dd>

   <dt><code data-x=""><var>metrics</var> = <var>context</var>.<span subdfn data-x="dom-context-2d-measureText">measureText</span>(<var>text</var>)</code></dt>

   <dd>
    <p>Returns a <code>TextMetrics</code> object with the metrics of the given text in the current
    font.</p>
   </dd>

   <dt><code data-x=""><var>metrics</var>.<span subdfn data-x="dom-textmetrics-width">width</span></code></dt>
   <dt><code data-x=""><var>metrics</var>.<span subdfn data-x="dom-textmetrics-actualBoundingBoxLeft">actualBoundingBoxLeft</span></code></dt>
   <dt><code data-x=""><var>metrics</var>.<span subdfn data-x="dom-textmetrics-actualBoundingBoxRight">actualBoundingBoxRight</span></code></dt>
   <dt><code data-x=""><var>metrics</var>.<span subdfn data-x="dom-textmetrics-fontBoundingBoxAscent">fontBoundingBoxAscent</span></code></dt>
   <dt><code data-x=""><var>metrics</var>.<span subdfn data-x="dom-textmetrics-fontBoundingBoxDescent">fontBoundingBoxDescent</span></code></dt>
   <dt><code data-x=""><var>metrics</var>.<span subdfn data-x="dom-textmetrics-actualBoundingBoxAscent">actualBoundingBoxAscent</span></code></dt>
   <dt><code data-x=""><var>metrics</var>.<span subdfn data-x="dom-textmetrics-actualBoundingBoxDescent">actualBoundingBoxDescent</span></code></dt>
   <dt><code data-x=""><var>metrics</var>.<span subdfn data-x="dom-textmetrics-emHeightAscent">emHeightAscent</span></code></dt>
   <dt><code data-x=""><var>metrics</var>.<span subdfn data-x="dom-textmetrics-emHeightDescent">emHeightDescent</span></code></dt>
   <dt><code data-x=""><var>metrics</var>.<span subdfn data-x="dom-textmetrics-hangingBaseline">hangingBaseline</span></code></dt>
   <dt><code data-x=""><var>metrics</var>.<span subdfn data-x="dom-textmetrics-alphabeticBaseline">alphabeticBaseline</span></code></dt>
   <dt><code data-x=""><var>metrics</var>.<span subdfn data-x="dom-textmetrics-ideographicBaseline">ideographicBaseline</span></code></dt>

   <dd><p>Returns the measurement described below.</p></dd>
  </dl>

  <div w-nodev>

  <p>Objects that implement the <code>CanvasText</code> interface provide the following methods for
  rendering text.</p>

  <p>The <dfn method for="CanvasText"><code
  data-x="dom-context-2d-fillText">fillText(<var>text</var>, <var>x</var>, <var>y</var>,
  <var>maxWidth</var>)</code></dfn> and <dfn method for="CanvasText"><code
  data-x="dom-context-2d-strokeText">strokeText(<var>text</var>, <var>x</var>, <var>y</var>,
  <var>maxWidth</var>)</code></dfn> methods render the given
  <var>text</var> at the given (<var>x</var>, <var>y</var>) coordinates ensuring that the text isn't
  wider than <var>maxWidth</var> if specified, using the current <code
  data-x="dom-context-2d-font">font</code>, <code
  data-x="dom-context-2d-textAlign">textAlign</code>, and <code
  data-x="dom-context-2d-textBaseline">textBaseline</code> values. Specifically, when the methods
  are invoked, the user agent must run these steps:</p>

  <ol>
   <li><p>If any of the arguments are infinite or NaN, then return.</p></li>

   <li><p>Run the <span>text preparation algorithm</span>, passing it <var>text</var>, the object
   implementing the <code>CanvasText</code> interface, and, if the <var>maxWidth</var> argument was
   provided, that argument. Let <var>glyphs</var> be the result.</p>

   <li><p>Move all the shapes in <var>glyphs</var> to the right by <var>x</var>
   <span data-x="'px'">CSS pixels</span> and down by <var>y</var> <span data-x="'px'">CSS
   pixels</span>.</p></li>

   <li>
    <p>Paint the shapes given in <var>glyphs</var>, as transformed by the <span
    data-x="dom-context-2d-transformation">current transformation matrix</span>, with each <span
    data-x="'px'">CSS pixel</span> in the coordinate space of <var>glyphs</var> mapped to one
    coordinate space unit.</p>

    <p>For <code data-x="dom-context-2d-fillText">fillText()</code>, <span>this</span>'s <span
    data-x="concept-CanvasFillStrokeStyles-fill-style">fill style</span> must be applied to the
    shapes and <span>this</span>'s <span data-x="concept-CanvasFillStrokeStyles-stroke-style">stroke
    style</span> must be ignored. For <code data-x="dom-context-2d-strokeText">strokeText()</code>,
    the reverse holds: <span>this</span>'s <span
    data-x="concept-CanvasFillStrokeStyles-stroke-style">stroke style</span> must be applied to the
    result of <span data-x="trace a path">tracing</span> the shapes using the object implementing
    the <code>CanvasText</code> interface for the line styles, and <span>this</span>'s <span
    data-x="concept-CanvasFillStrokeStyles-fill-style">fill style</span> must be ignored.</p>

    <p>These shapes are painted without affecting the current path, and are subject to <span
    data-x="shadows">shadow effects</span>, <span data-x="concept-canvas-global-alpha">global
    alpha</span>, the <span>clipping region</span>, and the <span>current compositing and blending
    operator</span>.</p>
   </li>
  </ol>

  <p>
  <!--INSERT TRACKING-->
  The <dfn method for="CanvasText"><code
  data-x="dom-context-2d-measureText">measureText(<var>text</var>)</code></dfn> method steps are to
  run the <span>text preparation algorithm</span>, passing it <var>text</var> and the object
  implementing the <code>CanvasText</code> interface, and then using the returned <span>inline
  box</span> must return a new <code>TextMetrics</code> object with members behaving as described in
  the following list: <ref>CSS</ref></p>

  </div>

  <dl>
   <dt><dfn attribute for="TextMetrics"><code data-x="dom-textmetrics-width">width</code></dfn>
   attribute</dt>

   <dd><p>The width of that <span>inline box</span>, in <span data-x="'px'">CSS pixels</span>. (The
   text's advance width.)</p></dd>

   <dt><dfn attribute for="TextMetrics"><code data-x="dom-textmetrics-actualBoundingBoxLeft">actualBoundingBoxLeft</code></dfn> attribute</dt>

   <dd>
    <p>The distance parallel to the baseline from the alignment point given by the <code
    data-x="dom-context-2d-textAlign">textAlign</code> attribute to the left side of the bounding
    rectangle of the given text, in <span data-x="'px'">CSS pixels</span>; positive numbers
    indicating a distance going left from the given alignment point.

    <p class="note">The sum of this value and the next (<code
    data-x="dom-textmetrics-actualBoundingBoxRight">actualBoundingBoxRight</code>) can be wider than
    the width of the <span>inline box</span> (<code data-x="dom-textmetrics-width">width</code>), in
    particular with slanted fonts where characters overhang their advance width.</p>
   </dd>

   <dt><dfn attribute for="TextMetrics"><code data-x="dom-textmetrics-actualBoundingBoxRight">actualBoundingBoxRight</code></dfn> attribute</dt>

   <dd>
    <p>The distance parallel to the baseline from the alignment point given by the <code
    data-x="dom-context-2d-textAlign">textAlign</code> attribute to the right side of the bounding
    rectangle of the given text, in <span data-x="'px'">CSS pixels</span>; positive numbers
    indicating a distance going right from the given alignment point.</p>
   </dd>

   <dt><dfn attribute for="TextMetrics"><code data-x="dom-textmetrics-fontBoundingBoxAscent">fontBoundingBoxAscent</code></dfn> attribute</dt>

   <dd>
    <p>The distance from the horizontal line indicated by the <code
    data-x="dom-context-2d-textBaseline">textBaseline</code> attribute to the <span>ascent
    metric</span> of the <span>first available font</span>, in <span data-x="'px'">CSS
    pixels</span>; positive numbers indicating a distance going up from the given baseline.</p>

    <p class="note">This value and the next are useful when rendering a background that have to have
    a consistent height even if the exact text being rendered changes. The <code
    data-x="dom-textmetrics-actualBoundingBoxAscent">actualBoundingBoxAscent</code> attribute (and
    its corresponding attribute for the descent) are useful when drawing a bounding box around
    specific text.</p>
   </dd>

   <dt><dfn attribute for="TextMetrics"><code data-x="dom-textmetrics-fontBoundingBoxDescent">fontBoundingBoxDescent</code></dfn> attribute</dt>

   <dd><p>The distance from the horizontal line indicated by the <code
   data-x="dom-context-2d-textBaseline">textBaseline</code> attribute to the <span>descent
   metric</span> of the <span>first available font</span>, in <span data-x="'px'">CSS pixels</span>;
   positive numbers indicating a distance going down from the given baseline.</p></dd>

   <dt><dfn attribute for="TextMetrics"><code data-x="dom-textmetrics-actualBoundingBoxAscent">actualBoundingBoxAscent</code></dfn> attribute</dt>

   <dd>
    <p>The distance from the horizontal line indicated by the <code
    data-x="dom-context-2d-textBaseline">textBaseline</code> attribute to the top of the bounding
    rectangle of the given text, in <span data-x="'px'">CSS pixels</span>; positive numbers
    indicating a distance going up from the given baseline.

    <p class="note">This number can vary greatly based on the input text, even if the first font
    specified covers all the characters in the input. For example, the <code
    data-x="dom-textmetrics-actualBoundingBoxAscent">actualBoundingBoxAscent</code> of a lowercase
    "o" from an <span>alphabetic baseline</span> would be less than that of an uppercase "F". The
    value can easily be negative; for example, the distance from the top of the em box (<code
    data-x="dom-context-2d-textBaseline">textBaseline</code> value "<code
    data-x="dom-context-2d-textBaseline-top">top</code>") to the top of the bounding rectangle when
    the given text is just a single comma "<code data-x="">,</code>" would likely (unless the font is
    quite unusual) be negative.</p>
   </dd>

   <dt><dfn attribute for="TextMetrics"><code data-x="dom-textmetrics-actualBoundingBoxDescent">actualBoundingBoxDescent</code></dfn> attribute</dt>

   <dd><p>The distance from the horizontal line indicated by the <code
   data-x="dom-context-2d-textBaseline">textBaseline</code> attribute to the bottom of the bounding
   rectangle of the given text, in <span data-x="'px'">CSS pixels</span>; positive numbers
   indicating a distance going down from the given baseline.</p></dd>

   <dt><dfn attribute for="TextMetrics"><code data-x="dom-textmetrics-emHeightAscent">emHeightAscent</code></dfn> attribute</dt>

   <dd><p>The distance from the horizontal line indicated by the <code
   data-x="dom-context-2d-textBaseline">textBaseline</code> attribute to the <span>em-over
   baseline</span> in the <span>inline box</span>, in <span data-x="'px'">CSS pixels</span>;
   positive numbers indicating that the given baseline is below the <span>em-over baseline</span>
   (so this value will usually be positive). Zero if the given baseline is the <span>em-over
   baseline</span>; half the font size if the given baseline is halfway between the <span>em-over
   baseline</span> and the <span>em-under baseline</span>.</p></dd>

   <dt><dfn attribute for="TextMetrics"><code data-x="dom-textmetrics-emHeightDescent">emHeightDescent</code></dfn> attribute</dt>

   <dd><p>The distance from the horizontal line indicated by the <code
   data-x="dom-context-2d-textBaseline">textBaseline</code> attribute to the <span>em-under
   baseline</span> in the <span>inline box</span>, in <span data-x="'px'">CSS pixels</span>;
   positive numbers indicating that the given baseline is above the <span>em-under baseline</span>.
   (Zero if the given baseline is the <span>em-under baseline</span>.)</p></dd>

   <dt><dfn attribute for="TextMetrics"><code data-x="dom-textmetrics-hangingBaseline">hangingBaseline</code></dfn> attribute</dt>

   <dd><p>The distance from the horizontal line indicated by the <code
   data-x="dom-context-2d-textBaseline">textBaseline</code> attribute to the <span>hanging
   baseline</span> of the <span>inline box</span>, in <span data-x="'px'">CSS pixels</span>;
   positive numbers indicating that the given baseline is below the <span>hanging baseline</span>.
   (Zero if the given baseline is the <span>hanging baseline</span>.)</p></dd>

   <dt><dfn attribute for="TextMetrics"><code data-x="dom-textmetrics-alphabeticBaseline">alphabeticBaseline</code></dfn> attribute</dt>

   <dd><p>The distance from the horizontal line indicated by the <code
   data-x="dom-context-2d-textBaseline">textBaseline</code> attribute to the <span>alphabetic
   baseline</span> of the <span>inline box</span>, in <span data-x="'px'">CSS pixels</span>;
   positive numbers indicating that the given baseline is below the <span>alphabetic
   baseline</span>. (Zero if the given baseline is the <span>alphabetic baseline</span>.)</p></dd>

   <dt><dfn attribute for="TextMetrics"><code data-x="dom-textmetrics-ideographicBaseline">ideographicBaseline</code></dfn> attribute</dt>

   <dd><p>The distance from the horizontal line indicated by the <code
   data-x="dom-context-2d-textBaseline">textBaseline</code> attribute to the <span>ideographic-under
   baseline</span> of the <span>inline box</span>, in <span data-x="'px'">CSS pixels</span>;
   positive numbers indicating that the given baseline is below the <span>ideographic-under
   baseline</span>. (Zero if the given baseline is the <span>ideographic-under
   baseline</span>.)</p></dd>
  </dl>

  <p class="note">Glyphs rendered using <code data-x="dom-context-2d-fillText">fillText()</code> and
  <code data-x="dom-context-2d-strokeText">strokeText()</code> can spill out of the box given by the
  font size and the width returned by <code data-x="dom-context-2d-measureText">measureText()</code>
  (the text width). Authors are encouraged to use the bounding box values described above if this is
  an issue.</p>

  <p class="note">A future version of the 2D context API might provide a way to render fragments of
  documents, rendered using CSS, straight to the canvas. This would be provided in preference to a
  dedicated way of doing multiline layout.</p>



  <h6>Drawing paths to the canvas</h6>

  <p>Objects that implement the <code>CanvasDrawPath</code> interface have a <dfn>current default
  path</dfn>. There is only one <span>current default path</span>, it is not part of the
  <span>drawing state</span>. The <span>current default path</span> is a <span
  data-x="concept-path">path</span>, as described above.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-beginPath">beginPath</span>()</code></dt>

   <dd><p>Resets the <span>current default path</span>.</p></dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-fill">fill</span>([ <var>fillRule</var> ])</code></dt>
   <dt><code data-x=""><var>context</var>.<span data-x="dom-context-2d-fill">fill</span>(<var>path</var> [, <var>fillRule</var> ])</code></dt>

   <dd>
    <p>Fills the subpaths of the <span>current default path</span> or the given path with the
    current fill style, obeying the given fill rule.</p>
   </dd>


   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-stroke">stroke</span>()</code></dt>
   <dt><code data-x=""><var>context</var>.<span data-x="dom-context-2d-stroke">stroke</span>(<var>path</var>)</code></dt>

   <dd>
    <p>Strokes the subpaths of the <span>current default path</span> or the given path with the
    current stroke style.</p>
   </dd>


   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-clip">clip</span>([ <var>fillRule</var> ])</code></dt>
   <dt><code data-x=""><var>context</var>.<span data-x="dom-context-2d-clip">clip</span>(<var>path</var> [, <var>fillRule</var> ])</code></dt>

   <dd>
    <p>Further constrains the clipping region to the <span>current default path</span> or the given
    path, using the given fill rule to determine what points are in the path.</p>
   </dd>


   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-isPointInPath">isPointInPath</span>(<var>x</var>, <var>y</var> [, <var>fillRule</var> ])</code></dt>
   <dt><code data-x=""><var>context</var>.<span data-x="dom-context-2d-isPointInPath">isPointInPath</span>(<var>path</var>, <var>x</var>, <var>y</var> [, <var>fillRule</var> ])</code></dt>

   <dd>
    <p>Returns true if the given point is in the <span>current default path</span> or the given
    path, using the given fill rule to determine what points are in the path.</p>
   </dd>


   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-isPointInStroke">isPointInStroke</span>(<var>x</var>, <var>y</var>)</code></dt>
   <dt><code data-x=""><var>context</var>.<span data-x="dom-context-2d-isPointInStroke">isPointInStroke</span>(<var>path</var>, <var>x</var>, <var>y</var>)</code></dt>

   <dd>
    <p>Returns true if the given point would be in the region covered by the stroke of the
    <span>current default path</span> or the given path, given the current stroke style.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="CanvasDrawPath"><code
  data-x="dom-context-2d-beginPath">beginPath()</code></dfn> method steps are to empty the list of
  subpaths in <span>this</span>'s <span>current default path</span> so that it once again has zero
  subpaths.</p>

  <p>Where the following method definitions use the term <dfn>intended path</dfn> for a
  <code>Path2D</code>-or-null <var>path</var>, it means <var>path</var> itself if it is a
  <code>Path2D</code> object, or the <span>current default path</span> otherwise.</p>

  <p>When the <span>intended path</span> is a <code>Path2D</code> object, the coordinates and lines
  of its subpaths must be transformed according to the <span
  data-x="dom-context-2d-transformation"> current transformation matrix</span> on the object
  implementing the <code>CanvasTransform</code> interface when used by these methods (without
  affecting the <code>Path2D</code> object itself). When the intended path is the <span>current
  default path</span>, it is not affected by the transform. (This is because transformations
  already affect the <span>current default path</span> when it is constructed, so applying it when
  it is painted as well would result in a double transformation.)</p>

  <p>The <dfn method for="CanvasDrawPath"><code
  data-x="dom-context-2d-fill">fill(<var>fillRule</var>)</code></dfn> method steps are to run the
  <span>fill steps</span> given <span>this</span>, null, and <var>fillRule</var>.</p>

  <p>The <dfn method for="CanvasDrawPath"><code
  data-x="dom-context-2d-fill-path">fill(<var>path</var>, <var>fillRule</var>)</code></dfn> method
  steps are to run the <span>fill steps</span> given <span>this</span>, <var>path</var>, and
  <var>fillRule</var>.</p>

  <p>The <dfn>fill steps</dfn>, given a <code>CanvasDrawPath</code> <var>context</var>, a
  <code>Path2D</code>-or-null <var>path</var>, and a <span>fill rule</span> <var>fillRule</var>,
  are to fill all the subpaths of the <span>intended path</span> for <var>path</var>, using
  <var>context</var>'s <span data-x="concept-CanvasFillStrokeStyles-fill-style">fill style</span>,
  and using the <span>fill rule</span> indicated by <var>fillRule</var>. Open subpaths must be
  implicitly closed when being filled (without affecting the actual subpaths).</p>

  <p>The <dfn method for="CanvasDrawPath"><code
  data-x="dom-context-2d-stroke">stroke()</code></dfn> method steps are to run the <span>stroke
  steps</span> given <span>this</span> and null.</p>

  <p>The <dfn method for="CanvasDrawPath"><code
  data-x="dom-context-2d-stroke-path">stroke(<var>path</var>)</code></dfn> method steps are to run
  the <span>stroke steps</span> given <span>this</span> and <var>path</var>.</p>

  <p>The <dfn>stroke steps</dfn>, given a <code>CanvasDrawPath</code> <var>context</var> and a
  <code>Path2D</code>-or-null <var>path</var>, are to <span data-x="trace a path">trace</span> the
  <span>intended path</span> for <var>path</var>, using <var>context</var>'s line styles as set by
  its <code>CanvasPathDrawingStyles</code> mixin, and then fill the resulting path using
  <var>context</var>'s <span data-x="concept-CanvasFillStrokeStyles-stroke-style">stroke
  style</span>, using the <span data-x="dom-context-2d-fillRule-nonzero">nonzero winding
  rule</span>.</p>

  <p class="note">As a result of how the algorithm to <span>trace a path</span> is defined,
  overlapping parts of the paths in one stroke operation are treated as if their union was what was
  painted.</p>

  <p class="note">The stroke <em>style</em> is affected by the transformation during painting, even
  if the <span>current default path</span> is used.</p>

  <p>Paths, when filled or stroked, must be painted without affecting the <span>current default
  path</span> or any <code>Path2D</code> objects, and must be subject to <span
  data-x="shadows">shadow effects</span>, <span data-x="concept-canvas-global-alpha">global
  alpha</span>, the <span>clipping region</span>, and the <span>current compositing and blending
  operator</span>. (The effect of transformations is described above and varies based on which path
  is being used.)</p>

  <hr>

  <p>The <dfn method for="CanvasDrawPath"><code
  data-x="dom-context-2d-clip">clip(<var>fillRule</var>)</code></dfn> method steps are to run the
  <span>clip steps</span> given <span>this</span>, null, and <var>fillRule</var>.</p>

  <p>The <dfn method for="CanvasDrawPath"><code
  data-x="dom-context-2d-clip-path">clip(<var>path</var>, <var>fillRule</var>)</code></dfn> method
  steps are to run the <span>clip steps</span> given <span>this</span>, <var>path</var>, and
  <var>fillRule</var>.</p>

  <p>The <dfn>clip steps</dfn>, given a <code>CanvasDrawPath</code> <var>context</var>, a
  <code>Path2D</code>-or-null <var>path</var>, and a <span>fill rule</span> <var>fillRule</var>,
  are to create a new <dfn>clipping region</dfn> by calculating the intersection of
  <var>context</var>'s current clipping region and the area described by the <span>intended
  path</span> for <var>path</var>, using the <span>fill rule</span> indicated by
  <var>fillRule</var>. Open subpaths must be implicitly closed when computing the clipping region,
  without affecting the actual subpaths. The new clipping region replaces the current clipping
  region.</p>

  <p>When the context is initialized, its current clipping region must be set to the largest
  infinite surface (i.e. by default, no clipping occurs).</p>

  <!-- v6
   Jordan OSETE suggests:
    * support ways of extending the clipping region (union instead of intersection)
       - also "add", "subtract", "replace", "intersect" and "xor"
       - maybe just support creating unions, intersections, and other -ions of paths
  -->

  <hr>

  <p>The <dfn method for="CanvasDrawPath"><code
  data-x="dom-context-2d-isPointInPath">isPointInPath(<var>x</var>, <var>y</var>,
  <var>fillRule</var>)</code></dfn> method steps are to return the result of the <span>is point in
  path steps</span> given <span>this</span>, null, <var>x</var>, <var>y</var>, and
  <var>fillRule</var>.</p>

  <p>The <dfn method for="CanvasDrawPath"><code
  data-x="dom-context-2d-isPointInPath-path">isPointInPath(<var>path</var>, <var>x</var>,
  <var>y</var>, <var>fillRule</var>)</code></dfn> method steps are to return the result of the
  <span>is point in path steps</span> given <span>this</span>, <var>path</var>, <var>x</var>, <var>y</var>,
  and <var>fillRule</var>.</p>

  <p>The <dfn>is point in path steps</dfn>, given a <code>CanvasDrawPath</code> <var>context</var>,
  a <code>Path2D</code>-or-null <var>path</var>, two numbers <var>x</var> and <var>y</var>, and a
  <span>fill rule</span> <var>fillRule</var>, are:</p>

  <ol>
   <li><p>If <var>x</var> or <var>y</var> are infinite or NaN, then return false.</p></li>

   <li><p>If the point given by the <var>x</var> and <var>y</var> coordinates, when treated as
   coordinates in the canvas coordinate space unaffected by the current transformation, is inside
   the <span>intended path</span> for <var>path</var> as determined by the <span>fill rule</span>
   indicated by <var>fillRule</var>, then return true. Open subpaths must be implicitly closed when
   computing the area inside the path, without affecting the actual subpaths. Points on the path
   itself must be considered to be inside the path.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  <hr>

  <p>The <dfn method for="CanvasDrawPath"><code
  data-x="dom-context-2d-isPointInStroke">isPointInStroke(<var>x</var>, <var>y</var>)</code></dfn>
  method steps are to return the result of the <span>is point in stroke steps</span> given
  <span>this</span>, null, <var>x</var>, and <var>y</var>.</p>

  <p>The <dfn method for="CanvasDrawPath"><code
  data-x="dom-context-2d-isPointInStroke-path">isPointInStroke(<var>path</var>, <var>x</var>,
  <var>y</var>)</code></dfn> method steps are to return the result of the <span>is point in stroke
  steps</span> given <span>this</span>, <var>path</var>, <var>x</var>, and <var>y</var>.</p>

  <p>The <dfn>is point in stroke steps</dfn>, given a <code>CanvasDrawPath</code>
  <var>context</var>, a <code>Path2D</code>-or-null <var>path</var>, and two numbers <var>x</var>
  and <var>y</var>, are:</p>

  <ol>
   <li><p>If <var>x</var> or <var>y</var> are infinite or NaN, then return false.</p></li>

   <li><p>If the point given by the <var>x</var> and <var>y</var> coordinates, when treated as
   coordinates in the canvas coordinate space unaffected by the current transformation, is inside
   the path that results from <span data-x="trace a path">tracing</span> the <span>intended
   path</span> for <var>path</var>, using the <span
   data-x="dom-context-2d-fillRule-nonzero">nonzero winding rule</span>, and using
   <var>context</var>'s line styles as set by its <code>CanvasPathDrawingStyles</code> mixin, then
   return true. Points on the resulting path must be considered to be inside the path.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  <hr>

  </div>

  <div class="example" id="drawCustomFocusRingExample">

   <p>This <code>canvas</code> element has a couple of checkboxes. The path-related commands are
   highlighted:</p>

   <pre><code class="html">&lt;canvas height=400 width=750>
 &lt;label>&lt;input type=checkbox id=showA> Show As&lt;/label>
 &lt;label>&lt;input type=checkbox id=showB> Show Bs&lt;/label>
 &lt;!-- ... -->
&lt;/canvas>
&lt;script>
 function drawCheckbox(context, element, x, y, paint) {
   context.save();
   context.font = '10px sans-serif';
   context.textAlign = 'left';
   context.textBaseline = 'middle';
   var metrics = context.measureText(element.labels[0].textContent);
   if (paint) {
<strong>     context.beginPath();
     context.strokeStyle = 'black';
     context.rect(x-5, y-5, 10, 10);
     context.stroke();
</strong>     if (element.checked) {
<strong>       context.fillStyle = 'black';
       context.fill();
</strong>     }
     context.fillText(element.labels[0].textContent, x+5, y);
   }
<strong>   context.beginPath();
   context.rect(x-7, y-7, 12 + metrics.width+2, 14);
</strong>
   context.drawFocusIfNeeded(element);
   context.restore();
 }
 function drawBase() { /* ... */ }
 function drawAs() { /* ... */ }
 function drawBs() { /* ... */ }
 function redraw() {
   var canvas = document.getElementsByTagName('canvas')[0];
   var context = canvas.getContext('2d');
   context.clearRect(0, 0, canvas.width, canvas.height);
   drawCheckbox(context, document.getElementById('showA'), 20, 40, true);
   drawCheckbox(context, document.getElementById('showB'), 20, 60, true);
   drawBase();
   if (document.getElementById('showA').checked)
     drawAs();
   if (document.getElementById('showB').checked)
     drawBs();
 }
 function processClick(event) {
   var canvas = document.getElementsByTagName('canvas')[0];
   var context = canvas.getContext('2d');
   var x = event.clientX;
   var y = event.clientY;
   var node = event.target;
   while (node) {
     x -= node.offsetLeft - node.scrollLeft;
     y -= node.offsetTop - node.scrollTop;
     node = node.offsetParent;
   }
   drawCheckbox(context, document.getElementById('showA'), 20, 40, false);
   if (<strong>context.isPointInPath(x, y)</strong>)
     document.getElementById('showA').checked = !(document.getElementById('showA').checked);
   drawCheckbox(context, document.getElementById('showB'), 20, 60, false);
   if (<strong>context.isPointInPath(x, y)</strong>)
     document.getElementById('showB').checked = !(document.getElementById('showB').checked);
   redraw();
 }
 document.getElementsByTagName('canvas')[0].addEventListener('focus', redraw, true);
 document.getElementsByTagName('canvas')[0].addEventListener('blur', redraw, true);
 document.getElementsByTagName('canvas')[0].addEventListener('change', redraw, true);
 document.getElementsByTagName('canvas')[0].addEventListener('click', processClick, false);
 redraw();
&lt;/script></code></pre>
<!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=340 -->

  </div>

  <h6 id="drawing-focus-rings-and-scrolling-paths-into-view">Drawing focus rings</h6>

  <dl class="domintro">
   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-drawFocusIfNeeded">drawFocusIfNeeded</span>(<var>element</var>)</code></dt>
   <dd>
    <p>If <var>element</var> is <span>focused</span>, draws a focus ring around the <span>current
    default path</span>, following the platform conventions for focus rings.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-drawFocusIfNeeded-path-element">drawFocusIfNeeded</span>(<var>path</var>, <var>element</var>)</code></dt>
   <dd>
    <p>If <var>element</var> is <span>focused</span>, draws a focus ring around <var>path</var>,
    following the platform conventions for focus rings.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>Objects that implement the <code>CanvasUserInterface</code> interface provide the following
  methods to draw focus rings.</p>

  <p id="dom-context-2d-drawosfocusring">The <dfn method for="CanvasUserInterface"><code
  data-x="dom-context-2d-drawFocusIfNeeded">drawFocusIfNeeded(<var>element</var>)</code></dfn>
  method steps are to <span>draw focus if needed</span> given <span>this</span>,
  <var>element</var>, and <span>this</span>'s <span>current default path</span>.</p>

  <!--
   This treats a Path2D as a concept-path. Ideally Path2Ds would have an internal slot holding
   a concept-path, but for now they don't...
  -->
  <p>The <dfn method for="CanvasUserInterface"><code
  data-x="dom-context-2d-drawFocusIfNeeded-path-element">drawFocusIfNeeded(<var>path</var>,
  <var>element</var>)</code></dfn> method steps are to <span>draw focus if needed</span> given
  <span>this</span>, <var>element</var>, and <var>path</var>.</p>

  <p>To <dfn>draw focus if needed</dfn>, given an object implementing
  <code>CanvasUserInterface</code> <var>context</var>, an element <var>element</var>, and a <span
  data-x="concept-path">path</span> <var>path</var>:</p>

  <ol>
   <li><p>If <var>element</var> is not <span>focused</span> or is not a descendant of
   <var>context</var>'s <code>canvas</code> element, then return.</p></li>

   <li>
    <p>Draw a focus ring of the appropriate style along <var>path</var>, following platform
    conventions.</p>

    <p class="note">Some platforms only draw focus rings around elements that have been focused from
    the keyboard, and not those focused from the mouse. Other platforms simply don't draw focus
    rings around some elements at all unless relevant accessibility features are enabled. This API
    is intended to follow these conventions. User agents that implement distinctions based on the
    manner in which the element was focused are encouraged to classify focus driven by the <code
    data-x="dom-focus">focus()</code> method based on the kind of user interaction event from which
    the call was triggered (if any).</p>

    <p>The focus ring should not be subject to the <span data-x="shadows">shadow effects</span>,
    the <span data-x="concept-canvas-global-alpha">global alpha</span>, the <span>current
    compositing and blending operator</span>, the <span
    data-x="concept-CanvasFillStrokeStyles-fill-style">fill style</span>, the <span
    data-x="concept-CanvasFillStrokeStyles-stroke-style">stroke style</span>, or any of the members
    in the <code>CanvasPathDrawingStyles</code>, <code>CanvasTextDrawingStyles</code> interfaces,
    but <em>should</em> be subject to the <span>clipping region</span>. (The effect of
    transformations is described above and varies based on which path is being used.)</p>
   </li>

   <li><p><a href="#inform">Inform the user</a> that the focus is at the location given by the
   intended path. User agents may wait until the next time the <span>event loop</span> reaches its
   <span>update the rendering</span> step to optionally inform the user.</p></li>
  </ol>

  <p>User agents should not implicitly close open subpaths in the intended path when drawing the
  focus ring.</p>

  <p class="note">This might be a moot point, however. For example, if the focus ring is drawn as an
  axis-aligned bounding rectangle around the points in the intended path, then whether the subpaths
  are closed or not has no effect. This specification intentionally does not specify precisely how
  focus rings are to be drawn: user agents are expected to honor their platform's native
  conventions.</p>

  <p id="inform">"Inform the user", as used in this section, does not imply any persistent state
  change. It could mean, for instance, calling a system accessibility API to notify assistive
  technologies such as magnification tools so that the user's magnifier moves to the given area of
  the canvas. However, it does not associate the path with the element, or provide a region for
  tactile feedback, etc.</p>

  </div>

  <h6>Drawing images</h6>

  <p>Objects that implement the <code>CanvasDrawImage</code> interface have the <dfn method
  for="CanvasDrawImage"><code data-x="dom-context-2d-drawImage">drawImage()</code></dfn> method to
  draw images.</p>

  <div w-nodev>

  <p>This method can be invoked with three different sets of arguments:</p>

  <ul class="brief">
   <li><code data-x="">drawImage(<var>image</var>, <var>dx</var>, <var>dy</var>)</code>
   <li><code data-x="">drawImage(<var>image</var>, <var>dx</var>, <var>dy</var>, <var>dw</var>, <var>dh</var>)</code>
   <li><code data-x="">drawImage(<var>image</var>, <var>sx</var>, <var>sy</var>, <var>sw</var>, <var>sh</var>, <var>dx</var>, <var>dy</var>, <var>dw</var>, <var>dh</var>)</code>
  </ul>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-drawImage">drawImage</span>(<var>image</var>, <var>dx</var>, <var>dy</var>)</code></dt>
   <dt><code data-x=""><var>context</var>.<span data-x="dom-context-2d-drawImage">drawImage</span>(<var>image</var>, <var>dx</var>, <var>dy</var>, <var>dw</var>, <var>dh</var>)</code></dt>
   <dt><code data-x=""><var>context</var>.<span data-x="dom-context-2d-drawImage">drawImage</span>(<var>image</var>, <var>sx</var>, <var>sy</var>, <var>sw</var>, <var>sh</var>, <var>dx</var>, <var>dy</var>, <var>dw</var>, <var>dh</var>)</code></dt>

   <dd>
    <p>Draws the given image onto the canvas. The arguments are interpreted as follows:</p>

    <p><img src="/images/drawImage.png" width="356" height="356" alt="The sx and sy parameters give
    the x and y coordinates of the source rectangle; the sw and sh arguments give the width and
    height of the source rectangle; the dx and dy give the x and y coordinates of the destination
    rectangle; and the dw and dh arguments give the width and height of the destination
    rectangle."></p>

    <p>If the image isn't yet fully decoded, then nothing is drawn. If the image is a canvas with no
    data, throws an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>When the <code data-x="dom-context-2d-drawImage">drawImage()</code> method is invoked, the user
  agent must run these steps:</p>

  <ol>
   <li><p>If any of the arguments are infinite or NaN, then return.</p></li>

   <li><p>Let <var>usability</var> be the result of <span data-x="check the usability of the image
   argument">checking the usability of <var>image</var></span>.</p></li>

   <li><p>If <var>usability</var> is <i>bad</i>, then return (without drawing anything).</p></li>

   <li>
    <p>Establish the source and destination rectangles as follows:</p>

    <p>If not specified, the <var>dw</var> and <var>dh</var> arguments must default to the values of
    <var>sw</var> and <var>sh</var>, interpreted such that one <span data-x="'px'">CSS pixel</span>
    in the image is treated as one unit in the <span>output bitmap</span>'s coordinate space. If the
    <var>sx</var>, <var>sy</var>, <var>sw</var>, and <var>sh</var> arguments are omitted, then they
    must default to 0, 0, the image's <span>natural width</span> in image pixels, and the image's
    <span>natural height</span> in image pixels, respectively. If the image has no <span>natural
    dimensions</span>, then the <i>concrete object size</i> must be used instead, as determined
    using the CSS "<a href="https://drafts.csswg.org/css-images/#default-sizing">Concrete Object
    Size Resolution</a>" algorithm, with the <i>specified size</i> having neither a definite width
    nor height, nor any additional constraints, the object's natural properties being those of the
    <var>image</var> argument, and the <span>default object size</span> being the size of the
    <span>output bitmap</span>. <ref>CSSIMAGES</ref></p>

    <p>The source rectangle is the rectangle whose corners are the four points (<var>sx</var>, <var>sy</var>), (<var>sx</var>+<var>sw</var>, <var>sy</var>), (<var>sx</var>+<var>sw</var>, <var>sy</var>+<var>sh</var>),
    (<var>sx</var>, <var>sy</var>+<var>sh</var>).</p>

    <p>The destination rectangle is the rectangle whose corners are the four points (<var>dx</var>, <var>dy</var>), (<var>dx</var>+<var>dw</var>, <var>dy</var>), (<var>dx</var>+<var>dw</var>, <var>dy</var>+<var>dh</var>),
    (<var>dx</var>, <var>dy</var>+<var>dh</var>).</p>

    <p>When the source rectangle is outside the source image, the source rectangle must be clipped
    to the source image and the destination rectangle must be clipped in the same proportion.</p>

    <p class="note">When the destination rectangle is outside the destination image (the
    <span>output bitmap</span>), the pixels that land outside the <span>output bitmap</span> are
    discarded, as if the destination was an infinite canvas whose rendering was clipped to the
    dimensions of the <span>output bitmap</span>.</p>
   </li>

   <li><p>If one of the <var>sw</var> or <var>sh</var> arguments is zero, then return. Nothing is
   painted.</p></li>

   <li>
    <p>Paint the region of the <var>image</var> argument specified by the source rectangle
    on the region of the rendering context's <span>output bitmap</span> specified by the
    destination rectangle, after applying the <span data-x="dom-context-2d-transformation">current
    transformation matrix</span> to the destination rectangle.</p>

    <p>The image data must be processed in the original direction, even if the dimensions given are
    negative. <!-- note: that last sentence has several times been needed to correct people who
    suggested that 5,5,-2,-2 was different than 3,3,2,2 --></p>

    <p>When scaling up, if the <code
    data-x="dom-context-2d-imageSmoothingEnabled">imageSmoothingEnabled</code> attribute is set to
    true, the user agent should attempt to apply a smoothing algorithm to the image data when it is
    scaled. User agents which support multiple filtering algorithms may use the value of the <code
    data-x="dom-context-2d-imageSmoothingQuality">imageSmoothingQuality</code> attribute to guide
    the choice of filtering algorithm when the <code
    data-x="dom-context-2d-imageSmoothingEnabled">imageSmoothingEnabled</code> attribute is set to
    true. Otherwise, the image must be rendered using nearest-neighbor interpolation.</p>

    <p class="note">This specification does not define the precise algorithm to use when scaling an
    image down, or when scaling an image up when the <code
    data-x="dom-context-2d-imageSmoothingEnabled">imageSmoothingEnabled</code> attribute is set to
    true.</p>

    <p class="note">When a <code>canvas</code> element is drawn onto itself, the <span>drawing
    model</span> requires the source to be copied before the image is drawn, so it is possible to
    copy parts of a <code>canvas</code> element onto overlapping parts of itself.</p>

    <p>If the original image data is a bitmap image, then the value painted at a point in the
    destination rectangle is computed by filtering the original image data. The user agent may use
    any filtering algorithm (for example bilinear interpolation or nearest-neighbor). When the
    filtering algorithm requires a pixel value from outside the original image data, it must instead
    use the value from the nearest edge pixel. (That is, the filter uses 'clamp-to-edge' behavior.)
    When the filtering algorithm requires a pixel value from outside the source rectangle but inside
    the original image data, then the value from the original image data must be used.</p>
    <!-- see CORE-32111 and:
         https://krijnhoetmer.nl/irc-logs/whatwg/20100818#l-737
         https://www.w3.org/Bugs/Public/show_bug.cgi?id=10799#c11
    -->
    <!-- createPattern() has a similar paragraph with different rules -->

    <p class="note">Thus, scaling an image in parts or in whole will have the same effect. This does
    mean that when sprites coming from a single sprite sheet are to be scaled, adjacent images in
    the sprite sheet can interfere. This can be avoided by ensuring each sprite in the sheet is
    surrounded by a border of <span>transparent black</span>, or by copying sprites to be scaled
    into temporary <code>canvas</code> elements and drawing the scaled sprites from there.</p>

    <p>Images are painted without affecting the current path, and are subject to <span
    data-x="shadows">shadow effects</span>, <span data-x="concept-canvas-global-alpha">global
    alpha</span>, the <span>clipping region</span>, and the <span>current compositing and blending
    operator</span>.</p>
   </li>

   <li><p>If <var>image</var> <span>is not origin-clean</span>, then set the
   <code>CanvasRenderingContext2D</code>'s <span
   data-x="concept-canvas-origin-clean">origin-clean</span> flag to false.</p></li>
  </ol>

  </div>


  <h6><dfn>Pixel manipulation</dfn></h6>

  <dl class="domintro">
   <dt><code data-x=""><var>imageData</var> = new <span subdfn data-x="dom-imagedata">ImageData</span>(<var>sw</var>, <var>sh</var> [, <var>settings</var>])</code></dt>
   <dd>
    <p>Returns an <code>ImageData</code> object with the given dimensions and the color space
    indicated by <var>settings</var>. All the pixels in the returned object are <span>transparent
    black</span>.</p>

    <p>Throws an <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code> if either of
    the width or height arguments are zero.</p>
   </dd>

   <dt><code data-x=""><var>imageData</var> = new <span subdfn data-x="dom-imagedata-with-data">ImageData</span>(<var>data</var>, <var>sw</var> [, <var>sh</var> [, <var>settings</var> ] ])</code></dt>
   <dd>
    <p>Returns an <code>ImageData</code> object using the data provided in the <code
    data-x="idl-Uint8ClampedArray">Uint8ClampedArray</code> argument, interpreted using the given
    dimensions and the color space indicated by <var>settings</var>.</p>

    <p>As each pixel in the data is represented by four numbers, the length of the data needs to be
    a multiple of four times the given width. If the height is provided as well, then the length
    needs to be exactly the width times the height times 4.</p>

    <p>Throws an <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code> if the given
    data and dimensions can't be interpreted consistently, or if either dimension is zero.</p>
   </dd>

   <dt><code data-x=""><var>imageData</var> = <var>context</var>.<span subdfn data-x="dom-context-2d-createImageData-imagedata">createImageData</span>(<var>imageData</var>)</code></dt>
   <dd><p>Returns an <code>ImageData</code> object with the same dimensions and color space as the
   argument. All the pixels in the returned object are <span>transparent black</span>.</p></dd>

   <dt><code data-x=""><var>imageData</var> = <var>context</var>.<span subdfn data-x="dom-context-2d-createImageData">createImageData</span>(<var>sw</var>, <var>sh</var> [, <var>settings</var>])</code></dt>
   <dd>
    <p>Returns an <code>ImageData</code> object with the given dimensions. The color space of the
    returned object is the <span data-x="concept-canvas-color-space">color space</span> of
    <var>context</var> unless overridden by <var>settings</var>. All the pixels in the returned
    object are <span>transparent black</span>.</p>

    <p>Throws an <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code> if either of
    the width or height arguments are zero.</p>
   </dd>

   <dt><code data-x=""><var>imageData</var> = <var>context</var>.<span subdfn data-x="dom-context-2d-getImageData">getImageData</span>(<var>sx</var>, <var>sy</var>, <var>sw</var>, <var>sh</var> [, <var>settings</var>])</code></dt>
   <dd>
    <p>Returns an <code>ImageData</code> object containing the image data for the given rectangle of
    the bitmap. The color space of the returned object is the <span
    data-x="concept-canvas-color-space">color space</span> of <var>context</var> unless overridden
    by <var>settings</var>.</p>

    <p>Throws an <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code> if the either
    of the width or height arguments are zero.</p>
   </dd>

   <dt><code data-x=""><var>imageData</var>.<span subdfn data-x="dom-imagedata-width">width</span></code></dt>
   <dt><code data-x=""><var>imageData</var>.<span subdfn data-x="dom-imagedata-height">height</span></code></dt>
   <dd><p>Returns the actual dimensions of the data in the <code>ImageData</code> object, in
   pixels.</p></dd>

   <dt><code data-x=""><var>imageData</var>.<span subdfn data-x="dom-imagedata-data">data</span></code></dt>
   <dd><p>Returns the one-dimensional array containing the data in RGBA order, as integers in the
   range 0 to 255.</p></dd>

   <dt><code data-x=""><var>imageData</var>.<span subdfn data-x="dom-imagedata-colorSpace">colorSpace</span></code></dt>
   <dd><p>Returns the color space of the pixels.</p></dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-putImageData">putImageData</span>(<var>imageData</var>, <var>dx</var>, <var>dy</var> [, <var>dirtyX</var>, <var>dirtyY</var>, <var>dirtyWidth</var>, <var>dirtyHeight</var> ])</code></dt>
   <dd>
    <p>Paints the data from the given <code>ImageData</code> object onto the bitmap. If a dirty
    rectangle is provided, only the pixels from that rectangle are painted.</p>

    <p>The <code data-x="dom-context-2d-globalAlpha">globalAlpha</code> and <code
    data-x="dom-context-2d-globalCompositeOperation">globalCompositeOperation</code> properties, as
    well as the <span data-x="shadows">shadow attributes</span>, are ignored for the purposes of
    this method call; pixels in the canvas are replaced wholesale, with no composition, alpha
    blending, no shadows, etc.</p>

    <p>Throws an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> if the
    <var>imageData</var> object's <code data-x="dom-imagedata-data">data</code> attribute value's
    [[ViewedArrayBuffer]] internal slot is detached.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>Objects that implement the <code>CanvasImageData</code> interface provide the following methods
  for reading and writing pixel data to the bitmap.</p>

  <p>The <dfn constructor for="ImageData"><code data-x="dom-imagedata">new ImageData(<var>sw</var>,
  <var>sh</var>, <var>settings</var>)</code></dfn> constructor steps are:</p>

  <ol>
   <li><p>If one or both of <var>sw</var> and <var>sh</var> are zero, then throw an
   <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p><span data-x="Initialize an ImageData object">Initialize</span> <span>this</span> given
   <var>sw</var>, <var>sh</var>, and <i><span
   data-x="initialize-imagedata-settings">settings</span></i> set to <var>settings</var>.</p></li>

   <li><p>Initialize the image data of <span>this</span> to <span>transparent black</span>.</p></li>
  </ol>

  <p>The <dfn constructor for="ImageData"><code data-x="dom-imagedata-with-data">new
  ImageData(<var>data</var>, <var>sw</var>, <var>sh</var>, <var>settings</var>)</code></dfn>
  constructor steps are:</p>

  <ol>
   <li><p>Let <var>length</var> be the number of bytes in <var>data</var>.</p></li>

   <li><p>If <var>length</var> is not a nonzero integral multiple of four, then throw an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>length</var> be <var>length</var> divided by four.</p>

   <li>
    <p>If <var>length</var> is not an integral multiple of <var>sw</var>, then throw an
    <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p>

    <p class="note">At this step, the length is guaranteed to be greater than zero (otherwise the
    second step above would have aborted the steps), so if <var>sw</var> is zero, this
    step will throw the exception and return.</p>
   </li>

   <li><p>Let <var>height</var> be <var>length</var> divided by <var>sw</var>.</p></li>

   <li><p>If <var>sh</var> was given and its value is not equal to <var>height</var>, then throw an
   <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p></li>

   <li>
    <p><span data-x="Initialize an ImageData object">Initialize</span> <span>this</span> given
    <var>sw</var>, <var>sh</var>, <i><span
    data-x="initialize-imagedata-settings">settings</span></i> set to <var>settings</var>, and
    <i><span data-x="initialize-imagedata-source">source</span></i> set to <var>data</var>.</p>

    <p class="note">This step does not set <span>this</span>'s data to a copy of <var>data</var>.
    It sets it to the actual <code data-x="idl-Uint8ClampedArray">Uint8ClampedArray</code> object
    passed as <var>data</var>.</p>
   </li>
  </ol>

  <p>The <dfn method for="CanvasImageData"><code
  data-x="dom-context-2d-createimagedata">createImageData(<var>sw</var>, <var>sh</var>,
  <var>settings</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>If one or both of <var>sw</var> and <var>sh</var> are zero, then throw an
   <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>newImageData</var> be a <span>new</span> <code>ImageData</code> object.</p></li>

   <li><p><span data-x="Initialize an ImageData object">Initialize</span> <var>newImageData</var>
   given the absolute magnitude of <var>sw</var>, the absolute magnitude of <var>sh</var>, <i><span
   data-x="initialize-imagedata-settings">settings</span></i> set to <var>settings</var>, and
   <i><span data-x="initialize-imagedata-defaultcolorspace">defaultColorSpace</span></i> set to
   <span>this</span>'s <span data-x="concept-canvas-color-space">color space</span>.</p></li>

   <li><p>Initialize the image data of <var>newImageData</var> to <span>transparent
   black</span>.</p></li>

   <li><p>Return <var>newImageData</var>.</p></li>
  </ol>

  <p>The <dfn method for="CanvasImageData"><code
  data-x="dom-context-2d-createimagedata-imagedata">createImageData(<var>imageData</var>)</code></dfn>
  method steps are:</p>

  <ol>
   <li><p>Let <var>newImageData</var> be a <span>new</span> <code>ImageData</code> object.</p></li>

   <li><p><span data-x="Initialize an ImageData object">Initialize</span> <var>newImageData</var>
   given the value of <var>imageData</var>'s <code data-x="dom-imagedata-width">width</code>
   attribute, the value of <var>imageData</var>'s <code data-x="dom-imagedata-height">height</code>
   attribute, and <i><span
   data-x="initialize-imagedata-defaultcolorspace">defaultColorSpace</span></i> set to the value of
   <var>imageData</var>'s <code data-x="dom-imagedata-colorSpace">colorSpace</code> attribute.</p></li>

   <li><p>Initialize the image data of <var>newImageData</var> to <span>transparent
   black</span>.</p></li>

   <li><p>Return <var>newImageData</var>.</p></li>
  </ol>

  <p>The <dfn method for="CanvasImageData"><code
  data-x="dom-context-2d-getImageData">getImageData(<var>sx</var>, <var>sy</var>, <var>sw</var>,
  <var>sh</var>, <var>settings</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>If either the <var>sw</var> or <var>sh</var> arguments are zero, then throw an
   <span>"<code>IndexSizeError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If the <code>CanvasRenderingContext2D</code>'s <span
   data-x="concept-canvas-origin-clean">origin-clean</span> flag is set to false, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>imageData</var> be a <span>new</span> <code>ImageData</code> object.</p></li>

   <li><p><span data-x="Initialize an ImageData object">Initialize</span> <var>imageData</var> given
   <var>sw</var>, <var>sh</var>, <i><span
   data-x="initialize-imagedata-settings">settings</span></i> set to <var>settings</var>, and
   <i><span data-x="initialize-imagedata-defaultcolorspace">defaultColorSpace</span></i> set to
   <span>this</span>'s <span data-x="concept-canvas-color-space">color space</span>.</p></li>

   <li><p>Let the source rectangle be the rectangle whose corners are the four points
   (<var>sx</var>, <var>sy</var>), (<span data-x=""><var>sx</var>+<var>sw</var></span>,
   <var>sy</var>), (<span data-x=""><var>sx</var>+<var>sw</var></span>, <span
   data-x=""><var>sy</var>+<var>sh</var></span>), (<var>sx</var>,
   <var>sy</var>+<var>sh</var>).</p></li>

   <li><p>Set the pixel values of <var>imageData</var> to be the pixels of <span>this</span>'s
   <span>output bitmap</span> in the area specified by the source rectangle in the bitmap's
   coordinate space units, converted from <span>this</span>'s <span
   data-x="concept-canvas-color-space">color space</span> to <var>imageData</var>'s <code
   data-x="dom-imagedata-colorSpace">colorSpace</code> using <span>'relative-colorimetric'</span>
   rendering intent.</p></li>

   <li><p>Set the pixels values of <var>imageData</var> for areas of the source rectangle that are
   outside of the <span>output bitmap</span> to <span>transparent black</span>.</p></li>

   <li><p>Return <var>imageData</var>.</p></li>
  </ol>


  <p>To <dfn>initialize an <code>ImageData</code> object</dfn> <var>imageData</var>, given a
  positive integer number of rows <var>rows</var>, a positive integer number of pixels per row
  <var>pixelsPerRow</var>, an optional <code>ImageDataSettings</code> <dfn
  data-x="initialize-imagedata-settings"><var>settings</var></dfn>, an optional <code
  data-x="idl-Uint8ClampedArray">Uint8ClampedArray</code> <dfn
  data-x="initialize-imagedata-source"><var>source</var></dfn>, and an optional
  <code>PredefinedColorSpace</code> <dfn
  data-x="initialize-imagedata-defaultcolorspace"><var>defaultColorSpace</var></dfn>:</p>

  <ol>
   <li><p>If <var>source</var> was given, then initialize the <dfn attribute for="ImageData"><code
   data-x="dom-imagedata-data">data</code></dfn> attribute of <var>imageData</var> to
   <var>source</var>.</p></li>

   <li>
    <p>Otherwise (<var>source</var> was not given), initialize the <code
    data-x="dom-imagedata-data">data</code> attribute of <var>imageData</var> to a new <code
    data-x="idl-Uint8ClampedArray">Uint8ClampedArray</code> object. The <code
    data-x="idl-Uint8ClampedArray">Uint8ClampedArray</code> object must use a new <span>Canvas
    Pixel <code data-x="idl-ArrayBuffer">ArrayBuffer</code></span> for its storage, and must have a
    zero start offset and a length equal to the length of its storage, in bytes. The <span>Canvas
    Pixel <code data-x="idl-ArrayBuffer">ArrayBuffer</code></span> must have the correct size to
    store <var>rows</var> × <var>pixelsPerRow</var> pixels.</p>

    <p>If the <span>Canvas Pixel <code data-x="idl-ArrayBuffer">ArrayBuffer</code></span> cannot be
    allocated, then rethrow the <code data-x="js-RangeError">RangeError</code> thrown by JavaScript,
    and return.</p>
   </li>

   <li><p>Initialize the <dfn attribute for="ImageData"><code
   data-x="dom-imagedata-width">width</code></dfn> attribute of <var>imageData</var> to
   <var>pixelsPerRow</var>.</p></li>

   <li><p>Initialize the <dfn attribute for="ImageData"><code
   data-x="dom-imagedata-height">height</code></dfn> attribute of <var>imageData</var> to
   <var>rows</var>.</p></li>

   <li><p>If <var>settings</var> was given and <var>settings</var>["<code
   data-x="dom-ImageDataSettings-colorSpace">colorSpace</code>"] <span data-x="map
   exists">exists</span>, then initialize the <dfn attribute for="ImageData"><code
   data-x="dom-imagedata-colorSpace">colorSpace</code></dfn> attribute of <var>imageData</var> to
   <var>settings</var>["<dfn dict-member for="ImageDataSettings"><code
   data-x="dom-ImageDataSettings-colorSpace">colorSpace</code></dfn>"].</p></li>

   <li><p>Otherwise, if <var>defaultColorSpace</var> was given, then initialize the <code
   data-x="dom-imagedata-colorSpace">colorSpace</code> attribute of <var>imageData</var> to
   <var>defaultColorSpace</var>.</p></li>

   <li><p>Otherwise, initialize the <code data-x="dom-imagedata-colorSpace">colorSpace</code>
   attribute of <var>imageData</var> to "<span
   data-x="dom-PredefinedColorSpace-srgb">srgb</span>".</p></li>
  </ol>

  <p><code>ImageData</code> objects are <span>serializable objects</span>. Their <span>serialization
  steps</span>, given <var>value</var> and <var>serialized</var>, are:</p>

  <ol>
   <li><p>Set <var>serialized</var>.[[Data]] to the <span>sub-serialization</span> of the value of
   <var>value</var>'s <code data-x="dom-imagedata-data">data</code> attribute.</p></li>

   <li><p>Set <var>serialized</var>.[[Width]] to the value of <var>value</var>'s <code
   data-x="dom-imagedata-width">width</code> attribute.</p></li>

   <li><p>Set <var>serialized</var>.[[Height]] to the value of <var>value</var>'s <code
   data-x="dom-imagedata-height">height</code> attribute.</p></li>

   <li><p>Set <var>serialized</var>.[[ColorSpace]] to the value of <var>value</var>'s <code
   data-x="dom-imagedata-colorSpace">colorSpace</code> attribute.</p></li>
  </ol>

  <p>Their <span>deserialization steps</span>, given <var>serialized</var>, <var>value</var>,
  and <var>targetRealm</var>, are:</p>

  <ol>
   <li><p>Initialize <var>value</var>'s <code data-x="dom-imagedata-data">data</code> attribute
   to the <span>sub-deserialization</span> of <var>serialized</var>.[[Data]].</p></li>

   <li><p>Initialize <var>value</var>'s <code data-x="dom-imagedata-width">width</code> attribute
   to <var>serialized</var>.[[Width]].</p></li>

   <li><p>Initialize <var>value</var>'s <code data-x="dom-imagedata-height">height</code> attribute
   to <var>serialized</var>.[[Height]].</p></li>

   <li><p>Initialize <var>value</var>'s <code data-x="dom-imagedata-colorSpace">colorSpace</code>
   attribute to <var>serialized</var>.[[ColorSpace]].</p></li>
  </ol>

  <p>A <dfn>Canvas Pixel <code data-x="idl-ArrayBuffer">ArrayBuffer</code></dfn> is an <code
  data-x="idl-ArrayBuffer">ArrayBuffer</code> whose data is represented in left-to-right order, row
  by row top to bottom, starting with the top left, with each pixel's red, green, blue, and alpha
  components being given in that order for each pixel. Each component of each pixel represented in
  this array must be in the range 0..255, representing the 8 bit value for that component. The
  components must be assigned consecutive indices starting with 0 for the top left pixel's red
  component.</p>

  <p>The <dfn method for="CanvasImageData"><code
  data-x="dom-context-2d-putImageData-short">putImageData(<var>imageData</var>,
  <var>dx</var>, <var>dy</var>)</code></dfn> method steps are to
  <span data-x="dom-context2d-putImageData-common">put pixels from an
  <code>ImageData</code> onto a bitmap</span>, given
  <var>imageData</var>, <span>this</span>'s <span>output bitmap</span>,
  <var>dx</var>, <var>dy</var>, 0, 0,
  <var>imageData</var>'s <code data-x="dom-imagedata-width">width</code>, and
  <var>imageData</var>'s <code data-x="dom-imagedata-height">height</code>.</p>

  <p>The <dfn method for="CanvasImageData"><code
  data-x="dom-context-2d-putImageData">putImageData(<var>imageData</var>,
  <var>dx</var>, <var>dy</var>, <var>dirtyX</var>, <var>dirtyY</var>,
  <var>dirtyWidth</var>, <var>dirtyHeight</var>)</code></dfn> method steps are to
  <span data-x="dom-context2d-putImageData-common">put pixels from an
  <code>ImageData</code> onto a bitmap</span>, given
  <var>imageData</var>, <span>this</span>'s <span>output bitmap</span>,
  <var>dx</var>, <var>dy</var>, <var>dirtyX</var>, <var>dirtyY</var>,
  <var>dirtyWidth</var>, and <var>dirtyHeight</var>.</p>

  <p>To <dfn data-x="dom-context2d-putImageData-common">put pixels from an <code>ImageData</code>
  onto a bitmap</dfn>, given an <code>ImageData</code> <var>imageData</var>,
  an <span>output bitmap</span> <var>bitmap</var>, and numbers <var>dx</var>, <var>dy</var>,
  <var>dirtyX</var>, <var>dirtyY</var>, <var>dirtyWidth</var>, and <var>dirtyHeight</var>:</p>

  <ol>
   <li><p>Let <var>buffer</var> be <var>imageData</var>'s <code
   data-x="dom-imagedata-data">data</code> attribute value's [[ViewedArrayBuffer]] internal
   slot.</p></li>

   <li><p>If <span>IsDetachedBuffer</span>(<var>buffer</var>) is true, then throw an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li>
    <p>If <var>dirtyWidth</var> is negative, then let <var>dirtyX</var> be <span
    data-x=""><var>dirtyX</var>+<var>dirtyWidth</var></span>, and let <var>dirtyWidth</var> be equal
    to the absolute magnitude of <var>dirtyWidth</var>.</p>

    <p>If <var>dirtyHeight</var> is negative, then let <var>dirtyY</var> be <span
    data-x=""><var>dirtyY</var>+<var>dirtyHeight</var></span>, and let <var>dirtyHeight</var> be
    equal to the absolute magnitude of <var>dirtyHeight</var>.</p>
   </li>

   <li>
    <p>If <var>dirtyX</var> is negative, then let <var>dirtyWidth</var> be <span
    data-x=""><var>dirtyWidth</var>+<var>dirtyX</var></span>, and let <var>dirtyX</var> be zero.</p>

    <p>If <var>dirtyY</var> is negative, then let <var>dirtyHeight</var> be <span
    data-x=""><var>dirtyHeight</var>+<var>dirtyY</var></span>, and let <var>dirtyY</var> be zero.</p>
   </li>

   <li>
    <p>If <var>dirtyX</var>+<var>dirtyWidth</var> is greater than the <code
    data-x="dom-imagedata-width">width</code> attribute of the <var>imageData</var> argument, then
    let <var>dirtyWidth</var> be the value of that <code data-x="dom-imagedata-width">width</code>
    attribute, minus the value of <var>dirtyX</var>.</p>

    <p>If <var>dirtyY</var>+<var>dirtyHeight</var> is greater than the <code
    data-x="dom-imagedata-height">height</code> attribute of the <var>imageData</var> argument, then
    let <var>dirtyHeight</var> be the value of that <code
    data-x="dom-imagedata-height">height</code> attribute, minus the value of <var>dirtyY</var>.</p>
   </li>

   <li>
    <p>If, after those changes, either <var>dirtyWidth</var> or <var>dirtyHeight</var> are negative
    or zero, then return without affecting any bitmaps.</p>
   </li>

   <li><p>For all integer values of <var>x</var> and <var>y</var> where <span
   data-x=""><var>dirtyX</var>&nbsp;&le;&nbsp;<var>x</var>&nbsp;&lt;&nbsp;<span
   data-x=""><var>dirtyX</var>+<var>dirtyWidth</var></span></span> and <span
   data-x=""><var>dirtyY</var>&nbsp;&le;&nbsp;<var>y</var>&nbsp;&lt;&nbsp;<span
   data-x=""><var>dirtyY</var>+<var>dirtyHeight</var></span></span>,
   set the pixel with coordinate (<span data-x=""><var>dx</var>+<var>x</var></span>,
   <var>dy</var>+<var>y</var>) in <var>bitmap</var> to the color of the pixel at coordinate
   (<var>x</var>, <var>y</var>) in the <var>imageData</var> data structure's
   <span>Canvas Pixel <code data-x="idl-ArrayBuffer">ArrayBuffer</code></span>,
   converted from <var>imageData</var>'s <code data-x="dom-imagedata-colorSpace">colorSpace</code>
   to the <span data-x="concept-canvas-color-space">color space</span> of <var>bitmap</var>
   using <span>'relative-colorimetric'</span> rendering intent.</p></li>
  </ol>

  <p class="note">Due to the lossy nature of converting between color spaces and converting to and
  from <span data-x="concept-premultiplied-alpha">premultiplied alpha</span> color values, pixels
  that have just been set using <code data-x="dom-context-2d-putImageData">putImageData()</code>,
  and are not completely opaque, might be returned to an equivalent <code
  data-x="dom-context-2d-getImageData">getImageData()</code> as different values.</p>

  <p>The current path, <span data-x="dom-context-2d-transformation">transformation matrix</span>,
  <span data-x="shadows">shadow attributes</span>, <span data-x="concept-canvas-global-alpha">global
  alpha</span>, the <span>clipping region</span>, and <span>current compositing and blending
  operator</span> must not affect the methods described in this section.</p>

  </div>

  <div class="example">
   <p>In the following example, the script generates an <code>ImageData</code> object so that it can
   draw onto it.</p>

   <pre><code class="js">// canvas is a reference to a &lt;canvas> element
var context = canvas.getContext('2d');

// create a blank slate
var data = context.createImageData(canvas.width, canvas.height);

// create some plasma
FillPlasma(data, 'green'); // green plasma

// add a cloud to the plasma
AddCloud(data, data.width/2, data.height/2); // put a cloud in the middle

// paint the plasma+cloud on the canvas
context.putImageData(data, 0, 0);

// support methods
function FillPlasma(data, color) { ... }
function AddCloud(data, x, y) { ... }</code></pre>
  </div>

  <div class="example">
   <p>Here is an example of using <code
   data-x="dom-context-2d-getImageData">getImageData()</code> and <code
   data-x="dom-context-2d-putImageData">putImageData()</code> to implement an edge detection
   filter.</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang="en">
 &lt;head>
  &lt;title>Edge detection demo&lt;/title>
  &lt;script>
   var image = new Image();
   function init() {
     image.onload = demo;
     image.src = "image.jpeg";
   }
   function demo() {
     var canvas = document.getElementsByTagName('canvas')[0];
     var context = canvas.getContext('2d');

     // draw the image onto the canvas
     context.drawImage(image, 0, 0);

     // get the image data to manipulate
     var input = context.getImageData(0, 0, canvas.width, canvas.height);

     // get an empty slate to put the data into
     var output = context.createImageData(canvas.width, canvas.height);

     // alias some variables for convenience
     // In this case input.width and input.height
     // match canvas.width and canvas.height
     // but we'll use the former to keep the code generic.
     var w = input.width, h = input.height;
     var inputData = input.data;
     var outputData = output.data;

     // edge detection
     for (var y = 1; y &lt; h-1; y += 1) {
       for (var x = 1; x &lt; w-1; x += 1) {
         for (var c = 0; c &lt; 3; c += 1) {
           var i = (y*w + x)*4 + c;
           outputData[i] = 127 + -inputData[i - w*4 - 4] -   inputData[i - w*4] - inputData[i - w*4 + 4] +
                                 -inputData[i - 4]       + 8*inputData[i]       - inputData[i + 4] +
                                 -inputData[i + w*4 - 4] -   inputData[i + w*4] - inputData[i + w*4 + 4];
         }
         outputData[(y*w + x)*4 + 3] = 255; // alpha
       }
     }

     // put the image data back after manipulation
     context.putImageData(output, 0, 0);
   }
  &lt;/script>
 &lt;/head>
 &lt;body onload="init()">
  &lt;canvas>&lt;/canvas>
 &lt;/body>
&lt;/html></code></pre>
  </div>

  <div class="example">
   <p>Here is an example of color space conversion applied when drawing a solid color and
   reading the result back using and <code data-x=
   "dom-context-2d-getImageData">getImageData()</code>.</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang="en">
&lt;title>Color space image data demo&lt;/title>

&lt;canvas>&lt;/canvas>

&lt;script>
const canvas = document.querySelector('canvas');
const context = canvas.getContext('2d', {colorSpace:'display-p3'});

// Draw a red rectangle. Note that the hex color notation
// specifies sRGB colors.
context.fillStyle = "#FF0000";
context.fillRect(0, 0, 64, 64);

// Get the image data.
const pixels = context.getImageData(0, 0, 1, 1);

// This will print 'display-p3', reflecting the default behavior
// of returning image data in the canvas's color space.
console.log(pixels.colorSpace);

// This will print the values 234, 51, and 35, reflecting the
// red fill color, converted to 'display-p3'.
console.log(pixels.data[0]);
console.log(pixels.data[1]);
console.log(pixels.data[2]);
&lt;/script></code></pre>
  </div>



  <h6>Compositing</h6>

  <dl class="domintro">
   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-globalAlpha">globalAlpha</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current <span data-x="concept-canvas-global-alpha">global alpha</span> value
    applied to rendering operations.</p>

    <p>Can be set, to change the <span data-x="concept-canvas-global-alpha">global alpha</span>
    value. Values outside of the range 0.0 .. 1.0 are ignored.</p>
   </dd>


   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-globalCompositeOperation">globalCompositeOperation</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the <span>current compositing and blending operator</span>, from the values defined
    in <cite>Compositing and Blending</cite>. <ref>COMPOSITE</ref></p>

    <p>Can be set, to change the <span>current compositing and blending operator</span>. Unknown
    values are ignored.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>Objects that implement the <code>CanvasCompositing</code> interface have a <span
  data-x="concept-canvas-global-alpha">global alpha</span> value and a <span>current compositing
  and blending operator</span> value that both affect all the drawing operations on this
  object.</p>

  <!-- conformance criteria for painting are described in the "drawing model" section below -->

  <p>The <dfn data-x="concept-canvas-global-alpha">global alpha</dfn> value gives an alpha value
  that is applied to shapes and images before they are composited onto the <span>output bitmap</span>.
  The value ranges from 0.0 (fully transparent) to 1.0 (no additional transparency). It must
  initially have the value 1.0.</p>

  <p>The <dfn attribute for="CanvasCompositing"><code
  data-x="dom-context-2d-globalAlpha">globalAlpha</code></dfn> getter steps are to return
  <span>this</span>'s <span data-x="concept-canvas-global-alpha">global alpha</span>.</p>

  <p>The <code data-x="dom-context-2d-globalAlpha">globalAlpha</code> setter steps are:

  <ol>
   <li><p>If the given value is either infinite, NaN, or not in the range 0.0 to 1.0, then
   return.</p></li>

   <li><p>Otherwise, set <span>this</span>'s <span data-x="concept-canvas-global-alpha">global
   alpha</span> to the given value.</p></li>
  </ol>

  <p>The <dfn>current compositing and blending operator</dfn> value controls how shapes and images
  are drawn onto the <span>output bitmap</span>, once they have had the <span
  data-x="concept-canvas-global-alpha">global alpha</span> and the <span>current transformation
  matrix</span> applied. Initially, it must be set to "<code
  data-x="gcop-source-over">source-over</code>".</p>

  <p>The <dfn attribute for="CanvasCompositing"><code
  data-x="dom-context-2d-globalCompositeOperation">globalCompositeOperation</code></dfn> getter
  steps are to return <span>this</span>'s <span>current compositing and blending
  operator</span>.</p>

  <p>The <code
  data-x="dom-context-2d-globalCompositeOperation">globalCompositeOperation</code> setter steps
  are:</p>

  <ol>
   <li><p>If the given value is not <span>identical to</span> any of the values that the
   <span>&lt;blend-mode&gt;</span> or the <span>&lt;composite-mode&gt;</span> properties are defined
   to take, then return. <ref>COMPOSITE</ref></p></li>

   <li><p>Otherwise, set <span>this</span>'s <span>current compositing and blending operator</span>
   to the given value.</p></li>
  </ol>

  </div>


  <h6>Image smoothing</h6>

  <dl class="domintro">
   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-imageSmoothingEnabled">imageSmoothingEnabled</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns whether pattern fills and the <code
    data-x="dom-context-2d-drawImage">drawImage()</code> method will attempt to smooth images if
    their pixels don't line up exactly with the display, when scaling images up.</p>

    <p>Can be set, to change whether images are smoothed (true) or not (false).</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-imageSmoothingQuality">imageSmoothingQuality</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current image-smoothing-quality preference.</p>

    <p>Can be set, to change the preferred quality of image smoothing. The possible values are
    "<code data-x="dom-context-2d-imageSmoothingQuality-low">low</code>", "<code
    data-x="dom-context-2d-imageSmoothingQuality-medium">medium</code>" and "<code
    data-x="dom-context-2d-imageSmoothingQuality-high">high</code>". Unknown values are ignored.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>Objects that implement the <code>CanvasImageSmoothing</code> interface have attributes that
  control how image smoothing is performed.</p>

  <p>The <dfn attribute for="CanvasImageSmoothing"><code
  data-x="dom-context-2d-imageSmoothingEnabled">imageSmoothingEnabled</code></dfn> attribute, on
  getting, must return the last value it was set to. On setting, it must be set to the new value.
  When the object implementing the <code>CanvasImageSmoothing</code> interface is created, the
  attribute must be set to true.</p>

  <p>The <dfn attribute for="CanvasImageSmoothing"><code
  data-x="dom-context-2d-imageSmoothingQuality">imageSmoothingQuality</code></dfn> attribute, on
  getting, must return the last value it was set to. On setting, it must be set to the new value.
  When the object implementing the <code>CanvasImageSmoothing</code> interface is created, the
  attribute must be set to "<code data-x="dom-context-2d-imageSmoothingQuality-low">low</code>".</p>

  </div>


  <h6><dfn>Shadows</dfn></h6>

  <p>All drawing operations on an object which implements the <code>CanvasShadowStyles</code>
  interface are affected by the four global shadow attributes.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-shadowColor">shadowColor</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current shadow color.</p>

    <p>Can be set, to change the shadow color. Values that cannot be parsed as CSS colors are
    ignored.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-shadowOffsetX">shadowOffsetX</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-shadowOffsetY">shadowOffsetY</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current shadow offset.</p>

    <p>Can be set, to change the shadow offset. Values that are not finite numbers are ignored.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-shadowBlur">shadowBlur</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current level of blur applied to shadows.</p>

    <p>Can be set, to change the blur level. Values that are not finite numbers greater than or
    equal to zero are ignored.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>Objects which implement the <code>CanvasShadowStyles</code> interface have an associated <dfn
  data-x="concept-CanvasShadowStyles-shadow-color">shadow color</dfn>, which is a CSS color.
  Initially, it must be <span>transparent black</span>.</p>

  <p>The <dfn attribute for="CanvasShadowStyles"><code
  data-x="dom-context-2d-shadowColor">shadowColor</code></dfn> getter steps are to return the <span
  data-x="serialize a CSS &lt;color> value">serialization</span> of <span>this</span>'s <span
  data-x="concept-CanvasShadowStyles-shadow-color">shadow color</span> with <span
  data-x="HTML-compatible serialization is requested">HTML-compatible serialization
  requested</span>.</p>

  <p>The <code data-x="dom-context-2d-shadowColor">shadowColor</code> setter steps are:</p>

  <ol>
   <li><p>Let <var>context</var> be <span>this</span>'s <code
   data-x="dom-context-2d-canvas">canvas</code> attribute's value, if that is an element; otherwise
   null.</p></li>

   <li><p>Let <var>parsedValue</var> be the result of <span data-x="parse a CSS &lt;color>
   value">parsing</span> the given value with <var>context</var> if non-null.</p></li>

   <li><p>If <var>parsedValue</var> is failure, then return.</p></li>

   <li><p>Set <span>this</span>'s <span data-x="concept-CanvasShadowStyles-shadow-color">shadow
   color</span> to <var>parsedValue</var>.</p></li>
  </ol>

  <p>The <dfn attribute for="CanvasShadowStyles"><code
  data-x="dom-context-2d-shadowOffsetX">shadowOffsetX</code></dfn> and <dfn attribute
  for="CanvasShadowStyles"><code data-x="dom-context-2d-shadowOffsetY">shadowOffsetY</code></dfn>
  attributes specify the distance that the shadow will be offset in the positive horizontal and
  positive vertical distance respectively. Their values are in coordinate space units. They are not
  affected by the current transformation matrix.</p>

  <p>When the context is created, the shadow offset attributes must initially have the value
  0.</p>

  <p>On getting, they must return their current value. On setting, the attribute being set must be
  set to the new value, except if the value is infinite or NaN, in which case the new value must be
  ignored.</p>

  <p>The <dfn attribute for="CanvasShadowStyles"><code
  data-x="dom-context-2d-shadowBlur">shadowBlur</code></dfn> attribute specifies the level of the
  blurring effect. (The units do not map to coordinate space units, and are not affected by the
  current transformation matrix.)</p>

  <p>When the context is created, the <code data-x="dom-context-2d-shadowBlur">shadowBlur</code>
  attribute must initially have the value 0.</p>

  <p>On getting, the attribute must return its current value. On setting, the attribute must be set
  to the new value, except if the value is negative, infinite or NaN, in which case the new value
  must be ignored.</p>

  <p><dfn data-x="when shadows are drawn">Shadows are only drawn if</dfn> the opacity component of
  the alpha component of the <span data-x="concept-CanvasShadowStyles-shadow-color">shadow
  color</span> is nonzero and either the <code data-x="dom-context-2d-shadowBlur">shadowBlur</code>
  is nonzero, or the <code data-x="dom-context-2d-shadowOffsetX">shadowOffsetX</code> is nonzero, or
  the <code data-x="dom-context-2d-shadowOffsetY">shadowOffsetY</code> is nonzero.</p>

  <p><span>When shadows are drawn</span>, they must be rendered as follows:</p>

  <ol>
   <li><p>Let <var>A</var> be an infinite <span>transparent black</span> bitmap on which the source
   image for which a shadow is being created has been rendered.</p></li>

   <li><p>Let <var>B</var> be an infinite <span>transparent black</span> bitmap, with a coordinate
   space and an origin identical to <var>A</var>.</p></li>

   <li><p>Copy the alpha component of <var>A</var> to <var>B</var>, offset by <code
   data-x="dom-context-2d-shadowOffsetX">shadowOffsetX</code> in the positive <var>x</var>
   direction, and <code data-x="dom-context-2d-shadowOffsetY">shadowOffsetY</code> in the positive
   <var>y</var> direction.</p></li>

   <li>
    <p>If <code data-x="dom-context-2d-shadowBlur">shadowBlur</code> is greater than 0:</p>

    <ol>
     <li><p>Let <var>&sigma;</var> be half the value of <code
     data-x="dom-context-2d-shadowBlur">shadowBlur</code>.</p></li>

     <li><p>Perform a 2D Gaussian Blur on <var>B</var>, using <var>&sigma;</var>
     as the standard deviation.</p> <!-- wish i could find a reference for this --> </li>
    </ol>

    <p>User agents may limit values of <var>&sigma;</var> to an implementation-specific
    maximum value to avoid exceeding hardware limitations during the Gaussian blur operation.</p>
   </li>

   <li><p>Set the red, green, and blue components of every pixel in <var>B</var> to the
   red, green, and blue components (respectively) of the <span
   data-x="concept-CanvasShadowStyles-shadow-color">shadow color</span>.</p></li>

   <li><p>Multiply the alpha component of every pixel in <var>B</var> by the alpha
   component of the <span data-x="concept-CanvasShadowStyles-shadow-color">shadow
   color</span>.</p></li>

   <li><p>The shadow is in the bitmap <var>B</var>, and is rendered as part of the
   <span>drawing model</span> described below.</p></li>
  </ol>

  </div>

  <p>If the <span>current compositing and blending operator</span> is "<code
  data-x="gcop-copy">copy</code>", then shadows effectively won't render (since the shape will
  overwrite the shadow).</p>

  <h6>Filters</h6>

  <p>All drawing operations on an object which implements the <code>CanvasFilters</code>
  interface are affected by the global <dfn attribute for="CanvasFilters"><code
  data-x="dom-context-2d-filter">filter</code></dfn> attribute.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-context-2d-filter">filter</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current filter.</p>


    <p>Can be set, to change the filter. Values can either be the string "<code
    data-x="">none</code>" or a string parseable as a <span>&lt;filter-value-list&gt;</span>. Other
    values are ignored.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>Such objects have an associated <dfn data-x="concept-canvas-current-filter">current
  filter</dfn>, which is a string. Initially the <span
  data-x="concept-canvas-current-filter">current filter</span> is set to the string "<code
  data-x="">none</code>". Whenever the value of the <span
  data-x="concept-canvas-current-filter">current filter</span> is the string "<code
  data-x="">none</code>" filters will be disabled for the context.</p>

  <p>The <code data-x="dom-context-2d-filter">filter</code> getter steps are to return
  <span>this</span>'s <span data-x="concept-canvas-current-filter">current filter</span>.</p>

  <p>The <code data-x="dom-context-2d-filter">filter</code> setter steps are:</p>

  <ol>
   <li><p>If the given value is "<code data-x="">none</code>", then set <span>this</span>'s <span
   data-x="concept-canvas-current-filter">current filter</span> to "<code data-x="">none</code>"
   and return.</p></li>

   <li><p>Let <var>parsedValue</var> be the result of <span data-x="parse something according to a
   CSS grammar">parsing</span> the given values as a <span>&lt;filter-value-list&gt;</span>. If any
   property-independent style sheet syntax like 'inherit' or 'initial' is present, then this
   parsing must return failure.</p></li>

   <li><p>If <var>parsedValue</var> is failure, then return.</p></li>

   <li><p>Set <span>this</span>'s <span data-x="concept-canvas-current-filter">current
   filter</span> to the given value.</p></li>
  </ol>

  </div>

  <p class="note">Though <code data-x=""><var>context</var>.<code
  data-x="dom-context-2d-filter">filter</code> = "<code data-x="">none</code>"</code> will disable
  filters for the context, <code data-x=""><var>context</var>.<code
  data-x="dom-context-2d-filter">filter</code> = ""</code>, <code
  data-x=""><var>context</var>.<code data-x="dom-context-2d-filter">filter</code> = null</code>,
  and <code data-x=""><var>context</var>.<code data-x="dom-context-2d-filter">filter</code> =
  undefined</code> are all treated as unparseable inputs and the value of the <span
  data-x="concept-canvas-current-filter">current filter</span> is left unchanged.</p>

  <p>Coordinates used in the value of the <span data-x="concept-canvas-current-filter">current
  filter</span> are interpreted such that one pixel is equivalent to one SVG user space unit and to
  one canvas coordinate space unit. Filter coordinates are not affected by the <span
  data-x="dom-context-2d-transformation">current transformation matrix</span>. The current
  transformation matrix affects only the input to the filter. Filters are applied in the
  <span>output bitmap</span>'s coordinate space.</p>

  <div w-nodev>

  <p>When the value of the <span data-x="concept-canvas-current-filter">current filter</span> is a
  string parsable as a <span>&lt;filter-value-list&gt;</span> which defines lengths using
  percentages or using <span>'em'</span> or <span>'ex'</span> units, these must be interpreted
  relative to the <span>computed value</span> of the <span>'font-size'</span> property of the
  <span>font style source object</span> at the time that the attribute is set. If the <span
  data-x="computed value">computed values</span> are undefined for a particular case (e.g. because
  the <span>font style source object</span> is not an element or is not <span>being
  rendered</span>), then the relative keywords must be interpreted relative to the default value of
  the <code data-x="dom-context-2d-font">font</code> attribute. The 'larger' and 'smaller' keywords
  are not supported.</p>

  <p>If the value of the <span data-x="concept-canvas-current-filter">current filter</span> is a
  string parseable as a <span>&lt;filter-value-list&gt;</span> with a reference to an SVG filter in
  the same document, and this SVG filter changes, then the changed filter is used for the next draw
  operation.</p>

  <p>If the value of the <span data-x="concept-canvas-current-filter">current filter</span> is a
  string parseable as a <span>&lt;filter-value-list&gt;</span> with a reference to an SVG filter in
  an external resource document and that document is not loaded when a drawing operation is
  invoked, then the drawing operation must proceed with no filtering.</p>

  </div>

  <h6>Working with externally-defined SVG filters</h6>

  <!-- NON-NORMATIVE SECTION -->

  <p>Since drawing is performed using filter value "<code data-x="">none</code>" until an
  externally-defined filter has finished loading, authors might wish to determine whether such a
  filter has finished loading before proceeding with a drawing operation. One way to accomplish
  this is to load the externally-defined filter elsewhere within the same page in some element that
  sends a <code data-x="">load</code> event (for example, an <span>SVG <code>use</code></span>
  element), and wait for the <code data-x="">load</code> event to be dispatched.</p>

  <div w-nodev>

  <h6><dfn>Drawing model</dfn></h6>

  <p>When a shape or image is painted, user agents must follow these steps, in the order given (or
  act as if they do):</p>

  <ol>
   <li><p>Render the shape or image onto an infinite <span>transparent black</span> bitmap, creating
   image <var>A</var>, as described in the previous sections. For shapes, the current fill, stroke,
   and line styles must be honored, and the stroke must itself also be subjected to the current
   transformation matrix.</p></li>

   <li><p>Multiply the alpha component of every pixel in <var>A</var> by <code
   data-x="concept-canvas-global-alpha">global alpha</code>.</p></li>

   <li>
    <p>When the <span data-x="concept-canvas-current-filter">current filter</span> is set to a
    value other than "<code data-x="">none</code>" and all the externally-defined filters it
    references, if any, are in documents that are currently loaded, then use image <var>A</var> as
    the input to the <span data-x="concept-canvas-current-filter">current filter</span>, creating
    image <var>B</var>. If the <span data-x="concept-canvas-current-filter">current filter</span>
    is a string parseable as a <span>&lt;filter-value-list&gt;</span>, then draw using the <span
    data-x="concept-canvas-current-filter">current filter</span> in the same manner as SVG.</p>

    <p>Otherwise, let <var>B</var> be an alias for <var>A</var>.</p>
   </li>

   <li><p><span>When shadows are drawn</span>, render the shadow from image <var>B</var>,
   using the current shadow styles, creating image <var>C</var>.</p></li>

   <li><p><span>When shadows are drawn</span>, composite <var>C</var> within the
   <span>clipping region</span> over the current <span>output bitmap</span> using the <span>current
   compositing and blending operator</span>.</p></li>

   <li><p>Composite <var>B</var> within the <span>clipping region</span> over the current
   <span>output bitmap</span> using the <span>current compositing and blending
   operator</span>.</p></li>
  </ol>

  <p>When compositing onto the <span>output bitmap</span>, pixels that would fall outside of the
  <span>output bitmap</span> must be discarded.</p>

  </div>




  <h6>Best practices</h6>

  <p>When a canvas is interactive, authors should include <span>focusable</span> elements in the
  element's fallback content corresponding to each <span>focusable</span> part of the canvas, as in
  the <a href="#drawCustomFocusRingExample">example above</a>.</p>

  <p>When rendering focus rings, to ensure that focus rings have the appearance of native focus
  rings, authors should use the <code
  data-x="dom-context-2d-drawFocusIfNeeded">drawFocusIfNeeded()</code> method, passing it the
  element for which a ring is being drawn. This method only draws the focus ring if the element is
  <span>focused</span>, so that it can simply be called whenever drawing the element, without
  checking whether the element is focused or not first.</p>

  <p id="no-text-editing-in-canvas-please">Authors should avoid implementing text editing controls
  using the <code>canvas</code> element. Doing so has a large number of disadvantages:</p>

  <ul>
   <li>Mouse placement of the caret has to be reimplemented.</li>

   <li>Keyboard movement of the caret has to be reimplemented (possibly across lines, for multiline
   text input).</li>

   <li>Scrolling of the text control has to be implemented (horizontally for long lines, vertically
   for multiline input).</li>

   <li>Native features such as copy-and-paste have to be reimplemented.</li>

   <li>Native features such as spell-checking have to be reimplemented.</li>

   <li>Native features such as drag-and-drop have to be reimplemented.</li>

   <li>Native features such as page-wide text search have to be reimplemented.</li>

   <li>Native features specific to the user, for example custom text services, have to be
   reimplemented. This is close to impossible since each user might have different services
   installed, and there is an unbounded set of possible such services.</li>

   <li>Bidirectional text editing has to be reimplemented.</li>

   <li>For multiline text editing, line wrapping has to be implemented for all relevant
   languages.</li>

   <li>Text selection has to be reimplemented.</li>

   <li>Dragging of bidirectional text selections has to be reimplemented.</li>

   <li>Platform-native keyboard shortcuts have to be reimplemented.</li>

   <li>Platform-native input method editors (IMEs) have to be reimplemented.</li>

   <li>Undo and redo functionality has to be reimplemented.</li>

   <li>Accessibility features such as magnification following the caret or selection have to be
   reimplemented.</li>
  </ul>

  <p>This is a huge amount of work, and authors are most strongly encouraged to avoid doing any of
  it by instead using the <code>input</code> element, the <code>textarea</code> element, or the
  <code data-x="attr-contenteditable">contenteditable</code> attribute.</p>


  <h6>Examples</h6>

  <!-- NON-NORMATIVE SECTION -->

  <div class="example">

  <p>Here is an example of a script that uses canvas to draw <a href="data:text/html;charset=utf-8;base64,PCFET0NUWVBFIEhUTUw%2BDQo8aHRtbCBsYW5nPSJlbiI%2BDQogPGhlYWQ%2BDQogIDx0aXRsZT5QcmV0dHkgR2xvd2luZyBMaW5lczwvdGl0bGU%2BDQogPC9oZWFkPg0KIDxib2R5Pg0KPGNhbnZhcyB3aWR0aD0iODAwIiBoZWlnaHQ9IjQ1MCI%2BPC9jYW52YXM%2BDQo8c2NyaXB0Pg0KDQogdmFyIGNvbnRleHQgPSBkb2N1bWVudC5nZXRFbGVtZW50c0J5VGFnTmFtZSgnY2FudmFzJylbMF0uZ2V0Q29udGV4dCgnMmQnKTsNCg0KIHZhciBsYXN0WCA9IGNvbnRleHQuY2FudmFzLndpZHRoICogTWF0aC5yYW5kb20oKTsNCiB2YXIgbGFzdFkgPSBjb250ZXh0LmNhbnZhcy5oZWlnaHQgKiBNYXRoLnJhbmRvbSgpOw0KIHZhciBodWUgPSAwOw0KIGZ1bmN0aW9uIGxpbmUoKSB7DQogICBjb250ZXh0LnNhdmUoKTsNCiAgIGNvbnRleHQudHJhbnNsYXRlKGNvbnRleHQuY2FudmFzLndpZHRoLzIsIGNvbnRleHQuY2FudmFzLmhlaWdodC8yKTsNCiAgIGNvbnRleHQuc2NhbGUoMC45LCAwLjkpOw0KICAgY29udGV4dC50cmFuc2xhdGUoLWNvbnRleHQuY2FudmFzLndpZHRoLzIsIC1jb250ZXh0LmNhbnZhcy5oZWlnaHQvMik7DQogICBjb250ZXh0LmJlZ2luUGF0aCgpOw0KICAgY29udGV4dC5saW5lV2lkdGggPSA1ICsgTWF0aC5yYW5kb20oKSAqIDEwOw0KICAgY29udGV4dC5tb3ZlVG8obGFzdFgsIGxhc3RZKTsNCiAgIGxhc3RYID0gY29udGV4dC5jYW52YXMud2lkdGggKiBNYXRoLnJhbmRvbSgpOw0KICAgbGFzdFkgPSBjb250ZXh0LmNhbnZhcy5oZWlnaHQgKiBNYXRoLnJhbmRvbSgpOw0KICAgY29udGV4dC5iZXppZXJDdXJ2ZVRvKGNvbnRleHQuY2FudmFzLndpZHRoICogTWF0aC5yYW5kb20oKSwNCiAgICAgICAgICAgICAgICAgICAgICAgICBjb250ZXh0LmNhbnZhcy5oZWlnaHQgKiBNYXRoLnJhbmRvbSgpLA0KICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnRleHQuY2FudmFzLndpZHRoICogTWF0aC5yYW5kb20oKSwNCiAgICAgICAgICAgICAgICAgICAgICAgICBjb250ZXh0LmNhbnZhcy5oZWlnaHQgKiBNYXRoLnJhbmRvbSgpLA0KICAgICAgICAgICAgICAgICAgICAgICAgIGxhc3RYLCBsYXN0WSk7DQoNCiAgIGh1ZSA9IGh1ZSArIDEwICogTWF0aC5yYW5kb20oKTsNCiAgIGNvbnRleHQuc3Ryb2tlU3R5bGUgPSAnaHNsKCcgKyBodWUgKyAnLCA1MCUsIDUwJSknOw0KICAgY29udGV4dC5zaGFkb3dDb2xvciA9ICd3aGl0ZSc7DQogICBjb250ZXh0LnNoYWRvd0JsdXIgPSAxMDsNCiAgIGNvbnRleHQuc3Ryb2tlKCk7DQogICBjb250ZXh0LnJlc3RvcmUoKTsNCiB9DQogc2V0SW50ZXJ2YWwobGluZSwgNTApOw0KDQogZnVuY3Rpb24gYmxhbmsoKSB7DQogICBjb250ZXh0LmZpbGxTdHlsZSA9ICdyZ2JhKDAsMCwwLDAuMSknOw0KICAgY29udGV4dC5maWxsUmVjdCgwLCAwLCBjb250ZXh0LmNhbnZhcy53aWR0aCwgY29udGV4dC5jYW52YXMuaGVpZ2h0KTsNCiB9DQogc2V0SW50ZXJ2YWwoYmxhbmssIDQwKTsNCg0KPC9zY3JpcHQ%2BDQogPC9ib2R5Pg0KPC9odG1sPg0K">pretty glowing lines</a>.</p>

  <pre><code class="html">&lt;canvas width="800" height="450">&lt;/canvas>
&lt;script>

 var context = document.getElementsByTagName('canvas')[0].getContext('2d');

 var lastX = context.canvas.width * Math.random();
 var lastY = context.canvas.height * Math.random();
 var hue = 0;
 function line() {
   context.save();
   context.translate(context.canvas.width/2, context.canvas.height/2);
   context.scale(0.9, 0.9);
   context.translate(-context.canvas.width/2, -context.canvas.height/2);
   context.beginPath();
   context.lineWidth = 5 + Math.random() * 10;
   context.moveTo(lastX, lastY);
   lastX = context.canvas.width * Math.random();
   lastY = context.canvas.height * Math.random();
   context.bezierCurveTo(context.canvas.width * Math.random(),
                         context.canvas.height * Math.random(),
                         context.canvas.width * Math.random(),
                         context.canvas.height * Math.random(),
                         lastX, lastY);

   hue = hue + 10 * Math.random();
   context.strokeStyle = 'hsl(' + hue + ', 50%, 50%)';
   context.shadowColor = 'white';
   context.shadowBlur = 10;
   context.stroke();
   context.restore();
 }
 setInterval(line, 50);

 function blank() {
   context.fillStyle = 'rgba(0,0,0,0.1)';
   context.fillRect(0, 0, context.canvas.width, context.canvas.height);
 }
 setInterval(blank, 40);

&lt;/script></code></pre>

  </div>

  <div class="example">

   <p>The 2D rendering context for <code>canvas</code> is often used for sprite-based games. The
   following example demonstrates this:</p>

   <iframe src="/demos/canvas/blue-robot/index-idle.html" width="396" height="216"></iframe>

   <p>Here is the source for this example:</p>

   <pre><code class="html">EXAMPLE canvas/blue-robot/index.html</code></pre>

  </div>




  <h5>The <code>ImageBitmap</code> rendering context</h5>

  <h6>Introduction</h6>

  <p><code>ImageBitmapRenderingContext</code> is a performance-oriented interface that provides a
  low overhead method for displaying the contents of <code>ImageBitmap</code> objects. It uses
  transfer semantics to reduce overall memory consumption. It also streamlines performance by
  avoiding intermediate compositing, unlike the <code
  data-x="dom-context-2d-drawImage">drawImage()</code> method of
  <code>CanvasRenderingContext2D</code>.</p>

  <p>Using an <code>img</code> element as an intermediate for getting an image resource into a
  canvas, for example, would result in two copies of the decoded image existing in memory at the
  same time: the <code>img</code> element's copy, and the one in the canvas's backing store. This
  memory cost can be prohibitive when dealing with extremely large images. This can be avoided by
  using <code>ImageBitmapRenderingContext</code>.</p>

  <div class="example">
   <p>Using <code>ImageBitmapRenderingContext</code>, here is how to transcode an image to the JPEG
   format in a memory- and CPU-efficient way:</p>

   <pre><code class="js">createImageBitmap(inputImageBlob).then(image => {
  const canvas = document.createElement('canvas');
  const context = canvas.getContext('bitmaprenderer');
  context.transferFromImageBitmap(image);

  canvas.toBlob(outputJPEGBlob => {
    // Do something with outputJPEGBlob.
  }, 'image/jpeg');
});</code></pre>
  </div>

  <h6>The <code>ImageBitmapRenderingContext</code> interface</h6>

  <pre><code class="idl">[Exposed=(Window,Worker)]
interface <dfn interface>ImageBitmapRenderingContext</dfn> {
  readonly attribute (<span>HTMLCanvasElement</span> or <span>OffscreenCanvas</span>) <span data-x="dom-ImageBitmapRenderingContext-canvas">canvas</span>;
  undefined <span data-x="dom-ImageBitmapRenderingContext-transferFromImageBitmap">transferFromImageBitmap</span>(<span>ImageBitmap</span>? bitmap);
};

dictionary <dfn dictionary>ImageBitmapRenderingContextSettings</dfn> {
  boolean <span data-x="dom-ImageBitmapRenderingContextSettings-alpha">alpha</span> = true;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>context</var> = <var>canvas</var>.<span data-x="dom-canvas-getContext">getContext</span>('bitmaprenderer' [, { [ <span data-x="dom-ImageBitmapRenderingContextSettings-alpha">alpha</span>: false ] } ])</code></dt>

   <dd>
    <p>Returns an <code>ImageBitmapRenderingContext</code> object that is permanently bound to a
    particular <code>canvas</code> element.</p>

    <p>If the <code data-x="dom-ImageBitmapRenderingContextSettings-alpha">alpha</code> setting is
    provided and set to false, then the canvas is forced to always be opaque.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-ImageBitmapRenderingContext-canvas">canvas</span></code></dt>

   <dd><p>Returns the <code>canvas</code> element that the context is bound to.</p></dd>

   <dt><code data-x=""><var>context</var>.<span subdfn data-x="dom-ImageBitmapRenderingContext-transferFromImageBitmap">transferFromImageBitmap</span>(imageBitmap)</code></dt>

   <dd>
    <p>Transfers the underlying <span data-x="concept-ImageBitmap-bitmap-data">bitmap data</span>
    from <var>imageBitmap</var> to <var>context</var>, and the bitmap becomes the contents of the
    <code>canvas</code> element to which <var>context</var> is bound.</p>
   </dd>

   <dt><code data-x=""><var>context</var>.<span data-x="dom-ImageBitmapRenderingContext-transferFromImageBitmap">transferFromImageBitmap</span>(null)</code></dt>

   <dd>
    <p>Replaces contents of the <code>canvas</code> element to which <var>context</var> is bound
    with a <span>transparent black</span> bitmap whose size corresponds to the <code
    data-x="attr-canvas-width">width</code> and <code data-x="attr-canvas-height">height</code>
    content attributes of the <code>canvas</code> element.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="ImageBitmapRenderingContext"><code
  data-x="dom-ImageBitmapRenderingContext-canvas">canvas</code></dfn> attribute must return the
  value it was initialized to when the object was created.</p>

  <p>An <code>ImageBitmapRenderingContext</code> object has an <dfn
  data-x="concept-ImageBitmapRenderingContext-output-bitmap">output bitmap</dfn>, which is a
  reference to <span data-x="concept-ImageBitmap-bitmap-data">bitmap data</span>.</p>

  <p>An <code>ImageBitmapRenderingContext</code> object has a <dfn
  data-x="concept-ImageBitmapRenderingContext-bitmap-mode">bitmap mode</dfn>, which can be set to
  <dfn data-x="concept-ImageBitmapRenderingContext-valid">valid</dfn> or <dfn
  data-x="concept-ImageBitmapRenderingContext-blank">blank</dfn>. A value of <span
  data-x="concept-ImageBitmapRenderingContext-valid">valid</span> indicates that the context's
  <span data-x="concept-ImageBitmapRenderingContext-output-bitmap">output bitmap</span> refers to
  <span data-x="concept-ImageBitmap-bitmap-data">bitmap data</span> that was acquired via <code
  data-x="dom-ImageBitmapRenderingContext-transferFromImageBitmap">transferFromImageBitmap()</code>.
  A value <span data-x="concept-ImageBitmapRenderingContext-blank">blank</span> indicates that the
  context's <span data-x="concept-ImageBitmapRenderingContext-output-bitmap">output
  bitmap</span> is a default transparent bitmap.</p>

  <p>An <code>ImageBitmapRenderingContext</code> object also has an <dfn
  data-x="concept-ImageBitmapRenderingContext-alpha">alpha</dfn> flag, which can be set to true or
  false. When an <code>ImageBitmapRenderingContext</code> object has its <span
  data-x="concept-ImageBitmapRenderingContext-alpha">alpha</span> flag set to false, the contents
  of the <code>canvas</code> element to which the context is bound are obtained by
  compositing the context's <span
  data-x="concept-ImageBitmapRenderingContext-output-bitmap">output bitmap</span> onto an
  <span>opaque black</span> bitmap of the same size using the <span
  data-x="gcop-source-over">source-over</span> compositing operator. If the <span
  data-x="concept-ImageBitmapRenderingContext-alpha">alpha</span> flag is set to true, then the
  <span data-x="concept-ImageBitmapRenderingContext-output-bitmap">output bitmap</span> is used as
  the contents of the <code>canvas</code> element to which the context is bound.
  <ref>COMPOSITE</ref></p>

  <p class="note">The step of compositing over an <span>opaque black</span> bitmap ought to be
  elided whenever equivalent results can be obtained more efficiently by other means.</p>

  <hr>

  <p>When a user agent is required to <dfn>set an <code>ImageBitmapRenderingContext</code>'s output
  bitmap</dfn>, with a <var>context</var> argument that is an
  <code>ImageBitmapRenderingContext</code> object and an optional argument <var>bitmap</var> that
  refers to <span data-x="concept-imagebitmap-bitmap-data">bitmap data</span>, it must run these
  steps:</p>

  <ol>
   <li><p>If a <var>bitmap</var> argument was not provided, then:</p>

    <ol>
     <li><p>Set <var>context</var>'s <span
     data-x="concept-ImageBitmapRenderingContext-bitmap-mode">bitmap mode</span> to <span
     data-x="concept-ImageBitmapRenderingContext-blank">blank</span>.</p></li>

     <li><p>Let <var>canvas</var> be the <code>canvas</code> element to which <var>context</var>
     is bound.</p></li>

     <li><p>Set <var>context</var>'s <span
     data-x="concept-ImageBitmapRenderingContext-output-bitmap">output bitmap</span> to be
     <span>transparent black</span> with a <span>natural width</span> equal to <span
     data-x="obtain-numeric-values">the numeric value</span> of <var>canvas</var>'s <code
     data-x="attr-canvas-width">width</code> attribute and a <span>natural height</span> equal
     to <span data-x="obtain-numeric-values">the numeric value</span> of <var>canvas</var>'s
     <code data-x="attr-canvas-height">height</code> attribute, those values being interpreted
     in <span data-x="'px'">CSS pixels</span>.</p></li>

     <li><p>Set the <span
     data-x="concept-ImageBitmapRenderingContext-output-bitmap">output bitmap</span>'s <span
     data-x="concept-canvas-origin-clean">origin-clean</span> flag to true.</p></li>
    </ol>
   </li>

   <li><p>If a <var>bitmap</var> argument was provided, then:</p>

    <ol>
     <li><p>Set <var>context</var>'s <span
     data-x="concept-ImageBitmapRenderingContext-bitmap-mode">bitmap mode</span> to <span
     data-x="concept-ImageBitmapRenderingContext-valid">valid</span>.</p></li>

     <li>
      <p>Set <var>context</var>'s <span
      data-x="concept-ImageBitmapRenderingContext-output-bitmap">output bitmap</span> to refer
      to the same underlying bitmap data as <var>bitmap</var>, without making a copy.</p>

      <p class="note">The <span data-x="concept-canvas-origin-clean">origin-clean</span> flag of
      <var>bitmap</var> is included in the bitmap data to be referenced by <var>context</var>'s
      <span data-x="concept-ImageBitmapRenderingContext-output-bitmap">output bitmap</span>.</p>
     </li>
    </ol>
   </li>
  </ol>

  <hr>

  <p>The <dfn><code>ImageBitmapRenderingContext</code> creation algorithm</dfn>, which is passed a
  <var>target</var> and <var>options</var>, consists of running these steps:</p>

  <ol>
   <li><p>Let <var>settings</var> be the result of <span
   data-x="concept-idl-convert">converting</span> <var>options</var> to the dictionary type
   <code>ImageBitmapRenderingContextSettings</code>. (This can throw an exception.)</p></li>

   <li><p>Let <var>context</var> be a new <code>ImageBitmapRenderingContext</code> object.</p></li>

   <li><p>Initialize <var>context</var>'s <code data-x="dom-context-2d-canvas">canvas</code>
   attribute to point to <var>target</var>.</p></li>

   <li><p>Set <var>context</var>'s <span
   data-x="concept-ImageBitmapRenderingContext-output-bitmap">output bitmap</span> to the same
   bitmap as <var>target</var>'s bitmap (so that they are shared).</p></li>

   <li><p>Run the steps to <span>set an <code>ImageBitmapRenderingContext</code>'s output
   bitmap</span> with <var>context</var>.</p></li>

   <li><p>Initialize <var>context</var>'s <span
   data-x="concept-ImageBitmapRenderingContext-alpha">alpha</span> flag to true.

   <li>
    <p>Process each of the members of <var>settings</var> as follows:</p>

    <dl>
     <dt><dfn dict-member for="ImageBitmapRenderingContextSettings"><code data-x="dom-ImageBitmapRenderingContextSettings-alpha">alpha</code></dfn></dt>
     <dd>If false, then set <var>context</var>'s <span
     data-x="concept-ImageBitmapRenderingContext-alpha">alpha</span> flag to false.</dd>
    </dl>
   </li>

   <li><p>Return <var>context</var>.</p></li>
  </ol>

  <hr>

  <p>The <dfn method for="ImageBitmapRenderingContext"><code
  data-x="dom-ImageBitmapRenderingContext-transferFromImageBitmap">transferFromImageBitmap(<var>bitmap</var>)</code></dfn>
  method, when invoked, must run these steps:</p>

  <ol>
   <li><p>Let <var>bitmapContext</var> be the <code>ImageBitmapRenderingContext</code> object on
   which the <code
   data-x="dom-ImageBitmapRenderingContext-transferFromImageBitmap">transferFromImageBitmap()</code>
   method was called.</p></li>

   <li><p>If <var>bitmap</var> is null, then run the steps to <span>set an
   ImageBitmapRenderingContext's output bitmap</span>, with <var>bitmapContext</var> as the
   <var>context</var> argument and no <var>bitmap</var> argument, then return.</p></li>

   <li><p>If the value of <var>bitmap</var>'s <span>[[Detached]]</span> internal slot is set to
   true, then throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Run the steps to <span>set an <code>ImageBitmapRenderingContext</code>'s output
   bitmap</span>, with the <var>context</var> argument equal to <var>bitmapContext</var>, and the
   <var>bitmap</var> argument referring to <var>bitmap</var>'s underlying <span
   data-x="concept-ImageBitmap-bitmap-data">bitmap data</span>.

   <li><p>Set the value of <var>bitmap</var>'s <span>[[Detached]]</span> internal slot to
   true.</p></li>

   <li><p>Unset <var>bitmap</var>'s <span data-x="concept-ImageBitmap-bitmap-data">bitmap
   data</span>.</p></li>
  </ol>

  </div>




  <h5>The <code>OffscreenCanvas</code> interface</h5>

  <pre><code class="idl">typedef (<span>OffscreenCanvasRenderingContext2D</span> or <span>ImageBitmapRenderingContext</span> or <span>WebGLRenderingContext</span> or <span>WebGL2RenderingContext</span> or <span>GPUCanvasContext</span>) <dfn typedef>OffscreenRenderingContext</dfn>;

dictionary <dfn dictionary>ImageEncodeOptions</dfn> {
  DOMString <span data-x="image-encode-options-type">type</span> = "image/png";
  unrestricted double <span data-x="image-encode-options-quality">quality</span>;
};

enum <dfn enum>OffscreenRenderingContextId</dfn> { "<span data-x="offscreen-context-type-2d">2d</span>", "<span data-x="offscreen-context-type-bitmaprenderer">bitmaprenderer</span>", "<span data-x="offscreen-context-type-webgl">webgl</span>", "<span data-x="offscreen-context-type-webgl2">webgl2</span>", "<span data-x="offscreen-context-type-webgpu">webgpu</span>" };

[Exposed=(Window,Worker), <span>Transferable</span>]
interface <dfn interface>OffscreenCanvas</dfn> : <span>EventTarget</span> {
  <span data-x="dom-OffscreenCanvas">constructor</span>([EnforceRange] unsigned long long width, [EnforceRange] unsigned long long height);

  attribute [EnforceRange] unsigned long long <span data-x="dom-OffscreenCanvas-width">width</span>;
  attribute [EnforceRange] unsigned long long <span data-x="dom-OffscreenCanvas-height">height</span>;

  <span>OffscreenRenderingContext</span>? <span data-x="dom-OffscreenCanvas-getContext">getContext</span>(<span>OffscreenRenderingContextId</span> contextId, optional any options = null);
  <span>ImageBitmap</span> <span data-x="dom-OffscreenCanvas-transferToImageBitmap">transferToImageBitmap</span>();
  <span data-x="idl-Promise">Promise</span>&lt;<span>Blob</span>> <span data-x="dom-OffscreenCanvas-convertToBlob">convertToBlob</span>(optional <span>ImageEncodeOptions</span> options = {});

  attribute <span>EventHandler</span> <span data-x="handler-offscreencanvas-oncontextlost">oncontextlost</span>;
  attribute <span>EventHandler</span> <span data-x="handler-offscreencanvas-oncontextrestored">oncontextrestored</span>;
};</code></pre>

  <p class="note"><code>OffscreenCanvas</code> is an <code>EventTarget</code>, so both
  <code>OffscreenCanvasRenderingContext2D</code> and WebGL can fire events at it.
  <code>OffscreenCanvasRenderingContext2D</code> can fire <code
  data-x="event-contextlost">contextlost</code> and <code
  data-x="event-contextrestored">contextrestored</code>, and WebGL can fire <code
  data-x="">webglcontextlost</code> and <code data-x="">webglcontextrestored</code>.
  <ref>WEBGL</ref></p>

  <p><code>OffscreenCanvas</code> objects are used to create rendering contexts, much like an
  <code>HTMLCanvasElement</code>, but with no connection to the DOM. This makes it possible to
  use canvas rendering contexts in <a href="#workers">workers</a>.</p>

  <p>An <code>OffscreenCanvas</code> object may hold a weak reference to a <dfn
  data-x="offscreencanvas-placeholder">placeholder <code>canvas</code> element</dfn>, which is
  typically in the DOM, whose embedded content is provided by the <code>OffscreenCanvas</code>
  object. The bitmap of the <code>OffscreenCanvas</code> object is pushed to the <span
  data-x="offscreencanvas-placeholder">placeholder <code>canvas</code> element</span> as part of
  the <code>OffscreenCanvas</code>'s <span>relevant agent</span>'s <span
  data-x="concept-agent-event-loop">event loop</span>'s <span>update the rendering</span>
  steps.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>offscreenCanvas</var> = new <span subdfn data-x="dom-OffscreenCanvas">OffscreenCanvas</span>(<var>width</var>,
   <var>height</var>)</code></dt>
   <dd>
    <p>Returns a new <code>OffscreenCanvas</code> object that is not linked to a <span
    data-x="offscreencanvas-placeholder">placeholder <code>canvas</code> element</span>, and whose
    bitmap's size is determined by the <var>width</var> and <var>height</var> arguments.</p>
   </dd>

   <dt><code data-x=""><var>context</var> = <var>offscreenCanvas</var>.<span subdfn data-x="dom-OffscreenCanvas-getContext">getContext</span>(<var>contextId</var> [,
   <var>options</var> ])</code></dt>
   <dd>
    <p>Returns an object that exposes an API for drawing on the <code>OffscreenCanvas</code> object.
    <var>contextId</var> specifies the desired API: "<code
    data-x="offscreen-context-type-2d">2d</code>", "<code
    data-x="offscreen-context-type-bitmaprenderer">bitmaprenderer</code>", "<code
    data-x="offscreen-context-type-webgl">webgl</code>", "<code
    data-x="offscreen-context-type-webgl2">webgl2</code>", or "<code
    data-x="offscreen-context-type-webgpu">webgpu</code>". <var>options</var> is handled by that
    API.</p>

    <p>This specification defines the "<code data-x="canvas-context-2d">2d</code>" context below,
    which is similar but distinct from the "<code data-x="offscreen-context-type-2d">2d</code>"
    context that is created from a <code>canvas</code> element. The WebGL specifications define the
    "<code data-x="offscreen-context-type-webgl">webgl</code>" and "<code
    data-x="offscreen-context-type-webgl2">webgl2</code>" contexts. <cite>WebGPU</cite> defines the
    "<code data-x="offscreen-context-type-webgpu">webgpu</code>" context. <ref>WEBGL</ref>
    <ref>WEBGPU</ref></p>

    <p>Returns null if the canvas has already been initialized with another context type (e.g.,
    trying to get a "<code data-x="offscreen-context-type-2d">2d</code>" context after getting a
    "<code data-x="offscreen-context-type-webgl">webgl</code>" context).</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>An <code>OffscreenCanvas</code> object has an internal <dfn
  data-x="offscreencanvas-bitmap">bitmap</dfn> that is initialized when the object
  is created. The width and height of the <span data-x="offscreencanvas-bitmap">bitmap</span> are
  equal to the values of the <code data-x="dom-OffscreenCanvas-width">width</code> and <code
  data-x="dom-OffscreenCanvas-height">height</code> attributes of the <code>OffscreenCanvas</code>
  object. Initially, all the bitmap's pixels are <span>transparent black</span>.</p>

  <p>An <code>OffscreenCanvas</code> object has an internal <dfn
  data-x="offscreencanvas-inherited-lang">inherited language</dfn> and <dfn
  data-x="offscreencanvas-inherited-direction">inherited direction</dfn> set when
  the <code>OffscreenCanvas</code> is created.</p>

  <p>An <code>OffscreenCanvas</code> object can have a rendering context bound to it. Initially,
  it does not have a bound rendering context. To keep track of whether it has a rendering context
  or not, and what kind of rendering context it is, an <code>OffscreenCanvas</code> object also
  has a <dfn data-x="offscreencanvas-context-mode">context mode</dfn>, which is initially <dfn
  data-x="offscreencanvas-context-none">none</dfn> but can be changed to either <dfn
  data-x="offscreencanvas-context-2d">2d</dfn>, <dfn
  data-x="offscreencanvas-context-bitmaprenderer">bitmaprenderer</dfn>, <dfn
  data-x="offscreencanvas-context-webgl">webgl</dfn>, <dfn
  data-x="offscreencanvas-context-webgl2">webgl2</dfn>, <dfn
  data-x="offscreencanvas-context-webgpu">webgpu</dfn>, or <dfn
  data-x="offscreencanvas-context-detached">detached</dfn> by algorithms defined in this
  specification.</p>

  <p>The <dfn constructor for="OffscreenCanvas"><code data-x="dom-OffscreenCanvas">new OffscreenCanvas(<var>width</var>,
  <var>height</var>)</code></dfn> constructor steps are:</p>

  <ol>
   <li><p>Initialize the <span data-x="offscreencanvas-bitmap">bitmap</span> of <span>this</span>
   to a rectangular array of <span>transparent black</span> pixels of the dimensions specified
   by <var>width</var> and <var>height</var>.</p></li>

   <li><p>Initialize the <code data-x="dom-OffscreenCanvas-width">width</code> of <span>this</span>
   to <var>width</var>.</p></li>

   <li><p>Initialize the <code data-x="dom-OffscreenCanvas-height">height</code> of
   <span>this</span> to <var>height</var>.</p></li>

   <li><p>Set <span>this</span>'s <span data-x="offscreencanvas-inherited-lang">inherited
   language</span> to <span data-x="concept-explicitly-unknown">explicitly unknown</span>.</p></li>

   <li><p>Set <span>this</span>'s <span data-x="offscreencanvas-inherited-direction">inherited
   direction</span> to "<code data-x="">ltr</code>".</p></li>

   <li><p>Let <var>global</var> be the <span>relevant global object</span> of
   <span>this</span>.</p></li>

   <li>
    <p>If <var>global</var> is a <code>Window</code> object:</p>

    <ol>
     <li><p>Let <var>element</var> be the <span>document element</span> of <var>global</var>'s
     <span data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

     <li>
      <p>If <var>element</var> is not null:</p>

      <ol>
       <li><p>Set the <span data-x="offscreencanvas-inherited-lang">inherited language</span>
       of <span>this</span> to <var>element</var>'s <span
       data-x="language">language</span>.</p></li>

       <li><p>Set the <span data-x="offscreencanvas-inherited-direction">inherited
       direction</span> of <span>this</span> to <var>element</var>'s <span
       data-x="the directionality">directionality</span>.</p></li>
      </ol>
     </li>
    </ol>
   </li>
  </ol>

  <p><code>OffscreenCanvas</code> objects are <span data-x="transferable
  objects">transferable</span>. Their <span>transfer steps</span>, given <var>value</var> and
  <var>dataHolder</var>, are as follows:</p>

  <ol>
   <li><p>If <var>value</var>'s <span data-x="offscreencanvas-context-mode">context mode</span> is
   not equal to <span data-x="offscreencanvas-context-none">none</span>, then throw an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Set <var>value</var>'s <span data-x="offscreencanvas-context-mode">context mode</span> to
   <span data-x="offscreencanvas-context-detached">detached</span>.</p></li>

   <li><p>Let <var>width</var> and <var>height</var> be the dimensions of <var>value</var>'s <span
   data-x="offscreencanvas-bitmap">bitmap</span>.</p></li>

   <li><p>Let <var>language</var> and <var>direction</var> be the values of <var>value</var>'s
   <span data-x="offscreencanvas-inherited-lang">inherited language</span> and
   <span data-x="offscreencanvas-inherited-direction">inherited direction</span>.</p></li>

   <li><p>Unset <var>value</var>'s <span data-x="offscreencanvas-bitmap">bitmap</span>.</p></li>

   <li><p>Set <var>dataHolder</var>.[[Width]] to <var>width</var> and
   <var>dataHolder</var>.[[Height]] to <var>height</var>.</p></li>

   <li><p>Set <var>dataHolder</var>.[[Language]] to <var>language</var> and
   <var>dataHolder</var>.[[Direction]] to <var>direction</var>.</p></li>

   <li><p>Set <var>dataHolder</var>.[[PlaceholderCanvas]] to be a weak reference to
   <var>value</var>'s <span data-x="offscreencanvas-placeholder">placeholder <code>canvas</code>
   element</span>, if <var>value</var> has one, or null if it does not.</p></li>
  </ol>

  <p>Their <span>transfer-receiving steps</span>, given <var>dataHolder</var> and <var>value</var>,
  are:</p>

  <ol>
   <li><p>Initialize <var>value</var>'s <span data-x="offscreencanvas-bitmap">bitmap</span> to a
   rectangular array of <span>transparent black</span> pixels with width given by
   <var>dataHolder</var>.[[Width]] and height given by <var>dataHolder</var>.[[Height]].</p></li>

   <li><p>Set <var>value</var>'s <span data-x="offscreencanvas-inherited-lang">inherited
   language</span> to <var>dataHolder</var>.[[Language]] and <span
   data-x="offscreencanvas-inherited-direction">inherited direction</span> to
   <var>dataHolder</var>.[[Direction]].</p></li>

   <li><p>If <var>dataHolder</var>.[[PlaceholderCanvas]] is not null, set <var>value</var>'s <span
   data-x="offscreencanvas-placeholder">placeholder <code>canvas</code> element</span> to
   <var>dataHolder</var>.[[PlaceholderCanvas]] (while maintaining the weak reference
   semantics).</p></li>
  </ol>

  <hr>

  <p>The <dfn method for="OffscreenCanvas"><code
  data-x="dom-OffscreenCanvas-getContext">getContext(<var>contextId</var>,
  <var>options</var>)</code></dfn> method of an <code>OffscreenCanvas</code> object, when invoked,
  must run these steps:</p>

  <ol>
   <li><p>If <var>options</var> is not an <span data-x="idl-object">object</span>, then set
   <var>options</var> to null.</p></li>

   <li><p>Set <var>options</var> to the result of <span
   data-x="concept-idl-convert">converting</span> <var>options</var> to a JavaScript value.</p></li>

   <li>
    <p>Run the steps in the cell of the following table whose column header matches this
    <code>OffscreenCanvas</code> object's <span data-x="offscreencanvas-context-mode">context
    mode</span> and whose row header matches <var>contextId</var>:</p>

    <table>
     <thead>
      <tr>
       <td>
       <th><span data-x="offscreencanvas-context-none">none</span>
       <th><span data-x="offscreencanvas-context-2d">2d</span>
       <th><span data-x="offscreencanvas-context-bitmaprenderer">bitmaprenderer</span>
       <th><span data-x="offscreencanvas-context-webgl">webgl</span> or <span
       data-x="offscreencanvas-context-webgl2">webgl2</span>
       <th><span data-x="offscreencanvas-context-webgpu">webgpu</span>
       <th><span data-x="offscreencanvas-context-detached">detached</span>
     <tbody>
      <tr>
       <th>"<dfn><code data-x="offscreen-context-type-2d">2d</code></dfn>"
       <td>
        <ol>
         <li><p>Let <var>context</var> be the result of running the
         <span>offscreen 2D context creation algorithm</span> given
         <span>this</span> and <var>options</var>.</p></li>

         <li><p>Set <span>this</span>'s <span data-x="offscreencanvas-context-mode">context
         mode</span> to <span data-x="offscreencanvas-context-2d">2d</span>.</p></li>

         <li><p>Return <var>context</var>.</p></li>
        </ol>
       <td>
        Return the same object as was returned the last time the method was invoked with this same
        first argument.
       <td>
        Return null.
       <td>
        Return null.
       <td>
        Return null.
       <td>
        Throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.
      <tr>
       <th>"<dfn><code data-x="offscreen-context-type-bitmaprenderer">bitmaprenderer</code></dfn>"
       <td>
        <ol>
         <li><p>Let <var>context</var> be the result of running the
         <span><code>ImageBitmapRenderingContext</code> creation
         algorithm</span> given <span>this</span> and <var>options</var>.</p></li>

         <li><p>Set <span>this</span>'s <span data-x="offscreencanvas-context-mode">context
         mode</span> to <span
         data-x="offscreencanvas-context-bitmaprenderer">bitmaprenderer</span>.</p></li>

         <li><p>Return <var>context</var>.</p></li>
        </ol>
       <td>
        Return null.
       <td>
        Return the same object as was returned the last time the method was invoked with this same
        first argument.
       <td>
        Return null.
       <td>
        Return null.
       <td>
        Throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.
      <tr>
       <th>"<dfn><code data-x="offscreen-context-type-webgl">webgl</code></dfn>" or "<dfn><code
       data-x="offscreen-context-type-webgl2">webgl2</code></dfn>"
       <td>
        <ol>
         <li><p>Let <var>context</var> be the result of following the instructions given in the
         WebGL specifications' <i>Context Creation</i> sections. <ref>WEBGL</ref></p></li>

         <li><p>If <var>context</var> is null, then return null; otherwise set
         <span>this</span>'s <span data-x="offscreencanvas-context-mode">context
         mode</span> to <span data-x="offscreencanvas-context-webgl">webgl</span> or <span
         data-x="offscreencanvas-context-webgl2">webgl2</span>.</p></li>

         <li><p>Return <var>context</var>.</p></li>
        </ol>
       <td>
        Return null.
       <td>
        Return null.
       <td>
        Return the same value as was returned the last time the method was invoked with this same
        first argument.
       <td>
        Return null.
       <td>
        Throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.
      <tr>
       <th>"<dfn><code data-x="offscreen-context-type-webgpu">webgpu</code></dfn>"
       <td>
        <ol>
         <li><p>Let <var>context</var> be the result of following the instructions given in
         <cite>WebGPU</cite>'s <a
         href="https://gpuweb.github.io/gpuweb/#canvas-rendering">Canvas Rendering</a> section.
         <ref>WEBGPU</ref></p></li>

         <li><p>If <var>context</var> is null, then return null; otherwise set <span>this</span>'s
         <span data-x="offscreencanvas-context-mode">context mode</span> to <span
         data-x="offscreencanvas-context-webgpu">webgpu</span>.</p></li>

         <li><p>Return <var>context</var>.</p></li>
        </ol>
       <td>
        Return null.
       <td>
        Return null.
       <td>
        Return null.
       <td>
        Return the same value as was returned the last time the method was invoked with this same
        first argument.
       <td>
        Throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.
    </table>
   </li>
  </ol>

  <hr>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>offscreenCanvas</var>.<span subdfn data-x="dom-OffscreenCanvas-width">width</span> [
   = <var>value</var> ]</code></dt>

   <dt><code data-x=""><var>offscreenCanvas</var>.<span subdfn data-x="dom-OffscreenCanvas-height">height</span> [
   = <var>value</var> ]</code></dt>

   <dd>
    <p>These attributes return the dimensions of the <code>OffscreenCanvas</code> object's <span
    data-x="offscreencanvas-bitmap">bitmap</span>.</p>

    <p>They can be set, to replace the <span data-x="offscreencanvas-bitmap">bitmap</span> with a
    new, <span>transparent black</span> bitmap of the specified dimensions (effectively resizing
    it).</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>If either the <dfn attribute for="OffscreenCanvas"><code
  data-x="dom-OffscreenCanvas-width">width</code></dfn> or <dfn attribute
  for="OffscreenCanvas"><code data-x="dom-OffscreenCanvas-height">height</code></dfn> attributes of
  an <code>OffscreenCanvas</code> object are set (to a new value or to the same value as before) and
  the <code>OffscreenCanvas</code> object's <span data-x="offscreencanvas-context-mode">context
  mode</span> is <span data-x="offscreencanvas-context-2d">2d</span>, then <span>reset the rendering
  context to its default state</span> and resize the <code>OffscreenCanvas</code> object's <span
  data-x="offscreencanvas-bitmap">bitmap</span> to the new values of the <code
  data-x="dom-OffscreenCanvas-width">width</code> and <code
  data-x="dom-OffscreenCanvas-height">height</code> attributes.</p>

  <p>The resizing behavior for "<code data-x="offscreen-context-type-webgl">webgl</code>" and "<code
  data-x="offscreen-context-type-webgl2">webgl2</code>" contexts is defined in the WebGL
  specifications. <ref>WEBGL</ref></p>

  <p>The resizing behavior for "<code data-x="offscreen-context-type-webgpu">webgpu</code>" context
  is defined in <cite>WebGPU</cite>. <ref>WEBGPU</ref></p>

  </div>

  <p class="note">If an <code>OffscreenCanvas</code> object whose dimensions were changed has
  a <span data-x="offscreencanvas-placeholder">placeholder <code>canvas</code> element</span>, then
  the <span data-x="offscreencanvas-placeholder">placeholder <code>canvas</code> element</span>'s
  <span data-x="natural dimensions">natural size</span> will only be updated during the
  <code>OffscreenCanvas</code>'s <span>relevant agent</span>'s <span
  data-x="concept-agent-event-loop">event loop</span>'s <span>update the rendering</span> steps.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>promise</var> = <var>offscreenCanvas</var>.<span subdfn data-x="dom-OffscreenCanvas-convertToBlob">convertToBlob</span>([<var>options</var>])</code></dt>

   <dd>
    <p>Returns a promise that will fulfill with a new <code>Blob</code> object representing a file
    containing the image in the <code>OffscreenCanvas</code> object.</p>

    <p>The argument, if provided, is a dictionary that controls the encoding options of the image
    file to be created. The <code data-x="image-encode-options-type">type</code> field specifies the
    file format and has a default value of "<code>image/png</code>"; that type is also used if the
    requested type isn't supported. If the image format supports variable quality (such as
    "<code>image/jpeg</code>"), then the <code data-x="image-encode-options-quality">quality</code>
    field is a number in the range 0.0 to 1.0 inclusive indicating the desired quality level for the
    resulting image.</p>
   </dd>

   <dt><code data-x=""><var>canvas</var>.<span subdfn data-x="dom-OffscreenCanvas-transferToImageBitmap">transferToImageBitmap</span>()</code></dt>

   <dd>
    <p>Returns a newly created <code>ImageBitmap</code> object with the image in the
    <code>OffscreenCanvas</code> object. The image in the <code>OffscreenCanvas</code> object is
    replaced with a new blank image.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="OffscreenCanvas"><code
  data-x="dom-OffscreenCanvas-convertToBlob">convertToBlob(<var>options</var>)</code></dfn> method
  steps are:</p>

  <ol>
   <li><p>If the value of <span>this</span>'s <span>[[Detached]]</span> internal slot is true, then
   return <span>a promise rejected with</span> an <span>"<code>InvalidStateError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>If <span>this</span>'s <span data-x="offscreencanvas-context-mode">context mode</span> is
   <span data-x="offscreencanvas-context-2d">2d</span> and the rendering context's <span>output
   bitmap</span>'s <span data-x="concept-canvas-origin-clean">origin-clean</span> flag is set to
   false, then return <span>a promise rejected with</span> a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p>

   <li><p>If <span>this</span>'s <span data-x="offscreencanvas-bitmap">bitmap</span> has no pixels
   (i.e., either its horizontal dimension or its vertical dimension is zero), then return <span>a
   promise rejected with</span> an <span>"<code>IndexSizeError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Let <var>bitmap</var> be a copy of <span>this</span>'s <span
   data-x="offscreencanvas-bitmap">bitmap</span>.</p></li>

   <li><p>Let <var>result</var> be a new promise object.</p></li>

   <li><p>Let <var>global</var> be <span>this</span>'s <span>relevant global object</span>.</p></li>

   <li>
    <p>Run these steps <span>in parallel</span>:</p>

    <ol>
     <li><p>Let <var>file</var> be <span data-x="a serialization of the bitmap as a file">a
     serialization of <var>bitmap</var> as a file</span>, with <var>options</var>'s <dfn><code
     data-x="image-encode-options-type">type</code></dfn> and <dfn><code
     data-x="image-encode-options-quality">quality</code></dfn> if present.</p></li>

     <li>
      <p><span>Queue a global task</span> on the <span>canvas blob serialization task source</span>
      given <var>global</var> to run these steps:</p>

      <ol>
       <li><p>If <var>file</var> is null, then reject <var>result</var> with an
       <span>"<code>EncodingError</code>"</span> <code>DOMException</code>.</p></li>

       <li><p>Otherwise, resolve <var>result</var> with a new <code>Blob</code> object, created in
       <var>global</var>'s <span data-x="concept-relevant-realm">relevant realm</span>, representing
       <var>file</var>. <ref>FILEAPI</ref></p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Return <var>result</var>.</p></li>
  </ol>

  <p>The <dfn method for="OffscreenCanvas"><code
  data-x="dom-OffscreenCanvas-transferToImageBitmap">transferToImageBitmap()</code></dfn> method,
  when invoked, must run the following steps:</p>
  <ol>
   <li><p>If the value of this <code>OffscreenCanvas</code> object's <span>[[Detached]]</span>
   internal slot is set to true, then throw an <span>"<code>InvalidStateError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>If this <code>OffscreenCanvas</code> object's <span
   data-x="offscreencanvas-context-mode">context mode</span> is set to <span
   data-x="offscreencanvas-context-none">none</span>, then throw an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>image</var> be a newly created <code>ImageBitmap</code> object that references
   the same underlying bitmap data as this <code>OffscreenCanvas</code> object's <span
   data-x="offscreencanvas-bitmap">bitmap</span>.</p></li>

   <li>
    <p>Set this <code>OffscreenCanvas</code> object's <span
    data-x="offscreencanvas-bitmap">bitmap</span> to reference a newly created bitmap of the same
    dimensions and color space as the previous bitmap, and with its pixels initialized to
    <span>transparent black</span>, or <span>opaque black</span> if the rendering context's <span
    data-x="concept-canvas-alpha">alpha</span> is false.</p>

    <p class="note">This means that if the rendering context of this <code>OffscreenCanvas</code> is
    a <code>WebGLRenderingContext</code>, the value of <code
    data-x="WebGLContextAttributes">preserveDrawingBuffer</code> will have no effect.
    <ref>WEBGL</ref></p>
   </li>

   <li><p>Return <var>image</var>.</p></li>
  </ol>
  </div>

  <p>The following are the <span>event handlers</span> (and their corresponding <span data-x="event
  handler event type">event handler event types</span>) <span w-nodev>that must be</span> supported,
  as <span>event handler IDL attributes</span>, by all objects implementing the
  <code>OffscreenCanvas</code> interface:</p>

  <table>
   <thead>
    <tr><th><span data-x="event handlers">Event handler</span> <th><span>Event handler event type</span>
   <tbody>
    <tr><td><dfn attribute for="OffscreenCanvas"><code data-x="handler-offscreencanvas-oncontextlost">oncontextlost</code></dfn> <td> <code data-x="event-contextlost">contextlost</code>
    <tr><td><dfn attribute for="OffscreenCanvas"><code data-x="handler-offscreencanvas-oncontextrestored">oncontextrestored</code></dfn> <td> <code data-x="event-contextrestored">contextrestored</code>
  </table>

  <h6>The offscreen 2D rendering context</h6>

  <pre><code class="idl">[Exposed=(Window,Worker)]
interface <dfn interface>OffscreenCanvasRenderingContext2D</dfn> {
  readonly attribute <span>OffscreenCanvas</span> <span data-x="offscreencontext2d-canvas">canvas</span>;
};

<span>OffscreenCanvasRenderingContext2D</span> includes <span>CanvasSettings</span>;
<span>OffscreenCanvasRenderingContext2D</span> includes <span>CanvasState</span>;
<span>OffscreenCanvasRenderingContext2D</span> includes <span>CanvasTransform</span>;
<span>OffscreenCanvasRenderingContext2D</span> includes <span>CanvasCompositing</span>;
<span>OffscreenCanvasRenderingContext2D</span> includes <span>CanvasImageSmoothing</span>;
<span>OffscreenCanvasRenderingContext2D</span> includes <span>CanvasFillStrokeStyles</span>;
<span>OffscreenCanvasRenderingContext2D</span> includes <span>CanvasShadowStyles</span>;
<span>OffscreenCanvasRenderingContext2D</span> includes <span>CanvasFilters</span>;
<span>OffscreenCanvasRenderingContext2D</span> includes <span>CanvasRect</span>;
<span>OffscreenCanvasRenderingContext2D</span> includes <span>CanvasDrawPath</span>;
<span>OffscreenCanvasRenderingContext2D</span> includes <span>CanvasText</span>;
<span>OffscreenCanvasRenderingContext2D</span> includes <span>CanvasDrawImage</span>;
<span>OffscreenCanvasRenderingContext2D</span> includes <span>CanvasImageData</span>;
<span>OffscreenCanvasRenderingContext2D</span> includes <span>CanvasPathDrawingStyles</span>;
<span>OffscreenCanvasRenderingContext2D</span> includes <span>CanvasTextDrawingStyles</span>;
<span>OffscreenCanvasRenderingContext2D</span> includes <span>CanvasPath</span>;</code></pre>

  <p>The <code>OffscreenCanvasRenderingContext2D</code> object is a rendering context for drawing to
  the <span data-x="offscreencanvas-bitmap">bitmap</span> of an <code>OffscreenCanvas</code> object.
  It is similar to the <code>CanvasRenderingContext2D</code> object, with the following
  differences:</p>

  <ul>
   <li><p>there is no support for <span data-x="CanvasUserInterface">user interface</span>
   features;</p></li>
   <li><p>its <code data-x="offscreencontext2d-canvas">canvas</code> attribute refers to an
   <code>OffscreenCanvas</code> object rather than a <code>canvas</code> element;</p></li>
  </ul>

  <p>An <code>OffscreenCanvasRenderingContext2D</code> object has an <dfn>associated
  <code>OffscreenCanvas</code> object</dfn>, which is the <code>OffscreenCanvas</code> object
  from which the <code>OffscreenCanvasRenderingContext2D</code> object was created.

  <dl class="domintro">
   <dt><code data-x=""><var>offscreenCanvas</var> = <var>offscreenCanvasRenderingContext2D</var>.<span subdfn data-x="offscreencontext2d-canvas">canvas</span></code></dt>

   <dd><p>Returns the <span>associated <code>OffscreenCanvas</code> object</span>.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn>offscreen 2D context creation algorithm</dfn>, which is passed a
  <var>target</var> (an <code>OffscreenCanvas</code> object) and optionally some arguments,
  consists of running the following steps:</p>

  <ol>
   <li><p>If the algorithm was passed some arguments, let <var>arg</var> be the first such
   argument. Otherwise, let <var>arg</var> be undefined.</p></li>

   <li><p>Let <var>settings</var> be the result of <span
   data-x="concept-idl-convert">converting</span> <var>arg</var> to the dictionary type
   <code>CanvasRenderingContext2DSettings</code>. (This can throw an exception.).</p></li>

   <li><p>Let <var>context</var> be a new <code>OffscreenCanvasRenderingContext2D</code>
   object.</p></li>

   <li><p>Set <var>context</var>'s <span>associated <code>OffscreenCanvas</code> object</span> to
   <var>target</var>.</p></li>

   <li><p>Run the <span data-x="canvas-setting-init-bitmap">canvas settings output bitmap
   initialization algorithm</span>, given <var>context</var> and <var>settings</var>.</li>

   <li><p>Set <var>context</var>'s <span>output bitmap</span> to a newly created bitmap with the
   dimensions specified by the <code data-x="dom-OffscreenCanvas-width">width</code> and <code
   data-x="dom-OffscreenCanvas-height">height</code> attributes of <var>target</var>, and set
   <var>target</var>'s bitmap to the same bitmap (so that they are shared).</p></li>

   <li><p>If <var>context</var>'s <span data-x="concept-canvas-alpha">alpha</span> flag is set to
   true, initialize all the pixels of <var>context</var>'s <span>output bitmap</span> to
   <span>transparent black</span>. Otherwise, initialize the pixels to <span>opaque
   black</span>.</p></li>

   <li><p>Return <var>context</var>.</p></li>
  </ol>

  <p class="note">Implementations are encouraged to short-circuit the graphics update steps of
  the <span>window event loop</span> for the purposes of updating the contents of a <span
  data-x="offscreencanvas-placeholder">placeholder <code>canvas</code> element</span> to the
  display. This could mean, for example, that the bitmap contents are copied directly to a graphics
  buffer that is mapped to the physical display location of the <span
  data-x="offscreencanvas-placeholder">placeholder <code>canvas</code> element</span>. This or
  similar short-circuiting approaches can significantly reduce display latency, especially in cases
  where the <code>OffscreenCanvas</code> is updated from a
  <span>worker event loop</span> and the <span>window event loop</span> of the <span
  data-x="offscreencanvas-placeholder">placeholder <code>canvas</code> element</span> is busy.
  However, such shortcuts cannot have any script-observable side-effects. This means that the
  committed bitmap still needs to be sent to the <span
  data-x="offscreencanvas-placeholder">placeholder <code>canvas</code> element</span>, in case the
  element is used as a <code>CanvasImageSource</code>, as an <code>ImageBitmapSource</code>, or in
  case <code data-x="dom-canvas-toDataURL">toDataURL()</code> or <code
  data-x="dom-canvas-toBlob">toBlob()</code> are called on it.</p>

  <p>The <dfn attribute for="OffscreenCanvas"><code
  data-x="offscreencontext2d-canvas">canvas</code></dfn> attribute, on getting, must return this
  <code>OffscreenCanvasRenderingContext2D</code>'s <span>associated <code>OffscreenCanvas</code>
  object</span>.

  </div>

  <div w-nodev>

  <!--en-GB--><h5 id="colour-spaces-and-colour-correction">Color spaces and color space
  conversion</h5>

  <p>The <code>canvas</code> APIs provide mechanisms for specifying the color space of the canvas's
  backing store. The default backing store color space for all canvas APIs is
  <span>'srgb'</span>.</p>

  <p><span data-x="Converting Colors">Color space conversion</span> must be applied to the canvas's
  backing store when rendering the canvas to the output device. This color space conversion must be
  identical to the color space conversion that would be applied to an <code>img</code> element with
  a color profile that specifies the same <span data-x="concept-canvas-color-space">color
  space</span> as the canvas's backing store.</p>

  <p>When drawing content to a 2D context, all inputs must be <span data-x="Converting
  Colors">converted</span> to the <span data-x="concept-canvas-color-space">context's color
  space</span> before drawing. Interpolation of gradient color stops must be performed on color
  values after conversion to the <span data-x="concept-canvas-color-space">context's color
  space</span>. Alpha blending must be performed on values after conversion to the <span
  data-x="concept-canvas-color-space">context's color space</span>.</p>

  <p class="note">There do not exist any inputs to a 2D context for which the color space is
  undefined. The color space for CSS colors is defined in <cite>CSS Color</cite>. The color space
  for images that specify no color profile information is assumed to be <span>'srgb'</span>, as
  specified in the <a href="https://drafts.csswg.org/css-color/#untagged">Color Spaces of Untagged
  Colors</a> section of <cite>CSS Color</cite>. <ref>CSSCOLOR</ref></p>

  </div>



  <!--en-GB--><h5 id="serialising-bitmaps-to-a-file">Serializing bitmaps to a file</h5>

  <div w-nodev>

  <p>When a user agent is to create <!--en-GB--><dfn id="a-serialisation-of-the-bitmap-as-a-file">a
  serialization of the bitmap as a file</dfn>, given a <var>type</var> and an optional
  <var>quality</var>, it must create an image file in the format given by <var>type</var>. If an
  error occurs during the creation of the image file (e.g. an internal encoder error), then the
  result of the serialization is null. <ref>PNG</ref></p>

  <p>The image file's pixel data must be the bitmap's pixel data scaled to one image pixel per
  coordinate space unit, and if the file format used supports encoding resolution metadata, the
  resolution must be given as 96dpi (one image pixel per <span data-x="'px'">CSS pixel</span>).</p>

  <p>If <var>type</var> is supplied, then it must be interpreted as a <span data-x="MIME type">MIME
  type</span> giving the format to use. If the type has any parameters, then it must be treated as
  not supported.</p>

  <p class="example">For example, the value "<code>image/png</code>" would mean to generate a PNG
  image, the value "<code>image/jpeg</code>" would mean to generate a JPEG image, and the value
  "<code>image/svg+xml</code>" would mean to generate an SVG image (which would require that the
  user agent track how the bitmap was generated, an unlikely, though potentially awesome,
  feature).</p>

  <p>User agents must support PNG ("<code>image/png</code>"). User agents may support other types.
  If the user agent does not support the requested type, then it must create the file using the PNG
  format. <ref>PNG</ref></p>

  <p>User agents must <span data-x="converted to ASCII lowercase">convert the provided type to ASCII
  lowercase</span> before establishing if they support that type.</p>

  <p>For image types that do not support an alpha component, the serialized image must be the bitmap
  image composited onto an <span>opaque black</span> background using the <span
  data-x="gcop-source-over">source-over</span> compositing operator.</p>

  <p>For image types that support color profiles, the serialized image must include a color profile
  indicating the color space of the underlying bitmap. For image types that do not support color
  profiles, the serialized image must be <span data-x="Converting Colors">converted</span> to the
  <span>'srgb'</span> color space using <span>'relative-colorimetric'</span> rendering intent.</p>

  <p class="note">Thus, in the 2D context, calling the <code
  data-x="dom-context-2d-drawImage">drawImage()</code> method to render the output of the <code
  data-x="dom-canvas-toDataURL">toDataURL()</code> or <code
  data-x="dom-canvas-toBlob">toBlob()</code> method to the canvas, given the appropriate dimensions,
  has no visible effect beyond, at most, clipping colors of the canvas to a more narrow gamut.</p>

  <p>For image types that support multiple bit depths, the serialized image must use the bit depth
  that best preserves content of the underlying bitmap.</p>

  <p class="example">For example, when serializing a 2D context that has
  <span data-x="concept-canvas-color-type">color type</span> of
  <span data-x="dom-CanvasColorType-float16">float16</span> to <var>type</var>
  "<code>image/png</code>", the resulting image would have 16 bits per sample.
  This serialization will still lose significant detail (all values less than 0.5/65535
  would be clamped to 0, and all values greater than 1 would be clamped to 1).</p>

  <p>If <var>type</var> is an image format that supports variable quality (such as
  "<code>image/jpeg</code>"), <var>quality</var> is given, and <var>type</var> is not
  "<code>image/png</code>", then, if <var>quality</var> <span data-x="js-Number">is a Number</span>
  in the range 0.0 to 1.0 inclusive, the user agent must treat <var>quality</var> as the desired
  quality level. Otherwise, the user agent must use its default quality value, as if the
  <var>quality</var> argument had not been given.</p>

  <p class="note">The use of type-testing here, instead of simply declaring <var>quality</var> as
  a Web IDL <code data-x="">double</code>, is a historical artifact.</p>

  <p class="note">Different implementations can have slightly different interpretations of
  "quality". When the quality is not specified, an implementation-specific default is used that
  represents a reasonable compromise between compression ratio, image quality, and encoding
  time.</p>

  </div>


  <div w-nodev>

  <h5>Security with <code>canvas</code> elements</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p><strong>Information leakage</strong> can occur if scripts from one <span>origin</span> can
  access information (e.g. read pixels) from images from another origin (one that isn't the <span
  data-x="same origin">same</span>).</p>

  <p>To mitigate this, bitmaps used with <code>canvas</code> elements, <code>OffscreenCanvas</code>
  objects, and <code>ImageBitmap</code> objects are defined to have a flag indicating whether they
  are <span data-x="concept-canvas-origin-clean">origin-clean</span>. All bitmaps start with their
  <span data-x="concept-canvas-origin-clean">origin-clean</span> set to true. The flag is set to
  false when cross-origin images are used.</p>

  <p>The <code data-x="dom-canvas-toDataURL">toDataURL()</code>, <code
  data-x="dom-canvas-toBlob">toBlob()</code>, and <code
  data-x="dom-context-2d-getImageData">getImageData()</code> methods check the flag and will
  throw a <span>"<code>SecurityError</code>"</span> <code>DOMException</code> rather than leak
  cross-origin data.</p>

  <p>The value of the <span data-x="concept-canvas-origin-clean">origin-clean</span> flag is
  propagated from a source's bitmap to a new <code>ImageBitmap</code> object by <code
  data-x="dom-createImageBitmap">createImageBitmap()</code>. Conversely, a destination <code
  >canvas</code> element's bitmap will have its <span
  data-x="concept-canvas-origin-clean">origin-clean</span> flags set to false by <code
  data-x="dom-context-2d-drawImage">drawImage</code> if the source image is an
  <code>ImageBitmap</code> object whose bitmap has its <span
  data-x="concept-canvas-origin-clean">origin-clean</span> flag set to false.</p>

  <p>The flag can be reset in certain situations; for example, when changing the value of the
  <code data-x="attr-canvas-width">width</code> or the <code
  data-x="attr-canvas-height">height</code> content attribute of the <code>canvas</code> element
  to which a <code>CanvasRenderingContext2D</code> is bound, the bitmap is
  cleared and its <span data-x="concept-canvas-origin-clean">origin-clean</span> flag is reset.</p>

  <p>When using an <code>ImageBitmapRenderingContext</code>, the value of the <span
  data-x="concept-canvas-origin-clean">origin-clean</span> flag is propagated from
  <code>ImageBitmap</code> objects when they are transferred to the <code>canvas</code>
  via <span
  data-x="dom-ImageBitmapRenderingContext-transferFromImageBitmap">transferFromImageBitmap()</span>.</p>

  </div>

  <h5>Premultiplied alpha and the 2D rendering context</h5>

  <p><dfn data-x="concept-premultiplied-alpha">Premultiplied alpha</dfn> refers to one way of
  representing transparency in an image, the other being non-premultiplied alpha.</p>

  <p>Under non-premultiplied alpha, the red, green, and blue components of a pixel represent that
  pixel's color, and its alpha component represents that pixel's opacity.</p>

  <p>Under premultiplied alpha, however, the red, green, and blue components of a pixel represent
  the amounts of color that the pixel adds to the image, and its alpha component represents the
  amount that the pixel obscures whatever is behind it.</p>

  <div class="example">
   <p>For instance, assuming the color components range from 0 (off) to 255 (full intensity), these
   example colors are represented in the following ways:</p>

   <table>
    <thead>
     <tr>
      <th>CSS color representation
      <th>Premultiplied representation
      <th>Non-premultiplied representation
      <th>Description of color
      <th>Image of color blended above other content
    <tbody>
     <tr>
      <td>rgba(255, 127, 0, 1)
      <td>255, 127, 0, 255
      <td>255, 127, 0, 255
      <td>Completely-opaque orange
      <td><img src="images/premultiplied-example-1.png" width="96" height="96" alt="An opaque orange circle sits atop a background">
     <tr>
      <td>rgba(255, 255, 0, 0.5)
      <td>127, 127, 0, 127
      <td>255, 255, 0, 127
      <td>Halfway-opaque yellow
      <td><img src="images/premultiplied-example-2.png" width="96" height="96" alt="A yellow circle, halfway transparent, sits atop a background">
     <tr>
      <td>Unrepresentable
      <td>255, 127, 0, 127
      <td>Unrepresentable
      <td>Additive halfway-opaque orange
      <td><img src="images/premultiplied-example-3.png" width="96" height="96" alt="An orange circle somewhat brightens the background that it sits atop">
     <tr>
      <td>Unrepresentable
      <td>255, 127, 0, 0
      <td>Unrepresentable
      <td>Additive fully-transparent orange
      <td><img src="images/premultiplied-example-4.png" width="96" height="96" alt="An orange circle completely brightens the background that it sits atop">
     <tr>
      <td>rgba(255, 127, 0, 0)
      <td>0, 0, 0, 0
      <td>255, 127, 0, 0
      <td>Fully-transparent ("invisible") orange
      <td><img src="images/premultiplied-example-5.png" width="96" height="96" alt="An empty background with nothing atop it">
     <tr>
      <td>rgba(0, 127, 255, 0)
      <td>0, 0, 0, 0
      <td>255, 127, 0, 0
      <td>Fully-transparent ("invisible") turquoise
      <td><img src="images/premultiplied-example-5.png" width="96" height="96" alt="An empty background with nothing atop it">
   </table>
  </div>

  <p><dfn data-x="convert-to-premultiplied">Converting a color value from a non-premultiplied
  representation to a premultiplied one</dfn> involves multiplying the color's red, green, and
  blue components by its alpha component (remapping the range of the alpha component such that
  "fully transparent" is 0, and "fully opaque" is 1).</p>

  <p><dfn data-x="convert-from-premultiplied">Converting a color value from a premultiplied
  representation to a non-premultiplied one</dfn> involves the inverse: dividing the color's red,
  green, and blue components by its alpha component.</p>

  <p>As certain colors can only be represented under premultiplied alpha (for instance, additive
  colors), and others can only be represented under non-premultiplied alpha (for instance,
  "invisible" colors which hold certain red, green, and blue values even with no opacity); and
  division and multiplication using finite precision entails a loss of accuracy, converting between
  premultiplied and non-premultiplied alpha is a lossy operation on colors that are not fully
  opaque.</p>

  <p>A <code>CanvasRenderingContext2D</code>'s <span>output bitmap</span> and an
  <code>OffscreenCanvasRenderingContext2D</code>'s <span>output bitmap</span> must use premultiplied
  alpha to represent transparent colors.</p>

  <p class="note">It is important for canvas bitmaps to represent colors using premultiplied alpha
  because it affects the range of representable colors. While additive colors cannot currently be
  drawn onto canvases directly because CSS colors are non-premultiplied and cannot represent them,
  it is still possible to, for instance, draw additive colors onto a WebGL canvas and then draw that
  WebGL canvas onto a 2D canvas via <code data-x="dom-context-2d-drawImage">drawImage()</code>.</p>

  <h3 split-filename="custom-elements" id="custom-elements">Custom elements</h3>

  <h4 id="custom-elements-intro">Introduction</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p><span data-x="custom element">Custom elements</span> provide a way for authors to build their
  own fully-featured DOM elements. Although authors could always use non-standard elements in their
  documents, with application-specific behavior added after the fact by scripting or similar, such
  elements have historically been non-conforming and not very functional. By <span
  data-x="element definition">defining</span> a custom element, authors can inform the parser how to
  properly construct an element and how elements of that class should react to changes.</p>

  <p>Custom elements are part of a larger effort to "rationalise the platform", by explaining
  existing platform features (like the elements of HTML) in terms of lower-level author-exposed
  extensibility points (like custom element definition). Although today there are many limitations
  on the capabilities of custom elements—both functionally and semantically—that prevent them from
  fully explaining the behaviors of HTML's existing elements, we hope to shrink this gap over
  time.</p>

  <h5 id="custom-elements-autonomous-example">Creating an autonomous custom element</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>For the purposes of illustrating how to create an <span>autonomous custom element</span>, let's
  define a custom element that encapsulates rendering a small icon for a country flag. Our goal is
  to be able to use it like so:</p>

  <pre><code class="html">&lt;flag-icon country="nl">&lt;/flag-icon></code></pre>

  <p>To do this, we first declare a class for the custom element, extending
  <code>HTMLElement</code>:</p>

  <pre><code class="js">class FlagIcon extends HTMLElement {
  constructor() {
    super();
    this._countryCode = null;
  }

  static observedAttributes = ["country"];

  attributeChangedCallback(name, oldValue, newValue) {
    // name will always be "country" due to observedAttributes
    this._countryCode = newValue;
    this._updateRendering();
  }
  connectedCallback() {
    this._updateRendering();
  }

  get country() {
    return this._countryCode;
  }
  set country(v) {
    this.setAttribute("country", v);
  }

  _updateRendering() {
    // Left as an exercise for the reader. But, you'll probably want to
    // check this.ownerDocument.defaultView to see if we've been
    // inserted into a document with a browsing context, and avoid
    // doing any work if not.
  }
}</code></pre>

  <p>We then need to use this class to define the element:</p>

  <pre><code class="js">customElements.define("flag-icon", FlagIcon);</code></pre>

  <p>At this point, our above code will work! The parser, whenever it sees the <code
  data-x="">flag-icon</code> tag, will construct a new instance of our <code
  data-x="">FlagIcon</code> class, and tell our code about its new <code data-x="">country</code>
  attribute, which we then use to set the element's internal state and update its rendering (when
  appropriate).</p>

  <p>You can also create <code data-x="">flag-icon</code> elements using the DOM API:</p>

  <pre><code class="js">const flagIcon = document.createElement("flag-icon")
flagIcon.country = "jp"
document.body.appendChild(flagIcon)</code></pre>

  <p>Finally, we can also use the <span>custom element constructor</span> itself. That is, the above
  code is equivalent to:</p>

  <pre><code class="js">const flagIcon = new FlagIcon()
flagIcon.country = "jp"
document.body.appendChild(flagIcon)</code></pre>

  <h5 id="custom-elements-face-example">Creating a form-associated custom element</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>Adding a static <code data-x="">formAssociated</code> property, with a true value, makes an
  <span>autonomous custom element</span> a <span>form-associated custom element</span>. The
  <code>ElementInternals</code> interface helps you to implement functions and properties common
  to form control elements.</p>

  <pre><code class="js">class MyCheckbox extends HTMLElement {
  static formAssociated = true;
  static observedAttributes = ['checked'];

  constructor() {
    super();
    this._internals = this.attachInternals();
    this.addEventListener('click', this._onClick.bind(this));
  }

  get form() { return this._internals.form; }
  get name() { return this.getAttribute('name'); }
  get type() { return this.localName; }

  get checked() { return this.hasAttribute('checked'); }
  set checked(flag) { this.toggleAttribute('checked', Boolean(flag)); }

  attributeChangedCallback(name, oldValue, newValue) {
    // name will always be "checked" due to observedAttributes
    this._internals.setFormValue(this.checked ? 'on' : null);
  }

  _onClick(event) {
    this.checked = !this.checked;
  }
}
customElements.define('my-checkbox', MyCheckbox);</code></pre>

  <p>You can use the custom element <code data-x="">my-checkbox</code> like a built-in
  form-associated element. For example, putting it in <code>form</code> or <code>label</code>
  associates the <code data-x="">my-checkbox</code> element with them, and submitting the
  <code>form</code> will send data provided by <code data-x="">my-checkbox</code> implementation.
  </p>

  <pre><code class="html">&lt;form action="..." method="...">
  &lt;label>&lt;my-checkbox name="agreed">&lt;/my-checkbox> I read the agreement.&lt;/label>
  &lt;input type="submit">
&lt;/form>
</code></pre>

  <h5 id="custom-elements-accessibility-example">Creating a custom element with default accessible roles, states, and properties</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>By using the appropriate properties of <code>ElementInternals</code>, your custom element can
  have default accessibility semantics. The following code expands our form-associated checkbox from
  the previous section to properly set its default role and checkedness, as viewed by accessibility
  technology:</p>

  <pre><code class="js" data-x="">class MyCheckbox extends HTMLElement {
  static formAssociated = true;
  static observedAttributes = ['checked'];

  constructor() {
    super();
    this._internals = this.attachInternals();
    this.addEventListener('click', this._onClick.bind(this));

<mark>    this._internals.role = 'checkbox';
    this._internals.ariaChecked = 'false';</mark>
  }

  get form() { return this._internals.form; }
  get name() { return this.getAttribute('name'); }
  get type() { return this.localName; }

  get checked() { return this.hasAttribute('checked'); }
  set checked(flag) { this.toggleAttribute('checked', Boolean(flag)); }

  attributeChangedCallback(name, oldValue, newValue) {
    // name will always be "checked" due to observedAttributes
    this._internals.setFormValue(this.checked ? 'on' : null);
<mark>    this._internals.ariaChecked = this.checked;</mark>
  }

  _onClick(event) {
    this.checked = !this.checked;
  }
}
customElements.define('my-checkbox', MyCheckbox);</code></pre>

  <p>Note that, like for built-in elements, these are only defaults, and can be overridden by the
  page author using the <code data-x="attr-aria-role">role</code> and <code
  data-x="attr-aria-*">aria-*</code> attributes:</p>

  <pre class="bad"><code class="html" data-x="">&lt;!-- This markup is non-conforming -->
&lt;input type="checkbox" checked role="button" aria-checked="false"></code></pre>

<pre class="bad"><code class="html" data-x="">&lt;!-- This markup is probably not what the custom element author intended -->
&lt;my-checkbox role="button" checked aria-checked="false"></code></pre>

  <p>Custom element authors are encouraged to state what aspects of their accessibility semantics
  are strong native semantics, i.e., should not be overridden by users of the custom element. In our
  example, the author of the <code data-x="">my-checkbox</code> element would state that its
  <span>role</span> and <code data-x="attr-aria-checked">aria-checked</code> values are strong
  native semantics, thus discouraging code such as the above.</p>

  <h5 id="custom-elements-customized-builtin-example">Creating a customized built-in element</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p><span data-x="customized built-in element">Customized built-in elements</span> are a distinct
  kind of <span>custom element</span>, which are defined slightly differently and used very
  differently compared to <span data-x="autonomous custom element">autonomous custom
  elements</span>. They exist to allow reuse of behaviors from the existing elements of HTML, by
  extending those elements with new custom functionality. This is important since many of the
  existing behaviors of HTML elements can unfortunately not be duplicated by using purely <span
  data-x="autonomous custom element">autonomous custom elements</span>. Instead, <span
  data-x="customized built-in element">customized built-in elements</span> allow the installation of
  custom construction behavior, lifecycle hooks, and prototype chain onto existing elements,
  essentially "mixing in" these capabilities on top of the already-existing element.</p>

  <p><span data-x="customized built-in element">Customized built-in elements</span> require a
  distinct syntax from <span data-x="autonomous custom element">autonomous custom elements</span>
  because user agents and other software key off an element's local name in order to identify the
  element's semantics and behavior. That is, the concept of <span data-x="customized built-in
  element">customized built-in elements</span> building on top of existing behavior depends
  crucially on the extended elements retaining their original local name.</p>

  <p>In this example, we'll be creating a <span>customized built-in element</span> named <code
  data-x="">plastic-button</code>, which behaves like a normal button but gets fancy animation
  effects added whenever you click on it. We start by defining a class, just like before, although
  this time we extend <code>HTMLButtonElement</code> instead of <code>HTMLElement</code>:</p>

  <pre><code class="js">class PlasticButton extends HTMLButtonElement {
  constructor() {
    super();

    this.addEventListener("click", () => {
      // Draw some fancy animation effects!
    });
  }
}</code></pre>

  <p>When defining our custom element, we have to also specify the <code
  data-x="dom-ElementDefinitionOptions-extends">extends</code> option:</p>

  <pre><code class="js">customElements.define("plastic-button", PlasticButton, { extends: "button" });</code></pre>

  <p>In general, the name of the element being extended cannot be determined simply by looking at
  what element interface it extends, as many elements share the same interface (such as
  <code>q</code> and <code>blockquote</code> both sharing <code>HTMLQuoteElement</code>).</p>

  <p>To construct our <span>customized built-in element</span> from parsed HTML source text, we use
  the <code data-x="attr-is">is</code> attribute on a <code>button</code> element:</p>

  <pre><code class="html">&lt;button is="plastic-button">Click Me!&lt;/button></code></pre>

  <p>Trying to use a <span>customized built-in element</span> as an <span>autonomous custom
  element</span> will <em>not</em> work; that is, <code data-x="">&lt;plastic-button>Click
  me?&lt;/plastic-button></code> will simply create an <code>HTMLElement</code> with no special
  behavior.</p>

  <p>If you need to create a customized built-in element programmatically, you can use the following
  form of <code data-x="dom-Document-createElement">createElement()</code>:</p>

  <pre><code class="js">const plasticButton = document.createElement("button", { is: "plastic-button" });
plasticButton.textContent = "Click me!";</code></pre>

  <p>And as before, the constructor will also work:</p>

  <pre><code class="js">const plasticButton2 = new PlasticButton();
console.log(plasticButton2.localName);  // will output "button"
console.assert(plasticButton2 instanceof PlasticButton);
console.assert(plasticButton2 instanceof HTMLButtonElement);</code></pre>

  <p>Note that when creating a customized built-in element programmatically, the <code
  data-x="attr-is">is</code> attribute will not be present in the DOM, since it was not explicitly
  set. However, <a href="#attr-is-during-serialization">it will be added to the output when
  serializing</a>:</p>

  <pre><code class="js">console.assert(!plasticButton.hasAttribute("is"));
console.log(plasticButton.outerHTML); // will output '&lt;button is="plastic-button">&lt;/button>'</code></pre>

  <p>Regardless of how it is created, all of the ways in which <code>button</code> is special
  apply to such "plastic buttons" as well: their focus behavior, ability to participate in <span
  data-x="concept-form-submit">form submission</span>, the <code
  data-x="attr-fe-disabled">disabled</code> attribute, and so on.</p>

  <p id="customized-built-in-element-restrictions"><span data-x="customized built-in
  element">Customized built-in elements</span> are designed to allow extension of existing HTML
  elements that have useful user-agent supplied behavior or APIs. As such, they can only extend
  existing HTML elements defined in this specification, and cannot extend legacy elements such as
  <code>bgsound</code>, <code>blink</code>, <code>isindex</code>, <code>keygen</code>,
  <code>multicol</code>, <code>nextid</code>, or <code>spacer</code> that have been defined to use
  <code>HTMLUnknownElement</code> as their <span>element interface</span>.</p>

  <p>One reason for this requirement is future-compatibility: if a <span>customized built-in
  element</span> was defined that extended a currently-unknown element, for example <code
  data-x="">combobox</code>, this would prevent this specification from defining a <code
  data-x="">combobox</code> element in the future, as consumers of the derived <span>customized
  built-in element</span> would have come to depend on their base element having no interesting
  user-agent-supplied behavior.</p>

  <h5 id="custom-elements-autonomous-drawbacks">Drawbacks of autonomous custom elements</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>As specified below, and alluded to above, simply defining and using an element called
  <code data-x="">taco-button</code> does not mean that such elements <span
  data-x="represents">represent</span> buttons. That is, tools such as web browsers, search engines,
  or accessibility technology will not automatically treat the resulting element as a button just
  based on its defined name.</p>

  <p>To convey the desired button semantics to a variety of users, while still using an
  <span>autonomous custom element</span>, a number of techniques would need to be employed:</p>

  <ul>
   <li><p>The addition of the <code data-x="attr-tabindex">tabindex</code> attribute would make the
   <code data-x="">taco-button</code> <span>focusable</span>. Note that if the
   <code data-x="">taco-button</code> were to become logically disabled, the <code
   data-x="attr-tabindex">tabindex</code> attribute would need to be removed.</p></li>

   <li><p>The addition of an ARIA role and various ARIA states and properties helps convey semantics
   to accessibility technology. For example, setting the <span>role</span> to "<code
   data-x="attr-aria-role-button">button</code>" will convey the semantics that this is a button,
   enabling users to successfully interact with the control using usual button-like interactions in
   their accessibility technology. Setting the <code data-x="attr-aria-label">aria-label</code>
   property is necessary to give the button an <span data-x="concept-accessible-name">accessible
   name</span>, instead of having accessibility technology traverse its child text nodes and
   announce them. And setting the <code data-x="attr-aria-disabled">aria-disabled</code> state to
   "<code data-x="">true</code>" when the button is logically disabled conveys to accessibility
   technology the button's disabled state.</p></li>

   <li><p>The addition of event handlers to handle commonly-expected button behaviors helps convey
   the semantics of the button to web browser users. In this case, the most relevant event handler
   would be one that proxies appropriate <code data-x="event-keydown">keydown</code> events to
   become <code data-x="event-click">click</code> events, so that you can activate the button both
   with keyboard and by clicking.</li>

   <li><p>In addition to any default visual styling provided for <code data-x="">taco-button</code>
   elements, the visual styling will also need to be updated to reflect changes in logical state,
   such as becoming disabled; that is, whatever style sheet has rules for <code
   data-x="">taco-button</code> will also need to have rules for <code
   data-x="">taco-button[disabled]</code>.</p></li>
  </ul>

  <p>With these points in mind, a full-featured <code data-x="">taco-button</code> that took on the
  responsibility of conveying button semantics (including the ability to be disabled) might look
  something like this:</p>

  <pre id="custom-elements-autonomous-drawbacks-example"><code
  class="js">class TacoButton extends HTMLElement {
  static observedAttributes = ["disabled"];

  constructor() {
    super();
    this._internals = this.attachInternals();
    this._internals.role = "button";

    this.addEventListener("keydown", e => {
      if (e.code === "Enter" || e.code === "Space") {
        this.dispatchEvent(new PointerEvent("click", {
          bubbles: true,
          cancelable: true
        }));
      }
    });

    this.addEventListener("click", e => {
      if (this.disabled) {
        e.preventDefault();
        e.stopImmediatePropagation();
      }
    });

    this._observer = new MutationObserver(() => {
      this._internals.ariaLabel = this.textContent;
    });
  }

  connectedCallback() {
    this.setAttribute("tabindex", "0");

    this._observer.observe(this, {
      childList: true,
      characterData: true,
      subtree: true
    });
  }

  disconnectedCallback() {
    this._observer.disconnect();
  }

  get disabled() {
    return this.hasAttribute("disabled");
  }
  set disabled(flag) {
    this.toggleAttribute("disabled", Boolean(flag));
  }

  attributeChangedCallback(name, oldValue, newValue) {
    // name will always be "disabled" due to observedAttributes
    if (this.disabled) {
      this.removeAttribute("tabindex");
      this._internals.ariaDisabled = "true";
    } else {
      this.setAttribute("tabindex", "0");
      this._internals.ariaDisabled = "false";
    }
  }
}</code></pre>

  <p>Even with this rather-complicated element definition, the element is not a pleasure to use for
  consumers: it will be continually "sprouting" <code data-x="attr-tabindex">tabindex</code>
  attributes of its own volition, and its choice of <code data-x="">tabindex="0"</code> focusability
  behavior may not match the <code>button</code> behavior on the current platform. This is because
  as of now there is no way to specify default focus behavior for custom elements, forcing the use
  of the <code data-x="attr-tabindex">tabindex</code> attribute to do so (even though it is usually
  reserved for allowing the consumer to override default behavior).</p>

  <p>In contrast, a simple <span>customized built-in element</span>, as shown in the previous
  section, would automatically inherit the semantics and behavior of the <code>button</code>
  element, with no need to implement these behaviors manually. In general, for any elements with
  nontrivial behavior and semantics that build on top of existing elements of HTML, <span
  data-x="customized built-in element">customized built-in elements</span> will be easier to
  develop, maintain, and consume.</p>

  <h5 id="custom-elements-upgrades-examples">Upgrading elements after their creation</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>Because <span>element definition</span> can occur at any time, a non-custom element could be
  <span data-x="create an element">created</span>, and then later become a <span>custom
  element</span> after an appropriate <span data-x="custom element definition">definition</span> is
  registered. We call this process "upgrading" the element, from a normal element into a custom
  element.</p>

  <p><span data-x="custom-element-upgrades">Upgrades</span> enable scenarios where it may be
  preferable for <span data-x="custom element definition">custom element definitions</span> to be
  registered after relevant elements have been initially created, such as by the parser. They allow
  progressive enhancement of the content in the custom element. For example, in the following HTML
  document the element definition for <code data-x="">img-viewer</code> is loaded
  asynchronously:</p>

  <pre><code class="html">&lt;!DOCTYPE html>
&lt;html lang="en">
&lt;title>Image viewer example&lt;/title>

&lt;img-viewer filter="Kelvin">
  &lt;img src="images/tree.jpg" alt="A beautiful tree towering over an empty savannah">
&lt;/img-viewer>

&lt;script src="js/elements/img-viewer.js" async>&lt;/script></code></pre>

  <p>The definition for the <code data-x="">img-viewer</code> element here is loaded using a
  <code>script</code> element marked with the <code data-x="attr-script-async">async</code>
  attribute, placed after the <code data-x="">&lt;img-viewer></code> tag in the markup. While the
  script is loading, the <code data-x="">img-viewer</code> element will be treated as an undefined
  element, similar to a <code>span</code>. Once the script loads, it will define the <code
  data-x="">img-viewer</code> element, and the existing <code data-x="">img-viewer</code> element on
  the page will be upgraded, applying the custom element's definition (which presumably includes
  applying an image filter identified by the string "Kelvin", enhancing the image's visual
  appearance).</p>

  <hr>

  <p>Note that <span data-x="custom-element-upgrades">upgrades</span> only apply to elements in the
  document tree. (Formally, elements that are <span>connected</span>.) An element that is not
  inserted into a document will stay un-upgraded. An example illustrates this point:</p>

  <pre><code class="html">&lt;!DOCTYPE html>
&lt;html lang="en">
&lt;title>Upgrade edge-cases example&lt;/title>

&lt;example-element>&lt;/example-element>

&lt;script>
  "use strict";

  const inDocument = document.querySelector("example-element");
  const outOfDocument = document.createElement("example-element");

  // Before the element definition, both are HTMLElement:
  console.assert(inDocument instanceof HTMLElement);
  console.assert(outOfDocument instanceof HTMLElement);

  class ExampleElement extends HTMLElement {}
  customElements.define("example-element", ExampleElement);

  // After element definition, the in-document element was upgraded:
  console.assert(inDocument instanceof ExampleElement);
  console.assert(!(outOfDocument instanceof ExampleElement));

  document.body.appendChild(outOfDocument);

  // Now that we've moved the element into the document, it too was upgraded:
  console.assert(outOfDocument instanceof ExampleElement);
&lt;/script></code></pre>

  <h5>Exposing custom element states</h5>

  <p>Built-in elements provided by user agents have certain states that can change over time
  depending on user interaction and other factors, and are exposed to web authors through <span
  data-x="pseudo-class">pseudo-classes</span>. For example, some form controls have the "invalid"
  state, which is exposed through the <code data-x="selector-invalid">:invalid</code>
  <span>pseudo-class</span>.</p>

  <p>Like built-in elements, <span data-x="custom element">custom elements</span> can have various
  states to be in too, and <span>custom element</span> authors want to expose these states in a
  similar fashion as the built-in elements.</p>

  <p>This is done via the <code data-x="selector-custom">:state()</code> pseudo-class. A custom
  element author can use the <code data-x="dom-ElementInternals-states">states</code> property of
  <code>ElementInternals</code> to add and remove such custom states, which are then exposed as
  arguments to the <code data-x="selector-custom">:state()</code> pseudo-class.

  <div class="example">
   <p>The following shows how <code data-x="selector-custom">:state()</code> can be used to style a
   custom checkbox element. Assume that <code data-x="">LabeledCheckbox</code> doesn't expose its
   "checked" state via a content attribute.</p>

   <pre><code class="html">&lt;script>
class LabeledCheckbox extends HTMLElement {
  constructor() {
    super();
    this._internals = this.attachInternals();
    this.addEventListener('click', this._onClick.bind(this));

    const shadowRoot = this.attachShadow({mode: 'closed'});
    shadowRoot.innerHTML =
      &#96;&lt;style>
       :host::before {
         content: '[ ]';
         white-space: pre;
         font-family: monospace;
       }
       :host(:state(checked))::before { content: '[x]' }
       &lt;/style>
       &lt;slot>Label&lt;/slot>&#96;;
  }

  get checked() { return this._internals.states.has('checked'); }

  set checked(flag) {
    if (flag)
      this._internals.states.add('checked');
    else
      this._internals.states.delete('checked');
  }

  _onClick(event) {
    this.checked = !this.checked;
  }
}

customElements.define('labeled-checkbox', LabeledCheckbox);
&lt;/script>

&lt;style>
labeled-checkbox { border: dashed red; }
labeled-checkbox:state(checked) { border: solid; }
&lt;/style>

&lt;labeled-checkbox>You need to check this&lt;/labeled-checkbox></code></pre>
  </div>

  <div class="example">
   <p>Custom pseudo-classes can even target shadow parts. An extension of the above example shows
   this:</p>

   <pre><code class="html">&lt;script>
class QuestionBox extends HTMLElement {
  constructor() {
    super();
    const shadowRoot = this.attachShadow({mode: 'closed'});
    shadowRoot.innerHTML =
      &#96;&lt;div>&lt;slot>Question&lt;/slot>&lt;/div>
       &lt;labeled-checkbox part='checkbox'>Yes&lt;/labeled-checkbox>&#96;;
  }
}
customElements.define('question-box', QuestionBox);
&lt;/script>

&lt;style>
question-box::part(checkbox) { color: red; }
question-box::part(checkbox):state(checked) { color: green; }
&lt;/style>

&lt;question-box>Continue?&lt;/question-box></code></pre>
  </div>

  <h4 id="custom-element-conformance">Requirements for custom element constructors and
  reactions</h4>

  <p>When authoring <span data-x="custom element constructor">custom element constructors</span>,
  authors are bound by the following conformance requirements:</p>

  <ul>
   <li><p>A parameter-less call to <code data-x="">super()</code> must be the first statement in the
   constructor body, to establish the correct prototype chain and <b>this</b> value before any
   further code is run.</p></li>

   <li><p>A <code data-x="">return</code> statement must not appear anywhere inside the constructor
   body, unless it is a simple early-return (<code data-x="">return</code> or <code data-x="">return
   this</code>).</p></li>

   <li><p>The constructor must not use the <code data-x="dom-document-write">document.write()</code>
   or <code data-x="dom-document-open">document.open()</code> methods.</p></li>

   <li><p>The element's attributes and children must not be inspected, as in the non-<span
   data-x="custom-element-upgrades">upgrade</span> case none will be present, and relying on
   upgrades makes the element less usable.</p></li>

   <li><p>The element must not gain any attributes or children, as this violates the expectations of
   consumers who use the <code data-x="dom-Document-createElement">createElement</code> or <code
   data-x="dom-Document-createElementNS">createElementNS</code> methods.</p></li>

   <li><p>In general, work should be deferred to <code data-x="">connectedCallback</code> as much as
   possible—especially work involving fetching resources or rendering. However, note that <code
   data-x="">connectedCallback</code> can be called more than once, so any initialization work that
   is truly one-time will need a guard to prevent it from running twice.</p></li>

   <li><p>In general, the constructor should be used to set up initial state and default values, and
   to set up event listeners and possibly a <span>shadow root</span>.</p></li>
  </ul>

  <p>Several of these requirements are checked during <span data-x="create an element">element
  creation</span>, either directly or indirectly, and failing to follow them will result in a custom
  element that cannot be instantiated by the parser or DOM APIs. This is true even if the work is
  done inside a constructor-initiated <span>microtask</span>, as a <span data-x="perform a microtask
  checkpoint">microtask checkpoint</span> can occur immediately after construction.</p>

  <p>When authoring <span data-x="concept-custom-element-reaction">custom element reactions</span>,
  authors should avoid manipulating the node tree as this can lead to unexpected results.</p>

  <div class="example">
   <p>An element's <code data-x="">connectedCallback</code> can be queued before the element is
   disconnected, but as the callback queue is still processed, it results in a <code
   data-x="">connectedCallback</code> for an element that is no longer connected:</p>

   <pre><code class="js">class CParent extends HTMLElement {
  connectedCallback() {
    this.firstChild.remove();
  }
}
customElements.define("c-parent", CParent);

class CChild extends HTMLElement {
  connectedCallback() {
    console.log("CChild connectedCallback: isConnected =", this.isConnected);
  }
}
customElements.define("c-child", CChild);

const parent = new CParent(),
      child = new CChild();
parent.append(child);
document.body.append(parent);

// Logs:
// CChild connectedCallback: isConnected = false</code></pre>
  </div>

  <h5>Preserving custom element state when moved</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>When manipulating the DOM tree, an element can be <span
  data-x="concept-node-move-ext">moved</span> in the tree while connected. This applies to custom
  elements as well. By default, the "<code data-x="">disconnectedCallback</code>" and "<code
  data-x="">connectedCallback</code>" would be called on the element, one after the other. This is
  done to maintain compatibility with existing custom elements that predate the <code
  data-x="dom-ParentNode-moveBefore">moveBefore()</code> method. This means that by default, custom
  elements reset their state as if they were removed and re-inserted. In the example <a
  href="#custom-elements-autonomous-drawbacks-example">above</a>, the impact would be that the
  observer would be disconnected and re-connected, and the tab index would be reset.</p>

  <p>To opt in to a state-preserving behavior while <span
  data-x="concept-node-move-ext">moving</span>, the author can implement a "<code
  data-x="">connectedMoveCallback</code>". The existence of this callback, even if empty, would
  supercede the default behavior of calling "<code data-x="">disconnectedCallback</code>" and "<code
  data-x="">connectedCallback</code>". "<code data-x="">connectedMoveCallback</code>" can also be an
  appropriate place to execute logic that depends on the element's ancestors. For example:</p>

  <pre><code class="js">class TacoButton extends HTMLElement {
  static observedAttributes = ["disabled"];

  constructor() {
    super();
    this._internals = this.attachInternals();
    this._internals.role = "button";

    this._observer = new MutationObserver(() => {
      this._internals.ariaLabel = this.textContent;
    });
  }

  _notifyMain() {
    if (this.parentElement.tagName === "MAIN") {
      // Execute logic that depends on ancestors.
    }
  }

  connectedCallback() {
    this.setAttribute("tabindex", "0");

    this._observer.observe(this, {
      childList: true,
      characterData: true,
      subtree: true
    });

    this._notifyMain();
  }

  disconnectedCallback() {
    this._observer.disconnect();
  }

  // Implementing this function would avoid resetting the tab index or re-registering the
  // mutation observer when this element is moved inside the DOM without being disconnected.
  connectedMoveCallback() {
    // The parent can change during a state-preserving move.
    this._notifyMain();
  }
}</code></pre>

  <h4 id="custom-elements-core-concepts">Core concepts</h4>

  <p>A <dfn export>custom element</dfn> is an element that is <span
  data-x="concept-element-custom">custom</span>. Informally, this means that its constructor and
  prototype are defined by the author, instead of by the user agent. This author-supplied
  constructor function is called the <dfn export>custom element constructor</dfn>.</p>

  <p>Two distinct types of <span data-x="custom element">custom elements</span> can be defined:</p>

  <ol>
   <li><p>An <dfn export>autonomous custom element</dfn>, which is defined with no <code
   data-x="dom-ElementDefinitionOptions-extends">extends</code> option. These types of custom
   elements have a local name equal to their <span
   data-x="concept-custom-element-definition-name">defined name</span>.</p></li>

   <li><p>A <dfn export>customized built-in element</dfn>, which is defined with an <code
   data-x="dom-ElementDefinitionOptions-extends">extends</code> option. These types of custom
   elements have a local name equal to the value passed in their <code
   data-x="dom-ElementDefinitionOptions-extends">extends</code> option, and their <span
   data-x="concept-custom-element-definition-name">defined name</span> is used as the value of the
   <dfn element-attr for="html-global"><code data-x="attr-is">is</code></dfn> attribute, which
   therefore must be a <span>valid custom element name</span>.</p></li>
  </ol>

  <p>After a <span>custom element</span> is <span data-x="create an element">created</span>,
  changing the value of the <code data-x="attr-is">is</code> attribute does not
  change the element's behavior<span w-nodev>, as it is saved on the element as its <span
  data-x="concept-element-is-value"><code data-x="">is</code> value</span></span>.</p>

  <p><span data-x="autonomous custom element">Autonomous custom elements</span> have the following
  element definition:</p>

  <dl class="element">
   <dt><span data-x="concept-element-categories">Categories</span>:</dt>
   <dd><span>Flow content</span>.</dd>
   <dd><span>Phrasing content</span>.</dd>
   <dd><span>Palpable content</span>.</dd>
   <dd>For <span data-x="form-associated custom element">form-associated custom elements</span>: <span data-x="category-listed">Listed</span>, <span data-x="category-label">labelable</span>, <span data-x="category-submit">submittable</span>, and <span data-x="category-reset">resettable</span> <span>form-associated element</span>.</dd>
   <dt><span data-x="concept-element-contexts">Contexts in which this element can be used</span>:</dt>
   <dd>Where <span>phrasing content</span> is expected.</dd>
   <dt><span data-x="concept-element-content-model">Content model</span>:</dt>
   <dd><span>Transparent</span>.</dd>
   <dt><span data-x="concept-element-attributes">Content attributes</span>:</dt>
   <dd><span>Global attributes</span>, except the <code data-x="attr-is">is</code> attribute<!-- no-annotate --></dd>
   <dd><code data-x="attr-fae-form">form</code>, for <span data-x="form-associated custom element">form-associated custom elements</span></dd>
   <dd><code data-x="attr-fe-disabled">disabled</code>, for <span data-x="form-associated custom element">form-associated custom elements</span></dd>
   <dd><code data-x="attr-face-readonly">readonly</code>, for <span data-x="form-associated custom element">form-associated custom elements</span></dd>
   <dd><code data-x="attr-fe-name">name</code>, for <span data-x="form-associated custom element">form-associated custom elements</span></dd>
   <dd>Any other attribute that has no namespace (see prose).</dd>
   <dt><span
   data-x="concept-element-accessibility-considerations">Accessibility considerations</span>:</dt>
   <dd>For <span data-x="form-associated custom element">form-associated custom elements</span>: <a
   href="https://w3c.github.io/html-aria/#el-form-associated-custom-element">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-form-associated-custom-element">for implementers</a>.</dd>
   <dd>Otherwise: <a
   href="https://w3c.github.io/html-aria/#el-autonomous-custom-element">for authors</a>; <a
   href="https://w3c.github.io/html-aam/#el-autonomous-custom-element">for implementers</a>.</dd>
   <dt><span data-x="concept-element-dom">DOM interface</span>:</dt>
   <dd>Supplied by the element's author (inherits from <code>HTMLElement</code>)</dd>
  </dl>


  <p>An <span>autonomous custom element</span> does not have any special meaning: it
  <span>represents</span> its children. A <span>customized built-in element</span> inherits the
  semantics of the element that it extends.</p>

  <p>Any namespace-less attribute that is relevant to the element's functioning, as determined by
  the element's author, may be specified on an <span>autonomous custom element</span>, so long as
  the attribute name is <span>XML-compatible</span> and contains no <span data-x="ASCII upper
  alpha">ASCII upper alphas</span>. The exception is the <code data-x="attr-is">is</code> attribute,
  which must not be specified on an <span>autonomous custom element</span> (and which will have no
  effect if it is).</p>

  <p><span data-x="customized built-in element">Customized built-in elements</span> follow the
  normal requirements for attributes, based on the elements they extend. To add custom
  attribute-based behavior, use <code data-x="attr-data-*">data-*</code> attributes.</p>

  <hr>

  <p>An <span>autonomous custom element</span> is called a <dfn export>form-associated custom
  element</dfn> if the element is associated with a <span>custom element definition</span> whose
  <span data-x="concept-custom-element-definition-form-associated">form-associated</span> field is
  set to true.</p>

  <p>The <code data-x="attr-fe-name">name</code> attribute represents the <span>form-associated
  custom element</span>'s name. The <code data-x="attr-fe-disabled">disabled</code> attribute is
  used to make the <span>form-associated custom element</span> non-interactive and to prevent its
  <span data-x="face-submission-value">submission value</span> from being submitted. The <code
  data-x="attr-fae-form">form</code> attribute is used to explicitly associate the
  <span>form-associated custom element</span> with its <span>form owner</span>.</p>

  <p>The <dfn element-attr for="form-associated custom elements"
  data-x="attr-face-readonly"><code>readonly</code></dfn> attribute of <span data-x="form-associated
  custom element">form-associated custom elements</span> specifies that the element is <span>barred
  from constraint validation</span>. User agents don't provide any other behavior for the attribute,
  but custom element authors should, where possible, use its presence to make their control
  non-editable in some appropriate fashion, similar to the behavior for the <span
  data-x="attr-input-readonly">readonly</span> attribute on built-in form controls.</p>

  <p><strong>Constraint validation</strong>: If the <code
  data-x="attr-face-readonly">readonly</code> attribute is specified on a <span>form-associated
  custom element</span>, the element is <span>barred from constraint validation</span>.</p>

  <p>The <span data-x="concept-form-reset-control">reset algorithm</span> for <span
  data-x="form-associated custom element">form-associated custom elements</span> is to <span>enqueue
  a custom element callback reaction</span> with the element, callback name "<code
  data-x="">formResetCallback</code>", and « ».</p>

  <hr>

  <p>A <dfn export>valid custom element name</dfn> is a sequence of characters <var>name</var> that
  meets all of the following requirements:</p>

  <ul>
   <li>
    <p><var>name</var> must match the <code
    data-x="prod-PotentialCustomElementName">PotentialCustomElementName</code> production:</p>

    <dl>
     <dt><code data-x=""><dfn
     data-x="prod-PotentialCustomElementName">PotentialCustomElementName</dfn> ::=</code></dt>
     <dd><code data-x="">[a-z] (<span data-x="prod-PCENChar">PCENChar</span>)* '-'
     (<span data-x="prod-PCENChar">PCENChar</span>)*</code></dd>

     <dt><code data-x=""><dfn data-x="prod-PCENChar">PCENChar</dfn> ::=</code></dt>
     <dd><code data-x="">"-" | "." | [0-9] | "_" | [a-z] | #xB7 | [#xC0-#xD6] | [#xD8-#xF6] |
     [#xF8-#x37D] | [#x37F-#x1FFF] | [#x200C-#x200D] | [#x203F-#x2040] | [#x2070-#x218F] |
     [#x2C00-#x2FEF] | [#x3001-#xD7FF] | [#xF900-#xFDCF] | [#xFDF0-#xFFFD] |
     [#x10000-#xEFFFF]</code></dd>
    </dl>

    <p>This uses the <a href="https://www.w3.org/TR/xml/#sec-notation">EBNF notation</a> from the
    <cite>XML</cite> specification. <ref>XML</ref></p>
   </li>

   <li>
    <p><var>name</var> must not be any of the following:</p>

    <ul class="brief">
     <li><code data-x="">annotation-xml</code></li>
     <li><code data-x="">color-profile</code></li>
     <li><code data-x="">font-face</code></li>
     <li><code data-x="">font-face-src</code></li>
     <li><code data-x="">font-face-uri</code></li>
     <li><code data-x="">font-face-format</code></li>
     <li><code data-x="">font-face-name</code></li>
     <li><code data-x="">missing-glyph</code></li>
    </ul>

    <p class="note">The list of names above is the summary of all hyphen-containing element names
    from the <span data-x="other applicable specifications">applicable specifications</span>, namely
    <cite>SVG 2</cite> and <cite>MathML</cite>. <ref>SVG</ref> <ref>MATHML</ref></p>
   </li>
  </ul>

  <div class="note">
   <p>These requirements ensure a number of goals for <span data-x="valid custom element name">valid
   custom element names</span>:</p>

   <ul>
    <li><p>They start with an <span>ASCII lower alpha</span>, ensuring that the HTML parser will
    treat them as tags instead of as text.</li>

    <li><p>They do not contain any <span data-x="ASCII upper alpha">ASCII upper alphas</span>,
    ensuring that the user agent can always treat HTML elements ASCII-case-insensitively.</p></li>

    <li><p>They contain a hyphen, used for namespacing and to ensure forward compatibility (since no
    elements will be added to HTML, SVG, or MathML with hyphen-containing local names in the
    future).</p></li>

    <li><p>They can always be created with <code
    data-x="dom-Document-createElement">createElement()</code> and <code
    data-x="dom-Document-createElementNS">createElementNS()</code>, which have restrictions that go
    beyond the parser's.</p></li>
   </ul>

   <p>Apart from these restrictions, a large variety of names is allowed, to give maximum
   flexibility for use cases like <code data-x="">&lt;math-α></code> or <code
   data-x="">&lt;emotion-😍></code>.</p>
  </div>

  <div w-nodev>

  <p>A <dfn>custom element definition</dfn> describes a <span>custom element</span> and consists
  of:</p>

  <dl>
   <dt>A <dfn data-x="concept-custom-element-definition-name" export
   for="custom element definition">name</dfn></dt>
   <dd>A <span>valid custom element name</span></dd>

   <dt>A <dfn data-x="concept-custom-element-definition-local-name" export
   for="custom element definition">local name</dfn></dt>
   <dd>A local name</dd>

   <dt>A <dfn data-x="concept-custom-element-definition-constructor" export
   for="custom element definition">constructor</dfn></dt>
   <dd>A Web IDL <code>CustomElementConstructor</code> callback function type value wrapping
   the <span>custom element constructor</span></dd>

   <dt>A list of <dfn data-x="concept-custom-element-definition-observed-attributes">observed
   attributes</dfn></dt>
   <dd>A <code data-x="">sequence&lt;DOMString></code></dd>

   <dt>A collection of <dfn
   data-x="concept-custom-element-definition-lifecycle-callbacks">lifecycle callbacks</dfn></dt>
   <dd>A map, whose keys are the strings "<code data-x="">connectedCallback</code>", "<code
   data-x="">disconnectedCallback</code>", "<code data-x="">adoptedCallback</code>", "<code
   data-x="">connectedMoveCallback</code>", "<code data-x="">attributeChangedCallback</code>",
   "<code data-x="">formAssociatedCallback</code>", "<code data-x="">formDisabledCallback</code>",
   "<code data-x="">formResetCallback</code>", and "<code
   data-x="">formStateRestoreCallback</code>". The corresponding values are either a Web IDL <code
   data-x="idl-Function">Function</code> callback function type value, or null. By default the value
   of each entry is null.</dd>

   <dt>A <dfn data-x="concept-custom-element-definition-construction-stack">construction
   stack</dfn></dt>
   <dd>A list, initially empty, that is manipulated by the <span
   data-x="concept-upgrade-an-element">upgrade an element</span> algorithm and the <a
   href="#html-element-constructors">HTML element constructors</a>. Each entry in the list will be
   either an element or an <dfn data-x="concept-already-constructed-marker"><i>already
   constructed</i> marker</dfn>.</dd>

   <dt>A <dfn data-x="concept-custom-element-definition-form-associated">form-associated</dfn>
   boolean</dt>
   <dd>If this is true, user agent treats elements associated to this
   <span>custom element definition</span> as <span data-x="form-associated custom
   element">form-associated custom elements</span>.</dd>

   <dt>A <dfn data-x="concept-custom-element-definition-disable-internals">disable
   internals</dfn> boolean</dt>
   <dd>Controls <code data-x="dom-attachInternals">attachInternals()</code>.

   <dt>A <dfn data-x="concept-custom-element-definition-disable-shadow" export
   for="custom element definition">disable shadow</dfn> boolean</dt>
   <dd>Controls <code data-x="dom-Element-attachShadow">attachShadow()</code>.
  </dl>

  <p>To <dfn export>look up a custom element definition</dfn>, given a <var>document</var>,
  <var>namespace</var>, <var>localName</var>, and <var>is</var>, perform the following steps. They
  will return either a <span>custom element definition</span> or null:</p>

  <ol>
   <li><p>If <var>namespace</var> is not the <span>HTML namespace</span>, then return null.</p></li>

   <li><p>If <var>document</var>'s <span data-x="concept-document-bc">browsing context</span> is
   null, then return null.</p></li>

   <li><p>Let <var>registry</var> be <var>document</var>'s <span>relevant global object</span>'s
   <span data-x="global-custom-element-registry">custom element registry</span>.</p></li>

   <li><p>If <var>registry</var>'s <span>custom element definition set</span> <span data-x="list
   contains">contains</span> an item with <span
   data-x="concept-custom-element-definition-name">name</span> and <span
   data-x="concept-custom-element-definition-local-name">local name</span> both equal to
   <var>localName</var>, then return that item.</p></li>

   <li><p>If <var>registry</var>'s <span>custom element definition set</span> <span data-x="list
   contains">contains</span> an item with <span
   data-x="concept-custom-element-definition-name">name</span> equal to <var>is</var> and <span
   data-x="concept-custom-element-definition-local-name">local name</span> equal to
   <var>localName</var>, then return that item.</p></li>

   <li><p>Return null.</p></li>
  </ol>

  </div>

  <h4 id="custom-elements-api">The <code>CustomElementRegistry</code> interface</h4>

  <p w-nodev>Each <code>Window</code> object has an associated <dfn
  data-x="global-custom-element-registry">custom element registry</dfn> (a
  <code>CustomElementRegistry</code> object). It is set to a new
  <code>CustomElementRegistry</code> object when the <code>Window</code> object is created.</p>

  <p class="note">Custom element registries are associated with <code>Window</code> objects, instead
  of <code>Document</code> objects, since each <span>custom element constructor</span> inherits from
  the <code>HTMLElement</code> interface, and there is exactly one <code>HTMLElement</code>
  interface per <code>Window</code> object.</p>

  <div w-nodev>

  <p>The <dfn attribute for="Window"><code
  data-x="dom-window-customElements">customElements</code></dfn> attribute of the
  <code>Window</code> interface must return the <code>CustomElementRegistry</code> object for that
  <code>Window</code> object.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>CustomElementRegistry</dfn> {
  [<span>CEReactions</span>] undefined <span data-x="dom-CustomElementRegistry-define">define</span>(DOMString name, <span>CustomElementConstructor</span> constructor, optional <span>ElementDefinitionOptions</span> options = {});
  (<span>CustomElementConstructor</span> or undefined) <span data-x="dom-CustomElementRegistry-get">get</span>(DOMString name);
  DOMString? <span data-x="dom-CustomElementRegistry-getName">getName</span>(CustomElementConstructor constructor);
  <span data-x="idl-Promise">Promise</span>&lt;<span>CustomElementConstructor</span>&gt; <span data-x="dom-CustomElementRegistry-whenDefined">whenDefined</span>(DOMString name);
  [<span>CEReactions</span>] undefined <span data-x="dom-CustomElementRegistry-upgrade">upgrade</span>(<span>Node</span> root);
};

callback <dfn callback>CustomElementConstructor</dfn> = <span>HTMLElement</span> ();

dictionary <dfn dictionary>ElementDefinitionOptions</dfn> {
  DOMString <dfn dict-member for="ElementDefinitionOptions" data-x="dom-ElementDefinitionOptions-extends">extends</dfn>;
};</code></pre>

  <p>Every <code>CustomElementRegistry</code> has a <dfn>custom element definition set</dfn>, a
  <span>set</span> of <span data-x="custom element definition">custom element definitions</span>,
  initially « ». Lookup of items in this <span>set</span> uses their <span
  data-x="concept-custom-element-definition-name">name</span>, <span
  data-x="concept-custom-element-definition-local-name">local name</span>, or <span
  data-x="concept-custom-element-definition-constructor">constructor</span>.</p>

  <p>Every <code>CustomElementRegistry</code> also has an <dfn>element definition is running</dfn>
  boolean which is used to prevent reentrant invocations of <span>element definition</span>. It is
  initially false.</p>

  <p>Every <code>CustomElementRegistry</code> also has a <dfn>when-defined promise map</dfn>, a
  <span data-x="ordered map">map</span> of <span data-x="valid custom element name">valid custom
  element names</span> to promises. It is used to implement the <code
  data-x="dom-CustomElementRegistry-whenDefined">whenDefined()</code> method.</p>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>window</var>.<span data-x="dom-window-customElements">customElements</span>.<span subdfn data-x="dom-CustomElementRegistry-define">define</span>(<var>name</var>,
   <var>constructor</var>)</code></dt>

   <dd>Defines a new <span>custom element</span>, mapping the given name to the given constructor
   as an <span>autonomous custom element</span>.</dd>


   <dt><code data-x=""><var>window</var>.<span data-x="dom-window-customElements">customElements</span>.<span data-x="dom-CustomElementRegistry-define">define</span>(<var>name</var>, <var>constructor</var>,
   { extends: <var>baseLocalName</var> })</code></dt>

   <dd>Defines a new <span>custom element</span>, mapping the given name to the given constructor as
   a <span>customized built-in element</span> for the <span>element type</span> identified by the
   supplied <var>baseLocalName</var>. A <span>"<code>NotSupportedError</code>"</span>
   <code>DOMException</code> will be thrown upon trying to extend a <span>custom element</span> or
   an unknown element.</dd>

   <dt><code data-x=""><var>window</var>.<span data-x="dom-window-customElements">customElements</span>.<span subdfn data-x="dom-CustomElementRegistry-get">get</span>(<var>name</var>)</code></dt>

   <dd>Retrieves the <span>custom element constructor</span> defined for the given <span data-x="concept-custom-element-definition-name">name</span>. Returns undefined if there is no
   <span>custom element definition</span> with the given <span data-x="concept-custom-element-definition-name">name</span>.</dd>

   <dt><code data-x=""><var>window</var>.<span data-x="dom-window-customElements">customElements</span>.<span subdfn data-x="dom-CustomElementRegistry-getName">getName</span>(<var>constructor</var>)</code></dt>

   <dd>Retrieves the given name for a <span>custom element</span> defined for the given <span data-x="concept-custom-element-definition-constructor">constructor</span>. Returns null if there is no
   <span>custom element definition</span> with the given <span data-x="concept-custom-element-definition-constructor">constructor</span>.</dd>

   <dt><code data-x=""><var>window</var>.<span data-x="dom-window-customElements">customElements</span>.<span subdfn data-x="dom-CustomElementRegistry-whenDefined">whenDefined</span>(<var>name</var>)</code></dt>

   <dd>Returns a promise that will be fulfilled with the <span>custom element</span>'s constructor
   when a <span>custom element</span> becomes defined with the given name. (If such a <span>custom
   element</span> is already defined, the returned promise will be immediately fulfilled.) Returns a
   promise rejected with a <span>"<code>SyntaxError</code>"</span> <code>DOMException</code> if not
   given a <span>valid custom element name</span>.</dd>

   <dt><code data-x=""><var>window</var>.<span data-x="dom-window-customElements">customElements</span>.<span subdfn data-x="dom-CustomElementRegistry-upgrade">upgrade</span>(<var>root</var>)</code></dt>

   <dd><span data-x="concept-try-upgrade">Tries to upgrade</span> all <span>shadow-including
   inclusive descendant</span> elements of <var>root</var>, even if they are not
   <span>connected</span>.</dd>
  </dl>

  <p><dfn>Element definition</dfn> is a process of adding a <span>custom element definition</span>
  to the <code>CustomElementRegistry</code>. This is accomplished by the <code
  data-x="dom-CustomElementRegistry-define">define()</code> method. <span w-nodev>The <dfn method
  for="CustomElementRegistry"><code
  data-x="dom-CustomElementRegistry-define">define(<var>name</var>, <var>constructor</var>,
  <var>options</var>)</code></dfn> method steps are:</span></p>

  <div w-nodev>

  <ol>
   <li><p>If <span>IsConstructor</span>(<var>constructor</var>) is false, then throw a
   <code>TypeError</code>.</p></li>

   <li><p>If <var>name</var> is not a <span>valid custom element name</span>, then throw a
   <span>"<code>SyntaxError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <span>this</span>'s <span>custom element definition set</span> <span data-x="list
   contains">contains</span> an item with <span
   data-x="concept-custom-element-definition-name">name</span> <var>name</var>, then throw a
   <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <span>this</span>'s <span>custom element definition set</span> <span data-x="list
   contains">contains</span> an item with <span
   data-x="concept-custom-element-definition-constructor">constructor</span> <var>constructor</var>,
   then throw a <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>localName</var> be <var>name</var>.</p></li>

   <li><p>Let <var>extends</var> be <var>options</var>["<code
   data-x="dom-ElementDefinitionOptions-extends">extends</code>"] if it <span data-x="map
   exists">exists</span>; otherwise null.</p></li>

   <li>
    <p>If <var>extends</var> is not null:</p>

    <ol>
     <li><p>If <var>extends</var> is a <span>valid custom element name</span>, then throw a
     <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>If the <span>element interface</span> for <var>extends</var> and the <span>HTML
     namespace</span> is <code>HTMLUnknownElement</code> (e.g., if <var>extends</var> does not
     indicate an element definition in this specification), then throw a
     <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>Set <var>localName</var> to <var>extends</var>.</p></li>
    </ol>
   </li>

   <li><p>If <span>this</span>'s <span>element definition is running</span> is true, then throw
   a <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Set <span>this</span>'s <span>element definition is running</span> to true.</p></li>

   <li><p>Let <var>formAssociated</var> be false.</p></li>

   <li><p>Let <var>disableInternals</var> be false.

   <li><p>Let <var>disableShadow</var> be false.

   <li><p>Let <var>observedAttributes</var> be an empty <code
   data-x="">sequence&lt;DOMString></code>.</p></li>

   <li>
    <p>Run the following steps while catching any exceptions:</p>

    <ol>
     <li><p>Let <var>prototype</var> be ? <span data-x="js-Get">Get</span>(<var>constructor</var>,
     "prototype").</p></li>

     <li><p>If <var>prototype</var> <span data-x="js-Object">is not an Object</span>, then throw a
     <code>TypeError</code> exception.</p></li>

     <li><p>Let <var>lifecycleCallbacks</var> be the <span>ordered map</span> «[ "<code
     data-x="">connectedCallback</code>" → null, "<code data-x="">disconnectedCallback</code>" →
     null, "<code data-x="">adoptedCallback</code>" → null, "<code
     data-x="">connectedMoveCallback</code>" → null, "<code
     data-x="">attributeChangedCallback</code>" → null ]».</p>

     <li>
      <p>For each <var>callbackName</var> of <span data-x="map get the keys">the keys</span> of
      <var>lifecycleCallbacks</var>:</p>

      <ol>
       <li><p>Let <var>callbackValue</var> be ? <span data-x="js-Get">Get</span>(<var>prototype</var>,
       <var>callbackName</var>).</p>

       <li><p>If <var>callbackValue</var> is not undefined, then <span data-x="map set">set</span>
       <var>lifecycleCallbacks</var>[<var>callbackName</var>] to the result of <span
       data-x="concept-idl-convert">converting</span> <var>callbackValue</var> to the Web IDL <code
       data-x="idl-Function">Function</code> callback type.</p></li>
      </ol>
     </li>

     <li>
      <p>If <var>lifecycleCallbacks</var>["<code data-x="">attributeChangedCallback</code>"] is not
      null:</p>

      <ol>
       <li><p>Let <var>observedAttributesIterable</var> be ? <span
       data-x="js-Get">Get</span>(<var>constructor</var>, "observedAttributes").</p></li>

       <li><p>If <var>observedAttributesIterable</var> is not undefined, then set
       <var>observedAttributes</var> to the result of <span
       data-x="concept-idl-convert">converting</span> <var>observedAttributesIterable</var> to a
       <code data-x="">sequence&lt;DOMString></code>. Rethrow any exceptions from the
       conversion.</p></li>
      </ol>

     <li><p>Let <var>disabledFeatures</var> be an empty <code
     data-x="">sequence&lt;DOMString></code>.</p></li>

     <li><p>Let <var>disabledFeaturesIterable</var> be ? <span
     data-x="js-Get">Get</span>(<var>constructor</var>, "disabledFeatures").</p></li>

     <li><p>If <var>disabledFeaturesIterable</var> is not undefined, then set
     <var>disabledFeatures</var> to the result of <span
     data-x="concept-idl-convert">converting</span> <var>disabledFeaturesIterable</var> to a <code
     data-x="">sequence&lt;DOMString></code>. Rethrow any exceptions from the conversion.</p></li>

     <li><p>If <var>disabledFeatures</var> <span data-x="list contains">contains</span> "<code
     data-x="">internals</code>", then set <var>disableInternals</var> to true.</p></li>

     <li><p>If <var>disabledFeatures</var> <span data-x="list contains">contains</span> "<code
     data-x="">shadow</code>", then set <var>disableShadow</var> to true.</p></li>

     <li><p>Let <var>formAssociatedValue</var> be ? <span data-x="js-Get">Get</span>(
     <var>constructor</var>, "formAssociated").</p></li>

     <li><p>Set <var>formAssociated</var> to the result of <span
     data-x="concept-idl-convert">converting</span> <var>formAssociatedValue</var> to a <code
     data-x="">boolean</code>.</p></li>

     <li>
      <p>If <var>formAssociated</var> is true, then for each <var>callbackName</var> of « "<code
      data-x="">formAssociatedCallback</code>", "<code data-x="">formResetCallback</code>", "<code
      data-x="">formDisabledCallback</code>", "<code data-x="">formStateRestoreCallback</code>"
      »:</p>

      <ol>
       <li><p>Let <var>callbackValue</var> be ? <span data-x="js-Get">Get</span>(<var>prototype</var>,
       <var>callbackName</var>).</p></li>

       <li><p>If <var>callbackValue</var> is not undefined, then <span data-x="map set">set</span>
       <var>lifecycleCallbacks</var>[<var>callbackName</var>] to the result of <span
       data-x="concept-idl-convert">converting</span> <var>callbackValue</var> to the Web IDL <code
       data-x="idl-Function">Function</code> callback type.</p></li>
      </ol>
     </li>
    </ol>

    <p>Then, regardless of whether the above steps threw an exception or not: set
    <span>this</span>'s <span>element definition is running</span> to false.</p>

    <p>Finally, if the steps threw an exception, rethrow that exception.</p>
   </li>

   <li><p>Let <var>definition</var> be a new <span>custom element definition</span> with <span
   data-x="concept-custom-element-definition-name">name</span> <var>name</var>, <span
   data-x="concept-custom-element-definition-local-name">local name</span> <var>localName</var>,
   <span data-x="concept-custom-element-definition-constructor">constructor</span>
   <var>constructor</var>, <span
   data-x="concept-custom-element-definition-observed-attributes">observed attributes</span>
   <var>observedAttributes</var>, <span
   data-x="concept-custom-element-definition-lifecycle-callbacks">lifecycle callbacks</span>
   <var>lifecycleCallbacks</var>,
   <span data-x="concept-custom-element-definition-form-associated">form-associated</span>
   <var>formAssociated</var>, <span
   data-x="concept-custom-element-definition-disable-internals">disable internals</span>
   <var>disableInternals</var>, and <span
   data-x="concept-custom-element-definition-disable-shadow">disable shadow</span>
   <var>disableShadow</var>.</p></li>

   <li><p><span data-x="set append">Append</span> <var>definition</var> to <span>this</span>'s
   <span>custom element definition set</span>.</p></li>

   <li><p>Let <var>document</var> be <span>this</span>'s <span>relevant global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>Let <var>upgradeCandidates</var> be all elements that are <span data-x="shadow-including
   descendant">shadow-including descendants</span> of <var>document</var>, whose namespace
   is the <span>HTML namespace</span> and whose local name is <var>localName</var>, in
   <span>shadow-including tree order</span>. Additionally, if <var>extends</var> is non-null, only
   include elements whose <span data-x="concept-element-is-value"><code data-x="">is</code>
   value</span> is equal to <var>name</var>.</p></li>

   <li><p>For each element <var>element</var> of <var>upgradeCandidates</var>, <span>enqueue a
   custom element upgrade reaction</span> given <var>element</var> and
   <var>definition</var>.</p></li>

   <!-- It is equivalent to just try to upgrade all elements in the document, and let "try to
   upgrade" bail out, but this seems a bit more explicit. -->

   <li>
    <p>If <span>this</span>'s <span>when-defined promise map</span>[<var>name</var>] <span data-x="map
    exists">exists</span>:</p>

    <ol>
     <li><p>Resolve <span>this</span>'s <span>when-defined promise map</span>[<var>name</var>] with
     <var>constructor</var>.</p></li>

     <li><p><span data-x="map remove">Remove</span> <span>this</span>'s <span>when-defined promise
     map</span>[<var>name</var>].</p></li>
    </ol>
   </li>
  </ol>

  <p>The <dfn method for="CustomElementRegistry"><code
  data-x="dom-CustomElementRegistry-get">get(<var>name</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>custom element definition set</span> <span data-x="list
   contains">contains</span> an item with <span
   data-x="concept-custom-element-definition-name">name</span> <var>name</var>, then return that
   item's <span data-x="concept-custom-element-definition-constructor">constructor</span>.</p></li>

   <li><p>Return undefined.</p></li>
  </ol>

  <p>The <dfn method for="CustomElementRegistry"><code
  data-x="dom-CustomElementRegistry-getName">getName(<var>constructor</var>)</code></dfn> method
  steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>custom element definition set</span> <span data-x="list
   contains">contains</span> an item with <span
   data-x="concept-custom-element-definition-constructor">constructor</span>
   <var>constructor</var>, then return that item's <span
   data-x="concept-custom-element-definition-name">name</span>.</p></li>

   <li><p>Return null.</p></li>
  </ol>

  <p>The <dfn method for="CustomElementRegistry"><code
  data-x="dom-CustomElementRegistry-whenDefined">whenDefined(<var>name</var>)</code></dfn> method
  steps are:</p>

  <ol>
   <li><p>If <var>name</var> is not a <span>valid custom element name</span>, then return <span>a
   promise rejected with</span> a <span>"<code>SyntaxError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>If <span>this</span>'s <span>custom element definition set</span> <span data-x="list
   contains">contains</span> an item with <span
   data-x="concept-custom-element-definition-name">name</span> <var>name</var>, then return <span>a
   promise resolved with</span> that item's <span
   data-x="concept-custom-element-definition-constructor">constructor</span>.</p></li>

   <li><p>If <span>this</span>'s <span>when-defined promise map</span>[<var>name</var>] does not
   <span data-x="map exists">exist</span>, then <span data-x="map set">set</span>
   <span>this</span>'s <span>when-defined promise map</span>[<var>name</var>] to a new
   promise.</p></li>

   <li><p>Return <span>this</span>'s <span>when-defined promise
   map</span>[<var>name</var>].</p></li>
  </ol>

  </div>

  <div class="example">
   <p>The <code data-x="dom-CustomElementRegistry-whenDefined">whenDefined()</code> method can be
   used to avoid performing an action until all appropriate <span data-x="custom element">custom
   elements</span> are <span data-x="concept-element-defined">defined</span>. In this example, we
   combine it with the <code data-x="selector-defined">:defined</code> pseudo-class to hide a
   dynamically-loaded article's contents until we're sure that all of the <span data-x="autonomous
   custom element">autonomous custom elements</span> it uses are defined.</p>

   <pre><code class="js">articleContainer.hidden = true;

fetch(articleURL)
  .then(response => response.text())
  .then(text => {
    articleContainer.innerHTML = text;

    return Promise.all(
      [...articleContainer.querySelectorAll(":not(:defined)")]
        .map(el => customElements.whenDefined(el.localName))
    );
  })
  .then(() => {
    articleContainer.hidden = false;
  });</code></pre>
  </div>

  <div w-nodev>

  <p>When invoked, the <dfn method for="CustomElementRegistry"><code
  data-x="dom-CustomElementRegistry-upgrade">upgrade(<var>root</var>)</code></dfn> method must run
  these steps:</p>

  <ol>
   <li><p>Let <var>candidates</var> be a <span>list</span> of all of <var>root</var>'s
   <span>shadow-including inclusive descendant</span> elements, in <span>shadow-including tree
   order</span>.</p></li>

   <li><p><span data-x="list iterate">For each</span> <var>candidate</var> of <var>candidates</var>,
   <span data-x="concept-try-upgrade">try to upgrade</span> <var>candidate</var>.</p></li>
  </ol>

  </div>

  <div class="example">
   <p>The <code data-x="dom-CustomElementRegistry-upgrade">upgrade()</code> method allows upgrading
   of elements at will. Normally elements are automatically upgraded when they become
   <span>connected</span>, but this method can be used if you need to upgrade before you're ready to
   connect the element.</p>

   <pre><code class="js">const el = document.createElement("spider-man");

class SpiderMan extends HTMLElement {}
customElements.define("spider-man", SpiderMan);

console.assert(!(el instanceof SpiderMan)); // not yet upgraded

customElements.upgrade(el);
console.assert(el instanceof SpiderMan);    // upgraded!</code></pre>
  </div>

  <div w-nodev>

  <h4><dfn data-x="custom-element-upgrades">Upgrades</dfn></h4>

  <p>To <dfn data-x="concept-upgrade-an-element" export>upgrade an element</dfn>, given as input a
  <span>custom element definition</span> <var>definition</var> and an element <var>element</var>,
  run the following steps:</p>

  <ol>
   <li id="concept-upgrade-an-element-early-exit">
    <p>If <var>element</var>'s <span>custom element state</span> is not "<code
    data-x="">undefined</code>" or "<code data-x="">uncustomized</code>", then return.</p>

    <div class="example">
     <p>One scenario where this can occur due to reentrant invocation of this algorithm, as in the
     following example:</p>

     <pre><code class="html">&lt;!DOCTYPE html>
&lt;x-foo id="a">&lt;/x-foo>
&lt;x-foo id="b">&lt;/x-foo>

&lt;script>
// Defining enqueues upgrade reactions for both "a" and "b"
customElements.define("x-foo", class extends HTMLElement {
  constructor() {
    super();

    const b = document.querySelector("#b");
    b.remove();

    // While this constructor is running for "a", "b" is still
    // undefined, and so inserting it into the document will enqueue a
    // second upgrade reaction for "b" in addition to the one enqueued
    // by defining x-foo.
    document.body.appendChild(b);
  }
})
&lt;/script></code></pre>

     <p>This step will thus bail out the algorithm early when <span
     data-x="concept-upgrade-an-element">upgrade an element</span> is invoked
     with "<code data-x="">b</code>" a second time.</p>
    </div>
   </li>

   <li><p>Set <var>element</var>'s <span data-x="concept-element-custom-element-definition">custom
   element definition</span> to <var>definition</var>.</p></li>

   <li>
    <p>Set <var>element</var>'s <span>custom element state</span> to "<code
    data-x="">failed</code>".</p>

    <p class="note">It will be set to "<code data-x="">custom</code>" <a
    href="#concept-upgrade-an-element-set-state-to-custom">after the upgrade succeeds</a>. For now,
    we set it to "<code data-x="">failed</code>" so that any reentrant invocations will hit <a
    href="#concept-upgrade-an-element-early-exit">the above early-exit step</a>.</p>
   </li>

   <li><p>For each <var>attribute</var> in <var>element</var>'s <span>attribute list</span>, in
   order, <span>enqueue a custom element callback reaction</span> with <var>element</var>, callback
   name "<code data-x="">attributeChangedCallback</code>", and « <var>attribute</var>'s local name,
   null, <var>attribute</var>'s value, <var>attribute</var>'s namespace ».</p></li>

   <li><p>If <var>element</var> is <span>connected</span>, then <span>enqueue a custom element
   callback reaction</span> with <var>element</var>, callback name "<code
   data-x="">connectedCallback</code>", and « ».</p></li>

   <li><p>Add <var>element</var> to the end of <var>definition</var>'s <span
   data-x="concept-custom-element-definition-construction-stack">construction stack</span>.</p></li>

   <li><p>Let <var>C</var> be <var>definition</var>'s <span
   data-x="concept-custom-element-definition-constructor">constructor</span>.</p></li>

   <li>
    <p>Run the following substeps while catching any exceptions:</p>

    <ol>
     <li>
      <p>If <var>definition</var>'s <span
      data-x="concept-custom-element-definition-disable-shadow">disable shadow</span> is true and
      <var>element</var>'s <span data-x="concept-element-shadow-root">shadow root</span> is
      non-null, then throw a <span>"<code>NotSupportedError</code>"</span>
      <code>DOMException</code>.</p>

      <p class="note">This is needed as <code
      data-x="dom-Element-attachShadow">attachShadow()</code> does not use <span>look up a custom
      element definition</span> while <code data-x="dom-attachInternals">attachInternals()</code>
      does.</p>
     </li>

     <li><p>Set <var>element</var>'s <span>custom element state</span> to "<code
     data-x="">precustomized</code>".</p></li>

     <li>
      <p>Let <var>constructResult</var> be the result of <span
      data-x="es-constructing-callback-functions">constructing</span> <var>C</var>, with no
      arguments.</p>

      <p class="note">If <var>C</var> <a href="#custom-element-conformance">non-conformantly</a>
      uses an API decorated with the <code data-x="CEReactions">[CEReactions]</code> extended
      attribute, then the reactions enqueued at the beginning of this algorithm will execute during
      this step, before <var>C</var> finishes and control returns to this algorithm. Otherwise, they
      will execute after <var>C</var> and the rest of the upgrade process finishes.</p>
     </li>

     <li>
      <p>If <span>SameValue</span>(<var>constructResult</var>, <var>element</var>) is false, then
      throw a <code>TypeError</code>.</p>

      <p class="note">This can occur if <var>C</var> constructs another instance of the same custom
      element before calling <code data-x="">super()</code>, or if <var>C</var> uses JavaScript's
      <code data-x="">return</code>-override feature to return an arbitrary <code>HTMLElement</code>
      object from the constructor.</p>
     </li>
    </ol>

    <p>Then, perform the following substep, regardless of whether the above steps threw an exception
    or not:</p>

    <ol>
     <li>
      <p>Remove the last entry from the end of <var>definition</var>'s <span
      data-x="concept-custom-element-definition-construction-stack">construction stack</span>.</p>

      <div class="note">
       <p>Assuming <var>C</var> calls <code data-x="">super()</code> (as it will if it is <a
       href="#custom-element-conformance">conformant</a>), and that the call succeeds, this will be
       the <span data-x="concept-already-constructed-marker"><i>already
       constructed</i> marker</span> that replaced the <var>element</var> we pushed at the beginning
       of this algorithm. (The <a href="#html-element-constructors">HTML element constructor</a>
       carries out this replacement.)</p>

       <p>If <var>C</var> does not call <code data-x="">super()</code> (i.e. it is not <a
       href="#custom-element-conformance">conformant</a>), or if any step in the <a
       href="#html-element-constructors">HTML element constructor</a> throws, then this entry will
       still be <var>element</var>.</p>
      </div>
     </li>
    </ol>

    <p>Finally, if the above steps threw an exception, then:</p>

    <ol>
     <li><p>Set <var>element</var>'s <span data-x="concept-element-custom-element-definition">custom
     element definition</span> to null.</p></li>

     <li><p>Empty <var>element</var>'s <span>custom element reaction queue</span>.</p>

     <li><p>Rethrow the exception (thus terminating this algorithm).</p></li>
    </ol>

    <p class="note">If the above steps threw an exception, then <var>element</var>'s <span>custom
    element state</span> will remain "<code data-x="">failed</code>" or "<code
    data-x="">precustomized</code>".</p>
   </li>

   <li>
    <p>If <var>element</var> is a <span>form-associated custom element</span>, then:</p>
    <ol>
     <li><p><span>Reset the form owner</span> of <var>element</var>. If <var>element</var> is
     associated with a <code>form</code> element, then
     <span>enqueue a custom element callback reaction</span> with <var>element</var>, callback
     name "<code data-x="">formAssociatedCallback</code>", and « the associated <code>form</code>
     ».</p></li>

     <li><p>If <var>element</var> is <span data-x="concept-fe-disabled">disabled</span>, then
     <span>enqueue a custom element callback reaction</span> with <var>element</var>, callback name
     "<code data-x="">formDisabledCallback</code>" and « true ».</p></li>
    </ol>
   </li>

   <li id="concept-upgrade-an-element-set-state-to-custom"><p>Set <var>element</var>'s <span>custom
   element state</span> to "<code data-x="">custom</code>".</p></li>
  </ol>

  <p>To <dfn data-x="concept-try-upgrade" export>try to upgrade an element</dfn>, given as input an
  element <var>element</var>, run the following steps:</p>

  <ol>
   <li><p>Let <var>definition</var> be the result of <span data-x="look up a custom element
   definition">looking up a custom element definition</span> given <var>element</var>'s <span>node
   document</span>, <var>element</var>'s namespace, <var>element</var>'s local name, and
   <var>element</var>'s <span data-x="concept-element-is-value"><code data-x="">is</code>
   value</span>.</p></li>

   <li><p>If <var>definition</var> is not null, then <span>enqueue a custom element upgrade
   reaction</span> given <var>element</var> and <var>definition</var>.</p></li>
  </ol>

  </div>

  <h4 id="custom-element-reactions">Custom element reactions</h4>

  <p>A <span>custom element</span> possesses the ability to respond to certain occurrences by
  running author code:</p>

  <ul>
   <li><p>When <span data-x="custom-element-upgrades">upgraded</span>, its <span data-x="custom
   element constructor">constructor</span> is run, with no arguments.</p></li>

   <li><p>When it <span>becomes connected</span>, its <code data-x="">connectedCallback</code> is
   called, with no arguments.</p></li>

   <li><p>When it <span>becomes disconnected</span>, its <code data-x="">disconnectedCallback</code>
   is called, with no arguments.</p></li>

   <li><p>When it is <span data-x="concept-node-move-ext">moved</span>, its <code
   data-x="">connectedMoveCallback</code> is called, with no arguments.</p></li>

   <li><p>When it is <span data-x="concept-node-adopt">adopted</span> into a new document, its <code
   data-x="">adoptedCallback</code> is called, given the old document and new document as
   arguments.</p></li>

   <li><p>When any of its attributes are <span
   data-x="concept-element-attributes-change">changed</span>, <span
   data-x="concept-element-attributes-append">appended</span>, <span
   data-x="concept-element-attributes-remove">removed</span>, or <span
   data-x="concept-element-attributes-replace">replaced</span>, its <code
   data-x="">attributeChangedCallback</code> is called, given the attribute's local name, old value,
   new value, and namespace as arguments. (An attribute's old or new value is considered to be null
   when the attribute is added or removed, respectively.)</p></li>

   <li><p>When the user agent <span data-x="reset the form owner">resets the form owner</span> of a
   <span>form-associated custom element</span> and doing so changes the form owner, its <code
   data-x="">formAssociatedCallback</code> is called, given the new form owner (or null if no owner)
   as an argument.</p></li>

   <li><p>When the form owner of a <span>form-associated custom element</span> is <span
   data-x="concept-form-reset">reset</span>, its <code data-x="">formResetCallback</code> is
   called.</p></li>

   <li><p>When the <span data-x="concept-fe-disabled">disabled</span> state of a
   <span>form-associated custom element</span> is changed, its <code
   data-x="">formDisabledCallback</code> is called, given the new state as an
   argument.</p></li>

   <li><p>When user agent updates a <span>form-associated custom element</span>'s value on behalf of
   a user or <span data-x="restore persisted state">as part of navigation</span>, its <code
   data-x="">formStateRestoreCallback</code> is called, given the new state and a string indicating
   a reason, "<code data-x="">autocomplete</code>" or "<code data-x="">restore</code>", as
   arguments.</p></li>
  </ul>

  <p>We call these reactions collectively <dfn data-x="concept-custom-element-reaction">custom
  element reactions</dfn>.</p>

  <p>The way in which <span data-x="concept-custom-element-reaction">custom element reactions</span>
  are invoked is done with special care, to avoid running author code during the middle of delicate
  operations. Effectively, they are delayed until "just before returning to user script". This means
  that for most purposes they appear to execute synchronously, but in the case of complicated
  composite operations (like <span data-x="concept-node-clone">cloning</span>, or <span
  data-x="concept-range">range</span> manipulation), they will instead be delayed until after all
  the relevant user agent processing steps have completed, and then run together as a batch.</p>

  <p><span w-nodev>Additionally, the precise ordering of these reactions is managed via a
  somewhat-complicated stack-of-queues system, described below. The intention behind this system is
  to guarantee</span><span w-dev>It is guaranteed</span> that <span
  data-x="concept-custom-element-reaction">custom element reactions</span> always are invoked in the
  same order as their triggering actions, at least within the local context of a single <span>custom
  element</span>. (Because <span data-x="concept-custom-element-reaction">custom element
  reaction</span> code can perform its own mutations, it is not possible to give a global ordering
  guarantee across multiple elements.)</p>

  <div w-nodev>

  <hr>

  <p>Each <span>similar-origin window agent</span> has a <dfn>custom element reactions stack</dfn>,
  which is initially empty. A <span>similar-origin window agent</span>'s <dfn>current element
  queue</dfn> is the <span>element queue</span> at the top of its <span>custom element reactions
  stack</span>. Each item in the stack is an <dfn>element queue</dfn>, which is initially empty as
  well. Each item in an <span>element queue</span> is an element. (The elements are not necessarily
  <span data-x="concept-element-custom">custom</span> yet, since this queue is used for <span
  data-x="custom-element-upgrades">upgrades</span> as well.)</p>

  <p>Each <span>custom element reactions stack</span> has an associated <dfn>backup element
  queue</dfn>, which is an initially-empty <span>element queue</span>. Elements are pushed onto the
  <span>backup element queue</span> during operations that affect the DOM without going through an
  API decorated with <code data-x="CEReactions">[CEReactions]</code>, or through the parser's
  <span>create an element for the token</span> algorithm. An example of this is a user-initiated
  editing operation which modifies the descendants or attributes of an <span>editable</span>
  element. To prevent reentrancy when processing the <span>backup element queue</span>, each
  <span>custom element reactions stack</span> also has a <dfn>processing the backup element
  queue</dfn> flag, initially unset.</p>

  <p>All elements have an associated <dfn>custom element reaction queue</dfn>, initially empty. Each
  item in the <span>custom element reaction queue</span> is of one of two types:</p>

  <ul>
   <li><p>An <dfn>upgrade reaction</dfn>, which will <span
   data-x="custom-element-upgrades">upgrade</span> the custom element and contains a <span>custom
   element definition</span>; or</p></li>

   <li><p>A <dfn>callback reaction</dfn>, which will call a lifecycle callback, and contains a
   callback function as well as a list of arguments.</p></li>
  </ul>

  <p>This is all summarized in the following schematic diagram:</p>

  <p><img src="/images/custom-element-reactions.svg" style="width: 80%; max-width: 580px;" alt="A custom element reactions stack consists of a stack of element queues. Zooming in on a particular queue, we see that it contains a number of elements (in our example, &lt;x-a&gt;, then &lt;x-b&gt;, then &lt;x-c&gt;). Any particular element in the queue then has a custom element reaction queue. Zooming in on the custom element reaction queue, we see that it contains a variety of queued-up reactions (in our example, upgrade, then attribute changed, then another attribute changed, then connected)." class="darkmode-aware"></p>

  <p>To <dfn>enqueue an element on the appropriate element queue</dfn>, given an element
  <var>element</var>, run the following steps:</p>

  <ol>
   <li><p>Let <var>reactionsStack</var> be <var>element</var>'s <span>relevant agent</span>'s
   <span>custom element reactions stack</span>.</p></li>

   <li>
    <p>If <var>reactionsStack</var> is empty, then:</p>

    <ol>
     <li><p>Add <var>element</var> to <var>reactionsStack</var>'s <span>backup element
     queue</span>.</p></li>

     <li><p>If <var>reactionsStack</var>'s <span>processing the backup element queue</span> flag is
     set, then return.</p></li>

     <li><p>Set <var>reactionsStack</var>'s <span>processing the backup element queue</span>
     flag.</p></li>

     <li>
      <p><span>Queue a microtask</span> to perform the following steps:</p>

      <ol>
       <li><p><span>Invoke custom element reactions</span> in <var>reactionsStack</var>'s
       <span>backup element queue</span>.</p></li>

       <li><p>Unset <var>reactionsStack</var>'s <span>processing the backup element queue</span>
       flag.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Otherwise, add <var>element</var> to <var>element</var>'s <span>relevant agent</span>'s
   <span>current element queue</span>.</p></li>
  </ol>

  <p>To <dfn export>enqueue a custom element callback reaction</dfn>, given a <span>custom
  element</span> <var>element</var>, a callback name <var>callbackName</var>, and a list of
  arguments <var>args</var>, run the following steps:</p>

  <ol>
   <li><p>Let <var>definition</var> be <var>element</var>'s <span
   data-x="concept-element-custom-element-definition">custom element definition</span>.</p>

   <li><p>Let <var>callback</var> be the value of the entry in <var>definition</var>'s <span
   data-x="concept-custom-element-definition-lifecycle-callbacks">lifecycle callbacks</span> with
   key <var>callbackName</var>.</p></li>

   <li>
    <p>If <var>callbackName</var> is "<code data-x="">connectedMoveCallback</code>" and
    <var>callback</var> is null:

    <ol>
     <li><p>Let <var>disconnectedCallback</var> be the value of the entry in <var>definition</var>'s
     <span data-x="concept-custom-element-definition-lifecycle-callbacks">lifecycle callbacks</span>
     with key "<code data-x="">disconnectedCallback</code>".</p></li>

     <li><p>Let <var>connectedCallback</var> be the value of the entry in <var>definition</var>'s
     <span data-x="concept-custom-element-definition-lifecycle-callbacks">lifecycle callbacks</span>
     with key "<code data-x="">connectedCallback</code>".</p></li>

     <li><p>If <var>connectedCallback</var> and <var>disconnectedCallback</var> are null, then
     return.</p></li>

     <li>
      <p>Set <var>callback</var> to the following steps:</p>

      <ol>
       <li><p>If <var>disconnectedCallback</var> is not null, then call
       <var>disconnectedCallback</var> with no arguments.</p></li>

       <li><p>If <var>connectedCallback</var> is not null, then call <var>connectedCallback</var>
       with no arguments.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>If <var>callback</var> is null, then return.</p></li>

   <li>
    <p>If <var>callbackName</var> is "<code data-x="">attributeChangedCallback</code>":</p>

    <ol>
     <li><p>Let <var>attributeName</var> be the first element of <var>args</var>.</p></li>

     <li><p>If <var>definition</var>'s <span
     data-x="concept-custom-element-definition-observed-attributes">observed attributes</span> does
     not contain <var>attributeName</var>, then return.</p></li>
    </ol>
   </li>

   <li><p>Add a new <span>callback reaction</span> to <var>element</var>'s <span>custom element
   reaction queue</span>, with callback function <var>callback</var> and arguments
   <var>args</var>.</p></li>

   <li><p><span>Enqueue an element on the appropriate element queue</span> given
   <var>element</var>.</p></li>
  </ol>

  <p>To <dfn export>enqueue a custom element upgrade reaction</dfn>, given an element
  <var>element</var> and <span>custom element definition</span> <var>definition</var>, run the
  following steps:</p>

  <ol>
   <li><p>Add a new <span>upgrade reaction</span> to <var>element</var>'s <span>custom element
   reaction queue</span>, with <span>custom element definition</span>
   <var>definition</var>.</p></li>

   <li><p><span>Enqueue an element on the appropriate element queue</span> given
   <var>element</var>.</p></li>
  </ol>

  <p>To <dfn>invoke custom element reactions</dfn> in an <span>element queue</span>
  <var>queue</var>, run the following steps:</p>

  <ol>
   <li>
    <p>While <var>queue</var> is not <span data-x="list is empty">empty</span>:</p>

    <ol>
     <li><p>Let <var>element</var> be the result of <span data-x="dequeue">dequeuing</span>
     from <var>queue</var>.</p></li>

     <li><p>Let <var>reactions</var> be <var>element</var>'s <span>custom element reaction
     queue</span>.</p></li>

     <li>
      <p>Repeat until <var>reactions</var> is empty:</p>

      <ol>
       <li>
        <p>Remove the first element of <var>reactions</var>, and let <var>reaction</var> be that
        element. Switch on <var>reaction</var>'s type:</p>

        <dl class="switch">
         <dt><span>upgrade reaction</span></dt>
         <dd>
          <p><span data-x="concept-upgrade-an-element">Upgrade</span> <var>element</var> using
          <var>reaction</var>'s <span>custom element definition</span>.</p>

          <p>If this throws an exception, catch it, and <span data-x="report an
          exception">report</span> it for <var>reaction</var>'s <span>custom element
          definition</span>'s <span
          data-x="concept-custom-element-definition-constructor">constructor</span>'s corresponding
          JavaScript object's <span>associated realm</span>'s <span
          data-x="concept-realm-global">global object</span>.</p>
         </dd>

         <dt><span>callback reaction</span></dt>
         <dd>
          <p><span data-x="es-invoking-callback-functions">Invoke</span> <var>reaction</var>'s
          callback function with <var>reaction</var>'s arguments and "<code
          data-x="">report</code>", and <i data-x="dfn-callback-this-value">callback this value</i>
          set to <var>element</var>.</p>
         </dd>
        </dl>
       </li>
      </ol>
     </li>
    </ol>
   </li>
  </ol>

  <hr>

  <p>To ensure <span data-x="concept-custom-element-reaction">custom element reactions</span> are
  triggered appropriately, we introduce the <dfn extended-attribute data-lt="CEReactions"
  data-x="CEReactions"><code>[CEReactions]</code></dfn> IDL <span>extended attribute</span>. It
  indicates that the relevant algorithm is to be supplemented with additional steps in order to
  appropriately track and invoke <span data-x="concept-custom-element-reaction">custom element
  reactions</span>.</p>

  <p>The <code data-x="CEReactions">[CEReactions]</code> extended attribute must take no
  arguments, and must not appear on anything other than an operation, attribute, setter, or deleter.
  Additionally, it must not appear on readonly attributes.</p>

  <p>Operations, attributes, setters, or deleters annotated with the <code
  data-x="CEReactions">[CEReactions]</code> extended attribute must run the following steps in place
  of the ones specified in their description:</p>

  <ol>
   <li><p><span data-x="stack push">Push</span> a new <span>element queue</span> onto this object's
   <span>relevant agent</span>'s <span>custom element reactions stack</span>.</p></li>

   <li><p>Run the originally-specified steps for this construct, catching any exceptions. If the
   steps return a value, let <var>value</var> be the returned value. If they throw an exception, let
   <var>exception</var> be the thrown exception.</p></li>

   <li><p>Let <var>queue</var> be the result of <span data-x="stack pop">popping</span> from this
   object's <span>relevant agent</span>'s <span>custom element reactions stack</span>.</p></li>

   <li><p><span>Invoke custom element reactions</span> in <var>queue</var>.</p></li>

   <li><p>If an exception <var>exception</var> was thrown by the original steps, rethrow
   <var>exception</var>.</p></li>

   <li><p>If a value <var>value</var> was returned from the original steps, return
   <var>value</var>.</p></li>
  </ol>

  <div class="note">
   <p>The intent behind this extended attribute is somewhat subtle. One way of accomplishing its
   goals would be to say that every operation, attribute, setter, and deleter on the platform
   <!--non-normative-->must have these steps inserted, and to allow implementers to optimize away
   unnecessary cases (where no DOM mutation is possible that could cause <span
   data-x="concept-custom-element-reaction">custom element reactions</span> to occur).</p>

   <p>However, in practice this imprecision could lead to non-interoperable implementations of <span
   data-x="concept-custom-element-reaction">custom element reactions</span>, as some implementations
   might forget to invoke these steps in some cases. Instead, we settled on the approach of
   explicitly annotating all relevant IDL constructs, as a way of ensuring interoperable behavior
   and helping implementations easily pinpoint all cases where these steps are necessary.</p>
  </div>

  <p>Any nonstandard APIs introduced by the user agent that could modify the DOM in such a way as to
  cause <span data-x="enqueue a custom element callback reaction">enqueuing a custom element
  callback reaction</span> or <span data-x="enqueue a custom element upgrade reaction">enqueuing a
  custom element upgrade reaction</span>, for example by modifying any attributes or child elements,
  must also be decorated with the <code data-x="CEReactions">[CEReactions]</code> attribute.</p>

  <div class="note">
   <p>As of the time of this writing, the following nonstandard or not-yet-standardized APIs are
   known to fall into this category:</p>

   <ul>
    <li><p><code>HTMLInputElement</code>'s <code data-x="">webkitdirectory</code> and <code
    data-x="">incremental</code> IDL attributes</p></li>

    <li><p><code>HTMLLinkElement</code>'s <code data-x="">scope</code> IDL attribute</p></li>
   </ul>
  </div>

  </div>

  <h4>Element internals</h4>

  <p>Certain capabilities are meant to be available to a custom element author, but not to a custom
  element consumer. These are provided by the <code
  data-x="dom-attachInternals">element.attachInternals()</code> method, which returns an instance of
  <code>ElementInternals</code>. The properties and methods of <code>ElementInternals</code> allow
  control over internal features which the user agent provides to all elements.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-attachInternals">attachInternals()</span></code></dt>

   <dd><p>Returns an <code>ElementInternals</code> object targeting the <span>custom element</span>
   <var>element</var>. Throws an exception if <var>element</var> is not a <span>custom
   element</span>, if the "<code data-x="">internals</code>" feature was disabled as part of the
   element definition, or if it is called twice on the same element.</p></dd>
  </dl>

  <div w-nodev>

  <p>Each <code>HTMLElement</code> has an <dfn>attached internals</dfn> (null or an
  <code>ElementInternals</code> object), initially null.</p>

  <p>The <dfn method for="HTMLElement"><code
  data-x="dom-attachInternals">attachInternals()</code></dfn> method steps are:</p>

  <ol>
   <!-- We could lift this restriction in the future. -->
   <li><p>If <span>this</span>'s <span data-x="concept-element-is-value"><code data-x="">is</code>
   value</span> is not null, then throw a <span>"<code>NotSupportedError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Let <var>definition</var> be the result of <span data-x="look up a custom element
   definition">looking up a custom element definition</span> given <span>this</span>'s <span>node
   document</span>, its namespace, its local name, and null as the <span
   data-x="concept-element-is-value"><code data-x="">is</code> value</span>.</p></li>

   <li><p>If <var>definition</var> is null, then throw an
   <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <var>definition</var>'s <span
   data-x="concept-custom-element-definition-disable-internals">disable internals</span> is true,
   then throw a <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <span>this</span>'s <span>attached internals</span> is non-null, then throw an
   <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <span>this</span>'s <span>custom element state</span> is not "<code
   data-x="">precustomized</code>" or "<code data-x="">custom</code>", then throw a
   <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Set <span>this</span>'s <span>attached internals</span> to a new
   <code>ElementInternals</code> instance whose <span data-x="internals-target">target
   element</span> is <span>this</span>.</p></li>

   <li><p>Return <span>this</span>'s <span>attached internals</span>.</p></li>
  </ol>

  <h5>The <code>ElementInternals</code> interface</h5>

  <p>The IDL for the <code>ElementInternals</code> interface is as follows, with the various operations and attributes
  defined in the following sections:</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>ElementInternals</dfn> {
  // <a href="#shadow-root-access">Shadow root access</a>
  readonly attribute <code>ShadowRoot</code>? <span data-x="dom-ElementInternals-shadowRoot">shadowRoot</span>;

  // <a href="#form-associated-custom-elements">Form-associated custom elements</a>
  undefined <span data-x="dom-ElementInternals-setFormValue">setFormValue</span>((<span>File</span> or <span data-x="idl-USVString">USVString</span> or <span>FormData</span>)? value,
                         optional (<span>File</span> or <span data-x="idl-USVString">USVString</span> or <span>FormData</span>)? state);

  readonly attribute <span>HTMLFormElement</span>? <span data-x="dom-ElementInternals-form">form</span>;

  undefined <span data-x="dom-ElementInternals-setValidity">setValidity</span>(optional <span>ValidityStateFlags</span> flags = {},
                        optional DOMString message,
                        optional <span>HTMLElement</span> anchor);
  readonly attribute boolean <span data-x="dom-ElementInternals-willValidate">willValidate</span>;
  readonly attribute <span>ValidityState</span> <span data-x="dom-ElementInternals-validity">validity</span>;
  readonly attribute DOMString <span data-x="dom-ElementInternals-validationMessage">validationMessage</span>;
  boolean <span data-x="dom-ElementInternals-checkValidity">checkValidity</span>();
  boolean <span data-x="dom-ElementInternals-reportValidity">reportValidity</span>();

  readonly attribute <span>NodeList</span> <span data-x="dom-ElementInternals-labels">labels</span>;

  // <a href="#custom-state-pseudo-class">Custom state pseudo-class</a>
  [SameObject] readonly attribute <span>CustomStateSet</span> <span data-x="dom-ElementInternals-states">states</span>;
};

// <a href="#accessibility-semantics">Accessibility semantics</a>
<span>ElementInternals</span> includes <span>ARIAMixin</span>;

dictionary <dfn dictionary>ValidityStateFlags</dfn> {
  boolean valueMissing = false;
  boolean typeMismatch = false;
  boolean patternMismatch = false;
  boolean tooLong = false;
  boolean tooShort = false;
  boolean rangeUnderflow = false;
  boolean rangeOverflow = false;
  boolean stepMismatch = false;
  boolean badInput = false;
  boolean customError = false;
};</code></pre>

  <p>Each <code>ElementInternals</code> has a <dfn data-x="internals-target">target element</dfn>,
  which is a <span>custom element</span>.</p>

  </div>

  <h5>Shadow root access</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ElementInternals-shadowRoot">shadowRoot</span></code></dt>
   <dd><p>Returns the <code>ShadowRoot</code> for <var>internals</var>'s <span
   data-x="internals-target">target element</span>, if the <span data-x="internals-target">target
   element</span> is a <span>shadow host</span>, or null otherwise.</p></dd>
  </dl>

  <div w-nodev>
  <p>The <dfn attribute for="ElementInternals"><code
  data-x="dom-ElementInternals-shadowRoot">shadowRoot</code></dfn> getter steps are:</p>

  <ol>
   <li><p>Let <var>target</var> be <span>this</span>'s <span data-x="internals-target">target
   element</span>.</p></li>

   <li><p>If <var>target</var> is not a <span>shadow host</span>, then return null.</p></li>

   <li><p>Let <var>shadow</var> be <var>target</var>'s <span
   data-x="concept-element-shadow-root">shadow root</span>.</p></li>

   <li><p>If <var>shadow</var>'s <span>available to element internals</span> is false, then return
   null.</p></li>

   <li><p>Return <var>shadow</var>.</p></li>
  </ol>
  </div>

  <h5>Form-associated custom elements</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ElementInternals-setFormValue">setFormValue</span>(<var>value</var>)</code></dt>
   <dd>
    <p>Sets both the <span data-x="face-state">state</span> and <span
    data-x="face-submission-value">submission value</span> of <var>internals</var>'s <span
    data-x="internals-target">target element</span> to <var>value</var>.</p>

    <p>If <var>value</var> is null, the element won't participate in form submission.</p>
   </dd>

   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ElementInternals-setFormValue">setFormValue</span>(<var>value</var>,
   <var>state</var>)</code></dt>
   <dd>
    <p>Sets the <span data-x="face-submission-value">submission value</span> of
    <var>internals</var>'s <span data-x="internals-target">target element</span> to
    <var>value</var>, and its <span data-x="face-state">state</span> to <var>state</var>.</p>

    <p>If <var>value</var> is null, the element won't participate in form submission.</p>
   </dd>

   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ElementInternals-form">form</span></code></dt>
   <dd><p>Returns the <span>form owner</span> of <var>internals</var>'s <span
   data-x="internals-target">target element</span>.</p></dd>

   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ElementInternals-setValidity">setValidity</span>(<var>flags</var>,
   <var>message</var> [, <var>anchor</var> ])</code></dt>
   <dd><p>Marks <var>internals</var>'s <span data-x="internals-target">target element</span> as
   suffering from the constraints indicated by the <var>flags</var> argument, and sets the element's
   validation message to <var>message</var>. If <var>anchor</var> is specified, the user agent might
   use it to indicate problems with the constraints of <var>internals</var>'s <span
   data-x="internals-target">target element</span> when the <span>form owner</span> is validated
   interactively or <code data-x="dom-ElementInternals-reportValidity">reportValidity()</code> is
   called.</p></dd>

   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ElementInternals-setValidity">setValidity</span>({})</code></dt>
   <dd><p>Marks <var>internals</var>'s <span data-x="internals-target">target element</span> as
   <span data-x="concept-fv-valid">satisfying its constraints</span>.</p></dd>

   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ElementInternals-willValidate">willValidate</span></code></dt>
   <dd><p>Returns true if <var>internals</var>'s <span data-x="internals-target">target
   element</span> will be validated when the form is submitted; false otherwise.</p></dd>

   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ElementInternals-validity">validity</span></code></dt>
   <dd><p>Returns the <code>ValidityState</code> object for <var>internals</var>'s <span
   data-x="internals-target">target element</span>.</p></dd>

   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ElementInternals-validationMessage">validationMessage</span></code></dt>
   <dd><p>Returns the error message that would be shown to the user if <var>internals</var>'s <span
   data-x="internals-target">target element</span> was to be checked for validity.</p></dd>

   <dt><code data-x=""><var>valid</var> = <var>internals</var>.<span data-x="dom-ElementInternals-checkValidity">checkValidity()</span></code></dt>
   <dd><p>Returns true if <var>internals</var>'s <span data-x="internals-target">target
   element</span> has no validity problems; false otherwise. Fires an <code
   data-x="event-invalid">invalid</code> event at the element in the latter case.</p></dd>

   <dt><code data-x=""><var>valid</var> = <var>internals</var>.<span data-x="dom-ElementInternals-reportValidity">reportValidity()</span></code></dt>
   <dd><p>Returns true if <var>internals</var>'s <span data-x="internals-target">target
   element</span> has no validity problems; otherwise, returns false, fires an <code
   data-x="event-invalid">invalid</code> event at the element, and (if the event isn't canceled)
   reports the problem to the user.</p></dd>

   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ElementInternals-labels">labels</span></code></dt>
   <dd><p>Returns a <code>NodeList</code> of all the <code>label</code> elements that
   <var>internals</var>'s <span data-x="internals-target">target element</span> is associated
   with.</p></dd>
  </dl>

  <p>Each <span>form-associated custom element</span> has <dfn
  data-x="face-submission-value">submission value</dfn>. It is used to provide one or more
  <span data-x="form entry">entries</span> on form submission.
  The initial value of <span data-x="face-submission-value">submission value</span> is null, and
  <span data-x="face-submission-value">submission value</span> can be null, a string, a
  <code>File</code>, or a <span>list</span> of <span data-x="form entry">entries</span>.</p>

  <p>Each <span>form-associated custom element</span> has <dfn data-x="face-state">state</dfn>.
  It is information with which the user agent can restore a user's input for the element.
  The initial value of <span data-x="face-state">state</span> is null, and <span
  data-x="face-state">state</span> can be null, a string, a <code>File</code>, or a
  <span>list</span> of <span data-x="form entry">entries</span>.</p>

  <p>The <code data-x="dom-ElementInternals-setFormValue">setFormValue()</code> method is used by
  the custom element author to set the element's <span data-x="face-submission-value">submission
  value</span> and <span data-x="face-state">state</span>, thus communicating these to the user
  agent.</p>

  <p>When the user agent believes it is a good idea to restore a <span>form-associated custom
  element</span>'s <span data-x="face-state">state</span>, for example <span data-x="restore
  persisted state">after navigation</span> or restarting the user agent, they may <span>enqueue a
  custom element callback reaction</span> with that element, callback name "<code
  data-x="">formStateRestoreCallback</code>", and « the state to be restored, "<code
  data-x="">restore</code>" ».</p>

  <p>If the user agent has a form-filling assist feature, then when the feature is invoked, it may
  <span>enqueue a custom element callback reaction</span> with a <span>form-associated custom
  element</span>, callback name "<code data-x="">formStateRestoreCallback</code>", and « the state
  value determined by history of state value and some heuristics, "<code
  data-x="">autocomplete</code>" ».</p>

  <p>In general, the <span data-x="face-state">state</span> is information specified by a user, and
  the <span data-x="face-submission-value">submission value</span> is a value after canonicalization
  or sanitization, suitable for submission to the server. The following examples makes this
  concrete:</p>

  <p class="example">Suppose that we have a <span>form-associated custom element</span> which asks a
  user to specify a date. The user specifies <kbd>"3/15/2019"</kbd>, but the control wishes to
  submit <code data-x="">"2019-03-15"</code> to the server. <kbd>"3/15/2019"</kbd> would be a <span
  data-x="face-state">state</span> of the element, and <code data-x="">"2019-03-15"</code> would be
  a <span data-x="face-submission-value">submission value</span>.</p>

  <p class="example">Suppose you develop a custom element emulating a the behavior of the existing
  <span data-x="attr-input-type-checkbox">checkbox</span> <code>input</code> type. Its <span
  data-x="face-submission-value">submission value</span> would be the value of its <code
  data-x="">value</code> content attribute, or the string <code data-x="">"on"</code>. Its <span
  data-x="face-state">state</span> would be one of <code data-x="">"checked"</code>, <code
  data-x="">"unchecked"</code>, <code data-x="">"checked/indeterminate"</code>, or <code
  data-x="">"unchecked/indeterminate"</code>.</p>

  <div w-nodev>

  <p>The <dfn method for="ElementInternals"><code
  data-x="dom-ElementInternals-setFormValue">setFormValue(<var>value</var>,
  <var>state</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>element</var> be <span>this</span>'s <span data-x="internals-target">target
   element</span>.</p></li>

   <li><p>If <var>element</var> is not a <span>form-associated custom element</span>, then throw a
   <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Set <span data-x="internals-target">target element</span>'s <span
   data-x="face-submission-value">submission value</span> to <var>value</var> if <var>value</var> is
   not a <code>FormData</code> object, or to a <span data-x="list clone">clone</span> of
   <var>value</var>'s <span data-x="FormData entry list">entry list</span> otherwise.</p></li>

   <li><p>If the <var>state</var> argument of the function is omitted, set <var>element</var>'s
   <span data-x="face-state">state</span> to its <span data-x="face-submission-value">submission
   value</span>.

   <li><p>Otherwise, if <var>state</var> is a <code>FormData</code> object, set <var>element</var>'s
   <span data-x="face-state">state</span> to a <span data-x="list clone">clone</span> of
   <var>state</var>'s <span data-x="FormData entry list">entry list</span>.</p></li>

   <li><p>Otherwise, set <var>element</var>'s <span data-x="face-state">state</span> to
   <var>state</var>.</p></li>
  </ol>

  <hr>

  <p>Each <span>form-associated custom element</span> has validity flags named
  <code data-x="">valueMissing</code>, <code data-x="">typeMismatch</code>,
  <code data-x="">patternMismatch</code>, <code data-x="">tooLong</code>,
  <code data-x="">tooShort</code>, <code data-x="">rangeUnderflow</code>,
  <code data-x="">rangeOverflow</code>, <code data-x="">stepMismatch</code>, and
  <code data-x="">customError</code>. They are false initially.</p>

  <p>Each <span>form-associated custom element</span> has a
  <dfn data-x="face-validation-message">validation message</dfn> string. It is the empty string
  initially.</p>

  <p>Each <span>form-associated custom element</span> has a
  <dfn data-x="face-validation-anchor">validation anchor</dfn> element. It is null initially.</p>

  <p>The <dfn method for="ElementInternals"><code
  data-x="dom-ElementInternals-setValidity">setValidity(<var>flags</var>, <var>message</var>,
  <var>anchor</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>element</var> be <span>this</span>'s <span data-x="internals-target">target
   element</span>.</p></li>

   <li><p>If <var>element</var> is not a <span>form-associated custom element</span>, then throw a
   <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <var>flags</var> contains one or more true values and <var>message</var> is not given
   or is the empty string, then throw a <code>TypeError</code>.</p></li>

   <li><p>For each entry <var>flag</var> → <var>value</var> of <var>flags</var>, set
   <var>element</var>'s validity flag with the name <var>flag</var> to <var>value</var>.</p></li>

   <li><p>Set <var>element</var>'s <span data-x="face-validation-message">validation message</span>
   to the empty string if <var>message</var> is not given or all of <var>element</var>'s validity
   flags are false, or to <var>message</var> otherwise.</p></li>

   <li><p>If <var>element</var>'s <code data-x="">customError</code> validity flag is true, then set
   <var>element</var>'s <span>custom validity error message</span> to <var>element</var>'s
   <span data-x="face-validation-message">validation message</span>. Otherwise, set
   <var>element</var>'s <span>custom validity error message</span> to the empty string.</p></li>

   <li><p>Set <var>element</var>'s <span data-x="face-validation-anchor">validation anchor</span> to
   null if <var>anchor</var> is not given. Otherwise, if <var>anchor</var> is not a
   <span>shadow-including descendant</span> of <var>element</var>, then throw a
   <span>"<code>NotFoundError</code>"</span> <code>DOMException</code>. Otherwise, set
   <var>element</var>'s <span data-x="face-validation-anchor">validation anchor</span> to
   <var>anchor</var>.</p></li>
  </ol>

  <p>The <dfn attribute for="ElementInternals"><code
  data-x="dom-ElementInternals-validationMessage">validationMessage</code></dfn> getter steps
  are:</p>

  <ol>
   <li><p>Let <var>element</var> be <span>this</span>'s <span data-x="internals-target">target
   element</span>.</p></li>

   <li><p>If <var>element</var> is not a <span>form-associated custom element</span>, then throw a
   <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Return <var>element</var>'s <span data-x="face-validation-message">validation
   message</span>.</p></li>
  </ol>

  <p>The <dfn data-x="face-entry-construction">entry construction algorithm</dfn> for a
  <span>form-associated custom element</span>, given an element <var>element</var> and an
  <span>entry list</span> <var>entry list</var>, consists of the following steps:</p>

  <ol>
   <li>
    <p>If <var>element</var>'s <span data-x="face-submission-value">submission value</span> is a
    <span>list</span> of <span data-x="form entry">entries</span>, then <span data-x="list
    append">append</span> each item of <var>element</var>'s <span
    data-x="face-submission-value">submission value</span> to <var>entry list</var>, and return.</p>

    <p class="note">In this case, user agent does not refer to the
    <code data-x="attr-fe-name">name</code> content attribute value. An implementation of
    <span>form-associated custom element</span> is responsible to decide names of
    <span data-x="form entry">entries</span>. They can be the
    <code data-x="attr-fe-name">name</code> content attribute value, they can be strings based on
    the <code data-x="attr-fe-name">name</code> content attribute value, or they can be unrelated
    to the <code data-x="attr-fe-name">name</code> content attribute.</p>
   </li>

   <li><p>If the element does not have a <code data-x="attr-fe-name">name</code> attribute
   specified, or its <code data-x="attr-fe-name">name</code> attribute's value is the empty string,
   then return.</p></li>

   <li><p>If the element's <span data-x="face-submission-value">submission value</span> is not
   null, <span>create an entry</span> with the <code data-x="attr-fe-name">name</code> attribute
   value and the <span data-x="face-submission-value">submission value</span>, and <span
   data-x="list append">append</span> it to <var>entry list</var>.</p></li>
  </ol>

  </div>

  <h5>Accessibility semantics</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ARIAMixin-role">role</span> [ = <var>value</var> ]</code></dt>
   <dd><p>Sets or retrieves the default ARIA role for <var>internals</var>'s <span
   data-x="internals-target">target element</span>, which will be used unless the page author
   overrides it using the <code data-x="attr-aria-role">role</code> attribute.</p></dd>

   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ARIAMixin-aria*">aria*</span> [ = <var>value</var> ]</code></dt>
   <dd><p>Sets or retrieves various default ARIA states or property values for
   <var>internals</var>'s <span data-x="internals-target">target element</span>, which will be used
   unless the page author overrides them using the <code data-x="attr-aria-*">aria-*</code>
   attributes.</p></dd>
  </dl>

  <p w-dev>By using the <code data-x="">role</code> and <code data-x="">aria*</code> properties of
  <code>ElementInternals</code>, custom element authors can set default accessibile roles, states,
  and property values for their custom element, similar to how native elements behave. See <a
  href="#custom-elements-accessibility-example">the example above</a> for more details.</p>

  <div w-nodev>

  <p>Each <span>custom element</span> has an <dfn>internal content attribute map</dfn>, which is a
  <span data-x="ordered map">map</span>, initially empty. See the <a href="#wai-aria">Requirements
  related to ARIA and to platform accessibility APIs</a> section for information on how this impacts
  platform accessibility APIs.</p>

  </div>

  <h5>Custom state pseudo-class</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ElementInternals-states">states</span>.add(<var>value</var>)</code></dt>
   <dd>
    <p>Adds the string <var>value</var> to the element's <span>states set</span> to be exposed as a
    pseudo-class.</p>
   </dd>

   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ElementInternals-states">states</span>.has(<var>value</var>)</code></dt>
   <dd>
    <p>Returns true if <var>value</var> is in the element's <span>states set</span>, otherwise
    false.</p>
   </dd>

   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ElementInternals-states">states</span>.delete(<var>value</var>)</code></dt>
   <dd>
    <p>If the element's <span>states set</span> has <var>value</var>, then it will be removed and
    true will be returned. Otherwise, false will be returned.</p>
   </dd>

   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ElementInternals-states">states</span>.clear()</code></dt>
   <dd>
    <p>Removes all values from the element's <span>states set</span>.</p>
   </dd>

   <dt><code data-x="">for (const <var>stateName</var> of <var>internals</var>.<span data-x="dom-ElementInternals-states">states</span>)</code></dt>
   <dt><code data-x="">for (const <var>stateName</var> of <var>internals</var>.<span data-x="dom-ElementInternals-states">states</span>.entries())</code></dt>
   <dt><code data-x="">for (const <var>stateName</var> of <var>internals</var>.<span data-x="dom-ElementInternals-states">states</span>.keys())</code></dt>
   <dt><code data-x="">for (const <var>stateName</var> of <var>internals</var>.<span data-x="dom-ElementInternals-states">states</span>.values())</code></dt>
   <dd>
    <p>Iterates over all values in the element's <span>states set</span>.</p>
   </dd>

   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ElementInternals-states">states</span>.forEach(<var>callback</var>)</code></dt>
   <dd>
    <p>Iterates over all values in the element's <span>states set</span> by calling
    <var>callback</var> once for each value.</p>
   </dd>

   <dt><code data-x=""><var>internals</var>.<span data-x="dom-ElementInternals-states">states</span>.size</code></dt>
   <dd>
    <p>Returns the number of values in the element's <span>states set</span>.</p>
   </dd>
  </dl>

  <div w-nodev>
   <p>Each <span>custom element</span> has a <dfn>states set</dfn>, which is a
   <code>CustomStateSet</code>, initially empty.</p>

   <pre><code class="idl">[Exposed=Window]
interface <dfn interface>CustomStateSet</dfn> {
  setlike&lt;DOMString>;
};</code></pre>

   <p>The <dfn attribute for="HTMLElement"><code
   data-x="dom-ElementInternals-states">states</code></dfn> getter steps are to return
   <span>this</span>'s <span data-x="internals-target">target element</span>'s <span>states
   set</span>.</p>
  </div>

  <div class="example">
   <p>The <span>states set</span> can expose boolean states represented by existence/non-existence
   of string values. If an author wants to expose a state which can have three values, it can be
   converted to three exclusive boolean states. For example, a state called <code
   data-x="">readyState</code> with <code data-x="">"loading"</code>, <code
   data-x="">"interactive"</code>, and <code data-x="">"complete"</code> values can be mapped to
   three exclusive boolean states, <code data-x="">"loading"</code>, <code
   data-x="">"interactive"</code>, and <code data-x="">"complete"</code>:</p>

   <pre><code class="js">// Change the readyState from anything to "complete".
this._readyState = "complete";
this._internals.states.delete("loading");
this._internals.states.delete("interactive");
this._internals.states.add("complete");</code></pre>
  </div>

  <h3 split-filename="semantics-other" id="common-idioms">Common idioms without dedicated elements</h3>

  <h4 id="rel-up">Breadcrumb navigation</h4>

  <p>This specification does not provide a machine-readable way of describing breadcrumb navigation
  menus. Authors are encouraged to just use a series of links in a paragraph. The <code>nav</code>
  element can be used to mark the section containing these paragraphs as being navigation
  blocks.</p>

  <!-- If there was implementation interest, we could go back to rel="up", rel="up up", rel="up up
  top" to programmatically indicate the actual breadcrumb structure. However, UAs and ATs don't seem
  interested in exposing this in a useful way to users. So... -->

  <div class="example">

   <p>In the following example, the current page can be reached via two paths.</p>

   <pre><code class="html">&lt;nav>
 &lt;p>
  &lt;a href="/">Main&lt;/a> &#x25B8;
  &lt;a href="/products/">Products&lt;/a> &#x25B8;
  &lt;a href="/products/dishwashers/">Dishwashers&lt;/a> &#x25B8;
  &lt;a>Second hand&lt;/a>
 &lt;/p>
 &lt;p>
  &lt;a href="/">Main&lt;/a> &#x25B8;
  &lt;a href="/second-hand/">Second hand&lt;/a> &#x25B8;
  &lt;a>Dishwashers&lt;/a>
 &lt;/p>
&lt;/nav></code></pre>

  </div>


  <h4>Tag clouds</h4>

  <p id="tag-cloud">This specification does not define any markup specifically for marking up lists
  of keywords that apply to a group of pages (also known as <i>tag clouds</i>). In general, authors
  are encouraged to either mark up such lists using <code>ul</code> elements with explicit inline
  counts that are then hidden and turned into a presentational effect using a style sheet, or to use
  SVG.</p>

  <div class="example">

   <p>Here, three tags are included in a short tag cloud:</p>

   <pre><code class="html">&lt;style>
.tag-cloud > li > span { display: none; }
.tag-cloud > li { display: inline; }
.tag-cloud-1 { font-size: 0.7em; }
.tag-cloud-2 { font-size: 0.9em; }
.tag-cloud-3 { font-size: 1.1em; }
.tag-cloud-4 { font-size: 1.3em; }
.tag-cloud-5 { font-size: 1.5em; }

@media speech {
  .tag-cloud > li > span { display:inline }
}
&lt;/style>
...
&lt;ul class="tag-cloud">
 &lt;li class="tag-cloud-4">&lt;a title="28 instances" href="/t/apple">apple&lt;/a> &lt;span>(popular)&lt;/span>
 &lt;li class="tag-cloud-2">&lt;a title="6 instances"  href="/t/kiwi">kiwi&lt;/a> &lt;span>(rare)&lt;/span>
 &lt;li class="tag-cloud-5">&lt;a title="41 instances" href="/t/pear">pear&lt;/a> &lt;span>(very popular)&lt;/span>
&lt;/ul></code></pre>

   <p>The actual frequency of each tag is given using the <code data-x="attr-title">title</code>
   attribute. A CSS style sheet is provided to convert the markup into a cloud of differently-sized
   words, but for user agents that do not support CSS or are not visual, the markup contains
   annotations like "(popular)" or "(rare)" to categorize the various tags by frequency, thus
   enabling all users to benefit from the information.</p>

   <p>The <code>ul</code> element is used (rather than <code>ol</code>) because the order is not
   particularly important: while the list is in fact ordered alphabetically, it would convey the
   same information if ordered by, say, the length of the tag.</p>

   <p>The <code data-x="rel-tag">tag</code> <code data-x="attr-hyperlink-rel">rel</code>-keyword is
   <em>not</em> used on these <code>a</code> elements because they do not represent tags that apply
   to the page itself; they are just part of an index listing the tags themselves.</p>

  </div>


  <h4>Conversations</h4>

  <!-- https://lists.w3.org/Archives/Public/public-whatwg-archive/2009Sep/0043.html -->

  <p>This specification does not define a specific element for marking up conversations, meeting
  minutes, chat transcripts, dialogues in screenplays, instant message logs, and other situations
  where different players take turns in discourse.</p>

  <p>Instead, authors are encouraged to mark up conversations using <code>p</code> elements and
  punctuation. Authors who need to mark the speaker for styling purposes are encouraged to use
  <code>span</code> or <code>b</code>. Paragraphs with their text wrapped in the <code>i</code>
  element can be used for marking up stage directions.</p>

  <div class="example">

   <p>This example demonstrates this using an extract from Abbot and Costello's famous sketch,
   <cite>Who's on first</cite>:</p>

<pre><code class="html">&lt;p> Costello: Look, you gotta first baseman?
&lt;p> Abbott: Certainly.
&lt;p> Costello: Who's playing first?
&lt;p> Abbott: That's right.
&lt;p> Costello becomes exasperated.
&lt;p> Costello: When you pay off the first baseman every month, who gets the money?
&lt;p> Abbott: Every dollar of it.</code></pre>

  </div>

  <div class="example">

   <p>The following extract shows how an IM conversation log could be marked up, using the
   <code>data</code> element to provide Unix timestamps for each line. Note that the timestamps are
   provided in a format that the <code>time</code> element does not support, so the
   <code>data</code> element is used instead (namely, Unix <code data-x="">time_t</code> timestamps).
   Had the author wished to mark up the data using one of the date and time formats supported by the
   <code>time</code> element, that element could have been used instead of <code>data</code>. This
   could be advantageous as it would allow data analysis tools to detect the timestamps
   unambiguously, without coordination with the page author.</p>

   <pre><code class="html">&lt;p> &lt;data value="1319898155">14:22&lt;/data> &lt;b>egof&lt;/b> I'm not that nerdy, I've only seen 30% of the star trek episodes
&lt;p> &lt;data value="1319898192">14:23&lt;/data> &lt;b>kaj&lt;/b> if you know what percentage of the star trek episodes you have seen, you are inarguably nerdy
&lt;p> &lt;data value="1319898200">14:23&lt;/data> &lt;b>egof&lt;/b> it's unarguably
&lt;p> &lt;data value="1319898228">14:23&lt;/data> &lt;i>* kaj blinks&lt;/i>
&lt;p> &lt;data value="1319898260">14:24&lt;/data> &lt;b>kaj&lt;/b> you are not helping your case</code></pre>
   <!-- with thanks to http://bash.org/?854262 -->

  </div>

  <div class="example">

   <p>HTML does not have a good way to mark up graphs, so descriptions of interactive conversations
   from games are more difficult to mark up. This example shows one possible convention using
   <code>dl</code> elements to list the possible responses at each point in the conversation.
   Another option to consider is describing the conversation in the form of a DOT file, and
   outputting the result as an SVG image to place in the document. <ref>DOT</ref></p>

   <pre><code class="html">&lt;p> Next, you meet a fisher. You can say one of several greetings:
&lt;dl>
 &lt;dt> "Hello there!"
 &lt;dd>
  &lt;p> She responds with "Hello, how may I help you?"; you can respond with:
  &lt;dl>
   &lt;dt> "I would like to buy a fish."
   &lt;dd> &lt;p> She sells you a fish and the conversation finishes.
   &lt;dt> "Can I borrow your boat?"
   &lt;dd>
    &lt;p> She is surprised and asks "What are you offering in return?".
    &lt;dl>
     &lt;dt> "Five gold." (if you have enough)
     &lt;dt> "Ten gold." (if you have enough)
     &lt;dt> "Fifteen gold." (if you have enough)
     &lt;dd> &lt;p> She lends you her boat. The conversation ends.
     &lt;dt> "A fish." (if you have one)
     &lt;dt> "A newspaper." (if you have one)
     &lt;dt> "A pebble." (if you have one)
     &lt;dd> &lt;p> "No thanks", she replies. Your conversation options
     at this point are the same as they were after asking to borrow
     her boat, minus any options you've suggested before.
    &lt;/dl>
   &lt;/dd>
  &lt;/dl>
 &lt;/dd>
 &lt;dt> "Vote for me in the next election!"
 &lt;dd> &lt;p> She turns away. The conversation finishes.
 &lt;dt> "Madam, are you aware that your fish are running away?"
 &lt;dd>
  &lt;p> She looks at you skeptically and says "Fish cannot run, miss".
  &lt;dl>
   &lt;dt> "You got me!"
   &lt;dd> &lt;p> The fisher sighs and the conversation ends.
   &lt;dt> "Only kidding."
   &lt;dd> &lt;p> "Good one!" she retorts. Your conversation options at this
   point are the same as those following "Hello there!" above.
   &lt;dt> "Oh, then what are they doing?"
   &lt;dd> &lt;p> She looks at her fish, giving you an opportunity to steal
   her boat, which you do. The conversation ends.
  &lt;/dl>
 &lt;/dd>
&lt;/dl></code></pre>

  </div>

  <div class="example">

   <p>In some games, conversations are simpler: each character merely has a fixed set of lines that
   they say. In this example, a game FAQ/walkthrough lists some of the known possible responses for
   each character:</p>

   <pre><code class="html">&lt;section>
 &lt;h1>Dialogue&lt;/h1>
 &lt;p>&lt;small>Some characters repeat their lines in order each time you interact
 with them, others randomly pick from amongst their lines. Those who respond in
 order have numbered entries in the lists below.&lt;/small>
 &lt;h2>The Shopkeeper&lt;/h2>
 &lt;ul>
  &lt;li>How may I help you?
  &lt;li>Fresh apples!
  &lt;li>A loaf of bread for madam?
 &lt;/ul>
 &lt;h2>The pilot&lt;/h2>
 &lt;p>Before the accident:
 &lt;ul>
  &lt;li>I'm about to fly out, sorry!
  &lt;li>Sorry, I'm just waiting for flight clearance and then I'll be off!
 &lt;/ul>
 &lt;p>After the accident:
 &lt;ol>
  &lt;li>I'm about to fly out, sorry!
  &lt;li>Ok, I'm not leaving right now, my plane is being cleaned.
  &lt;li>Ok, it's not being cleaned, it needs a minor repair first.
  &lt;li>Ok, ok, stop bothering me! Truth is, I had a crash.
 &lt;/ol>
 &lt;h2>Clan Leader&lt;/h2>
 &lt;p>During the first clan meeting:
 &lt;ul>
  &lt;li>Hey, have you seen my daughter? I bet she's up to something nefarious again...
  &lt;li>Nice weather we're having today, eh?
  &lt;li>The name is Bailey, Jeff Bailey. How can I help you today?
  &lt;li>A glass of water? Fresh from the well!
 &lt;/ul>
 &lt;p>After the earthquake:
 &lt;ol>
  &lt;li>Everyone is safe in the shelter, we just have to put out the fire!
  &lt;li>I'll go and tell the fire brigade, you keep hosing it down!
 &lt;/ol>
&lt;/section></code></pre>

  </div>


  <h4 id="footnotes">Footnotes</h4>

  <p>HTML does not have a dedicated mechanism for marking up footnotes. Here are the suggested
  alternatives.</p>

  <hr>

  <p>For short inline annotations, the <code data-x="attr-title">title</code> attribute could <!--
  SHOULD, modulo title-warning --> be used.</p>

  <div class="example">

   <p>In this example, two parts of a dialogue are annotated with footnote-like content using the
   <code data-x="attr-title">title</code> attribute.</p>

   <pre><code class="html">&lt;p> &lt;b>Customer&lt;/b>: Hello! I wish to register a complaint. Hello. Miss?
&lt;p> &lt;b>Shopkeeper&lt;/b>: <strong>&lt;span title="Colloquial pronunciation of 'What do you'"</strong>
>Watcha&lt;/span> mean, miss?
&lt;p> &lt;b>Customer&lt;/b>: Uh, I'm sorry, I have a cold. I wish to make a complaint.
&lt;p> &lt;b>Shopkeeper&lt;/b>: Sorry, &lt;span <strong>title="This is, of course, a lie."</strong>>we're
closing for lunch&lt;/span>.</code></pre>

  </div>

  <!-- search for title-warning if modifying this paragraph -->
  <p class="note">Unfortunately, relying on the <code data-x="attr-title">title</code> attribute is
  currently discouraged as many user agents do not expose the attribute in an accessible manner as
  required by this specification (e.g. requiring a pointing device such as a mouse to cause a
  tooltip to appear, which excludes keyboard-only users and touch-only users, such as anyone with a
  modern phone or tablet).</p>

  <p class="note">If the <code data-x="attr-title">title</code> attribute is used, CSS can be used to
  draw the reader's attention to the elements with the attribute.</p>

  <div class="example">

   <p>For example, the following CSS places a dashed line below elements that have a <code
   data-x="attr-title">title</code> attribute.</p>

   <pre><code class="css">[title] { border-bottom: thin dashed; }</code></pre>

  </div>

  <hr>

  <p>For longer annotations, the <code>a</code> element should be used, pointing to an element later
  in the document. The convention is that the contents of the link be a number in square
  brackets.</p>

  <div class="example">

   <p>In this example, a footnote in the dialogue links to a paragraph below the dialogue. The
   paragraph then reciprocally links back to the dialogue, allowing the user to return to the
   location of the footnote.</p>

   <pre><code class="html">&lt;p> Announcer: Number 16: The &lt;i>hand&lt;/i>.
&lt;p> Interviewer: Good evening. I have with me in the studio tonight
Mr Norman St John Polevaulter, who for the past few years has been
contradicting people. Mr Polevaulter, why &lt;em>do&lt;/em> you
contradict people?
&lt;p> Norman: I don't. &lt;sup>&lt;a href="#fn1" id="r1">[1]&lt;/a>&lt;/sup>
&lt;p> Interviewer: You told me you did!
<em>...</em>
&lt;section>
 &lt;p id="fn1">&lt;a href="#r1">[1]&lt;/a> This is, naturally, a lie,
 but paradoxically if it were true he could not say so without
 contradicting the interviewer and thus making it false.&lt;/p>
&lt;/section></code></pre>

  </div>

  <hr>

  <p>For side notes, longer annotations that apply to entire sections of the text rather than just
  specific words or sentences, the <code>aside</code> element should be used.</p>

  <div class="example">

   <p>In this example, a sidebar is given after a dialogue, giving it some context.</p>

   <pre><code class="html">&lt;p> &lt;span class="speaker">Customer&lt;/span>: I will not buy this record, it is scratched.
&lt;p> &lt;span class="speaker">Shopkeeper&lt;/span>: I'm sorry?
&lt;p> &lt;span class="speaker">Customer&lt;/span>: I will not buy this record, it is scratched.
&lt;p> &lt;span class="speaker">Shopkeeper&lt;/span>: No no no, this's'a tobacconist's.
&lt;aside>
 &lt;p>In 1970, the British Empire lay in ruins, and foreign
 nationalists frequented the streets &mdash; many of them Hungarians
 (not the streets &mdash; the foreign nationals). Sadly, Alexander
 Yalt has been publishing incompetently-written phrase books.
&lt;/aside></code></pre>

  </div>

  <hr>

  <p>For figures or tables, footnotes can be included in the relevant <code>figcaption</code> or
  <code>caption</code> element, or in surrounding prose.</p>

  <div class="example">

   <p>In this example, a <!-- round --> table has cells with footnotes that are given in prose. A
   <code>figure</code> element is used to give a single legend to the combination of the table and
   its footnotes.</p>

   <pre><code class="html">&lt;figure>
 &lt;figcaption>Table 1. Alternative activities for knights.&lt;/figcaption>
 &lt;table>
  &lt;tr>
   &lt;th> Activity
   &lt;th> Location
   &lt;th> Cost
  &lt;tr>
   &lt;td> Dance
   &lt;td> Wherever possible
   &lt;td> &#x00A3;0&lt;sup>&lt;a href="#fn1">1&lt;/a>&lt;/sup>
  &lt;tr>
   &lt;td> Routines, chorus scenes&lt;sup>&lt;a href="#fn2">2&lt;/a>&lt;/sup>
   &lt;td> Undisclosed
   &lt;td> Undisclosed
  &lt;tr>
   &lt;td> Dining&lt;sup>&lt;a href="#fn3">3&lt;/a>&lt;/sup>
   &lt;td> Camelot
   &lt;td> Cost of ham, jam, and spam&lt;sup>&lt;a href="#fn4">4&lt;/a>&lt;/sup>
 &lt;/table>
 &lt;p id="fn1">1. Assumed.&lt;/p>
 &lt;p id="fn2">2. Footwork impeccable.&lt;/p>
 &lt;p id="fn3">3. Quality described as "well".&lt;/p>
 &lt;p id="fn4">4. A lot.&lt;/p>
&lt;/figure></code></pre>

  </div>


  <h3>Disabled elements</h3>

  <p>An element is said to be <dfn data-x="concept-element-disabled">actually disabled</dfn> if it
  is one of the following:</p>

  <ul>
   <li>a <code>button</code> element that is <span data-x="concept-fe-disabled">disabled</span></li>

   <li>an <code>input</code> element that is <span data-x="concept-fe-disabled">disabled</span></li>

   <li>a <code>select</code> element that is <span data-x="concept-fe-disabled">disabled</span></li>

   <li>a <code>textarea</code> element that is <span data-x="concept-fe-disabled">disabled</span></li>

   <li>an <code>optgroup</code> element that has a <code data-x="attr-optgroup-disabled">disabled</code> attribute</li>

   <li>an <code>option</code> element that is <span data-x="concept-option-disabled">disabled</span></li>

   <li>a <code>fieldset</code> element that is a <span data-x="concept-fieldset-disabled">disabled fieldset</span></li>

   <li>a <span>form-associated custom element</span> that is
   <span data-x="concept-fe-disabled">disabled</span></li>
  </ul>

  <p class="note">This definition is used to determine what elements are <span>focusable</span> and
  which elements match the <code data-x="selector-enabled">:enabled</code> and
  <code data-x="selector-disabled">:disabled</code> <span data-x="pseudo-class">pseudo
  classes</span>.</p>


  <div w-nodev>

  <h3 id="selectors">Matching HTML elements using selectors and CSS</h3>

  <h4>Case-sensitivity of the CSS <span>'attr()'</span> function</h4>

  <p><cite>CSS Values and Units</cite> leaves the case-sensitivity of attribute names for the
  purpose of the <span>'attr()'</span> function to be defined by the host language.
  <ref>CSSVALUES</ref></p>

  <p>When comparing the attribute name part of a CSS <span>'attr()'</span> function to the names of
  namespace-less attributes on <span>HTML elements</span> in <span>HTML documents</span>, the name
  part of the CSS <span>'attr()'</span> function must first be <span>converted to ASCII
  lowercase</span>. The same function when compared to other attributes must be compared according
  to its original case. In both cases, to match the values must be <span>identical to</span> each
  other (and therefore the comparison is case sensitive).</p>

  <p class="note">This is the same as comparing the name part of a CSS <span>attribute
  selector</span>, specified in the next section.</p>


  <h4>Case-sensitivity of selectors</h4>

  <p><cite>Selectors</cite> leaves the case-sensitivity of element names, attribute names, and
  attribute values to be defined by the host language. <ref>SELECTORS</ref></p>

  <p>When comparing a CSS element <span>type selector</span> to the names of <span>HTML
  elements</span> in <span>HTML documents</span>, the CSS element <span>type selector</span> must
  first be <span>converted to ASCII lowercase</span>. The same selector when compared to other
  elements must be compared according to its original case. In both cases, to match the values must
  be <span>identical to</span> each other (and therefore the comparison is case sensitive).</p>
  <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2623 -->

  <p>When comparing the name part of a CSS <span>attribute selector</span> to the names of
  attributes on <span>HTML elements</span> in <span>HTML documents</span>, the name
  part of the CSS <span>attribute selector</span> must first be <span>converted to ASCII
  lowercase</span>. The same selector when compared to other attributes must be compared according
  to its original case. In both cases, the comparison is case-sensitive.</p>
  <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2624 -->

  <p><span data-x="attribute selector">Attribute selectors</span> on an <span
  data-x="HTML elements">HTML element</span> in an <span data-x="HTML documents">HTML
  document</span> must treat the <em>values</em> of attributes with the following names as
  <span>ASCII case-insensitive</span>:</p>

  <!-- based on Mozilla's list, which was itself based on HTML4 -->
  <!-- WebKit's was identical at the time of writing except as noted below -->
  <ul class="brief">
   <li><code data-x="">accept</code>
   <li><code data-x="">accept-charset</code>
   <li><code data-x="">align</code>
   <li><code data-x="">alink</code>
   <li><code data-x="">axis</code>
   <li><code data-x="">bgcolor</code>
   <li><code data-x="">charset</code>
   <li><code data-x="">checked</code>
   <li><code data-x="">clear</code>
   <li><code data-x="">codetype</code>
   <li><code data-x="">color</code>
   <li><code data-x="">compact</code>
   <li><code data-x="">declare</code>
   <li><code data-x="">defer</code>
   <li><code data-x="">dir</code>
   <li><code data-x="">direction</code> <!-- not in WebKit -->
   <li><code data-x="">disabled</code>
   <li><code data-x="">enctype</code>
   <li><code data-x="">face</code>
   <li><code data-x="">frame</code>
   <li><code data-x="">hreflang</code>
   <li><code data-x="">http-equiv</code>
   <li><code data-x="">lang</code>
   <li><code data-x="">language</code>
   <li><code data-x="">link</code>
   <li><code data-x="">media</code>
   <li><code data-x="">method</code>
   <li><code data-x="">multiple</code>
   <li><code data-x="">nohref</code>
   <li><code data-x="">noresize</code>
   <li><code data-x="">noshade</code>
   <li><code data-x="">nowrap</code>
   <li><code data-x="">readonly</code>
   <li><code data-x="">rel</code>
   <li><code data-x="">rev</code>
   <li><code data-x="">rules</code>
   <li><code data-x="">scope</code>
   <li><code data-x="">scrolling</code>
   <li><code data-x="">selected</code>
   <li><code data-x="">shape</code>
   <li><code data-x="">target</code>
   <li><code data-x="">text</code>
   <li><code data-x="">type</code>
   <li><code data-x="">valign</code>
   <li><code data-x="">valuetype</code>
   <li><code data-x="">vlink</code>
  </ul>

  <div class="example">
   <p>For example, the selector <code data-x="">[bgcolor="#ffffff"]</code> will match any HTML
   element with a <code data-x="">bgcolor</code> attribute with values including <code
   data-x="">#ffffff</code>, <code data-x="">#FFFFFF</code> and <code data-x="">#fffFFF</code>. This
   happens even if <code data-x="">bgcolor</code> has no effect for a given element (e.g.,
   <code>div</code>).</p>

   <p>The selector <code data-x="">[type=a <!--grammar-check-override-->s]</code> will match any
   HTML element with a <code data-x="">type</code> attribute whose value is <code
   data-x="">a</code>, but not whose value is <code data-x="">A</code>, due to the <code
   data-x="">s</code> flag.</p>
  </div>

  <p>All other attribute values and everything else must be treated as entirely <span>identical
  to</span> each other for the purposes of selector matching. This includes:</p>

  <ul>
   <li><p><span data-x="concept-id">IDs</span> and <span data-x="concept-class">classes</span>
   in <span>no-quirks mode</span> and <span>limited-quirks mode</span></p></li>

   <li><p>the names of elements not in the <span>HTML namespace</span></p></li>

   <li><p>the names of <span>HTML elements</span> in <span>XML documents</span></p></li>

   <li><p>the names of attributes of elements not in the <span>HTML namespace</span></p></li>

   <li><p>the names of attributes of <span>HTML elements</span> in <span>XML documents</span></p></li>

   <li><p>the names of attributes that themselves have namespaces</p></li>
  </ul>

  <p class="note"><cite>Selectors</cite> defines that ID and class selectors (such as <code
  data-x="">#foo</code> and <code data-x="">.bar</code>), when matched against elements in documents
  that are in <span>quirks mode</span>, will be matched in an <span>ASCII case-insensitive</span>
  manner. However, this does not apply for attribute selectors with "<code data-x="">id</code>" or
  "<code data-x="">class</code>" as the name part. The selector <code
  data-x="">[class="foobar"]</code> will treat its value as case-sensitive even in
  <span>quirks mode</span>.</p>


  <h4>Pseudo-classes</h4>

  <p>There are a number of dynamic selectors that can be used with HTML. This section defines when
  these selectors match HTML elements. <ref>SELECTORS</ref> <ref>CSSUI</ref></p>

  <dl>
   <dt><dfn selector noexport><code data-x="selector-defined">:defined</code></dfn></dt>
   <!-- noexport is used explicitly here and for other psuedo-classes below as the script
        scraping HTML for references (Shepherd) automatically exports terms that look like
        pseudo-classes. -->
   <dd>
    <p>The <code data-x="selector-defined">:defined</code> <span>pseudo-class</span> must match
    any element that is <span data-x="concept-element-defined">defined</span>.</p>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-link">:link</code></dfn></dt>
   <dt><dfn selector noexport><code data-x="selector-visited">:visited</code></dfn></dt>
   <dd>
    <p>All <code>a</code> elements that have an <code data-x="attr-hyperlink-href">href</code>
    attribute, and all <code>area</code> elements that have an <code
    data-x="attr-hyperlink-href">href</code> attribute, must match one of <code
    data-x="selector-link">:link</code> and <code data-x="selector-visited">:visited</code>.</p>

    <p>Other specifications might apply more specific rules regarding how these elements are to
    match these <span data-x="pseudo-class">pseudo-classes</span>, to mitigate some privacy concerns
    that apply with straightforward implementations of this requirement.</p>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-active">:active</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-active">:active</code> <span>pseudo-class</span> is defined to
    match an element <q cite="https://drafts.csswg.org/selectors/#the-active-pseudo">while an
    element is <dfn><i data-x="concept-selector-active">being activated</i></dfn> by the
    user</q>.</p>

    <p>To determine whether a particular element is <i data-x="concept-selector-active">being
    activated</i> for the purposes of defining the <code data-x="selector-active">:active</code>
    <span>pseudo-class</span> only, an HTML user agent must use the first relevant entry in the
    following list.</p>

    <dl>
     <dt>If the element is a <code>button</code> element</dt>
     <dt>If the element is an <code>input</code> element whose <code
     data-x="attr-input-type">type</code> attribute is in the <span
     data-x="attr-input-type-submit">Submit Button</span>, <span
     data-x="attr-input-type-image">Image Button</span>, <span data-x="attr-input-type-reset">Reset
     Button</span>, or <span data-x="attr-input-type-button">Button</span> state</dt>
     <dt>If the element is an <code>a</code> element that has an <code
     data-x="attr-hyperlink-href">href</code> attribute</dt>
     <dt>If the element is an <code>area</code> element that has an <code
     data-x="attr-hyperlink-href">href</code> attribute</dt>
     <dt>If the element is <span>focusable</span></dt>
     <dd>
      <p>The element is <i data-x="concept-selector-active">being activated</i> if it is <span>in a
      formal activation state</span>.</p>

      <p class="example">For example, if the user is using a keyboard to push a <code>button</code>
      element by pressing the space bar, the element would match this <span>pseudo-class</span> in
      between the time that the element received the <code data-x="event-keydown">keydown</code>
      event and the time the element received the <code data-x="event-keyup">keyup</code> event.</p>
     </dd>

     <dt>If the element is <span>being actively pointed at</span></dt>
     <dd><p>The element is <i data-x="concept-selector-active">being activated</i>.</p></dd>
    </dl>

    <p>An element is said to be <dfn>in a formal activation state</dfn> between the time the user
    begins to indicate an intent to trigger the element's <span>activation behavior</span> and
    either the time the user stops indicating an intent to trigger the element's <span>activation
    behavior</span>, or the time the element's <span>activation behavior</span> has finished
    running, which ever comes first.</p>

    <p>An element is said to be <dfn>being actively pointed at</dfn> while the user indicates the
    element using a pointing device while that pointing device is in the "down" state (e.g. for a
    mouse, between the time the mouse button is pressed and the time it is depressed; for a finger
    in a multitouch environment, while the finger is touching the display surface).</p>

    <p class="note">Per the definition in <cite>Selectors</cite>, <code
    data-x="selector-active">:active</code> also matches <span>flat tree</span> ancestors of
    elements that are <i data-x="concept-selector-active">being activated</i>.
    <ref>SELECTORS</ref></p>

    <p>Additionally, any element that is the <span>labeled control</span> of a <code>label</code>
    element that is currently matching <code data-x="selector-active">:active</code>, also matches
    <code data-x="selector-active">:active</code>. (But, it does not count as being <i
    data-x="concept-selector-active">being activated</i>.)</p>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-hover">:hover</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-hover">:hover</code> <span>pseudo-class</span> is defined to
    match an element <q cite="https://drafts.csswg.org/selectors/#the-hover-pseudo">while the user
    <dfn><i data-x="concept-selector-hover">designates</i></dfn> an element with a pointing
    device</q>. For the purposes of defining the <code data-x="selector-hover">:hover</code>
    <span>pseudo-class</span> only, an HTML user agent must consider an element as being one that
    the user <i data-x="concept-selector-hover">designates</i> if it is an element that the user
    indicates using a pointing device.</p>

    <p class="note">Per the definition in <cite>Selectors</cite>, <code
    data-x="selector-hover">:hover</code> also matches <span>flat tree</span> ancestors of elements
    that are <i data-x="concept-selector-hover">designated</i>. <ref>SELECTORS</ref></p>

    <p>Additionally, any element that is the <span>labeled control</span> of a <code>label</code>
    element that is currently matching <code data-x="selector-hover">:hover</code>, also matches
    <code data-x="selector-hover">:hover</code>. (But, it does not count as being <i
    data-x="concept-selector-hover">designated</i>.)</p>

<!--
Demos:
 https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1315
 https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1316
 https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1317
-->

    <div class="example">
     <p>Consider in particular a fragment such as:</p>

     <pre><code class="html">&lt;p> &lt;label for=c> &lt;input id=a> &lt;/label> &lt;span id=b> &lt;input id=c> &lt;/span> &lt;/p></code></pre>

     <p>If the user designates the element with ID "<code data-x="">a</code>" with their pointing
     device, then the <code>p</code> element (and all its ancestors not shown in the snippet
     above), the <code>label</code> element, the element with ID "<code data-x="">a</code>", and
     the element with ID "<code data-x="">c</code>" will match the <code
     data-x="selector-hover">:hover</code> <span>pseudo-class</span>. The element with ID "<code
     data-x="">a</code>" matches it by being <i data-x="concept-selector-hover">designated</i>; the
     <code>label</code> and <code>p</code> elements match it because of the condition in
     <cite>Selectors</cite> about flat tree ancestors; and the element with ID "<code
     data-x="">c</code>" matches it through the additional condition above on <span data-x="labeled
     control">labeled controls</span> (i.e., its <code>label</code> element matches <code
     data-x="selector-hover">:hover</code>). However, the element with ID "<code
     data-x="">b</code>" does <em>not</em> match <code data-x="selector-hover">:hover</code>: its
     flat tree descendant is not designated, even though that flat tree descendant matches <code
     data-x="selector-hover">:hover</code>.</p>
    </div>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-focus">:focus</code></dfn></dt>
   <dd>
    <p>For the purposes of the CSS <code data-x="selector-focus">:focus</code>
    <span>pseudo-class</span>, an <dfn>element has the focus</dfn> when:</p>

    <ul>
     <li><p>it is not itself a <span>navigable container</span>; and</p></li>
     <li>
      <p>any of the following are true:</p>
      <ul>
       <li><p>it is one of the elements listed in the <span data-x="current focus chain of a
       top-level traversable">current focus chain of the top-level traversable</span>; or</p></li>

       <li><p>its <span data-x="concept-element-shadow-root">shadow root</span>
       <var>shadowRoot</var> is not null and <var>shadowRoot</var> is the <span>root</span> of at
       least one element that <span data-x="element has the focus">has the focus</span>.</p></li>
      </ul>
     </li>
    </ul>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-target">:target</code></dfn></dt>
   <dd>
    <p>For the purposes of the CSS <code data-x="selector-target">:target</code>
    <span>pseudo-class</span>, the <code>Document</code>'s <i>target elements</i> are a list
    containing the <code>Document</code>'s <span>target element</span>, if it is not null, or
    containing no elements, if it is. <ref>SELECTORS</ref></p>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-popover-open">:popover-open</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-popover-open">:popover-open</code> <span>pseudo-class</span> is
    defined to match any <span data-x="html elements">HTML element</span> whose <code
    data-x="attr-popover">popover</code> attribute is not in the <span
    data-x="attr-popover-none-state">no popover state</span> and whose <span>popover visibility
    state</span> is <span data-x="popover-showing-state">showing</span>.</p>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-enabled">:enabled</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-enabled">:enabled</code> <span>pseudo-class</span> must match any
    <code>button</code>, <code>input</code>, <code>select</code>, <code>textarea</code>,
    <code>optgroup</code>, <code>option</code>, <code>fieldset</code> element, or
    <span>form-associated custom element</span> that is not <span
    data-x="concept-element-disabled">actually disabled</span>.</p>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-disabled">:disabled</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-disabled">:disabled</code> <span>pseudo-class</span> must match
    any element that is <span data-x="concept-element-disabled">actually disabled</span>.</p>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-checked">:checked</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-checked">:checked</code> <span>pseudo-class</span> must match any
    element falling into one of the following categories:</p>

    <ul>
     <li><code>input</code> elements whose <code data-x="attr-input-type">type</code> attribute is in
     the <span data-x="attr-input-type-checkbox">Checkbox</span> state and whose <span
     data-x="concept-fe-checked">checkedness</span> state is true</li>

     <li><code>input</code> elements whose <code data-x="attr-input-type">type</code> attribute is in
     the <span data-x="attr-input-type-radio">Radio Button</span> state and whose <span
     data-x="concept-fe-checked">checkedness</span> state is true</li>

     <li><code>option</code> elements whose <span
     data-x="concept-option-selectedness">selectedness</span> is true</li>
    </ul>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-indeterminate">:indeterminate</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-indeterminate">:indeterminate</code> <span>pseudo-class</span>
    must match any element falling into one of the following categories:</p>

    <ul>
     <li><code>input</code> elements whose <code data-x="attr-input-type">type</code> attribute is in
     the <span data-x="attr-input-type-checkbox">Checkbox</span> state and whose <code
     data-x="dom-input-indeterminate">indeterminate</code> IDL attribute is set to true</li>

     <li><code>input</code> elements whose <code data-x="attr-input-type">type</code> attribute is in
     the <span data-x="attr-input-type-radio">Radio Button</span> state and whose <span>radio button
     group</span> contains no <code>input</code> elements whose <span
     data-x="concept-fe-checked">checkedness</span> state is true.</li>

     <li><code>progress</code> elements with no <code data-x="attr-progress-value">value</code>
     content attribute</li>
    </ul>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-default">:default</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-default">:default</code> <span>pseudo-class</span> must match any
    element falling into one of the following categories:</p>

    <ul>
     <li><span data-x="concept-submit-button">Submit buttons</span> that are <span data-x="default
     button">default buttons</span> of their <span>form owner</span>.</li>

     <li><code>input</code> elements to which the <code data-x="attr-input-checked">checked</code>
     attribute applies and that have a <code data-x="attr-input-checked">checked</code>
     attribute</li>

     <li><code>option</code> elements that have a <code data-x="attr-option-selected">selected</code>
     attribute</li>
    </ul>
   </dd>

   <dt><dfn selector noexport><code
   data-x="selector-placeholder-shown">:placeholder-shown</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-placeholder-shown">:placeholder-shown</code>
    <span>pseudo-class</span> must match any element falling into one of the following
    categories:</p>

    <ul>
     <li><code>input</code> elements that have a <code
     data-x="attr-input-placeholder">placeholder</code> attribute whose value is currently being
     presented to the user</li>

     <li><code>textarea</code> elements that have a <code
     data-x="attr-textarea-placeholder">placeholder</code> attribute whose value is currently being
     presented to the user</li>
    </ul>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-valid">:valid</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-valid">:valid</code> <span>pseudo-class</span> must match any
    element falling into one of the following categories:</p>

    <ul>
     <li>elements that are <span data-x="candidate for constraint validation">candidates for
     constraint validation</span> and that <span data-x="concept-fv-valid">satisfy their
     constraints</span></li>

     <li><code>form</code> elements that are not the <span>form owner</span> of any elements that
     themselves are <span data-x="candidate for constraint validation">candidates for constraint
     validation</span> but do not <span data-x="concept-fv-valid">satisfy their
     constraints</span></li>

     <li><code>fieldset</code> elements that have no descendant elements that themselves are <span
     data-x="candidate for constraint validation">candidates for constraint validation</span> but do
     not <span data-x="concept-fv-valid">satisfy their constraints</span></li>
    </ul>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-invalid">:invalid</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-invalid">:invalid</code> <span>pseudo-class</span> must match any
    element falling into one of the following categories:</p>

    <ul>
     <li>elements that are <span data-x="candidate for constraint validation">candidates for
     constraint validation</span> but that do not <span data-x="concept-fv-valid">satisfy their
     constraints</span></li>

     <li><code>form</code> elements that are the <span>form owner</span> of one or more elements
     that themselves are <span data-x="candidate for constraint validation">candidates for constraint
     validation</span> but do not <span data-x="concept-fv-valid">satisfy their
     constraints</span></li>

     <li><code>fieldset</code> elements that have of one or more descendant elements that themselves
     are <span data-x="candidate for constraint validation">candidates for constraint
     validation</span> but do not <span data-x="concept-fv-valid">satisfy their
     constraints</span></li>
    </ul>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-user-valid">:user-valid</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-user-valid">:user-valid</code> <span>pseudo-class</span> must
    match <code>input</code>, <code>textarea</code>, and <code>select</code> elements whose
    <span>user validity</span> is true, are
    <span data-x="candidate for constraint validation">candidates for constraint validation</span>,
    and that <span data-x="concept-fv-valid">satisfy their constraints</span>.</p>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-user-invalid">:user-invalid</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-user-invalid">:user-invalid</code> <span>pseudo-class</span> must
    match <code>input</code>, <code>textarea</code>, and <code>select</code> elements whose
    <span>user validity</span> is true, are
    <span data-x="candidate for constraint validation">candidates for constraint validation</span>
    but do not <span data-x="concept-fv-valid">satisfy their constraints</span>.</p>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-in-range">:in-range</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-in-range">:in-range</code> <span>pseudo-class</span> must match
    all elements that are <span data-x="candidate for constraint validation">candidates for
    constraint validation</span>, <span>have range limitations</span>, and that are neither
    <span>suffering from an underflow</span> nor <span>suffering from an overflow</span>.</p>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-out-of-range">:out-of-range</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-out-of-range">:out-of-range</code> <span>pseudo-class</span> must
    match all elements that are <span data-x="candidate for constraint validation">candidates for
    constraint validation</span>, <span>have range limitations</span>, and that are either
    <span>suffering from an underflow</span> or <span>suffering from an overflow</span>.</p>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-required">:required</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-required">:required</code> <span>pseudo-class</span> must match
    any element falling into one of the following categories:</p>

    <ul>
     <li><code>input</code> elements that are <i data-x="concept-input-required">required</i></li>

     <li><code>select</code> elements that have a <code data-x="attr-select-required">required</code>
     attribute</li>

     <li><code>textarea</code> elements that have a <code
     data-x="attr-textarea-required">required</code> attribute</li>
    </ul>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-optional">:optional</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-optional">:optional</code> <span>pseudo-class</span> must match
    any element falling into one of the following categories:</p>

    <ul>
     <li><code>input</code> elements to which the <code data-x="attr-input-required">required</code>
     attribute applies that are not <i data-x="concept-input-required">required</i></li>

     <li><code>select</code> elements that do not have a <code
     data-x="attr-select-required">required</code> attribute</li>

     <li><code>textarea</code> elements that do not have a <code
     data-x="attr-textarea-required">required</code> attribute</li>
    </ul>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-autofill">:autofill</code></dfn></dt>
   <dt><dfn selector noexport><code
   data-x="selector-webkit-autofill">:-webkit-autofill</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-autofill">:autofill</code> and <code
    data-x="selector-webkit-autofill">:-webkit-autofill</code> <span
    data-x="pseudo-class">pseudo-classes</span> must match <code>input</code> elements which have
    been autofilled by user agent. These pseudo-classes must stop matching if the user edits the
    autofilled field.</p>

    <p class="note">One way such autofilling might happen is via the <code
    data-x="attr-fe-autocomplete">autocomplete</code> attribute, but user agents could autofill even
    without that attribute being involved.</p>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-read-only">:read-only</code></dfn></dt>
   <dt><dfn selector noexport><code data-x="selector-read-write">:read-write</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-read-write">:read-write</code> <span>pseudo-class</span> must
    match any element falling into one of the following categories, which for the purposes of
    Selectors are thus considered <i>user-alterable</i>: <ref>SELECTORS</ref></p>

    <ul>
     <li><code>input</code> elements to which the <code data-x="attr-input-readonly">readonly</code>
     attribute applies, and that are <i data-x="concept-fe-mutable">mutable</i> (i.e. that do not
     have the <code data-x="attr-input-readonly">readonly</code> attribute specified and that are not
     <span data-x="concept-fe-disabled">disabled</span>)</li>

     <li><code>textarea</code> elements that do not have a <code
     data-x="attr-textarea-readonly">readonly</code> attribute, and that are not <span
     data-x="concept-fe-disabled">disabled</span></li>

     <li>elements that are <span data-x="editing host">editing hosts</span> or <span>editable</span>
     and are neither <code>input</code> elements nor <code>textarea</code> elements</li>
    </ul>

    <p>The <code data-x="selector-read-only">:read-only</code> <span>pseudo-class</span> must match
    all other <span>HTML elements</span>.</p>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-modal">:modal</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-modal">:modal</code> <span>pseudo-class</span> must match any
    element falling into one of the following categories:</p>

    <ul>
     <li><code>dialog</code> elements whose <span>is modal</span> is true</li>

     <li>elements whose <span>fullscreen flag</span> is true</li>
    </ul>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-ltr">:dir(ltr)</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-ltr">:dir(ltr)</code> <span>pseudo-class</span> must match all
    elements whose <span data-x="the directionality">directionality</span> is '<span
    data-x="concept-ltr">ltr</span>'.</p>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-rtl">:dir(rtl)</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-rtl">:dir(rtl)</code> <span>pseudo-class</span> must match all
    elements whose <span data-x="the directionality">directionality</span> is '<span
    data-x="concept-rtl">rtl</span>'.</p>
   </dd>

   <dt><dfn selector noexport data-x="selector-custom">Custom state pseudo-class</dfn></dt>
   <dd>
    <p>The <code data-x="selector-custom">:state(<var>identifier</var>)</code> pseudo-class must
    match all <span>custom element</span>s whose <span>states set</span>'s <span>set entries</span>
    contains <var>identifier</var>.</p>
   </dd>

   <dt><dfn selector noexport><code data-x="selector-playing">:playing</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-playing">:playing</code> <span>pseudo-class</span> must match all
    <span data-x="media element">media elements</span> whose <code
    data-x="dom-media-paused">paused</code> attribute is false.</p>
    <!-- This should really use a non-public API. -->
   </dd>

   <dt><dfn selector noexport><code data-x="selector-paused">:paused</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-paused">:paused</code> <span>pseudo-class</span> must match all
    <span data-x="media element">media elements</span> whose <code
    data-x="dom-media-paused">paused</code> attribute is true.</p>
    <!-- This should really use a non-public API. -->
   </dd>

   <dt><dfn selector noexport><code data-x="selector-seeking">:seeking</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-seeking">:seeking</code> <span>pseudo-class</span> must match all
    <span data-x="media element">media elements</span> whose <code
    data-x="dom-media-seeking">seeking</code> attribute is true.</p>
    <!-- This should really use a non-public API. -->
   </dd>

   <dt><dfn selector noexport><code data-x="selector-buffering">:buffering</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-buffering">:buffering</code> <span>pseudo-class</span> must match
    all <span data-x="media element">media elements</span> whose <code
    data-x="dom-media-paused">paused</code> attribute is false, <code
    data-x="dom-media-networkState">networkState</code> attribute is <code
    data-x="dom-media-NETWORK_LOADING">NETWORK_LOADING</code>, and ready state is <code
    data-x="dom-media-HAVE_CURRENT_DATA">HAVE_CURRENT_DATA</code> or less.</p>
    <!-- This should really use a non-public API. -->
   </dd>

   <dt><dfn selector noexport><code data-x="selector-stalled">:stalled</code></dfn></dt>
   <dd><p>The <code data-x="selector-stalled">:stalled</code> <span>pseudo-class</span> must match
   all <span data-x="media element">media elements</span> that match the <code
   data-x="selector-buffering">:buffering</code> <span>pseudo-class</span> and whose <span>is
   currently stalled</span> is true.</p></dd>

   <dt><dfn selector noexport><code data-x="selector-muted">:muted</code></dfn></dt>
   <dd><p>The <code data-x="selector-muted">:muted</code> <span>pseudo-class</span> must match all
   <span data-x="media element">media elements</span> that are <span
   data-x="concept-media-muted">muted</span>.</p></dd>

   <dt><dfn selector noexport><code data-x="selector-volume-locked">:volume-locked</code></dfn></dt>
   <dd><p>The <code data-x="selector-volume-locked">:volume-locked</code> <span>pseudo-class</span>
   must match all <span data-x="media element">media elements</span> when the user agent's
   <span>volume locked</span> is true.</p></dd>

   <dt><dfn selector><code data-x="selector-open">:open</code></dfn></dt>
   <dd>
    <p>The <code data-x="selector-open">:open</code> <span>pseudo-class</span> must match any
    element falling into one of the following categories:</p>

    <ul>
     <li><p><code>details</code> elements that have an <code data-x="attr-details-open">open</code>
     attribute</p></li>

     <li><p><code>dialog</code> elements that have an <code data-x="attr-dialog-open">open</code>
     attribute</p></li>

     <li><p><code>select</code> elements that are a <span>drop-down box</span> and whose drop-down
     boxes are open</p></li>

     <li><p><code>input</code> elements that <span data-x="input-support-picker">support a
     picker</span> and whose pickers are open</p></li>
    </ul>
   </dd>
  </dl>

  <p class="note">This specification does not define when an element matches the <code undefined
  data-x="selector-lang()">:lang()</code> dynamic <span>pseudo-class</span>, as it is defined in
  sufficient detail in a language-agnostic fashion in <cite>Selectors</cite>.
  <ref>SELECTORS</ref></p>

  </div>



  <h2 split-filename="microdata" id="microdata"><dfn>Microdata</dfn></h2>

<!-- USE CASES

     Getting data out of poorly written web pages
     https://lists.whatwg.org/pipermail/whatwg-whatwg.org/2009-May/019595.html

     Allowing authors to keep track of where content originates
     https://lists.whatwg.org/pipermail/whatwg-whatwg.org/2009-May/019620.html

     Helping people searching for content filtered by license
     https://lists.whatwg.org/pipermail/whatwg-whatwg.org/2009-May/019668.html

     Allowing authors to annotate their documents to explain things for readers
     https://lists.whatwg.org/pipermail/whatwg-whatwg.org/2009-May/019672.html

     Annotating structured data that HTML has no semantics for
     https://lists.whatwg.org/pipermail/whatwg-whatwg.org/2009-May/019681.html

     Exposing known data types in a reusable way
     https://lists.whatwg.org/pipermail/whatwg-whatwg.org/2009-May/019833.html

     Providing enhanced search results
     https://lists.whatwg.org/pipermail/whatwg-whatwg.org/2009-May/019834.html

     Dragging or copying data between sites
     https://lists.whatwg.org/pipermail/whatwg-whatwg.org/2009-May/019835.html

     Writing authoring tools and validators for custom microdata vocabularies
     https://lists.whatwg.org/pipermail/whatwg-whatwg.org/2009-May/019839.html

     Removing the need for separate feeds
     https://lists.whatwg.org/pipermail/whatwg-whatwg.org/2009-May/019884.html

     Use cases for which I haven't been able to find solutions
     http://lists.whatwg.org/htdig.cgi/whatwg-whatwg.org/2009-May/019886.html

-->

<!-- v2

  * <itemref itemprop="foo" src="url#id"> to import the item with id="id" from url into the current
    microdata block as the value of property foo?

  * splitting the 'content' part of a property into multiple sub-bits, as in:

     <span itemprop="tel" item-content-in-bits>
       Telephone:
       <span content-bit>+44</span> (0) <span content-bit>1223 123 123</span>
     </span>

    maybe vocabs that need this can use a sub vocabulary specifically for this:

     <span itemprop="tel" itemscope itemtype="https://bits.example/">
       Telephone:
       <span itemprop=bit>+44</span> (0) <span itemprop=bit>1223 123 123</span>
     </span>

-->

  <h3>Introduction</h3>

  <h4>Overview</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>Sometimes, it is desirable to annotate content with specific machine-readable labels, e.g. to
  allow generic scripts to provide services that are customized to the page, or to enable content
  from a variety of cooperating authors to be processed by a single script in a consistent
  manner.</p>

  <p>For this purpose, authors can use the microdata features described in this section. Microdata
  allows nested groups of name-value pairs to be added to documents, in parallel with the existing
  content.</p>


  <h4>The basic syntax</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>At a high level, microdata consists of a group of name-value pairs. The groups are called <span
  data-x="concept-item">items</span>, and each name-value pair is a property. Items and properties
  are represented by regular elements.</p>

  <p>To create an item, the <code data-x="attr-itemscope">itemscope</code> attribute is used.</p>

  <p>To add a property to an item, the <code data-x="attr-itemprop">itemprop</code> attribute is used
  on one of the <span data-x="concept-item">item's</span> descendants.</p>

  <div class="example">

   <p>Here there are two items, each of which has the property "name":</p>

   <pre><code class="html">&lt;div itemscope>
 &lt;p>My name is &lt;span itemprop="name">Elizabeth&lt;/span>.&lt;/p>
&lt;/div>

&lt;div itemscope>
 &lt;p>My name is &lt;span itemprop="name">Daniel&lt;/span>.&lt;/p>
&lt;/div></code></pre>

  </div>

  <p>Markup without the microdata-related attributes does not have any effect on the microdata
  model.</p>

  <div class="example">

   <p>These two examples are exactly equivalent, at a microdata level, as the previous two examples
   respectively:</p>

   <pre><code class="html">&lt;div itemscope>
 &lt;p>My &lt;em>name&lt;/em> is &lt;span itemprop="name">E&lt;strong>liz&lt;/strong>abeth&lt;/span>.&lt;/p>
&lt;/div>

&lt;section>
 &lt;div itemscope>
  &lt;aside>
   &lt;p>My name is &lt;span itemprop="name">&lt;a href="/?user=daniel">Daniel&lt;/a>&lt;/span>.&lt;/p>
  &lt;/aside>
 &lt;/div>
&lt;/section></code></pre>

  </div>

  <p>Properties generally have values that are strings.</p>

  <div class="example">

   <p>Here the item has three properties:</p>

   <pre><code class="html">&lt;div itemscope>
 &lt;p>My name is &lt;span itemprop="name">Neil&lt;/span>.&lt;/p>
 &lt;p>My band is called &lt;span itemprop="band">Four Parts Water&lt;/span>.&lt;/p>
 &lt;p>I am &lt;span itemprop="nationality">British&lt;/span>.&lt;/p>
&lt;/div></code></pre>

  </div>

  <p>When a string value is a <span>URL</span>, it is expressed using the <code>a</code> element and
  its <code data-x="attr-hyperlink-href">href</code> attribute, the <code>img</code> element and its
  <code data-x="attr-img-src">src</code> attribute, or other elements that link to or embed external
  resources.</p>

  <div class="example">

   <p>In this example, the item has one property, "image", whose value is a URL:</p>

   <pre><code class="html">&lt;div itemscope>
 &lt;img itemprop="image" src="google-logo.png" alt="Google">
&lt;/div></code></pre>

  </div>

  <p>When a string value is in some machine-readable format unsuitable for human consumption, it is
  expressed using the <code data-x="attr-data-value">value</code> attribute of the <code>data</code>
  element, with the human-readable version given in the element's contents.</p>

  <div class="example">

   <p>Here, there is an item with a property whose value is a product ID. The ID is not
   human-friendly, so the product's name is used the human-visible text instead of the ID.</p>

   <pre><code class="html">&lt;h1 itemscope>
 &lt;data itemprop="product-id" value="9678AOU879">The Instigator 2000&lt;/data>
&lt;/h1></code></pre>

  </div>

  <p>For numeric data, the <code>meter</code> element and its <code
  data-x="attr-meter-value">value</code> attribute can be used instead.</p>

  <div class="example">

   <p>Here a rating is given using a <code>meter</code> element.</p>

   <pre><code class="html">&lt;div itemscope itemtype="http://schema.org/Product">
 &lt;span itemprop="name">Panasonic White 60L Refrigerator&lt;/span>
 &lt;img src="panasonic-fridge-60l-white.jpg" alt="">
  &lt;div itemprop="aggregateRating"
       itemscope itemtype="http://schema.org/AggregateRating">
   &lt;meter itemprop="ratingValue" min=0 value=3.5 max=5>Rated 3.5/5&lt;/meter>
   (based on &lt;span itemprop="reviewCount">11&lt;/span> customer reviews)
  &lt;/div>
&lt;/div></code></pre>

  </div>

  <p>Similarly, for date- and time-related data, the <code>time</code> element and its <code
  data-x="attr-time-datetime">datetime</code> attribute can be used instead.</p>

  <div class="example">

   <p>In this example, the item has one property, "birthday", whose value is a date:</p>

   <pre><code class="html">&lt;div itemscope>
 I was born on &lt;time itemprop="birthday" datetime="2009-05-10">May 10th 2009&lt;/time>.
&lt;/div></code></pre>

  </div>

  <p>Properties can also themselves be groups of name-value pairs, by putting the <code
  data-x="attr-itemscope">itemscope</code> attribute on the element that declares the property.</p>

  <p>Items that are not part of others are called <span>top-level microdata items</span>.</p>

  <div class="example">

   <p>In this example, the outer item represents a person, and the inner one represents a band:</p>

   <pre><code class="html">&lt;div itemscope>
 &lt;p>Name: &lt;span itemprop="name">Amanda&lt;/span>&lt;/p>
 &lt;p>Band: &lt;span itemprop="band" itemscope> &lt;span itemprop="name">Jazz Band&lt;/span> (&lt;span itemprop="size">12&lt;/span> players)&lt;/span>&lt;/p>
&lt;/div></code></pre>

   <p>The outer item here has two properties, "name" and "band". The "name" is "Amanda", and the
   "band" is an item in its own right, with two properties, "name" and "size". The "name" of the
   band is "Jazz Band", and the "size" is "12".</p>

   <p>The outer item in this example is a top-level microdata item.</p>

  </div>

  <p>Properties that are not descendants of the element with the <code
  data-x="attr-itemscope">itemscope</code> attribute can be associated with the <span
  data-x="concept-item">item</span> using the <code data-x="attr-itemref">itemref</code> attribute.
  This attribute takes a list of IDs of elements to crawl in addition to crawling the children of
  the element with the <code data-x="attr-itemscope">itemscope</code> attribute.</p>

  <div class="example">

   <p>This example is the same as the previous one, but all the properties are separated from their
   <span data-x="concept-item">items</span>:</p>

   <pre><code class="html">&lt;div itemscope id="amanda" itemref="a b">&lt;/div>
&lt;p id="a">Name: &lt;span itemprop="name">Amanda&lt;/span>&lt;/p>
&lt;div id="b" itemprop="band" itemscope itemref="c">&lt;/div>
&lt;div id="c">
 &lt;p>Band: &lt;span itemprop="name">Jazz Band&lt;/span>&lt;/p>
 &lt;p>Size: &lt;span itemprop="size">12&lt;/span> players&lt;/p>
&lt;/div></code></pre>

   <p>This gives the same result as the previous example. The first item has two properties, "name",
   set to "Amanda", and "band", set to another item. That second item has two further properties,
   "name", set to "Jazz Band", and "size", set to "12".</p>

  </div>

  <p>An <span data-x="concept-item">item</span> can have multiple properties with the same name and
  different values.</p>

  <div class="example">

   <p>This example describes an ice cream, with two flavors:</p>

   <pre><code class="html">&lt;div itemscope>
 &lt;p>Flavors in my favorite ice cream:&lt;/p>
 &lt;ul>
  &lt;li itemprop="flavor">Lemon sorbet&lt;/li>
  &lt;li itemprop="flavor">Apricot sorbet&lt;/li>
 &lt;/ul>
&lt;/div></code></pre>

   <p>This thus results in an item with two properties, both "flavor", having the values "Lemon
   sorbet" and "Apricot sorbet".</p>

  </div>

  <p>An element introducing a property can also introduce multiple properties at once, to avoid
  duplication when some of the properties have the same value.</p>

  <div class="example">

   <p>Here we see an item with two properties, "favorite-color" and "favorite-fruit", both set to
   the value "orange":</p>

   <pre><code class="html">&lt;div itemscope>
 &lt;span itemprop="favorite-color favorite-fruit">orange&lt;/span>
&lt;/div></code></pre>

  </div>

  <p>It's important to note that there is no relationship between the microdata and the content of
  the document where the microdata is marked up.</p>

  <div class="example">

   <p>There is no semantic difference, for instance, between the following two examples:</p>

   <pre><code class="html">&lt;figure>
 &lt;img src="castle.jpeg">
 &lt;figcaption>&lt;span itemscope>&lt;span itemprop="name">The Castle&lt;/span>&lt;/span> (1986)&lt;/figcaption>
&lt;/figure></code></pre>

   <pre><code class="html">&lt;span itemscope>&lt;meta itemprop="name" content="The Castle">&lt;/span>
&lt;figure>
 &lt;img src="castle.jpeg">
 &lt;figcaption>The Castle (1986)&lt;/figcaption>
&lt;/figure></code></pre>

   <p>Both have a figure with a caption, and both, completely unrelated to the figure, have an item
   with a name-value pair with the name "name" and the value "The Castle". The only difference is
   that if the user drags the caption out of the document, in the former case, the item will be
   included in the drag-and-drop data. In neither case is the image in any way associated with the
   item.</p>

  </div>


  <h4>Typed items</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>The examples in the previous section show how information could be marked up on a page that
  doesn't expect its microdata to be re-used. Microdata is most useful, though, when it is used in
  contexts where other authors and readers are able to cooperate to make new uses of the markup.</p>

  <p>For this purpose, it is necessary to give each <span data-x="concept-item">item</span> a type,
  such as "https://example.com/person", or "https://example.org/cat", or
  "https://band.example.net/". Types are identified as <span data-x="URL">URLs</span>.</p>

  <p>The type for an <span data-x="concept-item">item</span> is given as the value of an <code
  data-x="attr-itemtype">itemtype</code> attribute on the same element as the <code
  data-x="attr-itemscope">itemscope</code> attribute.</p>

  <div class="example">

   <p>Here, the item's type is "https://example.org/animals#cat":</p>

<pre><code class="html">&lt;section itemscope itemtype="https://example.org/animals#cat">
 &lt;h1 itemprop="name">Hedral&lt;/h1>
 &lt;p itemprop="desc">Hedral is a male american domestic
 shorthair, with a fluffy black fur with white paws and belly.&lt;/p>
 &lt;img itemprop="img" src="hedral.jpeg" alt="" title="Hedral, age 18 months">
&lt;/section></code></pre>

   <p>In this example the "https://example.org/animals#cat" item has three properties, a "name"
   ("Hedral"), a "desc" ("Hedral is..."), and an "img" ("hedral.jpeg").</p>

  </div>

  <p>The type gives the context for the properties, thus selecting a vocabulary: a property named
  "class" given for an item with the type "https://census.example/person" might refer to the economic
  class of an individual, while a property named "class" given for an item with the type
  "https://example.com/school/teacher" might refer to the classroom a teacher has been assigned.
  Several types can share a vocabulary. For example, the types "<code
  data-x="">https://example.org/people/teacher</code>" and "<code
  data-x="">https://example.org/people/engineer</code>" could be defined to use the same vocabulary
  (though maybe some properties would not be especially useful in both cases, e.g. maybe the "<code
  data-x="">https://example.org/people/engineer</code>" type might not typically be used with the
  "<code data-x="">classroom</code>" property). Multiple types defined to use the same vocabulary can
  be given for a single item by listing the URLs as a space-separated list in the attribute' value.
  An item cannot be given two types if they do not use the same vocabulary, however.</p>


  <h4>Global identifiers for items</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>Sometimes, an <span data-x="concept-item">item</span> gives information about a topic that has a
  global identifier. For example, books can be identified by their ISBN number.</p>

  <p>Vocabularies (as identified by the <code data-x="attr-itemtype">itemtype</code> attribute) can
  be designed such that <span data-x="concept-item">items</span> get associated with their global
  identifier in an unambiguous way by expressing the global identifiers as <span
  data-x="URL">URLs</span> given in an <code data-x="attr-itemid">itemid</code> attribute.</p>

  <p>The exact meaning of the <span data-x="URL">URLs</span> given in <code
  data-x="attr-itemid">itemid</code> attributes depends on the vocabulary used.</p>

  <div class="example">

   <p>Here, an item is talking about a particular book:</p>

<pre><code class="html">&lt;dl itemscope
    itemtype="https://vocab.example.net/book"
    <strong>itemid="urn:isbn:0-330-34032-8"</strong>>
 &lt;dt>Title
 &lt;dd itemprop="title">The Reality Dysfunction
 &lt;dt>Author
 &lt;dd itemprop="author">Peter F. Hamilton
 &lt;dt>Publication date
 &lt;dd>&lt;time itemprop="pubdate" datetime="1996-01-26">26 January 1996&lt;/time>
&lt;/dl></code></pre>

   <p>The "<code data-x="">https://vocab.example.net/book</code>" vocabulary in this example would
   define that the <code data-x="attr-itemid">itemid</code> attribute takes a <code data-x="urn
   protocol">urn:</code> <span>URL</span> pointing to the ISBN of the book.</p>

  </div>

  <!-- (commented out since itemtype="" defines the meaning of the
        identifier for now)
  <p>Although it is common practice, authors are encouraged not to use the same URL to identify
  multiple topics; in particular, an HTTP URL usually identifies a specific resource (such as an
  image or document), and authors are encouraged to not use them to identify people, non-Web
  resources like companies, movies, and products, or other abstract concepts. Using an HTTP URL as a
  global identifier for something other than the resource actually designated by the URL leads to a
  situation where it is ambiguous whether the identifier, and thus the set of properties specified
  in the microdata, references the page or something else.</p>
  -->



  <h4>Selecting names when defining vocabularies</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>Using microdata means using a vocabulary. For some purposes, an ad-hoc vocabulary is adequate.
  For others, a vocabulary will need to be designed. Where possible, authors are encouraged to
  re-use existing vocabularies, as this makes content re-use easier.</p>

  <p>When designing new vocabularies, identifiers can be created either using <span
  data-x="URL">URLs</span>, or, for properties, as plain words (with no dots or colons). For URLs,
  conflicts with other vocabularies can be avoided by only using identifiers that correspond to
  pages that the author has control over.</p>

  <div class="example">

   <p>For instance, if Jon and Adam both write content at <code
   data-x="">example.com</code>, at <code
   data-x="">https://example.com/~jon/...</code> and <code
   data-x="">https://example.com/~adam/...</code> respectively, then
   they could select identifiers of the form
   "https://example.com/~jon/name" and "https://example.com/~adam/name"
   respectively.</p>

  </div>

  <p>Properties whose names are just plain words can only be used within the context of the types
  for which they are intended; properties named using URLs can be reused in items of any type. If an
  item has no type, and is not part of another item, then if its properties have names that are just
  plain words, they are not intended to be globally unique, and are instead only intended for
  limited use. Generally speaking, authors are encouraged to use either properties with globally
  unique names (URLs) or ensure that their items are typed.</p>

  <div class="example">

   <p>Here, an item is an "https://example.org/animals#cat", and most of the properties have names
   that are words defined in the context of that type. There are also a few additional properties
   whose names come from other vocabularies.</p>

<pre><code class="html">&lt;section itemscope itemtype="https://example.org/animals#cat">
 &lt;h1 itemprop="name https://example.com/fn">Hedral&lt;/h1>
 &lt;p itemprop="desc">Hedral is a male American domestic
 shorthair, with a fluffy &lt;span
 itemprop="https://example.com/color">black&lt;/span> fur with &lt;span
 itemprop="https://example.com/color">white&lt;/span> paws and belly.&lt;/p>
 &lt;img itemprop="img" src="hedral.jpeg" alt="" title="Hedral, age 18 months">
&lt;/section></code></pre>

   <p>This example has one item with the type "https://example.org/animals#cat" and the following
   properties:</p>

   <table>
    <thead>
     <tr>
      <td>Property
      <td>Value
    <tbody>
     <tr>
      <td>name
      <td>Hedral
     <tr>
      <td>https://example.com/fn
      <td>Hedral
     <tr>
      <td>desc
      <td>Hedral is a male American domestic shorthair, with a fluffy black fur with white paws and belly.
     <tr>
      <td>https://example.com/color
      <td>black
     <tr>
      <td>https://example.com/color
      <td>white
     <tr>
      <td>img
      <td>.../hedral.jpeg
   </table>

  </div>



  <h3>Encoding microdata</h3>


  <h4>The microdata model</h4>

  <p>The microdata model consists of groups of name-value pairs known as <span
  data-x="concept-item">items</span>.</p>

  <p>Each group is known as an <span data-x="concept-item">item</span>. Each <span
  data-x="concept-item">item</span> can have <span>item types</span>, a <span>global
  identifier</span> (if the vocabulary specified by the <span>item types</span> <span>support global
  identifiers for items</span>), and a list of name-value pairs. Each name in the name-value pair is
  known as a <span data-x="the properties of an item">property</span>, and each <span data-x="the
  properties of an item">property</span> has one or more <span
  data-x="concept-property-value">values</span>. Each <span
  data-x="concept-property-value">value</span> is either a string or itself a group of name-value
  pairs (an <span data-x="concept-item">item</span>). The names are unordered relative to each other,
  but if a particular name has multiple values, they do have a relative order.</p>



  <h4>Items</h4>

  <p>Every <span data-x="HTML elements">HTML element</span> may have an <dfn element-attr
  for="html-global"><code data-x="attr-itemscope">itemscope</code></dfn> attribute specified. The
  <code data-x="attr-itemscope">itemscope</code> attribute is a <span>boolean attribute</span>.</p>

  <p>An element with the <code data-x="attr-itemscope">itemscope</code> attribute specified creates a
  new <dfn data-x="concept-item">item</dfn>, a group of name-value pairs.</p>

  <hr>

  <p>Elements with an <code data-x="attr-itemscope">itemscope</code> attribute may have an <dfn
  element-attr for="html-global"><code data-x="attr-itemtype">itemtype</code></dfn> attribute
  specified, to give the <span>item types</span> of the <span data-x="concept-item">item</span>.</p>

  <p>The <code data-x="attr-itemtype">itemtype</code> attribute, if specified, must have a value that
  is an <span>unordered set of unique space-separated tokens</span>, none of which are
  <span>identical to</span> another token and each of which is a <span>valid URL string</span> that
  is an <span>absolute URL</span>, and all of which are defined to use the same vocabulary. The
  attribute's value must have at least one token.</p>

  <p>The <dfn>item types</dfn> of an <span data-x="concept-item">item</span> are the tokens obtained
  by <span data-x="split a string on ASCII whitespace">splitting the element's <code
  data-x="attr-itemtype">itemtype</code> attribute's value on ASCII whitespace</span>. If the <code
  data-x="attr-itemtype">itemtype</code> attribute is missing or parsing it in this way finds no
  tokens, the <span data-x="concept-item">item</span> is said to have no <span>item types</span>.</p>

  <p>The <span>item types</span> must all be types defined in <span data-x="other applicable
  specifications">applicable specifications</span> and must all be defined to use the same
  vocabulary.</p>

  <div w-nodev>

   <p>Except if otherwise specified by that specification, the <span data-x="URL">URLs</span> given
   as the <span>item types</span> should not be automatically dereferenced.</p>

   <p class="note">A specification could define that its <span data-x="item types">item type</span>
   can be dereferenced to provide the user with help information, for example. In fact, vocabulary
   authors are encouraged to provide useful information at the given <span>URL</span>.</p>

   <p><span>Item types</span> are opaque identifiers, and user agents must not dereference unknown
   <span>item types</span>, or otherwise deconstruct them, in order to determine how to process
   <span data-x="concept-item">items</span> that use them.</p>

  </div>

  <p>The <code data-x="attr-itemtype">itemtype</code> attribute must not be specified on elements
  that do not have an <code data-x="attr-itemscope">itemscope</code> attribute specified.</p>

  <hr>

  <p>An <span data-x="concept-item">item</span> is said to be a <dfn>typed item</dfn> when either it
  has an <span data-x="item types">item type</span>, or it is the <span
  data-x="concept-property-value">value</span> of a <span data-x="the properties of an
  item">property</span> of a <span>typed item</span>. The <dfn>relevant types</dfn> for a
  <span>typed item</span> is the <span data-x="concept-item">item</span>'s <span>item types</span>,
  if it has any, or else is the <span>relevant types</span> of the <span
  data-x="concept-item">item</span> for which it is a <span data-x="the properties of an
  item">property</span>'s <span data-x="concept-property-value">value</span>.</p>

  <hr>

  <p>Elements with an <code data-x="attr-itemscope">itemscope</code> attribute and an <code
  data-x="attr-itemtype">itemtype</code> attribute that references a vocabulary that is defined to
  <dfn>support global identifiers for items</dfn> may also have an <dfn element-attr
  for="html-global"><code data-x="attr-itemid">itemid</code></dfn> attribute specified, to give a
  global identifier for the <span data-x="concept-item">item</span>, so that it can be related to
  other <span data-x="concept-item">items</span> on pages elsewhere on the web.</p>

  <p>The <code data-x="attr-itemid">itemid</code> attribute, if specified, must have a value that is
  a <span>valid URL potentially surrounded by spaces</span>.</p>

  <p>The <dfn>global identifier</dfn> of an <span data-x="concept-item">item</span> is the value of
  its element's <code data-x="attr-itemid">itemid</code> attribute, if it has one, <span
  data-x="encoding-parsing a URL">parsed</span> relative to the <span>node document</span> of the
  element on which the attribute is specified. If the <code data-x="attr-itemid">itemid</code>
  attribute is missing or if parsing it returns failure, it is said to have no <span>global
  identifier</span>.</p>

  <p>The <code data-x="attr-itemid">itemid</code> attribute must not be specified on elements that do
  not have both an <code data-x="attr-itemscope">itemscope</code> attribute and an <code
  data-x="attr-itemtype">itemtype</code> attribute specified, and must not be specified on elements
  with an <code data-x="attr-itemscope">itemscope</code> attribute whose <code
  data-x="attr-itemtype">itemtype</code> attribute specifies a vocabulary that does not <span>support
  global identifiers for items</span>, as defined by that vocabulary's specification.</p>

  <p>The exact meaning of a <span>global identifier</span> is determined by the vocabulary's
  specification. It is up to such specifications to define whether multiple items with the same
  global identifier (whether on the same page or on different pages) are allowed to exist, and what
  the processing rules for that vocabulary are with respect to handling the case of multiple items
  with the same ID.</p>

  <hr>

  <p>Elements with an <code data-x="attr-itemscope">itemscope</code> attribute may have an <dfn
  element-attr for="html-global"><code data-x="attr-itemref">itemref</code></dfn> attribute
  specified, to give a list of additional elements to crawl to find the name-value pairs of the
  <span data-x="concept-item">item</span>.</p>

  <p>The <code data-x="attr-itemref">itemref</code> attribute, if specified, must have a value that
  is an <span>unordered set of unique space-separated tokens</span> none of which are
  <span>identical to</span> another token and consisting of <span data-x="concept-ID">IDs</span> of
  elements in the same <span>tree</span>.</p>

  <p>The <code data-x="attr-itemref">itemref</code> attribute must not be specified on elements that
  do not have an <code data-x="attr-itemscope">itemscope</code> attribute specified.</p>

  <p class="note">The <code data-x="attr-itemref">itemref</code> attribute is not part of the
  microdata data model. It is merely a syntactic construct to aid authors in adding annotations to
  pages where the data to be annotated does not follow a convenient tree structure. For example, it
  allows authors to mark up data in a table so that each column defines a separate <span
  data-x="concept-item">item</span>, while keeping the properties in the cells.</p>

  <div class="example">

   <p>This example shows a simple vocabulary used to describe the products of a model railway
   manufacturer. The vocabulary has just five property names:</p>

   <dl>
    <dt>product-code</dt>
    <dd>An integer that names the product in the manufacturer's catalog.</dd>
    <dt>name</dt>
    <dd>A brief description of the product.</dd>
    <dt>scale</dt>
    <dd>One of "HO", "1", or "Z" (potentially with leading or trailing
    whitespace), indicating the scale of the product.</dd>
    <dt>digital</dt>
    <dd>If present, one of "Digital", "Delta", or "Systems"
    (potentially with leading or trailing whitespace) indicating that
    the product has a digital decoder of the given type.</dd>
    <dt>track-type</dt>
    <dd>For track-specific products, one of "K", "M", "C" (potentially
    with leading or trailing whitespace) indicating the type of track
    for which the product is intended.</dd>
   </dl>

   <p>This vocabulary has four defined <span>item types</span>:</p>

   <dl>
    <dt>https://md.example.com/loco</dt>
    <dd>Rolling stock with an engine</dd>
    <dt>https://md.example.com/passengers</dt>
    <dd>Passenger rolling stock</dd>
    <dt>https://md.example.com/track</dt>
    <dd>Track pieces</dd>
    <dt>https://md.example.com/lighting</dt>
    <dd>Equipment with lighting</dd>
   </dl>

   <p>Each <span data-x="concept-item">item</span> that uses this vocabulary can be given one or more
   of these types, depending on what the product is.</p>

   <p>Thus, a locomotive might be marked up as:</p>

<pre><code class="html">&lt;dl itemscope itemtype="https://md.example.com/loco
                        https://md.example.com/lighting">
 &lt;dt>Name:
 &lt;dd itemprop="name">Tank Locomotive (DB 80)
 &lt;dt>Product code:
 &lt;dd itemprop="product-code">33041
 &lt;dt>Scale:
 &lt;dd itemprop="scale">HO
 &lt;dt>Digital:
 &lt;dd itemprop="digital">Delta
&lt;/dl></code></pre>

   <p>A turnout lantern retrofit kit might be marked up as:</p>

   <pre><code class="html">&lt;dl itemscope itemtype="https://md.example.com/track
                        https://md.example.com/lighting">
 &lt;dt>Name:
 &lt;dd itemprop="name">Turnout Lantern Kit
 &lt;dt>Product code:
 &lt;dd itemprop="product-code">74470
 &lt;dt>Purpose:
 &lt;dd>For retrofitting 2 &lt;span itemprop="track-type">C&lt;/span> Track
 turnouts. &lt;meta itemprop="scale" content="HO">
&lt;/dl></code></pre>

   <p>A passenger car with no lighting might be marked up as:</p>

   <pre><code class="html">&lt;dl itemscope itemtype="https://md.example.com/passengers">
 &lt;dt>Name:
 &lt;dd itemprop="name">Express Train Passenger Car (DB Am 203)
 &lt;dt>Product code:
 &lt;dd itemprop="product-code">8710
 &lt;dt>Scale:
 &lt;dd itemprop="scale">Z
&lt;/dl></code></pre>

   <p>Great care is necessary when creating new vocabularies. Often, a hierarchical approach to
   types can be taken that results in a vocabulary where each item only ever has a single type,
   which is generally much simpler to manage.</p>

  </div>


  <h4>Names: the <dfn element-attr for="html-global"><code
  data-x="attr-itemprop">itemprop</code></dfn> attribute</h4>

  <p>Every <span data-x="HTML elements">HTML element</span> may have an <code
  data-x="attr-itemprop">itemprop</code> attribute specified, if doing so <span data-x="the properties
  of an item">adds one or more properties</span> to one or more <span
  data-x="concept-item">items</span> (as defined below).</p>

  <p>The <code data-x="attr-itemprop">itemprop</code> attribute, if specified, must have a value
  that is an <span>unordered set of unique space-separated tokens</span> none of which are
  <span>identical to</span> another token, representing the names of the name-value pairs that it
  adds. The attribute's value must have at least one token.</p>

  <p>Each token must be either:</p>

  <ul>
   <li>If the item is a <span>typed item</span>: a <dfn>defined property name</dfn> allowed in this
   situation according to the specification that defines the <span>relevant types</span> for the
   item, or</li>

   <li>A <span>valid URL string</span> that is an <span>absolute URL</span> defined as an item
   property name allowed in this situation by a vocabulary specification, or</li>

   <li>A <span>valid URL string</span> that is an <span>absolute URL</span>, used as a proprietary
   item property name (i.e. one used by the author for private purposes, not defined in a public
   specification), or</li>

   <li>If the item is not a <span>typed item</span>: a string that contains no U+002E FULL STOP
   characters (.) and no U+003A COLON characters (:), used as a proprietary item property name (i.e.
   one used by the author for private purposes, not defined in a public specification).</li>
  </ul>

  <p>Specifications that introduce <span data-x="defined property name">defined property names</span>
  must ensure all such property names contain no U+002E FULL STOP characters (.), no U+003A COLON
  characters (:), and no <span>ASCII whitespace</span>.</p>

  <p class="note">The rules above disallow U+003A COLON characters (:) in non-URL values because
  otherwise they could not be distinguished from URLs. Values with U+002E FULL STOP characters (.)
  are reserved for future extensions. <span>ASCII whitespace</span> are disallowed because otherwise
  the values would be parsed as multiple tokens.</p>

  <p>When an element with an <code data-x="attr-itemprop">itemprop</code> attribute <span data-x="the
  properties of an item">adds a property</span> to multiple <span data-x="concept-item">items</span>,
  the requirement above regarding the tokens applies for each <span data-x="concept-item">item</span>
  individually.</p>

  <!--

   e.g.:

     <div itemscope itemtype="https://example.com/a"> <ref refid="x"> </div>
     <div itemscope itemtype="https://example.com/b"> <ref refid="x"> </div>
     <meta id="x" itemprop="z" content="">

     "z" had better be valid for both https://example.com/a and https://example.com/b

  -->

  <p>The <dfn>property names</dfn> of an element are the tokens that the element's <code
  data-x="attr-itemprop">itemprop</code> attribute is found to contain when its value is <span
  data-x="split a string on ASCII whitespace">split on ASCII whitespace</span>, with the order
  preserved but with duplicates removed (leaving only the first occurrence of each name).</p>

  <p>Within an <span data-x="concept-item">item</span>, the properties are unordered with respect to
  each other, except for properties with the same name, which are ordered in the order they are
  given by the algorithm that defines <span>the properties of an item</span>.</p>

  <div class="example">

   <p>In the following example, the "a" property has the values "1" and "2", <em>in that order</em>,
   but whether the "a" property comes before the "b" property or not is not important:</p>

   <pre><code class="html">&lt;div itemscope>
 &lt;p itemprop="a">1&lt;/p>
 &lt;p itemprop="a">2&lt;/p>
 &lt;p itemprop="b">test&lt;/p>
&lt;/div></code></pre>

   <p>Thus, the following is equivalent:</p>

   <pre><code class="html">&lt;div itemscope>
 &lt;p itemprop="b">test&lt;/p>
 &lt;p itemprop="a">1&lt;/p>
 &lt;p itemprop="a">2&lt;/p>
&lt;/div></code></pre>

   <p>As is the following:</p>

   <pre><code class="html">&lt;div itemscope>
 &lt;p itemprop="a">1&lt;/p>
 &lt;p itemprop="b">test&lt;/p>
 &lt;p itemprop="a">2&lt;/p>
&lt;/div></code></pre>

   <p>And the following:</p>

   <pre><code class="html">&lt;div id="x">
 &lt;p itemprop="a">1&lt;/p>
&lt;/div>
&lt;div itemscope itemref="x">
 &lt;p itemprop="b">test&lt;/p>
 &lt;p itemprop="a">2&lt;/p>
&lt;/div></code></pre>

  </div>



  <h4>Values</h4>

  <p>The <dfn data-x="concept-property-value">property value</dfn> of a name-value pair added by an
  element with an <code data-x="attr-itemprop">itemprop</code> attribute is as given for the first
  matching case in the following list:</p>

  <dl class="switch">

   <dt>If the element also has an <code data-x="attr-itemscope">itemscope</code> attribute</dt>

   <dd>
    <p>The value is the <span data-x="concept-item">item</span> created by the element.</p>
   </dd>


   <dt>If the element is a <code>meta</code> element</dt>

   <dd>
    <p>The value is the value of the element's <code data-x="attr-meta-content">content</code>
    attribute, if any, or the empty string if there is no such attribute.</p>
   </dd>


   <dt>If the element is an <code>audio</code>, <code>embed</code>, <code>iframe</code>,
   <code>img</code>, <code>source</code>, <code>track</code>, or <code>video</code> element</dt>

   <dd>
    <p>The value is the result of <span>encoding-parsing-and-serializing a URL</span> given the
    element's <code data-x="">src</code> attribute's value, relative to the element's <span>node
    document</span>, at the time the attribute is set, or the empty string if there is no such
    attribute or the result is failure.</p>
   </dd>


   <dt>If the element is an <code>a</code>, <code>area</code>, or <code>link</code> element</dt>

   <dd>
    <p>The value is the result of <span>encoding-parsing-and-serializing a URL</span> given the
    element's <code data-x="">href</code> attribute's value, relative to the element's <span>node
    document</span>, at the time the attribute is set, or the empty string if there is no such
    attribute or the result is failure.</p>
   </dd>


   <dt>If the element is an <code>object</code> element</dt>

   <dd>
    <p>The value is the result of <span>encoding-parsing-and-serializing a URL</span> given the
    element's <code data-x="">data</code> attribute's value, relative to the element's <span>node
    document</span>, at the time the attribute is set, or the empty string if there is no such
    attribute or the result is failure.</p>
   </dd>


   <dt>If the element is a <code>data</code> element</dt>

   <dd>
    <p>The value is the value of the element's <code data-x="attr-data-value">value</code> attribute,
    if it has one, or the empty string otherwise.</p>
   </dd>


   <dt>If the element is a <code>meter</code> element</dt>

   <dd>
    <p>The value is the value of the element's <code data-x="attr-meter-value">value</code> attribute,
    if it has one, or the empty string otherwise.</p>
   </dd>


   <dt>If the element is a <code>time</code> element</dt>

   <dd>
    <p>The value is the element's <span>datetime value</span>.</p>
   </dd>


   <dt>Otherwise</dt>

   <dd>
    <p>The value is the element's <span>descendant text content</span>.</p>
   </dd>
  </dl>

  <p>The <dfn>URL property elements</dfn> are the <code>a</code>, <code>area</code>,
  <code>audio</code>, <code>embed</code>, <code>iframe</code>, <code>img</code>, <code>link</code>,
  <code>object</code>, <code>source</code>, <code>track</code>, and <code>video</code> elements.</p>

  <p>If a property's <span data-x="concept-property-value">value</span>, as defined by the property's
  definition, is an <span>absolute URL</span>, the property must be specified using a <span
  data-x="URL property elements">URL property element</span>.</p>

  <p class="note">These requirements do not apply just because a property value happens to match the
  syntax for a URL. They only apply if the property is explicitly defined as taking such a
  value.</p>

  <p class="example">For example, a book about the first moon landing <!-- 1969-07-20 --> could be
  called "mission:moon". A "title" property from a vocabulary that defines a title as being a string
  would not expect the title to be given in an <code>a</code> element, even though it looks like a
  <span>URL</span>. On the other hand, if there was a (rather narrowly scoped!) vocabulary for
  "books whose titles look like URLs" which had a "title" property defined to take a URL, then the
  property <em>would</em> expect the title to be given in an <code>a</code> element (or one of the
  other <span>URL property elements</span>), because of the requirement above.</p>


  <div w-nodev>

  <h4>Associating names with items</h4>

  <p>To find <dfn>the properties of an item</dfn> defined by the element <var>root</var>,
  the user agent must run the following steps. These steps are also used to flag <span
  data-x="microdata error">microdata errors</span>.</p>

  <ol>
   <li><p>Let <var>results</var>, <var>memory</var>, and <var>pending</var> be empty lists of elements.</p></li>

   <li><p>Add the element <var>root</var> to <var>memory</var>.</p></li>

   <li><p>Add the child elements of <var>root</var>, if any, to <var>pending</var>.</p></li>

   <li><p>If <var>root</var> has an <code data-x="attr-itemref">itemref</code> attribute, <span
   data-x="split a string on ASCII whitespace">split the value of that <code
   data-x="attr-itemref">itemref</code> attribute on ASCII whitespace</span>. For each resulting
   token <var>ID</var>, if there is an element in the <span>tree</span> of <var>root</var> with the
   <span data-x="concept-ID">ID</span> <var>ID</var>, then add the first such element to
   <var>pending</var>.</p></li>

   <li>
    <p>While <var>pending</var> is not empty:</p>

    <ol>
     <li><p>Remove an element from <var>pending</var> and let <var>current</var> be that
     element.</p></li>

     <li><p>If <var>current</var> is already in <var>memory</var>, there is a
     <span>microdata error</span>; <span>continue</span>.</p></li>

     <li><p>Add <var>current</var> to <var>memory</var>.</p></li>

     <li><p>If <var>current</var> does not have an <code data-x="attr-itemscope">itemscope</code>
     attribute, then: add all the child elements of <var>current</var> to
     <var>pending</var>.</p></li>

     <li><p>If <var>current</var> has an <code data-x="attr-itemprop">itemprop</code> attribute
     specified and has one or more <span>property names</span>, then add <var>current</var> to
     <var>results</var>.</p></li>
    </ol>

   <li><p>Sort <var>results</var> in <span>tree order</span>.</p></li>

   <li><p>Return <var>results</var>.</p></li>
  </ol>

  <p>A document must not contain any <span data-x="concept-item">items</span> for which the algorithm
  to find <span>the properties of an item</span> finds any <dfn data-x="microdata error">microdata
  errors</dfn>.</p>

  <p>An <span data-x="concept-item">item</span> is a <dfn data-x="top-level microdata items">top-level
  microdata item</dfn> if its element does not have an <code data-x="attr-itemprop">itemprop</code>
  attribute.</p>

<!--(this is currently redundant with "requirement itemprop" below)
  <p>An <span data-x="concept-item">item</span> is a <dfn data-x="used microdata items">used microdata
  item</dfn> if it is a <span data-x="top-level microdata items">top-level microdata item</span>, or
  if it has an <code data-x="attr-itemprop">itemprop</code> attribute and would be <span data-x="the
  properties of an item">found to be the property</span> of an <span
  data-x="concept-item">item</span> that is itself a <span data-x="used microdata items">used
  microdata item</span>.</p>

  <p>All <span data-x="concept-item">items</span> in a document must be <span>used microdata
  items</span>.</p>
-->

  <p>All <code data-x="attr-itemref">itemref</code> attributes in a <code>Document</code> must be
  such that there are no cycles in the graph formed from representing each <span
  data-x="concept-item">item</span> in the <code>Document</code> as a node in the graph and each
  <span data-x="the properties of an item">property</span> of an item whose <span
  data-x="concept-property-value">value</span> is another item as an edge in the graph connecting
  those two items.</p>

  <!-- "requirement itemprop" (see above) -->
  <p>A document must not contain any elements that have an <code
  data-x="attr-itemprop">itemprop</code> attribute that would not be found to be a property of any of
  the <span data-x="concept-item">items</span> in that document were their <span data-x="the
  properties of an item">properties</span> all to be determined.</p>

  <div class="example">

   <p>In this example, a single license statement is applied to two works, using <code
   data-x="attr-itemref">itemref</code> from the items representing the works:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang="en">
 &lt;head>
  &lt;title>Photo gallery&lt;/title>
 &lt;/head>
 &lt;body>
  &lt;h1>My photos&lt;/h1>
  &lt;figure itemscope itemtype="http://n.whatwg.org/work" itemref="licenses">
   &lt;img itemprop="work" src="images/house.jpeg" alt="A white house, boarded up, sits in a forest.">
   &lt;figcaption itemprop="title">The house I found.&lt;/figcaption>
  &lt;/figure>
  &lt;figure itemscope itemtype="http://n.whatwg.org/work" itemref="licenses">
   &lt;img itemprop="work" src="images/mailbox.jpeg" alt="Outside the house is a mailbox. It has a leaflet inside.">
   &lt;figcaption itemprop="title">The mailbox.&lt;/figcaption>
  &lt;/figure>
  &lt;footer>
   &lt;p id="licenses">All images licensed under the &lt;a itemprop="license"
   href="http://www.opensource.org/licenses/mit-license.php">MIT
   license&lt;/a>.&lt;/p>
  &lt;/footer>
 &lt;/body>
&lt;/html></code></pre>

   <p>The above results in two items with the type "<code data-x="">http://n.whatwg.org/work</code>",
   one with:</p>

   <dl class="brief">
    <dt>work
    <dd><code data-x="">images/house.jpeg</code>
    <dt>title
    <dd>The house I found.
    <dt>license
    <dd><code data-x="">http://www.opensource.org/licenses/mit-license.php</code>
   </dl>

   <p>...and one with:</p>

   <dl class="brief">
    <dt>work
    <dd><code data-x="">images/mailbox.jpeg</code>
    <dt>title
    <dd>The mailbox.
    <dt>license
    <dd><code data-x="">http://www.opensource.org/licenses/mit-license.php</code>
   </dl>

  </div>

  </div> <!-- see comment above -->


  <h4>Microdata and other namespaces</h4>

  <p>Currently, the <code data-x="attr-itemscope">itemscope</code>, <code
  data-x="attr-itemprop">itemprop</code>, and other microdata attributes are only defined for
  <span>HTML elements</span>. This means that attributes with the literal names "<code
  data-x="">itemscope</code>", "<code data-x="">itemprop</code>", etc, do not cause microdata
  processing to occur on elements in other namespaces, such as SVG.</p>

  <div class="example">

   <p>Thus, in the following example there is only one item, not
   two.</p>

   <pre class="bad"><code class="html">&lt;p itemscope>&lt;/p> &lt;!-- this is an item (with no properties and no type) -->
&lt;svg itemscope>&lt;/svg> &lt;!-- this is not, it's just an <span>SVG <code>svg</code></span> element with an invalid unknown attribute --></code></pre>

  </div>


  <h3 id="mdvocabs">Sample microdata vocabularies</h3>

  <p>The vocabularies in this section are primarily intended to demonstrate how a vocabulary is
  specified, though they are also usable in their own right.</p>


  <h4>vCard</h4>

  <p>An item with the <span data-x="item types">item type</span> <dfn><code data-x="md-vcard">http://microformats.org/profile/hcard</code></dfn> represents a person's or
  organization's contact information.</p>

  <p>This vocabulary does not <span>support global identifiers for items</span>.</p>

  <p>The following are the type's <span data-x="defined property name">defined property
  names</span>. They are based on the vocabulary defined in <cite>vCard Format Specification</cite>
  (<cite>vCard</cite>) and its extensions, where more information on how to interpret the values can
  be found. <ref>RFC6350</ref></p>

  <dl>
   <dt><dfn><code data-x="md-vcard-kind">kind</code></dfn></dt>

   <dd>
    <p>Describes what kind of contact the item represents.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text that is <span>identical
    to</span> one of the <span>kind strings</span>.</p>

    <p>A single property with the name <code data-x="md-vcard-kind">kind</code> may be present within
    each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <!-- XML not included since there's no good way to do it here -->
   <!-- Note: vCard->microdata->vCard doesn't count as "propagating vCards" according to Tantek, so
        we're not non-conforming when we drop it, despite vCard's requirement -->


   <dt><dfn><code data-x="md-vcard-fn">fn</code></dfn></dt>

   <dd>
    <p>Gives the formatted text corresponding to the name of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Exactly one property with the name <code data-x="md-vcard-fn">fn</code> must be present within
    each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-n">n</code></dfn></dt>

   <dd>
    <p>Gives the structured name of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be an <span
    data-x="concept-item">item</span> with zero or more of each of the <code
    data-x="md-vcard-n-family-name">family-name</code>, <code
    data-x="md-vcard-n-given-name">given-name</code>, <code
    data-x="md-vcard-n-additional-name">additional-name</code>, <code
    data-x="md-vcard-n-honorific-prefix">honorific-prefix</code>, and <code
    data-x="md-vcard-n-honorific-suffix">honorific-suffix</code> properties.</p>

    <p>Exactly one property with the name <code data-x="md-vcard-n">n</code> must be present within
    each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-n-family-name">family-name</code></dfn> (inside <code data-x="md-vcard-n">n</code>)</dt>

   <dd>
    <p>Gives the family name of the person, or the full name of the organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Any number of properties with the name <code
    data-x="md-vcard-n-family-name">family-name</code> may be present within the <span
    data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of the <code data-x="md-vcard-n">n</code> property of
    an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-n-given-name">given-name</code></dfn> (inside <code data-x="md-vcard-n">n</code>)</dt>

   <dd>
    <p>Gives the given-name of the person.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-n-given-name">given-name</code>
    may be present within the <span data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of the <code data-x="md-vcard-n">n</code> property of
    an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-n-additional-name">additional-name</code></dfn> (inside <code data-x="md-vcard-n">n</code>)</dt>

   <dd>
    <p>Gives the any additional names of the person.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Any number of properties with the name <code
    data-x="md-vcard-n-additional-name">additional-name</code> may be present within the <span
    data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of the <code data-x="md-vcard-n">n</code> property of
    an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-n-honorific-prefix">honorific-prefix</code></dfn> (inside <code data-x="md-vcard-n">n</code>)</dt>

   <dd>
    <p>Gives the honorific prefix of the person.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Any number of properties with the name <code
    data-x="md-vcard-n-honorific-prefix">honorific-prefix</code> may be present within the <span
    data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of the <code data-x="md-vcard-n">n</code> property of
    an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-n-honorific-suffix">honorific-suffix</code></dfn> (inside <code data-x="md-vcard-n">n</code>)</dt>

   <dd>
    <p>Gives the honorific suffix of the person.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Any number of properties with the name <code
    data-x="md-vcard-n-honorific-suffix">honorific-suffix</code> may be present within the <span
    data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of the <code data-x="md-vcard-n">n</code> property of
    an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-nickname">nickname</code></dfn></dt>

   <dd>
    <p>Gives the nickname of the person or organization.</p>

    <!-- copied from vCard -->
    <p class="note">The nickname is the descriptive name given instead of or in addition to the one
    belonging to a person, place, or thing. It can also be used to specify a familiar form of a
    proper name specified by the <code data-x="md-vcard-fn">fn</code> or <code
    data-x="md-vcard-n">n</code> properties.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-nickname">nickname</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-photo">photo</code></dfn></dt>

   <dd>
    <p>Gives a photograph of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be an <span>absolute
    URL</span>.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-photo">photo</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-bday">bday</code></dfn></dt>

   <dd>
    <p>Gives the birth date of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be a <span>valid date
    string</span>.</p>

    <p>A single property with the name <code data-x="md-vcard-bday">bday</code> may be present within
    each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-anniversary">anniversary</code></dfn></dt>

   <dd>
    <p>Gives the birth date of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be a <span>valid date
    string</span>.</p>

    <p>A single property with the name <code data-x="md-vcard-anniversary">anniversary</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-sex">sex</code></dfn></dt>

   <dd>
    <p>Gives the biological sex of the person.</p>

    <p>The <span data-x="concept-property-value">value</span> must be one of
    <code data-x="">F</code>, meaning "female",
    <code data-x="">M</code>, meaning "male",
    <code data-x="">N</code>, meaning "none or not applicable",
    <code data-x="">O</code>, meaning "other", or
    <code data-x="">U</code>, meaning "unknown".</p>

    <p>A single property with the name <code data-x="md-vcard-sex">sex</code> may be present within
    each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-gender-identity">gender-identity</code></dfn></dt>

   <dd>
    <p>Gives the gender identity of the person.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>A single property with the name <code data-x="md-vcard-gender-identity">gender-identity</code>
    may be present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-adr">adr</code></dfn></dt>

   <dd>
    <p>Gives the delivery address of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be an <span
    data-x="concept-item">item</span> with zero or more <code data-x="md-vcard-adr-type">type</code>,
    <code data-x="md-vcard-adr-post-office-box">post-office-box</code>, <code
    data-x="md-vcard-adr-extended-address">extended-address</code>, and <code
    data-x="md-vcard-adr-street-address">street-address</code> properties, and optionally a <code
    data-x="md-vcard-adr-locality">locality</code> property, optionally a <code
    data-x="md-vcard-adr-region">region</code> property, optionally a <code
    data-x="md-vcard-adr-postal-code">postal-code</code> property, and optionally a <code
    data-x="md-vcard-adr-country-name">country-name</code> property.</p>

    <p>If no <code data-x="md-vcard-adr-type">type</code> properties are present within an <span
    data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of an <code data-x="md-vcard-adr">adr</code> property
    of an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>, then the <span data-x="address
    type strings">address type string</span> <code data-x="md-vcard-type-adr-work">work</code> is
    implied.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-adr">adr</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-adr-type">type</code></dfn> (inside <code data-x="md-vcard-adr">adr</code>)</dt>

   <dd>
    <p>Gives the type of delivery address.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text that is <span>identical
    to</span> one of the <span>address type strings</span>.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-adr-type">type</code> may be
    present within the <span data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of an <code data-x="md-vcard-adr">adr</code> property
    of an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>, but within each such <code
    data-x="md-vcard-adr">adr</code> property <span data-x="concept-item">item</span> there must only
    be one <code data-x="md-vcard-adr-type">type</code> property per distinct value.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-adr-post-office-box">post-office-box</code></dfn> (inside <code data-x="md-vcard-adr">adr</code>)</dt>

   <dd>
    <p>Gives the post office box component of the delivery address of the person or
    organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Any number of properties with the name <code
    data-x="md-vcard-adr-post-office-box">post-office-box</code> may be present within the <span
    data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of an <code data-x="md-vcard-adr">adr</code> property
    of an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>

    <p class="note"><cite>vCard</cite> urges authors not to use this field.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-adr-extended-address">extended-address</code></dfn> (inside <code data-x="md-vcard-adr">adr</code>)</dt>

   <dd>
    <p>Gives an additional component of the delivery address of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Any number of properties with the name <code
    data-x="md-vcard-adr-extended-address">extended-address</code> may be present within the <span
    data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of an <code data-x="md-vcard-adr">adr</code> property
    of an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>

    <p class="note"><cite>vCard</cite> urges authors not to use this field.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-adr-street-address">street-address</code></dfn> (inside <code data-x="md-vcard-adr">adr</code>)</dt>

   <dd>
    <p>Gives the street address component of the delivery address of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Any number of properties with the name <code
    data-x="md-vcard-adr-street-address">street-address</code> may be present within the <span
    data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of an <code data-x="md-vcard-adr">adr</code> property
    of an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-adr-locality">locality</code></dfn> (inside <code data-x="md-vcard-adr">adr</code>)</dt>

   <dd>
    <p>Gives the locality component (e.g. city) of the delivery address of the person or
    organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>A single property with the name <code data-x="md-vcard-adr-locality">locality</code> may be
    present within the <span data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of an <code data-x="md-vcard-adr">adr</code> property
    of an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-adr-region">region</code></dfn> (inside <code data-x="md-vcard-adr">adr</code>)</dt>

   <dd>
    <p>Gives the region component (e.g. state or province) of the delivery address of the person or
    organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>A single property with the name <code data-x="md-vcard-adr-region">region</code> may be
    present within the <span data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of an <code data-x="md-vcard-adr">adr</code> property
    of an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-adr-postal-code">postal-code</code></dfn> (inside <code data-x="md-vcard-adr">adr</code>)</dt>

   <dd>
    <p>Gives the postal code component of the delivery address of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>A single property with the name <code data-x="md-vcard-adr-postal-code">postal-code</code> may
    be present within the <span data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of an <code data-x="md-vcard-adr">adr</code> property
    of an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-adr-country-name">country-name</code></dfn> (inside <code data-x="md-vcard-adr">adr</code>)</dt>

   <dd>
    <p>Gives the country name component of the delivery address of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>A single property with the name <code data-x="md-vcard-adr-country-name">country-name</code>
    may be present within the <span data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of an <code data-x="md-vcard-adr">adr</code> property
    of an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-tel">tel</code></dfn></dt>

   <dd>
    <p>Gives the telephone number of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be either text that can be
    interpreted as a telephone number as defined in the CCITT specifications E.163 and X.121, or an
    <span data-x="concept-item">item</span> with zero or more <code
    data-x="md-vcard-tel-type">type</code> properties and exactly one <code
    data-x="md-vcard-tel-value">value</code> property. <ref>E163</ref> <ref>X121</ref></p>

    <p>If no <code data-x="md-vcard-tel-type">type</code> properties are present within an <span
    data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of a <code data-x="md-vcard-tel">tel</code> property
    of an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>, or if the <span
    data-x="concept-property-value">value</span> of such a <code data-x="md-vcard-tel">tel</code>
    property is text, then the <span data-x="telephone type strings">telephone type string</span>
    <code data-x="md-vcard-type-tel-voice">voice</code> is implied.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-tel">tel</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-tel-type">type</code></dfn> (inside <code data-x="md-vcard-tel">tel</code>)</dt>

   <dd>
    <p>Gives the type of telephone number.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text that is <span>identical
    to</span> one of the <span>telephone type strings</span>.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-tel-type">type</code> may be
    present within the <span data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of a <code data-x="md-vcard-tel">tel</code> property
    of an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>, but within each such <code
    data-x="md-vcard-tel">tel</code> property <span data-x="concept-item">item</span> there must only
    be one <code data-x="md-vcard-tel-type">type</code> property per distinct value.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-tel-value">value</code></dfn> (inside <code data-x="md-vcard-tel">tel</code>)</dt>

   <dd>
    <p>Gives the actual telephone number of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text that can be interpreted
    as a telephone number as defined in the CCITT specifications E.163 and X.121. <ref>E163</ref>
    <ref>X121</ref></p>

    <p>Exactly one property with the name <code data-x="md-vcard-tel-value">value</code> must be
    present within the <span data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of a <code data-x="md-vcard-tel">tel</code> property
    of an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-email">email</code></dfn></dt>

   <dd>
    <p>Gives the email address of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-email">email</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-impp">impp</code></dfn></dt>

   <dd>
    <p>Gives a <span>URL</span> for instant messaging and presence protocol communications with the
    person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be an <span>absolute
    URL</span>.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-impp">impp</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-lang">lang</code></dfn></dt>

   <dd>
    <p>Gives a language understood by the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be a valid BCP 47 language tag.
    <ref>BCP47</ref></p>

    <p>Any number of properties with the name <code data-x="md-vcard-lang">lang</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-tz">tz</code></dfn></dt>

   <dd>
    <p>Gives the time zone of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text and must match the
    following syntax:</p>

    <ol>
     <li>Either a U+002B PLUS SIGN character (+) or a U+002D HYPHEN-MINUS character (-).</li>

     <li>A <span>valid non-negative integer</span> that is exactly two digits long and that
     represents a number in the range 00..23.</li>

     <li>A U+003A COLON character (:).</li>

     <li>A <span>valid non-negative integer</span> that is exactly two digits long and that
     represents a number in the range 00..59.</li>
    </ol>

    <p>Any number of properties with the name <code data-x="md-vcard-tz">tz</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-geo">geo</code></dfn></dt>

   <dd>
    <p>Gives the geographical position of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text and must match the
    following syntax:</p>

    <ol>
     <li>Optionally, either a U+002B PLUS SIGN character (+) or a U+002D HYPHEN-MINUS character
     (-).</li>

     <li>One or more <span>ASCII digits</span>.</li>

     <li>Optionally*, a U+002E FULL STOP character (.) followed by one or more <span>ASCII
     digits</span>.</li>

     <li>A U+003B SEMICOLON character (;).</li>

     <li>Optionally, either a U+002B PLUS SIGN character (+) or a U+002D HYPHEN-MINUS character
     (-).</li>

     <li>One or more <span>ASCII digits</span>.</li>

     <li>Optionally*, a U+002E FULL STOP character (.) followed by one or more <span>ASCII
     digits</span>.</li>
    </ol>

    <p>The optional components marked with an asterisk (*) should be included, and should have six
    digits each.</p>

    <!-- copied from vCard -->
    <p class="note">The value specifies latitude and longitude, in that order (i.e., "LAT LON"
    ordering), in decimal degrees. The longitude represents the location east and west of the prime
    meridian as a positive or negative real number, respectively. The latitude represents the
    location north and south of the equator as a positive or negative real number, respectively.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-geo">geo</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-title">title</code></dfn></dt>

   <dd>
    <p>Gives the job title, functional position or function of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-title">title</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-role">role</code></dfn></dt>

   <dd>
    <p>Gives the role, occupation, or business category of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-role">role</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-logo">logo</code></dfn></dt>

   <dd>
    <p>Gives the logo of the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be an <span>absolute
    URL</span>.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-logo">logo</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-agent">agent</code></dfn></dt>

   <dd>
    <p>Gives the contact information of another person who will act on behalf of the person or
    organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be either an <span
    data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>, or an <span>absolute URL</span>,
    or text.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-agent">agent</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-org">org</code></dfn></dt>

   <dd>
    <p>Gives the name and units of the organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be either text or an <span
    data-x="concept-item">item</span> with one <code
    data-x="md-vcard-org-organization-name">organization-name</code> property and zero or more <code
    data-x="md-vcard-org-organization-unit">organization-unit</code> properties.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-org">org</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-org-organization-name">organization-name</code></dfn> (inside <code data-x="md-vcard-org">org</code>)</dt>

   <dd>
    <p>Gives the name of the organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Exactly one property with the name <code
    data-x="md-vcard-org-organization-name">organization-name</code> must be present within the <span
    data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of an <code data-x="md-vcard-org">org</code> property
    of an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-org-organization-unit">organization-unit</code></dfn> (inside <code data-x="md-vcard-org">org</code>)</dt>

   <dd>
    <p>Gives the name of the organization unit.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Any number of properties with the name <code
    data-x="md-vcard-org-organization-unit">organization-unit</code> may be present within the <span
    data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of the <code data-x="md-vcard-org">org</code>
    property of an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-member">member</code></dfn></dt>

   <dd>
    <p>Gives a <span>URL</span> that represents a member of the group.</p>

    <p>The <span data-x="concept-property-value">value</span> must be an <span>absolute
    URL</span>.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-member">member</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code> if the <span
    data-x="concept-item">item</span> also has a property with the name <code
    data-x="md-vcard-kind">kind</code> whose value is "<code
    data-x="md-vcard-kind-group">group</code>".</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-related">related</code></dfn></dt>

   <dd>
    <p>Gives a relationship to another entity.</p>

    <p>The <span data-x="concept-property-value">value</span> must be an <span
    data-x="concept-item">item</span> with one <code data-x="md-vcard-related-url">url</code>
    property and one <code data-x="md-vcard-related-rel">rel</code> properties.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-related">related</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-related-url">url</code></dfn> (inside <code data-x="md-vcard-related">related</code>)</dt>

   <dd>
    <p>Gives the <span>URL</span> for the related entity.</p>

    <p>The <span data-x="concept-property-value">value</span> must be an <span>absolute
    URL</span>.</p>

    <p>Exactly one property with the name <code data-x="md-vcard-related-url">url</code> must be
    present within the <span data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of a <code data-x="md-vcard-related">related</code>
    property of an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-related-rel">rel</code></dfn> (inside <code data-x="md-vcard-related">related</code>)</dt>

   <dd>
    <p>Gives the relationship between the entity and the related entity.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text that is <span>identical
    to</span> one of the <span>relationship strings</span>.</p>

    <p>Exactly one property with the name <code data-x="md-vcard-related-rel">rel</code> must be
    present within the <span data-x="concept-item">item</span> that forms the <span
    data-x="concept-property-value">value</span> of a <code data-x="md-vcard-related">related</code>
    property of an <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-categories">categories</code></dfn></dt>

   <dd>
    <p>Gives the name of a category or tag that the person or organization could be classified
    as.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-categories">categories</code>
    may be present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-note">note</code></dfn></dt>

   <dd>
    <p>Gives supplemental information or a comment about the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-note">note</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <!-- PRODID not included -->


   <dt><dfn><code data-x="md-vcard-rev">rev</code></dfn></dt>

   <dd>
    <p>Gives the revision date and time of the contact information.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text that is a <span>valid
    global date and time string</span>.</p>

    <!-- copied from vCard -->
    <p class="note">The value distinguishes the current revision of the information for other
    renditions of the information.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-rev">rev</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-sound">sound</code></dfn></dt>

   <dd>
    <p>Gives a sound file relating to the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be an <span>absolute
    URL</span>.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-sound">sound</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vcard-uid">uid</code></dfn></dt>

   <dd>
    <p>Gives a globally unique identifier corresponding to the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>A single property with the name <code data-x="md-vcard-uid">uid</code> may be present within
    each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <!-- CLIENTPIDMAP not included since the global parameters are not included -->


   <dt><dfn><code data-x="md-vcard-url">url</code></dfn></dt>

   <dd>
    <p>Gives a <span>URL</span> relating to the person or organization.</p>

    <p>The <span data-x="concept-property-value">value</span> must be an <span>absolute
    URL</span>.</p>

    <p>Any number of properties with the name <code data-x="md-vcard-url">url</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p>
   </dd>


   <!-- VERSION not included - always 4.0 -->

   <!-- KEY not included - no way to include binary data at this time -->

   <!-- FBURL, CALADRURI, CALURI not included - unclear how calendars would be exposed on the web -->
  </dl>

  <p>The <dfn>kind strings</dfn> are:</p>

  <dl>
   <dt><dfn><code data-x="md-vcard-kind-individual">individual</code></dfn></dt>
   <dd>
    <p>Indicates a single entity (e.g. a person).</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-kind-group">group</code></dfn></dt>
   <dd>
    <p>Indicates multiple entities (e.g. a mailing list).</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-kind-org">org</code></dfn></dt>
   <dd>
    <p>Indicates a single entity that is not a person (e.g. a company).</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-kind-location">location</code></dfn></dt>
   <dd>
    <p>Indicates a geographical place (e.g. an office building).</p>
   </dd>

   <!-- update this when https://www.iana.org/assignments/vcard-elements/vcard-elements.xml#property-values is updated -->
  </dl>

  <p>The <dfn>address type strings</dfn> are:</p>

  <dl>
   <dt><dfn><code data-x="md-vcard-type-adr-home">home</code></dfn></dt>
   <dd>
    <p>Indicates a delivery address for a residence.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-type-adr-work">work</code></dfn></dt>
   <dd>
    <p>Indicates a delivery address for a place of work.</p>
   </dd>
  </dl>

  <p>The <dfn>telephone type strings</dfn> are:</p>

  <dl>
   <dt><dfn><code data-x="md-vcard-type-tel-home">home</code></dfn></dt>
   <dd>
    <p>Indicates a residential number.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-type-tel-work">work</code></dfn></dt>
   <dd>
    <p>Indicates a telephone number for a place of work.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-type-tel-text">text</code></dfn></dt>
   <dd>
    <p>Indicates that the telephone number supports text messages (SMS).</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-type-tel-voice">voice</code></dfn></dt>
   <dd>
    <p>Indicates a voice telephone number.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-type-tel-fax">fax</code></dfn></dt>
   <dd>
    <p>Indicates a facsimile telephone number.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-type-tel-cell">cell</code></dfn></dt>
   <dd>
    <p>Indicates a cellular telephone number.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-type-tel-video">video</code></dfn></dt>
   <dd>
    <p>Indicates a video conferencing telephone number.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-type-tel-pager">pager</code></dfn></dt>
   <dd>
    <p>Indicates a paging device telephone number.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-type-tel-textphone">textphone</code></dfn></dt>
   <dd>
    <p>Indicates a telecommunication device for people with hearing or speech difficulties.</p>
   </dd>
  </dl>

  <p>The <dfn>relationship strings</dfn> are:</p>

  <dl>
   <dt><dfn><code data-x="md-vcard-rel-emergency">emergency</code></dfn></dt>
   <dd>
    <p>An emergency contact.</p>
   </dd>

   <dt><dfn><code data-x="md-vcard-rel-agent">agent</code></dfn></dt>
   <dd>
    <p>Another entity that acts on behalf of this entity.</p>
   </dd>

   <dt><dfn data-x="md-vcard-rel-contact">contact</dfn></dt>
   <dt><dfn data-x="md-vcard-rel-acquaintance">acquaintance</dfn></dt>
   <dt><dfn data-x="md-vcard-rel-friend">friend</dfn></dt>
   <dt><dfn data-x="md-vcard-rel-met">met</dfn></dt>
   <dt><dfn data-x="md-vcard-rel-co-worker">worker</dfn></dt>
   <dt><dfn data-x="md-vcard-rel-colleague">colleague</dfn></dt>
   <dt><dfn data-x="md-vcard-rel-co-resident">resident</dfn></dt>
   <dt><dfn data-x="md-vcard-rel-neighbor">neighbor</dfn></dt>
   <dt><dfn data-x="md-vcard-rel-child">child</dfn></dt>
   <dt><dfn data-x="md-vcard-rel-parent">parent</dfn></dt>
   <dt><dfn data-x="md-vcard-rel-sibling">sibling</dfn></dt>
   <dt><dfn data-x="md-vcard-rel-spouse">spouse</dfn></dt>
   <dt><dfn data-x="md-vcard-rel-kin">kin</dfn></dt>
   <dt><dfn data-x="md-vcard-rel-muse">muse</dfn></dt>
   <dt><dfn data-x="md-vcard-rel-crush">crush</dfn></dt>
   <dt><dfn data-x="md-vcard-rel-date">date</dfn></dt>
   <dt><dfn data-x="md-vcard-rel-sweetheart">sweetheart</dfn></dt>
   <dt><dfn data-x="md-vcard-rel-me">me</dfn></dt>
   <dd>
    <p>Has the meaning defined in XFN. <ref>XFN</ref></p>
   </dd>

   <!-- update this when https://www.iana.org/assignments/vcard-elements/vcard-elements.xml#parameter-values is updated -->
  </dl>


  <h5>Conversion to vCard</h5>

  <p>Given a list of nodes <var>nodes</var> in a <code>Document</code>, a user agent must
  run the following algorithm to <dfn data-x="extracting a vCard">extract any vCard data represented
  by those nodes</dfn> (only the first vCard is returned):</p>

  <ol>
   <li><p>If none of the nodes in <var>nodes</var> are <span
   data-x="concept-item">items</span> with the <span data-x="item types">item type</span> <code
   data-x="md-vcard">http://microformats.org/profile/hcard</code>, then there is no vCard. Abort the
   algorithm, returning nothing.</p></li>

   <li><p>Let <var>node</var> be the first node in <var>nodes</var> that is an
   <span data-x="concept-item">item</span> with the <span data-x="item types">item type</span> <code
   data-x="md-vcard">http://microformats.org/profile/hcard</code>.</p></li>

   <li><p>Let <var>output</var> be an empty string.</p></li>

   <li><p><span>Add a vCard line</span> with the type "<code data-x="">BEGIN</code>" and the value
   "<code data-x="">VCARD</code>" to <var>output</var>.</p></li>

   <li><p><span>Add a vCard line</span> with the type "<code data-x="">PROFILE</code>" and the value
   "<code data-x="">VCARD</code>" to <var>output</var>.</p></li>

   <li><p><span>Add a vCard line</span> with the type "<code data-x="">VERSION</code>" and the value
   "<code data-x="">4.0</code>" to <var>output</var>.</p></li>

   <li><p><span>Add a vCard line</span> with the type "<code data-x="">SOURCE</code>" and the result
   of <span>escaping the vCard text string</span> that is the document's <span
   data-x="concept-document-url">URL</span> as the value to <var>output</var>.</p></li>

   <li><p>If <span>the <code>title</code> element</span> is not null, <span>add a vCard line</span>
   with the type "<code data-x="">NAME</code>" and with the result of <span>escaping the vCard text
   string</span> obtained from <span>the <code>title</code> element</span>'s <span>descendant
   text content</span> as the value to <var>output</var>.</p></li>

   <li><p>Let <var>sex</var> be the empty string.</p></li>

   <li><p>Let <var>gender-identity</var> be the empty string.</p></li>

   <li>
    <p>For each element <var>element</var> that is <span data-x="the properties of an
    item">a property of the item</span> <var>node</var>: for each name <var>name</var> in <var>element</var>'s <span>property names</span>, run the
    following substeps:</p>

    <ol>
     <li><p>Let <var>parameters</var> be an empty set of name-value pairs.</p></li>

     <li>
      <p>Run the appropriate set of substeps from the following list. The steps will set a variable
      <var>value</var>, which is used in the next step.</p>

      <dl>
       <dt>If the property's <span data-x="concept-property-value">value</span> is an <span
       data-x="concept-item">item</span> <var>subitem</var> and <var>name</var> is
       <code data-x="md-vcard-n">n</code></dt>

       <dd>
        <ol>
         <li><p>Let <var>value</var> be the empty string.</p></li>

         <li><p>Append to <var>value</var> the result of <span>collecting the first vCard
         subproperty</span> named <code data-x="md-vcard-n-family-name">family-name</code> in <var>subitem</var>.</p>

         <li>Append a U+003B SEMICOLON character (;) to <var>value</var>.</li>

         <li><p>Append to <var>value</var> the result of <span>collecting the first vCard
         subproperty</span> named <code data-x="md-vcard-n-given-name">given-name</code> in <var>subitem</var>.</p>

         <li>Append a U+003B SEMICOLON character (;) to <var>value</var>.</li>

         <li><p>Append to <var>value</var> the result of <span>collecting the first vCard
         subproperty</span> named <code data-x="md-vcard-n-additional-name">additional-name</code> in
         <var>subitem</var>.</p>

         <li>Append a U+003B SEMICOLON character (;) to <var>value</var>.</li>

         <li><p>Append to <var>value</var> the result of <span>collecting the first vCard
         subproperty</span> named <code data-x="md-vcard-n-honorific-prefix">honorific-prefix</code>
         in <var>subitem</var>.</p>

         <li>Append a U+003B SEMICOLON character (;) to <var>value</var>.</li>

         <li><p>Append to <var>value</var> the result of <span>collecting the first vCard
         subproperty</span> named <code data-x="md-vcard-n-honorific-suffix">honorific-suffix</code>
         in <var>subitem</var>.</p>
        </ol>
       </dd>

       <dt>If the property's <span data-x="concept-property-value">value</span> is an <span
       data-x="concept-item">item</span> <var>subitem</var> and <var>name</var> is
       <code data-x="md-vcard-adr">adr</code></dt>

       <dd>
        <ol>
         <li><p>Let <var>value</var> be the empty string.</p></li>

         <li><p>Append to <var>value</var> the result of <span>collecting vCard
         subproperties</span> named <code
         data-x="md-vcard-adr-post-office-box">post-office-box</code> in <var>subitem</var>.</p>

         <li>Append a U+003B SEMICOLON character (;) to <var>value</var>.</li>

         <li><p>Append to <var>value</var> the result of <span>collecting vCard
         subproperties</span> named <code
         data-x="md-vcard-adr-extended-address">extended-address</code> in <var>subitem</var>.</p>

         <li>Append a U+003B SEMICOLON character (;) to <var>value</var>.</li>

         <li><p>Append to <var>value</var> the result of <span>collecting vCard
         subproperties</span> named <code data-x="md-vcard-adr-street-address">street-address</code>
         in <var>subitem</var>.</p>

         <li>Append a U+003B SEMICOLON character (;) to <var>value</var>.</li>

         <li><p>Append to <var>value</var> the result of <span>collecting the first vCard
         subproperty</span> named <code data-x="md-vcard-adr-locality">locality</code> in <var>subitem</var>.</p>

         <li>Append a U+003B SEMICOLON character (;) to <var>value</var>.</li>

         <li><p>Append to <var>value</var> the result of <span>collecting the first vCard
         subproperty</span> named <code data-x="md-vcard-adr-region">region</code> in <var>subitem</var>.</p>

         <li>Append a U+003B SEMICOLON character (;) to <var>value</var>.</li>

         <li><p>Append to <var>value</var> the result of <span>collecting the first vCard
         subproperty</span> named <code data-x="md-vcard-adr-postal-code">postal-code</code> in <var>subitem</var>.</p>

         <li>Append a U+003B SEMICOLON character (;) to <var>value</var>.</li>

         <li><p>Append to <var>value</var> the result of <span>collecting the first vCard
         subproperty</span> named <code data-x="md-vcard-adr-country-name">country-name</code> in
         <var>subitem</var>.</p>

         <li><p>If there is a property named <code data-x="md-vcard-adr-type">type</code> in
         <var>subitem</var>, and the first such property has a <span
         data-x="concept-property-value">value</span> that is not an <span
         data-x="concept-item">item</span> and whose value consists only of <span data-x="ASCII
         alphanumeric">ASCII alphanumerics</span>, then add a parameter named "<code
         data-x="">TYPE</code>" whose value is the <span
         data-x="concept-property-value">value</span> of that property to
         <var>parameters</var>.</p></li>
        </ol>
       </dd>

       <dt>If the property's <span data-x="concept-property-value">value</span> is an <span
       data-x="concept-item">item</span> <var>subitem</var> and <var>name</var> is
       <code data-x="md-vcard-org">org</code></dt>

       <dd>
        <ol>
         <li><p>Let <var>value</var> be the empty string.</p></li>

         <li><p>Append to <var>value</var> the result of <span>collecting the first vCard
         subproperty</span> named <code
         data-x="md-vcard-org-organization-name">organization-name</code> in <var>subitem</var>.</p>

         <li>
          <p>For each property named <code
          data-x="md-vcard-org-organization-unit">organization-unit</code> in <var>subitem</var>, run the following steps:</p>

          <ol>
           <li><p>If the <span data-x="concept-property-value">value</span> of the property is an
           <span data-x="concept-item">item</span>, then skip this property.</p></li>

           <li><p>Append a U+003B SEMICOLON character (;) to <var>value</var>.</p></li>

           <li><p>Append the result of <span>escaping the vCard text string</span> given by the
           <span data-x="concept-property-value">value</span> of the property to <var>value</var>.</p></li>
          </ol>
         </li>
        </ol>
       </dd>

       <dt>If the property's <span data-x="concept-property-value">value</span> is an <span
       data-x="concept-item">item</span> <var>subitem</var> with the <span data-x="item
       types">item type</span> <code data-x="md-vcard">http://microformats.org/profile/hcard</code>
       and <var>name</var> is <code data-x="md-vcard-related">related</code></dt>

       <dd>
        <ol>
         <li><p>Let <var>value</var> be the empty string.</p></li>

         <li><p>If there is a property named <code data-x="md-vcard-related-url">url</code> in
         <var>subitem</var>, and its element is a <span data-x="URL property elements">URL property
         element</span>, then append the result of <span>escaping the vCard text string</span> given
         by the <span data-x="concept-property-value">value</span> of the first such property to
         <var>value</var>, and add a parameter with the name "<code data-x="">VALUE</code>" and the
         value "<code data-x="">URI</code>" to <var>parameters</var>.</p></li>

         <li><p>If there is a property named <code data-x="md-vcard-related-rel">rel</code> in
         <var>subitem</var>, and the first such property has a <span
         data-x="concept-property-value">value</span> that is not an <span
         data-x="concept-item">item</span> and whose value consists only of <span data-x="ASCII
         alphanumeric">ASCII alphanumerics</span>, then add a parameter named "<code
         data-x="">RELATION</code>" whose value is the <span
         data-x="concept-property-value">value</span> of that property to
         <var>parameters</var>.</p></li>
        </ol>
       </dd>

       <dt>If the property's <span data-x="concept-property-value">value</span> is an <span
       data-x="concept-item">item</span> and <var>name</var> is none of the above</dt>

       <!-- tel, adr -->

       <dd>
        <ol>
         <li><p>Let <var>value</var> be the result of <span>collecting the first vCard
         subproperty</span> named <code data-x="">value</code> in <var>subitem</var>.</p>

         <li><p>If there is a property named <code data-x="">type</code> in <var>subitem</var>, and
         the first such property has a <span data-x="concept-property-value">value</span> that is
         not an <span data-x="concept-item">item</span> and whose value consists only of <span
         data-x="ASCII alphanumeric">ASCII alphanumerics</span>, then add a parameter named "<code
         data-x="">TYPE</code>" whose value is the <span
         data-x="concept-property-value">value</span> of that property to
         <var>parameters</var>.</p></li>
        </ol>
       </dd>


       <dt>If the property's <span data-x="concept-property-value">value</span> is not an <span
       data-x="concept-item">item</span> and its <var>name</var> is <code
       data-x="md-vcard-sex">sex</code></dt>

       <dd>
        <p>If this is the first such property to be found, set <var>sex</var> to the
        property's <span data-x="concept-property-value">value</span>.</p>
       </dd>


       <dt>If the property's <span data-x="concept-property-value">value</span> is not an <span
       data-x="concept-item">item</span> and its <var>name</var> is <code
       data-x="md-vcard-gender-identity">gender-identity</code></dt>

       <dd>
        <p>If this is the first such property to be found, set <var>gender-identity</var>
        to the property's <span data-x="concept-property-value">value</span>.</p>
       </dd>


       <dt>Otherwise (the property's <span data-x="concept-property-value">value</span> is not an
       <span data-x="concept-item">item</span>)</dt>

       <dd>
        <ol>
         <li><p>Let <var>value</var> be the property's <span
         data-x="concept-property-value">value</span>.</p></li>

         <li><p>If <var>element</var> is one of the <span>URL property elements</span>, add
         a parameter with the name "<code data-x="">VALUE</code>" and the value "<code
         data-x="">URI</code>" to <var>parameters</var>.</p></li>

         <li><p>Otherwise, if <var>name</var> is <code data-x="md-vcard-bday">bday</code> or
         <code data-x="md-vcard-anniversary">anniversary</code> and the <var>value</var> is
         a <span>valid date string</span>, add a parameter with the name "<code
         data-x="">VALUE</code>" and the value "<code data-x="">DATE</code>" to <var>parameters</var>.</p></li>

         <li><p>Otherwise, if <var>name</var> is <code data-x="md-vcard-rev">rev</code> and
         the <var>value</var> is a <span>valid global date and time string</span>, add a
         parameter with the name "<code data-x="">VALUE</code>" and the value "<code
         data-x="">DATE-TIME</code>" to <var>parameters</var>.</p></li>

         <li><p>Prefix every U+005C REVERSE SOLIDUS character (\) in <var>value</var> with
         another U+005C REVERSE SOLIDUS character (\).</p></li>

         <li><p>Prefix every U+002C COMMA character (,) in <var>value</var> with a U+005C
         REVERSE SOLIDUS character (\).</p></li>

         <li><p>Unless <var>name</var> is <code data-x="md-vcard-geo">geo</code>, prefix
         every U+003B SEMICOLON character (;) in <var>value</var> with a U+005C REVERSE
         SOLIDUS character (\).</p></li>

         <li><p>Replace every U+000D CARRIAGE RETURN U+000A LINE FEED character pair (CRLF) in <var>value</var> with a U+005C REVERSE SOLIDUS character (\) followed by a U+006E LATIN
         SMALL LETTER N character (n).</p></li>

         <li><p>Replace every remaining U+000D CARRIAGE RETURN (CR) or U+000A LINE FEED (LF)
         character in <var>value</var> with a U+005C REVERSE SOLIDUS character (\) followed
         by a U+006E LATIN SMALL LETTER N character (n).</p></li>
        </ol>
       </dd>
      </dl>
     </li>

     <li>
      <p><span>Add a vCard line</span> with the type <var>name</var>, the parameters <var>parameters</var>, and the value <var>value</var> to <var>output</var>.</p>
     </li>
    </ol>
   </li>

   <li><p>If either <var>sex</var> or <var>gender-identity</var> has a value that
   is not the empty string, <span>add a vCard line</span> with the type "<code
   data-x="">GENDER</code>" and the value consisting of the concatenation of <var>sex</var>,
   a U+003B SEMICOLON character (;), and <var>gender-identity</var> to <var>output</var>.</p></li>

   <li><p><span>Add a vCard line</span> with the type "<code data-x="">END</code>" and the value
   "<code data-x="">VCARD</code>" to <var>output</var>.</p></li>
  </ol>

  <p>When the above algorithm says that the user agent is to <dfn>add a vCard line</dfn> consisting
  of a type <var>type</var>, optionally some parameters, and a value <var>value</var> to a string <var>output</var>, it must run the following steps:</p>

  <ol>
   <li><p>Let <var>line</var> be an empty string.

   <li><p>Append <var>type</var>, <span>converted to ASCII uppercase</span>, to <var>line</var>.</p></li>

   <li>
    <p>If there are any parameters, then for each parameter, in the order that they were added, run
    these substeps:</p>

    <ol>
     <li><p>Append a U+003B SEMICOLON character (;) to <var>line</var>.</p></li>

     <li><p>Append the parameter's name to <var>line</var>.</p></li>

     <li><p>Append a U+003D EQUALS SIGN character (=) to <var>line</var>.</p></li>

     <li><p>Append the parameter's value to <var>line</var>.</p></li>
    </ol>
   </li>

   <li><p>Append a U+003A COLON character (:) to <var>line</var>.</p></li>

   <li><p>Append <var>value</var> to <var>line</var>.</p></li>

   <li><p>Let <var>maximum length</var> be 75.</p></li>

   <li>
    <p>While <var>line</var>'s <span>code point length</span> is greater than <var>maximum
    length</var>:</p>

    <ol>
     <li><p>Append the first <var>maximum length</var> code points of <var>line</var> to
     <var>output</var>.</p></li>

     <li><p>Remove the first <var>maximum length</var> code points from <var>line</var>.</p></li>

     <li><p>Append a U+000D CARRIAGE RETURN character (CR) to <var>output</var>.</p></li>

     <li><p>Append a U+000A LINE FEED character (LF) to <var>output</var>.</p></li>

     <li><p>Append a U+0020 SPACE character to <var>output</var>.</p></li>

     <li><p>Let <var>maximum length</var> be 74.</p></li>
    </ol>
   </li>

   <li><p>Append (what remains of) <var>line</var> to <var>output</var>.</p></li>

   <li><p>Append a U+000D CARRIAGE RETURN character (CR) to <var>output</var>.</p></li>

   <li><p>Append a U+000A LINE FEED character (LF) to <var>output</var>.</p></li>
  </ol>

  <p>When the steps above require the user agent to obtain the result of <dfn>collecting vCard
  subproperties</dfn> named <var>subname</var> in <var>subitem</var>, the user
  agent must run the following steps:</p>

  <ol>
   <li><p>Let <var>value</var> be the empty string.</p></li>

   <li>
    <p>For each property named <var>subname</var> in the item <var>subitem</var>,
    run the following substeps:</p>

    <ol>
     <li><p>If the <span data-x="concept-property-value">value</span> of the property is itself an
     <span data-x="concept-item">item</span>, then skip this property.</p></li>

     <li><p>If this is not the first property named <var>subname</var> in <var>subitem</var> (ignoring any that were skipped by the previous step), then append a
     U+002C COMMA character (,) to <var>value</var>.</p></li>

     <li><p>Append the result of <span>escaping the vCard text string</span> given by the <span
     data-x="concept-property-value">value</span> of the property to <var>value</var>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>value</var>.</p></li>
  </ol>

  <p>When the steps above require the user agent to obtain the result of <dfn>collecting the first
  vCard subproperty</dfn> named <var>subname</var> in <var>subitem</var>, the user
  agent must run the following steps:</p>

  <ol>
   <li><p>If there are no properties named <var>subname</var> in <var>subitem</var>, then return the
   empty string.</p></li>

   <li><p>If the <span data-x="concept-property-value">value</span> of the first property named
   <var>subname</var> in <var>subitem</var> is an <span data-x="concept-item">item</span>, then
   return the empty string.</p></li>

   <li><p>Return the result of <span>escaping the vCard text string</span> given by the <span
   data-x="concept-property-value">value</span> of the first property named <var>subname</var> in
   <var>subitem</var>.</p></li>
  </ol>

  <p>When the above algorithms say the user agent is to <dfn data-x="escaping the vCard text
  string">escape the vCard text string</dfn> <var>value</var>, the user agent must use the
  following steps:</p>

  <ol>
   <li><p>Prefix every U+005C REVERSE SOLIDUS character (\) in <var>value</var> with
   another U+005C REVERSE SOLIDUS character (\).</p></li>

   <li><p>Prefix every U+002C COMMA character (,) in <var>value</var> with a U+005C REVERSE
   SOLIDUS character (\).</p></li>

   <li><p>Prefix every U+003B SEMICOLON character (;) in <var>value</var> with a U+005C
   REVERSE SOLIDUS character (\).</p></li>

   <li><p>Replace every U+000D CARRIAGE RETURN U+000A LINE FEED character pair (CRLF) in <var>value</var> with a U+005C REVERSE SOLIDUS character (\) followed by a U+006E LATIN SMALL
   LETTER N character (n).</p></li>

   <li><p>Replace every remaining U+000D CARRIAGE RETURN (CR) or U+000A LINE FEED (LF) character in
   <var>value</var> with a U+005C REVERSE SOLIDUS character (\) followed by a U+006E LATIN
   SMALL LETTER N character (n).</p></li>

   <li><p>Return the mutated <var>value</var>.</p></li>
  </ol>

  <p class="note">This algorithm can generate invalid vCard output, if the input does not conform to
  the rules described for the <code data-x="md-vcard">http://microformats.org/profile/hcard</code>
  <span data-x="item types">item type</span> and <span data-x="defined property name">defined property
  names</span>.</p> <!-- of course since vcard doesn't define error handling, this is somewhat
  problematic. -->



  <h5>Examples</h5>

  <!-- NON-NORMATIVE SECTION -->

  <div class="example">

   <p>Here is a long example vCard for a fictional character called "Jack Bauer":</p>

   <pre><code class="html">&lt;section id="jack" itemscope itemtype="http://microformats.org/profile/hcard">
 &lt;h1 itemprop="fn">
  &lt;span itemprop="n" itemscope>
   &lt;span itemprop="given-name">Jack&lt;/span>
   &lt;span itemprop="family-name">Bauer&lt;/span>
  &lt;/span>
 &lt;/h1>
 &lt;img itemprop="photo" alt="" src="jack-bauer.jpg">
 &lt;p itemprop="org" itemscope>
  &lt;span itemprop="organization-name">Counter-Terrorist Unit&lt;/span>
  (&lt;span itemprop="organization-unit">Los Angeles Division&lt;/span>)
 &lt;/p>
 &lt;p>
  &lt;span itemprop="adr" itemscope>
   &lt;span itemprop="street-address">10201 W. Pico Blvd.&lt;/span>&lt;br>
   &lt;span itemprop="locality">Los Angeles&lt;/span>,
   &lt;span itemprop="region">CA&lt;/span>
   &lt;span itemprop="postal-code">90064&lt;/span>&lt;br>
   &lt;span itemprop="country-name">United States&lt;/span>&lt;br>
  &lt;/span>
  &lt;span itemprop="geo">34.052339;-118.410623&lt;/span>
 &lt;/p>
 &lt;h2>Assorted Contact Methods&lt;/h2>
 &lt;ul>
  &lt;li itemprop="tel" itemscope>
   &lt;span itemprop="value">+1 (310) 597 3781&lt;/span> &lt;span itemprop="type">work&lt;/span>
   &lt;meta itemprop="type" content="voice">
  &lt;/li>
  &lt;li>&lt;a itemprop="url" href="https://en.wikipedia.org/wiki/Jack_Bauer">I'm on Wikipedia&lt;/a>
  so you can leave a message on my user talk page.&lt;/li>
  &lt;li>&lt;a itemprop="url" href="http://www.jackbauerfacts.com/">Jack Bauer Facts&lt;/a>&lt;/li>
  &lt;li itemprop="email">&lt;a href="mailto:j.bauer@la.ctu.gov.invalid">j.bauer@la.ctu.gov.invalid&lt;/a>&lt;/li>
  &lt;li itemprop="tel" itemscope>
   &lt;span itemprop="value">+1 (310) 555 3781&lt;/span> &lt;span>
   &lt;meta itemprop="type" content="cell">mobile phone&lt;/span>
  &lt;/li>
 &lt;/ul>
 &lt;ins datetime="2008-07-20 21:00:00+01:00">
  &lt;meta itemprop="rev" content="2008-07-20 21:00:00+01:00">
  &lt;p itemprop="tel" itemscope>&lt;strong>Update!&lt;/strong>
  My new &lt;span itemprop="type">home&lt;/span> phone number is
  &lt;span itemprop="value">01632 960 123&lt;/span>.&lt;/p>
 &lt;/ins>
&lt;/section></code></pre>

   <p>The odd line wrapping is needed because newlines are meaningful in microdata: newlines would
   be preserved in a conversion to, for example, the vCard format.</p>

  </div>

  <div class="example">

   <p>This example shows a site's contact details (using the <code>address</code> element)
   containing an address with two street components:</p>

   <pre><code class="html">&lt;address itemscope itemtype="http://microformats.org/profile/hcard">
 &lt;strong itemprop="fn">&lt;span itemprop="n" itemscope>&lt;span itemprop="given-name">Alfred&lt;/span>
 &lt;span itemprop="family-name">Person&lt;/span>&lt;/span>&lt;/strong> &lt;br>
 &lt;span itemprop="adr" itemscope>
  &lt;span itemprop="street-address">1600 Amphitheatre Parkway&lt;/span> &lt;br>
  &lt;span itemprop="street-address">Building 43, Second Floor&lt;/span> &lt;br>
  &lt;span itemprop="locality">Mountain View&lt;/span>,
   &lt;span itemprop="region">CA&lt;/span> &lt;span itemprop="postal-code">94043&lt;/span>
 &lt;/span>
&lt;/address></code></pre>

  </div>

  <div class="example">

   <p>The vCard vocabulary can be used to just mark up people's names:</p>

   <pre><code class="html">&lt;span itemscope itemtype="http://microformats.org/profile/hcard"
>&lt;span itemprop=fn>&lt;span itemprop="n" itemscope>&lt;span itemprop="given-name"
>George&lt;/span> &lt;span itemprop="family-name">Washington&lt;/span>&lt;/span
>&lt;/span>&lt;/span></code></pre>

   <p>This creates a single item with a two name-value pairs, one with the name "fn" and the value
   "George Washington", and the other with the name "n" and a second item as its value, the second
   item having the two name-value pairs "given-name" and "family-name" with the values "George" and
   "Washington" respectively. This is defined to map to the following vCard:</p>

   <pre>BEGIN:VCARD
PROFILE:VCARD
VERSION:4.0
SOURCE:<var>document's address</var>
FN:George Washington
N:Washington;George;;;
END:VCARD</pre>

  </div>


  <h4>vEvent</h4>

  <p>An item with the <span data-x="item types">item type</span> <dfn><code data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code></dfn> represents
  an event.</p>

  <p>This vocabulary does not <span>support global identifiers for items</span>.</p>

  <p>The following are the type's <span data-x="defined property name">defined property
  names</span>. They are based on the vocabulary defined in <cite>Internet Calendaring and
  Scheduling Core Object Specification</cite> (<cite>iCalendar</cite>), where more information on
  how to interpret the values can be found. <ref>RFC5545</ref></p>

  <p class="note">Only the parts of the iCalendar vocabulary relating to events are used here; this
  vocabulary cannot express a complete iCalendar instance.</p>

  <dl>

   <dt><dfn><code data-x="md-vevent-attach">attach</code></dfn></dt>

   <dd>
    <p>Gives the address of an associated document for the event.</p>

    <p>The <span data-x="concept-property-value">value</span> must be an <span>absolute
    URL</span>.</p>

    <p>Any number of properties with the name <code data-x="md-vevent-attach">attach</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-categories">categories</code></dfn></dt>

   <dd>
    <p>Gives the name of a category or tag that the event could be classified as.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Any number of properties with the name <code data-x="md-vevent-categories">categories</code>
    may be present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-class">class</code></dfn></dt>

   <dd>
    <p>Gives the access classification of the information regarding the event.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text with one of the following
    values:</p>

    <ul class="brief">
     <li><code data-x="">public</code></li>
     <li><code data-x="">private</code></li>
     <li><code data-x="">confidential</code></li>
    </ul>

    <p class="warning">This is merely advisory and cannot be considered a confidentiality
    measure.</p>

    <p>A single property with the name <code data-x="md-vevent-class">class</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-comment">comment</code></dfn></dt>

   <dd>
    <p>Gives a comment regarding the event.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>Any number of properties with the name <code data-x="md-vevent-comment">comment</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-description">description</code></dfn></dt>

   <dd>
    <p>Gives a detailed description of the event.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>A single property with the name <code data-x="md-vevent-description">description</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-geo">geo</code></dfn></dt>

   <dd>
    <p>Gives the geographical position of the event.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text and must match the
    following syntax:</p>

    <ol>
     <li>Optionally, either a U+002B PLUS SIGN character (+) or a U+002D HYPHEN-MINUS character
     (-).</li>

     <li>One or more <span>ASCII digits</span>.</li>

     <li>Optionally*, a U+002E FULL STOP character (.) followed by one or more <span>ASCII
     digits</span>.</li>

     <li>A U+003B SEMICOLON character (;).</li>

     <li>Optionally, either a U+002B PLUS SIGN character (+) or a U+002D HYPHEN-MINUS character
     (-).</li>

     <li>One or more <span>ASCII digits</span>.</li>

     <li>Optionally*, a U+002E FULL STOP character (.) followed by one or more <span>ASCII
     digits</span>.</li>
    </ol>

    <p>The optional components marked with an asterisk (*) should be included, and should have six
    digits each.</p> <!-- iCalendar actually limits the range to -91.0 < lat < 91.0; -181.0 < long <
    181.0, which seems weird. It also gives special meanings to +90/-90 lat, and +180 long. -->

    <!-- copied from vcard (not icalendar!) -->
    <p class="note">The value specifies latitude and longitude, in that order (i.e., "LAT LON"
    ordering), in decimal degrees. The longitude represents the location east and west of the prime
    meridian as a positive or negative real number, respectively. The latitude represents the
    location north and south of the equator as a positive or negative real number, respectively.</p>

    <p>A single property with the name <code data-x="md-vevent-geo">geo</code> may be present within
    each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-location">location</code></dfn></dt>

   <dd>
    <p>Gives the location of the event.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p> <!-- v2: support
    vcard here -->

    <p>A single property with the name <code data-x="md-vevent-location">location</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <!-- PRIORITY not included - always 0 - doesn't make sense for single events -->


   <dt><dfn><code data-x="md-vevent-resources">resources</code></dfn></dt>

   <dd>
    <p>Gives a resource that will be needed for the event.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p> <!-- v2: support
    vcard here -->

    <p>Any number of properties with the name <code data-x="md-vevent-resources">resources</code> may
    be present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-status">status</code></dfn></dt>

   <dd>
    <p>Gives the confirmation status of the event.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text with one of the following
    values:</p>

    <ul class="brief">
     <li><code data-x="">tentative</code></li>
     <li><code data-x="">confirmed</code></li>
     <li><code data-x="">canceled</code></li> <!-- two Ls as per iCalendar spec -->
    </ul>

    <p>A single property with the name <code data-x="md-vevent-status">status</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-summary">summary</code></dfn></dt>

   <dd>
    <p>Gives a short summary of the event.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>User agents should replace U+000A LINE FEED (LF) characters in the <span
    data-x="concept-property-value">value</span> by U+0020 SPACE characters when using the value.</p>

    <p>A single property with the name <code data-x="md-vevent-summary">summary</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-dtend">dtend</code></dfn></dt>

   <dd>
    <p>Gives the date and time by which the event ends.</p>

    <p>If the property with the name <code data-x="md-vevent-dtend">dtend</code> is present within an
    <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code> that has a property
    with the name <code data-x="md-vevent-dtstart">dtstart</code> whose value is a <span>valid date
    string</span>, then the <span data-x="concept-property-value">value</span> of the property with
    the name <code data-x="md-vevent-dtend">dtend</code> must be text that is a <span>valid date
    string</span> also. Otherwise, the <span data-x="concept-property-value">value</span> of the
    property must be text that is a <span>valid global date and time string</span>.</p>

    <p>In either case, the <span data-x="concept-property-value">value</span> be later in time than
    the value of the <code data-x="md-vevent-dtstart">dtstart</code> property of the same <span
    data-x="concept-item">item</span>.</p>

    <p class="note">The time given by the <code data-x="md-vevent-dtend">dtend</code> property is not
    inclusive. For day-long events, therefore, the <code data-x="md-vevent-dtend">dtend</code>
    property's <span data-x="concept-property-value">value</span> will be the day <em>after</em> the
    end of the event.</p>

    <p>A single property with the name <code data-x="md-vevent-dtend">dtend</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>, so long as that <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code> does not have a
    property with the name <code data-x="md-vevent-duration">duration</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-dtstart">dtstart</code></dfn></dt>

   <dd>
    <p>Gives the date and time at which the event starts.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text that is either a
    <span>valid date string</span> or a <span>valid global date and time string</span>.</p>

    <p>Exactly one property with the name <code data-x="md-vevent-dtstart">dtstart</code> must be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-duration">duration</code></dfn></dt>

   <dd>
    <p>Gives the duration of the event.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text that is a <span>valid
    vevent duration string</span>.</p>

    <p>The duration represented is the sum of all the durations represented by integers in the
    value.</p>

    <p>A single property with the name <code data-x="md-vevent-duration">duration</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>, so long as that <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code> does not have a
    property with the name <code data-x="md-vevent-dtend">dtend</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-transp">transp</code></dfn></dt>

   <dd>
    <p>Gives whether the event is to be considered as consuming time on a calendar, for the purpose
    of free-busy time searches.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text with one of the following
    values:</p>

    <ul class="brief">
     <li><code data-x="">opaque</code></li>
     <li><code data-x="">transparent</code></li>
    </ul>

    <p>A single property with the name <code data-x="md-vevent-transp">transp</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <!-- ATTENDEE not included; iCalendar says MUST NOT be included in
   published events -->


   <dt><dfn><code data-x="md-vevent-contact">contact</code></dfn></dt>

   <dd>
    <p>Gives the contact information for the event.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p> <!-- v2: support
    vcard here -->

    <p>Any number of properties with the name <code data-x="md-vevent-contact">contact</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <!-- ORGANIZER not included, as it is calendar-system-specific (gives a CAL-ADDRESS value); use
   CONTACT instead -->


   <!-- RECURRENCE-ID not included... unclear how to make it usefully work here -->


   <!-- RELATED-TO not included, as it only makes sense for complete calendars -->


   <dt><dfn><code data-x="md-vevent-url">url</code></dfn></dt>

   <dd>
    <p>Gives a <span>URL</span> for the event.</p>

    <p>The <span data-x="concept-property-value">value</span> must be an <span>absolute
    URL</span>.</p>

    <p>A single property with the name <code data-x="md-vevent-url">url</code> may be present within
    each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-uid">uid</code></dfn></dt>

   <dd>
    <p>Gives a globally unique identifier corresponding to the event.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text.</p>

    <p>A single property with the name <code data-x="md-vevent-uid">uid</code> may be present within
    each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-exdate">exdate</code></dfn></dt>

   <dd>
    <p>Gives a date and time at which the event does not occur despite the recurrence rules.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text that is either a
    <span>valid date string</span> or a <span>valid global date and time string</span>.</p>

    <p>Any number of properties with the name <code data-x="md-vevent-exdate">exdate</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-rdate">rdate</code></dfn></dt>

   <dd>
    <p>Gives a date and time at which the event recurs.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text that is one of the
    following:

    <ul>
     <li>A <span>valid date string</span>.</li>

     <li>A <span>valid global date and time string</span>.</li>

     <li>A <span>valid global date and time string</span> followed by a U+002F SOLIDUS character (/)
     followed by a second <span>valid global date and time string</span> representing a later
     time.</li>

     <li>A <span>valid global date and time string</span> followed by a U+002F SOLIDUS character (/)
     followed by a <span>valid vevent duration string</span>.</li>
    </ul>

    <p>Any number of properties with the name <code data-x="md-vevent-rdate">rdate</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-rrule">rrule</code></dfn></dt>

   <dd>
    <p>Gives a rule for finding dates and times at which the event occurs.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text that matches the RECUR
    value type defined in <cite>iCalendar</cite>. <ref>RFC5545</ref></p>

    <p>A single property with the name <code data-x="md-vevent-rrule">rrule</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-created">created</code></dfn></dt>

   <dd>
    <p>Gives the date and time at which the event information was first created in a calendaring
    system.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text that is a <span>valid
    global date and time string</span>.</p>

    <p>A single property with the name <code data-x="md-vevent-created">created</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <!-- DTSTAMP not included, it gets added when serializing -->


   <dt><dfn><code data-x="md-vevent-last-modified">last-modified</code></dfn></dt>

   <dd>
    <p>Gives the date and time at which the event information was last modified in a calendaring
    system.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text that is a <span>valid
    global date and time string</span>.</p>

    <p>A single property with the name <code data-x="md-vevent-last-modified">last-modified</code>
    may be present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-vevent-sequence">sequence</code></dfn></dt>

   <dd>
    <p>Gives a revision number for the event information.</p>

    <p>The <span data-x="concept-property-value">value</span> must be text that is a <span>valid
    non-negative integer</span>.</p>

    <p>A single property with the name <code data-x="md-vevent-sequence">sequence</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>.</p>
   </dd>


   <!-- REQUEST-STATUS not included, calendaring system interop
   feature -->
  </dl>

  <p>A string is a <dfn>valid vevent duration string</dfn> if it matches the following pattern:</p>

  <ol>
   <li><p>A U+0050 LATIN CAPITAL LETTER P character (P).</p></li>

   <li>
    <p>One of the following:</p>

    <ul>
     <li><p>A <span>valid non-negative integer</span> followed by a U+0057 LATIN CAPITAL LETTER W
     character (W). The integer represents a duration of that number of weeks.</p></li>

     <li>
      <p>At least one, and possible both in this order, of the following:</p>

      <ol>
       <li><p>A <span>valid non-negative integer</span> followed by a U+0044 LATIN CAPITAL LETTER D
       character (D). The integer represents a duration of that number of days.</p></li>

       <li>
        <p>A U+0054 LATIN CAPITAL LETTER T character (T) followed by any one of the following, or
        the first and second of the following in that order, or the second and third of the
        following in that order, or all three of the following in this order:</p>

        <ol>
         <li><p>A <span>valid non-negative integer</span> followed by a U+0048 LATIN CAPITAL LETTER
         H character (H). The integer represents a duration of that number of hours.</p></li>

         <li><p>A <span>valid non-negative integer</span> followed by a U+004D LATIN CAPITAL LETTER
         M character (M). The integer represents a duration of that number of minutes.</p></li>

         <li><p>A <span>valid non-negative integer</span> followed by a U+0053 LATIN CAPITAL LETTER
         S character (S). The integer represents a duration of that number of seconds.</p></li>
        </ol>
      </ol>
     </li>
    </ul>
   </li>
  </ol>


  <h5>Conversion to iCalendar</h5>

  <p>Given a list of nodes <var>nodes</var> in a <code>Document</code>, a user agent must
  run the following algorithm to <dfn data-x="extracting vEvent data">extract any vEvent data
  represented by those nodes</dfn>:</p>

  <ol>
   <li><p>If none of the nodes in <var>nodes</var> are <span
   data-x="concept-item">items</span> with the type <code
   data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>, then there is no
   vEvent data. Abort the algorithm, returning nothing.</p></li>

   <li><p>Let <var>output</var> be an empty string.</p></li>

   <li><p><span>Add an iCalendar line</span> with the type "<code data-x="">BEGIN</code>" and the
   value "<code data-x="">VCALENDAR</code>" to <var>output</var>.</p></li>

   <li><p><span>Add an iCalendar line</span> with the type "<code data-x="">PRODID</code>" and the
   value equal to a user-agent-specific string representing the user agent to <var>output</var>.</p></li>

   <li><p><span>Add an iCalendar line</span> with the type "<code data-x="">VERSION</code>" and the
   value "<code data-x="">2.0</code>" to <var>output</var>.</p></li>

   <li>
    <p>For each node <var>node</var> in <var>nodes</var> that is an <span
    data-x="concept-item">item</span> with the type <code
    data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code>, run the following
    steps:</p>

    <ol>
     <li><p><span>Add an iCalendar line</span> with the type "<code data-x="">BEGIN</code>" and the
     value "<code data-x="">VEVENT</code>" to <var>output</var>.</p></li>

     <li><p><span>Add an iCalendar line</span> with the type "<code data-x="">DTSTAMP</code>" and a
     value consisting of an iCalendar DATE-TIME string representing the current date and time, with
     the annotation "<code data-x="">VALUE=DATE-TIME</code>", to <var>output</var>.
     <ref>RFC5545</ref></p></li>

     <li>
      <p>For each element <var>element</var> that is <span data-x="the properties of an
      item">a property of the item</span> <var>node</var>: for each name <var>name</var> in <var>element</var>'s <span>property names</span>, run the
      appropriate set of substeps from the following list:</p>

      <dl>
       <dt>If the property's <span data-x="concept-property-value">value</span> is an <span
       data-x="concept-item">item</span></dt>

       <dd>
        <p>Skip the property.</p>
       </dd>


       <dt>If the property is <code data-x="md-vevent-dtend">dtend</code></dt>
       <dt>If the property is <code data-x="md-vevent-dtstart">dtstart</code></dt>
       <dt>If the property is <code data-x="md-vevent-exdate">exdate</code></dt>
       <dt>If the property is <code data-x="md-vevent-rdate">rdate</code></dt>
       <dt>If the property is <code data-x="md-vevent-created">created</code></dt>
       <dt>If the property is <code data-x="md-vevent-last-modified">last-modified</code></dt>

       <dd>
        <p>Let <var>value</var> be the result of stripping all U+002D HYPHEN-MINUS (-) and
        U+003A COLON (:) characters from the property's <span
        data-x="concept-property-value">value</span>.</p>

        <p>If the property's <span data-x="concept-property-value">value</span> is a <span>valid date
        string</span> then <span>add an iCalendar line</span> with the type <var>name</var>
        and the value <var>value</var> to <var>output</var>, with the annotation
        "<code data-x="">VALUE=DATE</code>".</p>

        <p>Otherwise, if the property's <span data-x="concept-property-value">value</span> is a
        <span>valid global date and time string</span> then <span>add an iCalendar line</span> with
        the type <var>name</var> and the value <var>value</var> to <var>output</var>, with the annotation "<code data-x="">VALUE=DATE-TIME</code>".</p>

        <p>Otherwise, skip the property.</p>
       </dd>


       <dt>Otherwise</dt>

       <dd>
        <p><span>Add an iCalendar line</span> with the type <var>name</var> and the
        property's <span data-x="concept-property-value">value</span> to <var>output</var>.</p>
       </dd>
      </dl>
     </li>

     <li><p><span>Add an iCalendar line</span> with the type "<code data-x="">END</code>" and the
     value "<code data-x="">VEVENT</code>" to <var>output</var>.</p></li>
    </ol>
   </li>

   <li><p><span>Add an iCalendar line</span> with the type "<code data-x="">END</code>" and the value
   "<code data-x="">VCALENDAR</code>" to <var>output</var>.</p></li>
  </ol>

  <p>When the above algorithm says that the user agent is to <dfn>add an iCalendar line</dfn>
  consisting of a type <var>type</var>, a value <var>value</var>, and optionally
  an annotation, to a string <var>output</var>, it must run the following steps:</p>

  <ol>
   <li><p>Let <var>line</var> be an empty string.</p></li>

   <li><p>Append <var>type</var>, <span>converted to ASCII uppercase</span>, to <var>line</var>.</p></li>

   <li>
    <p>If there is an annotation:</p>

    <ol>
     <li><p>Append a U+003B SEMICOLON character (;) to <var>line</var>.</p></li>

     <li><p>Append the annotation to <var>line</var>.</p></li>
    </ol>
   </li>

   <li><p>Append a U+003A COLON character (:) to <var>line</var>.</p></li>

   <li><p>Prefix every U+005C REVERSE SOLIDUS character (\) in <var>value</var> with
   another U+005C REVERSE SOLIDUS character (\).</p></li>

   <li><p>Prefix every U+002C COMMA character (,) in <var>value</var> with a U+005C REVERSE
   SOLIDUS character (\).</p></li>

   <li><p>Prefix every U+003B SEMICOLON character (;) in <var>value</var> with a U+005C
   REVERSE SOLIDUS character (\).</p></li>

   <li><p>Replace every U+000D CARRIAGE RETURN U+000A LINE FEED character pair (CRLF) in <var>value</var> with a U+005C REVERSE SOLIDUS character (\) followed by a U+006E LATIN SMALL
   LETTER N character (n).</p></li>

   <li><p>Replace every remaining U+000D CARRIAGE RETURN (CR) or U+000A LINE FEED (LF) character in
   <var>value</var> with a U+005C REVERSE SOLIDUS character (\) followed by a U+006E LATIN
   SMALL LETTER N character (n).</p></li>

   <li><p>Append <var>value</var> to <var>line</var>.</p></li>

   <li><p>Let <var>maximum length</var> be 75.</p></li>

   <li>
    <p>While <var>line</var>'s <span>code point length</span> is greater than <var>maximum
    length</var>:</p>

    <ol>
     <li><p>Append the first <var>maximum length</var> code points of <var>line</var> to
     <var>output</var>.</p></li>

     <li><p>Remove the first <var>maximum length</var> code points from <var>line</var>.</p></li>

     <li><p>Append a U+000D CARRIAGE RETURN character (CR) to <var>output</var>.</p></li>

     <li><p>Append a U+000A LINE FEED character (LF) to <var>output</var>.</p></li>

     <li><p>Append a U+0020 SPACE character to <var>output</var>.</p></li>

     <li><p>Let <var>maximum length</var> be 74.</p></li>
    </ol>
   </li>

   <li><p>Append (what remains of) <var>line</var> to <var>output</var>.</p></li>

   <li><p>Append a U+000D CARRIAGE RETURN character (CR) to <var>output</var>.</p></li>

   <li><p>Append a U+000A LINE FEED character (LF) to <var>output</var>.</p></li>
  </ol>

  <p class="note">This algorithm can generate invalid iCalendar output, if the input does not
  conform to the rules described for the <code
  data-x="md-vevent">http://microformats.org/profile/hcalendar#vevent</code> <span data-x="item
  types">item type</span> and <span data-x="defined property name">defined property names</span>.</p>
  <!-- of course since iCalendar doesn't define error handling, this is somewhat problematic. -->



  <h5>Examples</h5>

  <!-- NON-NORMATIVE SECTION -->

<!-- get more from https://www.rfc-editor.org/rfc/rfc5545 -->

  <div class="example">

   <p>Here is an example of a page that uses the vEvent vocabulary to mark up an event:</p>

   <pre><code class="html">&lt;body itemscope itemtype="http://microformats.org/profile/hcalendar#vevent">
 ...
 &lt;h1 itemprop="summary">Bluesday Tuesday: Money Road&lt;/h1>
 ...
 &lt;time itemprop="dtstart" datetime="2009-05-05T19:00:00Z">May 5th @ 7pm&lt;/time>
 (until &lt;time itemprop="dtend" datetime="2009-05-05T21:00:00Z">9pm&lt;/time>)
 ...
 &lt;a href="http://livebrum.co.uk/2009/05/05/bluesday-tuesday-money-road"
    rel="bookmark" itemprop="url">Link to this page&lt;/a>
 ...
 &lt;p>Location: &lt;span itemprop="location">The RoadHouse&lt;/span>&lt;/p>
 ...
 &lt;p>&lt;input type=button value="Add to Calendar"
           onclick="location = getCalendar(this)">&lt;/p>
 ...
 &lt;meta itemprop="description" content="via livebrum.co.uk">
&lt;/body></code></pre>

   <p>The <code data-x="">getCalendar()</code> function is left as an exercise for the reader.</p>

   <p>The same page could offer some markup, such as the following,
   for copy-and-pasting into blogs:</p>

   <pre><code class="html">&lt;div itemscope itemtype="http://microformats.org/profile/hcalendar#vevent">
 &lt;p>I'm going to
 &lt;strong itemprop="summary">Bluesday Tuesday: Money Road&lt;/strong>,
 &lt;time itemprop="dtstart" datetime="2009-05-05T19:00:00Z">May 5th at 7pm&lt;/time>
 to &lt;time itemprop="dtend" datetime="2009-05-05T21:00:00Z">9pm&lt;/time>,
 at &lt;span itemprop="location">The RoadHouse&lt;/span>!&lt;/p>
 &lt;p>&lt;a href="http://livebrum.co.uk/2009/05/05/bluesday-tuesday-money-road"
       itemprop="url">See this event on livebrum.co.uk&lt;/a>.&lt;/p>
 &lt;meta itemprop="description" content="via livebrum.co.uk">
&lt;/div></code></pre>

  </div>


  <h4>Licensing works</h4>

  <p>An item with the <span data-x="item types">item type</span> <dfn><code data-x="md-work">http://n.whatwg.org/work</code></dfn> represents a work (e.g. an article, an
  image, a video, a song, etc.). This type is primarily intended to allow authors to include
  licensing information for works.</p>

  <p>The following are the type's <span data-x="defined property name">defined property
  names</span>.</p>

  <dl>

   <dt><dfn><code data-x="md-work-work">work</code></dfn></dt>

   <dd>
    <p>Identifies the work being described.</p>

    <p>The <span data-x="concept-property-value">value</span> must be an <span>absolute
    URL</span>.</p>

    <p>Exactly one property with the name <code data-x="md-work-work">work</code> must be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-work">http://n.whatwg.org/work</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-work-title">title</code></dfn></dt>

   <dd>
    <p>Gives the name of the work.</p>

    <p>A single property with the name <code data-x="md-work-title">title</code> may be present
    within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-work">http://n.whatwg.org/work</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-work-author">author</code></dfn></dt>

   <dd>
    <p>Gives the name or contact information of one of the authors or creators of the work.</p>

    <p>The <span data-x="concept-property-value">value</span> must be either an <span
    data-x="concept-item">item</span> with the type <code
    data-x="md-vcard">http://microformats.org/profile/hcard</code>, or text.</p>

    <p>Any number of properties with the name <code data-x="md-work-author">author</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-work">http://n.whatwg.org/work</code>.</p>
   </dd>


   <dt><dfn><code data-x="md-work-license">license</code></dfn></dt>

   <dd>
    <p>Identifies one of the licenses under which the work is available.</p>

    <p>The <span data-x="concept-property-value">value</span> must be an <span>absolute
    URL</span>.</p>

    <p>Any number of properties with the name <code data-x="md-work-license">license</code> may be
    present within each <span data-x="concept-item">item</span> with the type <code
    data-x="md-work">http://n.whatwg.org/work</code>.</p>
   </dd>
  </dl>




  <h5>Examples</h5>

  <!-- NON-NORMATIVE SECTION -->

  <div class="example">

   <p>This example shows an embedded image entitled <cite>My Pond</cite>, licensed under the
   Creative Commons Attribution-Share Alike 4.0 International License and the MIT license
   simultaneously.</p>

   <pre><code class="html">&lt;figure <strong>itemscope itemtype="http://n.whatwg.org/work"</strong>>
 &lt;img <strong>itemprop="work"</strong> src="mypond.jpeg">
 &lt;figcaption>
  &lt;p>&lt;cite <strong>itemprop="title"</strong>>My Pond&lt;/cite>&lt;/p>
  &lt;p>&lt;small>Licensed under the &lt;a <strong>itemprop="license"</strong>
  href="https://creativecommons.org/licenses/by-sa/4.0/">Creative
  Commons Attribution-Share Alike 4.0 International License&lt;/a>
  and the &lt;a <strong>itemprop="license"</strong>
  href="http://www.opensource.org/licenses/mit-license.php">MIT
  license&lt;/a>.&lt;/small>
 &lt;/figcaption>
&lt;/figure></code></pre>

  </div>


  <div w-nodev>

  <h3>Converting HTML to other formats</h3>

  <h4>JSON</h4>

  <p>Given a list of nodes <var>nodes</var> in a <code>Document</code>, a user agent must
  run the following algorithm to <dfn data-x="extracting JSON">extract the microdata from those nodes
  into a JSON form</dfn>:</p>

  <ol>
   <li><p>Let <var>result</var> be an empty object.</p></li>

   <li><p>Let <var>items</var> be an empty array.</p></li>

   <li><p>For each <var>node</var> in <var>nodes</var>, check if the element is a
   <span data-x="top-level microdata items">top-level microdata item</span>, and if it is then
   <span>get the object</span> for that element and add it to <var>items</var>.</p></li>

   <li><p>Add an entry to <var>result</var> called "<code data-x="">items</code>" whose
   value is the array <var>items</var>.</p></li>

   <li><p>Return the result of serializing <var>result</var> to JSON in the shortest
   possible way (meaning no whitespace between tokens, no unnecessary zero digits in numbers, and
   only using Unicode escapes in strings for characters that do not have a dedicated escape
   sequence), and with a lowercase "<code data-x="">e</code>" used, when appropriate, in the
   representation of any numbers. <ref>JSON</ref></p></li>
  </ol>

  <p class="note">This algorithm returns an object with a single property that is an array, instead
  of just returning an array, so that it is possible to extend the algorithm in the future if
  necessary.</p>

  <p>When the user agent is to <dfn>get the object</dfn> for an item <var>item</var>,
  optionally with a list of elements <var>memory</var>, it must run the following
  substeps:</p>

  <ol>
   <li><p>Let <var>result</var> be an empty object.</p></li>

   <li><p>If no <var>memory</var> was passed to the algorithm, let <var>memory</var> be an empty list.</p></li>

   <li><p>Add <var>item</var> to <var>memory</var>.</p></li>

   <li><p>If the <var>item</var> has any <span>item types</span>, add an entry to <var>result</var> called "<code data-x="">type</code>" whose value is an array listing the
   <span>item types</span> of <var>item</var>, in the order they were specified on the
   <code data-x="attr-itemtype">itemtype</code> attribute.</p>

   <li><p>If the <var>item</var> has a <span>global identifier</span>, add an entry to <var>result</var> called "<code data-x="">id</code>" whose value is the <span>global
   identifier</span> of <var>item</var>.</p></li>

   <li><p>Let <var>properties</var> be an empty object.</p></li>

   <li>
    <p>For each element <var>element</var> that has one or more <span>property names</span>
    and is one of <span data-x="the properties of an item">the properties of the item</span> <var>item</var>, in the order those elements are given by the algorithm that returns
    <span>the properties of an item</span>, run the following substeps:</p>

    <ol>
     <li><p>Let <var>value</var> be the <span data-x="concept-property-value">property
     value</span> of <var>element</var>.</p></li>

     <li><p>If <var>value</var> is an <span data-x="concept-item">item</span>, then: If <var>value</var> is in <var>memory</var>, then let <var>value</var> be
     the string "<code data-x="">ERROR</code>". Otherwise, <span>get the object</span> for <var>value</var>, passing a copy of <var>memory</var>, and then replace <var>value</var> with the object returned from those steps.</p></li>

     <li>
      <p>For each name <var>name</var> in <var>element</var>'s <span>property
      names</span>, run the following substeps:</p>

      <ol>
       <li><p>If there is no entry named <var>name</var> in <var>properties</var>,
       then add an entry named <var>name</var> to <var>properties</var> whose
       value is an empty array.</p></li>

       <li><p>Append <var>value</var> to the entry named <var>name</var> in <var>properties</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Add an entry to <var>result</var> called "<code data-x="">properties</code>" whose
   value is the object <var>properties</var>.</p></li>

   <li><p>Return <var>result</var>.</p></li>
  </ol>

  <div class="example">

   <p>For example, take this markup:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang="en">
&lt;title>My Blog&lt;/title>
&lt;article itemscope itemtype="http://schema.org/BlogPosting">
 &lt;header>
  &lt;h1 itemprop="headline">Progress report&lt;/h1>
  &lt;p>&lt;time itemprop="datePublished" datetime="2013-08-29">today&lt;/time>&lt;/p>
  &lt;link itemprop="url" href="?comments=0">
 &lt;/header>
 &lt;p>All in all, he's doing well with his swim lessons. The biggest thing was he had trouble
 putting his head in, but we got it down.&lt;/p>
 &lt;section>
  &lt;h1>Comments&lt;/h1>
  &lt;article itemprop="comment" itemscope itemtype="http://schema.org/UserComments" id="c1">
   &lt;link itemprop="url" href="#c1">
   &lt;footer>
    &lt;p>Posted by: &lt;span itemprop="creator" itemscope itemtype="http://schema.org/Person">
     &lt;span itemprop="name">Greg&lt;/span>
    &lt;/span>&lt;/p>
    &lt;p>&lt;time itemprop="commentTime" datetime="2013-08-29">15 minutes ago&lt;/time>&lt;/p>
   &lt;/footer>
   &lt;p>Ha!&lt;/p>
  &lt;/article>
  &lt;article itemprop="comment" itemscope itemtype="http://schema.org/UserComments" id="c2">
   &lt;link itemprop="url" href="#c2">
   &lt;footer>
    &lt;p>Posted by: &lt;span itemprop="creator" itemscope itemtype="http://schema.org/Person">
     &lt;span itemprop="name">Charlotte&lt;/span>
    &lt;/span>&lt;/p>
    &lt;p>&lt;time itemprop="commentTime" datetime="2013-08-29">5 minutes ago&lt;/time>&lt;/p>
   &lt;/footer>
   &lt;p>When you say "we got it down"...&lt;/p>
  &lt;/article>
 &lt;/section>
&lt;/article></code></pre>

   <p>It would be turned into the following JSON by the algorithm above (supposing that the page's
   URL was <code data-x="">https://blog.example.com/progress-report</code>):</p>

   <pre><code data-x="" class="json">{
  "items": [
    {
      "type": [ "http://schema.org/BlogPosting" ],
      "properties": {
        "headline": [ "Progress report" ],
        "datePublished": [ "2013-08-29" ],
        "url": [ "https://blog.example.com/progress-report?comments=0" ],
        "comment": [
          {
            "type": [ "http://schema.org/UserComments" ],
            "properties": {
              "url": [ "https://blog.example.com/progress-report#c1" ],
              "creator": [
                {
                  "type": [ "http://schema.org/Person" ],
                  "properties": {
                    "name": [ "Greg" ]
                  }
                }
              ],
              "commentTime": [ "2013-08-29" ]
            }
          },
          {
            "type": [ "http://schema.org/UserComments" ],
            "properties": {
              "url": [ "https://blog.example.com/progress-report#c2" ],
              "creator": [
                {
                  "type": [ "http://schema.org/Person" ],
                  "properties": {
                    "name": [ "Charlotte" ]
                  }
                }
              ],
              "commentTime": [ "2013-08-29" ]
            }
          }
        ]
      }
    }
  ]
}</code></pre>

  </div>

  </div>




  <h2 split-filename="interaction" id="editing"><dfn>User interaction</dfn></h2>


  <h3>The <code data-x="attr-hidden">hidden</code> attribute</h3>

  <p>All <span>HTML elements</span> may have the <dfn element-attr for="html-global"><code
  data-x="attr-hidden">hidden</code></dfn> content attribute set. The <code
  data-x="attr-hidden">hidden</code> attribute is an <span>enumerated attribute</span> with the
  following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="html-global/hidden"><code data-x="attr-hidden-hidden">hidden</code></dfn>
     <td rowspan=2><dfn data-x="attr-hidden-hidden-state">hidden</dfn>
     <td rowspan=2>Will not be rendered.
    <tr>
     <td>(the empty string)
    <tr>
     <td><dfn attr-value for="html-global/hidden"><code data-x="attr-hidden-until-found">until-found</code></dfn>
     <td><dfn data-x="attr-hidden-until-found-state">hidden until found</dfn>
     <td>Will not be rendered, but content inside will be accessible to <span>find-in-page</span>
     and <span data-x="navigate-fragid">fragment navigation</span>.
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> is the <dfn
  data-x="attr-hidden-not-hidden-state">not hidden</dfn> state, and its <i data-x="invalid value
  default">invalid value default</i> is the <span data-x="attr-hidden-hidden-state">hidden</span>
  state.</p>

  <p>When an element has the <code data-x="attr-hidden">hidden</code> attribute in the <span
  data-x="attr-hidden-hidden-state">hidden</span> state, it indicates that the element is not yet, or
  is no longer, directly relevant to the page's current state, or that it is being used to declare
  content to be reused by other parts of the page as opposed to being directly accessed by the user.
  User agents should not render elements that are in the <span
  data-x="attr-hidden-hidden-state">hidden</span> state.</p>

  <div w-nodev>

  <p class="note">The requirement for user agents not to render elements that are in the <span
  data-x="attr-hidden-hidden-state">hidden</span> state can be implemented indirectly through the
  style layer. For example, a web browser could implement these requirements <a
  href="#hiddenCSS">using the rules suggested in the Rendering section</a>.</p>

  </div>

  <p>When an element has the <code data-x="attr-hidden">hidden</code> attribute in the <span
  data-x="attr-hidden-until-found-state">hidden until found</span> state, it indicates that the
  element is hidden like the <span data-x="attr-hidden-hidden-state">hidden</span> state but the
  content inside the element will be accessible to <span>find-in-page</span> and <span
  data-x="navigate-fragid">fragment navigation</span>. When these features attempt to scroll to a
  target which is in the element's subtree, the user agent will remove the <code
  data-x="attr-hidden">hidden</code> attribute in order to reveal the content before scrolling to
  it by running the <span>ancestor hidden-until-found revealing algorithm</span> on the target
  node.</p>

  <div w-nodev>

  <p class="note">Web browsers will use 'content-visibility: hidden' instead of 'display: none'
  when the <code data-x="attr-hidden">hidden</code> attribute is in the <span
  data-x="attr-hidden-until-found-state">hidden until found</span> state, as specified in the <a
  href="#hiddenCSS">Rendering section</a>.</p>

  </div>


  <div class="note">
   <p>Because this attribute is typically implemented using CSS, it's also possible to override it
   using CSS. For instance, a rule that applies 'display: block' to all elements will cancel the
   effects of the <span data-x="attr-hidden-hidden-state">hidden</span> state. Authors therefore
   have to take care when writing their style sheets to make sure that the attribute is still styled
   as expected. In addition, legacy user agents which don't support the <span
   data-x="attr-hidden-until-found-state">hidden until found</span> state will have 'display: none'
   instead of 'content-visibility: hidden', so authors are encouraged to make sure that their style
   sheets don't change the 'display' or 'content-visibility' properties of <span
   data-x="attr-hidden-until-found-state">hidden until found</span> elements.</p>

   <p>Since elements with the <code data-x="attr-hidden">hidden</code> attribute in the <span
   data-x="attr-hidden-until-found-state">hidden until found</span> state use 'content-visibility:
   hidden' instead of 'display: none', there are two caveats of the <span
   data-x="attr-hidden-until-found-state">hidden until found</span> state that make it different
   from the <span data-x="attr-hidden-hidden-state">hidden</span> state:</p>

   <ol>
    <li><p>The element needs to be affected by <span>layout containment</span> in order to be
    revealed by find-in-page. This means that if the element in the <span
    data-x="attr-hidden-until-found-state">hidden until found</span> state has a 'display' value of
    'none', 'contents', or 'inline', then the element will not be revealed by find-in-page.</p></li>

    <li><p>The element will still have a <span data-x="'visibility'">generated box</span> when in
    the <span data-x="attr-hidden-until-found-state">hidden until found</span> state, which means
    that borders, margin, and padding will still be rendered around the element.</p></li>
   </ol>
  </div>

  <div class="example">

   <p>In the following skeletal example, the attribute is used to hide the web game's main screen
   until the user logs in:</p>

   <pre><code class="html">  &lt;h1>The Example Game&lt;/h1>
  &lt;section id="login">
   &lt;h2>Login&lt;/h2>
   &lt;form>
    ...
    &lt;!-- calls login() once the user's credentials have been checked -->
   &lt;/form>
   &lt;script>
    function login() {
      // switch screens
      document.getElementById('login').hidden = true;
      document.getElementById('game').hidden = false;
    }
   &lt;/script>
  &lt;/section>
  &lt;section id="game" hidden>
   ...
  &lt;/section></code></pre>

  </div>

  <p>The <code data-x="attr-hidden">hidden</code> attribute must not be used to hide content that
  could legitimately be shown in another presentation. For example, it is incorrect to use <code
  data-x="attr-hidden">hidden</code> to hide panels in a tabbed dialog, because the tabbed interface
  is merely a kind of overflow presentation &mdash; one could equally well just show all the form
  controls in one big page with a scrollbar. It is similarly incorrect to use this attribute to hide
  content just from one presentation &mdash; if something is marked <code
  data-x="attr-hidden">hidden</code>, it is hidden from all presentations, including, for instance,
  screen readers.</p>

  <!-- for example, "<a hidden href=#content>Skip to content</a>" would be inappropriate. -->
  <!-- (but only add that example if you first add some more good valid examples -->

  <p>Elements that are not themselves <code data-x="attr-hidden">hidden</code> must not
  <span>hyperlink</span> to elements that are <code data-x="attr-hidden">hidden</code>. The <code
  data-x="">for</code> attributes of <code>label</code> and <code>output</code> elements that are not
  themselves <code data-x="attr-hidden">hidden</code> must similarly not refer to elements that are
  <code data-x="attr-hidden">hidden</code>. In both cases, such references would cause user
  confusion.</p>

  <p>Elements and scripts may, however, refer to elements that are <code
  data-x="attr-hidden">hidden</code> in other contexts.</p>

  <div class="example">

   <p>For example, it would be incorrect to use the <code data-x="attr-hyperlink-href">href</code>
   attribute to link to a section marked with the <code data-x="attr-hidden">hidden</code> attribute.
   If the content is not applicable or relevant, then there is no reason to link to it.</p>

   <p>It would be fine, however, to use the ARIA <code
   data-x="attr-aria-describedby">aria-describedby</code> attribute to refer to descriptions that are
   themselves <code data-x="attr-hidden">hidden</code>. While hiding the descriptions implies that
   they are not useful alone, they could be written in such a way that they are useful in the
   specific context of being referenced from the elements that they describe.</p>

   <p>Similarly, a <code>canvas</code> element with the <code data-x="attr-hidden">hidden</code>
   attribute could be used by a scripted graphics engine as an off-screen buffer, and a form control
   could refer to a hidden <code>form</code> element using its <code
   data-x="attr-fae-form">form</code> attribute.</p>

  </div>

  <p>Elements in a section hidden by the <code data-x="attr-hidden">hidden</code> attribute are still
  active, e.g. scripts and form controls in such sections still execute and submit respectively.
  Only their presentation to the user changes.</p>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLElement"><code data-x="dom-hidden">hidden</code></dfn> getter steps
  are:</p>

  <ol>
   <li><p>If the <code data-x="attr-hidden">hidden</code> attribute is in the <span
   data-x="attr-hidden-until-found-state">hidden until found</span> state, then return "<code
   data-x="attr-hidden-until-found">until-found</code>".</p></li>

   <li><p>If the <code data-x="attr-hidden">hidden</code> attribute is set, then return
   true.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  <p>The <code data-x="dom-hidden">hidden</code> setter steps are:</p>

  <ol>
   <li><p>If the given value is a string that is an <span>ASCII case-insensitive</span> match for
   "<code data-x="attr-hidden-until-found">until-found</code>", then set the <code
   data-x="attr-hidden">hidden</code> attribute to "<code
   data-x="attr-hidden-until-found">until-found</code>".</p></li>

   <li><p>Otherwise, if the given value is false, then remove the <code
   data-x="attr-hidden">hidden</code> attribute.</p></li>

   <li><p>Otherwise, if the given value is the empty string, then remove the <code
   data-x="attr-hidden">hidden</code> attribute.</p></li>

   <li><p>Otherwise, if the given value is null, then remove the <code
   data-x="attr-hidden">hidden</code> attribute.</p></li>

   <li><p>Otherwise, if the given value is 0, then remove the <code
   data-x="attr-hidden">hidden</code> attribute.</p></li>

   <li><p>Otherwise, if the given value is NaN, then remove the <code
   data-x="attr-hidden">hidden</code> attribute.</p></li>

   <li><p>Otherwise, set the <code data-x="attr-hidden">hidden</code> attribute to the empty
   string.</p></li>
  </ol>

  <p>The <dfn>ancestor hidden-until-found revealing algorithm</dfn> is to run the following steps on
  <var>currentNode</var>:</p>

  <ol>
   <li>
    <p>While <var>currentNode</var> has a parent node within the <span>flat tree</span>:</p>

    <ol>
     <li>
      <p>If <var>currentNode</var> has the <code data-x="attr-hidden">hidden</code> attribute in the
      <span data-x="attr-hidden-until-found-state">hidden until found</span> state, then:</p>

      <ol>
       <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
       data-x="event-beforematch">beforematch</code> at <var>currentNode</var> with the <code
       data-x="dom-Event-bubbles">bubbles</code> attribute initialized to true.</p></li>

       <li><p>Remove the <code data-x="attr-hidden">hidden</code> attribute from
       <var>currentNode</var>.</p></li>
      </ol>
     </li>

     <li><p>Set <var>currentNode</var> to the parent node of <var>currentNode</var> within the
     <span>flat tree</span>.</p></li>
    </ol>
   </li>
  </ol>

  </div>

  <h3>Page visibility</h3>

  <p>A <span>traversable navigable</span>'s <span>system visibility state</span>, including its
  initial value upon creation, is determined by the user agent. It represents, for example, whether
  the browser window is minimized, a browser tab is currently in the background, or a system
  element such as a task switcher obscures the page.</p>

  <p>When a user-agent determines that the <span>system visibility state</span> for
  <span>traversable navigable</span> <var>traversable</var> has changed to <var>newState</var>, it
  must run the following steps:</p>

  <ol>
   <li><p>Let <var>navigables</var> be the <span>inclusive descendant navigables</span> of
   <var>traversable</var>'s <span data-x="nav-document">active document</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>navigable</var> of <var>navigables</var>
    <span data-x="" class="XXX">in what order?</span>:</p>

    <ol>
     <li><p>Let <var>document</var> be <var>navigable</var>'s <span data-x="nav-document">active
     document</span>.</p></li>

     <li><p><span>Queue a global task</span> on the <span>user interaction task source</span> given
     <var>document</var>'s <span>relevant global object</span> to <span>update the visibility
     state</span> of <var>document</var> with <var>newState</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>A <code>Document</code> has a <dfn for="Document" export>visibility state</dfn>, which is
  either "<code data-x="">hidden</code>" or "<code data-x="">visible</code>", initially set to
  "<code data-x="">hidden</code>".</p>

  <p>The <dfn attribute for="Document"><code
  data-x="dom-document-visibilityState">visibilityState</code></dfn> getter steps are to return
  <span>this</span>'s <span>visibility state</span>.</p>

  <p>The <dfn attribute for="Document"><code data-x="dom-document-hidden">hidden</code></dfn> getter
  steps are to return true if <span>this</span>'s <span>visibility state</span> is
  "<code data-x="">hidden</code>", otherwise false.</p>

  <p>To <dfn>update the visibility state</dfn> of <code>Document</code> <var>document</var> to
  <var>visibilityState</var>:</p>

  <ol>
   <li><p>If <var>document</var>'s <span>visibility state</span> equals <var>visibilityState</var>,
   then return.</p></li>

   <li><p>Set <var>document</var>'s <span>visibility state</span> to
   <var>visibilityState</var>.</p></li>

   <li><p><span data-x="queue a performance entry">Queue</span> a new
   <code>VisibilityStateEntry</code> whose
   <span data-x="VisibilityStateEntry-state">visibility state</span> is
   <var>visibilityState</var> and whose <span
   data-x="VisibilityStateEntry-timestamp">timestamp</span> is
   the <span>current high resolution time</span> given <var>document</var>'s
   <span>relevant global object</span>.</p>

   <li><p>Run the <span>screen orientation change steps</span> with <var>document</var>.
   <ref>SCREENORIENTATION</ref></p></li>

   <li><p>Run the <span>view transition page visibility change steps</span> with
   <var>document</var>.</p></li>

   <li>
    <p>Run any <dfn export>page visibility change steps</dfn> which may be defined in other
    specifications, with <span>visibility state</span> and <var>document</var>.</p>

    <p class="XXX">It would be better if specification authors sent a pull request to add calls
    from here into their specifications directly, instead of using the <span>page visibility change
    steps</span> hook, to ensure well-defined cross-specification call order. As of the time of
    this writing the following specifications are known to have <span>page visibility change
    steps</span>, which will be run in an unspecified order: <cite>Device Posture API</cite> and
    <cite>Web NFC</cite>. <ref>DEVICEPOSTURE</ref> <ref>WEBNFC</ref></p>
   </li>

   <li><p>Fire an event named <code data-x="event-visibilitychange">visibilitychange</code> at
   <var>document</var>, with its <code data-x="dom-Event-bubbles">bubbles</code> attribute
   initialized to true.</p></li>
  </ol>

  <h4>The <code>VisibilityStateEntry</code> interface</h4>

  <p>The <code>VisibilityStateEntry</code> interface exposes visibility changes to the document,
  from the moment the document becomes active.</p>

  <div class="example">For example, this allows JavaScript code in the page to examine correlation
  between visibility changes and paint timing:

  <pre><code class="js">function wasHiddenBeforeFirstContentfulPaint() {
    const fcpEntry = performance.getEntriesByName("first-contentful-paint")[0];
    const visibilityStateEntries = performance.getEntriesByType("visibility-state");
    return visibilityStateEntries.some(e =>
                                            e.startTime &lt; fcpEntry.startTime &&
                                            e.name === "hidden");
}</code></pre>
  </div>

  <p class="note">Since hiding a page can cause throttling of rendering and other user-agent
  operations, it is common to use visibility changes as an indication that such throttling has
  occurred. However, other things could also cause throttling in different browsers, such as
  long periods of inactivity.</p>

  <pre><code class="idl">[Exposed=(Window)]
interface <dfn interface>VisibilityStateEntry</dfn> : <span>PerformanceEntry</span> {
  readonly attribute DOMString <span data-x="VisibilityStateEntry-name">name</span>;                 // shadows inherited <span data-x="PerformanceEntry-name">name</span>
  readonly attribute DOMString <span data-x="VisibilityStateEntry-entryType">entryType</span>;            // shadows inherited <span data-x="PerformanceEntry-entryType">entryType</span>
  readonly attribute DOMHighResTimeStamp <span data-x="VisibilityStateEntry-startTime">startTime</span>;  // shadows inherited <span data-x="PerformanceEntry-startTime">startTime</span>
  readonly attribute unsigned long <span data-x="VisibilityStateEntry-duration">duration</span>;         // shadows inherited <span data-x="PerformanceEntry-duration">duration</span>
};</code></pre>

  <p>The <code>VisibilityStateEntry</code> has an associated
  <span><code>DOMHighResTimeStamp</code></span>
  <dfn data-x="VisibilityStateEntry-timestamp">timestamp</dfn>.</p>

  <p>The <code>VisibilityStateEntry</code> has an associated "<code data-x="">visible</code>" or
  "<code data-x="">hidden</code>" <dfn data-x="VisibilityStateEntry-state">visibility
  state</dfn>.</p>

  <p>The <dfn attribute for="VisibilityStateEntry"
  data-x="VisibilityStateEntry-name"><code>name</code></dfn> getter steps are to return
  <span>this</span>'s <span data-x="VisibilityStateEntry-state">visibility state</span>.</p>

  <p>The <dfn attribute for="VisibilityStateEntry"
  data-x="VisibilityStateEntry-entryType"><code>entryType</code></dfn> getter steps are to return
  "<code data-x="">visibility-state</code>".</p>

  <p>The <dfn attribute for="VisibilityStateEntry"
  data-x="VisibilityStateEntry-startTime"><code>startTime</code></dfn> getter steps are to return
  <span>this</span>'s <span data-x="VisibilityStateEntry-timestamp">timestamp</span>.</p>

  <p>The <dfn attribute for="VisibilityStateEntry"
  data-x="VisibilityStateEntry-duration"><code>duration</code></dfn> getter steps are to return
  zero.</p>


  <h3>Inert subtrees</h3>

  <p class="note">See also <code data-x="attr-inert">inert</code> for an explanation of the
  attribute of the same name.</p>

  <p>A node (in particular elements and text nodes) can be <dfn>inert</dfn>. When a node is
  <span>inert</span>:</p>

  <ul>
   <li><p>Hit-testing must act as if the <span>'pointer-events'</span> CSS property were set to
   'none'.</p></li>

   <li><p>Text selection functionality must act as if the <span>'user-select'</span> CSS property
   were set to 'none'.</p></li>

   <li><p>If it is <span>editable</span>, the node behaves as if it were non-editable.</p></li>

   <li><p>The user agent should ignore the node for the purposes of <span>find-in-page</span>.</p></li>
  </ul>

  <p class="note">Inert nodes generally cannot be focused, and user agents do not expose the inert
  nodes to accessibility APIs or assistive technologies. Inert nodes that are <span
  data-x="concept-command">commands</span> will become inoperable to users, in the manner described
  above.</p>

  <p>User agents may allow the user to override the restrictions on <span>find-in-page</span> and
  text selection, however.</p>

  <p>By default, a node is not <span>inert</span>.</p>

  <h4>Modal dialogs and inert subtrees</h4>

  <p>A <code>Document</code> <var>document</var> is <dfn>blocked by a modal dialog</dfn>
  <var>subject</var> if <var>subject</var> is the topmost <code>dialog</code> element in
  <var>document</var>'s <span>top layer</span>. While <var>document</var> is so blocked, every node
  that is <span>connected</span> to <var>document</var>, with the exception of the
  <var>subject</var> element and its <span>flat tree</span> descendants, must become
  <span>inert</span>.</p>

  <p><var>subject</var> can additionally become <span>inert</span> via the <code
  data-x="attr-inert">inert</code> attribute, but only if specified on <var>subject</var> itself
  (i.e., <var>subject</var> escapes inertness of ancestors); <var>subject</var>'s <span>flat
  tree</span> descendants can become <span>inert</span> in a similar fashion.</p>

  <p class="note">The <code>dialog</code> element's <code
  data-x="dom-dialog-showModal">showModal()</code> method causes this mechanism to trigger, by <span
  data-x="add an element to the top layer">adding</span> the <code>dialog</code> element to its
  <span>node document</span>'s <span>top layer</span>.</p>

  <h4>The <dfn data-x="attr-inert"><code>inert</code></dfn> attribute</h4>

  <p>The <code data-x="attr-inert">inert</code> attribute is a <span>boolean attribute</span> that
  indicates, by its presence, that the element and all its <span>flat tree</span> descendants which
  don't otherwise escape inertness (such as modal dialogs) are to be made <span>inert</span> by the
  user agent.</p>

  <p>An inert subtree should not contain any content or controls which are critical to
  understanding or using aspects of the page which are not in the inert state. Content in an inert
  subtree will not be perceivable by all users, or interactive. Authors should not specify elements
  as inert unless the content they represent are also visually obscured in some way. In most cases,
  authors should not specify the <code>inert</code> attribute on individual form controls. In these
  instances, the <code data-x="attr-fe-disabled">disabled</code> attribute is probably more
  appropriate.</p>

  <div class="example">
   <p>The following example shows how to mark partially loaded content, visually obscured by a
   "loading" message, as inert.</p>

   <pre><code class="html">&lt;section aria-labelledby=s1>
  &lt;h3 id=s1>Population by City&lt;/h3>
  &lt;div class=container>
    &lt;div class=loading>&lt;p>Loading...&lt;/p>&lt;/div>
    &lt;div inert>
      &lt;form>
        &lt;fieldset>
          &lt;legend>Date range&lt;/legend>
          &lt;div>
            &lt;label for=start>Start&lt;/label>
            &lt;input type=date id=start>
          &lt;/div>
          &lt;div>
            &lt;label for=end>End&lt;/label>
            &lt;input type=date id=end>
          &lt;/div>
          &lt;div>
            &lt;button>Apply&lt;/button>
          &lt;/div>
        &lt;/fieldset>
      &lt;/form>
      &lt;table>
        &lt;caption>From 20-- to 20--&lt;/caption>
        &lt;thead>
          &lt;tr>
            &lt;th>City&lt;/th>
            &lt;th>State&lt;/th>
            &lt;th>20-- Population&lt;/th>
            &lt;th>20-- Population&lt;/th>
            &lt;th>Percentage change&lt;/th>
          &lt;/tr>
        &lt;/thead>
        &lt;tbody>
          &lt;!-- ... -->
        &lt;/tbody>
      &lt;/table>
    &lt;/div>
  &lt;/div>
&lt;/section></code></pre>

   <img src="/images/inert-example-loading-section.png" width="947" height="243" alt="Screenshot of Population by City content with an overlaid loading message which visually obscures the form controls and data table which have not fully rendered, and thus are in the inert state.">

   <p>The "loading" overlay obscures the inert content, making it visually apparent that the inert
   content is not presently accessible. Notice that the heading and "loading" text are not
   descendants of the element with the <code>inert</code> attribute. This will ensure this text is
   accessible to all users, while the inert content cannot be interacted with by anyone.</p>
  </div>

  <div class="note">
   <p>By default, there is no persistent visual indication of an element or its subtree being
   inert. Appropriate visual styles for such content is often context-dependent. For instance, an
   inert off-screen navigation panel would not require a default style, as its off-screen position
   visually obscures the content. Similarly, a modal <code>dialog</code> element's backdrop will
   serve as the means to visually obscure the inert content of the web page, rather than styling
   the inert content specifically.</p>

   <p>However, for many other situations authors are strongly 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 using particularly
   small windows might not be able to see the active part of a page and might get frustrated if
   inert sections are not obviously inert.</p>
  </div>

  <p>The <dfn attribute for="HTMLElement" data-x="dom-inert"><code>inert</code></dfn> IDL attribute
  must <span>reflect</span> the content attribute of the same name.</p>

  <h3>Tracking <dfn export>user activation</dfn></h3>

  <p>To prevent abuse of certain APIs that could be annoying to users (e.g., opening popups or
  vibrating phones), user agents allow these APIs only when the user is actively interacting with
  the web page or has interacted with the page at least once. This "active interaction" state is
  maintained through the mechanisms defined in this section.</p>

  <div w-nodev>

  <h4 id="user-activation-data-model">Data model</h4>

  <p>For the purpose of tracking user activation, each <code>Window</code> <var>W</var> has two
  relevant values:</p>

  <ul>
   <li><p>A <dfn>last activation timestamp</dfn>, which is either a
   <code>DOMHighResTimeStamp</code>, positive infinity (indicating that <var>W</var> has never been
   activated), or negative infinity (indicating that the activation has been <span data-x="consume
   user activation">consumed</span>). Initially positive infinity.</p></li>

   <li><p>A <dfn>last history-action activation timestamp</dfn>, which is either a
   <code>DOMHighResTimeStamp</code> or positive infinity, initially positive infinity.</p></li>
  </ul>

  <p>A user agent also defines a <dfn>transient activation duration</dfn>, which is a constant
  number indicating how long a user activation is available for certain <a
  href="#user-activation-gated-apis">user activation-gated APIs</a> (e.g., for opening popups).</p>

  <p class="note">The <span>transient activation duration</span> is expected be at most a few
  seconds, so that the user can possibly perceive the link between an interaction with the page and
  the page calling the activation-gated API.</p>

  <p>We then have the following boolean user activation states for <var>W</var>:</p>

  <dl>
   <dt><dfn export>Sticky activation</dfn></dt>
   <dd>
    <p>When the <span>current high resolution time</span> given <var>W</var> is greater than or
    equal to the <span>last activation timestamp</span> in <var>W</var>, <var>W</var> is said to
    have <span>sticky activation</span>.</p>

    <p>This is <var>W</var>'s historical activation state, indicating whether the user has ever
    interacted in <var>W</var>. It starts false, then changes to true (and never changes back to
    false) when <var>W</var> gets the very first <span>activation notification</span>.</p>
   </dd>

   <dt><dfn export>Transient activation</dfn></dt>
   <dd>
    <p>When the <span>current high resolution time</span> given <var>W</var> is greater than or
    equal to the <span>last activation timestamp</span> in <var>W</var>, and less than the
    <span>last activation timestamp</span> in <var>W</var> plus the <span>transient activation
    duration</span>, then <var>W</var> is said to have <span>transient activation</span>.</p>

    <p>This is <var>W</var>'s current activation state, indicating whether the user has interacted
    in <var>W</var> recently. This starts with a false value, and remains true for a limited time
    after every <span>activation notification</span> <var>W</var> gets.</p>

    <p>The <span>transient activation</span> state is considered <dfn
    data-x="activation-expiry">expired</dfn> if it becomes false because the <span>transient
    activation duration</span> time has elapsed since the last user activation. Note that it can
    become false even before the expiry time through an <span data-x="consume user
    activation">activation consumption</span>.</p>
   </dd>

   <dt><dfn>History-action activation</dfn></dt>
   <dd>
    <p>When the <span>last history-action activation timestamp</span> of <var>W</var> is not equal
    to the <span>last activation timestamp</span> of <var>W</var>, then <var>W</var> is said to have
    <span>history-action activation</span>.</p>

    <p>This is a special variant of user activation, used to allow access to certain session history
    APIs which, if used too frequently, would make it harder for the user to traverse back using <a
    href="#nav-traversal-ui">browser UI</a>. It starts with a false value, and becomes true whenever
    the user interacts with <var>W</var>, but is reset to false through <span data-x="consume
    history-action user activation">history-action activation consumption</span>. This ensures such
    APIs cannot be used multiple times in a row without an intervening user activation. But unlike
    <span>transient activation</span>, there is no time limit within which such APIs must be
    used.</p>
   </dd>
  </dl>

  <p class="note">The <span>last activation timestamp</span> and <span>last history-action
  activation timestamp</span> are retained even after the <code>Document</code> changes its
  <span>fully active</span> status (e.g., after navigating away from a <code>Document</code>, or
  navigating to a cached <code>Document</code>). This means <span>sticky activation</span> state
  spans multiple navigations as long as the same <code>Document</code> gets reused. For the
  transient activation state, the original <span data-x="activation-expiry">expiry</span> time
  remains unchanged (i.e., the state still expires within the <span>transient activation
  duration</span> limit from the original <span>activation triggering input event</span>). It is
  important to consider this when deciding whether to base certain things off <span>sticky
  activation</span> or <span>transient activation</span>.</p>

  <h4 id="user-activation-processing-model">Processing model</h4>

  <p>When a user interaction causes firing of an <span>activation triggering input
  event</span> in a <code>Document</code> <var>document</var>, the user agent must perform the
  following <dfn>activation notification</dfn> steps <em>before</em> <span
  data-x="concept-event-dispatch">dispatching</span> the event:</p>

  <ol>
   <li><p><span>Assert</span>: <var>document</var> is <span>fully active</span>.</p></li>

   <li><p>Let <var>windows</var> be « <var>document</var>'s <span>relevant global object</span>
   ».</p></li>

   <li><p><span data-x="list extend">Extend</span> <var>windows</var> with the <span
   data-x="nav-window">active window</span> of each of <var>document</var>'s <span>ancestor
   navigables</span>.</p></li>

   <li><p><span data-x="list extend">Extend</span> <var>windows</var> with the <span
   data-x="nav-window">active window</span> of each of <var>document</var>'s <span>descendant
   navigables</span>, filtered to include only those <span data-x="navigable">navigables</span>
   whose <span data-x="nav-document">active document</span>'s <span
   data-x="concept-document-origin">origin</span> is <span>same origin</span> with
   <var>document</var>'s <span data-x="concept-document-origin">origin</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>window</var> in <var>windows</var>:</p>

    <ol>
     <li><p>Set <var>window</var>'s <span>last activation timestamp</span> to the <span>current
     high resolution time</span>.</p></li>

     <li><p><span>Notify the close watcher manager about user activation</span> given
     <var>window</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>An <dfn>activation triggering input event</dfn> is any event whose <code
  data-x="dom-Event-isTrusted">isTrusted</code> attribute is true and whose <code
  data-x="dom-Event-type">type</code> is one of:</p>

  <ul>
   <li><p>"<code data-x="event-keydown">keydown</code>", provided the key is neither the
   <kbd>Esc</kbd> key nor a shortcut key reserved by the user agent;</li>

   <li><p>"<code data-x="event-mousedown">mousedown</code>";</p></li>

   <li><p>"<code data-x="event-pointerdown">pointerdown</code>", provided the event's
   <code>pointerType</code> is "<code data-x="">mouse</code>";</p></li>

   <li><p>"<code data-x="event-pointerup">pointerup</code>", provided the event's
   <code>pointerType</code> is not "<code data-x="">mouse</code>"; or</p></li>

   <li><p>"<code data-x="event-touchend">touchend</code>".</p></li>
  </ul>

  <p><span data-x="activation consuming API">Activation consuming APIs</span> defined in this and
  other specifications can <dfn export>consume user activation</dfn> by performing the following
  steps, given a <code>Window</code> <var>W</var>:</p>

  <ol>
   <li><p>If <var>W</var>'s <span data-x="window navigable">navigable</span> is null, then
   return.</p></li>

   <li><p>Let <var>top</var> be <var>W</var>'s <span data-x="window navigable">navigable</span>'s
   <span data-x="nav-top">top-level traversable</span>.</p></li>

   <li><p>Let <var>navigables</var> be the <span>inclusive descendant navigables</span> of
   <var>top</var>'s <span data-x="nav-document">active document</span>.</p></li>

   <li><p>Let <var>windows</var> be the list of <code>Window</code> objects constructed by taking
   the <span data-x="nav-window">active window</span> of each item in
   <var>navigables</var>.</p></li>

   <li><p><span data-x="list iterate">For each</span> <var>window</var> in <var>windows</var>, if
   <var>window</var>'s <span>last activation timestamp</span> is not positive infinity, then set
   <var>window</var>'s <span>last activation timestamp</span> to negative infinity.</p></li>
  </ol>

  <p><span data-x="history-action activation-consuming API">History-action activation-consuming
  APIs</span> can <dfn>consume history-action user activation</dfn> by performing the following
  steps, given a <code>Window</code> <var>W</var>:</p>

  <ol>
   <li><p>If <var>W</var>'s <span data-x="window navigable">navigable</span> is null, then
   return.</p></li>

   <li><p>Let <var>top</var> be <var>W</var>'s <span data-x="window navigable">navigable</span>'s
   <span data-x="nav-top">top-level traversable</span>.</p></li>

   <li><p>Let <var>navigables</var> be the <span>inclusive descendant navigables</span> of
   <var>top</var>'s <span data-x="nav-document">active document</span>.</p></li>

   <li><p>Let <var>windows</var> be the list of <code>Window</code> objects constructed by taking
   the <span data-x="nav-window">active window</span> of each item in
   <var>navigables</var>.</p></li>

   <li><p><span data-x="list iterate">For each</span> <var>window</var> in <var>windows</var>, set
   <var>window</var>'s <span>last history-action activation timestamp</span> to <var>window</var>'s
   <span>last activation timestamp</span>.</p></li>
  </ol>

  <p class="note">Note the asymmetry in the sets of <span data-x="browsing context">browsing
  contexts</span> in the page that are affected by an <span>activation notification</span> vs an
  <span data-x="consume user activation">activation consumption</span>: an activation consumption
  changes (to false) the <span>transient activation</span> states for all browsing contexts in the
  page, but an activation notification changes (to true) the states for a subset of those browsing
  contexts. The exhaustive nature of consumption here is deliberate: it prevents malicious sites
  from making multiple calls to an <span>activation consuming API</span> from a single user
  activation (possibly by exploiting a deep hierarchy of <code>iframe</code>s).</p>

  <h4 id="user-activation-gated-apis">APIs gated by user activation</h4>

  </div>

  <p>APIs that are dependent on user activation are classified into different levels:</p>

  <dl>
   <dt><dfn export data-x="sticky activation gated api" data-lt="sticky activation-gated API">Sticky activation-gated APIs</dfn></dt>
   <dd><p>These APIs require the <span>sticky activation</span> state to be true, so they are
   blocked until the very first user activation.</p></dd>

   <dt><dfn export data-x="transient activation gated api" data-lt="transient activation-gated
   API">Transient activation-gated APIs</dfn></dt>
   <dd><p>These APIs require the <span>transient activation</span> state to be true, but they don't
   <span data-x="consume user activation">consume</span> it, so multiple calls are allowed per user
   activation until the transient state <span data-x="activation-expiry">expires</span>.</p></dd>

   <dt><dfn export data-x="activation consuming api" data-lt="transient activation-consuming API">Transient activation-consuming APIs</dfn></dt>
   <dd><p>These APIs require the <span>transient activation</span> state to be true, and they
   <span>consume user activation</span> in each call to prevent multiple calls per user
   activation.</p></dd>

   <dt><dfn export data-x="history-action activation-consuming API" data-lt="history-action activation-consuming API">History-action activation-consuming APIs</dfn></dt>
   <dd><p>These APIs require the <span>history-action activation</span> state to be true, and they
   <span>consume history-action user activation</span> in each call to prevent multiple calls per
   user activation.</p></dd>
  </dl>


  <h4>The <code>UserActivation</code> interface</h4>

  <p>Each <code>Window</code> has an <dfn>associated <code>UserActivation</code></dfn>, which is a
  <code>UserActivation</code> object. Upon creation of the <code>Window</code> object, its
  <span>associated <code>UserActivation</code></span> must be set to a <span>new</span>
  <code>UserActivation</code> object created in the <code>Window</code> object's <span
  data-x="concept-relevant-realm">relevant realm</span>.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>UserActivation</dfn> {
  readonly attribute boolean <span data-x="dom-UserActivation-hasBeenActive">hasBeenActive</span>;
  readonly attribute boolean <span data-x="dom-UserActivation-isActive">isActive</span>;
};

partial interface <span id="NavigatorUserActivation-partial">Navigator</span> {
  [SameObject] readonly attribute <code>UserActivation</code> <span data-x="dom-Navigator-userActivation">userActivation</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-Navigator-userActivation">userActivation</span>.<span subdfn data-x="dom-UserActivation-hasBeenActive">hasBeenActive</span></code></dt>
   <dd>
    <p>Returns whether the window has <span>sticky activation</span>.</p>
   </dd>

   <dt><code data-x=""><span data-x="dom-navigator">navigator</span>.<span data-x="dom-Navigator-userActivation">userActivation</span>.<span subdfn data-x="dom-UserActivation-isActive">isActive</span></code></dt>
   <dd>
    <p>Returns whether the window has <span>transient activation</span>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="Navigator"><code
  data-x="dom-Navigator-userActivation">userActivation</code></dfn> getter steps are to return
  <span>this</span>'s <span>relevant global object</span>'s <span>associated
  <code>UserActivation</code></span>.</p>

  <p>The <dfn attribute for="UserActivation"><code
  data-x="dom-UserActivation-hasBeenActive">hasBeenActive</code></dfn> getter steps are to return
  true if <span>this</span>'s <span>relevant global object</span> has <span>sticky
  activation</span>, and false otherwise.</p>

  <p>The <dfn attribute for="UserActivation"><code
  data-x="dom-UserActivation-isActive">isActive</code></dfn> getter steps are to return true if
  <span>this</span>'s <span>relevant global object</span> has <span>transient activation</span>,
  and false otherwise.</p>


  <h4 id="user-activation-user-agent-automation">User agent automation</h4>

  <p>For the purposes of user-agent automation and application testing, this specification defines
  the following <span>extension command</span> for the <cite>Web Driver</cite> specification. It is
  optional for a user agent to support the following <span>extension command</span>.
  <ref>WEBDRIVER</ref></p>

  <table class="data">
   <thead>
    <tr>
     <th>HTTP Method</th>
     <th>URI Template</th>
    </tr>
   </thead>
   <tbody>
    <tr>
     <td>`<code data-x="">POST</code>`</td>
     <td><code data-x="">/session/{<var>session id</var>}/window/consume-user-activation</code></td>
    </tr>
   </tbody>
  </table>

  <p>The <span>remote end steps</span> are:</p>

  <ol>
   <li><p>Let <var>window</var> be <span data-x="webdriver-current-browsing-context">current
   browsing context</span>'s <span>active window</span>.</p></li>

   <li><p>Let <var>consume</var> be true if <var>window</var> has <span>transient activation</span>;
   otherwise false.</p></li>

   <li><p>If <var>consume</var> is true, then <span>consume user activation</span> of
   <var>window</var>.</p></li>

   <li><p>Return <span data-x="success-value">success</span> with data <var>consume</var>.</p></li>
  </ol>

  </div>


  <h3 id="activation">Activation behavior of elements</h3>

  <p>Certain elements in HTML have an <span>activation behavior</span>, which means that the user
  can activate them. This is always caused by a <code data-x="event-click">click</code> event.</p>

  <div w-nodev>

  <p>The user agent should allow the user to manually trigger elements that have an <span>activation
  behavior</span>, for instance using keyboard or voice input, or through mouse clicks. When the
  user triggers an element with a defined <span>activation behavior</span> in a manner other than
  clicking it, the default action of the interaction event must be to <span>fire a <code
  data-x="event-click">click</code> event</span> at the element.</p>

  </div>

  <!-- v2 idea: HTMLImageElement.click(x, y); or clickPoint(), if click() can't be done in IE; can
       this be emulated in IE by posting a synthetic pointer click event with those X and Y coords?
       (ack Csaba Gabor)
  -->

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-click">click</span>()</code></dt>

   <dd><p>Acts as if the element was clicked.</p></dd>
  </dl>

  <div w-nodev>

  <p>Each element has an associated <dfn>click in progress flag</dfn>, which is initially unset.</p>

  <p>The <dfn method for="HTMLElement"><code data-x="dom-click">click()</code></dfn> method must run
  the following steps:</p>

  <ol>
   <li><p>If this element is a form control that is <span
   data-x="concept-fe-disabled">disabled</span>, then return.</p></li>

   <li><p>If this element's <span>click in progress flag</span> is set, then return.</p></li>

   <li><p>Set this element's <span>click in progress flag</span>.</p></li>

   <li><p><span>Fire a synthetic pointer event</span> named <code data-x="event-click">click</code>
   at this element, with the <var>not trusted flag</var> set.</p></li>

   <li><p>Unset this element's <span>click in progress flag</span>.</p></li>
  </ol>

  </div>

  <h4>The <code>ToggleEvent</code> interface</h4>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>ToggleEvent</dfn> : <span>Event</span> {
  constructor(DOMString type, optional <span>ToggleEventInit</span> eventInitDict = {});
  readonly attribute DOMString <span data-x="dom-ToggleEvent-oldState">oldState</span>;
  readonly attribute DOMString <span data-x="dom-ToggleEvent-newState">newState</span>;
};

dictionary <dfn dictionary>ToggleEventInit</dfn> : <span>EventInit</span> {
  DOMString oldState = "";
  DOMString newState = "";
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>event</var>.<span subdfn
   data-x="dom-ToggleEvent-oldState">oldState</span></code></dt>

   <dd>
    <p>Set to "<code data-x="">closed</code>" when transitioning from closed to open, or set to
    "<code data-x="">open</code>" when transitioning from open to closed.</p>
   </dd>

   <dt><code data-x=""><var>event</var>.<span subdfn
   data-x="dom-ToggleEvent-newState">newState</span></code></dt>

   <dd>
    <p>Set to "<code data-x="">open</code>" when transitioning from closed to open, or set to "<code
    data-x="">closed</code>" when transitioning from open to closed.</p>
   </dd>
  </dl>

  <p>The <dfn attribute for="ToggleEvent"><code
  data-x="dom-ToggleEvent-oldState">oldState</code></dfn> and <dfn attribute for="ToggleEvent"><code
  data-x="dom-ToggleEvent-newState">newState</code></dfn> attributes must return the values they are
  initialized to.</p>

  <p>A <dfn>toggle task tracker</dfn> is a <span>struct</span> which has:</p>

  <dl>
   <dt><dfn data-x="toggle-task-task">task</dfn></dt>
   <dd>A <span data-x="concept-task">task</span> which fires a <code>ToggleEvent</code>.</dd>

   <dt><dfn data-x="toggle-task-old-state">old state</dfn></dt>
   <dd>A string which represents the <span data-x="toggle-task-task">task</span>'s event's value for
   the <code data-x="dom-ToggleEvent-oldState">oldState</code> attribute.</dd>
  </dl>

  <h4>The <code>CommandEvent</code> interface</h4>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>CommandEvent</dfn> : <span>Event</span> {
  constructor(DOMString type, optional <span>CommandEventInit</span> eventInitDict = {});
  readonly attribute Element? <span data-x="dom-CommandEvent-source">source</span>;
  readonly attribute DOMString <span data-x="dom-CommandEvent-command">command</span>;
};

dictionary <dfn dictionary>CommandEventInit</dfn> : <span>EventInit</span> {
  Element? source = null;
  DOMString command = "";
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>event</var>.<span subdfn
   data-x="dom-CommandEvent-command">command</span></code></dt>

   <dd>
    <p>Returns what action the element can take.</p>
   </dd>

   <dt><code data-x=""><var>event</var>.<span subdfn
   data-x="dom-CommandEvent-source">source</span></code></dt>

   <dd>
    <p>Returns the <code>Element</code> that was interacted with in order to cause this event.</p>
   </dd>
  </dl>

  <p>The <dfn attribute for="CommandEvent"><code
  data-x="dom-CommandEvent-command">command</code></dfn> attribute must return the value it was
  initialized to.</p>

  <p>The <dfn attribute
  for="CommandEvent"><code data-x="dom-CommandEvent-source">source</code></dfn> getter steps are to
  return the result of <span data-x="dom-retarget">retargeting</span> <code
  data-x="dom-CommandEvent-source">source</code> against <span>this</span>'s <code
  data-x="dom-Event-currentTarget">currentTarget</code>.</p>

  <p class="XXX"><a href="https://github.com/whatwg/dom/issues/1328">DOM standard issue #1328</a>
  tracks how to better standardize associated event data in a way which makes sense on Events.
  Currently an event attribute initialized to a value cannot also have a getter, and so an internal
  slot (or map of additional fields) is required to properly specify this.</p>

  <h3>Focus</h3>

  <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2807 -->
  <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2811 -->
  <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2809 -->
  <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2812 -->

  <!-- v2: more things to define, if no other specs define them:
   - define onfocus/onblur behavior for Window
   - Other things to look at are IE's focus APIs (HTMLElement.setActive(),
     onBeforeActivate, onActivate, onBeforeDeactivate, onDeactivate):
       https://bugzilla.mozilla.org/show_bug.cgi?id=296471
       https://bugzilla.mozilla.org/show_bug.cgi?id=296469
       http://msdn.microsoft.com/workshop/author/dhtml/reference/methods/setactive.asp
       http://msdn.microsoft.com/workshop/author/dhtml/reference/events/onbeforeactivate.asp
  -->

  <h4>Introduction</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>An HTML user interface typically consists of multiple interactive widgets, such as form
  controls, scrollable regions, links, dialog boxes, browser tabs, and so forth. These widgets form
  a hierarchy, with some (e.g. browser tabs, dialog boxes) containing others (e.g. links, form
  controls).</p>

  <p>When interacting with an interface using a keyboard, key input is channeled from the system,
  through the hierarchy of interactive widgets, to an active widget, which is said to be
  <span>focused</span>.</p>

  <div class="example">

   <p>Consider an HTML application running in a browser tab running in a graphical environment.
   Suppose this application had a page with some text controls and links, and was currently showing
   a modal dialog, which itself had a text control and a button.</p>

   <p>The hierarchy of focusable widgets, in this scenario, would include the browser window, which
   would have, amongst its children, the browser tab containing the HTML application. The tab itself
   would have as its children the various links and text controls, as well as the dialog. The dialog
   itself would have as its children the text control and the button.</p>

   <p><img src="/images/focus-tree.png" alt="" width=800 height=450> <!-- purely decorative: it repeats the previous
   paragraph, in graphical form -->

   <p>If the widget with <span data-x="focused">focus</span> in this example was the text control in
   the dialog box, then key input would be channeled from the graphical system to &#x2460; the web
   browser, then to &#x2461; the tab, then to &#x2462; the dialog, and finally to &#x2463; the text
   control.</p>

  </div>

  <p>Keyboard <em>events</em> are always targeted at this <span>focused</span> element.</p>


  <h4>Data model</h4>

  <p id="tlbc-system-focus">A <span>top-level traversable</span> has <dfn export for="top-level
  traversable">system focus</dfn> when it can receive keyboard input channeled from the operating
  system, possibly targeted at one of its <span data-x="nav-document">active document</span>'s
  <span>descendant navigables</span>.</p>

  <p>A <span>top-level traversable</span> has <dfn for="top-level traversable">user attention</dfn>
  when its <span>system visibility state</span> is "<code data-x="">visible</code>", and it either
  has <span>system focus</span> or user agent widgets directly related to it can receive keyboard
  input channeled from the operating system.</p>
  <!-- This language is borrowed from system focus -->

  <p class="note">User attention is lost when a browser window loses focus, whereas system focus
  might also be lost to other system widgets in the browser window such as a location bar.</p>

  <p>A <code>Document</code> <var>d</var> is a <dfn export for="Document">fully active descendant of
  a top-level traversable with user attention</dfn> when <var>d</var> is <span>fully active</span>
  and <var>d</var>'s <span>node navigable</span>'s <span data-x="nav-top">top-level
  traversable</span> has <span>user attention</span>.</p>

  <p>The term <dfn>focusable area</dfn> is used to refer to regions of the interface that can further become
  the target of such keyboard input. Focusable areas can be elements, parts of elements, or other regions
  managed by the user agent.</p>

  <p>Each <span>focusable area</span> has a <dfn>DOM anchor</dfn>, which is a <code>Node</code> object
  that represents the position of the <span>focusable area</span> in the DOM. (When the <span>focusable
  area</span> is itself a <code>Node</code>, it is its own <span>DOM anchor</span>.) The <span>DOM anchor</span> is
  used in some APIs as a substitute for the <span>focusable area</span> when there is no other DOM object
  to represent the <span>focusable area</span>.</p>

  <p>The following table describes what objects can be <span data-x="focusable area">focusable
  areas</span>. The cells in the left column describe objects that can be <span data-x="focusable
  area">focusable areas</span>; the cells in the right column describe the <span data-x="DOM anchor">DOM
  anchors</span> for those elements. (The cells that span both columns are non-normative examples.)</p>

  <table id="table-fa">
   <thead>
    <tr>
     <th id="th-fa-area"><span>Focusable area</span>
     <th id="th-fa-dom-rep"><span>DOM anchor</span>
    <tr>
     <th id="th-fa-examples" colspan=2>Examples

   <tbody>
    <tr>
     <th class="data-header" headers="th-fa-area" id="td-fa-1">
     Elements that meet all the following criteria:

     <ul class="brief">
      <li>the element's <span>tabindex value</span> is non-null, or the element is determined by the
      user agent to be focusable;</li>

      <li>the element is either not a <span>shadow host</span>, or has a <span
      data-x="concept-element-shadow-root">shadow root</span> whose <span>delegates focus</span> is
      false;</li>

      <li>the element is not <span data-x="concept-element-disabled">actually disabled</span>;</li>

      <li>the element is not <span>inert</span>;</li>

      <li>the element is either <span>being rendered</span>, <span>delegating its rendering to its
      children</span>, or <span>being used as relevant canvas fallback content</span>.</li>
     </ul>
     <td headers="td-fa-1 th-fa-dom-rep">
     The element itself.
    <tr>
     <td headers="td-fa-1 th-fa-examples" colspan=2>
     <p class="example"><code>iframe</code>, <code>dialog</code>, <code
     data-x="attr-input-type-text">&lt;input type=text></code>, sometimes <code data-x="a">&lt;a
     href=""></code> (depending on platform conventions).

   <tbody>
    <tr>
     <th class="data-header" headers="th-fa-area" id="td-fa-2">
     The shapes of <code>area</code> elements in an <span>image map</span> associated with an
     <code>img</code> element that is <span>being rendered</span> and is not <span>inert</span>.
     <td headers="td-fa-2 th-fa-dom-rep">
     The <code>img</code> element.
    <tr>
     <td headers="td-fa-2 th-fa-examples" colspan=2>
     <div class="example">
      <p>In the following example, the <code>area</code> element creates two shapes, one on each
      image. The <span>DOM anchor</span> of the first shape is the first <code>img</code> element, and the
      <span>DOM anchor</span> of the second shape is the second <code>img</code> element.</p>
      <pre><code class="html">&lt;map id=wallmap>&lt;area alt="Enter Door" coords="10,10,100,200" href="door.html">&lt;/map>
...
&lt;img src="images/innerwall.jpeg" alt="There is a white wall here, with a door." usemap="#wallmap">
...
&lt;img src="images/outerwall.jpeg" alt="There is a red wall here, with a door." usemap="#wallmap"></code></pre>
     </div>

   <tbody>
    <tr>
     <th class="data-header" headers="th-fa-area" id="td-fa-3">
     The user-agent provided subwidgets of elements that are <span>being rendered</span> and are not
     <span data-x="concept-element-disabled">actually disabled</span> or <span>inert</span>.
     <td headers="td-fa-3 th-fa-dom-rep">
     The element for which the <span>focusable area</span> is a subwidget.
    <tr>
     <td headers="td-fa-3 th-fa-examples" colspan=2>
     <p class="example">The <span data-x="expose a user interface to the user">controls in the user
     interface</span> for a <code>video</code> element, the up and down buttons in a spin-control
     version of <code data-x="attr-input-type-number">&lt;input type=number></code>, the part of a
     <code>details</code> element's rendering that enables the element to be opened or closed using
     keyboard input.</p>

   <tbody>
    <tr>
     <th class="data-header" headers="th-fa-area" id="td-fa-4">
     The scrollable regions of elements that are <span>being rendered</span> and are not
     <span>inert</span>. <!-- the being rendered part is kinda redundant, a scrollable region is a
     box generated for the element so by definition if the element has a scrollable region it is
     being rendered -->
     <td headers="td-fa-4 th-fa-dom-rep">
     The element for which the box that the scrollable region scrolls was created.
    <tr>
     <td headers="td-fa-4 th-fa-examples" colspan=2>
     <p class="example">The CSS <span>'overflow'</span> property's 'scroll' value typically creates
     a scrollable region.</p>

   <tbody>
    <tr>
     <th class="data-header" headers="th-fa-area" id="td-fa-5">
     The <span>viewport</span> of a <code>Document</code> that has a non-null <span
     data-x="concept-document-bc">browsing context</span> and is not <span>inert</span>.
     <td headers="td-fa-5 th-fa-dom-rep">
     The <code>Document</code> for which the <span>viewport</span> was created.
    <tr>
     <td headers="td-fa-5 th-fa-examples" colspan=2>
     <p class="example">The contents of an <code>iframe</code>.</p>

   <tbody>
    <tr>
     <th class="data-header" headers="th-fa-area" id="td-fa-6">
     Any other element or part of an element determined by the user agent to be a focusable area,
     especially to aid with accessibility or to better match platform conventions.
     <td headers="td-fa-6 th-fa-dom-rep">
     The element.
    <tr>
     <td headers="td-fa-6 th-fa-examples" colspan=2>
     <p class="example">A user agent could make all list item bullets <span>sequentially
     focusable</span>, so that a user can more easily navigate lists.</p>
     <p class="example">Similarly, a user agent could make all elements with <code
     data-x="attr-title">title</code> attributes <span>sequentially focusable</span>, so that their
     advisory information can be accessed.</p>

  </table>

  <p id="bc-focus-ergo-bcc-focus" class="note">A <span>navigable container</span> (e.g. an
  <code>iframe</code>) is a <span>focusable area</span>, but key events routed to a <span>navigable
  container</span> get immediately routed to its <span>content navigable</span>'s <span
  data-x="nav-document">active document</span>. Similarly, in sequential focus navigation a
  <span>navigable container</span> essentially acts merely as a placeholder for its <span>content
  navigable</span>'s <span data-x="nav-document">active document</span>.</p>

  <hr>

  <p>One <span>focusable area</span> in each <code>Document</code> is designated the <dfn>focused
  area of the document</dfn>. Which control is so designated changes over time, based on algorithms
  in this specification.</p>

  <p class="note">Even if a document is not <span>fully active</span> and not shown to the user, it
  can still have a <span>focused area of the document</span>. If a document's <span>fully
  active</span> state changes, its <span>focused area of the document</span> will stay the same.</p>

  <p id="currently-focused-area-of-a-top-level-browsing-context">The <dfn>currently focused area of
  a top-level traversable</dfn> <var>traversable</var> is the <span>focusable area</span>-or-null
  returned by this algorithm:</p>

  <ol>
   <li><p>If <var>traversable</var> does not have <span>system focus</span>, then return
   null.</p></li>

   <li><p>Let <var>candidate</var> be <var>traversable</var>'s <span data-x="nav-document">active
   document</span>.</p></li>

   <li><p>While <var>candidate</var>'s <span data-x="focused area of the document">focused
   area</span> is a <span>navigable container</span> with a non-null <span>content
   navigable</span>: set <var>candidate</var> to the <span data-x="nav-document">active
   document</span> of that <span>navigable container</span>'s <span>content
   navigable</span>.</p></li>

   <li><p>If <var>candidate</var>'s <span data-x="focused area of the document">focused area</span>
   is non-null, set <var>candidate</var> to <var>candidate</var>'s <span data-x="focused area of
   the document">focused area</span>.</p></li>
   <!-- Can it be null though? See https://github.com/whatwg/html/issues/5835. -->

   <li><p>Return <var>candidate</var>.</p></li>
  </ol>

  <p id="current-focus-chain-of-a-top-level-browsing-context">The <dfn>current focus chain of a
  top-level traversable</dfn> <var>traversable</var> is the <span>focus chain</span> of the <span
  data-x="currently focused area of a top-level traversable">currently focused area</span> of
  <var>traversable</var>, if <var>traversable</var> is non-null, or an empty list otherwise.</p>

  <p>An element that is the <span>DOM anchor</span> of a <span>focusable area</span> is said to
  <dfn data-x="gains focus">gain focus</dfn> when that <span>focusable area</span> becomes the
  <span>currently focused area of a top-level traversable</span>. When an element is the <span>DOM
  anchor</span> of a <span>focusable area</span> of the <span>currently focused area of a top-level
  traversable</span>, it is <dfn>focused</dfn>.</p>

  <div w-nodev>

  <p>The <dfn>focus chain</dfn> of a <span>focusable area</span> <var>subject</var> is the ordered
  list constructed as follows:</p>

  <ol>
   <li><p>Let <var>output</var> be an empty <span>list</span>.</p></li>

   <li><p>Let <var>currentObject</var> be <var>subject</var>.</p></li>

   <li>
    <p>While true:</p>

    <ol>
     <li><p><span data-x="list append">Append</span> <var>currentObject</var> to
     <var>output</var>.</p></li>

     <li>
      <p>If <var>currentObject</var> is an <code>area</code> element's shape, then <span
      data-x="list append">append</span> that <code>area</code> element to <var>output</var>.</p>

      <p>Otherwise, if <var>currentObject</var>'s <span>DOM anchor</span> is an element that is not
      <var>currentObject</var> itself, then <span data-x="list append">append</span>
      <var>currentObject</var>'s <span>DOM anchor</span> to <var>output</var>.</p>
     </li>

     <li>
      <p>If <var>currentObject</var> is a <span>focusable area</span>, then set
      <var>currentObject</var> to <var>currentObject</var>'s <span>DOM anchor</span>'s <span>node
      document</span>.</p>

      <p>Otherwise, if <var>currentObject</var> is a <code>Document</code> whose <span>node
      navigable</span>'s <span data-x="nav-parent">parent</span> is non-null, then set
      <var>currentObject</var> to <var>currentObject</var>'s <span>node navigable</span>'s <span
      data-x="nav-parent">parent</span>.</p>

      <p>Otherwise, <span>break</span>.</p>
     </li>
    </ol>
   </li>

   <li>
    <p>Return <var>output</var>.</p>

    <p class="note">The chain starts with <var>subject</var> and (if <var>subject</var> is or can be
    the <span>currently focused area of a top-level traversable</span>) continues up the focus
    hierarchy up to the <code>Document</code> of the <span>top-level traversable</span>.</p>
   </li>
  </ol>

  <p>All elements that are <span data-x="focusable area">focusable areas</span> are said to be
  <dfn>focusable</dfn>.</p>

  <p>There are two special types of focusability for <span data-x="focusable area">focusable
  areas</span>:</p>
  <ul>
   <li>
    <p>A <span>focusable area</span> is said to be <dfn>sequentially focusable</dfn> if it is
    included in its <code>Document</code>'s <span>sequential focus navigation order</span> and
    the user agent determines that it is sequentially focusable.</p>
   </li>
   <li>
    <p>A <span>focusable area</span> is said to be <dfn>click focusable</dfn> if the user agent
    determines that it is click focusable. User agents should consider focusable areas with non-null
    <span data-x="tabindex value">tabindex values</span> to be click focusable.</p>
   </li>
  </ul>

  <p class="note">Elements which are not <span>focusable</span> are not <span data-x="focusable
  area">focusable areas</span>, and thus not <span>sequentially focusable</span> and not <span>click
  focusable</span>.</p>

  <div class="note">
   <p>Being <span>focusable</span> is a statement about whether an element can be focused
   programmatically, e.g. via the <code data-x="dom-focus">focus()</code> method or <code
   data-x="attr-fe-autofocus">autofocus</code> attribute. In contrast, <span>sequentially
   focusable</span> and <span>click focusable</span> govern how the user agent responds to user
   interaction: respectively, to <span>sequential focus navigation</span> and as
   <span>activation behavior</span>.</p>

   <p>The user agent might determine that an element is not <span>sequentially focusable</span> even
   if it is <span>focusable</span> and is included in its <code>Document</code>'s <span>sequential
   focus navigation order</span>, according to user preferences. For example, macOS users can set
   the user agent to skip non-form control elements, or can skip links when doing <span>sequential
   focus navigation</span> with just the <kbd>Tab</kbd> key (as opposed to using both the
   <kbd>Option</kbd> and <kbd>Tab</kbd> keys).</p>

   <p>Similarly, the user agent might determine that an element is not <span>click focusable</span>
   even if it is <span>focusable</span>. For example, in some user agents, clicking on a
   non-editable form control does not focus it, i.e. the user agent has determined that such
   controls are not click focusable.</p>

   <p>Thus, an element can be <span>focusable</span>, but neither <span>sequentially
   focusable</span> nor <span>click focusable</span>. For example, in some user agents, a
   non-editable form-control with a negative-integer <span>tabindex value</span> would not be
   focusable via user interaction, only via programmatic APIs.</p>
  </div>

  <p>When a user <a href="#activation">activates</a> a <span>click focusable</span> <span>focusable
  area</span>, the user agent must run the <span>focusing steps</span> on the <span>focusable
  area</span> with <var>focus trigger</var> set to "<code data-x="">click</code>".</p>

  <p class="note">Note that focusing is not an <span>activation behavior</span>, i.e. calling the
  <code data-x="dom-click">click()</code> method on an element or dispatching a synthetic <code
  data-x="event-click">click</code> event on it won't cause the element to get focused.</p>

  <hr>

  <p>A node is a <dfn>focus navigation scope owner</dfn> if it is a <code>Document</code>, a
  <span>shadow host</span>, a <span>slot</span>, or an element which is the <span>popover
  invoker</span> of an element in the <span data-x="popover-showing-state">popover showing
  state</span>.</p>

  <p>Each <span>focus navigation scope owner</span> has a <dfn>focus navigation scope</dfn>, which
  is a list of elements. Its contents are determined as follows:</p>

  <p>Every element <var>element</var> has an <dfn>associated focus navigation owner</dfn>, which is
  either null or a <span>focus navigation scope owner</span>. It is determined by the following
  algorithm:</p>

  <ol>
   <li><p>If <var>element</var>'s parent is null, then return null.</p></li>

   <li><p>If <var>element</var>'s parent is a <span>shadow host</span>, then return
   <var>element</var>'s <span>assigned slot</span>.</p></li>

   <li><p>If <var>element</var>'s parent is a <span>shadow root</span>, then return the parent's
   <span data-x="concept-DocumentFragment-host">host</span>.</p></li>

   <li><p>If <var>element</var>'s parent is the <span>document element</span>, then return the
   parent's <span>node document</span>.</p></li>

   <li><p>If <var>element</var> is in the <span data-x="popover-showing-state">popover showing
   state</span> and has a <span>popover invoker</span> set, then return <var>element</var>'s
   <span>popover invoker</span>.</p></li>

   <li><p>Return <var>element</var>'s parent's <span>associated focus navigation owner</span>.</p></li>
  </ol>

  <p>Then, the contents of a given <span>focus navigation scope owner</span> <var>owner</var>'s
  <span>focus navigation scope</span> are all elements whose <span>associated focus navigation
  owner</span> is <var>owner</var>.</p>

  <p class="note">The order of elements within a <span>focus navigation scope</span> does not impact
  any of the algorithms in this specification. Ordering only becomes important for the
  <span>tabindex-ordered focus navigation scope</span> and <span>flattened tabindex-ordered focus
  navigation scope</span> concepts defined below.</p>

  <p>A <dfn>tabindex-ordered focus navigation scope</dfn> is a list of <span data-x="focusable
  area">focusable areas</span> and <span data-x="focus navigation scope owner">focus navigation
  scope owners</span>. Every <span>focus navigation scope owner</span> <var>owner</var> has
  <span>tabindex-ordered focus navigation scope</span>, whose contents are determined as
  follows:</p>

  <ul>
   <li><p>It contains all elements in <var>owner</var>'s <span>focus navigation scope</span> that
   are themselves <span data-x="focus navigation scope owner">focus navigation scope owners</span>,
   except the elements whose <span>tabindex value</span> is a negative integer.</p></li>

   <li><p>It contains all of the <span data-x="focusable area">focusable areas</span> whose
   <span>DOM anchor</span> is an element in <var>owner</var>'s <span>focus navigation scope</span>,
   except the <span data-x="focusable area">focusable areas</span> whose <span>tabindex value</span>
   is a negative integer.</p></li>
  </ul>

  <p>The order within a <span>tabindex-ordered focus navigation scope</span> is determined by each
  element's <span>tabindex value</span>, as described in the section below.</p>

  <p class="note">The rules there do not give a precise ordering, as they are composed mostly of
  "<!--non-normative-->should" statements and relative orderings.</p>

  <p>A <dfn>flattened tabindex-ordered focus navigation scope</dfn> is a list of <span
  data-x="focusable area">focusable areas</span>. Every <span>focus navigation scope owner</span>
  <var>owner</var> owns a distinct <span>flattened tabindex-ordered focus navigation scope</span>,
  whose contents are determined by the following algorithm:</p>

  <ol>
   <li><p>Let <var>result</var> be a <span data-x="list clone">clone</span> of <var>owner</var>'s
   <span>tabindex-ordered focus navigation scope</span>.</p></li>

   <li>
    <p>For each <var>item</var> of <var>result</var>:</p>

    <ol>
     <li><p>If <var>item</var> is not a <span>focus navigation scope owner</span>, then
     <span>continue</span>.</p></li>

     <li><p>If <var>item</var> is not a <span>focusable area</span>, then replace <var>item</var>
     with all of the items in <var>item</var>'s <span>flattened tabindex-ordered focus navigation
     scope</span>.</p></li>

     <li><p>Otherwise, insert the contents of <var>item</var>'s <span>flattened tabindex-ordered
     focus navigation scope</span> after <var>item</var>.</p></li>
    </ol>
   </li>
  </ol>

  </div>

  <h4>The <code data-x="attr-tabindex">tabindex</code> attribute</h4>

  <p>The <dfn element-attr for="htmlsvg-global"><code data-x="attr-tabindex">tabindex</code></dfn>
  content attribute allows authors to make an element and regions that have the element as its
  <span>DOM anchor</span> be <span data-x="focusable area">focusable areas</span>, allow or prevent
  them from being <span>sequentially focusable</span>, and determine their relative ordering for
  <span>sequential focus navigation</span>.</p>

  <p>The name "tab index" comes from the common use of the <kbd>Tab</kbd> key to navigate through
  the focusable elements. The term "tabbing" refers to moving forward through <span>sequentially
  focusable</span> <span data-x="focusable area">focusable areas</span>.</p>

  <p>The <code data-x="attr-tabindex">tabindex</code> attribute, if specified, must have a value
  that is a <span>valid integer</span>. Positive numbers specify the relative position of the
  element's <span data-x="focusable area">focusable areas</span> in the <span>sequential focus
  navigation order</span>, and negative numbers indicate that the control is not
  <span>sequentially focusable</span>.</p>

  <p>Developers should use caution when using values other than 0 or &#x2212;1 for their <code
  data-x="attr-tabindex">tabindex</code> attributes as this is complicated to do correctly.</p>

  <div class="note">
   <p>The following provides a <span w-nodev>non-normative</span> summary of the behaviors of the
   possible <code data-x="attr-tabindex">tabindex</code> attribute values.<span w-nodev> The below
   processing model gives the more precise rules.</span></p>

   <dl>
    <dt>omitted (or non-integer values)</dt>
    <dd>The user agent will decide whether the element is <span>focusable</span>, and if it is,
    whether it is <span>sequentially focusable</span> or <span>click focusable</span> (or both).</dd>

    <dt>&#x2212;1 (or other negative integer values)</dt>
    <dd>Causes the element to be <span>focusable</span>, and indicates that the author would prefer
    the element to be <span>click focusable</span> but not <span>sequentially focusable</span>. The
    user agent might ignore this preference for click and sequential focusability, e.g., for
    specific element types according to platform conventions, or for keyboard-only users.</dd>

    <dt>0</dt>
    <dd>Causes the element to be <span>focusable</span>, and indicates that the author would prefer
    the element to be both <span>click focusable</span> and <span>sequentially focusable</span>. The
    user agent might ignore this preference for click and sequential focusability.</dd>

    <dt>positive integer values</dt>
    <dd>Behaves the same as 0, but in addition creates a relative ordering within a
    <span>tabindex-ordered focus navigation scope</span>, so that elements with higher <code
    data-x="attr-tabindex">tabindex</code> attribute value come later.</dd>
   </dl>

   <p>Note that the <code data-x="attr-tabindex">tabindex</code> attribute cannot be used to make
   an element non-focusable. The only way a page author can do that is by <span
   data-x="concept-element-disabled">disabling</span> the element, or making it
   <span>inert</span>.</p>
  </div>

  <div w-nodev>

  <hr>

  <p>The <dfn>tabindex value</dfn> of an element is the value of its <code
  data-x="attr-tabindex">tabindex</code> attribute, parsed using the <span>rules for parsing
  integers</span>. If parsing fails or the attribute is not specified, then the <span>tabindex
  value</span> is null.</p>

  <p>The <span>tabindex value</span> of a <span>focusable area</span> is the <span>tabindex
  value</span> of its <span>DOM anchor</span>.</p>

  <p>The <span>tabindex value</span> of an element must be interpreted as follows:</p>

  <dl>
   <dt>If the value is null</dt>

   <dd>
    <p>The user agent should follow platform conventions to determine if the element should be
    considered as a <span>focusable area</span> and if so, whether the element and any <span
    data-x="focusable area">focusable areas</span> that have the element as their
    <span>DOM anchor</span> are <span>sequentially focusable</span>, and if so, what their relative
    position in their <span>tabindex-ordered focus navigation scope</span> is to be. If the element
    is a <span>focus navigation scope owner</span>, it must be included in its
    <span>tabindex-ordered focus navigation scope</span> even if it is not a <span>focusable
    area</span>.</p>

    <p>The relative ordering within a <span>tabindex-ordered focus navigation scope</span> for
    elements and <span data-x="focusable area">focusable areas</span> that belong to the same
    <span>focus navigation scope</span> and whose <span>tabindex value</span> is null should be in
    <span>shadow-including tree order</span>.</p>

    <p>Modulo platform conventions, it is suggested that the following elements should be considered
    as <span data-x="focusable area">focusable areas</span> and be <span>sequentially
    focusable</span>:</p>

    <ul>
     <li><code>a</code> elements that have an <code data-x="attr-hyperlink-href">href</code>
     attribute</li>

     <li><code>button</code> elements</li>

     <li><code>input</code> elements whose <code data-x="attr-input-type">type</code> attribute are
     not in the <span data-x="attr-input-type-hidden">Hidden</span> state</li>

     <li><code>select</code> elements</li>

     <li><code>textarea</code> elements</li>

     <li><code>summary</code> elements that are the first <code>summary</code> element child of a
     <code>details</code> element</li>

     <li>Elements with a <code data-x="attr-draggable">draggable</code> attribute set, if that would
     enable the user agent to allow the user to begin drag operations for those elements without
     the use of a pointing device</li>

     <li><span data-x="editing host">Editing hosts</span></li>

     <li><span data-x="navigable container">Navigable containers</span></li>
    </ul>
   </dd>

   <dt id="negative-tabindex">If the value is a negative integer</dt>

   <dd>
    <p>The user agent must consider the element as a <span>focusable area</span>, but should omit
    the element from any <span>tabindex-ordered focus navigation scope</span>.</p>

    <p class="note">One valid reason to ignore the requirement that sequential focus navigation not
    allow the author to lead to the element would be if the user's only mechanism for moving the
    focus is sequential focus navigation. For instance, a keyboard-only user would be unable to
    click on a text control with a negative <code data-x="attr-tabindex">tabindex</code>, so that
    user's user agent would be well justified in allowing the user to tab to the control
    regardless.</p>
   </dd>

   <dt>If the value is a zero</dt>

   <dd>
    <p>The user agent must allow the element to be considered as a <span>focusable area</span> and
    should allow the element and any <span data-x="focusable area">focusable areas</span> that have
    the element as their <span>DOM anchor</span> to be <span>sequentially focusable</span>.</p>

    <p>The relative ordering within a <span>tabindex-ordered focus navigation scope</span> for
    elements and <span data-x="focusable area">focusable areas</span> that belong to the same
    <span>focus navigation scope</span> and whose <span>tabindex value</span> is zero should be in
    <span>shadow-including tree order</span>.</p>
   </dd>

   <dt>If the value is greater than zero</dt>

   <dd>
    <p>The user agent must allow the element to be considered as a <span>focusable area</span> and
    should allow the element and any <span data-x="focusable area">focusable areas</span> that have
    the element as their <span>DOM anchor</span> to be <span>sequentially focusable</span>, and
    should place the element &mdash; referenced as <var>candidate</var> below &mdash; and the
    aforementioned <span data-x="focusable area">focusable areas</span> in the
    <span>tabindex-ordered focus navigation scope</span> where the element is a part of so that,
    relative to other elements and <span data-x="focusable area">focusable areas</span> that belong
    to the same <span>focus navigation scope</span>, they are:</p>

    <ul>
     <li>before any <span>focusable area</span> whose <span>DOM anchor</span> is an element whose <code
     data-x="attr-tabindex">tabindex</code> attribute has been omitted or whose value, when parsed,
     returns an error,</li>

     <li>before any <span>focusable area</span> whose <span>DOM anchor</span> is an element whose <code
     data-x="attr-tabindex">tabindex</code> attribute has a value less than or equal to zero,</li>

     <li>after any <span>focusable area</span> whose <span>DOM anchor</span> is an element whose <code
     data-x="attr-tabindex">tabindex</code> attribute has a value greater than zero but less than
     the value of the <code data-x="attr-tabindex">tabindex</code> attribute on <var>candidate</var>,</li>

     <li>after any <span>focusable area</span> whose <span>DOM anchor</span> is an element whose <code
     data-x="attr-tabindex">tabindex</code> attribute has a value equal to the value of the <code
     data-x="attr-tabindex">tabindex</code> attribute on <var>candidate</var> but that is
     located earlier than <var>candidate</var> in <span>shadow-including tree order</span>,</li>

     <li>before any <span>focusable area</span> whose <span>DOM anchor</span> is an element whose <code
     data-x="attr-tabindex">tabindex</code> attribute has a value equal to the value of the <code
     data-x="attr-tabindex">tabindex</code> attribute on <var>candidate</var> but that is
     located later than <var>candidate</var> in <span>shadow-including tree order</span>, and</li>

     <li>before any <span>focusable area</span> whose <span>DOM anchor</span> is an element whose <code
     data-x="attr-tabindex">tabindex</code> attribute has a value greater than the value of the
     <code data-x="attr-tabindex">tabindex</code> attribute on <var>candidate</var>.</li>
    </ul>
   </dd>
  </dl>

  <p>The <dfn attribute for="HTMLOrSVGElement"><code data-x="dom-tabIndex">tabIndex</code></dfn> IDL
  attribute must <span>reflect</span> the value of the <code data-x="attr-tabindex">tabindex</code>
  content attribute. The <span>default value</span> is 0 if the element is an <code>a</code>,
  <code>area</code>, <code>button</code>, <code>frame</code>, <code>iframe</code>,
  <code>input</code>, <code>object</code>, <code>select</code>, <code>textarea</code>, or <span>SVG
  <code>a</code></span> element, or is a <code>summary</code> element that is a <span>summary for
  its parent details</span>. The <span>default value</span> is &#x2212;1 otherwise.</p>

  <p class="note">The varying default value based on element type is a historical artifact.</p>

  </div>


  <div w-nodev>

  <h4 id="focus-processing-model"><span id="processing-model-5"></span>Processing model</h4>

  <p>To <dfn>get the focusable area</dfn> for a <var>focus target</var> that is either an element
  that is not a <span>focusable area</span>, or is a <span>navigable</span>, given an
  optional string <var>focus trigger</var> (default "<code data-x="">other</code>"), run the first
  matching set of steps from the following list:</p>

  <dl class="switch">
   <dt>If <var>focus target</var> is an <code>area</code> element with one or more shapes that are
   <span data-x="focusable area">focusable areas</span></dt>

   <dd><p>Return the shape corresponding to the first <code>img</code> element in <span>tree
   order</span> that uses the image map to which the <code>area</code> element belongs.</p></dd>

   <dt>If <var>focus target</var> is an element with one or more scrollable regions that are <span
   data-x="focusable area">focusable areas</span></dt>

   <dd><p>Return the element's first scrollable region, according to a pre-order, depth-first
   traversal of the <span>flat tree</span>. <ref>CSSSCOPING</ref></p></dd>

   <dt>If <var>focus target</var> is the <span>document element</span> of its
   <code>Document</code></dt>

   <dd><p>Return the <code>Document</code>'s <span>viewport</span>.</p></dd>

   <dt>If <var>focus target</var> is a <span>navigable</span></dt>

   <dd><p>Return the <span>navigable</span>'s <span data-x="nav-document">active
   document</span>.</p></dd>

   <dt>If <var>focus target</var> is a <span>navigable container</span> with a non-null
   <span>content navigable</span></dt>

   <dd><p>Return the <span>navigable container</span>'s <span>content navigable</span>'s <span
   data-x="nav-document">active document</span>.</p></dd>

   <dt>If <var>focus target</var> is a <span>shadow host</span> whose <span
   data-x="concept-element-shadow-root">shadow root</span>'s <span>delegates focus</span> is
   true</dt>

   <dd>
    <ol>
     <li><p>Let <var>focusedElement</var> be the <span>currently focused area of a top-level
     traversable</span>'s <span>DOM anchor</span>.</p></li>

     <li><p>If <var>focus target</var> is a <span>shadow-including inclusive ancestor</span> of
     <var>focusedElement</var>, then return <var>focusedElement</var>.</p></li>

     <li><p>Return the <span>focus delegate</span> for <var>focus target</var> given <var>focus
     trigger</var>.</p></li>
    </ol>

    <p class="note">For <span data-x="sequentially focusable">sequential focusability</span>, the
    handling of <span data-x="shadow host">shadow hosts</span> and <span>delegates focus</span> is
    done when constructing the <span>sequential focus navigation order</span>. That is, the
    <span>focusing steps</span> will never be called on such <span data-x="shadow host">shadow
    hosts</span> as part of sequential focus navigation.</p>
   </dd>

   <dt>Otherwise</dt>

   <dd><p>Return null.</p></dd>
  </dl>

  <p>The <dfn>focus delegate</dfn> for a <var>focusTarget</var>, given an optional string
  <var>focusTrigger</var> (default "<code data-x="">other</code>"), is given by the following steps:</p>

  <ol>
   <li><p>If <var>focusTarget</var> is a <span>shadow host</span> and its <span
   data-x="concept-element-shadow-root">shadow root</span>'s <span>delegates focus</span> is false,
   then return null.</p></li>

   <li><p>Let <var>whereToLook</var> be <var>focusTarget</var>.</p></li>

   <li><p>If <var>whereToLook</var> is a <span>shadow host</span>, then set <var>whereToLook</var>
   to <var>whereToLook</var>'s <span data-x="concept-element-shadow-root">shadow
   root</span>.</p></li>

   <li><p>Let <var>autofocusDelegate</var> be the <span>autofocus delegate</span> for
   <var>whereToLook</var> given <var>focusTrigger</var>.</p></li>

   <li><p>If <var>autofocusDelegate</var> is not null, then return
   <var>autofocusDelegate</var>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>descendant</var> of
    <var>whereToLook</var>'s <span data-x="descendant">descendants</span>, in <span>tree
    order</span>:</p>

    <ol>
     <li><p>Let <var>focusableArea</var> be null.</p></li>

     <li><p>If <var>focusTarget</var> is a <code>dialog</code> element and <var>descendant</var> is
     <span>sequentially focusable</span>, then set <var>focusableArea</var> to
     <var>descendant</var>.</p></li>

     <li><p>Otherwise, if <var>focusTarget</var> is not a <code>dialog</code> and
     <var>descendant</var> is a <span>focusable area</span>, set <var>focusableArea</var> to
     <var>descendant</var>.</p></li>

     <li>
      <p>Otherwise, set <var>focusableArea</var> to the result of <span data-x="get the focusable
      area">getting the focusable area</span> for <var>descendant</var> given
      <var>focusTrigger</var>.</p>

      <p class="note">This step can end up recursing, i.e., the <span>get the focusable area</span>
      steps might return the <span>focus delegate</span> of <var>descendant</var>.</p>
     </li>

     <li><p>If <var>focusableArea</var> is not null, then return <var>focusableArea</var>.</p></li>
    </ol>

    <p class="note">It's important that we are <em>not</em> looking at the <span
    data-x="shadow-including descendant">shadow-including descendants</span> here, but instead only
    at the <span data-x="descendant">descendants</span>. <span data-x="shadow host">Shadow
    hosts</span> are instead handled by the recursive case mentioned above.</p>
   </li>

   <li><p>Return null.</p></li>
  </ol>

  <p class="note">The above algorithm essentially returns the first suitable <span>focusable
  area</span> where the path between its <span>DOM anchor</span> and <var>focusTarget</var>
  delegates focus at any shadow tree boundaries.</p>

  <p>The <dfn>autofocus delegate</dfn> for a <var>focus target</var> given a <var>focus
  trigger</var> is given by the following steps:</p>

  <ol>
   <li>
    <p>For each <span>descendant</span> <var>descendant</var> of <var>focus target</var>, in
    <span>tree order</span>:</p>

    <ol>
     <li><p>If <var>descendant</var> does not have an <code
     data-x="attr-fe-autofocus">autofocus</code> content attribute, then
     <span>continue</span>.</p></li>

     <li><p>Let <var>focusable area</var> be <var>descendant</var>, if <var>descendant</var> is a
     <span>focusable area</span>; otherwise let <var>focusable area</var> be the result of <span
     data-x="get the focusable area">getting the focusable area</span> for <var>descendant</var>
     given <var>focus trigger</var>.</p></li>

     <li><p>If <var>focusable area</var> is null, then <span>continue</span>.</p></li>

     <li><p>If <var>focusable area</var> is not <span>click focusable</span> and <var>focus
     trigger</var> is "<code data-x="">click</code>", then <span>continue</span>.</p></li>

     <li><p>Return <var>focusable area</var>.</p></li>
    </ol>
   </li>

   <li><p>Return null.</p></li>
  </ol>

  <p>The <dfn export>focusing steps</dfn> for an object <var>new focus target</var> that is either a
  <span>focusable area</span>, or an element that is not a <span>focusable area</span>, or a
  <span>navigable</span>, are as follows. They can optionally be run with a <var>fallback
  target</var> and a string <var>focus trigger</var>.</p>

  <ol>
   <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2816 div.focus() when it's got a scroll region -->
   <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2817 img.focus() when it's got an image map -->

   <li><p>If <var>new focus target</var> is not a <span>focusable area</span>, then set <var>new
   focus target</var> to the result of <span data-x="get the focusable area">getting the focusable
   area</span> for <var>new focus target</var>, given <var>focus trigger</var> if it was
   passed.</p></li>

   <li>
    <p>If <var>new focus target</var> is null, then:</p>

    <ol>
     <li><p>If no <var>fallback target</var> was specified, then return.</p></li>

     <li><p>Otherwise, set <var>new focus target</var> to the <var>fallback
     target</var>.</p></li>
    </ol>
   </li>

   <li><p>If <var>new focus target</var> is a <span>navigable container</span> with non-null
   <span>content navigable</span>, then set <var>new focus target</var> to the <span>content
   navigable</span>'s <span data-x="nav-document">active document</span>.</p></li>
   <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2818 -->

   <li><p>If <var>new focus target</var> is a <span>focusable area</span> and its <span>DOM
   anchor</span> is <span>inert</span>, then return.</p></li>

   <li><p>If <var>new focus target</var> is the <span>currently focused area of a top-level
   traversable</span>, then return.</p></li>

   <li><p>Let <var>old chain</var> be the <span data-x="current focus chain of a top-level
   traversable">current focus chain of the top-level traversable</span> in which <var>new focus
   target</var> finds itself.</p></li>

   <li><p>Let <var>new chain</var> be the <span>focus chain</span> of <var>new
   focus target</var>.</p></li>

   <li><p>Run the <span>focus update steps</span> with <var>old chain</var>, <var>new chain</var>,
   and <var>new focus target</var> respectively.</p></li>
  </ol>

  <p>User agents must <span>immediately</span> run the <span>focusing steps</span> for a
  <span>focusable area</span> or <span>navigable</span> <var>candidate</var> whenever the
  user attempts to move the focus to <var>candidate</var>.</p>

  <p>The <dfn>unfocusing steps</dfn> for an object <var>old focus target</var> that is
  either a <span>focusable area</span> or an element that is not a <span>focusable area</span> are as
  follows:</p>

  <ol>
   <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2819 -->
   <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2822 -->

   <li><p>If <var>old focus target</var> is a <span>shadow host</span> whose <span
   data-x="concept-element-shadow-root">shadow root</span>'s <span>delegates focus</span> is true,
   and <var>old focus target</var>'s <span data-x="concept-element-shadow-root">shadow root</span>
   is a <span>shadow-including inclusive ancestor</span> of the <span>currently focused area of a
   top-level traversable</span>'s <span>DOM anchor</span>, then set <var>old focus target</var> to
   that <span>currently focused area of a top-level traversable</span>.</p></li>

   <li><p>If <var>old focus target</var> is <span>inert</span>, then return.</p></li>

   <li><p>If <var>old focus target</var> is an <code>area</code> element and one of its shapes is
   the <span>currently focused area of a top-level traversable</span>, or, if <var>old focus
   target</var> is an element with one or more scrollable regions, and one of them is the
   <span>currently focused area of a top-level traversable</span>, then let <var>old focus
   target</var> be that <span>currently focused area of a top-level traversable</span>.</p></li>

   <li><p>Let <var>old chain</var> be the <span data-x="current focus chain of a top-level
   traversable">current focus chain of the top-level traversable</span> in which <var>old focus
   target</var> finds itself.</p></li>

   <li><p>If <var>old focus target</var> is not one of the entries in <var>old
   chain</var>, then return.</p></li>

   <li><p>If <var>old focus target</var> is not a <span>focusable area</span>, then return.</p></li>

   <li><p>Let <var>topDocument</var> be <var>old chain</var>'s last entry.</p></li>

   <li>
    <p>If <var>topDocument</var>'s <span>node navigable</span> has <span>system focus</span>, then
    run the <span>focusing steps</span> for <var>topDocument</var>'s <span>viewport</span>.</p>

    <p>Otherwise, apply any relevant platform-specific conventions for removing <span>system
    focus</span> from <var>topDocument</var>'s <span>node navigable</span>, and run the <span>focus
    update steps</span> given <var>old chain</var>, an empty list, and null.</p>
   </li>
  </ol>

  <p class="note">The <span>unfocusing steps</span> do not always result in the focus changing, even
  when applied to the <span>currently focused area of a top-level traversable</span>. For
  example, if the <span>currently focused area of a top-level traversable</span> is a
  <span>viewport</span>, then it will usually keep its focus regardless until another
  <span>focusable area</span> is explicitly focused with the <span>focusing steps</span>.</p>

  <hr>

  <p>The <dfn>focus update steps</dfn>, given an <var>old chain</var>, a <var>new chain</var>, and a <var>new focus target</var> respectively, are as
  follows:</p>

  <ol>
   <!-- focusin/focusout?: https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=723 -->

   <li><p>If the last entry in <var>old chain</var> and the last entry in <var>new chain</var> are the same, pop the last entry from <var>old chain</var>
   and the last entry from <var>new chain</var> and redo this step.</p></li>

   <li>
    <p>For each entry <var>entry</var> in <var>old chain</var>, in order, run
    these substeps:</p>

    <ol>
     <li>
      <p>If <var>entry</var> is an <code>input</code> element, and the <code
      data-x="event-change">change</code> event <span data-x="concept-input-apply">applies</span>
      to the element, and the element does not have a defined <span>activation behavior</span>, and
      the user has changed the element's <span data-x="concept-fe-value">value</span> or its list
      of <span data-x="concept-input-type-file-selected">selected files</span> while the control
      was focused without committing that change (such that it is different to what it was when the
      control was first focused), then:</p>

      <ol>
       <li><p>Set <var>entry</var>'s <span>user validity</span> to true.</p></li>

       <li id="unfocus-causes-change-event"><p><span data-x="concept-event-fire">Fire an
       event</span> named <code data-x="event-change">change</code> at the element, with the <code
       data-x="dom-Event-bubbles">bubbles</code> attribute initialized to true.</p></li>
      </ol>
     </li>

     <li>
      <p>If <var>entry</var> is an element, let <var>blur event target</var> be
      <var>entry</var>.</p>

      <p>If <var>entry</var> is a <code>Document</code> object, let <var>blur event target</var> be
      that <code>Document</code> object's <span>relevant global object</span>.</p>

      <p>Otherwise, let <var>blur event target</var> be null.</p>
     </li>

     <li><p>If <var>entry</var> is the last entry in <var>old chain</var>, and
     <var>entry</var> is an <code>Element</code>, and the last entry in <var>new
     chain</var> is also an <code>Element</code>, then let <var>related blur target</var>
     be the last entry in <var>new chain</var>. Otherwise, let <var>related blur
     target</var> be null.</p></li>

     <li>
      <p>If <var>blur event target</var> is not null, <span>fire a focus event</span>
      named <code data-x="event-blur">blur</code> at <var>blur event target</var>, with
      <var>related blur target</var> as the related target.</p>

      <p class="note" id="note-sometimes-no-blur-event">In some cases, e.g., if <var>entry</var> is
      an <code>area</code> element's shape, a scrollable region, or a <span>viewport</span>, no
      event is fired.</p>
     </li>
    </ol>
   </li>

   <li><p>Apply any relevant platform-specific conventions for focusing <var>new focus target</var>.
   (For example, some platforms select the contents of a text control when that control is
   focused.)</p></li>

   <li>
    <p>For each entry <var>entry</var> in <var>new chain</var>, in reverse
    order, run these substeps:</p>

    <ol>
     <li>
      <p>If <var>entry</var> is a <span>focusable area</span>, and the <span>focused area of the
      document</span> is not <var>entry</var>:</p>

      <ol>
       <li><p>Set <var>document</var>'s <span>relevant global object</span>'s <span
       data-x="window-navigation-api">navigation API</span>'s <span>focus changed during ongoing
       navigation</span> to true.</p></li>

       <li><p>Designate <var>entry</var> as the <span>focused area of the document</span>.</p>
      </ol>
     </li>

     <li>
      <p>If <var>entry</var> is an element, let <var>focus event target</var> be
      <var>entry</var>.</p>

      <p>If <var>entry</var> is a <code>Document</code> object, let <var>focus event target</var> be
      that <code>Document</code> object's <span>relevant global object</span>.</p>

      <p>Otherwise, let <var>focus event target</var> be null.</p>
     </li>

     <li><p>If <var>entry</var> is the last entry in <var>new chain</var>, and
     <var>entry</var> is an <code>Element</code>, and the last entry in <var>old
     chain</var> is also an <code>Element</code>, then let <var>related focus target</var>
     be the last entry in <var>old chain</var>. Otherwise, let <var>related
     focus target</var> be null.</p></li>

     <li>
      <p>If <var>focus event target</var> is not null, <span>fire a focus event</span>
      named <code data-x="event-focus">focus</code> at <var>focus event target</var>, with
      <var>related focus target</var> as the related target.</p>

      <p class="note">In some cases, e.g. if <var>entry</var> is an <code>area</code>
      element's shape, a scrollable region, or a <span>viewport</span>, no event is fired.</p>
     </li>
    </ol>
   </li>
  </ol>

  <p>To <dfn>fire a focus event</dfn> named <var>e</var> at an element <var>t</var> with a given
  related target <var>r</var>, <span data-x="concept-event-fire">fire an event</span> named
  <var>e</var> at <var>t</var>, using <code>FocusEvent</code>, with the <code
  data-x="dom-FocusEvent-relatedTarget">relatedTarget</code> attribute initialized to <var>r</var>,
  the <code data-x="dom-UIEvent-view">view</code> attribute initialized to <var>t</var>'s
  <span>node document</span>'s <span>relevant global object</span>, and the <span>composed
  flag</span> set.</p>

  <hr>

  <p>When a key event is to be routed in a <span>top-level traversable</span>, the user agent must
  run the following steps:</p>

  <!-- interaction event spec point -->

  <ol>
   <li><p>Let <var>target area</var> be the <span data-x="currently focused area of a top-level
   traversable">currently focused area of the top-level traversable</span>.</p></li>

   <li><p><span>Assert</span>: <var>target area</var> is not null, since key events are only routed
   to <span data-x="top-level traversable">top-level traversables</span> that have <span>system
   focus</span>. Therefore, <var>target area</var> is a <span>focusable area</span>.</p></li>

   <li><p>Let <var>target node</var> be <var>target area</var>'s <span>DOM anchor</span>.</p></li>

   <li>
    <p>If <var>target node</var> is a <code>Document</code> that has a <span data-x="the
    body element">body element</span>, then let <var>target node</var> be <span>the body
    element</span> of that <code>Document</code>.</p>

    <p>Otherwise, if <var>target node</var> is a <code>Document</code> object that has a non-null
    <span>document element</span>, then let <var>target node</var> be that <span>document
    element</span>.</p>
   </li>

   <li>
    <p>If <var>target node</var> is not <span>inert</span>, then:</p>
    <ol>
     <li><p>Let <var>canHandle</var> be the result of <span
     data-x="concept-event-dispatch">dispatching</span> the key event at <var>target
     node</var>.</p></li>

     <li><p>If <var>canHandle</var> is true, then let <var>target area</var> handle the key event.
     This might include <span data-x="fire a click event">firing a <code
     data-x="event-click">click</code> event</span> at <var>target node</var>.</p></li>
    </ol>
   </li>
  </ol>

  <hr>

  <p>The <dfn export>has focus steps</dfn>, given a <code>Document</code> object <var>target</var>,
  are as follows:</p>

  <ol>
   <li><p>If <var>target</var>'s <span>node navigable</span>'s <span data-x="nav-top">top-level
   traversable</span> does not have <span>system focus</span>, then return false.</p></li>

   <li><p>Let <var>candidate</var> be <var>target</var>'s <span>node navigable</span>'s <span
   data-x="nav-top">top-level traversable</span>'s <span>active document</span>.</p></li>

   <li>
    <p>While true:</p>

    <ol>
     <li><p>If <var>candidate</var> is <var>target</var>, then return true.</p></li>

     <li><p>If the <span data-x="focused area of the document">focused area</span> of
     <var>candidate</var> is a <span>navigable container</span> with a non-null <span>content
     navigable</span>, then set <var>candidate</var> to the <span data-x="nav-document">active
     document</span> of that <span>navigable container</span>'s <span>content
     navigable</span>.</p></li>

     <li><p>Otherwise, return false.</p></li>
    </ol>
   </li>
  </ol>



  <h4><dfn>Sequential focus navigation</dfn></h4>

  <p>Each <code>Document</code> has a <dfn>sequential focus navigation order</dfn>, which orders
  some or all of the <span data-x="focusable area">focusable areas</span> in the
  <code>Document</code> relative to each other. Its contents and ordering are given by the
  <span>flattened tabindex-ordered focus navigation scope</span> of the <code>Document</code>.</p>

  <p class="note">Per the rules defining the <span>flattened tabindex-ordered focus navigation
  scope</span>, the ordering is not necessarily related to the <span>tree order</span> of the
  <code>Document</code>.</p>

  <p>If a <span>focusable area</span> is omitted from the <span>sequential focus navigation
  order</span> of its <code>Document</code>, then it is unreachable via <span>sequential focus
  navigation</span>.</p>

  <p>There can also be a <dfn>sequential focus navigation starting point</dfn>. It is initially
  unset. The user agent may set it when the user indicates that it should be moved.</p>

  <p class="example">For example, the user agent could set it to the position of the user's click if
  the user clicks on the document contents.</p>

  <p class="note">User agents are required to set the <span>sequential focus navigation starting
  point</span> to the <span>target element</span> when <span data-x="navigate-fragid">navigating
  to a fragment</span>.</p>

  <p>A <dfn>sequential focus direction</dfn> is one of two possible values: "<dfn><code
  data-x="sequential-focus-forward">forward</code></dfn>", or "<dfn><code
  data-x="sequential-focus-backward">backward</code></dfn>". They are used in the below algorithms
  to describe the direction in which sequential focus travels at the user's request.</p>

  <p>A <dfn>selection mechanism</dfn> is one of two possible values: "<dfn><code
  data-x="selection-mechanism-DOM">DOM</code></dfn>", or
  "<dfn><code data-x="selection-mechanism-sequential">sequential</code></dfn>". They are used to
  describe how the <span>sequential navigation search algorithm</span> finds the <span>focusable
  area</span> it returns.</p>

  <p>When the user requests that focus move from the <span>currently focused area of a top-level
  traversable</span> to the next or previous <span>focusable area</span> (e.g., as the default
  action of pressing the <kbd><kbd>tab</kbd></kbd> key), or when the user requests that focus
  sequentially move to a <span>top-level traversable</span> in the first place (e.g., from the
  browser's location bar), the user agent must use the following algorithm:</p>

  <ol>
   <li><p>Let <var>starting point</var> be the <span>currently focused area of a top-level
   traversable</span>, if the user requested to move focus sequentially from there, or else the
   <span>top-level traversable</span> itself, if the user instead requested to move focus from
   outside the <span>top-level traversable</span>.</p></li>

   <li><p>If there is a <span>sequential focus navigation starting point</span> defined and it is
   inside <var>starting point</var>, then let <var>starting point</var> be the <span>sequential
   focus navigation starting point</span> instead.</p></li>

   <li>
    <p>Let <var>direction</var> be "<code data-x="sequential-focus-forward">forward</code>" if the
    user requested the next control, and "<code
    data-x="sequential-focus-backward">backward</code>" if the user requested the previous
    control.</p>

    <p class="note">Typically, pressing <kbd><kbd>tab</kbd></kbd> requests the next control, and
    pressing <kbd><kbd>shift</kbd> + <kbd>tab</kbd></kbd> requests the previous control.</p>
   </li>

   <li>
    <p><i>Loop</i>: Let <var>selection mechanism</var> be "<code
    data-x="selection-mechanism-sequential">sequential</code>" if <var>starting point</var> is a
    <span>navigable</span> or if <var>starting point</var> is in its <code>Document</code>'s
    <span>sequential focus navigation order</span>.</p>

    <p>Otherwise, <var>starting point</var> is not in its <code>Document</code>'s <span>sequential
    focus navigation order</span>; let <var>selection mechanism</var> be "<code
    data-x="selection-mechanism-DOM">DOM</code>".</p>
   </li>

   <li><p>Let <var>candidate</var> be the result of running the <span>sequential navigation search
   algorithm</span> with <var>starting point</var>, <var>direction</var>, and <var>selection
   mechanism</var>.</p></li>

   <li><p>If <var>candidate</var> is not null, then run the <span>focusing steps</span> for
   <var>candidate</var> and return.</p></li>

   <li><p>Otherwise, unset the <span>sequential focus navigation starting point</span>.</p></li>

   <li>
    <p>If <var>starting point</var> is a <span>top-level traversable</span>, or a
    <span>focusable area</span> in the <span>top-level traversable</span>, the user agent
    should transfer focus to its own controls appropriately (if any), honouring
    <var>direction</var>, and then return.</p>

    <p class="example">For example, if <var>direction</var> is <i>backward</i>, then the last
    <span>sequentially focusable</span> control before the browser's rendering area would be the
    control to focus.</p>

    <p>If the user agent has no <span>sequentially focusable</span> controls &mdash; a kiosk-mode
    browser, for instance &mdash; then the user agent may instead restart these steps with the
    <var>starting point</var> being the <span>top-level traversable</span> itself.</p>

    <!-- in theory, the top-level traversable _always_ has at least one focusable area: the
    viewport. Even a "blocked by a modal dialog" doesn't disable the viewport (since the Document is
    its DOM anchor, and the Document isn't made inert by "blocked by a modal dialog"). Note that
    child navigables can have inert viewports, though (if the navigable container
    itself is inert, for example) -->
   </li>

   <li><p>Otherwise, <var>starting point</var> is a <span>focusable area</span> in a <span>child
   navigable</span>. Set <var>starting point</var> to that <span>child navigable</span>'s <span
   data-x="nav-parent">parent</span> and return to the step labeled <i>loop</i>.</p></li>
   <!--
    https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2856
    https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2857
   -->
  </ol>


  <p>The <dfn>sequential navigation search algorithm</dfn>, given a <span>focusable area</span>
  <var>starting point</var>, <span>sequential focus direction</span> <var>direction</var>, and
  <span>selection mechanism</span> <var>selection mechanism</var>, consists of the following steps.
  They return a <span>focusable area</span>-or-null.</p>

  <ol>
   <li>
    <p>Pick the appropriate cell from the following table, and follow the instructions in that
    cell.</p>

    <p>The appropriate cell is the one that is from the column whose header describes
    <var>direction</var> and from the first row whose header describes <var>starting point</var> and
    <var>selection mechanism</var>.</p>

    <table>
     <thead>
      <tr>
       <th>
       <th><var>direction</var> is "<code data-x="sequential-focus-forward">forward</code>"
       <th><var>direction</var> is "<code data-x="sequential-focus-backward">backward</code>"
     <tbody>
      <tr>
       <th><var>starting point</var> is a <span>navigable</span>
       <td>Let <var>candidate</var> be the first <span>suitable sequentially focusable area</span>
       in <var>starting point</var>'s <span data-x="nav-document">active document</span>, if any; or
       else null
       <td>Let <var>candidate</var> be the last <span>suitable sequentially focusable area</span>
       in <var>starting point</var>'s <span data-x="nav-document">active document</span>, if any; or
       else null
      <tr>
       <th><var>selection mechanism</var> is "<code data-x="selection-mechanism-DOM">DOM</code>"
       <td>
        <p>Let <var>candidate</var> be the <span>suitable sequentially focusable area</span>, that
        appears nearest after <var>starting point</var> in <var>starting point</var>'s
        <code>Document</code>, in <span>shadow-including tree order</span>, if any; or else null</p>

        <p class="note">In this case, <var>starting point</var> does not necessarily belong to its
        <code>Document</code>'s <span>sequential focus navigation order</span>, so we'll select the
        <span data-x="suitable sequentially focusable area">suitable</span> <span data-x="list
        item">item</span> from that list that comes <em>after</em> <var>starting point</var> in
        <span>shadow-including tree order</span>.</p>
       </td>
       <td>Let <var>candidate</var> be the <span>suitable sequentially focusable area</span>, that
       appears nearest before <var>starting point</var> in <var>starting point</var>'s
       <code>Document</code>, in <span>shadow-including tree order</span>, if any; or else null
      <tr>
       <th><var>selection mechanism</var> is "<code
       data-x="selection-mechanism-sequential">sequential</code>"
       <td>Let <var>candidate</var> be the first <span>suitable sequentially focusable area</span>
       after <var>starting point</var>, in <var>starting point</var>'s <code>Document</code>'s
       <span>sequential focus navigation order</span>, if any; or else null
       <td>Let <var>candidate</var> be the last <span>suitable sequentially focusable area</span>
       before <var>starting point</var>, in <var>starting point</var>'s <code>Document</code>'s
       <span>sequential focus navigation order</span>, if any; or else null
    </table>

    <p>A <dfn>suitable sequentially focusable area</dfn> is a <span>focusable area</span> whose
    <span>DOM anchor</span> is not <span>inert</span> and is <span>sequentially
    focusable</span>.</p>
   </li>

   <li>
    <p>If <var>candidate</var> is a <span>navigable container</span> with a non-null <span>content
    navigable</span>, then:</p>

    <ol>
     <li><p>Let <var>recursive candidate</var> be the result of running the <span>sequential
     navigation search algorithm</span> with <var>candidate</var>'s <span>content navigable</span>,
     <var>direction</var>, and "<code
     data-x="selection-mechanism-sequential">sequential</code>".<!-- shift-tab from the end in
     https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2855 --></p></li>

     <li><p>If <var>recursive candidate</var> is null, then return the result of running the
     <span>sequential navigation search algorithm</span> with <var>candidate</var>,
     <var>direction</var>, and <var>selection mechanism</var>.</p></li>

     <li><p>Otherwise, set <var>candidate</var> to <var>recursive candidate</var>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>candidate</var>.</p></li>
  </ol>

  </div>


  <h4>Focus management APIs</h4>

  <pre><code class="idl">dictionary <dfn dictionary>FocusOptions</dfn> {
  boolean <span data-x="dom-focusoptions-preventscroll">preventScroll</span> = false;
  boolean <span data-x="dom-focusoptions-focusvisible">focusVisible</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>documentOrShadowRoot</var>.<span subdfn data-x="dom-documentorshadowroot-activeElement">activeElement</span></code></dt>

   <dd>
    <p>Returns the deepest element in the document through which or to which key events are being
    routed. This is, roughly speaking, the focused element in the document.</p>

    <p>For the purposes of this API, when a <span>child navigable</span> is focused, its <span
    data-x="nav-container">container</span> is <a href="#bc-focus-ergo-bcc-focus">focused</a>
    within its <span data-x="nav-parent">parent</span>'s <span data-x="nav-document">active
    document</span>. For example, if the user moves the focus to a text control in an
    <code>iframe</code>, the <code>iframe</code> is the element returned by the <code
    data-x="dom-documentorshadowroot-activeElement">activeElement</code> API in the
    <code>iframe</code>'s <span>node document</span>.</p>

    <p>Similarly, when the focused element is in a different <span>node tree</span> than
    <var>documentOrShadowRoot</var>, the element returned will be the <span
    data-x="concept-DocumentFragment-host">host</span> that's located in the same <span>node
    tree</span> as <var>documentOrShadowRoot</var> if  <var>documentOrShadowRoot</var> is a
    <span>shadow-including inclusive ancestor</span> of the focused element, and null if not.</p>
   </dd>

   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-hasFocus">hasFocus</span>()</code></dt>

   <dd>
    <p>Returns true if key events are being routed through or to the document; otherwise, returns
    false. Roughly speaking, this corresponds to the document, or a document nested inside this one,
    being focused.</p>
   </dd>

   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-window-focus">focus</span>()</code></dt>

   <dd>
    <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2823 -->

    <p>Moves the focus to the window's <span data-x="window navigable">navigable</span>, if any.</p>
   </dd>

<!--
   <dt><var>window</var>.<code subdfn data-x="dom-window-blur">blur</code>()</dt>

   <dd><p>No effect.</p></dd>
-->

   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-focus">focus</span>([ { <span data-x="dom-focusoptions-preventscroll">preventScroll</span>: true } ])</code></dt>

   <dd>
    <p>Moves the focus to the element.</p>

    <p>If the element is a <span>navigable container</span>, moves the focus to its <span>content
    navigable</span> instead.</p>

    <p>By default, this method also scrolls the element into view. Providing the <code
    data-x="dom-focusoptions-preventscroll">preventScroll</code> option and setting it to true
    prevents this behavior.</p>
   </dd>

   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-blur">blur</span>()</code></dt>

   <dd>
    <p>Moves the focus to the <span>viewport</span>. Use of this method is discouraged; if you want
    to focus the <span>viewport</span>, call the <code data-x="dom-focus">focus()</code> method on
    the <code>Document</code>'s <span>document element</span>.</p>

    <p>Do not use this method to hide the focus ring if you find the focus ring unsightly. Instead,
    use the <code>:focus-visible</code> pseudo-class to override the <span>'outline'</span>
    property, and provide a different way to show what element is focused. Be aware that if an
    alternative focusing style isn't made available, the page will be significantly less usable for
    people who primarily navigate pages using a keyboard, or those with reduced vision who use focus
    outlines to help them navigate the page.</p>

    <!-- we suggest using CSS here because users can override CSS, so it's no the end of the world,
    unlike using .blur(), which cannot be easily overridden by users and completely breaks tab
    navigation. -->

    <div class="example">

     <p>For example, to hide the outline from <code>textarea</code> elements and instead use a
     yellow background to indicate focus, you could use:</p>

     <pre><code class="css">textarea:focus-visible { outline: none; background: yellow; color: black; }</code></pre>

    </div>
   </dd>
  </dl>

  <div w-nodev>

  <p id="dom-document-activeelement">The <dfn attribute for="DocumentOrShadowRoot"><code
  data-x="dom-documentorshadowroot-activeElement">activeElement</code></dfn> attribute's getter must
  run these steps:</p>

  <ol>
   <li><p>Let <var>candidate</var> be the <span>DOM anchor</span> of the <span data-x="focused area
   of the document">focused area</span> of this <code>DocumentOrShadowRoot</code>'s <span>node
   document</span>.</p></li>

   <li><p>Set <var>candidate</var> to the result of <span data-x="dom-retarget">retargeting</span>
   <var>candidate</var> against this <code>DocumentOrShadowRoot</code>.</p></li>

   <li><p>If <var>candidate</var>'s <span>root</span> is not this <code>DocumentOrShadowRoot</code>,
   then return null.</p></li>

   <li><p>If <var>candidate</var> is not a <code>Document</code> object, then return
   <var>candidate</var>.</p></li>

   <li><p>If <var>candidate</var> has a <span data-x="the body element">body element</span>, then
   return that <span data-x="the body element">body element</span>.</p></li>

   <li><p>If <var>candidate</var>'s <span>document element</span> is non-null, then return that
   <span>document element</span>.</p></li>

   <li><p>Return null.</p></li>
  </ol>

  <p>The <dfn method for="Document"><code data-x="dom-document-hasFocus">hasFocus()</code></dfn>
  method on the <code>Document</code> object, when invoked, must return the result of running the
  <span>has focus steps</span> with the <code>Document</code> object as the argument.</p>

  <p>The <dfn method for="Window"><code data-x="dom-window-focus">focus()</code></dfn> method, when
  invoked, must run these steps:</p>

  <ol>
   <li><p>Let <var>current</var> be this <code>Window</code> object's <span data-x="window
   navigable">navigable</span>.</p></li>

   <li><p>If <var>current</var> is null, then return.</p></li>

   <li><p>If the <span>allow focus steps</span> given <var>current</var>'s <span
   data-x="nav-document">active document</span> return false, then return.</p></li>

   <li><p>Run the <span>focusing steps</span> with <var>current</var>.</p></li>

   <li><p>If <var>current</var> is a <span>top-level traversable</span>, user agents are encouraged
   to trigger some sort of notification to indicate to the user that the page is attempting to gain
   focus.</p></li>
  </ol>

  <p>The <dfn method for="Window"><code data-x="dom-window-blur">blur()</code></dfn> method steps
  are to do nothing.</p>

  <p class="note">Historically, the <code data-x="dom-window-focus">focus()</code> and <code
  data-x="dom-window-blur">blur()</code> methods actually affected the system-level focus of the
  system widget (e.g., tab or window) that contained the <span>navigable</span>, but hostile sites
  widely abuse this behavior to the user's detriment.</p>

  <p>The <dfn method for="HTMLOrSVGElement"><code data-x="dom-focus">focus(<var>options</var>)</code></dfn>
  method on elements, when invoked, must run the following steps:</p>

  <ol>
   <li><p>If the <span>allow focus steps</span> given the element's <span>node document</span>
   return false, then return.</p></li>

   <li><p>If the element is marked as <i data-x="locked for focus">locked for focus</i>, then return.</p></li>

   <li><p>Mark the element as <dfn>locked for focus</dfn>.</p></li>

   <li><p>Run the <span>focusing steps</span> for the element.</p></li>

   <li><p>If the value of the <dfn dict-member for="FocusOptions"><code
   data-x="dom-focusoptions-focusvisible">focusVisible</code></dfn> dictionary member of
   <var>options</var> is true, or is not present but in an <span>implementation-defined</span> way
   the user agent determines it would be best to do so, then <span>indicate focus</span>.</p></li>

   <li><p>If the value of the <dfn dict-member for="FocusOptions"><code
   data-x="dom-focusoptions-preventscroll">preventScroll</code></dfn> dictionary member of
   <var>options</var> is false, then <span data-x="scroll a target into view">scroll the element
   into view</span> given "auto", "center", and "center".</p></li>

   <li><p>Unmark the element as <i data-x="locked for focus">locked for focus</i>.</p></li>
  </ol>

  <p>The <dfn method for="HTMLOrSVGElement"><code data-x="dom-blur">blur()</code></dfn> method, when
  invoked, should run the <span>unfocusing steps</span> for the element on which the method was
  called. User agents may selectively or uniformly ignore calls to this method for usability
  reasons.</p>

  <p class="example">For example, if the <code data-x="dom-blur">blur()</code> method is unwisely
  being used to remove the focus ring for aesthetics reasons, the page would become unusable by
  keyboard users. Ignoring calls to this method would thus allow keyboard users to interact with the
  page.</p>

  <p>The <dfn export>allow focus steps</dfn>, given a <code>Document</code> object <var>target</var>,
  are as follows:</p>

  <ol>
   <li><p>If <var>target</var> is <span>allowed to use</span> the "<code
   data-x="focus-without-user-activation-feature">focus-without-user-activation</code>" feature,
   then return true.</p></li>

   <li>
    <p>If any of the following are true:</p>

    <ul>
     <li><p><var>target</var>'s <span>relevant global object</span> has <span data-x="transient
     activation">transient user activation</span>; or</p></li>

     <li><p><var>target</var>'s <span>node navigable</span>'s <span data-x="nav-container">container</span>,
     if any, is marked as <i data-x="locked for focus">locked for focus</i>,</p></li>
    </ul>

    <p>then return true.</p>
   </li>

   <li><p>Return false.</p></li>
  </ol>

  </div>


  <h4><span id="autofocusing-a-form-control:-the-autofocus-attribute"></span>The <code
  data-x="attr-fe-autofocus">autofocus</code> attribute</h4>

  <p>The <dfn element-attr for="html-global"><code data-x="attr-fe-autofocus">autofocus</code></dfn>
  content attribute allows the author to indicate that an element is to be focused as soon as the
  page is loaded, allowing the user to just start typing without having to manually focus the main
  element.</p>

  <p>When the <code data-x="attr-fe-autofocus">autofocus</code> attribute is specified on an element
  inside <code>dialog</code> elements or <span>HTML elements</span> whose <code
  data-x="attr-popover">popover</code> attribute is set, then it will be focused when the dialog or
  popover becomes shown.</p>

  <p>The <code data-x="attr-fe-autofocus">autofocus</code> attribute is a <span>boolean
  attribute</span>.</p>

  <p>To find the <dfn>nearest ancestor autofocus scoping root element</dfn> given an
  <code>Element</code> <var>element</var>:</p>

  <ol>
   <li><p>If <var>element</var> is a <code>dialog</code> element, then return
   <var>element</var>.</p></li>

   <li><p>If <var>element</var>'s <code data-x="attr-popover">popover</code> attribute is not in the
   <span data-x="attr-popover-none-state">no popover state</span>, then return
   <var>element</var>.</p></li>

   <li><p>Let <var>ancestor</var> be <var>element</var>.</p></li>

   <li>
    <p>While <var>ancestor</var> has a <span>parent element</span>:</p>

    <ol>
     <li><p>Set <var>ancestor</var> to <var>ancestor</var>'s <span>parent element</span>.</p></li>

     <li><p>If <var>ancestor</var> is a <code>dialog</code> element, then return
     <var>ancestor</var>.</p></li>

     <li><p>If <var>ancestor</var>'s <code data-x="attr-popover">popover</code> attribute is not in
     the <span data-x="attr-popover-none-state">no popover state</span>, then return
     <var>ancestor</var>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>ancestor</var>.</p></li>
  </ol>

  <p>There must not be two elements with the same <span>nearest ancestor autofocus scoping root
  element</span> that both have the <code data-x="attr-fe-autofocus">autofocus</code> attribute
  specified.</p>

  <div w-nodev>

  <p>Each <code>Document</code> has an <dfn>autofocus candidates</dfn> <span>list</span>,
  initially empty.</p>

  <p>Each <code>Document</code> has an <dfn>autofocus processed flag</dfn> boolean, initially
  false.</p>

  <p>When an element with the <code data-x="attr-fe-autofocus">autofocus</code> attribute specified
  is <span data-x="node is inserted into a document">inserted into a document</span>, run the
  following steps:</p>

  <ol>
   <li><p>If the user has indicated (for example, by starting to type in a form control) that they
   do not wish focus to be changed, then optionally return.</p></li>

   <li><p>Let <var>target</var> be the element's <span>node document</span>.</p></li>

   <li><p>If <var>target</var> is not <span>fully active</span>, then return.</p></li>

   <li><p>If <var>target</var>'s <span>active sandboxing flag set</span> has the
   <span>sandboxed automatic features browsing context flag</span>, then return.</p></li>

   <li><p>If the <span>allow focus steps</span> given <var>target</var> return false, then
   return.</p></li>

   <li><p><span data-x="list iterate">For each</span> <var>ancestorNavigable</var> of
   <var>target</var>'s <span>ancestor navigables</span>: if <var>ancestorNavigable</var>'s <span
   data-x="nav-document">active document</span>'s <span
   data-x="concept-document-origin">origin</span> is not <span>same origin</span> with
   <var>target</var>'s <span data-x="concept-document-origin">origin</span>, then return.</p></li>

   <li><p>Let <var>topDocument</var> be <var>target</var>'s <span>node navigable</span>'s <span
   data-x="nav-top">top-level traversable</span>'s <span data-x="nav-document">active
   document</span>.</p></li>

   <li><p>If <var>topDocument</var>'s <span>autofocus processed flag</span> is false, then
   <span data-x="list remove">remove</span> the element from <var>topDocument</var>'s
   <span>autofocus candidates</span>, and <span data-x="list append">append</span> the element to
   <var>topDocument</var>'s <span>autofocus candidates</span>.</p></li>
  </ol>

  <p class="note">We do not check if an element is a <span>focusable area</span> before storing it
  in the <span>autofocus candidates</span> list, because even if it is not a focusable area when it
  is inserted, it could become one by the time <span>flush autofocus candidates</span> sees it.</p>

  <p>To <dfn>flush autofocus candidates</dfn> for a document <var>topDocument</var>, run these
  steps:</p>

  <ol>
   <li><p>If <var>topDocument</var>'s <span>autofocus processed flag</span> is true, then
   return.</p></li>

   <li><p>Let <var>candidates</var> be <var>topDocument</var>'s <span>autofocus
   candidates</span>.</p></li>

   <li><p>If <var>candidates</var> <span data-x="list is empty">is empty</span>, then
   return.</p></li>

   <li>
    <p>If <var>topDocument</var>'s <span data-x="focused area of the document">focused area</span>
    is not <var>topDocument</var> itself, or <var>topDocument</var> has non-null
    <span>target element</span>, then:</p>

    <ol>
     <li><p><span data-x="list empty">Empty</span> <var>candidates</var>.</p></li>

     <li><p>Set <var>topDocument</var>'s <span>autofocus processed flag</span> to true.</p></li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li>
    <p>While <var>candidates</var> is not <span data-x="list is empty">empty</span>:</p>

    <ol>
     <li><p>Let <var>element</var> be <var>candidates</var>[0].</p></li>

     <li><p>Let <var>doc</var> be <var>element</var>'s <span>node document</span>.</p></li>

     <li><p>If <var>doc</var> is not <span>fully active</span>, then
     <span data-x="list remove">remove</span> <var>element</var> from <var>candidates</var>, and
     <span>continue</span>.</p></li>

     <li><p>If <var>doc</var>'s <span>node navigable</span>'s <span data-x="nav-top">top-level
     traversable</span> is not the same as <var>topDocument</var>'s <span>node navigable</span>,
     then <span data-x="list remove">remove</span> <var>element</var> from <var>candidates</var>,
     and <span>continue</span>.</p></li>

     <li>
      <p>If <var>doc</var>'s <span>script-blocking style sheet set</span> is not <span data-x="list
      empty">empty</span>, then return.</p>

      <p class="note">In this case, <var>element</var> is the currently-best candidate, but
      <var>doc</var> is not ready for autofocusing. We'll try again next time <span>flush
      autofocus candidates</span> is called.</p>
     </li>

     <li><p><span data-x="list remove">Remove</span> <var>element</var> from
     <var>candidates</var>.</p></li>

     <li><p>Let <var>inclusiveAncestorDocuments</var> be a <span>list</span> consisting of the
     <span data-x="nav-document">active document</span> of <var>doc</var>'s <span>inclusive
     ancestor navigables</span>.</p></li>

     <li><p>If any <code>Document</code> in <var>inclusiveAncestorDocuments</var> has non-null
     <span>target element</span>, then <span>continue</span>.</p></li>

     <li><p>Let <var>target</var> be <var>element</var>.</p></li>

     <li>
      <p>If <var>target</var> is not a <span>focusable area</span>, then set <var>target</var> to
      the result of <span data-x="get the focusable area">getting the focusable area</span> for
      <var>target</var>.</p>

      <p class="note"><span>Autofocus candidates</span> can <span data-x="list
      contains">contain</span> elements which are not <span data-x="focusable area">focusable
      areas</span>. In addition to the special cases handled in the <span>get the focusable
      area</span> algorithm, this can happen because a non-<span>focusable area</span> element with
      an <code data-x="attr-fe-autofocus">autofocus</code> attribute was <span data-x="node is
      inserted into a document">inserted into a document</span> and it never became focusable, or
      because the element was focusable but its status changed while it was stored in
      <span>autofocus candidates</span>.</p>
     </li>

     <li>
      <p>If <var>target</var> is not null, then:</p>

      <ol>
       <li><p><span data-x="list empty">Empty</span> <var>candidates</var>.</p></li>

       <li><p>Set <var>topDocument</var>'s <span>autofocus processed flag</span> to true.</p></li>

       <li><p>Run the <span>focusing steps</span> for <var>target</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>
  </ol>

  <p class="note">This handles the automatic focusing during document load. The <code
  data-x="dom-dialog-show">show()</code> and <code data-x="dom-dialog-showModal">showModal()</code>
  methods of <code>dialog</code> elements also processes the <code
  data-x="attr-fe-autofocus">autofocus</code> attribute.</p>

  <p class="note">Focusing the element does not imply that the user agent has to focus the browser
  window if it has lost focus.</p>

  <p>The <dfn attribute for="HTMLOrSVGElement"><code
  data-x="dom-fe-autofocus">autofocus</code></dfn> IDL attribute must <span>reflect</span> the
  content attribute of the same name.</p>

  </div>

  <div class="example">
   <p>In the following snippet, the text control would be focused when
   the document was loaded.</p>
   <pre><code class="html">&lt;input maxlength="256" name="q" value="" autofocus&gt;
&lt;input type="submit" value="Search"&gt;</code></pre>
  </div>

  <div class="example">
   <p>The <code data-x="attr-fe-autofocus">autofocus</code> attribute applies to all elements, not
   just to form controls. This allows examples such as the following:</p>
   <pre><code class="html">&lt;div contenteditable autofocus&gt;Edit &lt;strong>me!&lt;/strong>&lt;div></code></pre>
  </div>



  <h3>Assigning keyboard shortcuts</h3>

  <h4>Introduction</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>Each element that can be activated or focused can be assigned a single key combination to
  activate it, using the <code data-x="attr-accesskey">accesskey</code> attribute.</p>

  <p>The exact shortcut is determined by the user agent, based on information about the user's
  keyboard, what keyboard shortcuts already exist on the platform, and what other shortcuts have
  been specified on the page, using the information provided in the <code
  data-x="attr-accesskey">accesskey</code> attribute as a guide.</p>

  <p>In order to ensure that a relevant keyboard shortcut is available on a wide variety of input
  devices, the author can provide a number of alternatives in the <code
  data-x="attr-accesskey">accesskey</code> attribute.</p>

  <p>Each alternative consists of a single character, such as a letter or digit.</p>

  <p>User agents can provide users with a list of the keyboard shortcuts, but authors are encouraged
  to do so also. The <code data-x="dom-accessKeyLabel">accessKeyLabel</code> IDL attribute returns a
  string representing the actual key combination assigned by the user agent.</p>

  <div class="example">

   <p>In this example, an author has provided a button that can be invoked using a shortcut key. To
   support full keyboards, the author has provided "C" as a possible key. To support devices
   equipped only with numeric keypads, the author has provided "1" as another possible key.</p>

   <pre><code class="html">&lt;input type=button value=Collect onclick="collect()"
       <strong>accesskey="C 1"</strong> id=c></code></pre>

  </div>

  <div class="example">

   <p>To tell the user what the shortcut key is, the author has this script here opted to explicitly
   add the key combination to the button's label:</p>

   <pre><code class="js">function addShortcutKeyLabel(button) {
<strong>  if (button.accessKeyLabel != '')
    button.value += ' (' + button.accessKeyLabel + ')';</strong>
}
addShortcutKeyLabel(document.getElementById('c'));</code></pre>

   <p>Browsers on different platforms will show different labels, even for the same key combination,
   based on the convention prevalent on that platform. For example, if the key combination is the
   Control key, the Shift key, and the letter C, a Windows browser might display
   "<samp>Ctrl+Shift+C</samp>", whereas a Mac browser might display "<samp>^&#x21E7;C</samp>", while
   an Emacs browser might just display "<samp>C-C</samp>". Similarly, if the key combination is the
   Alt key and the Escape key, Windows might use "<samp>Alt+Esc</samp>", Mac might use
   "<samp>&#x2325;&#x238B;</samp>", and an Emacs browser might use "<samp>M-ESC</samp>" or
   "<samp>ESC ESC</samp>".</p>

   <p>In general, therefore, it is unwise to attempt to parse the value returned from the <code
   data-x="dom-accessKeyLabel">accessKeyLabel</code> IDL attribute.</p>

  </div>


  <h4>The <dfn element-attr for="html-global"><code data-x="attr-accesskey">accesskey</code></dfn>
  attribute</h4>

  <p>All <span>HTML elements</span> may have the <code data-x="attr-accesskey">accesskey</code>
  content attribute set. The <code data-x="attr-accesskey">accesskey</code> attribute's value is used
  by the user agent as a guide for creating a keyboard shortcut that activates or focuses the
  element.</p>

  <p>If specified, the value must be an <span>ordered set of unique space-separated tokens</span>
  none of which are <span>identical to</span> another token and each of which must be exactly one
  code point in length.</p>

  <div class="example">

   <p>In the following example, a variety of links are given with access keys so that keyboard users
   familiar with the site can more quickly navigate to the relevant pages:</p>

   <pre><code class="html">&lt;nav>
 &lt;p>
  &lt;a title="Consortium Activities" accesskey="A" href="/Consortium/activities">Activities&lt;/a> |
  &lt;a title="Technical Reports and Recommendations" accesskey="T" href="/TR/">Technical Reports&lt;/a> |
  &lt;a title="Alphabetical Site Index" accesskey="S" href="/Consortium/siteindex">Site Index&lt;/a> |
  &lt;a title="About This Site" accesskey="B" href="/Consortium/">About Consortium&lt;/a> |
  &lt;a title="Contact Consortium" accesskey="C" href="/Consortium/contact">Contact&lt;/a>
 &lt;/p>
&lt;/nav></code></pre>

  </div>

  <div class="example">

   <p>In the following example, the search field is given two possible access keys, "s" and "0" (in
   that order). A user agent on a device with a full keyboard might pick <kbd><kbd>Ctrl</kbd> +
   <kbd>Alt</kbd> + <kbd>S</kbd></kbd> as the shortcut key, while a user agent on a small device
   with just a numeric keypad might pick just the plain unadorned key <kbd><kbd>0</kbd></kbd>:</p>

   <pre><code class="html">&lt;form action="/search">
 &lt;label>Search: &lt;input type="search" name="q" accesskey="s 0">&lt;/label>
 &lt;input type="submit">
&lt;/form></code></pre>

  </div>

  <div class="example">

   <p>In the following example, a button has possible access keys described. A script then tries to
   update the button's label to advertise the key combination the user agent selected.</p>

   <pre><code class="html">&lt;input type=submit accesskey="N @ 1" value="Compose">
...
&lt;script>
 function labelButton(button) {
   if (button.accessKeyLabel)
     button.value += ' (' + button.accessKeyLabel + ')';
 }
 var inputs = document.getElementsByTagName('input');
 for (var i = 0; i &lt; inputs.length; i += 1) {
   if (inputs[i].type == "submit")
     labelButton(inputs[i]);
 }
&lt;/script></code></pre>

   <p>On one user agent, the button's label might become "<samp>Compose (&#x2318;N)</samp>". On
   another, it might become "<samp>Compose (Alt+&#x21E7;+1)</samp>". If the user agent doesn't
   assign a key, it will be just "<samp>Compose</samp>". The exact string depends on what the
   <span>assigned access key</span> is, and on how the user agent represents that key
   combination.</p>

  </div>


  <div w-nodev>

  <h4 id="keyboard-shortcuts-processing-model"><span id="processing-model-6"></span>Processing
  model</h4>

  <p>An element's <dfn>assigned access key</dfn> is a key combination derived from the element's
  <code data-x="attr-accesskey">accesskey</code> content attribute. Initially, an element must not
  have an <span>assigned access key</span>.</p>

  <p>Whenever an element's <code data-x="attr-accesskey">accesskey</code> attribute is set, changed,
  or removed, the user agent must update the element's <span>assigned access key</span> by running
  the following steps:</p>

  <ol>
   <li><p>If the element has no <code data-x="attr-accesskey">accesskey</code> attribute, then skip
   to the <i>fallback</i> step below.</p></li>

   <li><p>Otherwise, <span data-x="split a string on ASCII whitespace">split the attribute's value
   on ASCII whitespace</span>, and let <var>keys</var> be the resulting tokens.</p></li>

   <li>
    <p>For each value in <var>keys</var> in turn, in the order the tokens appeared in the
    attribute's value, run the following substeps:</p>

    <ol>
     <li><p>If the value is not a string exactly one code point in length, then skip the remainder
     of these steps for this value.</p></li>

     <li><p>If the value does not correspond to a key on the system's keyboard, then skip the
     remainder of these steps for this value.</p></li>

     <li><p>
     <!--INSERT TRACKING-->
     If the user agent can find a mix of zero or more modifier keys that, combined with the key that
     corresponds to the value given in the attribute, can be used as the access key, then the user
     agent may assign that combination of keys as the element's <span>assigned access key</span> and
     return.</p></li>
    </ol>
   </li>

   <li><p><i>Fallback</i>: Optionally, the user agent may assign a key combination of its choosing
   as the element's <span>assigned access key</span> and then return.</p></li>

   <li><p>If this step is reached, the element has no <span>assigned access key</span>.</p></li>
  </ol>

  <p>Once a user agent has selected and assigned an access key for an element, the user agent should
  not change the element's <span>assigned access key</span> unless the <code
  data-x="attr-accesskey">accesskey</code> content attribute is changed or the element is moved to
  another <code>Document</code>.</p>

  <p>When the user presses the key combination corresponding to the <span>assigned access key</span>
  for an element, if the element <span data-x="concept-command">defines a command</span>, the
  command's <span data-x="command-facet-HiddenState">Hidden State</span> facet is false (visible),
  the command's <span data-x="command-facet-DisabledState">Disabled State</span> facet is also false
  (enabled), the element is <span>in a document</span> that has a non-null <span
  data-x="concept-document-bc">browsing context</span>, and neither the element nor any of its
  ancestors has a <code data-x="attr-hidden">hidden</code> attribute specified, then the user agent
  must trigger the <span data-x="command-facet-Action">Action</span> of the command.</p>

  <p class="note">User agents <a href="#expose-commands-in-ui">might expose</a> elements that have
  an <code data-x="attr-accesskey">accesskey</code> attribute in other ways as well, e.g. in a menu
  displayed in response to a specific key combination.</p> <!-- the actual conformance criteria for
  this is in the section that defines commands -->

  <hr>

  <p>The <dfn attribute for="HTMLElement"><code data-x="dom-accessKey">accessKey</code></dfn> IDL
  attribute must <span>reflect</span> the <code data-x="attr-accesskey">accesskey</code> content
  attribute.</p>

  <p>The <dfn attribute for="HTMLElement"><code
  data-x="dom-accessKeyLabel">accessKeyLabel</code></dfn> IDL attribute must return a string that
  represents the element's <span>assigned access key</span>, if any. If the element does not have
  one, then the IDL attribute must return the empty string.</p>

  </div>



  <h3>Editing</h3>

  <h4 id="contenteditable">Making document regions editable: The <code
  data-x="attr-contenteditable">contenteditable</code> content attribute</h4>

  <pre><code class="idl">interface mixin <dfn interface>ElementContentEditable</dfn> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-contentEditable">contentEditable</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-enterKeyHint">enterKeyHint</span>;
  readonly attribute boolean <span data-x="dom-isContentEditable">isContentEditable</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-inputMode">inputMode</span>;
};</code></pre>

  <p>The <dfn element-attr for="html-global"><code
  data-x="attr-contenteditable">contenteditable</code></dfn> content attribute is an
  <span>enumerated attribute</span> with the following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="html-global/contenteditable"><code
     data-x="attr-contenteditable-true">true</code></dfn>
     <td rowspan=2><dfn data-x="attr-contenteditable-true-state">true</dfn>
     <td rowspan=2>The element is editable.
    <tr>
     <td>(the empty string)
    <tr>
     <td><dfn attr-value for="html-global/contenteditable"><code
     data-x="attr-contenteditable-false">false</code></dfn>
     <td><dfn data-x="attr-contenteditable-false-state">false</dfn>
     <td>The element is not editable.
    <tr>
     <td><dfn attr-value for="html-global/contenteditable"><code
     data-x="attr-contenteditable-plaintextonly">plaintext-only</code></dfn>
     <td><dfn data-x="attr-contenteditable-plaintextonly-state">plaintext-only</dfn>
     <td>Only the element's raw text content is editable; rich formatting is disabled.
  </table>


  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <dfn
  data-x="attr-contenteditable-inherit-state">inherit</dfn> state. The inherit state indicates that
  the element is editable (or not) based on the parent element's state.</p>

  <div class="example">
   <p>For example, consider a page that has a <code>form</code> and a <code>textarea</code> to
   publish a new article, where the user is expected to write the article using HTML:</p>

   <pre><code class="html">&lt;form method=POST>
 &lt;fieldset>
  &lt;legend>New article&lt;/legend>
  &lt;textarea name=article>&amp;lt;p>Hello world.&amp;lt;/p>&lt;/textarea>
 &lt;/fieldset>
 &lt;p>&lt;button>Publish&lt;/button>&lt;/p>
&lt;/form></code></pre>

   <p>When scripting is enabled, the <code>textarea</code> element could be replaced with a rich
   text control instead, using the <code data-x="attr-contenteditable">contenteditable</code>
   attribute:</p>

   <pre><code class="html">&lt;form method=POST>
 &lt;fieldset>
  &lt;legend>New article&lt;/legend>
  &lt;textarea id=textarea name=article>&amp;lt;p>Hello world.&amp;lt;/p>&lt;/textarea>
  &lt;div id=div style="white-space: pre-wrap" hidden>&lt;p>Hello world.&lt;/p>&lt;/div>
  &lt;script>
   let textarea = document.getElementById("textarea");
   let div = document.getElementById("div");
   textarea.hidden = true;
   div.hidden = false;
   div.contentEditable = "true";
   div.oninput = (e) => {
     textarea.value = div.innerHTML;
   };
  &lt;/script>
 &lt;/fieldset>
 &lt;p>&lt;button>Publish&lt;/button>&lt;/p>
&lt;/form></code></pre>

   <p>Features to enable, e.g., inserting links, can be implemented using the <code
   data-x="dom-document-execCommand">document.execCommand()</code> API, or using
   <code>Selection</code> APIs and other DOM APIs. <ref>EXECCOMMAND</ref> <ref>SELECTION</ref>
   <ref>DOM</ref></p>
  </div>

  <div class="example">
   <p>The <code data-x="attr-contenteditable">contenteditable</code> attribute can also be used to
   great effect:</p>

   <pre><code class="html">&lt;!doctype html>
&lt;html lang=en>
&lt;title>Live CSS editing!&lt;/title>
&lt;style style=white-space:pre contenteditable>
html { margin:.2em; font-size:2em; color:lime; background:purple }
head, title, style { display:block }
body { display:none }
&lt;/style></code></pre>
  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-contentEditable">contentEditable</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns "<code data-x="">true</code>", "<code data-x="">plaintext-only</code>", "<code
    data-x="">false</code>", or "<code data-x="attr-contenteditable-inherit-state">inherit</code>",
    based on the state of the <code data-x="attr-contenteditable">contenteditable</code>
    attribute.</p>

    <p>Can be set, to change that state.</p>

    <p>Throws a <span>"<code>SyntaxError</code>"</span> <code>DOMException</code> if the new value
    isn't one of those strings.</p>
   </dd>

   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-isContentEditable">isContentEditable</span></code></dt>

   <dd><p>Returns true if the element is editable; otherwise, returns false.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="ElementContentEditable"><code
  data-x="dom-contentEditable">contentEditable</code></dfn> IDL attribute, on getting, must return
  the string "<code data-x="">true</code>" if the content attribute is set to the <span
  data-x="attr-contenteditable-true-state">true</span> state, "<code
  data-x="">plaintext-only</code>" if the content attribute is set to the <span
  data-x="attr-contenteditable-plaintextonly-state">plaintext-only</span> state, "<code
  data-x="">false</code>" if the content attribute is set to the <span
  data-x="attr-contenteditable-false-state">false</span> state, and "<code
  data-x="attr-contenteditable-inherit-state">inherit</code>" otherwise. On setting, if the new
  value is an <span>ASCII case-insensitive</span> match for the string "<code
  data-x="">inherit</code>" then the content attribute must be removed, if the new value is an
  <span>ASCII case-insensitive</span> match for the string "<code data-x="">true</code>" then the
  content attribute must be set to the string "<code data-x="">true</code>", if the new value is an
  <span>ASCII case-insensitive</span> match for the string "<code data-x="">plaintext-only</code>"
  then the content attribute must be set to the string "<code data-x="">plaintext-only</code>", if
  the new value is an <span>ASCII case-insensitive</span> match for the string "<code
  data-x="">false</code>" then the content attribute must be set to the string "<code
  data-x="">false</code>", and otherwise the attribute setter must throw a
  <span>"<code>SyntaxError</code>"</span> <code>DOMException</code>.</p>

  <p>The <dfn attribute for="ElementContentEditable"><code
  data-x="dom-isContentEditable">isContentEditable</code></dfn> IDL attribute, on getting, must
  return true if the element is either an <span>editing host</span> or <span>editable</span>, and
  false otherwise.</p>

  </div>



  <h4 id="making-entire-documents-editable:-the-designmode-idl-attribute">Making entire documents
  editable: the <code data-x="dom-document-designMode">designMode</code> getter and setter</h4>

  <dl class="domintro">
   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-designMode">designMode</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns "<code data-x="">on</code>" if the document is editable, and "<code
    data-x="">off</code>" if it isn't.</p>

    <p>Can be set, to change the document's current state. This focuses the document and resets the
    selection in that document.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p><code>Document</code> objects have an associated <dfn>design mode enabled</dfn>, which is a
  boolean. It is initially false.</p>

  <p>The <dfn attribute for="Document" id="designMode"><code
  data-x="dom-document-designMode">designMode</code></dfn> getter steps are to return "<code
  data-x="">on</code>" if <span>this</span>'s <span>design mode enabled</span> is true; otherwise
  "<code data-x="">off</code>".</p>

  <p>The <code data-x="dom-document-designMode">designMode</code> setter steps are:</p>

  <ol>
   <li><p>Let <var>value</var> be the given value, <span>converted to ASCII
   lowercase</span>.</p></li>

   <li>
    <p>If <var>value</var> is "<code data-x="">on</code>" and <span>this</span>'s <span>design mode
    enabled</span> is false, then:</p>

    <ol>
     <li><p>Set <span>this</span>'s <span>design mode enabled</span> to true.</p></li>

     <li><p>Reset <span>this</span>'s <span>active range</span>'s start and end boundary points to
     be at the start of <span>this</span>.</p></li>

     <li><p>Run the <span>focusing steps</span> for <span>this</span>'s <span>document
     element</span>, if non-null.</p></li>
    </ol>
   </li>

   <li><p>If <var>value</var> is "<code data-x="">off</code>", then set <span>this</span>'s
   <span>design mode enabled</span> to false.</p></li>
  </ol>

  </div>


  <h4>Best practices for in-page editors</h4>

  <p>Authors are encouraged to set the <span>'white-space'</span> property on <span data-x="editing
  host">editing hosts</span> and on markup that was originally created through these editing
  mechanisms to the value 'pre-wrap'. Default HTML whitespace handling is not well suited to WYSIWYG
  editing, and line wrapping will not work correctly in some corner cases if
  <span>'white-space'</span> is left at its default value.</p>

  <div class="example">

   <p>As an example of problems that occur if the default 'normal' value is used instead, consider
   the case of the user typing "<kbd>yellow&#x2423;&#x2423;ball</kbd>", with two spaces (here
   represented by "&#x2423;") between the words. With the editing rules in place for the default
   value of <span>'white-space'</span> ('normal'), the resulting markup will either consist of
   "<samp>yellow&amp;nbsp;&nbsp;ball</samp>" or "<samp>yellow&nbsp;&amp;nbsp;ball</samp>"; i.e.,
   there will be a non-breaking space between the two words in addition to the regular space. This
   is necessary because the 'normal' value for <span>'white-space'</span> requires adjacent regular
   spaces to be collapsed together.</p>

   <p>In the former case, "<samp>yellow&#x237D;</samp>" might wrap to the next line ("&#x237D;"
   being used here to represent a non-breaking space) even though "<samp>yellow</samp>" alone might
   fit at the end of the line; in the latter case, "<samp>&#x237D;ball</samp>", if wrapped to the
   start of the line, would have visible indentation from the non-breaking space.</p>

   <p>When <span>'white-space'</span> is set to 'pre-wrap', however, the editing rules will instead
   simply put two regular spaces between the words, and should the two words be split at the end of
   a line, the spaces would be neatly removed from the rendering.</p>

  </div>

  <div w-nodev>

  <h4>Editing APIs</h4>

  <p>An <dfn export>editing host</dfn> is either an <span data-x="HTML elements">HTML element</span>
  with its <code data-x="attr-contenteditable">contenteditable</code> attribute in the <i>true</i>
  state or <i>plaintext-only</i> state, or a <span data-x="concept-tree-child">child</span> <span
  data-x="HTML elements">HTML element</span> of a <code>Document</code> whose <span>design mode
  enabled</span> is true.</p>

  <p>The definition of the terms <dfn
  data-x-href="https://w3c.github.io/editing/docs/execCommand/#active-range">active range</dfn>, <dfn
  data-x-href="https://w3c.github.io/editing/docs/execCommand/#editing-host-of">editing host
  of</dfn>, and <dfn
  data-x-href="https://w3c.github.io/editing/docs/execCommand/#editable">editable</dfn>, the user
  interface requirements of elements that are <span data-x="editing host">editing hosts</span> or
  <span>editable</span>, the

  <dfn data-x="dom-document-execCommand" id="execCommand" data-x-href="https://w3c.github.io/editing/docs/execCommand/#execcommand%28%29"><code>execCommand()</code></dfn>,
  <dfn data-x-href="https://w3c.github.io/editing/docs/execCommand/#querycommandenabled%28%29"><code data-x="dom-document-queryCommandEnabled">queryCommandEnabled()</code></dfn>,
  <dfn data-x-href="https://w3c.github.io/editing/docs/execCommand/#querycommandindeterm%28%29"><code data-x="dom-document-queryCommandIndeterm">queryCommandIndeterm()</code></dfn>,
  <dfn data-x-href="https://w3c.github.io/editing/docs/execCommand/#querycommandstate%28%29"><code data-x="dom-document-queryCommandState">queryCommandState()</code></dfn>,
  <dfn data-x-href="https://w3c.github.io/editing/docs/execCommand/#querycommandsupported%28%29"><code data-x="dom-document-queryCommandSupported">queryCommandSupported()</code></dfn>, and
  <dfn data-x-href="https://w3c.github.io/editing/docs/execCommand/#querycommandvalue%28%29"><code data-x="dom-document-queryCommandValue">queryCommandValue()</code></dfn>

  methods, text selections, and the <dfn
  data-x-href="https://w3c.github.io/editing/docs/execCommand/#delete-the-selection">delete the
  selection</dfn> algorithm are defined in <cite>execCommand</cite>. <ref>EXECCOMMAND</ref></p>

  </div>

  <h4>Spelling and grammar checking</h4>

  <div w-nodev>

  <p>User agents can support the checking of spelling and grammar of editable text, either in form
  controls (such as the value of <code>textarea</code> elements), or in elements in an <span>editing
  host</span> (e.g. using <code data-x="attr-contenteditable">contenteditable</code>).</p>

  <p>For each element, user agents must establish a <dfn data-x="concept-spellcheck-default">default
  behavior</dfn>, either through defaults or through preferences expressed by the user. There are
  three possible default behaviors for each element:</p>

  <dl>
   <dt><dfn data-x="concept-spellcheck-default-true">true-by-default</dfn>

   <dd>The element will be checked for spelling and grammar if its contents are editable and
   spellchecking is not explicitly disabled through the <code
   data-x="attr-spellcheck">spellcheck</code> attribute.

   <dt><dfn data-x="concept-spellcheck-default-false">false-by-default</dfn>

   <dd>The element will never be checked for spelling and grammar unless spellchecking is
   explicitly enabled through the <code data-x="attr-spellcheck">spellcheck</code> attribute.

   <dt><dfn data-x="concept-spellcheck-default-inherit">inherit-by-default</dfn>

   <dd>The element's default behavior is the same as its parent element's. Elements that have no
   parent element cannot have this as their default behavior.
  </dl>

  <hr>

  </div>

  <p>The <dfn element-attr for="html-global"><code data-x="attr-spellcheck">spellcheck</code></dfn>
  attribute is an <span>enumerated attribute</span> with the following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="html-global/spellcheck"><code
     data-x="attr-spellcheck-true">true</code></dfn>
     <td rowspan=2><dfn data-x="attr-spellcheck-true-state">true</dfn>
     <td rowspan=2>Spelling and grammar will be checked.
    <tr>
     <td>(the empty string)
    <tr>
     <td><dfn attr-value for="html-global/spellcheck"><code
     data-x="attr-spellcheck-false">false</code></dfn>
     <td><dfn data-x="attr-spellcheck-false-state">false</dfn>
     <td>Spelling and grammar will not be checked.
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <dfn
  data-x="attr-spellcheck-default-state">default</dfn> state. The default state indicates that the
  element is to act according to a default behavior, possibly based on the parent element's own
  <code data-x="attr-spellcheck">spellcheck</code> state, as defined below.</p>

  <div w-nodev>

  <hr>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-spellcheck">spellcheck</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns true if the element is to have its spelling and grammar checked; otherwise, returns
    false.</p>

    <p>Can be set, to override the default and set the <code
    data-x="attr-spellcheck">spellcheck</code> content attribute.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLElement"><code data-x="dom-spellcheck">spellcheck</code></dfn> IDL
  attribute, on getting, must return true if the element's <code
  data-x="attr-spellcheck">spellcheck</code> content attribute is in the <span
  data-x="attr-spellcheck-true-state">true</span> state, or if the element's <code
  data-x="attr-spellcheck">spellcheck</code> content attribute is in the <span
  data-x="attr-spellcheck-default-state">default</span> state and the element's <span
  data-x="concept-spellcheck-default">default behavior</span> is <span
  data-x="concept-spellcheck-default-true">true-by-default</span>, or if the element's <code
  data-x="attr-spellcheck">spellcheck</code> content attribute is in the <span
  data-x="attr-spellcheck-default-state">default</span> state and the element's <span
  data-x="concept-spellcheck-default">default behavior</span> is <span
  data-x="concept-spellcheck-default-inherit">inherit-by-default</span> and the element's parent
  element's <code data-x="dom-spellcheck">spellcheck</code> IDL attribute would return true;
  otherwise, if none of those conditions applies, then the attribute must instead return false.</p>

  <p class="note">The <code data-x="dom-spellcheck">spellcheck</code> IDL attribute is not affected
  by user preferences that override the <code data-x="attr-spellcheck">spellcheck</code> content
  attribute, and therefore might not reflect the actual spellchecking state.</p>

  <p>On setting, if the new value is true, then the element's <code
  data-x="attr-spellcheck">spellcheck</code> content attribute must be set to
  "<code data-x="">true</code>", otherwise it must be set to "<code data-x="">false</code>".

  <hr>

  <p>User agents should only consider the following pieces of text as checkable for the purposes of
  this feature:</p>

  <ul>
   <li>The <span data-x="concept-fe-value">value</span> of <code>input</code> elements whose <code
   data-x="attr-input-type">type</code> attributes are in the <span
   data-x="attr-input-type-text">Text</span>, <span data-x="attr-input-type-search">Search</span>,
   <span data-x="attr-input-type-url">URL</span>, or <span
   data-x="attr-input-type-email">Email</span> states and that are <i
   data-x="concept-fe-mutable">mutable</i> (i.e. that do not have the <code
   data-x="attr-input-readonly">readonly</code> attribute specified and that are not <span
   data-x="concept-fe-disabled">disabled</span>).</li>

   <li>The <span data-x="concept-fe-value">value</span> of <code>textarea</code> elements that do not
   have a <code data-x="attr-textarea-readonly">readonly</code> attribute and that are not <span
   data-x="concept-fe-disabled">disabled</span>.</li>

   <li>Text in <code>Text</code> nodes that are children of <span data-x="editing host">editing
   hosts</span> or <span>editable</span> elements.</li>

   <li>Text in attributes of <span>editable</span> elements.</li>
  </ul>

  <p>For text that is part of a <code>Text</code> node, the element with which the text is
  associated is the element that is the immediate parent of the first character of the word,
  sentence, or other piece of text. For text in attributes, it is the attribute's element. For the
  values of <code>input</code> and <code>textarea</code> elements, it is the element itself.</p>

  <p>To determine if a word, sentence, or other piece of text in an applicable element (as defined
  above) is to have spelling- and grammar-checking enabled, the UA must use the following
  algorithm:</p>

  <ol>
   <!-- user override -->

   <li>If the user has disabled the checking for this text, then the checking is disabled.</li>

   <li>Otherwise, if the user has forced the checking for this text to always be enabled, then the
   checking is enabled.</li>

   <!-- content attribute: on, off -->

   <li>Otherwise, if the element with which the text is associated has a <code
   data-x="attr-spellcheck">spellcheck</code> content attribute, then: if that attribute is in the
   <span data-x="attr-spellcheck-true-state">true</span> state, then checking is enabled; otherwise, if that attribute is in the <span data-x="attr-spellcheck-false-state">false</span>
   state, then checking is disabled.</li>

   <!-- inherit, if there is one to inherit from -->

   <li>Otherwise, if there is an ancestor element with a <code
   data-x="attr-spellcheck">spellcheck</code> content attribute that is not in the <span
   data-x="attr-spellcheck-default-state">default</span>
   state, then: if the nearest such ancestor's <code data-x="attr-spellcheck">spellcheck</code>
   content attribute is in the <span data-x="attr-spellcheck-true-state">true</span> state, then
   checking is enabled; otherwise, checking is disabled.</li>

   <!-- default -->

   <li>Otherwise, if the element's <span data-x="concept-spellcheck-default">default behavior</span>
   is <span data-x="concept-spellcheck-default-true">true-by-default</span>, then checking is
   enabled.</li>

   <li>Otherwise, if the element's <span data-x="concept-spellcheck-default">default behavior</span>
   is <span data-x="concept-spellcheck-default-false">false-by-default</span>, then checking is
   disabled.</li>

   <!-- default inheritance -->

   <li>Otherwise, if the element's parent element has <em>its</em> checking enabled, then checking
   is enabled.</li>

   <li>Otherwise, checking is disabled.</li>
  </ol>

  <p>If the checking is enabled for a word/sentence/text, the user agent should indicate spelling
  and grammar errors in that text. User agents should take into account the other semantics given in
  the document when suggesting spelling and grammar corrections. User agents may use the language of
  the element to determine what spelling and grammar rules to use, or may use the user's preferred
  language settings. UAs should use <code>input</code> element attributes such as <code
  data-x="attr-input-pattern">pattern</code> to ensure that the resulting value is valid, where
  possible.</p>

  <p>If checking is disabled, the user agent should not indicate spelling or grammar errors for that
  text.</p>

  <div class="example">

   <p>The element with ID "a" in the following example would be the one used to determine if the
   word "Hello" is checked for spelling errors. In this example, it would not be.</p>

   <pre><code class="html">&lt;div contenteditable="true">
 &lt;span spellcheck="false" id="a">Hell&lt;/span>&lt;em>o!&lt;/em>
&lt;/div></code></pre>

   <p>The element with ID "b" in the following example would have checking enabled (the leading
   space character in the attribute's value on the <code>input</code> element causes the attribute
   to be ignored, so the ancestor's value is used instead, regardless of the default).</p>

   <pre class="bad"><code class="html">&lt;p spellcheck="true">
 &lt;label>Name: &lt;input spellcheck=" false" id="b">&lt;/label>
&lt;/p></code></pre>

  </div>

  </div>

  <p class="note">This specification does not define the user interface for spelling and grammar
  checkers. A user agent could offer on-demand checking, could perform continuous checking while the
  checking is enabled, or could use other interfaces.</p>

  <h4>Writing suggestions</h4>

  <p>User agents offer writing suggestions as users type into editable regions, either in form
  controls (e.g., the <code>textarea</code> element) or in elements in an <span  data-x="editing
  host"> editing host</span>.</p>

  <p>The <dfn element-attr for="html-global"><code
  data-x="attr-writingsuggestions">writingsuggestions</code></dfn> content attribute is an
  <span>enumerated attribute</span> with the following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="html-global/writingsuggestions"><code
     data-x="attr-writingsuggestions-true">true</code></dfn>
     <td rowspan="2"><dfn data-x="attr-writingsuggestions-true-state">true</dfn>
     <td rowspan="2">Writing suggestions should be offered on this element.
    </tr>
    <tr>
     <td>(the empty string)
    <tr>
     <td><dfn attr-value for="html-global/writingsuggestions"><code
      data-x="attr-writingsuggestions-false">false</code></dfn>
     <td><dfn data-x="attr-writingsuggestions-false-state">false</dfn>
     <td>Writing suggestions should not be offered on this element.
    </tr>
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> is the <dfn
  data-x="attr-writingsuggestions-default-state">default</dfn> state. The default state indicates
  that the element is to act according to a default behavior, possibly based on the parent
  element's own <code data-x="attr-writingsuggestions">writingsuggestions</code> state, as defined
  below.</p>

  <p>The attribute's <i data-x="invalid value default">invalid value default</i> is the <span
  data-x="attr-writingsuggestions-true-state">true</span> state.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-writingSuggestions">writingSuggestions</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns "<code data-x="">true</code>" if the user agent is to offer writing suggestions under
    the scope of the element; otherwise, returns "<code data-x="">false</code>".</p>

    <p>Can be set, to override the default and set the <code data-x="attr-writingsuggestions">
    writingsuggestions</code> content attribute.</p>
   </dd>
  </dl>

  <div w-nodev>
  <p>The <dfn>computed writing suggestions value</dfn> of a given <var>element</var> is
  determined  by running the following steps:</p>

  <ol>
   <li><p>If <var>element</var>'s <code data-x="attr-writingsuggestions">writingsuggestions</code>
   content attribute is in the <span data-x="attr-writingsuggestions-false-state">false</span>
   state, return "<code data-x="">false</code>".</p></li>

   <li><p>If <var>element</var>'s <code
   data-x="attr-writingsuggestions">writingsuggestions</code> content attribute is in the <span
   data-x="attr-writingsuggestions-default-state">default</span> state, <var>element</var> has a
   parent element, and the <span>computed writing suggestions value</span> of
   <var>element</var>'s parent element is "<code data-x="">false</code>", then return
   "<code data-x="">false</code>".</p></li>

   <li><p>Return "<code data-x="">true</code>".</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLElement"><code
  data-x="dom-writingSuggestions">writingSuggestions</code></dfn> getter steps are:</p>

  <ol>
   <li><p>Return <span>this</span>'s <span>computed writing suggestions value</span>.</p></li>
  </ol>

  <p class="note">The <code data-x="dom-writingSuggestions">writingSuggestions</code> IDL
  attribute is not affected by user preferences that override the <code
  data-x="attr-writingsuggestions">writingsuggestions</code> content attribute, and therefore
  might not reflect the actual writing suggestions state.</p>

  <p>The <code data-x="dom-writingSuggestions">writingSuggestions</code> setter steps are:</p>

  <ol>
   <li><p>Set <span>this</span>'s <code data-x="attr-writingsuggestions">writingsuggestions</code>
   content attribute to the given value.</p></li>
  </ol>
  </div>

  <hr>

  <p>User agents should only offer suggestions within an element's scope if the
  result of running the following algorithm given <var>element</var> returns true:</p>

  <ol>
   <!-- user override -->

   <li><p>If the user has disabled writing suggestions, then return false.</p></li>

   <!-- element type -->
   <li>
    <p>If none of the following conditions are true:</p>

    <ul>
     <li><p><var>element</var> is an <code>input</code> element whose <code
     data-x="attr-input-type">type</code> attribute is in either the <span
     data-x="attr-input-type-text">Text</span>, <span
     data-x="attr-input-type-search">Search</span>, <span
     data-x="attr-input-type-tel">Telephone</span>, <span data-x="attr-input-type-url">URL</span>,
     or <span data-x="attr-input-type-email">Email</span> state and is <i
     data-x="concept-fe-mutable">mutable</i>;</p></li>

     <li><p><var>element</var> is a <code>textarea</code> element that is <i
     data-x="concept-fe-mutable">mutable</i>; or</p></li>

     <li><p><var>element</var> is an <span data-x="editing host">editing host</span> or is
     <span>editable</span>,</p></li>
    </ul>

    <p>then return false.</p>
   </li>

   <!--
    content attribute: false on this element, or inherit if there is an ancestor to inherit from
   -->

   <li><p>If <var>element</var> has an <span>inclusive ancestor</span> with a <code
   data-x="attr-writingsuggestions">writingsuggestions</code> content attribute that's not in the
   <span data-x="attr-writingsuggestions-default-state">default</span> and the nearest such
   ancestor's <code data-x="attr-writingsuggestions">writingsuggestions</code> content attribute is
   in the <span data-x="attr-writingsuggestions-false-state">false</span> state, then return
   false.</p></li>

   <li><p>Otherwise, return true.</p></li>
  </ol>

  <p class="note">This specification does not define the user interface for writing suggestions.
  A user agent could offer on-demand suggestions, continuous suggestions as the user types, inline
  suggestions, autofill-like suggestions in a popup, or could use other interfaces.</p>

  <h4>Autocapitalization</h4>

  <p>Some methods of entering text, for example virtual keyboards on mobile devices, and also voice
  input, often assist users by automatically capitalizing the first letter of sentences (when
  composing text in a language with this convention). A virtual keyboard that implements
  autocapitalization might automatically switch to showing uppercase letters (but allow the user to
  toggle it back to lowercase) when a letter that should be autocapitalized is about to be typed.
  Other types of input, for example voice input, may perform autocapitalization in a way that does
  not give users an option to intervene first. The <code
  data-x="attr-autocapitalize">autocapitalize</code> attribute allows authors to control such
  behavior.</p>

  <p>The <code data-x="attr-autocapitalize">autocapitalize</code> attribute, as typically
  implemented, does not affect behavior when typing on a physical keyboard. (For this reason, as
  well as the ability for users to override the autocapitalization behavior in some cases or edit
  the text after initial input, the attribute must not be relied on for any sort of input
  validation.)</p>

  <p>The <code data-x="attr-autocapitalize">autocapitalize</code> attribute can be used on an <span
  data-x="editing host">editing host</span> to control autocapitalization behavior for the hosted
  editable region, on an <code>input</code> or <code>textarea</code> element to control the behavior
  for inputting text into that element, or on a <code>form</code> element to control the default
  behavior for all <span
  data-x="category-autocapitalize-and-autocorrect">autocapitalize-and-autocorrect inheriting
  elements</span> associated with the <code>form</code> element.

  <p>The <code data-x="attr-autocapitalize">autocapitalize</code> attribute never causes
  autocapitalization to be enabled for <code>input</code> elements whose <code
  data-x="attr-input-type">type</code> attribute is in one of the <span
  data-x="attr-input-type-url">URL</span>, <span
  data-x="attr-input-type-email">Email</span>, or <span
  data-x="attr-input-type-password">Password</span> states.<span w-nodev> (This behavior is included
  in the <span data-x="used autocapitalization hint">used autocapitalization hint</span> algorithm
  below.)</span></p>

  <p>The autocapitalization processing model is based on selecting among five
  <dfn data-x="autocapitalization hint">autocapitalization hints</dfn>, defined as follows:</p>

  <dl>
   <dt><dfn data-x="autocap-hint-default">default</dfn></dt>
   <dd><p>The user agent and input method should make their own determination of whether or not to
   enable autocapitalization.</p></dd>

   <dt><dfn data-x="autocap-hint-none">none</dfn></dt>
   <dd><p>No autocapitalization should be applied (all letters should default to
   lowercase).</p></dd>

   <dt><dfn data-x="autocap-hint-sentences">sentences</dfn></dt>
   <dd><p>The first letter of each sentence should default to a capital letter; all other letters
   should default to lowercase.</p></dd>

   <dt><dfn data-x="autocap-hint-words">words</dfn></dt>
   <dd><p>The first letter of each word should default to a capital letter; all other letters should
   default to lowercase.</p></dd>

   <dt><dfn data-x="autocap-hint-characters">characters</dfn></dt>
   <dd><p>All letters should default to uppercase.</p></dd>
  </dl>

  <p>The <dfn element-attr for="html-global"><code
  data-x="attr-autocapitalize">autocapitalize</code></dfn> attribute is an <span>enumerated
  attribute</span> whose states are the possible <span data-x="autocapitalization
  hint">autocapitalization hints</span>. The <span>autocapitalization hint</span> specified by the
  attribute's state combines with other considerations to form the <span>used autocapitalization
  hint</span>, which informs the behavior of the user agent. The keywords for this attribute and
  their state mappings are as follows:</p>

  <table>
   <thead>
    <tr>
     <th> Keyword
     <th> State
   <tbody>
    <tr>
     <td><dfn attr-value for="html-global/autocapitalize"><code
     data-x="attr-autocapitalize-off">off</code></dfn>

     <td rowspan="2"><span data-x="autocap-hint-none">none</span>
    <tr>
     <td><dfn attr-value for="html-global/autocapitalize"><code
     data-x="attr-autocapitalize-none">none</code></dfn>
    <tr>
     <td><dfn attr-value for="html-global/autocapitalize"><code
     data-x="attr-autocapitalize-on">on</code></dfn>

     <td rowspan="2"><span data-x="autocap-hint-sentences">sentences</span>
    <tr>
     <td><dfn attr-value for="html-global/autocapitalize"><code
     data-x="attr-autocapitalize-sentences">sentences</code></dfn>
    <tr>
     <td><dfn attr-value for="html-global/autocapitalize"><code
     data-x="attr-autocapitalize-words">words</code></dfn>

     <td><span data-x="autocap-hint-words">words</span>
    <tr>
     <td><dfn attr-value for="html-global/autocapitalize"><code
     data-x="attr-autocapitalize-characters">characters</code></dfn>

     <td><span data-x="autocap-hint-characters">characters</span>
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> is the <span
  data-x="autocap-hint-default">default</span> state, and its <i data-x="invalid value
  default">invalid value default</i> is the <span data-x="autocap-hint-sentences">sentences</span>
  state.

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-autocapitalize">autocapitalize</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the current autocapitalization state for the element, or an empty string if it hasn't
    been set. Note that for <code>input</code> and <code>textarea</code> elements that inherit their
    state from a <code>form</code> element, this will return the autocapitalization state of the
    <code>form</code> element, but for an element in an editable region, this will not return the
    autocapitalization state of the editing host (unless this element is, in fact, the <span>editing
    host</span>).</p>

    <p>Can be set, to set the <code data-x="attr-autocapitalize">autocapitalize</code> content
    attribute (and thereby change the autocapitalization behavior for the element).</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>To compute the <dfn>own autocapitalization hint</dfn> of an element <var>element</var>, run the
  following steps:</p>

  <ol>
   <li><p>If the <code data-x="attr-autocapitalize">autocapitalize</code> content attribute is
   present on <var>element</var>, and its value is not the empty string, return the state of the
   attribute.</p></li>

   <li><p>If <var>element</var> is an <span
   data-x="category-autocapitalize-and-autocorrect">autocapitalize-and-autocorrect inheriting
   element</span> and has a non-null <span>form owner</span>, return the <span>own
   autocapitalization hint</span> of <var>element</var>'s <span>form owner</span>.</p></li>

   <li><p>Return <span data-x="autocap-hint-default">default</span>.</p></li>
  </ol>

  <p>The <dfn attribute for="HTMLElement"><code
  data-x="dom-autocapitalize">autocapitalize</code></dfn> getter steps are to:</p>

  <ol>
   <li><p>Let <var>state</var> be the <span>own autocapitalization hint</span> of
   <span>this</span>.</p></li>

   <li><p>If <var>state</var> is <span data-x="autocap-hint-default">default</span>, then return the
   empty string.</p></li>

   <li><p>If <var>state</var> is <span data-x="autocap-hint-none">none</span>, then return "<code
   data-x="attr-autocapitalize-none">none</code>".</p></li>

   <li><p>If <var>state</var> is <span data-x="autocap-hint-sentences">sentences</span>, then return
   "<code data-x="attr-autocapitalize-sentences">sentences</code>".</p></li>

   <li><p>Return the keyword value corresponding to <var>state</var>.</p></li>
  </ol>

  <p>The <code data-x="dom-autocapitalize">autocapitalize</code> setter steps are to set the <code
  data-x="attr-autocapitalize">autocapitalize</code> content attribute to the given value.</p>

  <hr>

  <p>User agents that support customizable autocapitalization behavior for a text input method and
  wish to allow web developers to control this functionality should, during text input into an
  element, compute the <dfn>used autocapitalization hint</dfn> for the element. This will be an
  <span>autocapitalization hint</span> that describes the recommended autocapitalization behavior
  for text input into the element.</p>

  <p>User agents or input methods may choose to ignore or override the <span>used autocapitalization
  hint</span> in certain circumstances.</p>

  <p>The <span>used autocapitalization hint</span> for an element <var>element</var> is computed
  using the following algorithm:</p>

  <ol>
   <li><p>If <var>element</var> is an <code>input</code> element whose <code
   data-x="attr-input-type">type</code> attribute is in one of the <span
   data-x="attr-input-type-url">URL</span>, <span data-x="attr-input-type-email">Email</span>, or
   <span data-x="attr-input-type-password">Password</span> states, then return <span
   data-x="autocap-hint-default">default</span>.</p></li>

   <li><p>If <var>element</var> is an <code>input</code> element or a <code>textarea</code> element,
   then return <var>element</var>'s <span>own autocapitalization hint</span>.</p></li>

   <li><p>If <var>element</var> is an <span>editing host</span> or an <span>editable</span> element,
   then return the <span>own autocapitalization hint</span> of the <span>editing host of</span>
   <var>element</var>.</p></li>

   <li><p><span>Assert</span>: this step is never reached, since text input only occurs in elements
   that meet one of the above criteria.</p></li>
  </ol>

  </div>


  <h4>Autocorrection</h4>

  <p>Some methods of entering text assist users by automatically correcting misspelled words while
  typing, a process also known as autocorrection. User agents can support autocorrection of editable
  text, either in form controls (such as the value of <code>textarea</code> elements), or in
  elements in an <span>editing host</span> (e.g., using <code
  data-x="attr-contenteditable">contenteditable</code>). Autocorrection may be accompanied by user
  interfaces indicating that text is about to be autocorrected or has been autocorrected, and is
  commonly performed when inserting punctuation characters, spaces, or new paragraphs after
  misspelled words. The <code data-x="attr-autocorrect">autocorrect</code> attribute allows authors
  to control such behavior.</p>

  <p>The <code data-x="attr-autocorrect">autocorrect</code> attribute can be used on an editing host
  to control autocorrection behavior for the hosted editable region, on an <code>input</code> or
  <code>textarea</code> element to control the behavior when inserting text into that element, or on
  a <code>form</code> element to control the default behavior for all <span
  data-x="category-autocapitalize-and-autocorrect">autocapitalize-and-autocorrect inheriting
  elements</span> associated with the <code>form</code> element.</p>

  <p>The <code data-x="attr-autocorrect">autocorrect</code> attribute never causes autocorrection to
  be enabled for <code>input</code> elements whose <code data-x="attr-input-type">type</code>
  attribute is in one of the <span data-x="attr-input-type-url">URL</span>, <span
  data-x="attr-input-type-email">E-mail</span>, or <span
  data-x="attr-input-type-password">Password</span> states.<span w-nodev> (This behavior is included
  in the <span>used autocorrection state</span> algorithm below.)</span></p>

  <p>The <dfn element-attr for="html-global"><code
  data-x="attr-autocorrect">autocorrect</code></dfn> attribute is an enumerated attribute with the
  following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="html-global/autocorrect"><code
     data-x="attr-autocorrect-on">on</code></dfn>
     <td rowspan="2"><dfn data-x="concept-autocorrection-on">on</dfn>
     <td rowspan="2">The user agent is permitted to automatically correct spelling errors while the user types.
     Whether spelling is automatically corrected while typing left is for the user agent to decide,
     and may depend on the element as well as the user's preferences.
    <tr>
     <td>(the empty string)
    <tr>
     <td><dfn attr-value for="html-global/autocorrect"><code data-x="attr-autocorrect-off">off</code></dfn>
     <td><dfn data-x="concept-autocorrection-off">off</dfn>
     <td>The user agent is not allowed to automatically correct spelling while the user types.
  </table>

  <p>The attribute's <i data-x="invalid value default">invalid value default</i> and <i
  data-x="missing value default">missing value default</i> are both the <span
  data-x="concept-autocorrection-on">on</span> state.</p>

  <p>The <dfn attribute for="HTMLElement"><code data-x="dom-autocorrect">autocorrect</code></dfn>
  getter steps are: return true if the element's <span>used autocorrection state</span> is <span
  data-x="concept-autocorrection-on">on</span> and false if the element's <span>used autocorrection
  state</span> is <span data-x="concept-autocorrection-off">off</span>. The setter steps are: if the
  given value is true, then the element's <code data-x="attr-autocorrect">autocorrect</code>
  attribute must be set to "<code data-x="">on</code>"; otherwise it must be set to "<code
  data-x="">off</code>".</p>

  <p>To compute the <dfn>used autocorrection state</dfn> of an element <var>element</var>, run these
  steps:</p>

  <ol>
   <li><p>If <var>element</var> is an <code>input</code> element whose <code
   data-x="attr-input-type">type</code> attribute is in one of the <span
   data-x="attr-input-type-url">URL</span>, <span data-x="attr-input-type-email">E-mail</span>, or
   <span data-x="attr-input-type-password">Password</span> states, then return <span
   data-x="concept-autocorrection-off">off</span>.</p></li>

   <li><p>If the <code data-x="attr-autocorrect">autocorrect</code> content attribute is present on
   <var>element</var>, then return the state of the attribute.</p></li>

   <li><p>If <var>element</var> is an <span
   data-x="category-autocapitalize-and-autocorrect">autocapitalize-and-autocorrect inheriting
   element</span> and has a non-null <span>form owner</span>, then return the state of
   <var>element</var>'s <span>form owner</span>'s <code data-x="attr-autocorrect">autocorrect</code>
   attribute.</p></li>

   <li><p>Return <span data-x="concept-autocorrection-on">on</span>.</p></li>
  </ol>

  <dl class="domintro">
   <dt><var>element</var> . <code subdfn data-x="dom-autocorrect">autocorrect</code></dt>
   <dd><p>Returns the autocorrection behavior of the element. Note that for <span
   data-x="category-autocapitalize-and-autocorrect">autocapitalize-and-autocorrect inheriting
   elements</span> that inherit their state from a <code>form</code> element, this will return the
   autocorrection behavior of the <code>form</code> element, but for an element in an editable
   region, this will not return the autocorrection behavior of the <span>editing host</span> (unless
   this element is, in fact, the <span>editing host</span>).</p></dd>

   <dt><var>element</var> . <code data-x="dom-autocorrect">autocorrect</code> =
   <var>value</var></dt>
   <dd><p>Updates the <code data-x="attr-autocorrect">autocorrect</code> content attribute (and
   thereby changes the autocorrection behavior of the element).</p></dd>
  </dl>

  <div class="example">
   <p>The <code>input</code> element in the following example would not allow autocorrection, since
   it does not have an <code data-x="attr-autocorrect">autocorrect</code> content attribute and
   therefore inherits from the <code>form</code> element, which has an attribute of "<code
   data-x="attr-autocorrect-off">off</code>". However, the <code>textarea</code> element would allow
   autocorrection, since it has an <code data-x="attr-autocorrect">autocorrect</code> content
   attribute with a value of "<code data-x="attr-autocorrect-on">on</code>".</p>

   <pre><code class="html" data-x="">&lt;form autocorrect="off"&gt;
 &lt;input type="search"&gt;
 &lt;textarea autocorrect="on"&gt;&lt;/textarea&gt;
&lt;/form&gt;</code></pre>
  </div>


  <h4>Input modalities: the <code data-x="attr-inputmode">inputmode</code> attribute</h4>

  <p>User agents can support the <code data-x="attr-inputmode">inputmode</code> attribute on form
  controls (such as the value of <code>textarea</code> elements), or in elements in an <span>editing
  host</span> (e.g., using <code data-x="attr-contenteditable">contenteditable</code>).</p>

  <p>The <dfn element-attr for="html-global"><code data-x="attr-inputmode">inputmode</code></dfn>
  content attribute is an <span>enumerated attribute</span> that specifies what kind of input
  mechanism would be most helpful for users entering content.</p>

  <table>
   <thead>
    <tr>
     <th> Keyword
     <th> Description
   <tbody>
    <tr>
     <td><dfn attr-value for="html-global/inputmode"><code
     data-x="attr-inputmode-keyword-none">none</code></dfn>

     <td>The user agent should not display a virtual keyboard. This keyword is useful for content
     that renders its own keyboard control.
    <tr>
     <td><dfn attr-value for="html-global/inputmode"><code
     data-x="attr-inputmode-keyword-text">text</code></dfn>

     <td>The user agent should display a virtual keyboard capable of text input in the user's locale.
    <tr>
     <td><dfn attr-value for="html-global/inputmode"><code
     data-x="attr-inputmode-keyword-tel">tel</code></dfn>

     <td> The user agent should display a virtual keyboard capable of telephone number input. This
     should including keys for the digits 0 to 9, the "#" character, and the "*" character. In some
     locales, this can also include alphabetic mnemonic labels (e.g., in the US, the key labeled
     "2" is historically also labeled with the letters A, B, and C).
    <tr>
     <td><dfn attr-value for="html-global/inputmode"><code
     data-x="attr-inputmode-keyword-url">url</code></dfn>

     <td>The user agent should display a virtual keyboard capable of text input in the user's locale,
     with keys for aiding in the input of <span data-x="URL">URLs</span>, such as that for the "/"
     and "." characters and for quick input of strings commonly found in domain names such as
     "www." or ".com".
    <tr>
     <td><dfn attr-value for="html-global/inputmode"><code
     data-x="attr-inputmode-keyword-email">email</code></dfn>

     <td>The user agent should display a virtual keyboard capable of text input in the user's locale,
     with keys for aiding in the input of email addresses, such as that for the "@" character
     and the "." character.
    <tr>
     <td><dfn attr-value for="html-global/inputmode"><code
     data-x="attr-inputmode-keyword-numeric">numeric</code></dfn>

     <td> The user agent should display a virtual keyboard capable of numeric input. This keyword
     is useful for PIN entry.
    <tr>
     <td><dfn attr-value for="html-global/inputmode"><code
     data-x="attr-inputmode-keyword-decimal">decimal</code></dfn>

     <td> The user agent should display a virtual keyboard capable of fractional numeric input.
     Numeric keys and the format separator for the locale should be shown.
    <tr>
     <td><dfn attr-value for="html-global/inputmode"><code
     data-x="attr-inputmode-keyword-search">search</code></dfn>

     <td>The user agent should display a virtual keyboard optimized for search.
  </table>

  <div w-nodev>

  <p>The <dfn attribute for="ElementContentEditable"><code
  data-x="dom-inputMode">inputMode</code></dfn> IDL attribute must <span>reflect</span> the <code
  data-x="attr-inputmode">inputmode</code> content attribute, <span>limited to only known
  values</span>.</p>

  <p>When <code data-x="attr-inputmode">inputmode</code> is unspecified (or is in a state not
  supported by the user agent), the user agent should determine the default virtual keyboard to be
  shown. Contextual information such as the input <code data-x="attr-input-type">type</code> or
  <code data-x="attr-input-pattern">pattern</code> attributes should be used to determine which type
  of virtual keyboard should be presented to the user.</p>

  </div>

  <h4>Input modalities: the <code data-x="attr-enterkeyhint">enterkeyhint</code>
  attribute</h4>

  <p>User agents can support the <code data-x="attr-enterkeyhint">enterkeyhint</code>
  attribute on form controls (such as the value of <code>textarea</code> elements), or in elements
  in an <span>editing host</span> (e.g., using <code
  data-x="attr-contenteditable">contenteditable</code>).</p>

  <p>The <dfn element-attr for="html-global"><code
  data-x="attr-enterkeyhint">enterkeyhint</code></dfn> content attribute is an <span>enumerated
  attribute</span> that specifies what action label (or icon) to present for the enter key on
  virtual keyboards. This allows authors to customize the presentation of the enter key in order to
  make it more helpful for users.</p>

  <table>
   <thead>
    <tr>
     <th> Keyword
     <th> Description
   <tbody>
    <tr>
     <td><dfn attr-value for="html-global/enterkeyhint"><code
     data-x="attr-enterkeyhint-keyword-enter">enter</code></dfn>

     <td>The user agent should present a cue for the operation 'enter', typically
     inserting a new line.
    <tr>
     <td><dfn attr-value for="html-global/enterkeyhint"><code
     data-x="attr-enterkeyhint-keyword-done">done</code></dfn>

     <td>The user agent should present a cue for the operation 'done', typically
     meaning there is nothing more to input and the input method editor (IME)
     will be closed.
    <tr>
     <td><dfn attr-value for="html-global/enterkeyhint"><code
     data-x="attr-enterkeyhint-keyword-go">go</code></dfn>

     <td> The user agent should present a cue for the operation 'go', typically
     meaning to take the user to the target of the text they typed.
    <tr>
     <td><dfn attr-value for="html-global/enterkeyhint"><code
     data-x="attr-enterkeyhint-keyword-next">next</code></dfn>

     <td>The user agent should present a cue for the operation 'next', typically
     taking the user to the next field that will accept text.
    <tr>
     <td><dfn attr-value for="html-global/enterkeyhint"><code
     data-x="attr-enterkeyhint-keyword-previous">previous</code></dfn>

     <td>The user agent should present a cue for the operation 'previous', typically
     taking the user to the previous field that will accept text.
    <tr>
     <td><dfn attr-value for="html-global/enterkeyhint"><code
     data-x="attr-enterkeyhint-keyword-search">search</code></dfn>

     <td>The user agent should present a cue for the operation 'search', typically
     taking the user to the results of searching for the text they have typed.
    <tr>
     <td><dfn attr-value for="html-global/enterkeyhint"><code
     data-x="attr-enterkeyhint-keyword-send">send</code></dfn>

     <td> The user agent should present a cue for the operation 'send', typically
     delivering the text to its target.
  </table>

  <div w-nodev>

  <p>The <dfn attribute for="ElementContentEditable"><code
  data-x="dom-enterKeyHint">enterKeyHint</code></dfn> IDL attribute must <span>reflect</span> the
  <code data-x="attr-enterkeyhint">enterkeyhint</code> content attribute, <span>limited to only
  known values</span>.</p>

  <p>When <code data-x="attr-enterkeyhint">enterkeyhint</code> is unspecified (or is in a state not
  supported by the user agent), the user agent should determine the default action label (or icon)
  to present. Contextual information such as the <code data-x="attr-inputmode">inputmode</code>,
  <code data-x="attr-input-type">type</code>, or <code data-x="attr-input-pattern">pattern</code>
  attributes should be used to determine which action label (or icon) to present on the virtual
  keyboard.</p>

  </div>

  <h3>Find-in-page</h3>

  <h4>Introduction</h4>

  <p>This section defines <dfn>find-in-page</dfn> — a common user-agent mechanism which allows users
  to search through the contents of the page for particular information.</p>

  <p>Access to the <span>find-in-page</span> feature is provided via a <dfn>find-in-page
  interface</dfn>. This is a user-agent provided user interface, which allows the user to specify
  input and the parameters of the search. This interface can appear as a result of a shortcut or a
  menu selection.</p>

  <p>A combination of text input and settings in the <span>find-in-page interface</span> represents
  the user <dfn data-x="fip-query">query</dfn>. This typically includes the text that the user wants
  to search for, as well as optional settings (e.g., the ability to restrict the search to whole
  words only).</p>

  <p>The user-agent processes page contents for a given <span data-x="fip-query">query</span>, and
  identifies zero or more <dfn data-x="fip-matches">matches</dfn>, which are content ranges that
  satisfy the user <span data-x="fip-query">query</span>.</p>

  <p>One of the <span data-x="fip-matches">matches</span> is identified to the user as the <dfn
  data-x="fip-active-match">active match</dfn>. It is highlighted and scrolled into view. The user
  can navigate through the <span data-x="fip-matches">matches</span> by advancing the <span
  data-x="fip-active-match">active match</span> using the <span>find-in-page interface</span>.</p>

  <p class="XXX"><a href="https://github.com/whatwg/html/issues/3539">Issue #3539</a> tracks
  standardizing how <span>find-in-page</span> underlies the currently-unspecified <code
  data-x="">window.find()</code> API.</p>

  <h4>Interaction with <code>details</code> and <code
  data-x="attr-hidden-until-found-state">hidden=until-found</code></h4>

  <p>When find-in-page begins searching for matches, all <code>details</code> elements in the page
  which do not have their <code data-x="attr-details-open">open</code> attribute set should have the
  <span data-x="skips its contents">skipped contents</span> of their second slot become accessible,
  without modifying the <code data-x="attr-details-open">open</code> attribute, in order to make
  find-in-page able to search through it. Similarly, all HTML elements with the <code
  data-x="attr-hidden">hidden</code> attribute in the <span
  data-x="attr-hidden-until-found-state">hidden until found</span> state should have their <span
  data-x="skips its contents">skipped contents</span> become accessible without modifying the <code
  data-x="attr-hidden">hidden</code> attribute in order to make find-in-page able to search through
  them. After find-in-page finishes searching for matches, the <code>details</code> elements and the
  elements with the <code data-x="attr-hidden">hidden</code> attribute in the <span
  data-x="attr-hidden-until-found-state">hidden until found</span> state should have their contents
  become skipped again. This entire process must happen synchronously (and so is not observable to
  users or to author code). <ref>CSSCONTAIN</ref></p>

  <p>When find-in-page chooses a new <span data-x="fip-active-match">active match</span>, perform
  the following steps:</p>

  <ol>
   <li><p>Let <var>node</var> be the first node in the <span data-x="fip-active-match">active
   match</span>.</p></li>

   <li><p><span>Queue a global task</span> on the <span>user interaction task source</span> given
   <var>node</var>'s <span>relevant global object</span> to run the following steps:</p>

    <ol>
     <li><p>Run the <span>ancestor details revealing algorithm</span> on <var>node</var>.</p></li>

     <li><p>Run the <span>ancestor hidden-until-found revealing algorithm</span> on
     <var>node</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p class="warning">
  <!--INSERT TRACKING-->
  When find-in-page auto-expands a <code>details</code> element like this, it will fire a <code
  data-x="event-toggle">toggle</code> event. As with the separate <code
  data-x="event-scroll">scroll</code> event that find-in-page fires, this event could be used by the
  page to discover what the user is typing into the find-in-page dialog. If the page creates a tiny
  scrollable area with the current search term and every possible next character the user could type
  separated by a gap, and observes which one the browser scrolls to, it can add that character to
  the search term and update the scrollable area to incrementally build the search term. By wrapping
  each possible next match in a closed <code>details</code> element, the page could listen to <code
  data-x="event-toggle">toggle</code> events instead of <code data-x="event-scroll">scroll</code>
  events. This attack could be addressed for both events by not acting on every character the user
  types into the find-in-page dialog.</p>

  <h4>Interaction with selection</h4>

  <p>The find-in-page process is invoked in the context of a document, and may have an effect on
  the <span data-x="document-selection">selection</span> of that document. Specifically, the range
  that defines the <span data-x="fip-active-match">active match</span> can dictate the current
  selection. These selection updates, however, can happen at different times during the
  find-in-page process (e.g. upon the <span>find-in-page interface</span> dismissal or upon a change
  in the <span data-x="fip-active-match">active match</span> range).

  <!-- TODO: Define scroll and highlight timing, as well as algorithms (what is and isn't found) -->


  <h3>Close requests and close watchers</h3>

  <h4>Close requests</h4>

  <p>In an <span>implementation-defined</span> (and likely device-specific) manner, a user can send
  a <dfn export>close request</dfn> to the user agent. This indicates that the user wishes to close
  something that is currently being shown on the screen, such as a popover, menu, dialog, picker, or
  display mode.</p>

  <div class="example">
   <p>Some example close requests are:</p>

   <ul>
    <li><p>The <kbd>Esc</kbd> key on desktop platforms.</p></li>

    <li><p>The back button or gesture on certain mobile platforms such as Android.</p></li>

    <li><p>Any assistive technology's dismiss gesture, such as iOS VoiceOver's two-finger scrub "z"
    gesture.</p></li>

    <li><p>A game controller's canonical "back" button, such as the circle button on a DualShock
    gamepad.</p></li>
   </ul>
  </div>

  <div w-nodev>

  <p>Whenever the user agent receives a potential close request targeted at a <code>Document</code>
  <var>document</var>, it must <span>queue a global task</span> on the <span>user interaction task
  source</span> given <var>document</var>'s <span>relevant global object</span> to perform the
  following <dfn export>close request steps</dfn>:</p>

  <ol>
   <li>
    <p>If <var>document</var>'s <span>fullscreen element</span> is not null, then:</p>

    <ol>
     <li><p><span>Fully exit fullscreen</span> given <var>document</var>'s <span>node
     navigable</span>'s <span data-x="nav-top">top-level traversable</span>'s <span
     data-x="nav-document">active document</span>.</p></li>

     <li><p>Return.</p>
    </ol>

    <p class="note">This does <em>not</em> fire any relevant event, such as <code
    data-x="event-keydown">keydown</code>; it only causes <code
    data-x="event-fullscreenchange">fullscreenchange</code> to be eventually fired.</p>
   </li>

   <li>
    <p>Optionally, skip to the step labeled <a href="#close-request-fallback">alternative
    processing</a>.</p>

    <p class="note">For example, if the user agent detects user frustration at repeated close
    request interception by the current web page, it might take this path.</p>
   </li>

   <li>
    <p>Fire any relevant events, per <cite>UI Events</cite> or other relevant specifications.
    <ref>UIEVENTS</ref></p>

    <p class="example">An example of a relevant event in the <cite>UI Events</cite> model would be
    the <code data-x="event-keydown">keydown</code>  event that <cite>UI Events</cite> <a
    href="https://w3c.github.io/uievents/#events-keyboard-event-order">suggests</a> firing when the
    user presses the <kbd>Esc</kbd> key on their keyboard. On most platforms with keyboards, this
    is treated as a <span>close request</span>, and so would trigger these <span>close request
    steps</span>.</p>

    <p class="example">An example of relevant events that are outside of the model given in
    <cite>UI Events</cite> would be assistive technology synthesizing an <kbd>Esc</kbd> <code
    data-x="event-keydown">keydown</code> event when the user sends a <span>close request</span> by
    using a dismiss gesture.</p>
   </li>

   <li><p>Let <var>event</var> be null if no such events are fired, or the <code>Event</code>
   object representing one of the fired events otherwise. If multiple events are fired, which one
   is chosen is <span>implementation-defined</span>.</p></li>

   <li><p>If <var>event</var> is not null, and its <span>canceled flag</span> is set, then
   return.</p></li>

   <li>
    <p>If <var>document</var> is not <span>fully active</span>, then return.</p>

    <p class="note">This step is necessary because, if <var>event</var> is not null, then an event
    listener might have caused <var>document</var> to no longer be <span>fully active</span>.</p>
   </li>

   <li><p>Let <var>closedSomething</var> be the result of <span data-x="process close
   watchers">processing close watchers</span> on <var>document</var>'s <span>relevant global
   object</span>.</p></li>

   <li><p>If <var>closedSomething</var> is true, then return.</p></li>

   <li id="close-request-fallback"><p><i>Alternative processing</i>: Otherwise, there was nothing
   watching for a <span>close request</span>. The user agent may instead interpret this interaction
   as some other action, instead of interpreting it as a close request.</p></li>
  </ol>

  </div>

  <p>On platforms where pressing the <kbd>Esc</kbd> key is interpreted as a <span>close
  request</span>, the user agent must interpret the key being pressed <em>down</em> as the close
  request, instead of the key being released. Thus, in the above algorithm, the "relevant events"
  that are fired must be the single <code data-x="event-keydown">keydown</code> event.</p>

  <p class="example">On platforms where <kbd>Esc</kbd> is the <span>close request</span>, the user
  agent will first fire an appropriately-initialized <code data-x="event-keydown">keydown</code>
  event. If the web developer cancels the event by calling <code
  data-x="dom-Event-preventDefault">preventDefault()</code>, then nothing further happens. But if
  the event fires without being canceled, then the user agent proceeds to <span>process close
  watchers</span>.</p>

  <p class="example">On platforms where a back button is a potential <span>close request</span>, no
  event is involved, so when the back button is pressed, the user agent proceeds directly to
  <span>process close watchers</span>. If there is an <span
  data-x="close-watcher-active">active</span> <span>close watcher</span>, then that will get
  triggered. If there is not, then the user agent can interpret the back button press in another
  way, for example as a request to <span>traverse the history by a delta</span> of &minus;1.</p>


  <div w-nodev>

  <h4>Close watcher infrastructure</h4>

  <p>Each <code>Window</code> has a <dfn>close watcher manager</dfn>, which is a
  <span>struct</span> with the following <span data-x="struct item">items</span>:</p>

  <ul>
   <li><p><dfn data-x="close watcher manager groups">Groups</dfn>, a <span>list</span> of <span
   data-x="list">lists</span> of <span data-x="close watcher">close watchers</span>, initially
   empty.</p></li>

   <li><p><dfn data-x="close watcher manager allowed number of groups">Allowed number of
   groups</dfn>, a number, initially 1.</p></li>

   <li><p><dfn data-x="close watcher manager next">Next user interaction allows a new group</dfn>,
   a boolean, initially true.</p></li>
  </ul>

  <p>Most of the complexity of the <span>close watcher manager</span> comes from anti-abuse
  protections designed to prevent developers from disabling users' history traversal abilities, for
  platforms where a <span>close request</span>'s <a href="#close-request-fallback">fallback
  action</a> is the main mechanism of history traversal. In particular:</p>

  <p>The grouping of <span data-x="close watcher">close watchers</span> is designed so that if
  multiple close watchers are created without <span>history-action activation</span>, they are
  grouped together, so that a user-triggered <span>close request</span> will close all of the close
  watchers in a group. This ensures that web developers can't intercept an unlimited number of
  close requests by creating close watchers; instead they can create a number equal to at most 1 +
  the number of times the <span data-x="user activation">user activates the page</span>.</p>

  <p>The <span data-x="close watcher manager next">next user interaction allows a new group</span>
  boolean encourages web developers to create <span data-x="close watcher">close watchers</span> in
  a way that is tied to individual <span data-x="user activation">user activations</span>. Without
  it, each user activation would increase the <span data-x="close watcher manager allowed number of
  groups">allowed number of groups</span>, even if the web developer isn't "using" those user
  activations to create close watchers. In short:</p>

  <ul>
   <li><p>Allowed: user interaction; create a close watcher in its own group; user interaction;
   create a close watcher in a second independent group.</p></li>

   <li><p>Disallowed: user interaction; user interaction; create a close watcher in its own group;
   create a close watcher in a second independent group.</p></li>

   <li><p>Allowed: user interaction; user interaction; create a close watcher in its own group;
   create a close watcher grouped with the previous one.</p></li>
  </ul>

  <p>This protection is <em>not</em> important for upholding our desired invariant of creating at
  most (1 + the number of times the <span data-x="user activation">user activates the page</span>)
  groups. A determined abuser will just create one close watcher per user interaction, "banking"
  them for future abuse. But this system causes more predictable behavior for the normal case, and
  encourages non-abusive developers to create close watchers directly in response to user
  interactions.</p>

  <p>To <dfn>notify the close watcher manager about user activation</dfn> given a
  <code>Window</code> <var>window</var>:</p>

  <ol>
   <li><p>Let <var>manager</var> be <var>window</var>'s <span>close watcher manager</span>.</p></li>

   <li><p>If <var>manager</var>'s <span data-x="close watcher manager next">next user interaction
   allows a new group</span> is true, then increment <var>manager</var>'s <span data-x="close
   watcher manager allowed number of groups">allowed number of groups</span>.</p></li>

   <li><p>Set <var>manager</var>'s <span data-x="close watcher manager next">next user interaction
   allows a new group</span> to false.</p></li>
  </ol>

  <hr>

  <p>A <dfn export>close watcher</dfn> is a <span>struct</span> with the following <span
  data-x="struct item">items</span>:</p>

  <ul>
   <li><p>A <dfn data-x="close-watcher-window">window</dfn>, a <code>Window</code>.</p></li>

   <li><p>A <dfn data-x="close-watcher-cancel-action">cancel action</dfn>, an algorithm accepting a
   boolean argument and returning a boolean. The argument indicates whether or not the cancel
   action algorithm can prevent the close request from proceeding via the algorithm's return value.
   If the boolean argument is true, then the algorithm can return either true to indicate that the
   caller will proceed to the <span data-x="close-watcher-close-action">close action</span>, or
   false to indicate that the caller will bail out. If the argument is false, then the return value
   is always false. This algorithm can never throw an exception.</p></li>

   <li><p>A <dfn data-x="close-watcher-close-action">close action</dfn>, an algorithm accepting no
   arguments and returning nothing. This algorithm can never throw an exception.</p></li>

   <li><p>An <dfn data-x="close-watcher-is-running-cancel">is running cancel action</dfn>
   boolean.</p></li>

   <li><p>A <dfn data-x="close-watcher-get-enabled-state">get enabled state</dfn>, an algorithm
   accepting no arguments and returning a boolean. This algorithm can never throw an
   exception.</p></li>
  </ul>

  <p>A <span>close watcher</span> <var>closeWatcher</var> <dfn data-x="close-watcher-active">is
  active</dfn> if <var>closeWatcher</var>'s <span data-x="close-watcher-window">window</span>'s
  <span>close watcher manager</span> <span data-x="list contains">contains</span> any list which
  <span data-x="list contains">contains</span> <var>closeWatcher</var>.</p>

  <hr>

  <p>To <dfn>establish a close watcher</dfn> given a <code>Window</code> <var>window</var>, a list
  of steps <dfn data-x="create-close-watcher-cancelAction"><var>cancelAction</var></dfn>, a
  list of steps <dfn data-x="create-close-watcher-closeAction"><var>closeAction</var></dfn>,
  and an algorithm that returns a boolean <dfn
  data-x="create-close-watcher-getEnabledState"><var>getEnabledState</var></dfn>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>window</var>'s <span
   data-x="concept-document-window">associated <code>Document</code></span> is <span>fully
   active</span>.</p></li>

   <li>
    <p>Let <var>closeWatcher</var> be a new <span>close watcher</span>, with</p>

    <dl class="props">
     <dt><span data-x="close-watcher-window">window</span></dt>
     <dd><var>window</var></dd>

     <dt><span data-x="close-watcher-cancel-action">cancel action</span></dt>
     <dd><var>cancelAction</var></dd>

     <dt><span data-x="close-watcher-close-action">close action</span></dt>
     <dd><var>closeAction</var></dd>

     <dt><span data-x="close-watcher-is-running-cancel">is running cancel action</span></dt>
     <dd>false</dd>

     <dt><span data-x="close-watcher-get-enabled-state">get enabled state</span></dt>
     <dd><var>getEnabledState</var></dd>
    </dl>
   </li>

   <li><p>Let <var>manager</var> be <var>window</var>'s <span>close watcher manager</span>.</p></li>

   <li><p>If <var>manager</var>'s <span data-x="close watcher manager groups">groups</span>'s <span
   data-x="list size">size</span> is less than <var>manager</var>'s <span data-x="close watcher
   manager allowed number of groups">allowed number of groups</span>, then <span data-x="list
   append">append</span> « <var>closeWatcher</var> » to <var>manager</var>'s <span data-x="close
   watcher manager groups">groups</span>.</p></li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p><span>Assert</span>: <var>manager</var>'s <span data-x="close watcher manager
     groups">groups</span>'s <span data-x="list size">size</span> is at least 1 in this branch,
     since <var>manager</var>'s <span data-x="close watcher manager allowed number of
     groups">allowed number of groups</span> is always at least 1.</p></li>

     <li><p><span data-x="list append">Append</span> <var>closeWatcher</var> to
     <var>manager</var>'s <span data-x="close watcher manager groups">groups</span>'s last <span
     data-x="list item">item</span>.</p></li>
    </ol>
   </li>

   <li><p>Set <var>manager</var>'s <span data-x="close watcher manager next">next user interaction
   allows a new group</span> to true.</p></li>

   <li><p>Return <var>closeWatcher</var>.</p></li>
  </ol>

  <p>To <dfn data-x="close-watcher-request-close">request to close</dfn> a <span>close
  watcher</span> <var>closeWatcher</var> with boolean <var>requireHistoryActionActivation</var>:</p>

  <ol>
   <li><p>If <var>closeWatcher</var> <span data-x="close-watcher-active">is not active</span>, then
   return true.</p></li>

   <li><p>If the result of running <var>closeWatcher</var>'s <span
   data-x="close-watcher-get-enabled-state">get enabled state</span> is false, then return
   true.</p></li>

   <li><p>If <var>closeWatcher</var>'s <span data-x="close-watcher-is-running-cancel">is running
   cancel action</span> is true, then return true.</p></li>

   <li><p>Let <var>window</var> be <var>closeWatcher</var>'s <span
   data-x="close-watcher-window">window</span>.</p></li>

   <li><p>If <var>window</var>'s <span data-x="concept-document-window">associated
   <code>Document</code></span> is not <span>fully active</span>, then return true.</p></li>

<!-- Help me here - I never know how to add logical parenthesis, but in spec English. This should be:
     canPreventClose = ((A || B) && C) -->

   <li><p>Let <var>canPreventClose</var> be true if <var>requireHistoryActionActivation</var> is
   false, or if <var>window</var>'s <span>close watcher manager</span>'s <span
   data-x="close watcher manager groups">groups</span>'s <span data-x="list size">size</span> is
   less than <var>window</var>'s <span>close watcher manager</span>'s <span
   data-x="close watcher manager allowed number of groups">allowed number of groups</span>, and
   <var>window</var> has <span>history-action activation</span>; otherwise false.</p></li>

   <li><p>Set <var>closeWatcher</var>'s <span data-x="close-watcher-is-running-cancel">is running
   cancel action</span> to true.</p></li>

   <li><p>Let <var>shouldContinue</var> be the result of running <var>closeWatcher</var>'s <span
   data-x="close-watcher-cancel-action">cancel action</span> given
   <var>canPreventClose</var>.</p></li>

   <li><p>Set <var>closeWatcher</var>'s <span data-x="close-watcher-is-running-cancel">is running
   cancel action</span> to false.</p></li>

   <li>
    <p>If <var>shouldContinue</var> is false, then:</p>

    <ol>
     <li><p><span>Assert</span>: <var>canPreventClose</var> is true.</p></li>

     <li><p><span>Consume history-action user activation</span> given <var>window</var>.</p></li>

     <li><p>Return false.</p></li>
    </ol>

    <p class="note">Note that since these substeps <span>consume history-action user
    activation</span>, <span data-x="close-watcher-request-close">requesting to close</span> a
    <span>close watcher</span> twice without any intervening <span>user activation</span> will
    result in <var>canPreventClose</var> being false the second time.</p>
   </li>

   <li><p><span data-x="close-watcher-close">Close</span> <var>closeWatcher</var>.</p></li>

   <li><p>Return true.</p></li>
  </ol>

  <p>To <dfn data-x="close-watcher-close">close</dfn> a <span>close watcher</span>
  <var>closeWatcher</var>:</p>

  <ol>
   <li><p>If <var>closeWatcher</var> <span data-x="close-watcher-active">is not active</span>, then
   return.</p></li>

   <li><p>If the result of running <var>closeWatcher</var>'s <span
   data-x="close-watcher-get-enabled-state">get enabled state</span> is false, then return.</p></li>

   <li><p>If <var>closeWatcher</var>'s <span data-x="close-watcher-window">window</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span> is not <span>fully
   active</span>, then return.</p></li>

   <li><p><span data-x="close-watcher-destroy">Destroy</span> <var>closeWatcher</var>.</p></li>

   <li><p>Run <var>closeWatcher</var>'s <span data-x="close-watcher-close-action">close
   action</span>.</p></li>
  </ol>

  <p>To <dfn data-x="close-watcher-destroy">destroy</dfn> a <span>close watcher</span>
  <var>closeWatcher</var>:</p>

  <ol>
   <li><p>Let <var>manager</var> be <var>closeWatcher</var>'s <span
   data-x="close-watcher-window">window</span>'s <span>close watcher manager</span>.</p></li>

   <li><p><span data-x="list iterate">For each</span> <var>group</var> of <var>manager</var>'s
   <span data-x="close watcher manager groups">groups</span>: <span data-x="list
   remove">remove</span> <var>closeWatcher</var> from <var>group</var>.</p></li>

   <li><p><span data-x="list remove">Remove</span> any item from <var>manager</var>'s <span
   data-x="close watcher manager groups">groups</span> that <span data-x="list is empty">is
   empty</span>.</p></li>
  </ol>

  <hr>

  <p>To <dfn>process close watchers</dfn> given a <code>Window</code> <var>window</var>:</p>

  <ol>
   <li><p>Let <var>processedACloseWatcher</var> be false.</p></li>

   <li>
    <p>If <var>window</var>'s <span>close watcher manager</span>'s <span data-x="close watcher
    manager groups">groups</span> is not empty:</p>

    <ol>
     <li><p>Let <var>group</var> be the last <span data-x="list item">item</span> in
     <var>window</var>'s <span>close watcher manager</span>'s <span data-x="close watcher manager
     groups">groups</span>.</p></li>

     <li>
      <p><span data-x="list iterate">For each</span> <var>closeWatcher</var> of <var>group</var>,
      in reverse order:</p>

      <ol>
       <li><p>If the result of running <var>closeWatcher</var>'s <span
       data-x="close-watcher-get-enabled-state">get enabled state</span> is true, set
       <var>processedACloseWatcher</var> to true.</p></li>

       <li><p>Let <var>shouldProceed</var> be the result of <span
       data-x="close-watcher-request-close">requesting to close</span>
       <var>closeWatcher</var> with true.</p></li>

       <li><p>If <var>shouldProceed</var> is false, then <span>break</span>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>If <var>window</var>'s <span>close watcher manager</span>'s <span data-x="close watcher
   manager allowed number of groups">allowed number of groups</span> is greater than 1, decrement
   it by 1.</p></li>

   <li><p>Return <var>processedACloseWatcher</var>.</p></li>
  </ol>

  </div>


  <h4>The <code>CloseWatcher</code> interface</h4>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>CloseWatcher</dfn> : <span>EventTarget</span> {
  <span data-x="dom-CloseWatcher">constructor</span>(optional <span>CloseWatcherOptions</span> options = {});

  undefined <span data-x="dom-CloseWatcher-requestClose">requestClose</span>();
  undefined <span data-x="dom-CloseWatcher-close">close</span>();
  undefined <span data-x="dom-CloseWatcher-destroy">destroy</span>();

  attribute <span>EventHandler</span> <span data-x="handler-CloseWatcher-oncancel">oncancel</span>;
  attribute <span>EventHandler</span> <span data-x="handler-CloseWatcher-onclose">onclose</span>;
};

dictionary <dfn dictionary>CloseWatcherOptions</dfn> {
  <span>AbortSignal</span> <dfn dict-member for="CloseWatcherOptions" data-x="dom-CloseWatcherOptions-signal">signal</dfn>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>watcher</var> = new <span subdfn data-x="dom-CloseWatcher">CloseWatcher</span>()</code></dt>
   <dt><code data-x=""><var>watcher</var> = new <span data-x="dom-CloseWatcher">CloseWatcher</span>({ <span subdfn data-x="dom-CloseWatcherOptions-signal">signal</span> })</code></dt>
   <dd>
    <p>Creates a new <code>CloseWatcher</code> instance.</p>

    <p>If the <code data-x="dom-CloseWatcherOptions-signal">signal</code> option is provided, then
    <var>watcher</var> can be destroyed (as if by <code
    data-x="dom-CloseWatcher-destroy">watcher.destroy()</code>) by aborting the given
    <code>AbortSignal</code>.</p>

    <p>If any <span>close watcher</span> is already active, and the <code>Window</code> does not
    have <span>history-action activation</span>, then the resulting <code>CloseWatcher</code> will
    be closed together with that already-active <span>close watcher</span> in response to any
    <span>close request</span>. (This already-active <span>close watcher</span> does not
    necessarily have to be a <code>CloseWatcher</code> object; it could be a modal
    <code>dialog</code> element, or a popover generated by an element with the <code
    data-x="attr-popover">popover</code> attribute.)</p>
   </dd>

   <dt><code data-x=""><var>watcher</var>.<span subdfn data-x="dom-CloseWatcher-requestClose">requestClose</span>()</code></dt>
   <dd>
    <p>Acts as if a <span>close request</span> was sent targeting <var>watcher</var>, by first
    firing a <code data-x="event-cancel">cancel</code> event, and if that event is not canceled
    with <code data-x="dom-Event-preventDefault">preventDefault()</code>, proceeding to fire a
    <code data-x="event-close">close</code> event before deactivating the close watcher as if <code
    data-x="dom-CloseWatcher-destroy">watcher.destroy()</code> was called.</p>

    <p>This is a helper utility that can be used to consolidate cancelation and closing logic into
    the <code data-x="event-cancel">cancel</code> and <code data-x="event-close">close</code> event
    handlers, by having all non-<span>close request</span> closing affordances call this method.</p>
   </dd>

   <dt><code data-x=""><var>watcher</var>.<span subdfn data-x="dom-CloseWatcher-close">close</span>()</code></dt>
   <dd>
    <p>Immediately fires the <code data-x="event-close">close</code> event, and then deactivates
    the close watcher as if <code data-x="dom-CloseWatcher-destroy">watcher.destroy()</code> was
    called.</p>

    <p>This is a helper utility that can be used trigger the closing logic into the <code
    data-x="event-close">close</code> event handler, skipping any logic in the <code
    data-x="event-cancel">cancel</code> event handler.</p>
   </dd>

   <dt><code data-x=""><var>watcher</var>.<span subdfn data-x="dom-CloseWatcher-destroy">destroy</span>()</code></dt>
   <dd>
    <p>Deactivates <var>watcher</var>, so that it will no longer receive <code
    data-x="event-close">close</code> events and so that new independent <code>CloseWatcher</code>
    instances can be constructed.</p>

    <p>This is intended to be called if the relevant UI element is torn down in some other way than
    being closed.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>Each <code>CloseWatcher</code> instance has an <dfn>internal close watcher</dfn>, which is a
  <span>close watcher</span>.</p>

  <p>The <dfn><code data-x="dom-CloseWatcher">new CloseWatcher(<var>options</var>)</code></dfn>
  constructor steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span> is not <span>fully
   active</span>, then throw an <span>"<code>InvalidStateError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li>
    <p>Let <var>closeWatcher</var> be the result of <span data-x="establish a close
    watcher">establishing a close watcher</span> given <span>this</span>'s <span>relevant global
    object</span>, with:</p>

    <ul>
     <li><p><i data-x="create-close-watcher-cancelAction">cancelAction</i> given
     <var>canPreventClose</var> being to return the result of <span
     data-x="concept-event-fire">firing an event</span> named <code
     data-x="event-cancel">cancel</code> at <span>this</span>, with the <code
     data-x="dom-Event-cancelable">cancelable</code> attribute initialized to
     <var>canPreventClose</var>.</p></li>

     <li><p><i data-x="create-close-watcher-closeAction">closeAction</i> being to <span
     data-x="concept-event-fire">fire an event</span> named <code data-x="event-close">close</code>
     at <span>this</span>.</p></li>

     <li><p><i data-x="close-watcher-get-enabled-state">getEnabledState</i> being to return
     true.</li>
    </ul>
   </li>

   <li>
    <p>If <var>options</var>["<code data-x="dom-CloseWatcherOptions-signal">signal</code>"] <span
    data-x="map exists">exists</span>, then:</p>

    <ol>
     <li><p>If <var>options</var>["<code data-x="dom-CloseWatcherOptions-signal">signal</code>"] is
     <span data-x="AbortSignal-aborted">aborted</span>, then <span
     data-x="close-watcher-destroy">destroy</span> <var>closeWatcher</var>.</p></li>

     <li>
      <p><span data-x="AbortSignal-add">Add</span> the following steps to <var>options</var>["<code
      data-x="dom-CloseWatcherOptions-signal">signal</code>"]:</p>

      <ol>
       <li><p><span data-x="close-watcher-destroy">Destroy</span> <var>closeWatcher</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Set <span>this</span>'s <span>internal close watcher</span> to
   <var>closeWatcher</var>.</p></li>
  </ol>

  <p>The <dfn method for="CloseWatcher"><code
  data-x="dom-CloseWatcher-requestClose">requestClose()</code></dfn> method steps are to <span
  data-x="close-watcher-request-close">request to close</span> <span>this</span>'s <span>internal
  close watcher</span> with false.</p>

  <p>The <dfn method for="CloseWatcher"><code data-x="dom-CloseWatcher-close">close()</code></dfn>
  method steps are to <span data-x="close-watcher-close">close</span> <span>this</span>'s
  <span>internal close watcher</span>.</p>

  <p>The <dfn method for="CloseWatcher"><code
  data-x="dom-CloseWatcher-destroy">destroy()</code></dfn> method steps are to <span
  data-x="close-watcher-destroy">destroy</span> <span>this</span>'s <span>internal close
  watcher</span>.</p>

  </div>

  <p>The following are the <span>event handlers</span> (and their corresponding <span data-x="event
  handler event type">event handler event types</span>) <span w-nodev>that must be</span> supported,
  as <span>event handler IDL attributes</span>, by all objects implementing the
  <code>CloseWatcher</code> interface:</p>

  <table>
   <thead>
    <tr><th><span data-x="event handlers">Event handler</span> <th><span>Event handler event type</span>
   <tbody>
    <tr><td><dfn attribute for="CloseWatcher"><code data-x="handler-CloseWatcher-oncancel">oncancel</code></dfn> <td> <code data-x="event-cancel">cancel</code>
    <tr><td><dfn attribute for="CloseWatcher"><code data-x="handler-CloseWatcher-onclose">onclose</code></dfn> <td> <code data-x="event-close">close</code>
  </table>

  <div class="example" id="example-CloseWatcher-basic">
   <p>If one wanted to implement a custom picker control, which closed itself on a user-provided
   <span>close request</span> as well as when a close button is pressed, the following code shows how
   one would use the <code>CloseWatcher</code> API to process close requests:</p>

   <pre><code class="js">const watcher = new CloseWatcher();
const picker = setUpAndShowPickerDOMElement();

let chosenValue = null;

watcher.onclose = () => {
  chosenValue = picker.querySelector('input').value;
  picker.remove();
};

picker.querySelector('.close-button').onclick = () => watcher.requestClose();</code></pre>

   <p>Note how the logic to gather the chosen value is centralized in the <code>CloseWatcher</code>
   object's <code data-x="event-close">close</code> event handler, with the <code
   data-x="event-click">click</code> event handler for the close button delegating to that logic by
   calling <code data-x="dom-CloseWatcher-requestClose">requestClose()</code>.</p>
  </div>

  <div class="example" id="example-CloseWatcher-cancel">
   <p>The <code data-x="event-cancel">cancel</code> event on <code>CloseWatcher</code> objects can
   be used to prevent the <code data-x="event-close">close</code> event from firing, and the
   <code>CloseWatcher</code> from being destroying. A typical use case is as follows:</p>

   <pre><code class="js">watcher.oncancel = async (e) => {
  if (hasUnsavedData && e.cancelable) {
    e.preventDefault();

    const userReallyWantsToClose = await askForConfirmation("Are you sure you want to close?");
    if (userReallyWantsToClose) {
      hasUnsavedData = false;
      watcher.close();
    }
  }
};</code></pre>

   <p>For abuse prevention purposes, this event is only <code
   data-x="dom-Event-cancelable">cancelable</code> if the page has <span>history-action
   activation</span>, which will be lost after any given <span>close request</span>. This ensures
   that if the user sends a close request twice in a row without any intervening user activation,
   the request definitely succeeds; the second request ignores any <code
   data-x="event-cancel">cancel</code> event handler's attempt to call <code
   data-x="dom-Event-preventDefault">preventDefault()</code> and proceeds to close the
   <code>CloseWatcher</code>.</p>
  </div>

  <p>Combined, the above two examples show how <code
  data-x="dom-CloseWatcher-requestClose">requestClose()</code> and <code
  data-x="dom-CloseWatcher-close">close()</code> differ. Because we used <code
  data-x="dom-CloseWatcher-requestClose">requestClose()</code> in the <code
  data-x="event-click">click</code> event handler for the close button, clicking that button will
  trigger the <code>CloseWatcher</code>'s <code data-x="event-cancel">cancel</code> event, and thus
  potentially ask the user for confirmation if there is unsaved data. If we had used <code
  data-x="dom-CloseWatcher-close">close()</code>, then this check would be skipped. Sometimes that
  is appropriate, but usually <code data-x="dom-CloseWatcher-requestClose">requestClose()</code> is
  the better option for user-triggered close requests.</p>

  <div class="example" id="example-CloseWatcher-grouping">
   <p>In addition to the <span>user activation</span> restrictions for <code
   data-x="event-cancel">cancel</code> events, there is a more subtle form of user activation gating
   for <code>CloseWatcher</code> construction. If one creates more than one
   <code>CloseWatcher</code> without user activation, then the newly-created one will get grouped
   together with the most-recently-created <span>close watcher</span>, so that a single <span>close
   request</span> will close them both:</p>

   <pre><code class="js">window.onload = () => {
  // This will work as normal: it is the first close watcher created without user activation.
  (new CloseWatcher()).onclose = () => { /* ... */ };
};

button1.onclick = () => {
  // This will work as normal: the button click counts as user activation.
  (new CloseWatcher()).onclose = () => { /* ... */ };
};

button2.onclick = () => {
  // These will be grouped together, and both will close in response to a single close request.
  (new CloseWatcher()).onclose = () => { /* ... */ };
  (new CloseWatcher()).onclose = () => { /* ... */ };
};</code></pre>

   <p>This means that calling <code data-x="dom-CloseWatcher-destroy">destroy()</code>, <code
   data-x="dom-CloseWatcher-close">close()</code>, or <code
   data-x="dom-CloseWatcher-requestClose">requestClose()</code> properly is important. Doing so is
   the only way to get back the "free" ungrouped close watcher slot. Such close watchers created
   without user activation are useful for cases like session inactivity timeout dialogs or urgent
   notifications of server-triggered events, which are not generated in response to user
   activation.</p>
  </div>


  <h3 split-filename="dnd" id="dnd"><dfn>Drag and drop</dfn></h3>

<!-- v2: ideas for drag and drop:

     * being able to animate a drop target:

       > To implement this with simple interface I've proposed, events should be handled either by
       > existing elements (like list items that compare their size and position of dragged element
       > to decide whether element should be dropped before or after) or handled by container that
       > would probably need to calculate positions of it's children and create new element to show
       > drop target. Smooth Mac-like drag'n'drop can be implemented by animating drop target's
       > padding/margin. So that's quite a bit of code that's going to be reinvented each time
       > someone implements reordering.

       <hyatt> :droptarget
       <hyatt> or something
       <hyatt> we don't support a pseudo-class for the drop target but that's a great idea
       <Hixie_> yeah, thinking about that too
       <Hixie_> :drop-target, :drop-target(above), :drop-target(below) and having ondragover be able to say "not on me, but next to me maybe"

       - some way to be able to match an element that is being dragged over.

       - some way to be able to animate an element as it goes into and out of this state (CSS
         transitions?), e.g. to be able to animate something "getting out of the way" to let you
         drop an item between others.

       - as an extension to the previous feature, a way to distinguish being dragged above or to the
         left of the drag target vs below or to the right of the drag target.

     * We should let drop targets communicate back to drag sources if they want to communicate.
       (e.g. expose Window, and thus postMessage(), on the dataTransfer object on drop.) Or maybe
       just use a MessagePort!

       We should let drag sources provide a set of options via a context menu when the drop happens.
       (So that, e.g., the source can know whether a capabilities URI that it is passing along is
       supposed to be read-write access or read-only access to the object being dragged.)

       We should let potential drop targets see the types (but not the contents!) of dragged data so
       they can establish if they care or not. (dataTransfer.hasType())

       Ack: Ben Laurie (@g)

     * Interop with native apps. In particular, we probably want to safelist the list of types that
       a web page can see, since otherwise we'll end up exposing things like the username (if a user
       drags a file from their desktop, the path is exposed on some OSes).

     * Alex Velkov suggests that drags to and from the same origin should be able to read the data
       even before the drop.

     Other things listed below:
      DND-v2: more native support: text/html from selections, etc
      DND-v3: add Blob support
      DND-v4: add structured clone support
      DND-v5: add promises (should be able to say "if you accept this
              drop, then I can provide the File object that
              corresponds to it eventually")

         DataTransferPromise.type = 'string' or 'file' or 'blob' or 'data'
                            .onneeddata - can wait until this fires to provide data
                            .setData() - call this once you have data, must be the right type

-->

  <p>This section defines an event-based drag-and-drop mechanism.</p>

  <p>This specification does not define exactly what a <em>drag-and-drop operation</em> actually
  is.</p>

  <p>On a visual medium with a pointing device, a drag operation could be the default action of a
  <code data-x="event-mousedown">mousedown</code> event that is followed by a series of <code
  data-x="event-mousemove">mousemove</code> events, and the drop could be triggered by the mouse
  being released.</p>

  <p>When using an input modality other than a pointing device, users would probably have to
  explicitly indicate their intention to perform a drag-and-drop operation, stating what they wish
  to drag and where they wish to drop it, respectively.</p>

  <div w-nodev>

  <p>However it is implemented, drag-and-drop operations must have a starting point (e.g. where the
  mouse was clicked, or the start of the selection or element that was selected for the drag), may
  have any number of intermediate steps (elements that the mouse moves over during a drag, or
  elements that the user picks as possible drop points as they cycle through possibilities), and must
  either have an end point (the element above which the mouse button was released, or the element
  that was finally selected), or be canceled. The end point must be the last element selected as a
  possible drop point before the drop occurs (so if the operation is not canceled, there must be at
  least one element in the middle step).</p>

  </div>


  <h4 id="event-drag">Introduction</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>To make an element draggable, give the element a <code data-x="attr-draggable">draggable</code>
  attribute, and set an event listener for <code data-x="event-dnd-dragstart">dragstart</code> that
  stores the data being dragged.</p>

  <p>The event handler typically needs to check that it's not a text selection that is being
  dragged, and then needs to store data into the <code>DataTransfer</code> object and set the
  allowed effects (copy, move, link, or some combination).</p>

  <p>For example:</p>

  <pre><code class="html">&lt;p>What fruits do you like?&lt;/p>
&lt;ol ondragstart="dragStartHandler(event)">
 &lt;li draggable="true" data-value="fruit-apple">Apples&lt;/li>
 &lt;li draggable="true" data-value="fruit-orange">Oranges&lt;/li>
 &lt;li draggable="true" data-value="fruit-pear">Pears&lt;/li>
&lt;/ol>
&lt;script>
  var internalDNDType = 'text/x-example'; // set this to something specific to your site
  function dragStartHandler(event) {
    if (event.target instanceof HTMLLIElement) {
      // use the element's data-value="" attribute as the value to be moving:
      event.dataTransfer.setData(internalDNDType, event.target.dataset.value);
      event.dataTransfer.effectAllowed = 'move'; // only allow moves
    } else {
      event.preventDefault(); // don't allow selection to be dragged
    }
  }
&lt;/script></code></pre>

  <hr>

  <p>To accept a drop, the drop target has to listen to the following events:</p>

  <ol>
    <li>The <code data-x="event-dnd-dragenter">dragenter</code> event handler reports
    whether or not the drop target is potentially willing to accept the drop, by canceling the
    event.</li>

    <li>The <code data-x="event-dnd-dragover">dragover</code> event handler specifies what feedback
    will be shown to the user, by setting the <code
    data-x="dom-DataTransfer-DropEffect">dropEffect</code> attribute of the
    <code>DataTransfer</code> associated with the event. This event also needs to be canceled.</li>

    <li>The <code data-x="event-dnd-drop">drop</code> event handler has a final chance to accept or
    reject the drop. If the drop is accepted, the event handler must perform the drop operation on
    the target. This event needs to be canceled, so that the <code
    data-x="dom-DataTransfer-DropEffect">dropEffect</code> attribute's value can be used by the
    source. Otherwise, the drop operation is rejected.</li>
  </ol>

  <p>For example:</p>

  <pre><code class="html">&lt;p>Drop your favorite fruits below:&lt;/p>
&lt;ol ondragenter="dragEnterHandler(event)" ondragover="dragOverHandler(event)"
    ondrop="dropHandler(event)">
&lt;/ol>
&lt;script>
  var internalDNDType = 'text/x-example'; // set this to something specific to your site
  function dragEnterHandler(event) {
    var items = event.dataTransfer.items;
    for (var i = 0; i &lt; items.length; ++i) {
      var item = items[i];
      if (item.kind == 'string' && item.type == internalDNDType) {
        event.preventDefault();
        return;
      }
    }
  }
  function dragOverHandler(event) {
    event.dataTransfer.dropEffect = 'move';
    event.preventDefault();
  }
  function dropHandler(event) {
    var li = document.createElement('li');
    var data = event.dataTransfer.getData(internalDNDType);
    if (data == 'fruit-apple') {
      li.textContent = 'Apples';
    } else if (data == 'fruit-orange') {
      li.textContent = 'Oranges';
    } else if (data == 'fruit-pear') {
      li.textContent = 'Pears';
    } else {
      li.textContent = 'Unknown Fruit';
    }
    event.target.appendChild(li);
  }
&lt;/script></code></pre>

  <hr>

  <p>To remove the original element (the one that was dragged) from the display, the <code
  data-x="event-dnd-dragend">dragend</code> event can be used.</p>

  <p>For our example here, that means updating the original markup to handle that event:</p>

  <pre><code class="html">&lt;p>What fruits do you like?&lt;/p>
&lt;ol ondragstart="dragStartHandler(event)" ondragend="dragEndHandler(event)">
 <em>...as before...</em>
&lt;/ol>
&lt;script>
  function dragStartHandler(event) {
    // <em>...as before...</em>
  }
  function dragEndHandler(event) {
    if (event.dataTransfer.dropEffect == 'move') {
      // remove the dragged element
      event.target.parentNode.removeChild(event.target);
    }
  }
&lt;/script></code></pre>



  <h4>The drag data store</h4>

  <p>The data that underlies a drag-and-drop operation, known as the <dfn>drag data store</dfn>,
  consists of the following information:</p>

  <ul>
   <li><p>A <dfn>drag data store item list</dfn>, which is a list of items representing the dragged
   data, each consisting of the following information:</p>

    <dl>
     <dt><dfn>The drag data item kind</dfn></dt>

     <dd>
      <p>The kind of data:</p>

      <dl>
       <dt><i>Text</i></dt>
       <dd>
        <p>Text.</p>
       </dd>

<!-- DND-v3:
       <dt><i>Blob</i></dt>
       <dd>
        <p>Binary data.</p>
       </dd>
-->

       <dt><i>File</i></dt>
       <dd>
        <p>Binary data with a filename.</p>
       </dd>

<!-- DND-v4:
       <dt><i>Structured object</i></dt>
       <dd>
        <p>An object that will be cloned using the <span>structured clone</span> algorithm.</p>
       </dd>
-->
      </dl>
     </dd>

     <dt><dfn>The drag data item type string</dfn></dt>

     <dd>
      <p>A Unicode string giving the type or format of the data, generally given by a <span>MIME
      type</span>. Some values that are not <span data-x="MIME type">MIME types</span> are
      special-cased for legacy reasons. The API does not enforce the use of <span data-x="MIME
      type">MIME types</span>; other values can be used as well. In all cases, however, the values
      are all <span>converted to ASCII lowercase</span> by the API.</p>

      <p>There is a limit of one <i>text</i> item per <span data-x="The drag data
      item type string">item type string</span>.</p> <!-- DND-v4: consider limiting the structured
      objects too -->
     </dd>

     <dt>The actual data</dt>

     <dd><p>A Unicode or binary string, in some cases with a filename (itself a Unicode string),
     <!-- (DND-v4:) or an object, --> as per <span>the drag data item kind</span>.</p></dd>
    </dl>

    <p>The <span>drag data store item list</span> is ordered in the order that the items were added
    to the list; most recently added last.</p>
   </li>

   <li>
    <p>The following information, used to generate the UI feedback during the drag:</p>

    <ul>
     <li>User-agent-defined default feedback information, known as the <dfn>drag data store default
     feedback</dfn>.</li>

     <li>Optionally, a bitmap image and the coordinate of a point within that image, known as the
     <dfn>drag data store bitmap</dfn> and <dfn>drag data store hot spot coordinate</dfn>.</li>
    </ul>
   </li>

   <li>
    <p>A <dfn>drag data store mode</dfn>, which is one of the following:</p>

    <dl>
     <dt><dfn data-x="concept-dnd-rw">Read/write mode</dfn></dt>
     <dd>
      <p>For the <code data-x="event-dnd-dragstart">dragstart</code> event. New data can be added to the
      <span>drag data store</span>.</p>
     </dd>

     <dt><dfn data-x="concept-dnd-ro">Read-only mode</dfn></dt>
     <dd>
      <p>For the <code data-x="event-dnd-drop">drop</code> event. The list of items representing dragged
      data can be read, including the data. No new data can be added.</p>
     </dd>

     <dt><dfn data-x="concept-dnd-p">Protected mode</dfn></dt>
     <dd>
      <p>For all other events. The formats and kinds in the <span>drag data store</span> list of
      items representing dragged data can be enumerated, but the data itself is unavailable and no
      new data can be added.</p>
     </dd>
    </dl>
   </li>

   <li>
    <p>A <dfn>drag data store allowed effects state</dfn>, which is a string.</p>
   </li>
  </ul>

  <p>When a <span>drag data store</span> is <dfn data-x="create a drag data store">created</dfn>, it
  must be initialized such that its <span>drag data store item list</span> is empty, it has no
  <span>drag data store default feedback</span>, it has no <span>drag data store bitmap</span> and
  <span>drag data store hot spot coordinate</span>, its <span>drag data store mode</span> is <span
  data-x="concept-dnd-p">protected mode</span>, and its <span>drag data store allowed effects
  state</span> is the string "<code
  data-x="dom-DataTransfer-effectAllowed-uninitialized">uninitialized</code>".</p>


  <h4>The <code>DataTransfer</code> interface</h4>

  <p><code>DataTransfer</code> objects are used to expose the <span>drag data store</span> that
  underlies a drag-and-drop operation.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>DataTransfer</dfn> {
  <span data-x="dom-DataTransfer">constructor</span>();

  attribute DOMString <span data-x="dom-DataTransfer-dropEffect">dropEffect</span>;
  attribute DOMString <span data-x="dom-DataTransfer-effectAllowed">effectAllowed</span>;

  [SameObject] readonly attribute <span>DataTransferItemList</span> <span data-x="dom-DataTransfer-items">items</span>;

  undefined <span data-x="dom-DataTransfer-setDragImage">setDragImage</span>(Element image, long x, long y);

  /* old interface */
  readonly attribute FrozenArray&lt;DOMString> <span data-x="dom-DataTransfer-types">types</span>;
  DOMString <span data-x="dom-DataTransfer-getData">getData</span>(DOMString format);
  undefined <span data-x="dom-DataTransfer-setData">setData</span>(DOMString format, DOMString data);
  undefined <span data-x="dom-DataTransfer-clearData">clearData</span>(optional DOMString format);
  [SameObject] readonly attribute <span>FileList</span> <span data-x="dom-DataTransfer-files">files</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>dataTransfer</var> = new <span subdfn data-x="dom-DataTransfer">DataTransfer</span>()</code></dt>

   <dd>
    <p>Creates a new <code>DataTransfer</code> object with an empty <span>drag data
    store</span>.</p>
   </dd>

   <dt><code data-x=""><var>dataTransfer</var>.<span subdfn data-x="dom-DataTransfer-dropEffect">dropEffect</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the kind of operation that is currently selected. If the kind of operation isn't one
    of those that is allowed by the <code
    data-x="dom-DataTransfer-effectAllowed">effectAllowed</code> attribute, then the operation will
    fail.</p>

    <p>Can be set, to change the selected operation.</p>

    <p>The possible values are "<code data-x="dom-DataTransfer-dropEffect-none">none</code>", "<code
    data-x="dom-DataTransfer-dropEffect-copy">copy</code>", "<code
    data-x="dom-DataTransfer-dropEffect-link">link</code>", and "<code
    data-x="dom-DataTransfer-dropEffect-move">move</code>".</p>
   </dd>


   <dt><code data-x=""><var>dataTransfer</var>.<span subdfn data-x="dom-DataTransfer-effectAllowed">effectAllowed</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns the kinds of operations that are to be allowed.</p>

    <p>Can be set (during the <code data-x="event-dnd-dragstart">dragstart</code> event), to change
    the allowed operations.</p>

    <p>The possible values are "<code data-x="dom-DataTransfer-effectAllowed-none">none</code>",
    "<code data-x="dom-DataTransfer-effectAllowed-copy">copy</code>", "<code
    data-x="dom-DataTransfer-effectAllowed-copyLink">copyLink</code>", "<code
    data-x="dom-DataTransfer-effectAllowed-copyMove">copyMove</code>", "<code
    data-x="dom-DataTransfer-effectAllowed-link">link</code>", "<code
    data-x="dom-DataTransfer-effectAllowed-linkMove">linkMove</code>", "<code
    data-x="dom-DataTransfer-effectAllowed-move">move</code>", "<code
    data-x="dom-DataTransfer-effectAllowed-all">all</code>", and "<code
    data-x="dom-DataTransfer-effectAllowed-uninitialized">uninitialized</code>",</p>
   </dd>


   <dt><code data-x=""><var>dataTransfer</var>.<span subdfn data-x="dom-DataTransfer-items">items</span></code></dt>

   <dd><p>Returns a <code>DataTransferItemList</code> object, with the drag data.</p></dd>


   <dt><code data-x=""><var>dataTransfer</var>.<span subdfn data-x="dom-DataTransfer-setDragImage">setDragImage</span>(<var>element</var>, <var>x</var>, <var>y</var>)</code></dt>

   <dd>
    <p>Uses the given element to update the drag feedback, replacing any previously specified
    feedback.</p>
   </dd>


   <dt><code data-x=""><var>dataTransfer</var>.<span subdfn data-x="dom-DataTransfer-types">types</span></code></dt>

   <dd>
    <p>Returns a <span>frozen array</span> listing the formats that were set in the <code
    data-x="event-dnd-dragstart">dragstart</code> event. In addition, if any files are being
    dragged, then one of the types will be the string "<code data-x="">Files</code>".</p>
   </dd>


   <dt><code data-x=""><var>data</var> = <var>dataTransfer</var>.<span subdfn data-x="dom-DataTransfer-getData">getData</span>(<var>format</var>)</code></dt>

   <dd><p>Returns the specified data. If there is no such data, returns the empty string.</p></dd>


   <dt><code data-x=""><var>dataTransfer</var>.<span subdfn data-x="dom-DataTransfer-setData">setData</span>(<var>format</var>, <var>data</var>)</code></dt>

   <dd><p>Adds the specified data.</p></dd>


   <dt><code data-x=""><var>dataTransfer</var>.<span subdfn data-x="dom-DataTransfer-clearData">clearData</span>([ <var>format</var> ])</code></dt>

   <dd><p>Removes the data of the specified formats. Removes all data if the argument is
   omitted.</p></dd>


   <dt><code data-x=""><var>dataTransfer</var>.<span subdfn data-x="dom-DataTransfer-files">files</span></code></dt>

   <dd><p>Returns a <code>FileList</code> of the files being dragged, if any.</p></dd>
  </dl>

  <p><code>DataTransfer</code> objects that are created as part of <a
  href="#dndevents">drag-and-drop events</a> are only valid while those events are being fired.</p>

  <div w-nodev>

  <p>A <code>DataTransfer</code> object is associated with a <span>drag data store</span> while it
  is valid.</p>

  <p>A <code>DataTransfer</code> object has an associated <dfn
  data-x="concept-DataTransfer-types">types array</dfn>, which is a
  <code data-x="frozen array">FrozenArray&lt;DOMString></code>, initially empty. When the contents
  of the <code>DataTransfer</code> object's <span>drag data store item list</span> change, or when
  the <code>DataTransfer</code> object becomes no longer associated with a <span>drag data
  store</span>, run the following steps:</p>

  <ol>
   <li><p>Let <var>L</var> be an empty sequence.</p></li>

   <li>
    <p>If the <code>DataTransfer</code> object is still associated with a <span>drag data
    store</span>, then:</p>

    <ol>
     <li><p>For each item in the <code>DataTransfer</code> object's <span>drag data store item
     list</span> <!-- in some order...? --> whose <span data-x="the drag data item kind">kind</span>
     is <i>text</i>, add an entry to <var>L</var> consisting of the item's <span
     data-x="the drag data item type string">type string</span>.</p></li>

     <li><p>If there are any items in the <code>DataTransfer</code> object's <span>drag data store
     item list</span> whose <span data-x="the drag data item kind">kind</span> is <i>File</i>, then
     add an entry to <var>L</var> consisting of the string "<code data-x="">Files</code>". (This
     value can be distinguished from the other values because it is not lowercase.)</p></li>

     <!-- <li><p>Sort the list...?</p></li> -->
    </ol>
   </li>

   <li><p>Set the <code>DataTransfer</code> object's <span data-x="concept-DataTransfer-types">types
   array</span> to the result of <span>creating a frozen array</span> from <var>L</var>.</p></li>
  </ol>

  <p>The <dfn constructor for="DataTransfer"><code
  data-x="dom-DataTransfer">DataTransfer()</code></dfn> constructor, when invoked, must return a
  newly created <code>DataTransfer</code> object initialized as follows:</p>

  <ol>
   <li><p>Set the <span>drag data store</span>'s <span data-x="drag data store item list">item
   list</span> to be an empty list.</p></li>

   <li><p>Set the <span>drag data store</span>'s <span data-x="drag data store mode">mode</span> to
   <span data-x="concept-dnd-rw">read/write mode</span>.</p></li>

   <li><p>Set the <code data-x="dom-DataTransfer-dropEffect">dropEffect</code> and
   <code data-x="dom-DataTransfer-effectAllowed">effectAllowed</code> to "none".</p></li>
  </ol>

  <p>The <dfn attribute for="DataTransfer"><code
  data-x="dom-DataTransfer-dropEffect">dropEffect</code></dfn> attribute controls the drag-and-drop
  feedback that the user is given during a drag-and-drop operation. When the
  <code>DataTransfer</code> object is created, the <code
  data-x="dom-DataTransfer-dropEffect">dropEffect</code> attribute is set to a string value. On
  getting, it must return its current value. On setting, if the new value is one of "<dfn><code
  data-x="dom-DataTransfer-dropEffect-none">none</code></dfn>", "<dfn><code
  data-x="dom-DataTransfer-dropEffect-copy">copy</code></dfn>", "<dfn><code
  data-x="dom-DataTransfer-dropEffect-link">link</code></dfn>", or "<dfn><code
  data-x="dom-DataTransfer-dropEffect-move">move</code></dfn>", then the attribute's current value
  must be set to the new value. Other values must be ignored.</p>

  <p>The <dfn attribute for="DataTransfer"><code
  data-x="dom-DataTransfer-effectAllowed">effectAllowed</code></dfn> attribute is used in the
  drag-and-drop processing model to initialize the <code
  data-x="dom-DataTransfer-dropEffect">dropEffect</code> attribute during the <code
  data-x="event-dnd-dragenter">dragenter</code> and <code
  data-x="event-dnd-dragover">dragover</code> events. When the <code>DataTransfer</code> object is
  created, the <code data-x="dom-DataTransfer-effectAllowed">effectAllowed</code> attribute is set
  to a string value. On getting, it must return its current value. On setting, if <span>drag data
  store</span>'s <span data-x="drag data store mode">mode</span> is the <span
  data-x="concept-dnd-rw">read/write mode</span> and the new value is one of "<dfn><code
  data-x="dom-DataTransfer-effectAllowed-none">none</code></dfn>", "<dfn><code
  data-x="dom-DataTransfer-effectAllowed-copy">copy</code></dfn>", "<dfn><code
  data-x="dom-DataTransfer-effectAllowed-copyLink">copyLink</code></dfn>", "<dfn><code
  data-x="dom-DataTransfer-effectAllowed-copyMove">copyMove</code></dfn>", "<dfn><code
  data-x="dom-DataTransfer-effectAllowed-link">link</code></dfn>", "<dfn><code
  data-x="dom-DataTransfer-effectAllowed-linkMove">linkMove</code></dfn>", "<dfn><code
  data-x="dom-DataTransfer-effectAllowed-move">move</code></dfn>", "<dfn><code
  data-x="dom-DataTransfer-effectAllowed-all">all</code></dfn>", or "<dfn><code
  data-x="dom-DataTransfer-effectAllowed-uninitialized">uninitialized</code></dfn>", then the attribute's
  current value must be set to the new value. Otherwise, it must be left unchanged.</p>

  <p>The <dfn attribute for="DataTransfer"><code data-x="dom-DataTransfer-items">items</code></dfn>
  attribute must return a <code>DataTransferItemList</code> object associated with the
  <code>DataTransfer</code> object.</p>

  <p>The <dfn method for="DataTransfer"><code
  data-x="dom-DataTransfer-setDragImage">setDragImage(<var>image</var>, <var>x</var>,
  <var>y</var>)</code></dfn> method must run the following steps:</p>

  <ol>
   <li><p>If the <code>DataTransfer</code> object is no longer associated with a <span>drag data
   store</span>, return. Nothing happens.</p></li>

   <li><p>If the <span>drag data store</span>'s <span data-x="drag data store mode">mode</span> is
   not the <span data-x="concept-dnd-rw">read/write mode</span>, return. Nothing
   happens.</p></li>

   <li><p>If <var>image</var> is an <code>img</code> element, then set the <span>drag data store
   bitmap</span> to the element's image (at its <span data-x="natural dimensions">natural
   size</span>); otherwise, set the <span>drag data store bitmap</span> to an image generated from
   the given element (the exact mechanism for doing so is not currently specified).</p></li>

   <li><p>Set the <span>drag data store hot spot coordinate</span> to the given <var>x</var>,
   <var>y</var> coordinate.</p></li>
  </ol>

  <p>The <dfn attribute for="DataTransfer"><code data-x="dom-DataTransfer-types">types</code></dfn>
  attribute must return this <code>DataTransfer</code> object's <span
  data-x="concept-DataTransfer-types">types array</span>.</p>

  <p>The <dfn method for="DataTransfer"><code
  data-x="dom-DataTransfer-getData">getData(<var>format</var>)</code></dfn> method must run the
  following steps:</p>

  <ol>
   <li><p>If the <code>DataTransfer</code> object is no longer associated with a <span>drag data
   store</span>, then return the empty string.</p></li>

   <li><p>If the <span>drag data store</span>'s <span data-x="drag data store mode">mode</span> is
   the <span data-x="concept-dnd-p">protected mode</span>, then return the empty string.</p></li>

   <li><p>Let <var>format</var> be the first argument, <span>converted to ASCII
   lowercase</span>.</p></li>

   <li><p>Let <var>convert-to-URL</var> be false.</p></li>

   <li><p>If <var>format</var> equals "<code data-x="">text</code>", change it to "<code
   data-x="">text/plain</code>".</p></li>

   <li><p>If <var>format</var> equals "<code data-x="">url</code>", change it to "<code
   data-x="">text/uri-list</code>" and set <var>convert-to-URL</var> to true.</p></li>

   <li><p>If there is no item in the <span>drag data store item list</span> whose <span data-x="the
   drag data item kind">kind</span> is <i>text</i> and whose <span data-x="the drag
   data item type string">type string</span> is equal to <var>format</var>, return the empty string.</p></li>

   <li><p>Let <var>result</var> be the data of the item in the <span>drag data store item
   list</span> whose <span data-x="the drag data item kind">kind</span> is <i>Plain Unicode
   string</i> and whose <span data-x="the drag data item type string">type string</span> is equal to
   <var>format</var>.</p></li>

   <li><p>If <var>convert-to-URL</var> is true, then parse <var>result</var> as appropriate for
   <code data-x="">text/uri-list</code> data, and then set <var>result</var> to the first URL from
   the list, if any, or the empty string otherwise. <ref>RFC2483</ref></p></li>

   <li><p>Return <var>result</var>.</p></li>
  </ol>

  <p>The <dfn method for="DataTransfer"><code
  data-x="dom-DataTransfer-setData">setData(<var>format</var>, <var>data</var>)</code></dfn> method
  must run the following steps:</p>

  <ol>
   <li><p>If the <code>DataTransfer</code> object is no longer associated with a <span>drag data
   store</span>, return. Nothing happens.</p></li>

   <li><p>If the <span>drag data store</span>'s <span data-x="drag data store mode">mode</span> is
   not the <span data-x="concept-dnd-rw">read/write mode</span>, return. Nothing
   happens.</p></li>

   <li><p>Let <var>format</var> be the first argument, <span>converted to ASCII
   lowercase</span>.</p></li>

   <li>
    <p>If <var>format</var> equals "<code data-x="">text</code>", change it to "<code
    data-x="">text/plain</code>".</p>

    <p>If <var>format</var> equals "<code data-x="">url</code>", change it to "<code
    data-x="">text/uri-list</code>".</p>
   </li>

   <li><p>Remove the item in the <span>drag data store item list</span> whose <span data-x="the drag
   data item kind">kind</span> is <i>text</i> and whose <span data-x="the drag data
   item type string">type string</span> is equal to <var>format</var>, if there is
   one.</p></li>

   <li><p>Add an item to the <span>drag data store item list</span> whose <span data-x="the drag data
   item kind">kind</span> is <i>text</i>, whose <span data-x="the drag data item type
   string">type string</span> is equal to <var>format</var>, and whose data is the string
   given by the method's second argument.</p></li>
  </ol>

  <p>The <dfn method for="DataTransfer"><code
  data-x="dom-DataTransfer-clearData">clearData(<var>format</var>)</code></dfn> method must run the
  following steps:</p>

  <ol>
   <li><p>If the <code>DataTransfer</code> object is no longer associated with a <span>drag data
   store</span>, return. Nothing happens.</p></li>

   <li><p>If the <span>drag data store</span>'s <span data-x="drag data store mode">mode</span> is
   not the <span data-x="concept-dnd-rw">read/write mode</span>, return. Nothing
   happens.</p></li>

   <li><p>If the method was called with no arguments, remove each item in the <span>drag data store
   item list</span> whose <span data-x="the drag data item kind">kind</span> is <i>Plain Unicode
   string</i>, and return.</p></li>

   <li><p>Set <var>format</var> to <var>format</var>, <span>converted to ASCII
   lowercase</span>.</p></li>

   <li>
    <p>If <var>format</var> equals "<code data-x="">text</code>", change it to "<code
    data-x="">text/plain</code>".</p>

    <p>If <var>format</var> equals "<code data-x="">url</code>", change it to "<code
    data-x="">text/uri-list</code>".</p>
   </li>

   <li><p>Remove the item in the <span>drag data store item list</span> whose <span data-x="the drag
   data item kind">kind</span> is <i>text</i> and whose <span data-x="the drag data
   item type string">type string</span> is equal to <var>format</var>, if there is
   one.</p></li>
  </ol>

  <p class="note">The <code data-x="dom-DataTransfer-clearData">clearData()</code> method does not
  affect whether any files were included in the drag, so the <code
  data-x="dom-DataTransfer-types">types</code> attribute's list might still not be empty after
  calling <code data-x="dom-DataTransfer-clearData">clearData()</code> (it would still contain the
  "<code data-x="">Files</code>" string if any files were included in the drag).</p>

  <p>The <dfn attribute for="DataTransfer"><code data-x="dom-DataTransfer-files">files</code></dfn>
  attribute must return a <span>live</span> <code>FileList</code> sequence consisting of
  <code>File</code> objects representing the files found by the following steps.  Furthermore, for a
  given <code>FileList</code> object and a given underlying file, the same <code>File</code> object
  must be used each time.</p>

  <ol>
   <li><p>Start with an empty list <var>L</var>.</p></li>

   <li><p>If the <code>DataTransfer</code> object is no longer associated with a <span>drag data
   store</span>, the <code>FileList</code> is empty. Return the empty list <var>L</var>.</p></li>

   <li><p>If the <span>drag data store</span>'s <span data-x="drag data store mode">mode</span> is
   the <span data-x="concept-dnd-p">protected mode</span>, return the empty list
   <var>L</var>.</p></li>

   <li><p>For each item in the <span>drag data store item list</span> <!-- in some order...? -->
   whose <span data-x="the drag data item kind">kind</span> is <i>File</i><!-- DND-v3: (not
   <i>Blob</i>) -->, add the item's data (the file, in particular its name and contents, as well as
   its <span data-x="the drag data item type string">type</span>) to the list <var>L</var>.</p></li>

   <!-- <li><p>Sort the list...?</p></li> -->

   <li><p>The files found by these steps are those in the list <var>L</var>.</p></li>
  </ol>

  <p class="note">This version of the API does not expose the types of the files during the
  drag.</p>

  </div>


  <h5>The <code>DataTransferItemList</code> interface</h5>

  <p>Each <code>DataTransfer</code> object is associated with a <code>DataTransferItemList</code>
  object.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>DataTransferItemList</dfn> {
  readonly attribute unsigned long <span data-x="dom-DataTransferItemList-length">length</span>;
  <a href="#dom-datatransferitemlist-item">getter</a> <span>DataTransferItem</span> (unsigned long index);
  <span>DataTransferItem</span>? <span data-x="dom-DataTransferItemList-add">add</span>(DOMString data, DOMString type);<!--
DND-v3:  <span>DataTransferItem</span>? <span data-x="dom-DataTransferItemList-add">add</span>(<span>Blob</span> data);-->
  <span>DataTransferItem</span>? <span data-x="dom-DataTransferItemList-add">add</span>(<span>File</span> data);<!--
DND-v4:  <span>DataTransferItem</span>? <span data-x="dom-DataTransferItemList-add">add</span>(any data, DOMString type);--><!--
DND-v5:  <span>DataTransferItem</span>? <span data-x="dom-DataTransferItemList-add">add</span>(<span>DataTransferPromise</span> data);-->
  undefined <span data-x="dom-DataTransferItemList-remove">remove</span>(unsigned long index);
  undefined <span data-x="dom-DataTransferItemList-clear">clear</span>();
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>items</var>.<span subdfn data-x="dom-DataTransferItemList-length">length</span></code></dt>

   <dd><p>Returns the number of items in the <span>drag data store</span>.</p></dd>


   <dt><code data-x=""><var>items</var>[<var>index</var>]</code></dt>

   <dd>
    <p>Returns the <code>DataTransferItem</code> object representing the <var>index</var>th entry in
    the <span>drag data store</span>.</p>
   </dd>


   <dt><code data-x=""><var>items</var>.<span subdfn data-x="dom-DataTransferItemList-remove">remove</span>(<var>index</var>)</code></dt>

   <dd><p>Removes the <var>index</var>th entry in the <span>drag data store</span>.</p></dd>


   <dt><code data-x=""><var>items</var>.<span subdfn data-x="dom-DataTransferItemList-clear">clear</span>()</code></dt>

   <dd><p>Removes all the entries in the <span>drag data store</span>.</p></dd>


   <dt><code data-x=""><var>items</var>.<span subdfn data-x="dom-DataTransferItemList-add">add</span>(<var>data</var>)</code></dt>
   <dt><code data-x=""><var>items</var>.<span data-x="dom-DataTransferItemList-add">add</span>(<var>data</var>, <var>type</var>)</code></dt>

   <dd>
    <p>Adds a new entry for the given data to the <span>drag data store</span>. If the data is plain
    text <!-- DND-v4: or an object --> then a <var>type</var> string has to be provided also.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>While the <code>DataTransferItemList</code> object's <code>DataTransfer</code> object is
  associated with a <span>drag data store</span>, the <code>DataTransferItemList</code> object's
  <i>mode</i> is the same as the <span>drag data store mode</span>. When the
  <code>DataTransferItemList</code> object's <code>DataTransfer</code> object is <em>not</em>
  associated with a <span>drag data store</span>, the <code>DataTransferItemList</code> object's
  <i>mode</i> is the <i>disabled mode</i>. The <span>drag data store</span> referenced in this
  section (which is used only when the <code>DataTransferItemList</code> object is not in the
  <i>disabled mode</i>) is the <span>drag data store</span> with which the
  <code>DataTransferItemList</code> object's <code>DataTransfer</code> object is associated.</p>

  <p>The <dfn attribute for="DataTransferItemList"><code
  data-x="dom-DataTransferItemList-length">length</code></dfn> attribute must return zero if the
  object is in the <i>disabled mode</i>; otherwise it must return the number of items in the
  <span>drag data store item list</span>.</p>

  <p>When a <code>DataTransferItemList</code> object is not in the <i>disabled mode</i>, its
  <span>supported property indices</span> are the <span>indices</span> of the <span>drag data store
  item list</span>.</p>

  <p id="dom-datatransferitemlist-item">To <span>determine the value of an indexed property</span>
  <var>i</var> of a <code>DataTransferItemList</code> object, the user agent must return a
  <code>DataTransferItem</code> object representing the <var>i</var>th item in the
  <span>drag data store</span>. The same object must be returned each time a particular item is
  obtained from this <code>DataTransferItemList</code> object. The <code>DataTransferItem</code>
  object must be associated with the same <code>DataTransfer</code> object as the
  <code>DataTransferItemList</code> object when it is first created.</p>

  <p>The <dfn method for="DataTransferItemList"><code
  data-x="dom-DataTransferItemList-add">add()</code></dfn> method must run the following steps:</p>

  <ol>
   <li><p>If the <code>DataTransferItemList</code> object is not in the <i
   data-x="concept-dnd-rw">read/write mode</i>, return null.</p></li>

   <li>
    <p>Jump to the appropriate set of steps from the following list:</p>

    <dl class="switch">

     <dt>If the first argument to the method is a string</dt>

     <dd>
      <p>If there is already an item in the <span>drag data store item list</span> whose <span
      data-x="the drag data item kind">kind</span> is <i>text</i> and whose <span
      data-x="the drag data item type string">type string</span> is equal to the value of the
      method's second argument, <span>converted to ASCII lowercase</span>, then throw a
      <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p>

      <p>Otherwise, add an item to the <span>drag data store item list</span> whose <span data-x="the
      drag data item kind">kind</span> is <i>text</i>, whose <span data-x="the drag
      data item type string">type string</span> is equal to the value of the method's second
      argument, <span>converted to ASCII lowercase</span>, and whose data is the string given by the
      method's first argument.</p>
     </dd>

<!--DND-v3:
     <dt>If the first argument to the method is a <code>Blob</code></dt>

     <dd>
      <p>Add an item to the <span>drag data store item list</span> whose <span data-x="the drag data
      item kind">kind</span> is <i>Blob</i>, whose <span data-x="the drag data item type string">type
      string</span> is the <code data-x="dom-Blob-type">type</code> of the <code>Blob</code>,
      <span>converted to ASCII lowercase</span>, and whose data is the same as the
      <code>Blob</code>'s data.</p>
     </dd>
-->

     <dt>If the first argument to the method is a <code>File</code></dt>

     <dd>
      <p>Add an item to the <span>drag data store item list</span> whose <span data-x="the drag data
      item kind">kind</span> is <i>File</i>, whose <span data-x="the drag data item type string">type
      string</span> is the <code data-x="dom-Blob-type">type</code> of the <code>File</code>,
      <span>converted to ASCII lowercase</span>, and whose data is the same as the
      <code>File</code>'s data.</p>
     </dd>

<!--DND-v4: (might want to prevent duplicates like for strings; see above)
[make sure that the cloning happens before any side-effects can happen]
     <dt>Otherwise

     <dd>
      <p>Add an item to the <span>drag data store item list</span> whose <span data-x="the drag data
      item kind">kind</span> is <i>Object</i>, whose <span data-x="the drag data item type
      string">type string</span> is equal to the value of the method's second argument,
      <span>converted to ASCII lowercase</span>, and whose data is a <span>structured clone</span>
      of the method's first argument. If creating the clone throws an exception, then throw that
      exception.</p>
     </dd>
-->
    </dl>
   </li>

   <li><p><a href="#dom-datatransferitemlist-item">Determine the value of the indexed property</a>
   corresponding to the newly added item, and return that value (a newly created
   <code>DataTransferItem</code> object).</p></li>
  </ol>

  <p>The <dfn method for="DataTransferItemList"><code
  data-x="dom-DataTransferItemList-remove">remove(<var>index</var>)</code></dfn> method must run
  these steps:</p>

  <ol>
   <li><p>If the <code>DataTransferItemList</code> object is not in the <i
   data-x="concept-dnd-rw">read/write mode</i>, throw an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If the <span>drag data store</span> does not contain an <var>index</var>th item, then
   return.</p></li>

   <li><p>Remove the <var>index</var>th item from the <span>drag data store</span>.</p></li>
  </ol>

  <p>The <dfn method for="DataTransferItemList"><code
  data-x="dom-DataTransferItemList-clear">clear()</code></dfn> method, if the
  <code>DataTransferItemList</code> object is in the <i data-x="concept-dnd-rw">read/write mode</i>,
  must remove all the items from the <span>drag data store</span>. Otherwise, it must do
  nothing.</p>

  </div>


  <h5>The <code>DataTransferItem</code> interface</h5>

  <p>Each <code>DataTransferItem</code> object is associated with a <code>DataTransfer</code>
  object.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>DataTransferItem</dfn> {
  readonly attribute DOMString <span data-x="dom-DataTransferItem-kind">kind</span>;
  readonly attribute DOMString <span data-x="dom-DataTransferItem-type">type</span>;
  undefined <span data-x="dom-DataTransferItem-getAsString">getAsString</span>(<span>FunctionStringCallback</span>? _callback);<!--
DND-v3:  <span>Blob</span> <span data-x="dom-DataTransferItem-getAsBlob">getAsBlob</span>();-->
  <span>File</span>? <span data-x="dom-DataTransferItem-getAsFile">getAsFile</span>();<!--
DND-v4:  undefined <span data-x="dom-DataTransferItem-getAsObject">getAsObject</span>(<span>FunctionObjectCallback</span> _callback);-->
};

callback <dfn callback>FunctionStringCallback</dfn> = undefined (DOMString data);<!--

// DND-v4:
callback <dfn>FunctionObjectCallback</dfn> = undefined (any data);--></code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>item</var>.<span subdfn data-x="dom-DataTransferItem-kind">kind</span></code></dt>
   <dd>
    <p>Returns <span>the drag data item kind</span>, one of: "string", <!-- DND-v3: "blob", -->
    "file"<!-- DND-v4: , "object" -->.</p>
   </dd>

   <dt><code data-x=""><var>item</var>.<span subdfn data-x="dom-DataTransferItem-type">type</span></code></dt>
   <dd><p>Returns <span>the drag data item type string</span>.</p></dd>

   <dt><code data-x=""><var>item</var>.<span subdfn data-x="dom-DataTransferItem-getAsString">getAsString</span>(<var>callback</var>)</code></dt>
   <dd>
    <p>Invokes the callback with the string data as the argument, if <span>the drag data item
    kind</span> is <i>text</i>.</p>
   </dd>

<!-- DND-v3:
   <dt><var>file</var> = <var>item</var>.<code subdfn data-x="dom-DataTransferItem-getAsBlob">getAsBlob</code>()</dt>
   <dd>
    <p>Returns a <code>Blob</code> object, if <span>the drag data item kind</span> is <i>Blob</i> or
    <i>File</i>.</p>
   </dd>
-->

   <dt><code data-x=""><var>file</var> = <var>item</var>.<span subdfn data-x="dom-DataTransferItem-getAsFile">getAsFile</span>()</code></dt>
   <dd>
    <p>Returns a <code>File</code> object, if <span>the drag data item kind</span> is
    <i>File</i>.</p>
   </dd>

<!-- DND-v4:
   <dt><var>file</var> = <var>item</var>.<code subdfn data-x="dom-DataTransferItem-getAsObject">getAsObject</code>()</dt>
   <dd>
    <p>Invokes the callback with the cloned object data as the argument, if <span>the drag data item
    kind</span> is <i>text</i> or <i>Object</i>.</p>
   </dd>
-->
  </dl>

  <div w-nodev>

  <p>While the <code>DataTransferItem</code> object's <code>DataTransfer</code> object is associated
  with a <span>drag data store</span> and that <span>drag data store</span>'s <span>drag data store
  item list</span> still contains the item that the <code>DataTransferItem</code> object represents,
  the <code>DataTransferItem</code> object's <i>mode</i> is the same as the <span>drag data store
  mode</span>. When the <code>DataTransferItem</code> object's <code>DataTransfer</code> object is
  <em>not</em> associated with a <span>drag data store</span>, or if the item that the
  <code>DataTransferItem</code> object represents has been removed from the relevant <span>drag data
  store item list</span>, the <code>DataTransferItem</code> object's <i>mode</i> is the <i>disabled
  mode</i>. The <span>drag data store</span> referenced in this section (which is used only when the
  <code>DataTransferItem</code> object is not in the <i>disabled mode</i>) is the <span>drag data
  store</span> with which the <code>DataTransferItem</code> object's <code>DataTransfer</code>
  object is associated.</p>

  <p>The <dfn attribute for="DataTransferItem"><code
  data-x="dom-DataTransferItem-kind">kind</code></dfn> attribute must return the empty string if the
  <code>DataTransferItem</code> object is in the <i>disabled mode</i>; otherwise it must return the
  string given in the cell from the second column of the following table from the row whose cell in
  the first column contains <span>the drag data item kind</span> of the item represented by the
  <code>DataTransferItem</code> object:</p>

  <table>
   <thead>
    <tr> <th> Kind <th> String
   <tbody>
    <tr> <td> <i>Text</i> <td> "<code data-x="">string</code>"
<!-- DND-v3:    <tr> <td> <i>Blob</i> <td> "<code data-x="">blob</code>"-->
    <tr> <td> <i>File</i> <td> "<code data-x="">file</code>"
<!-- DND-v4:    <tr> <td> <i>Object</i> <td> "<code data-x="">object</code>"-->
  </table>

  <p>The <dfn attribute for="DataTransferItem"><code
  data-x="dom-DataTransferItem-type">type</code></dfn> attribute must return the empty string if the
  <code>DataTransferItem</code> object is in the <i>disabled mode</i>; otherwise it must return
  <span>the drag data item type string</span> of the item represented by the
  <code>DataTransferItem</code> object.</p>

  <p>The <dfn method for="DataTransferItem"><code
  data-x="dom-DataTransferItem-getAsString">getAsString(<var>callback</var>)</code></dfn> method
  must run the following steps:</p>

  <ol>
   <li><p>If the <var>callback</var> is null, return.</p></li>

   <li><p>If the <code>DataTransferItem</code> object is not in the <i
   data-x="concept-dnd-rw">read/write mode</i> or the <i data-x="concept-dnd-ro">read-only mode</i>,
   return. The callback is never invoked.</p></li>

   <li><p>If <span>the drag data item kind</span> is not <i>text</i>, then return.
   The callback is never invoked.</p></li>

   <li><p>Otherwise, <span>queue a task</span> to invoke <var>callback</var>, passing the
   actual data of the item represented by the <code>DataTransferItem</code> object as the
   argument.</p></li>
  </ol>

<!--DND-v3:
  <p>The <dfn><code data-x="dom-DataTransferItem-getAsBlob">getAsBlob()</code></dfn> method must run
  the following steps:</p>

  <ol>
   <li><p>If the <code>DataTransferItem</code> object is not in the <i
   data-x="concept-dnd-rw">read/write mode</i> or the <i data-x="concept-dnd-ro">read-only mode</i>,
   return null.</p></li>

   <li>
--><!--DND-v4:
    <p>If <span>the drag data item kind</span> is <i>Object</i>, return null.</p></li>
--><!--DND-v3:

    <p>If <span>the drag data item kind</span> is <i>File</i>, then return a new <code>File</code>
    object representing the actual data of the item represented by the <code>DataTransferItem</code>
    object.</p>

    <p>If <span>the drag data item kind</span> is <i>Unicode Data string</i>, then return a new
    <code>Blob</code> object representing the actual data of the item represented by the
    <code>DataTransferItem</code> object, with the <code data-x="dom-Blob-type">type</code> of the
    <code>Blob</code> being <span>the drag data item type string</span> and with the binary data of
    the <code>Blob</code> object being the Unicode string encoded as UTF-8. <ref>ENCODING</ref></p>

    <p>Otherwise, <span>the drag data item kind</span> is <i>Blob</i>; return a new
    <code>Blob</code> object representing the actual data of the item represented by the
    <code>DataTransferItem</code> object.</p>
   </li>
  </ol>
-->

  <p>The <dfn method for="DataTransferItem"><code
  data-x="dom-DataTransferItem-getAsFile">getAsFile()</code></dfn> method must run the following
  steps:</p>

  <ol>
   <li><p>If the <code>DataTransferItem</code> object is not in the <i
   data-x="concept-dnd-rw">read/write mode</i> or the <i data-x="concept-dnd-ro">read-only mode</i>,
   then return null.</p></li>

   <li><p>If <span>the drag data item kind</span> is not <i>File</i>, then return null.</p></li>

   <li><p>Return a new <code>File</code> object representing the actual data of the item represented
   by the <code>DataTransferItem</code> object.</p>
  </ol>

<!--DND-v4:
  <p>The <dfn><code data-x="dom-DataTransferItem-getAsObject">getAsObject(<var>callback</var>)</code></dfn> method must run the following steps:</p>

  <ol>
   <li><p>If the <code>DataTransferItem</code> object is not in the <i
   data-x="concept-dnd-rw">read/write mode</i> or the <i data-x="concept-dnd-ro">read-only mode</i>,
   return null.</p></li>

   <li>
    <p>Let <var>data</var> be the actual data of the item represented by the
    <code>DataTransferItem</code> object.</p>

    <p>If <span>the drag data item kind</span> is <i>Unicode Data string</i>, then <var>data</var>
    is a <code data-x="idl-DOMString">DOMString</code> containing the actual data.</p>

    <p>If <span>the drag data item kind</span> is <i>Blob</i>, then <var>data</var> is a
    <code>Blob</code> representing the actual data.</p>

    <p>If <span>the drag data item kind</span> is <i>File</i>, then <var>data</var> is a
    <code>File</code> representing the actual data.</p>

    <p>If <span>the drag data item kind</span> is <i>Object</i>, then <var>data</var> is the object
    that is the actual data.</p>
   </li>

   <li><span>Queue a task</span> to invoke <var>callback</var>, passing a <span>structured
   clone</span> of <var>data</var> as the argument.</p></li>
  </ol>
-->

  </div>



  <h4>The <code>DragEvent</code> interface</h4>

  <p>The drag-and-drop processing model involves several events. They all use the
  <code>DragEvent</code> interface.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>DragEvent</dfn> : <span>MouseEvent</span> {
  constructor(DOMString type, optional <span>DragEventInit</span> eventInitDict = {});

  readonly attribute <span>DataTransfer</span>? <span data-x="dom-DragEvent-dataTransfer">dataTransfer</span>;
};

dictionary <dfn dictionary>DragEventInit</dfn> : <span>MouseEventInit</span> {
  <span>DataTransfer</span>? dataTransfer = null;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-DragEvent-dataTransfer">dataTransfer</span></code></dt>

   <dd><p>Returns the <code>DataTransfer</code> object for the event.</p></dd>
  </dl>

  <p class="note">Although, for consistency with other event interfaces, the <code>DragEvent</code>
  interface has a constructor, it is not particularly useful. In particular, there's no way to
  create a useful <code>DataTransfer</code> object from script, as <code>DataTransfer</code> objects
  have a processing and security model that is coordinated by the browser during drag-and-drops.</p>

  <div w-nodev>

  <p>The <dfn attribute for="DragEvent"><code
  data-x="dom-DragEvent-dataTransfer">dataTransfer</code></dfn> attribute of the
  <code>DragEvent</code> interface must return the value it was initialized to. It represents the
  context information for the event.</p>

  </div>

  <div w-nodev>

  <p>When a user agent is required to <dfn>fire a DND event</dfn> named <var>e</var> at an element,
  using a particular <span>drag data store</span>, and optionally with a specific <var>related
  target</var>, the user agent must run the following steps:</p>

  <ol>
   <li>Let <var>dataDragStoreWasChanged</var> be false.</li>

   <li><p>If no specific <var>related target</var> was provided, set <var>related target</var> to
   null.</p></li>

   <li><p>Let <var>window</var> be the <span>relevant global object</span> of the
   <code>Document</code> object of the specified target element.</p></li>

   <li>
    <p>If <var>e</var> is <code data-x="event-dnd-dragstart">dragstart</code>, then set the
    <span>drag data store mode</span> to the <span data-x="concept-dnd-rw">read/write mode</span>
    and set <var>dataDragStoreWasChanged</var> to true.</p>

    <p>If <var>e</var> is <code data-x="event-dnd-drop">drop</code>, set the <span>drag data store
    mode</span> to the <span data-x="concept-dnd-ro">read-only mode</span>.</p>
   </li>

   <li><p>Let <var>dataTransfer</var> be a newly created <code>DataTransfer</code> object
   associated with the given <span>drag data store</span>.</p></li>

   <li><!--en-GB--><p id="effectAllowed-initialisation">Set the <code
   data-x="dom-DataTransfer-effectAllowed">effectAllowed</code> attribute to the <span>drag data
   store</span>'s <span>drag data store allowed effects state</span>.</p></li>

   <li>
    <!--en-GB--><p id="dropEffect-initialisation">Set the <code
    data-x="dom-DataTransfer-dropEffect">dropEffect</code> attribute to "<code
    data-x="dom-DataTransfer-dropEffect-none">none</code>" if <var>e</var> is <code
    data-x="event-dnd-dragstart">dragstart</code>, <code data-x="event-dnd-drag">drag</code>, or
    <code data-x="event-dnd-dragleave">dragleave</code>; to the value corresponding to the
    <span>current drag operation</span> if <var>e</var> is <code data-x="event-dnd-drop">drop</code>
    or <code data-x="event-dnd-dragend">dragend</code>; and to a value based on the <code
    data-x="dom-DataTransfer-effectAllowed">effectAllowed</code> attribute's value and the
    drag-and-drop source, as given by the following table, otherwise (i.e. if <var>e</var> is <code
    data-x="event-dnd-dragenter">dragenter</code> or <code
    data-x="event-dnd-dragover">dragover</code>):</p>

    <table>
     <thead>
      <tr>
       <th><code data-x="dom-DataTransfer-effectAllowed">effectAllowed</code></th>
       <th><code data-x="dom-DataTransfer-dropEffect">dropEffect</code></th>
      </tr>
     </thead>
     <tr>
      <td>"<code data-x="dom-DataTransfer-effectAllowed-none">none</code>"</td>
      <td>"<code data-x="dom-DataTransfer-dropEffect-none">none</code>"</td>
     </tr>
     <tr>
      <td>"<code data-x="dom-DataTransfer-effectAllowed-copy">copy</code>"</td>
      <td>"<code data-x="dom-DataTransfer-dropEffect-copy">copy</code>"</td>
     </tr>
     <tr>
      <td>"<code data-x="dom-DataTransfer-effectAllowed-copyLink">copyLink</code>"</td>
      <td>"<code data-x="dom-DataTransfer-dropEffect-copy">copy</code>", or, <span data-x="concept-platform-dropEffect-override">if appropriate</span>, "<code data-x="dom-DataTransfer-dropEffect-link">link</code>"</td>
     </tr>
     <tr>
      <td>"<code data-x="dom-DataTransfer-effectAllowed-copyMove">copyMove</code>"</td>
      <td>"<code data-x="dom-DataTransfer-dropEffect-copy">copy</code>", or, <span data-x="concept-platform-dropEffect-override">if appropriate</span>, "<code data-x="dom-DataTransfer-dropEffect-move">move</code>"</td>
     </tr>
     <tr>
      <td>"<code data-x="dom-DataTransfer-effectAllowed-all">all</code>"</td>
      <td>"<code data-x="dom-DataTransfer-dropEffect-copy">copy</code>", or, <span data-x="concept-platform-dropEffect-override">if appropriate</span>, either "<code data-x="dom-DataTransfer-dropEffect-link">link</code>" or "<code data-x="dom-DataTransfer-dropEffect-move">move</code>"</td>
     </tr>
     <tr>
      <td>"<code data-x="dom-DataTransfer-effectAllowed-link">link</code>"</td>
      <td>"<code data-x="dom-DataTransfer-dropEffect-link">link</code>"</td>
     </tr>
     <tr>
      <td>"<code data-x="dom-DataTransfer-effectAllowed-linkMove">linkMove</code>"</td>
      <td>"<code data-x="dom-DataTransfer-dropEffect-link">link</code>", or, <span data-x="concept-platform-dropEffect-override">if appropriate</span>, "<code data-x="dom-DataTransfer-dropEffect-move">move</code>"</td>
     </tr>
     <tr>
      <td>"<code data-x="dom-DataTransfer-effectAllowed-move">move</code>"</td>
      <td>"<code data-x="dom-DataTransfer-dropEffect-move">move</code>"</td>
     </tr>
     <tr>
      <td>"<code data-x="dom-DataTransfer-effectAllowed-uninitialized">uninitialized</code>" when what is being dragged is a selection from a text control</td>
      <td>"<code data-x="dom-DataTransfer-dropEffect-move">move</code>", or, <span data-x="concept-platform-dropEffect-override">if appropriate</span>, either "<code data-x="dom-DataTransfer-dropEffect-copy">copy</code>" or "<code data-x="dom-DataTransfer-dropEffect-link">link</code>"</td>
     </tr>
     <tr>
      <td>"<code data-x="dom-DataTransfer-effectAllowed-uninitialized">uninitialized</code>" when what is being dragged is a selection</td>
      <td>"<code data-x="dom-DataTransfer-dropEffect-copy">copy</code>", or, <span data-x="concept-platform-dropEffect-override">if appropriate</span>, either "<code data-x="dom-DataTransfer-dropEffect-link">link</code>" or "<code data-x="dom-DataTransfer-dropEffect-move">move</code>"</td>
     </tr>
     <tr>
      <td>"<code data-x="dom-DataTransfer-effectAllowed-uninitialized">uninitialized</code>" when what is being dragged is an <code>a</code> element with an <code data-x="attr-hyperlink-href">href</code> attribute</td>
      <td>"<code data-x="dom-DataTransfer-dropEffect-link">link</code>", or, <span data-x="concept-platform-dropEffect-override">if appropriate</span>, either "<code data-x="dom-DataTransfer-dropEffect-copy">copy</code>" or "<code data-x="dom-DataTransfer-dropEffect-move">move</code>"</td>
     </tr>
     <tr>
      <td>Any other case</td>
      <td>"<code data-x="dom-DataTransfer-dropEffect-copy">copy</code>", or, <span data-x="concept-platform-dropEffect-override">if appropriate</span>, either "<code data-x="dom-DataTransfer-dropEffect-link">link</code>" or "<code data-x="dom-DataTransfer-dropEffect-move">move</code>"</td>
     </tr>
    </table>

    <p>Where the table above provides <dfn data-x="concept-platform-dropEffect-override">possibly
    appropriate alternatives</dfn>, user agents may instead use the listed alternative values if
    platform conventions dictate that the user has requested those alternate effects.</p>

    <p class="example">For example, Windows platform conventions are such that dragging while
    holding the "alt" key indicates a preference for linking the data, rather than moving or copying
    it. Therefore, on a Windows system, if "<code data-x="dom-DataTransfer-dropEffect-link">link</code>" is an option according to
    the table above while the "alt" key is depressed, the user agent could select that instead of
    "<code data-x="dom-DataTransfer-dropEffect-copy">copy</code>" or "<code data-x="dom-DataTransfer-dropEffect-move">move</code>".</p>
   </li>

   <li><p>Let <var>event</var> be the result of <span>creating an event</span> using
   <code>DragEvent</code>.</p></li>

   <li><p>Initialize <var>event</var>'s <code data-x="dom-Event-type">type</code> attribute to
   <var>e</var>, its <code data-x="dom-Event-bubbles">bubbles</code> attribute to true, its <code
   data-x="dom-UIEvent-view">view</code> attribute to <var>window</var>, its <code
   data-x="dom-MouseEvent-relatedTarget">relatedTarget</code> attribute to <var>related
   target</var>, and its <code data-x="dom-DragEvent-dataTransfer">dataTransfer</code> attribute to
   <var>dataTransfer</var>.</p></li>

   <li><p>If <var>e</var> is not <code data-x="event-dnd-dragleave">dragleave</code> or <code
   data-x="event-dnd-dragend">dragend</code>, then initialize <var>event</var>'s <code
   data-x="dom-Event-cancelable">cancelable</code> attribute to true.</p></li>

   <li>
    <p>Initialize <var>event</var>'s mouse and key attributes initialized according to the state of
    the input devices as they would be for user interaction events.</p>

    <!-- interaction event spec point -->

    <p>If there is no relevant pointing device, then initialize <var>event</var>'s <code
    data-x="">screenX</code>, <code data-x="">screenY</code>, <code
    data-x="mouseevent-clientx">clientX</code>, <code data-x="mouseevent-clienty">clientY</code>,
    and <code data-x="">button</code> attributes to 0.</p>
   </li>

   <li><p><span data-x="concept-event-dispatch">Dispatch</span> <var>event</var> at the specified
   target element.</p></li>

   <li><p>Set the <span>drag data store allowed effects state</span> to the current value of
   <var>dataTransfer</var>'s <code data-x="dom-DataTransfer-effectAllowed">effectAllowed</code>
   attribute. (It can only have changed value if <var>e</var> is <code
   data-x="event-dnd-dragstart">dragstart</code>.)</p></li>

   <li><p>If <var>dataDragStoreWasChanged</var> is true, then set the <span>drag data store
   mode</span> back to the <span data-x="concept-dnd-p">protected mode</span>.</p></li>

   <li><p>Break the association between <var>dataTransfer</var> and the <span>drag data
   store</span>.</p></li>
  </ol>

  </div>


  <div w-nodev>

  <h4 id="drag-and-drop-processing-model">Processing model</h4>

  <p>When the user attempts to begin a drag operation, the user agent must run the following steps.
  User agents must act as if these steps were run even if the drag actually started in another
  document or application and the user agent was not aware that the drag was occurring until it
  intersected with a document under the user agent's purview.</p>

  <ol>
   <li>
    <p>Determine what is being dragged, as follows:</p>

    <p>If the drag operation was invoked on a selection, then it is the selection that is being
    dragged.</p>

    <p>Otherwise, if the drag operation was invoked on a <code>Document</code>, it is the first
    element, going up the ancestor chain, starting at the node that the user tried to drag, that has
    the IDL attribute <code data-x="dom-draggable">draggable</code> set to true. If there is no such
    element, then nothing is being dragged; return, the drag-and-drop operation is never
    started.</p>

    <p>Otherwise, the drag operation was invoked outside the user agent's purview. What is being
    dragged is defined by the document or application where the drag was started.</p>

    <p class="note"><code>img</code> elements and <code>a</code> elements with an <code
    data-x="attr-hyperlink-href">href</code> attribute have their <code
    data-x="dom-draggable">draggable</code> attribute set to true by default.</p>
   </li>

   <li><p><span>Create a drag data store</span>. All the DND events fired subsequently by the steps
   in this section must use this <span>drag data store</span>.</p></li>

   <li>
    <p>Establish which DOM node is the <dfn>source node</dfn>, as follows:</p>

    <p>If it is a selection that is being dragged, then the <span>source node</span> is the
    <code>Text</code> node that the user started the drag on (typically the <code>Text</code> node
    that the user originally clicked). If the user did not specify a particular node, for example if
    the user just told the user agent to begin a drag of "the selection", then the <span>source
    node</span> is the first <code>Text</code> node containing a part of the selection.</p>

    <p>Otherwise, if it is an element that is being dragged, then the <span>source node</span> is
    the element that is being dragged.</p>

    <p>Otherwise, the <span>source node</span> is part of another document or application. When this
    specification requires that an event be dispatched at the <span>source node</span> in this case,
    the user agent must instead follow the platform-specific conventions relevant to that
    situation.</p>

    <p class="note">Multiple events are fired on the <span>source node</span> during the course of
    the drag-and-drop operation.</p>
   </li>

   <li>
    <p>Determine the <dfn>list of dragged nodes</dfn>, as follows:</p>

    <p>If it is a selection that is being dragged, then the <span>list of dragged nodes</span>
    contains, in <span>tree order</span>, every node that is partially or completely included in the
    selection (including all their ancestors).</p>

    <p>Otherwise, the <span>list of dragged nodes</span> contains only the <span>source node</span>,
    if any.</p>
   </li>

   <li>
    <p>If it is a selection that is being dragged, then add an item to the <span>drag data store
    item list</span>, with its properties set as follows:</p>

    <dl>
     <dt><span>The drag data item type string</span>
     <dd>"<code>text/plain</code>"</dd>

     <dt><span>The drag data item kind</span>
     <dd><i>Text</i></dd>

     <dt>The actual data</dt>
     <dd>The text of the selection</dd>
    </dl>

    <p>Otherwise, if any files are being dragged, then add one item per file to the <span>drag data
    store item list</span>, with their properties set as follows:</p>

    <dl>
     <dt><span>The drag data item type string</span>
     <dd>The MIME type of the file, if known, or "<code>application/octet-stream</code>" otherwise.</dd>

     <dt><span>The drag data item kind</span>
     <dd><i>File</i></dd>

     <dt>The actual data</dt>
     <dd>The file's contents and name.</dd>
    </dl>

    <p class="note">Dragging files can currently only happen from outside a <span>navigable</span>,
    for example from a file system manager application.</p>

    <p>If the drag initiated outside of the application, the user agent must add items to the
    <span>drag data store item list</span> as appropriate for the data being dragged, honoring
    platform conventions where appropriate; however, if the platform conventions do not use <span
    data-x="MIME type">MIME types</span> to label dragged data, the user agent must make a
    best-effort attempt to map the types to MIME types, and, in any case, all the <span data-x="the
    drag data item type string">drag data item type strings</span> must be <span>converted to ASCII
    lowercase</span>.</p>

    <p>User agents may also add one or more items representing the selection or dragged element(s)
    in other forms, e.g. as HTML.</p>
   </li>

   <!-- DND-v2: text/html as an export format -->

   <li>
    <p>If the <span>list of dragged nodes</span> is not empty, then <span data-x="extracting
    JSON">extract the microdata from those nodes into a JSON form</span>, and add one item to the
    <span>drag data store item list</span>, with its properties set as follows:</p>

    <dl>
     <dt><span>The drag data item type string</span>
     <dd><code>application/microdata+json</code></dd>

     <dt><span>The drag data item kind</span>
     <dd><i>Text</i></dd>

     <dt>The actual data</dt>
     <dd>The resulting JSON string.</dd>
    </dl>
   </li>

   <li>
    <p>Run the following substeps:</p>

    <ol>
     <li><p>Let <var>urls</var> be « ».</p></li>

     <li>
      <p>For each <var>node</var> in the <span>list of dragged nodes</span>:</p>

      <dl>
       <dt>If the node is an <code>a</code> element with an <code
       data-x="attr-hyperlink-href">href</code> attribute</dt>

       <dd>Add to <var>urls</var> the result of <span>encoding-parsing-and-serializing a URL</span>
       given the element's <code data-x="attr-hyperlink-href">href</code> content attribute's value,
       relative to the element's <span>node document</span>.</dd>

       <dt>If the node is an <code>img</code> element with a <code data-x="attr-img-src">src</code>
       attribute</dt>

       <dd>Add to <var>urls</var> the result of <span>encoding-parsing-and-serializing a URL</span>
       given the element's <code data-x="attr-img-src">src</code> content attribute's value, relative to
       the element's <span>node document</span>.</dd>
      </dl>
     </li>

     <li><p>If <var>urls</var> is still empty, then return.</p></li>

     <li><p>Let <var>url string</var> be the result of concatenating the strings in <var>urls</var>,
     in the order they were added, separated by a U+000D CARRIAGE RETURN U+000A LINE FEED character
     pair (CRLF).</p></li>

     <li><p>Add one item to the <span>drag data store item list</span>, with its properties set as
     follows:</p>

      <dl>
       <dt><span>The drag data item type string</span>
       <dd><code>text/uri-list</code></dd>

       <dt><span>The drag data item kind</span>
       <dd><i>Text</i></dd>

       <dt>The actual data</dt>
       <dd><var>url string</var></dd>
      </dl>
     </li>
    </ol>
   </li>

   <li>
    <p>Update the <span>drag data store default feedback</span> as appropriate for the user agent
    (if the user is dragging the selection, then the selection would likely be the basis for this
    feedback; if the user is dragging an element, then that element's rendering would be used; if
    the drag began outside the user agent, then the platform conventions for determining the drag
    feedback should be used).</p>
   </li>

   <li>
    <p><span>Fire a DND event</span> named <code data-x="event-dnd-dragstart">dragstart</code> at the
    <span>source node</span>.</p>

    <p>If the event is canceled, then the drag-and-drop operation should not occur; return.</p>
    <!-- only a "should" because the UA can always allow the user to drag without the page knowing -->

    <p class="note">Since events with no event listeners registered are, almost by definition, never
    canceled, drag-and-drop is always available to the user if the author does not specifically
    prevent it.</p>
   </li>

   <li><p><span>Fire a pointer event</span> at the <span>source node</span> named <code
   data-x="event-pointercancel">pointercancel</code>, and fire any other follow-up events as
   required by <cite>Pointer Events</cite>. <ref>POINTEREVENTS</ref></p></li>

   <li>
    <p><span>Initiate the drag-and-drop operation</span> in a manner consistent with platform
    conventions, and as described below.</p>

    <p id="base-dnd-feedback">The drag-and-drop feedback must be generated from the first of the
    following sources that is available:</p>

    <ol>
     <li>The <span>drag data store bitmap</span>, if any. In this case, the <span>drag data store
     hot spot coordinate</span> should be used as hints for where to put the cursor relative to the
     resulting image. The values are expressed as distances in <span data-x="'px'">CSS pixels</span>
     from the left side and from the top side of the image respectively. <ref>CSS</ref></li>

     <li>The <span>drag data store default feedback</span>.</li>
    </ol>
   </li>
  </ol>

  <p>From the moment that the user agent is to <dfn>initiate the drag-and-drop operation</dfn>,
  until the end of the drag-and-drop operation, device input events (e.g. mouse and keyboard events)
  must be suppressed.</p>

  <p>During the drag operation, the element directly indicated by the user as the drop target is
  called the <dfn>immediate user selection</dfn>. (Only elements can be selected by the user; other
  nodes must not be made available as drop targets.) However, the <span>immediate user
  selection</span> is not necessarily the <dfn>current target element</dfn>, which is the element
  currently selected for the drop part of the drag-and-drop operation.</p>

  <p>The <span>immediate user selection</span> changes as the user selects different elements
  (either by pointing at them with a pointing device, or by selecting them in some other way). The
  <span>current target element</span> changes when the <span>immediate user selection</span>
  changes, based on the results of event listeners in the document, as described below.</p>

  <p>Both the <span>current target element</span> and the <span>immediate user selection</span> can
  be null, which means no target element is selected. They can also both be elements in other
  (DOM-based) documents, or other (non-web) programs altogether. (For example, a user could drag
  text to a word-processor.) The <span>current target element</span> is initially null.</p>

  <p>In addition, there is also a <dfn>current drag operation</dfn>, which can take on the values
  "<dfn><code data-x="concept-current-drag-operation-none">none</code></dfn>", "<dfn><code
  data-x="concept-current-drag-operation-copy">copy</code></dfn>", "<dfn><code
  data-x="concept-current-drag-operation-link">link</code></dfn>", and "<dfn><code
  data-x="concept-current-drag-operation-move">move</code></dfn>". Initially, it has the value
  "<code data-x="concept-current-drag-operation-none">none</code>". It is updated by the user agent
  as described in the steps below.</p>

  <p>User agents must, as soon as the drag operation is <span data-x="initiate the drag-and-drop
  operation">initiated</span> and every 350ms (&#xB1;200ms) thereafter for as long as the drag
  operation is ongoing, <span>queue a task</span> to perform the following steps in sequence:</p>

  <ol>
   <li>
    <p>If the user agent is still performing the previous iteration of the sequence (if any) when
    the next iteration becomes due, return for this iteration (effectively "skipping
    missed frames" of the drag-and-drop operation).</p>
   </li>

   <li>
    <p><span>Fire a DND event</span> named <code data-x="event-dnd-drag">drag</code> at the
    <span>source node</span>. If this event is canceled, the user agent must set the <span>current
    drag operation</span> to "<code data-x="concept-current-drag-operation-none">none</code>" (no
    drag operation).</p>
   </li>

   <li>
    <p>If the <code data-x="event-dnd-drag">drag</code> event was not canceled and the user has not
    ended the drag-and-drop operation, check the state of the drag-and-drop operation, as
    follows:</p>

    <ol>
     <li>
      <p>If the user is indicating a different <span>immediate user selection</span> than during the
      last iteration (or if this is the first iteration), and if this <span>immediate user
      selection</span> is not the same as the <span>current target element</span>, then update the
      <span>current target element</span> as follows:</p>

      <dl class="switch">

       <dt>If the new <span>immediate user selection</span> is null</dt>

       <dd><p>Set the <span>current target element</span> to null also.</p></dd>

       <dt>If the new <span>immediate user selection</span> is in a non-DOM document or
       application</dt>

       <dd><p>Set the <span>current target element</span> to the <span>immediate user
       selection</span>.</p></dd>

       <dt>Otherwise</dt>

       <dd>
        <p><span>Fire a DND event</span> named <code data-x="event-dnd-dragenter">dragenter</code>
        at the <span>immediate user selection</span>.</p>

        <p>If the event is canceled, then set the <span>current target element</span> to the
        <span>immediate user selection</span>.</p>

        <p>Otherwise, run the appropriate step from the following list:</p>

        <dl class="switch">

         <dt>If the <span>immediate user selection</span> is a text control (e.g.,
         <code>textarea</code>, or an <code>input</code> element whose <code
         data-x="attr-input-type">type</code> attribute is in the <span
         data-x="attr-input-type-text">Text</span> state) or an <span>editing host</span> or
         <span>editable</span> element, and the <span>drag data store item list</span> has an item
         with <span>the drag data item type string</span> "<code>text/plain</code>" and <span>the
         drag data item kind</span> <i>text</i></dt>

         <dd><p>Set the <span>current target element</span> to the <span>immediate user
         selection</span> anyway.</p></dd>

         <dt>If the <span>immediate user selection</span> is <span>the body element</span></dt>

         <dd><p>Leave the <span>current target element</span> unchanged.</p></dd>

         <dt>Otherwise</dt>

         <dd>
          <p><span>Fire a DND event</span> named <code data-x="event-dnd-dragenter">dragenter</code>
          at <span>the body element</span>, if there is one, or at the <code>Document</code> object,
          if not. Then, set the <span>current target element</span> to <span>the body
          element</span>, regardless of whether that event was canceled or not.</p>
         </dd>
        </dl>
       </dd>
      </dl>
     </li>

     <li>
      <p>If the previous step caused the <span>current target element</span> to change, and if the
      previous target element was not null or a part of a non-DOM document, then <span>fire a DND
      event</span> named <code data-x="event-dnd-dragleave">dragleave</code> at the previous target
      element, with the new <span>current target element</span> as the specific <var>related
      target</var>.</p>
     </li>

     <li>
      <p>If the <span>current target element</span> is a DOM element, then <span>fire a DND
      event</span> named <code data-x="event-dnd-dragover">dragover</code> at this <span>current
      target element</span>.</p>

      <p>If the <code data-x="event-dnd-dragover">dragover</code> event is not canceled, run the
      appropriate step from the following list:</p>

      <dl class="switch">

       <dt>If the <span>current target element</span> is a text control (e.g.,
       <code>textarea</code>, or an <code>input</code> element whose <code
       data-x="attr-input-type">type</code> attribute is in the <span
       data-x="attr-input-type-text">Text</span> state) or an <span>editing host</span> or
       <span>editable</span> element, and the <span>drag data store item list</span> has an item
       with <span>the drag data item type string</span> "<code>text/plain</code>" and <span>the drag
       data item kind</span> <i>text</i></dt>

       <dd><p>Set the <span>current drag operation</span> to either "<code
       data-x="concept-current-drag-operation-copy">copy</code>" or "<code
       data-x="concept-current-drag-operation-move">move</code>", as appropriate given the platform
       conventions.</p></dd>


       <dt>Otherwise</dt>

       <dd><p>Reset the <span>current drag operation</span> to "<code
       data-x="concept-current-drag-operation-none">none</code>".</p></dd>
      </dl>

      <p>Otherwise (if the <code data-x="event-dnd-dragover">dragover</code> event <em>is</em>
      canceled), set the <span>current drag operation</span> based on the values of the <code
      data-x="dom-DataTransfer-effectAllowed">effectAllowed</code> and <code
      data-x="dom-DataTransfer-dropEffect">dropEffect</code> attributes of the
      <code>DragEvent</code> object's <code data-x="dom-DragEvent-dataTransfer">dataTransfer</code>
      object as they stood after the event <span data-x="concept-event-dispatch">dispatch</span>
      finished, as per the following table:</p>

      <table>
       <thead>
        <tr>
         <th><code data-x="dom-DataTransfer-effectAllowed">effectAllowed</code></th>
         <th><code data-x="dom-DataTransfer-dropEffect">dropEffect</code></th>
         <th>Drag operation</th>
        </tr>
       </thead>
       <tr>
        <td>"<code data-x="dom-DataTransfer-effectAllowed-uninitialized">uninitialized</code>", "<code data-x="dom-DataTransfer-effectAllowed-copy">copy</code>", "<code data-x="dom-DataTransfer-effectAllowed-copyLink">copyLink</code>", "<code data-x="dom-DataTransfer-effectAllowed-copyMove">copyMove</code>", or "<code data-x="dom-DataTransfer-effectAllowed-all">all</code>"</td>
        <td>"<code data-x="dom-DataTransfer-dropEffect-copy">copy</code>"</td>
        <td>"<code data-x="concept-current-drag-operation-copy">copy</code>"</td>
       </tr>
       <tr>
        <td>"<code data-x="dom-DataTransfer-effectAllowed-uninitialized">uninitialized</code>", "<code data-x="dom-DataTransfer-effectAllowed-link">link</code>", "<code data-x="dom-DataTransfer-effectAllowed-copyLink">copyLink</code>", "<code data-x="dom-DataTransfer-effectAllowed-linkMove">linkMove</code>", or "<code data-x="dom-DataTransfer-effectAllowed-all">all</code>"</td>
        <td>"<code data-x="dom-DataTransfer-dropEffect-link">link</code>"</td>
        <td>"<code data-x="concept-current-drag-operation-link">link</code>"</td>
       </tr>
       <tr>
        <td>"<code data-x="dom-DataTransfer-effectAllowed-uninitialized">uninitialized</code>", "<code data-x="dom-DataTransfer-effectAllowed-move">move</code>", "<code data-x="dom-DataTransfer-effectAllowed-copyMove">copyMove</code>", "<code data-x="dom-DataTransfer-effectAllowed-linkMove">linkMove</code>", or "<code data-x="dom-DataTransfer-effectAllowed-all">all</code>"</td>
        <td>"<code data-x="dom-DataTransfer-dropEffect-move">move</code>"</td>
        <td>"<code data-x="concept-current-drag-operation-move">move</code>"</td>
       </tr>
       <tr>
        <td colspan="2">Any other case</td>
        <td>"<code data-x="concept-current-drag-operation-none">none</code>"</td>
       </tr>
      </table>
     </li>

     <li>
      <p>Otherwise, if the <span>current target element</span> is not a DOM element, use
      platform-specific mechanisms to determine what drag operation is being performed (none, copy,
      link, or move), and set the <span>current drag operation</span> accordingly.</p>
     </li>

     <li>
      <p>Update the drag feedback (e.g. the mouse cursor) to match the <span>current drag
      operation</span>, as follows:</p>

      <table>
       <thead>
        <tr>
         <th>Drag operation</th>
         <th>Feedback</th>
        </tr>
       </thead>
       <tr>
        <td>"<code data-x="concept-current-drag-operation-copy">copy</code>"</td>
        <td>Data will be copied if dropped here.</td>
       </tr>
       <tr>
        <td>"<code data-x="concept-current-drag-operation-link">link</code>"</td>
        <td>Data will be linked if dropped here.</td>
       </tr>
       <tr>
        <td>"<code data-x="concept-current-drag-operation-move">move</code>"</td>
        <td>Data will be moved if dropped here.</td>
       </tr>
       <tr>
        <td>"<code data-x="concept-current-drag-operation-none">none</code>"</td>
        <td>No operation allowed, dropping here will cancel the drag-and-drop operation.</td>
       </tr>
      </table>
     </li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if the user ended the drag-and-drop operation (e.g. by releasing the mouse button
    in a mouse-driven drag-and-drop interface), or if the <code data-x="event-dnd-drag">drag</code>
    event was canceled, then this will be the last iteration. Run the following steps, then stop the
    drag-and-drop operation:</p>

    <ol>
     <li>
      <p>If the <span>current drag operation</span> is "<code
      data-x="concept-current-drag-operation-none">none</code>" (no drag operation), or, if the user
      ended the drag-and-drop operation by canceling it (e.g. by hitting the <kbd><kbd>Escape</kbd></kbd> key),
      or if the <span>current target element</span> is null, then the drag operation failed. Run
      these substeps:</p>

      <ol>
       <li><p>Let <var>dropped</var> be false.</p></li>

       <li><p>If the <span>current target element</span> is a DOM element, <span>fire a DND
       event</span> named <code data-x="event-dnd-dragleave">dragleave</code> at it; otherwise, if
       it is not null, use platform-specific conventions for drag cancelation.</p></li>

       <li><p>Set the <span>current drag operation</span> to "<code
       data-x="concept-current-drag-operation-none">none</code>".</p></li>
      </ol>

      <p>Otherwise, the drag operation might be a success; run these substeps:</p>

      <ol>
       <li><p>Let <var>dropped</var> be true.</p></li>

       <li><p>If the <span>current target element</span> is a DOM element, <span>fire a DND
       event</span> named <code data-x="event-dnd-drop">drop</code> at it; otherwise, use
       platform-specific conventions for indicating a drop.</p></li>

       <li>
        <p>If the event is canceled, set the <span>current drag operation</span> to the value of the
        <code data-x="dom-DataTransfer-dropEffect">dropEffect</code> attribute of the
        <code>DragEvent</code> object's <code data-x="dom-DragEvent-dataTransfer">dataTransfer</code>
        object as it stood after the event <span data-x="concept-event-dispatch">dispatch</span>
        finished.</p>

        <p>Otherwise, the event is not canceled; perform the event's default action, which depends
        on the exact target as follows:</p>

        <dl class="switch">

         <dt>If the <span>current target element</span> is a text control (e.g.,
         <code>textarea</code>, or an <code>input</code> element whose <code
         data-x="attr-input-type">type</code> attribute is in the <span
         data-x="attr-input-type-text">Text</span> state) or an <span>editing host</span> or
         <span>editable</span> element, and the <span>drag data store item list</span> has an item
         with <span>the drag data item type string</span> "<code>text/plain</code>" and <span>the
         drag data item kind</span> <i>text</i></dt>

         <dd><p>Insert the actual data of the first item in the <span>drag data store item
         list</span> to have <span data-x="the drag data item type string">a drag data item type
         string</span> of "<code>text/plain</code>" and <span data-x="the drag data item kind">a drag
         data item kind</span> that is <i>text</i> into the text control or
         <span>editing host</span> or <span>editable</span> element in a manner consistent with
         platform-specific conventions (e.g. inserting it at the current mouse cursor position, or
         inserting it at the end of the field).</p></dd>

         <dt>Otherwise</dt>

         <dd><p>Reset the <span>current drag operation</span> to "<code
         data-x="concept-current-drag-operation-none">none</code>".</p></dd>
        </dl>
       </li>
      </ol>
     </li>

     <li>
      <p><span>Fire a DND event</span> named <code data-x="event-dnd-dragend">dragend</code> at the
      <span>source node</span>.</p>
     </li>

     <li>
      <p>Run the appropriate steps from the following list as the default action of the <code
      data-x="event-dnd-dragend">dragend</code> event:</p>

      <dl class="switch">

       <dt>If <var>dropped</var> is true, the <span>current target element</span> is a <i>text
       control</i> (see below), the <span>current drag operation</span> is "<code
       data-x="concept-current-drag-operation-move">move</code>", and the source of the
       drag-and-drop operation is a selection in the DOM that is entirely contained within an
       <span>editing host</span></dt>

       <dd><p><span>Delete the selection</span>.</p></dd>

       <dt>If <var>dropped</var> is true, the <span>current target element</span> is a <i>text
       control</i> (see below), the <span>current drag operation</span> is "<code
       data-x="concept-current-drag-operation-move">move</code>", and the source of the
       drag-and-drop operation is a selection in a text control</dt>

       <dd><p>The user agent should delete the dragged selection from the relevant text
       control.</p></dd>

       <dt>If <var>dropped</var> is false or if the <span>current drag operation</span> is "<code
       data-x="concept-current-drag-operation-none">none</code>"</dt>

       <dd><p>The drag was canceled. If the platform conventions dictate that this be represented to
       the user (e.g. by animating the dragged selection going back to the source of the
       drag-and-drop operation), then do so.</p></dd>

       <dt>Otherwise</dt>

       <dd><p>The event has no default action.</p></dd>
      </dl>

      <p>For the purposes of this step, a <i>text control</i> is a <code>textarea</code> element or
      an <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
      one of the
      <span data-x="attr-input-type-text">Text</span>,
      <span data-x="attr-input-type-search">Search</span>,
      <span data-x="attr-input-type-tel">Tel</span>,
      <span data-x="attr-input-type-url">URL</span>,
      <span data-x="attr-input-type-email">Email</span>,
      <span data-x="attr-input-type-password">Password</span>, or
      <span data-x="attr-input-type-number">Number</span>
      states.</p>
     </li>
    </ol>
   </li>
  </ol>

  <p class="note">User agents are encouraged to consider how to react to drags near the edge of
  scrollable regions. For example, if a user drags a link to the bottom of the <span>viewport</span>
  on a long page, it might make sense to scroll the page so that the user can drop the link lower on
  the page.</p>

  <p class="note">This model is independent of which <code>Document</code> object the nodes involved
  are from; the events are fired as described above and the rest of the processing model runs as
  described above, irrespective of how many documents are involved in the operation.</p>

  </div>


  <h4 id="dndevents">Events summary</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>The following events are involved in the drag-and-drop
  model.</p>

  <table>

   <thead>
    <tr>
     <th> Event name </th>
     <th> Target </th>
     <th> Cancelable? </th>
     <th> <span>Drag data store mode</span> </th>
     <th> <code data-x="dom-DataTransfer-dropEffect">dropEffect</code> </th>
     <th> Default Action </th>
    </tr>
   </thead>

   <tbody>

    <tr>
     <td><dfn event for="GlobalEventHandlers,Text"><code data-x="event-dnd-dragstart">dragstart</code></dfn></td>
     <td><span>Source node</span></td>
     <td>&#x2713; Cancelable</td>
     <td><span data-x="concept-dnd-rw">Read/write mode</span>
     <td>"<code data-x="dom-DataTransfer-dropEffect-none">none</code>"</td>
     <td>Initiate the drag-and-drop operation</td>
    </tr>

    <tr>
     <td><dfn event for="GlobalEventHandlers,Text"><code data-x="event-dnd-drag">drag</code></dfn></td>
     <td><span>Source node</span></td>
     <td>&#x2713; Cancelable</td>
     <td><span data-x="concept-dnd-p">Protected mode</span>
     <td>"<code data-x="dom-DataTransfer-dropEffect-none">none</code>"</td>
     <td>Continue the drag-and-drop operation</td>
    </tr>

    <tr>
     <td><dfn event for="GlobalEventHandlers"><code data-x="event-dnd-dragenter">dragenter</code></dfn></td>
     <td><span>Immediate user selection</span> or <span>the body element</span></td>
     <td>&#x2713; Cancelable</td>
     <td><span data-x="concept-dnd-p">Protected mode</span>
     <td><!--en-GB--><a href="#dropEffect-initialisation">Based on <code>effectAllowed</code> value</a></td>
     <td>Reject <span>immediate user selection</span> as potential <span data-x="current target element">target element</span></td>
    </tr>

    <tr>
     <td><dfn event for="GlobalEventHandlers"><code data-x="event-dnd-dragleave">dragleave</code></dfn></td>
     <td><span data-x="current target element">Previous target element</span></td>
     <td>&mdash;</td>
     <td><span data-x="concept-dnd-p">Protected mode</span>
     <td>"<code data-x="dom-DataTransfer-dropEffect-none">none</code>"</td>
     <td>None</td>
    </tr>

    <tr>
     <td><dfn event for="GlobalEventHandlers"><code data-x="event-dnd-dragover">dragover</code></dfn></td>
     <td><span>Current target element</span></td>
     <td>&#x2713; Cancelable</td>
     <td><span data-x="concept-dnd-p">Protected mode</span>
     <td><!--en-GB--><a href="#dropEffect-initialisation">Based on <code>effectAllowed</code> value</a></td>
     <td>Reset the <span>current drag operation</span> to "none"</td>
    </tr>

    <tr>
     <td><dfn event for="GlobalEventHandlers"><code data-x="event-dnd-drop">drop</code></dfn></td>
     <td><span>Current target element</span></td>
     <td>&#x2713; Cancelable</td>
     <td><span data-x="concept-dnd-ro">Read-only mode</span>
     <td><span>Current drag operation</span></td>
     <td>Varies</td>
    </tr>

    <tr>
     <td><dfn event for="GlobalEventHandlers,Text"><code data-x="event-dnd-dragend">dragend</code></dfn></td>
     <td><span>Source node</span></td>
     <td>&mdash;</td>
     <td><span data-x="concept-dnd-p">Protected mode</span>
     <!-- <td>Same as last event</td> -->
     <td><span>Current drag operation</span></td>
     <td>Varies</td>
    </tr>

   </tbody>

  </table>

  <p>All of these events bubble, are composed, and the <code
  data-x="dom-DataTransfer-effectAllowed">effectAllowed</code> attribute always has the value it had
  after the <code data-x="event-dnd-dragstart">dragstart</code> event, defaulting to "<code
  data-x="dom-DataTransfer-effectAllowed-uninitialized">uninitialized</code>" in the <code
  data-x="event-dnd-dragstart">dragstart</code> event.</p>



  <h4>The <code data-x="attr-draggable">draggable</code> attribute</h4>

  <p>All <span>HTML elements</span> may have the <dfn element-attr for="html-global"><code
  data-x="attr-draggable">draggable</code></dfn> content attribute set. The <code
  data-x="attr-draggable">draggable</code> attribute is an <span>enumerated attribute</span> with
  the following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="html-global/draggable"><code
     data-x="attr-draggable-true">true</code></dfn>
     <td><dfn data-x="attr-draggable-true-state">true</dfn>
     <td>The element will be draggable.
    <tr>
     <td><dfn attr-value for="html-global/draggable"><code
     data-x="attr-draggable-false">false</code></dfn>
     <td><dfn data-x="attr-draggable-false-state">false</dfn>
     <td>The element will not be draggable.
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <dfn
  data-x="attr-draggable-auto-state">auto</dfn> state. The auto state uses the default behavior of
  the user agent.</p>

  <p>An element with a <code data-x="attr-draggable">draggable</code> attribute should also have a
  <code data-x="attr-title">title</code> attribute that names the element for the purpose of
  non-visual interactions. <!-- "should", not "must", only because this is a relatively new
  attribute and its design implications are not entirely obvious yet. For example, what happens if
  you use an element with text as a drag source? Is that sufficiently clear for ATs? Indeed,
  shouldn't the element generally be distinguishable anyway for it to be useful to drag? --></p>

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-draggable">draggable</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns true if the element is draggable; otherwise, returns false.</p>

    <p>Can be set, to override the default and set the <code
    data-x="attr-draggable">draggable</code> content attribute.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HTMLElement"><code data-x="dom-draggable">draggable</code></dfn> IDL
  attribute, whose value depends on the content attribute's in the way described below, controls
  whether or not the element is draggable. Generally, only text selections are draggable, but
  elements whose <code data-x="dom-draggable">draggable</code> IDL attribute is true become
  draggable as well.</p>

  <p>If an element's <code data-x="attr-draggable">draggable</code> content attribute has the state
  <span data-x="attr-draggable-true-state">true</span>, the <code
  data-x="dom-draggable">draggable</code> IDL attribute must return true.</p>

  <p>Otherwise, if the element's <code data-x="attr-draggable">draggable</code> content attribute
  has the state <span data-x="attr-draggable-false-state">false</span>, the <code
  data-x="dom-draggable">draggable</code> IDL attribute must return false.</p>

  <p>Otherwise, the element's <code data-x="attr-draggable">draggable</code> content attribute has
  the state <span data-x="attr-draggable-auto-state">auto</span>. If the element is an
  <code>img</code> element, an <code>object</code> element that <span>represents</span> an image, or
  an <code>a</code> element with an <code data-x="attr-hyperlink-href">href</code> content
  attribute, the <code data-x="dom-draggable">draggable</code> IDL attribute must return true;
  otherwise, the <code data-x="dom-draggable">draggable</code> IDL attribute must return false.</p>

  <p>If the <code data-x="dom-draggable">draggable</code> IDL attribute is set to the value false,
  the <code data-x="attr-draggable">draggable</code> content attribute must be set to the literal
  value "<code data-x="">false</code>". If the <code data-x="dom-draggable">draggable</code> IDL
  attribute is set to the value true, the <code data-x="attr-draggable">draggable</code> content
  attribute must be set to the literal value "<code data-x="">true</code>".</p>

  </div>



  <div w-nodev>

  <h4>Security risks in the drag-and-drop model</h4>

  <p>User agents must not make the data added to the <code>DataTransfer</code> object during the
  <code data-x="event-dnd-dragstart">dragstart</code> event available to scripts until the <code
  data-x="event-dnd-drop">drop</code> event, because otherwise, if a user were to drag sensitive
  information from one document to a second document, crossing a hostile third document in the
  process, the hostile document could intercept the data.</p>

  <p>For the same reason, user agents must consider a drop to be successful only if the user
  specifically ended the drag operation &mdash; if any scripts end the drag operation, it must be
  considered unsuccessful (canceled) and the <code data-x="event-dnd-drop">drop</code> event must not be
  fired.</p>

  <p>User agents should take care to not start drag-and-drop operations in response to script
  actions. For example, in a mouse-and-window environment, if a script moves a window while the user
  has their mouse button depressed, the UA would not consider that to start a drag. This is important
  because otherwise UAs could cause data to be dragged from sensitive sources and dropped into
  hostile documents without the user's consent.</p>

  <p>User agents should filter potentially active (scripted) content (e.g. HTML) when it is dragged
  and when it is dropped, using a safelist of known-safe features. Similarly, <span data-x="relative
  URL">relative URLs</span> should be turned into absolute URLs to avoid references changing in
  unexpected ways. This specification does not specify how this is performed.</p>

  <div class="example">

   <p>Consider a hostile page providing some content and getting the user to select and drag and
   drop (or indeed, copy and paste) that content to a victim page's <code
   data-x="attr-contenteditable">contenteditable</code> region. If the browser does not ensure that
   only safe content is dragged, potentially unsafe content such as scripts and event handlers in
   the selection, once dropped (or pasted) into the victim site, get the privileges of the victim
   site. This would thus enable a cross-site scripting attack.</p>

  </div>

  </div>

  <h3 split-filename="popover">The <code data-x="attr-popover">popover</code> attribute</h3>

  <p>All <span>HTML elements</span> may have the <dfn element-attr for="html-global"><code
  data-x="attr-popover">popover</code></dfn> content attribute set. When specified, the element
  won't be rendered until it becomes shown, at which point it will be rendered on top of other page
  content.</p>

  <div class="example">
   <p>The <code data-x="attr-popover">popover</code> attribute is a global attribute that allows
   authors flexibility to ensure popover functionality can be applied to elements with the most
   relevant semantics.</p>

   <p>The following demonstrates how one might create a popover sub-navigation list of
   links, within the global navigation for a website.</p>

   <pre><code class="html">&lt;ul>
  &lt;li>
    &lt;a href=...>All Products&lt;/a>
    &lt;button popovertarget=sub-nav>
     &lt;img src=down-arrow.png alt="Product pages">
    &lt;/button>
    &lt;ul popover id=sub-nav>
     &lt;li>&lt;a href=...>Shirts&lt;/a>
     &lt;li>&lt;a href=...>Shoes&lt;/a>
     &lt;li>&lt;a href=...>Hats etc.&lt;/a>
    &lt;/ul>
  &lt;/li>
  &lt;!-- other list items and links here -->
&lt;/ul></code></pre>
  </div>

  <p>When using <code data-x="attr-popover">popover</code> on elements without accessibility
  semantics, for instance the <code>div</code> element, authors should use the appropriate ARIA
  attributes to ensure the popover is accessible.</p>

  <div class="example">
   <p>The following shows the baseline markup to create a custom menu popover, where the first
   menuitem will receive keyboard focus when the popover is invoked due to the use of the
   <code data-x="">autofocus</code> attribute. Navigating the menuitems with arrow keys and
   activation behaviors would still need author scripting. Additional requirements for building
   custom menus widgets are defined in the <a href="https://w3c.github.io/aria/#menu">WAI-ARIA
   specification</a>.</p>

   <pre><code class="html">&lt;button popovertarget=m>Actions&lt;/button>
&lt;div role=menu id=m popover>
  &lt;button role=menuitem tabindex=-1 autofocus>Edit&lt;/button>
  &lt;button role=menuitem tabindex=-1>Hide&lt;/button>
  &lt;button role=menuitem tabindex=-1>Delete&lt;/button>
&lt;/div></code></pre>

   <p>A popover can be useful for rendering a status message, confirming the action performed by
   the user. The following demonstrates how one could reveal a popover in an <code>output</code>
   element.</p>

   <pre><code class="html">&lt;button id=submit>Submit&lt;/button>
&lt;p>&lt;output>&lt;span popover=manual>&lt;/span>&lt;/output>&lt;/p>

&lt;script>
 const sBtn = document.getElementById("submit");
 const outSpan = document.querySelector("output [popover=manual]");
 let successMessage;
 let errorMessage;

 /* define logic for determining success of action
  and determining the appropriate success or error
  messages to use */

 sBtn.addEventListener("click", ()=> {
  if ( success ) {
   outSpan.textContent = successMessage;
  }
  else {
   outSpan.textContent = errorMessage;
  }
  outSpan.showPopover();

  setTimeout(function () {
   outSpan.hidePopover();
  }, 10000);
 });
&lt;/script></code></pre>
  </div>

  <p class="note">Inserting a popover element into an <code>output</code> element will generally
  cause screen readers to announce the content when it becomes visible. Depending on the complexity
  or frequency of the content, this could be either useful or annoying to users of these
  assistive technologies. Keep this in mind when using the <code>output</code> element or other
  ARIA live regions to ensure the best user experience.</p>

  <p>The <code data-x="attr-popover">popover</code> attribute is an <span>enumerated
  attribute</span> with the following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="html-global/popover"><code
     data-x="attr-popover-auto">auto</code></dfn>
     <td rowspan=2><dfn data-x="attr-popover-auto-state">auto</dfn>
     <td rowspan=2>Closes other popovers when opened; has <span data-x="popover light dismiss">light
     dismiss</span> and responds to <span data-x="close request">close requests</span>.
    <tr>
     <td>(the empty string)
    <tr>
     <td><dfn attr-value for="html-global/popover"><code
     data-x="attr-popover-manual">manual</code></dfn>
     <td><dfn data-x="attr-popover-manual-state">manual</dfn>
     <td>Does not close other popovers; does not <span data-x="popover light dismiss">light
     dismiss</span> or respond to <span data-x="close request">close requests</span>.
    </tr>
    <tr>
     <td><dfn attr-value for="html-global/popover"><code
     data-x="attr-popover-hint">hint</code></dfn></td>
     <td><dfn data-x="attr-popover-hint-state">hint</dfn></td>
     <td>Closes other hint popovers when opened, but not other auto popovers; has <span
     data-x="popover light dismiss">light dismiss</span> and responds to <span data-x="close
     request">close requests</span>.</td>
    </tr>
   </tbody>
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> is the <dfn
  data-x="attr-popover-none-state">no popover</dfn> state, and its <i data-x="invalid value
  default">invalid value default</i> is the <span data-x="attr-popover-manual-state">manual</span>
  state.</p>

  <p>The <dfn attribute for="HTMLElement"><code data-x="dom-popover">popover</code></dfn> IDL
  attribute must <span>reflect</span> the <span data-x="attr-popover">popover</span> attribute,
  <span>limited to only known values</span>.</p>

  <p>Every <span data-x="HTML elements">HTML element</span> has a <dfn export>popover visibility
  state</dfn>, initially <span data-x="popover-hidden-state">hidden</span>, with these potential
  values:</p>

  <ul>
   <li><p><dfn export for="popover visibility state"
   data-x="popover-hidden-state">hidden</dfn></p></li>

   <li><p><dfn export for="popover visibility state"
   data-x="popover-showing-state">showing</dfn></p></li>
  </ul>

  <p>Every <code>Document</code> has a <dfn>popover pointerdown target</dfn>, which is an <span
  data-x="HTML elements">HTML element</span> or null, initially null.</p>

  <p>Every <span data-x="HTML elements">HTML element</span> has a <dfn>popover invoker</dfn>, which
  is an <span data-x="HTML elements">HTML element</span> or null, initially set to null.</p>

  <p>Every <span data-x="HTML elements">HTML element</span> has a <dfn>popover showing or
  hiding</dfn>, which is a boolean, initially set to false.</p>

  <p>Every <span data-x="HTML elements">HTML element</span> <dfn>popover toggle task tracker</dfn>,
  which is a <span>toggle task tracker</span> or null, initially null.</p>

  <p>Every <span data-x="HTML elements">HTML element</span> has a <dfn>popover close watcher</dfn>,
  which is a <span>close watcher</span> or null, initially null.</p>

  <p>Every <span data-x="HTML elements">HTML element</span> has an <dfn>opened in popover
  mode</dfn>, which is a string or null, initially null.</p>

  <p>The following <span data-x="concept-element-attributes-change-ext">attribute change
  steps</span>, given <var>element</var>, <var>localName</var>, <var>oldValue</var>,
  <var>value</var>, and <var>namespace</var>, are used for all <span>HTML elements</span>:</p>

  <ol>
   <li><p>If <var>namespace</var> is not null, then return.</p></li>

   <li><p>If <var>localName</var> is not <code data-x="attr-popover">popover</code>, then
   return.</p></li>

   <li><p>If <var>element</var>'s <span>popover visibility state</span> is in the <span
   data-x="popover-showing-state">showing state</span> and <var>oldValue</var> and <var>value</var>
   are in different <span data-x="attr-popover">states</span>, then run the <span>hide popover
   algorithm</span> given <var>element</var>, true, true, and false.</p></li>
  </ol>

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn
   data-x="dom-showPopover">showPopover</span>()</code></dt>

   <dd>Shows the popover <var>element</var> by adding it to the top layer. If <var>element</var>'s
   <code data-x="attr-popover">popover</code> attribute is in the <span
   data-x="attr-popover-auto-state">auto</span> state, then this will also close all other <span
   data-x="attr-popover-auto-state">auto</span> popovers unless they are an ancestor of
   <var>element</var> according to the <span>topmost popover ancestor</span> algorithm.</dd>

   <dt><code data-x=""><var>element</var>.<span subdfn
   data-x="dom-hidePopover">hidePopover</span>()</code></dt>

   <dd>Hides the popover <var>element</var> by removing it from the top layer and applying
   <code data-x="">display: none</code> to it.</dd>

   <dt><code data-x=""><var>element</var>.<span subdfn
   data-x="dom-togglePopover">togglePopover</span>()</code></dt>

   <dd>If the popover <var>element</var> is not showing, then this method shows it. Otherwise, this
   method hides it. This method returns true if the popover is open after calling it, otherwise
   false.</dd>
  </dl>

  <p>The <dfn method for="HTMLElement"><code
  data-x="dom-showPopover">showPopover(<var>options</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>invoker</var> be <var>options</var>["<code
   data-x="dom-ShowPopoverOptions-source">source</code>"] if it <span data-x="map
   exists">exists</span>; otherwise, null.</p></li>

   <li><p>Run <span>show popover</span> given <span>this</span>, true, and
   <var>invoker</var>.</p></li>
  </ol>

  <p>To <dfn>show popover</dfn>, given an <span data-x="HTML elements">HTML element</span>
  <var>element</var>, a boolean <var>throwExceptions</var>, and an <span data-x="HTML elements">HTML
  element</span> or null <var>invoker</var>:</p>

  <ol>
   <li><p>If the result of running <span>check popover validity</span> given <var>element</var>,
   false, <var>throwExceptions</var>, and null is false, then return.</p></li>

   <li><p>Let <var>document</var> be <var>element</var>'s <span>node document</span>.</p></li>

   <li><p><span>Assert</span>: <var>element</var>'s <span>popover invoker</span> is null.</p></li>

   <li><p><span>Assert</span>: <var>element</var> is not in <var>document</var>'s <span>top
   layer</span>.</p></li>

   <li><p>Let <var>nestedShow</var> be <var>element</var>'s <span>popover showing or
   hiding</span>.</p></li>

   <li><p>Let <var>fireEvents</var> be the boolean negation of <var>nestedShow</var>.</p></li>

   <li><p>Set <var>element</var>'s <span>popover showing or hiding</span> to true.</p></li>

   <li>
    <p>Let <var>cleanupShowingFlag</var> be the following steps:</p>

    <ol>
     <li><p>If <var>nestedShow</var> is false, then set <var>element</var>'s <span>popover showing
     or hiding</span> to false.</p></li>
    </ol>
   </li>

   <li><p>If the result of <span data-x="concept-event-fire">firing an event</span> named <code
   data-x="event-beforetoggle">beforetoggle</code>, using <code>ToggleEvent</code>, with the <code
   data-x="dom-Event-cancelable">cancelable</code> attribute initialized to true, the <code
   data-x="dom-ToggleEvent-oldState">oldState</code> attribute initialized to "<code
   data-x="">closed</code>", and the <code data-x="dom-ToggleEvent-newState">newState</code>
   attribute initialized to "<code data-x="">open</code>" at <var>element</var> is false, then
   run <var>cleanupShowingFlag</var> and return.</p></li>

   <li>
    <p>If the result of running <span>check popover validity</span> given <var>element</var>, false,
    <var>throwExceptions</var>, and <var>document</var> is false, then run
    <var>cleanupShowingFlag</var> and return.</p>

    <p class="note"><span>Check popover validity</span> is called again because firing the <code
    data-x="event-beforetoggle">beforetoggle</code> event could have disconnected this element or
    changed its <code data-x="attr-popover">popover</code> attribute.</p>
   </li>

   <li><p>Let <var>shouldRestoreFocus</var> be false.</p></li>

   <li><p>Let <var>originalType</var> be the current state of <var>element</var>'s <code
   data-x="attr-popover">popover</code> attribute.</p></li>

   <li><p>Let <var>stackToAppendTo</var> be null.</p></li>

   <li><p>Let <var>autoAncestor</var> be the result of running the <span>topmost popover
   ancestor</span> algorithm given <var>element</var>, <var>document</var>'s <span>showing auto
   popover list</span>, <var>invoker</var>, and true.</p></li>

   <li><p>Let <var>hintAncestor</var> be the result of running the <span>topmost popover
   ancestor</span> algorithm given <var>element</var>, <var>document</var>'s <span>showing hint
   popover list</span>, <var>invoker</var>, and true.</p></li>

   <li>
    <p>If <var>originalType</var> is the <span data-x="attr-popover-auto-state">auto</span> state,
    then:</p>

    <ol>
     <li><p>Run <span>close entire popover list</span> given <var>document</var>'s <span>showing
     hint popover list</span>, <var>shouldRestoreFocus</var>, and <var>fireEvents</var>.</p></li>

     <li><p>Let <var>ancestor</var> be the result of running the <span>topmost popover
     ancestor</span> algorithm given <var>element</var>, <var>document</var>'s <span>showing auto
     popover list</span>, <var>invoker</var>, and true.</p></li>

     <li><p>If <var>ancestor</var> is null, then set <var>ancestor</var> to
     <var>document</var>.</p></li>

     <li><p>Run <span data-x="hide-all-popovers-until">hide all popovers until</span> given
     <var>ancestor</var>, <var>shouldRestoreFocus</var>, and <var>fireEvents</var>.</p></li>

     <li><p>Set <var>stackToAppendTo</var> to "<code data-x="">auto</code>".</p></li>
    </ol>
   </li>

   <li>
    <p>If <var>originalType</var> is the <span data-x="attr-popover-hint-state">hint</span> state,
    then:</p>

    <ol>
     <li>
      <p>If <var>hintAncestor</var> is not null, then:</p>

      <ol>
       <li><p>Run <span data-x="hide-all-popovers-until">hide all popovers until</span> given
       <var>hintAncestor</var>, <var>shouldRestoreFocus</var>, and <var>fireEvents</var>.</p></li>

       <li><p>Set <var>stackToAppendTo</var> to "<code data-x="">hint</code>".</p></li>
      </ol>
     </li>

     <li>
      <p>Otherwise:</p>

      <ol>
       <li><p>Run <span>close entire popover list</span> given <var>document</var>'s <span>showing
       hint popover list</span>, <var>shouldRestoreFocus</var>, and <var>fireEvents</var>.</p></li>

       <li>
        <p>If <var>autoAncestor</var> is not null, then:</p>

        <ol>
         <li><p>Run <span data-x="hide-all-popovers-until">hide all popovers until</span> given
         <var>autoAncestor</var>, <var>shouldRestoreFocus</var>, and <var>fireEvents</var>.</p></li>

         <li><p>Set <var>stackToAppendTo</var> to "<code data-x="">auto</code>".</p></li>
        </ol>
       </li>

       <li><p>Otherwise, set <var>stackToAppendTo</var> to "<code data-x="">hint</code>".</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li>
    <p>If <var>originalType</var> is <span data-x="attr-popover-auto-state">auto</span> or <span
    data-x="attr-popover-hint-state">hint</span>, then:</p>

    <ol>
     <li><p><span>Assert</span>: <var>stackToAppendTo</var> is not null.</p></li>

     <li>
      <p>If <var>originalType</var> is not equal to the value of <var>element</var>'s <code
      data-x="attr-popover">popover</code> attribute, then:</p>

      <ol>
       <li><p>If <var>throwExceptions</var> is true, then throw a
       <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

       <li><p>Return.</p></li>
      </ol>
     </li>

     <li>
      <p>If the result of running <span>check popover validity</span> given <var>element</var>,
      false, <var>throwExceptions</var>, and <var>document</var> is false, then run
      <var>cleanupShowingFlag</var> and return.</p>

      <p class="note"><span>Check popover validity</span> is called again because running <span
      data-x="hide-all-popovers-until">hide all popovers until</span> above could have fired the
      <code data-x="event-beforetoggle">beforetoggle</code> event, and an event handler could have
      disconnected this element or changed its <code data-x="attr-popover">popover</code>
      attribute.</p>
     </li>

     <li>
      <p>If the result of running <span data-x="topmost auto popover">topmost auto or hint
      popover</span> on <var>document</var> is null, then set <var>shouldRestoreFocus</var> to
      true.</p>

      <p class="note">This ensures that focus is returned to the previously-focused element only for
      the first popover in a stack.</p>
     </li>

     <li>
      <p>If <var>stackToAppendTo</var> is "<code data-x="">auto</code>":</p>

      <ol>
       <li><p><span>Assert</span>: <var>document</var>'s <span>showing auto popover list</span> does
       not contain <var>element</var>.</p></li>

       <li><p>Set <var>element</var>'s <span>opened in popover mode</span> to "<code
       data-x="">auto</code>".</p></li>
      </ol>

      <p>Otherwise:</p>

      <ol>
       <li><p><span>Assert</span>: <var>stackToAppendTo</var> is "<code
       data-x="">hint</code>".</p></li>

       <li><p><span>Assert</span>: <var>document</var>'s <span>showing hint popover list</span> does
       not contain <var>element</var>.</p></li>

       <li><p>Set <var>element</var>'s <span>opened in popover mode</span> to "<code
       data-x="">hint</code>".</p></li>
      </ol>
     </li>

     <li id="canceling-popovers">
      <p>Set <var>element</var>'s <span>popover close watcher</span> to the result of <span
      data-x="establish a close watcher">establishing a close watcher</span> given
      <var>element</var>'s <span>relevant global object</span>, with:</p>

      <ul>
       <li><p><i data-x="create-close-watcher-cancelAction">cancelAction</i> being to return
       true.</p></li>

       <li><p><i data-x="create-close-watcher-closeAction">closeAction</i> being to <span
       data-x="hide popover algorithm">hide a popover</span> given <var>element</var>, true, true,
       and false.</p></li>

       <li><p><i data-x="create-close-watcher-getEnabledState">getEnabledState</i> being to return
       true.</p></li>
      </ul>
     </li>
    </ol>
   </li>

   <li><p>Set <var>element</var>'s <span>previously focused element</span> to null.</p></li>

   <li><p>Let <var>originallyFocusedElement</var> be <var>document</var>'s <span>focused area of the
   document</span>'s <span>DOM anchor</span>.</p></li>

   <li><p><span>Add an element to the top layer</span> given <var>element</var>.</p></li>

   <li><p>Set <var>element</var>'s <span>popover visibility state</span> to <span
   data-x="popover-showing-state">showing</span>.</p></li>

   <li><p>Set <var>element</var>'s <span>popover invoker</span> to <var>invoker</var>.</p></li>

   <li><p>Set <var>element</var>'s <span>implicit anchor element</span> to
   <var>invoker</var>.</p></li>

   <li><p>Run the <span>popover focusing steps</span> given <var>element</var>.</p></li>

   <li><p>If <var>shouldRestoreFocus</var> is true and <var>element</var>'s <code
   data-x="attr-popover">popover</code> attribute is not in the <span
   data-x="attr-popover-none-state">no popover</span> state, then set <var>element</var>'s
   <span>previously focused element</span> to <var>originallyFocusedElement</var>.</p></li>

   <li><p><span>Queue a popover toggle event task</span> given <var>element</var>, "<code
   data-x="">closed</code>", and "<code data-x="">open</code>".</p></li>

   <li><p>Run <var>cleanupShowingFlag</var>.</p></li>
  </ol>

  <p>To <dfn>queue a popover toggle event task</dfn> given an element <var>element</var>, a string
  <var>oldState</var>, and a string <var>newState</var>:

  <ol>
   <li>
    <p>If <var>element</var>'s <span>popover toggle task tracker</span> is not null, then:</p>

    <ol>
     <li><p>Set <var>oldState</var> to <var>element</var>'s <span>popover toggle task
     tracker</span>'s <span data-x="toggle-task-old-state">old state</span>.</p></li>

     <li><p>Remove <var>element</var>'s <span>popover toggle task tracker</span>'s <span
     data-x="toggle-task-task">task</span> from its <span>task queue</span>.</p></li>

     <li><p>Set <var>element</var>'s <span>popover toggle task tracker</span> to null.</p></li>
    </ol>
   </li>

   <li>
    <p><span>Queue an element task</span> given the <span>DOM manipulation task source</span> and
    <var>element</var> to run the following steps:</p>

    <ol>
     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-toggle">toggle</code> at <var>element</var>, using <code>ToggleEvent</code>, with
     the <code data-x="dom-ToggleEvent-oldState">oldState</code> attribute initialized to
     <var>oldState</var> and the <code data-x="dom-ToggleEvent-newState">newState</code> attribute
     initialized to <var>newState</var>.</p></li>

     <li><p>Set <var>element</var>'s <span>popover toggle task tracker</span> to null.</p></li>
    </ol>
   </li>

   <li><p>Set <var>element</var>'s <span>popover toggle task tracker</span> to a struct with <span
   data-x="toggle-task-task">task</span> set to the just-queued <span
   data-x="concept-task">task</span> and <span data-x="toggle-task-old-state">old state</span> set
   to <var>oldState</var>.</p></li>
  </ol>

  <p>The <dfn method for="HTMLElement"><code data-x="dom-hidePopover">hidePopover()</code></dfn>
  method steps are:</p>

  <ol>
   <li><p>Run the <span>hide popover algorithm</span> given <span>this</span>, true, true, and
   true.</p></li>
  </ol>

  <p>To <dfn data-x="hide popover algorithm">hide a popover</dfn> given an <span data-x="HTML
  elements">HTML element</span> <var>element</var>, a boolean <var>focusPreviousElement</var>, a
  boolean <var>fireEvents</var>, and a boolean <var>throwExceptions</var>:</p>

  <ol>
   <li><p>If the result of running <span>check popover validity</span> given <var>element</var>,
   true, <var>throwExceptions</var>, and null is false, then return.</p></li>

   <li><p>Let <var>document</var> be <var>element</var>'s <span>node document</span>.</p></li>

   <li><p>Let <var>nestedHide</var> be <var>element</var>'s <span>popover showing or
   hiding</span>.</p></li>

   <li><p>Set <var>element</var>'s <span>popover showing or hiding</span> to true.</p></li>

   <li><p>If <var>nestedHide</var> is true, then set <var>fireEvents</var> to false.</p></li>

   <li>
    <p>Let <var>cleanupSteps</var> be the following steps:</p>

    <ol>
     <li><p>If <var>nestedHide</var> is false, then set <var>element</var>'s <span>popover showing
     or hiding</span> to false.</p></li>

     <li>
      <p>If <var>element</var>'s <span>popover close watcher</span> is not null, then:</p>

      <ol>
       <li><p><span data-x="close-watcher-destroy">Destroy</span> <var>element</var>'s <span>popover
       close watcher</span>.</p></li>

       <li><p>Set <var>element</var>'s <span>popover close watcher</span> to null.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li>
    <p>If <var>element</var>'s <code data-x="attr-popover">popover</code> attribute is in the <span
    data-x="attr-popover-auto-state">auto</span> state or the <span
    data-x="attr-popover-hint-state">hint</span> state, then:</p>

    <ol>
     <li><p>Run <span data-x="hide-all-popovers-until">hide all popovers until</span> given
     <var>element</var>, <var>focusPreviousElement</var>, and <var>fireEvents</var>.</p></li>

     <li>
      <p>If the result of running <span>check popover validity</span> given <var>element</var>,
      true, and <var>throwExceptions</var> is false, then run <var>cleanupSteps</var> and
      return.</p>

      <p class="note"><span>Check popover validity</span> is called again because running <span
      data-x="hide-all-popovers-until">hide all popovers until</span> could have disconnected
      <var>element</var> or changed its <code data-x="attr-popover">popover</code> attribute.</p>
     </li>
    </ol>
   </li>

   <li><p>Let <var>autoPopoverListContainsElement</var> be true if <var>document</var>'s
   <span>showing auto popover list</span>'s last item is <var>element</var>, otherwise
   false.</p></li>

   <li><p>Set <var>element</var>'s <span>popover invoker</span> to null.</p></li>

   <li>
    <p>If <var>fireEvents</var> is true:</p>

    <ol>
     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-beforetoggle">beforetoggle</code>, using <code>ToggleEvent</code>, with the <code
     data-x="dom-ToggleEvent-oldState">oldState</code> attribute initialized to "<code
     data-x="">open</code>" and the <code data-x="dom-ToggleEvent-newState">newState</code>
     attribute initialized to "<code data-x="">closed</code>" at <var>element</var>.</p></li>

     <li><p>If <var>autoPopoverListContainsElement</var> is true and <var>document</var>'s
     <span>showing auto popover list</span>'s last item is not <var>element</var>, then run <span
     data-x="hide-all-popovers-until">hide all popovers until</span> given <var>element</var>,
     <var>focusPreviousElement</var>, and false.</p></li>

     <li>
      <p>If the result of running <span>check popover validity</span> given <var>element</var>,
      true, <var>throwExceptions</var>, and null is false, then run <var>cleanupSteps</var> and
      return.</p>

      <p class="note"><span>Check popover validity</span> is called again because firing the <code
      data-x="event-beforetoggle">beforetoggle</code> event could have disconnected
      <var>element</var> or changed its <code data-x="attr-popover">popover</code> attribute.</p>
     </li>

     <li><p><span>Request an element to be removed from the top layer</span> given
     <var>element</var>.</p></li>

     <li><p>Set <var>element</var>'s <span>implicit anchor element</span> to null.</p></li>
    </ol>
   </li>

   <li><p>Otherwise, <span>remove an element from the top layer immediately</span> given
   <var>element</var>.</p></li>

   <li><p>Set <var>element</var>'s <span>opened in popover mode</span> to null.</p></li>

   <li><p>Set <var>element</var>'s <span>popover visibility state</span> to <span
   data-x="popover-hidden-state">hidden</span>.</p></li>

   <li><p>If <var>fireEvents</var> is true, then <span>queue a popover toggle event task</span>
   given <var>element</var>, "<code data-x="">open</code>", and "<code
   data-x="">closed</code>".</p></li>

   <li><p>Let <var>previouslyFocusedElement</var> be <var>element</var>'s <span>previously focused
   element</span>.</p></li>

   <li>
    <p>If <var>previouslyFocusedElement</var> is not null, then:</p>

    <ol>
     <li><p>Set <var>element</var>'s <span>previously focused element</span> to null.</p></li>

     <li><p>If <var>focusPreviousElement</var> is true and <var>document</var>'s <span>focused area
     of the document</span>'s <span>DOM anchor</span> is a <span>shadow-including inclusive
     descendant</span> of <var>element</var>, then run the <span>focusing steps</span> for
     <var>previouslyFocusedElement</var>; the viewport should not be scrolled by doing this
     step.</p></li>
    </ol>
   </li>

   <li><p>Run <var>cleanupSteps</var>.</p></li>
  </ol>

  <p>The <dfn method for="HTMLElement"><code
  data-x="dom-togglePopover">togglePopover(<var>options</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>force</var> be null.</p></li>

   <li><p>If <var>options</var> is a boolean, set <var>force</var> to <var>options</var>.</p></li>

   <li><p>Otherwise, if <var>options</var>["<code
   data-x="dom-TogglePopoverOptions-force">force</code>"] <span data-x="map exists">exists</span>,
   set <var>force</var> to <var>options</var>["<code
   data-x="dom-TogglePopoverOptions-force">force</code>"].</p></li>

   <li><p>Let <var>invoker</var> be <var>options</var>["<code
   data-x="dom-ShowPopoverOptions-source">source</code>"] if it <span data-x="map
   exists">exists</span>; otherwise, null.</p></li>

   <li><p>If <span>this</span>'s <span>popover visibility state</span> is <span
   data-x="popover-showing-state">showing</span>, and <var>force</var> is null or false, then run
   the <span>hide popover algorithm</span> given <span>this</span>, true, true, and true.</p></li>

   <li><p>Otherwise, if <var>force</var> is null or true, then run <span>show popover</span> given
   <span>this</span>, true, and <var>invoker</var>.</p></li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p>Let <var>expectedToBeShowing</var> be true if <span>this</span>'s <span>popover
     visibility state</span> is <span data-x="popover-showing-state">showing</span>; otherwise
     false.</p></li>

     <li><p>Run <span>check popover validity</span> given <var>expectedToBeShowing</var>, true, and
     null.</p></li>
    </ol>
   </li>

   <li><p>Return true if <span>this</span>'s <span>popover visibility state</span> is <span
   data-x="popover-showing-state">showing</span>; otherwise false.</p></li>
  </ol>

  <p>To <dfn export data-x="hide-all-popovers-until">hide all popovers until</dfn>, given an <span
  data-x="HTML elements">HTML element</span> or <code>Document</code> <var>endpoint</var>, a boolean
  <var>focusPreviousElement</var>, and a boolean <var>fireEvents</var>:</p>

  <ol>
   <li><p>If <var>endpoint</var> is an <span data-x="HTML elements">HTML element</span> and
   <var>endpoint</var> is not in the <span data-x="popover-showing-state">popover showing
   state</span>, then return.</p></li>

   <li><p>Let <var>document</var> be <var>endpoint</var>'s <span>node document</span>.</p></li>

   <li><p><span>Assert</span>: <var>endpoint</var> is a <code>Document</code> or
   <var>endpoint</var>'s <span>popover visibility state</span> is <span
   data-x="popover-showing-state">showing</span>.</p></li>

   <li><p><span>Assert</span>: <var>endpoint</var> is a <code>Document</code> or
   <var>endpoint</var>'s <code data-x="attr-popover">popover</code> attribute is in the <span
   data-x="attr-popover-auto-state">auto</span> state or <var>endpoint</var>'s <code
   data-x="attr-popover">popover</code> attribute is in the <span
   data-x="attr-popover-hint-state">hint</span> state.</p></li>

   <li>
    <p>If <var>endpoint</var> is a <code>Document</code>:</p>

    <ol>
     <li><p>Run <span>close entire popover list</span> given <var>document</var>'s <span>showing
     hint popover list</span>, <var>focusPreviousElement</var>, and <var>fireEvents</var>.</p></li>

     <li><p>Run <span>close entire popover list</span> given <var>document</var>'s <span>showing
     auto popover list</span>, <var>focusPreviousElement</var>, and <var>fireEvents</var>.</p></li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li>
    <p>If <var>document</var>'s <span>showing hint popover list</span> contains
    <var>endpoint</var>:</p>

    <ol>
     <li><p><span>Assert</span>: <var>endpoint</var>'s <code data-x="attr-popover">popover</code>
     attribute is in the <span data-x="attr-popover-hint-state">hint</span> state.</p></li>

     <li><p>Run <span>hide popover stack until</span> given <var>endpoint</var>,
     <var>document</var>'s <span>showing hint popover list</span>, <var>focusPreviousElement</var>,
     and <var>fireEvents</var>.</p></li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li><p>Run <span>close entire popover list</span> given <var>document</var>'s <span>showing hint
   popover list</span>, <var>focusPreviousElement</var>, and <var>fireEvents</var>.</p></li>

   <li><p>If <var>document</var>'s <span>showing auto popover list</span> does not contain
   <var>endpoint</var>, then return.</p></li>

   <li><p>Run <span>hide popover stack until</span> given <var>endpoint</var>, <var>document</var>'s
   <span>showing auto popover list</span>, <var>focusPreviousElement</var>, and
   <var>fireEvents</var>.</p></li>
  </ol>

  <p>To <dfn>hide popover stack until</dfn>, given an <span data-x="html elements">HTML
  element</span> <var>endpoint</var>, a <span>list</span> <var>popoverList</var>, a boolean
  <var>focusPreviousElement</var>, and a boolean <var>fireEvents</var>:</p>

  <ol>
   <li><p>Let <var>repeatingHide</var> be false.</p></li>

   <li>
    <p>Perform the following steps at least once:</p>

    <ol>
     <li><p>Let <var>lastToHide</var> be null.</p></li>

     <li>
      <p>For each <var>popover</var> in <var>popoverList</var>:</p>

      <ol>
       <li><p>If <var>popover</var> is <var>endpoint</var>, then <span>break</span>.</p></li>

       <li><p>Set <var>lastToHide</var> to <var>popover</var>.</p></li>
      </ol>
     </li>

     <li><p>If <var>lastToHide</var> is null, then return.</p></li>

     <li>
      <p><span>While</span> <var>lastToHide</var>'s <span>popover visibility state</span> is <span
      data-x="popover-showing-state">showing</span>:</p>

      <ol>
       <li><p><span>Assert</span>: <var>popoverList</var> is not empty.</p></li>

       <li><p>Run the <span>hide popover algorithm</span> given the last item in
       <var>popoverList</var>, <var>focusPreviousElement</var>, <var>fireEvents</var>, and
       false.</p></li>
      </ol>
     </li>

     <li><p><span>Assert</span>: <var>repeatingHide</var> is false or <var>popoverList</var>'s last
     item is <var>endpoint</var>.</p></li>

     <li><p>Set <var>repeatingHide</var> to true if <var>popoverList</var> contains
     <var>endpoint</var> and <var>popoverList</var>'s last item is not <var>endpoint</var>,
     otherwise false.</p></li>

     <li><p>If <var>repeatingHide</var> is true, then set <var>fireEvents</var> to false.</p></li>
    </ol>

    <p>and keep performing them <span>while</span> <var>repeatingHide</var> is true.</p>
   </li>
  </ol>

  <p class="note">The <span data-x="hide-all-popovers-until">hide all popovers until
  algorithm</span> is used in several cases to hide all popovers that don't stay open when something
  happens. For example, during light-dismiss of a popover, this algorithm ensures that we close only
  the popovers that aren't related to the node clicked by the user.</p>

  <p>To find the <dfn export>topmost popover ancestor</dfn>, given a <code>Node</code>
  <var>newPopoverOrTopLayerElement</var>, a <span>list</span> <var>popoverList</var>, an <span
  data-x="HTML elements">HTML element</span> or null <var>invoker</var>, and a boolean
  <var>isPopover</var>, perform the following steps. They return an <span data-x="HTML
  elements">HTML element</span> or null.</p>

  <div class="note">
   <p>The <span>topmost popover ancestor</span> algorithm will return the topmost (latest in the
   <span>showing auto popover list</span>) ancestor popover for the provided popover or top layer
   element. Popovers can be related to each other in several ways, creating a tree of popovers.
   There are two paths through which one popover (call it the "child" popover) can have a topmost
   ancestor popover (call it the "parent" popover):</p>

   <ol>
    <li><p>The popovers are nested within each other in the node tree. In this case, the descendant
    popover is the "child" and its topmost ancestor popover is the "parent".</p></li>

    <li><p>An invoking element (e.g., a <code>button</code>) has a <code
    data-x="attr-popovertarget">popovertarget</code> attribute pointing to a popover. In this case,
    the popover is the "child", and the popover subtree the invoking element is in is the
    "parent". The invoker has to be in a popover and reference an open popover.</p></li>
   </ol>

   <p>In each of the relationships formed above, the parent popover has to be strictly earlier in
   the <span>showing auto popover list</span> than the child popover, or it does not form a valid
   ancestral relationship. This eliminates non-showing popovers and self-pointers (e.g., a popover
   containing an invoking element that points back to the containing popover), and it allows for the
   construction of a well-formed tree from the (possibly cyclic) graph of connections. Only <span
   data-x="attr-popover-auto-state">auto</span> popovers are considered.</p>

   <p>If the provided element is a top layer element such as a <code>dialog</code> which is not
   showing as a popover, then <span>topmost popover ancestor</span> will only look in the node tree
   to find the first popover.</p>
  </div>

  <ol>
   <li>
    <p>If <var>isPopover</var> is true:</p>

    <ol>
     <li><p><span>Assert</span>: <var>newPopoverOrTopLayerElement</var> is an <span data-x="HTML
     elements">HTML element</span>.</p></li>

     <li><p><span>Assert</span>: <var>newPopoverOrTopLayerElement</var>'s <code
     data-x="attr-popover">popover</code> attribute is not in the <span
     data-x="attr-popover-none-state">no popover state</span> or the <span
     data-x="attr-popover-manual-state">manual</span> state.</p></li>

     <li><p><span>Assert</span>: <var>newPopoverOrTopLayerElement</var>'s <span>popover visibility
     state</span> is not in the <span data-x="popover-showing-state">popover showing
     state</span>.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p><span>Assert</span>: <var>invoker</var> is null.</p></li>
    </ol>
   </li>

   <li><p>Let <var>popoverPositions</var> be an empty <span>ordered map</span>.</p></li>

   <li><p>Let <var>index</var> be 0.</p></li>

   <li>
    <p>For each <var>popover</var> of <var>popoverList</var>:</p>

    <ol>
     <li><p><span data-x="map set">Set</span> <var>popoverPositions</var>[<var>popover</var>] to
     <var>index</var>.</p></li>

     <li><p>Increment <var>index</var> by 1.</p></li>
    </ol>
   </li>

   <li><p>If <var>isPopover</var> is true, then <span data-x="map set">set</span>
   <var>popoverPositions</var>[<var>newPopoverOrTopLayerElement</var>] to <var>index</var>.</p></li>

   <li><p>Increment <var>index</var> by 1.</p></li>

   <li><p>Let <var>topmostPopoverAncestor</var> be null.</p></li>

   <li>
    <p>Let <var>checkAncestor</var> be an algorithm which performs the following steps given
    <var>candidate</var>:</p>

    <ol>
     <li><p>If <var>candidate</var> is null, then return.</p></li>

     <li><p>Let <var>okNesting</var> be false.</p></li>

     <li><p>Let <var>candidateAncestor</var> be null.</p></li>

     <li>
      <p><span>While</span> <var>okNesting</var> is false:</p>

      <ol>
       <li><p>Set <var>candidateAncestor</var> to the result of running <span>nearest inclusive open
       popover</span> given <var>candidate</var>.</p></li>

       <li><p>If <var>candidateAncestor</var> is null or <var>popoverPositions</var> does not
       contain <var>candidateAncestor</var>, then return.</p></li>

       <li><p><span>Assert</span>: <var>candidateAncestor</var>'s <code
       data-x="attr-popover">popover</code> attribute is not in the <span
       data-x="attr-popover-manual-state">manual</span> or <span
       data-x="attr-popover-none-state">none</span> state.</p></li>

       <li><p>Set <var>okNesting</var> to true if <var>newPopoverOrTopLayerElement</var>'s <code
       data-x="attr-popover">popover</code> attribute is in the <span
       data-x="attr-popover-hint-state">hint</span> state or <var>candidateAncestor</var>'s <code
       data-x="attr-popover">popover</code> attribute is in the <span
       data-x="attr-popover-auto-state">auto</span> state.</p></li>

       <li><p>If <var>okNesting</var> is false, then set <var>candidate</var> to
       <var>candidateAncestor</var>'s parent in the <span>flat tree</span>.</p></li>
      </ol>
     </li>

     <li><p>Let <var>candidatePosition</var> be
     <var>popoverPositions</var>[<var>candidateAncestor</var>].</p></li>

     <li><p>If <var>topmostPopoverAncestor</var> is null or
     <var>popoverPositions</var>[<var>topmostPopoverAncestor</var>] is less than
     <var>candidatePosition</var>, then set <var>topmostPopoverAncestor</var> to
     <var>candidateAncestor</var>.</p></li>
    </ol>
   </li>

   <li><p>Run <var>checkAncestor</var> given <var>newPopoverOrTopLayerElement</var>'s parent node
   within the <span>flat tree</span>.</p></li>

   <li><p>Run <var>checkAncestor</var> given <var>invoker</var>.</p></li>

   <li><p>Return <var>topmostPopoverAncestor</var>.</p></li>
  </ol>

  <p>To find the <dfn>nearest inclusive open popover</dfn> given a <code>Node</code>
  <var>node</var>, perform the following steps. They return an <span data-x="html elements">HTML
  element</span> or null.</p>

  <ol>
   <li><p>Let <var>currentNode</var> be <var>node</var>.</p></li>

   <li>
    <p>While <var>currentNode</var> is not null:</p>

    <ol>
     <li><p>If <var>currentNode</var>'s <code data-x="attr-popover">popover</code> attribute is in
     the <span data-x="attr-popover-auto-state">auto</span> state and <var>currentNode</var>'s
     <span>popover visibility state</span> is <span data-x="popover-showing-state">showing</span>,
     then return <var>currentNode</var>.</p></li>

     <li><p>Set <var>currentNode</var> to <var>currentNode</var>'s parent in the <span>flat
     tree</span>.</p></li>
    </ol>
   </li>

   <li><p>Return null.</p></li>
  </ol>

  <p>To <dfn data-x="topmost auto popover">find the topmost auto or hint popover</dfn> given a
  <code>Document</code> <var>document</var>, perform the following steps. They return an <span
  data-x="HTML elements">HTML element</span> or null.</p>

  <ol>
   <li><p>If <var>document</var>'s <span>showing hint popover list</span> is not empty, then return
   <var>document</var>'s <span>showing hint popover list</span>'s last element.</p></li>

   <li><p>If <var>document</var>'s <span>showing auto popover list</span> is not empty, then return
   <var>document</var>'s <span>showing auto popover list</span>'s last element.</p></li>

   <li><p>Return null.</p></li>
  </ol>

  <p>To perform the <dfn>popover focusing steps</dfn> for an <span data-x="HTML elements">HTML
  element</span> <var>subject</var>:</p>

  <ol>
   <li><p>If the <span>allow focus steps</span> given <var>subject</var>'s <span>node
   document</span> return false, then return.</p></li>

   <li><p>If <var>subject</var> is a <code>dialog</code> element, then run the <span>dialog focusing
   steps</span> given <var>subject</var> and return.</p></li>

   <li><p>If <var>subject</var> has the <code data-x="attr-fe-autofocus">autofocus</code>
   attribute, then let <var>control</var> be <var>subject</var>.

   <li><p>Otherwise, let <var>control</var> be the <span>autofocus delegate</span> for
   <var>subject</var> given "<code data-x="">other</code>".</p></li>

   <li><p>If <var>control</var> is null, then return.</p></li>

   <li><p>Run the <span>focusing steps</span> given <var>control</var>.</p></li>

   <li><p>Let <var>topDocument</var> be the <span>active document</span> of <var>control</var>'s
   <span>node document</span>'s <span data-x="concept-document-bc">browsing context</span>'s
   <span>top-level browsing context</span>.</p></li>

   <li><p>If <var>control</var>'s <span>node document</span>'s <span
   data-x="concept-document-origin">origin</span> is not the <span data-x="same origin">same</span>
   as the <span data-x="concept-document-origin">origin</span> of <var>topDocument</var>, then
   return.</p></li>

   <li><p><span data-x="list empty">Empty</span> <var>topDocument</var>'s
   <span>autofocus candidates</span>.</p></li>

   <li><p>Set <var>topDocument</var>'s <span>autofocus processed flag</span> to true.</p></li>
  </ol>

  <p>To <dfn>check popover validity</dfn> for an <span data-x="HTML elements">HTML element</span>
  <var>element</var> given a boolean <var>expectedToBeShowing</var>, a boolean
  <var>throwExceptions</var>, and a <code>Document</code> or null <var>expectedDocument</var>,
  perform the following steps. They throw an exception or return a boolean.</p>

  <ol>
   <li>
    <p>If <var>element</var>'s <code data-x="attr-popover">popover</code> attribute is in the <span
    data-x="attr-popover-none-state">no popover</span> state, then:</p>

    <ol>
     <li><p>If <var>throwExceptions</var> is true, then throw a
     <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>Return false.</p></li>
    </ol>
   </li>

   <li>
    <p>If any of the following are true:</p>

    <ul>
     <li><p><var>expectedToBeShowing</var> is true and <var>element</var>'s <span>popover visibility
     state</span> is not <span data-x="popover-showing-state">showing</span>; or</p></li>

     <li><p><var>expectedToBeShowing</var> is false and <var>element</var>'s <span>popover
     visibility state</span> is not <span data-x="popover-hidden-state">hidden</span>,</p></li>
    </ul>

    <p>then return false.</p>
   </li>

   <li>
    <p>If any of the following are true:</p>

    <ul>
     <li><p><var>element</var> is not <span>connected</span>;</p></li>

     <li><p><var>element</var>'s <span>node document</span> is not <span>fully
     active</span>;</p></li>

     <li><p><var>expectedDocument</var> is not null and <var>element</var>'s <span>node
     document</span> is not <var>expectedDocument</var>;</p></li>

     <li><p><var>element</var> is a <code>dialog</code> element and its <span>is modal</span>
     is set to true; or</li>

     <li><p><var>element</var>'s <span>fullscreen flag</span> is set,</p></li>
    </ul>

    <p>then:</p>

    <ol>
     <li><p>If <var>throwExceptions</var> is true, then throw a
     <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>Return false.</p></li>
    </ol>
   </li>

   <li><p>Return true.</p></li>
  </ol>

  <p>To get the <dfn id="auto-popover-list">showing auto popover list</dfn> for a
  <code>Document</code> <var>document</var>:</p>

  <ol>
   <li><p>Let <var>popovers</var> be « ».</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <code>Element</code> <var>element</var> in
    <var>document</var>'s <span>top layer</span>:</p>

    <ol>
     <li>
      <p>If all of the following are true:</p>

      <ul>
       <li><p><var>element</var> is an <span data-x="HTML elements">HTML element</span>;</p></li>

       <li><p><var>element</var>'s <span>opened in popover mode</span> is "<code
       data-x="">auto</code>"; and</p></li>

       <li><p><var>element</var>'s <span>popover visibility state</span> is <span
       data-x="popover-showing-state">showing</span>,</p></li>
      </ul>

      <p>then <span data-x="list append">append</span> <var>element</var> to
      <var>popovers</var>.</p>
     </li>
    </ol>
   </li>

   <li><p>Return <var>popovers</var>.</p></li>
  </ol>

  <p>To get the <dfn>showing hint popover list</dfn> for a <code>Document</code>
  <var>document</var>:</p>

  <ol>
   <li><p>Let <var>popovers</var> be « ».</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <code>Element</code> <var>element</var> in
    <var>document</var>'s <span>top layer</span>:</p>

    <ol>
     <li>
      <p>If all of the following are true:</p>

      <ul>
       <li><p><var>element</var> is an <span data-x="HTML elements">HTML element</span>;</p></li>

       <li><p><var>element</var>'s <span>opened in popover mode</span> is "<code
       data-x="">hint</code>"; and</p></li>

       <li><p><var>element</var>'s <span>popover visibility state</span> is <span
       data-x="popover-showing-state">showing</span>,</p></li>
      </ul>

      <p>then <span data-x="list append">append</span> <var>element</var> to
      <var>popovers</var>.</p>
     </li>
    </ol>
   </li>

   <li><p>Return <var>popovers</var>.</p></li>
  </ol>

  <p>To <dfn>close entire popover list</dfn> given a <span>list</span> <var>popoverList</var>, a
  boolean <var>focusPreviousElement</var>, and a boolean <var>fireEvents</var>:</p>

  <ol>
   <li>
    <p>While <var>popoverList</var> is not empty:</p>

    <ol>
     <li><p>Run the <span>hide popover algorithm</span> given <var>popoverList</var>'s last item,
     <var>focusPreviousElement</var>, <var>fireEvents</var>, and false.</p></li>
    </ol>
   </li>
  </ol>

  <h4>The popover target attributes</h4>

  <p><span data-x="concept-button">Buttons</span> may have the following content attributes:</p>

  <ul>
   <li><p><dfn element-attr for="html-global"><code
   data-x="attr-popovertarget">popovertarget</code></dfn></p></li>

   <li><p><dfn element-attr for="html-global"><code
   data-x="attr-popovertargetaction">popovertargetaction</code></dfn></p></li>
  </ul>

  <p>If specified, the <code data-x="attr-popovertarget">popovertarget</code> attribute value must
  be the <span data-x="concept-ID">ID</span> of an element with a <code
  data-x="attr-popover">popover</code> attribute in the same <span>tree</span> as the <span
  data-x="concept-button">button</span> with the <code
  data-x="attr-popovertarget">popovertarget</code> attribute.</p>

  <p>The <code data-x="attr-popovertargetaction">popovertargetaction</code> attribute is an
  <span>enumerated attribute</span> with the following keywords and states:</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
     <th>Brief description
   <tbody>
    <tr>
     <td><dfn attr-value for="html-global/popovertargetaction"
     data-x="attr-popovertargetaction-toggle"><code>toggle</code></dfn>
     <td><dfn data-x="attr-popovertargetaction-toggle-state">toggle</dfn>
     <td>Shows or hides the targeted popover element.
    <tr>
     <td><dfn attr-value for="html-global/popovertargetaction"
     data-x="attr-popovertargetaction-show"><code>show</code></dfn>
     <td><dfn data-x="attr-popovertargetaction-show-state">show</dfn>
     <td>Shows the targeted popover element.
    <tr>
     <td><dfn attr-value for="html-global/popovertargetaction"
     data-x="attr-popovertargetaction-hide"><code>hide</code></dfn>
     <td><dfn data-x="attr-popovertargetaction-hide-state">hide</dfn>
     <td>Hides the targeted popover element.
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <span
  data-x="attr-popovertargetaction-toggle-state">toggle</span> state.</p>

  <p class="note">Whenever possible ensure the popover element is placed immediately after its
  triggering element in the DOM. Doing so will help ensure that the popover is exposed in a logical
  programmatic reading order for users of assistive technology, such as screen readers.</p>

  <div class="example">
   <p>The following shows how the <code data-x="attr-popovertarget">popovertarget</code> attribute
   in combination with the <code data-x="attr-popovertargetaction">popovertargetaction</code>
   attribute can be used to show and close a popover:</p>

   <pre><code class="html">&lt;button popovertarget="foo" popovertargetaction="show">
  Show a popover
&lt;/button>

&lt;article popover="auto" id="foo">
  This is a popover article!
  &lt;button popovertarget="foo" popovertargetaction="hide">Close&lt;/button>
&lt;/article></code></pre>

   <p>If a <code data-x="attr-popovertargetaction">popovertargetaction</code> attribute is not
   specified, the default action will be to toggle the associated popover. The following shows
   how only specifying the <code data-x="attr-popovertarget">popovertarget</code> attribute on its
   invoking button can toggle a manual popover between its opened and closed states. A manual
   popover will not respond to <span data-x="popover light dismiss">light dismiss</span> or <span
   data-x="close request">close requests</span>:</p>

   <pre><code class="html">&lt;input type="button" popovertarget="foo" value="Toggle the popover">

&lt;div popover=manual id="foo">
  This is a popover!
&lt;/div></code></pre>
  </div>

  <span data-x="concept-element-dom">DOM interface</span>:
  <pre><code class="idl">interface mixin <dfn interface>PopoverInvokerElement</dfn> {
  [<span>CEReactions</span>] attribute Element? <span data-x="dom-popoverTargetElement">popoverTargetElement</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-popoverTargetAction">popoverTargetAction</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLElement"><code
  data-x="dom-popoverTargetElement">popoverTargetElement</code></dfn> IDL attribute must
  <span>reflect</span> the <code data-x="attr-popovertarget">popovertarget</code> attribute.</p>

  <p>The <dfn attribute for="HTMLElement"><code
  data-x="dom-popoverTargetAction">popoverTargetAction</code></dfn> IDL attribute must
  <span>reflect</span> the <code data-x="attr-popovertargetaction">popovertargetaction</code>
  attribute, <span>limited to only known values</span>.</p>

  <p>To run the <dfn>popover target attribute activation behavior</dfn> given a <code>Node</code>
  <var>node</var> and a <code>Node</code> <var>eventTarget</var>:</p>

  <ol>
   <li><p>Let <var>popover</var> be <var>node</var>'s <span>popover target element</span>.</p></li>

   <li><p>If <var>popover</var> is null, then return.</p></li>

   <li><p>If <var>eventTarget</var> is a <span>shadow-including inclusive descendant</span> of
   <var>popover</var> and <var>popover</var> is a
   <span data-x="shadow-including descendant">shadow-including descendant</span> of <var>node</var>,
   then return.</p></li>

   <li><p>If <var>node</var>'s <code data-x="attr-popovertargetaction">popovertargetaction</code>
   attribute is in the <span data-x="attr-popovertargetaction-show">show</span> state and
   <var>popover</var>'s <span>popover visibility state</span> is <span
   data-x="popover-showing-state">showing</span>, then return.</p></li>

   <li><p>If <var>node</var>'s <code data-x="attr-popovertargetaction">popovertargetaction</code>
   attribute is in the <span data-x="attr-popovertargetaction-hide">hide</span> state and
   <var>popover</var>'s <span>popover visibility state</span> is <span
   data-x="popover-hidden-state">hidden</span>, then return.</p></li>

   <li><p>If <var>popover</var>'s <span>popover visibility state</span> is <span
   data-x="popover-showing-state">showing</span>, then run the <span>hide popover algorithm</span>
   given <var>popover</var>, true, true, and false.</p></li>

   <li><p>Otherwise, if <var>popover</var>'s <span>popover visibility state</span> is <span
   data-x="popover-hidden-state">hidden</span> and the result of running <span>check popover
   validity</span> given <var>popover</var>, false, false, and null is true, then run <span>show
   popover</span> given <var>popover</var>, false, and <var>node</var>.</p></li>
  </ol>

  <p>To get the <dfn>popover target element</dfn> given a <code>Node</code> <var>node</var>, perform
  the following steps. They return an <span data-x="HTML elements">HTML element</span> or null.</p>

  <ol>
   <li><p>If <var>node</var> is not a <span data-x="concept-button">button</span>, then return
   null.</p></li>

   <li><p>If <var>node</var> is <span data-x="concept-fe-disabled">disabled</span>, then return
   null.</p></li>

   <li><p>If <var>node</var> has a <span>form owner</span> and <var>node</var> is a <span
   data-x="concept-submit-button">submit button</span>, then return null.</p></li>

   <li><p>Let <var>popoverElement</var> be the result of running <var>node</var>'s <span data-x="get
   the attr-associated element">get the <code data-x="">popovertarget</code>-associated
   element</span>.</p></li>

   <li><p>If <var>popoverElement</var> is null, then return null.</p></li>

   <li><p>If <var>popoverElement</var>'s <code data-x="attr-popover">popover</code> attribute is in
   the <span data-x="attr-popover-none-state">no popover</span> state, then return null.</p></li>

   <li><p>Return <var>popoverElement</var>.</p></li>
  </ol>

  <h4><dfn>Popover light dismiss</dfn></h4>

  <p class="note">"Light dismiss" means that clicking outside of a popover whose <code
  data-x="attr-popover">popover</code> attribute is in the <span
  data-x="attr-popover-auto-state">auto</span> state will close the popover. This is in addition to
  how such popovers respond to <span data-x="close request">close requests</span>.</p>

  <p>To <dfn>light dismiss open popovers</dfn>, given a <code>PointerEvent</code>
  <var>event</var>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>event</var>'s <code
   data-x="dom-Event-isTrusted">isTrusted</code> attribute is true.</p></li>

   <li><p>Let <var>target</var> be <var>event</var>'s <span
   data-x="concept-event-target">target</span>.</p></li>

   <li><p>Let <var>document</var> be <var>target</var>'s <span>node document</span>.</p></li>

   <li><p>Let <var>topmostPopover</var> be the result of running <span>topmost auto popover</span>
   given <var>document</var>.</p></li>

   <li><p>If <var>topmostPopover</var> is null, then return.</p></li>

   <li><p>If <var>event</var>'s <code data-x="dom-Event-type">type</code> is "<code
   data-x="event-pointerdown">pointerdown</code>", then: set <var>document</var>'s <span>popover
   pointerdown target</span> to the result of running <span>topmost clicked popover</span> given
   <var>target</var>.</p></li>

   <li>
    <p>If <var>event</var>'s <code data-x="dom-Event-type">type</code> is "<code
    data-x="event-pointerup">pointerup</code>", then:</p>

    <ol>
     <li><p>Let <var>ancestor</var> be the result of running <span>topmost clicked popover</span>
     given <var>target</var>.</p></li>

     <li><p>Let <var>sameTarget</var> be true if <var>ancestor</var> is <var>document</var>'s
     <span>popover pointerdown target</span>.</p></li>

     <li><p>Set <var>document</var>'s <span>popover pointerdown target</span> to null.</p></li>

     <li><p>If <var>ancestor</var> is null, then set <var>ancestor</var> to
     <var>document</var>.</p></li>

     <li><p>If <var>sameTarget</var> is true, then run <span data-x="hide-all-popovers-until">hide
     all popovers until</span> given <var>ancestor</var>, false, and true.</p></li>
    </ol>
   </li>
  </ol>

  <p>To find the <dfn>topmost clicked popover</dfn>, given a <code>Node</code> <var>node</var>:</p>

  <ol>
   <li><p>Let <var>clickedPopover</var> be the result of running <span>nearest inclusive open
   popover</span> given <var>node</var>.</p></li>

   <li><p>Let <var>invokerPopover</var> be the result of running <span>nearest inclusive target
   popover for invoker</span> given <var>node</var>.</p></li>

   <li><p>If the result of <span data-x="get the popover stack position">getting the popover stack
   position</span> given <var>clickedPopover</var> is greater than the result of <span data-x="get
   the popover stack position">getting the popover stack position</span> given
   <var>invokerPopover</var>, then return <var>clickedPopover</var>.</p></li>

   <li><p>Return <var>invokerPopover</var>.</p></li>
  </ol>

  <p>To <dfn>get the popover stack position</dfn>, given an <span data-x="html elements">HTML
  element</span> <var>popover</var>:</p>

  <ol>
   <li><p>Let <var>hintList</var> be <var>popover</var>'s <span>node document</span>'s <span>showing
   hint popover list</span>.</p></li>

   <li><p>Let <var>autoList</var> be <var>popover</var>'s <span>node document</span>'s <span>showing
   auto popover list</span>.</p></li>

   <li><p>If <var>popover</var> is in <var>hintList</var>, then return the index of
   <var>popover</var> in <var>hintList</var> + the size of <var>autoList</var> + 1.</p></li>

   <li><p>If <var>popover</var> is in <var>autoList</var>, then return the index of
   <var>popover</var> in <var>autoList</var> + 1.</p></li>

   <li><p>Return 0.</p></li>
  </ol>

  <p>To find the <dfn>nearest inclusive target popover for invoker</dfn> given a <code>Node</code>
  <var>node</var>:</p>

  <ol>
   <li><p>Let <var>currentNode</var> be <var>node</var>.</p></li>

   <li>
    <p>While <var>currentNode</var> is not null:</p>

    <ol>
     <li><p>Let <var>targetPopover</var> be <var>currentNode</var>'s <span>popover target
     element</span>.</p></li>

     <li><p>If <var>targetPopover</var> is not null and <var>targetPopover</var>'s <code
     data-x="attr-popover">popover</code> attribute is in the <span
     data-x="attr-popover-auto-state">auto</span> state and <var>targetPopover</var>'s <span>popover
     visibility state</span> is <span data-x="popover-showing-state">showing</span>, then return
     <var>targetPopover</var>.</p></li>

     <li><p>Set <var>currentNode</var> to <var>currentNode</var>'s ancestor in the <span>flat
     tree</span>.</p></li>
    </ol>
   </li>
  </ol>


  <h2 split-filename="browsers" id="browsers">Loading web pages</h2>

  <div w-nodev>

  <p>This section describes features that apply most directly to web browsers. Having said that,
  except where specified otherwise, the requirements defined in this section <em>do</em> apply to
  all user agents, whether they are web browsers or not.</p>

  </div>

  <h3 id="loading-web-pages-supporting-concepts">Supporting concepts</h3>

  <h4 id="origin">Origins</h4>
  <!-- Hallowed are the Ori -->

  <p>Origins are the fundamental currency of the web's security model. Two actors in the web
  platform that share an origin are assumed to trust each other and to have the same authority.
  Actors with differing origins are considered potentially hostile versus each other, and are
  isolated from each other to varying degrees.</p>

  <p class="example">For example, if Example Bank's web site, hosted at <code
  data-x="">bank.example.com</code>, tries to examine the DOM of Example Charity's web site, hosted
  at <code data-x="">charity.example.org</code>, a <span>"<code>SecurityError</code>"</span>
  <code>DOMException</code> will be raised.</p>

  <hr>

  <p id="origin-2">An <dfn id="concept-origin" export>origin</dfn> is one of the following:</p>

  <dl>
   <dt>An <dfn export data-x="concept-origin-opaque">opaque origin</dfn></dt>
   <dd><p>An internal value, with no serialization it can be recreated from (it is serialized as
   "<code data-x="">null</code>" per <span>serialization of an origin</span>), for which the only
   meaningful operation is testing for equality.</p></dd>

   <dt>A <dfn export data-x="concept-origin-tuple">tuple origin</dfn></dt>
   <dd>
    <p>A <span>tuple</span> consisting of:

    <ul class="brief">
     <li>A <dfn export for="origin" data-x="concept-origin-scheme">scheme</dfn> (an <span>ASCII
     string</span>).</li>
     <li>A <dfn export for="origin" data-x="concept-origin-host">host</dfn> (a <span
     data-x="concept-host">host</span>).</li>
     <li>A <dfn export for="origin" data-x="concept-origin-port">port</dfn> (null or a 16-bit
     unsigned integer).</li>
     <li>A <dfn export for="origin" data-x="concept-origin-domain">domain</dfn> (null or a <span
     data-x="concept-domain">domain</span>). Null unless stated otherwise.</li>
    </ul>
   </dd>
  </dl>

  <p class="note"><span data-x="origin">Origins</span> can be shared, e.g., among multiple
  <code>Document</code> objects. Furthermore, <span data-x="origin">origins</span> are generally
  immutable. Only the <span data-x="concept-origin-domain">domain</span> of a <span
  data-x="concept-origin-tuple">tuple origin</span> can be changed, and only through the <code
  data-x="dom-document-domain">document.domain</code> API.</p>

  <div w-nodev>

  <p>The <dfn export data-x="concept-origin-effective-domain">effective domain</dfn> of an
  <span>origin</span> <var>origin</var> is computed as follows:</p>

  <ol>
   <li><p>If <var>origin</var> is an <span data-x="concept-origin-opaque">opaque origin</span>,
   then return null.</p></li>

   <li><p>If <var>origin</var>'s <span data-x="concept-origin-domain">domain</span> is non-null,
   then return <var>origin</var>'s <span data-x="concept-origin-domain">domain</span>.</p></li>

   <li><p>Return <var>origin</var>'s <span data-x="concept-origin-host">host</span>.</p></li>
  </ol>

  <p>The <!--en-GB--><dfn id="ascii-serialisation-of-an-origin" export data-lt="serialization of an
  origin|ASCII serialization of an origin">serialization of an origin</dfn> is the string obtained
  by applying the following algorithm to the given <span>origin</span> <var>origin</var>:</p>

  <ol>
   <li><p>If <var>origin</var> is an <span data-x="concept-origin-opaque">opaque origin</span>,
   then return "<code data-x="">null</code>".</p></li>

   <li><p>Otherwise, let <var>result</var> be <var>origin</var>'s <span
   data-x="concept-origin-scheme">scheme</span>.</p></li>

   <li><p>Append "<code data-x="">://</code>" to <var>result</var>.</p></li>

   <li><p>Append <var>origin</var>'s <span data-x="concept-origin-host">host</span>, <span
   data-x="host serializer">serialized</span>, to <var>result</var>.</p></li>

   <li><p>If <var>origin</var>'s <span data-x="concept-origin-port">port</span> is non-null, append
   a U+003A COLON character (:), and <var>origin</var>'s <span
   data-x="concept-origin-port">port</span>, <span data-x="serialize an integer">serialized</span>,
   to <var>result</var>.</p></li>

   <li><p>Return <var>result</var>.</p></li>
  </ol>

  </div>

  <p class="example">The <span data-x="serialization of an origin">serialization</span> of ("<code
  data-x="">https</code>", "<code data-x="">xn--maraa-rta.example</code>", null, null) is "<code
  data-x="">https://xn--maraa-rta.example</code>".</p>

  <div w-nodev>

  <!--en-GB--><p class="note" id="unicode-serialisation-of-an-origin">There used to also be a
  <i>Unicode serialization of an origin</i>. However, it was never widely adopted.</p>

  <hr>

  <p>Two <span data-x="origin">origins</span>, <var>A</var> and <var>B</var>, are said to be <dfn
  export>same origin</dfn> if the following algorithm returns true:</p>

  <ol>
   <li><p>If <var>A</var> and <var>B</var> are the same <span data-x="concept-origin-opaque">opaque
   origin</span>, then return true.</p></li>

   <li><p>If <var>A</var> and <var>B</var> are both <span data-x="concept-origin-tuple">tuple
   origins</span> and their <span data-x="concept-origin-scheme">schemes</span>, <span
   data-x="concept-origin-host">hosts</span>, and <span data-x="concept-origin-port">port</span>
   are identical, then return true.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  <p>Two <span data-x="origin">origins</span>, <var>A</var> and <var>B</var>, are said to be <dfn
  export>same origin-domain</dfn> if the following algorithm returns true:</p>

  <ol>
   <li><p>If <var>A</var> and <var>B</var> are the same <span data-x="concept-origin-opaque">opaque
   origin</span>, then return true.</p></li>

   <li>
    <p>If <var>A</var> and <var>B</var> are both <span data-x="concept-origin-tuple">tuple
    origins</span>:</p>

    <ol>
     <li><p>If <var>A</var> and <var>B</var>'s <span data-x="concept-origin-scheme">schemes</span>
     are identical, and their <span data-x="concept-origin-domain">domains</span> are identical and
     non-null, then return true.</p></li>

     <li><p>Otherwise, if <var>A</var> and <var>B</var> are <span>same origin</span> and their
     <span data-x="concept-origin-domain">domains</span> are both null, return true.</p></li>
    </ol>

   <li><p>Return false.</p></li>
  </ol>

  </div>

  <div class="example">
   <p w-dev>The following table shows examples of when two <span data-x="concept-origin-tuple">tuple
   origins</span> are <span subdfn>same origin</span> and <span subdfn>same
   origin-domain</span>.</p>

   <table>
    <tr>
     <th><var>A</var>
     <th><var>B</var>
     <th><span>same origin</span>
     <th><span>same origin-domain</span>
    <tr>
     <td>("<code data-x="">https</code>", "<code data-x="">example.org</code>", null, null)
     <td>("<code data-x="">https</code>", "<code data-x="">example.org</code>", null, null)
     <td>✅
     <td>✅
    <tr>
     <td>("<code data-x="">https</code>", "<code data-x="">example.org</code>", 314, null)
     <td>("<code data-x="">https</code>", "<code data-x="">example.org</code>", 420, null)
     <td>❌
     <td>❌
    <tr>
     <td>("<code data-x="">https</code>", "<code data-x="">example.org</code>", 314, "<code data-x="">example.org</code>")
     <td>("<code data-x="">https</code>", "<code data-x="">example.org</code>", 420, "<code data-x="">example.org</code>")
     <td>❌
     <td>✅
    <tr>
     <td>("<code data-x="">https</code>", "<code data-x="">example.org</code>", null, null)
     <td>("<code data-x="">https</code>", "<code data-x="">example.org</code>", null, "<code data-x="">example.org</code>")
     <td>✅
     <td>❌
    <tr>
     <td>("<code data-x="">https</code>", "<code data-x="">example.org</code>", null, "<code data-x="">example.org</code>")
     <td>("<code data-x="">http</code>", "<code data-x="">example.org</code>", null, "<code data-x="">example.org</code>")
     <td>❌
     <td>❌
   </table>
  </div>


  <h5>Sites</h5>

  <p>A <dfn>scheme-and-host</dfn> is a <span>tuple</span> of a <dfn
  data-x="concept-scheme-and-host-scheme">scheme</dfn> (an <span>ASCII string</span>) and a <dfn
  data-x="concept-scheme-and-host-host">host</dfn> (a <span data-x="concept-host">host</span>).</p>

  <p>A <dfn export>site</dfn> is an <span data-x="concept-origin-opaque">opaque origin</span> or a
  <span>scheme-and-host</span>.</p>

  <div w-nodev>

  <p>To <dfn export>obtain a site</dfn>, given an origin <var>origin</var>, run these steps:</p>

  <ol>
   <li><p>If <var>origin</var> is an <span data-x="concept-origin-opaque">opaque origin</span>,
   then return <var>origin</var>.</p></li>

   <li><p>If <var>origin</var>'s <span data-x="concept-origin-host">host</span>'s <span>registrable
   domain</span> is null, then return (<var>origin</var>'s <span
   data-x="concept-origin-scheme">scheme</span>, <var>origin</var>'s <span
   data-x="concept-origin-host">host</span>).</p></li>

   <li><p>Return (<var>origin</var>'s <span data-x="concept-origin-scheme">scheme</span>,
   <var>origin</var>'s <span data-x="concept-origin-host">host</span>'s <span>registrable
   domain</span>).</p></li>
  </ol>

  <p>Two <span data-x="site">sites</span>, <var>A</var> and <var>B</var>, are said to be <dfn
  export for="site" data-x="concept-site-same-site">same site</dfn> if the following algorithm
  returns true:</p>

  <ol>
   <li><p>If <var>A</var> and <var>B</var> are the same <span data-x="concept-origin-opaque">opaque
   origin</span>, then return true.</p></li>

   <li><p>If <var>A</var> or <var>B</var> is an <span data-x="concept-origin-opaque">opaque
   origin</span>, then return false.</p></li>

   <li><p>If <var>A</var>'s and <var>B</var>'s <span
   data-x="concept-scheme-and-host-scheme">scheme</span> values are different, then return
   false.</p></li>

   <li><p>If <var>A</var>'s and <var>B</var>'s <span
   data-x="concept-scheme-and-host-host">host</span> values are not <span data-x="host
   equals">equal</span>, then return false.</p></li>

   <li><p>Return true.</p></li>
  </ol>

  <p>The <dfn export>serialization of a site</dfn> is the string obtained by applying the following
  algorithm to the given <span>site</span> <var>site</var>:</p>

  <ol>
   <li><p>If <var>site</var> is an <span data-x="concept-origin-opaque">opaque origin</span>, then
   return "<code data-x="">null</code>".</p></li>

   <li><p>Let <var>result</var> be <var>site</var>[0].</p></li>

   <li><p>Append "<code data-x="">://</code>" to <var>result</var>.</p></li>

   <li><p>Append <var>site</var>[1], <span data-x="host serializer">serialized</span>, to
   <var>result</var>.</p></li>

   <li><p>Return <var>result</var>.</p></li>
  </ol>

  <p class="warning">It needs to be clear from context that the serialized value is a site, not an
  origin, as there is not necessarily a syntactic difference between the two. For example, the
  origin ("<code data-x="">https</code>", "<code data-x="">shop.example</code>", null, null) and
  the site ("<code data-x="">https</code>", "<code data-x="">shop.example</code>") have the same
  serialization: "<code data-x="">https://shop.example</code>".</p>

  <p>Two <span data-x="origin">origins</span>, <var>A</var> and <var>B</var>, are said to be <dfn
  export>schemelessly same site</dfn> if the following algorithm returns true:</p>

  <ol>
   <li><p>If <var>A</var> and <var>B</var> are the same <span data-x="concept-origin-opaque">opaque
   origin</span>, then return true.</p></li>

   <li>
    <p>If <var>A</var> and <var>B</var> are both <span data-x="concept-origin-tuple">tuple
    origins</span>, then:</p>

    <ol>
     <li><p>Let <var>hostA</var> be <var>A</var>'s <span data-x="concept-origin-host">host</span>,
     and let <var>hostB</var> be <var>B</var>'s <span
     data-x="concept-origin-host">host</span>.</p></li>

     <li><p>If <var>hostA</var> <span data-x="host equals">equals</span> <var>hostB</var> and
     <var>hostA</var>'s <span>registrable domain</span> is null, then return true.</p></li>

     <li><p>If <var>hostA</var>'s <span>registrable domain</span> <span data-x="host
     equals">equals</span> <var>hostB</var>'s <span>registrable domain</span> and is non-null, then
     return true.</p></li>
    </ol>
   </li>

   <li><p>Return false.</p></li>
  </ol>

  <p>Two <span data-x="origin">origins</span>, <var>A</var> and <var>B</var>, are said to be <dfn
  export>same site</dfn> if the following algorithm returns true:</p>

  <ol>
   <li><p>Let <var>siteA</var> be the result of <span data-x="obtain a site">obtaining a
   site</span> given <var>A</var>.</p></li>

   <li><p>Let <var>siteB</var> be the result of <span data-x="obtain a site">obtaining a
   site</span> given <var>B</var>.</p></li>

   <li><p>If <var>siteA</var> is <span data-x="concept-site-same-site">same site</span> with
   <var>siteB</var>, then return true.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  </div>

  <p class="note">Unlike the <span>same origin</span> and <span>same origin-domain</span> concepts,
  for <span>schemelessly same site</span> and <span>same site</span>, the <span
  data-x="concept-origin-port">port</span> and <span data-x="concept-origin-domain">domain</span>
  components are ignored.</p>

  <p class="warning">For the reasons <a
  href="https://url.spec.whatwg.org/#warning-avoid-psl">explained in <cite>URL</cite></a>, the
  <span>same site</span> and <span>schemelessly same site</span> concepts should be avoided when
  possible, in favor of <span>same origin</span> checks.</p>

  <div class="example" id="example-same-site">
   <p w-dev>The following table shows examples of when two <span data-x="concept-origin-tuple">tuple
   origins</span> are <span subdfn>schemelessly same site</span> and <span subdfn>same
   site</span>.</p>

  <p>Given that <code data-x="">wildlife.museum</code>, <code data-x="">museum</code>, and <code
  data-x="">com</code> are <span data-x="public suffix">public suffixes</span> and that <code
  data-x="">example.com</code> is not:</p>

   <table>
    <tr>
     <th><var>A</var>
     <th><var>B</var>
     <th><span>schemelessly same site</span>
     <th><span>same site</span>
    <tr>
     <td>("<code data-x="">https</code>", "<code data-x="">example.com</code>")
     <td>("<code data-x="">https</code>", "<code data-x="">sub.example.com</code>")
     <td>✅
     <td>✅
    <tr>
     <td>("<code data-x="">https</code>", "<code data-x="">example.com</code>")
     <td>("<code data-x="">https</code>", "<code data-x="">sub.other.example.com</code>")
     <td>✅
     <td>✅
    <tr>
     <td>("<code data-x="">https</code>", "<code data-x="">example.com</code>")
     <td>("<code data-x="">http</code>", "<code data-x="">non-secure.example.com</code>")
     <td>✅
     <td>❌
    <tr>
     <td>("<code data-x="">https</code>", "<code data-x="">r.wildlife.museum</code>")
     <td>("<code data-x="">https</code>", "<code data-x="">sub.r.wildlife.museum</code>")
     <td>✅
     <td>✅
    <tr>
     <td>("<code data-x="">https</code>", "<code data-x="">r.wildlife.museum</code>")
     <td>("<code data-x="">https</code>", "<code data-x="">sub.other.r.wildlife.museum</code>")
     <td>✅
     <td>✅
    <tr>
     <td>("<code data-x="">https</code>", "<code data-x="">r.wildlife.museum</code>")
     <td>("<code data-x="">https</code>", "<code data-x="">other.wildlife.museum</code>")
     <td>❌
     <td>❌
    <tr>
     <td>("<code data-x="">https</code>", "<code data-x="">r.wildlife.museum</code>")
     <td>("<code data-x="">https</code>", "<code data-x="">wildlife.museum</code>")
     <td>❌
     <td>❌
    <tr>
     <td>("<code data-x="">https</code>", "<code data-x="">wildlife.museum</code>")
     <td>("<code data-x="">https</code>", "<code data-x="">wildlife.museum</code>")
     <td>✅
     <td>✅
    <tr>
     <td>("<code data-x="">https</code>", "<code data-x="">example.com</code>")
     <td>("<code data-x="">https</code>", "<code data-x="">example.com.</code>")
     <td>❌
     <td>❌
   </table>

   <p>(Here we have omitted the <span data-x="concept-origin-port">port</span> and <span
   data-x="concept-origin-domain">domain</span> components since they are not considered.)</p>
  </div>


  <h5>Relaxing the same-origin restriction</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-domain">domain</span> [ = <var>domain</var> ]</code></dt>

   <dd>
    <p>Returns the current domain used for security checks.</p>

    <p>Can be set to a value that removes subdomains, to change the <span>origin</span>'s <span
    data-x="concept-origin-domain">domain</span> to allow pages on other subdomains of the same
    domain (if they do the same thing) to access each other. This enables pages on different hosts
    of a domain to synchronously access each other's DOMs.</p>

    <p>In sandboxed <code>iframe</code>s, <code>Document</code>s with <span
    data-x="concept-origin-opaque">opaque origins</span>, and <code>Document</code>s without a <span
    data-x="concept-document-bc">browsing context</span>, the setter will
    throw a <span>"<code>SecurityError</code>"</span> exception. In cases where <code
    data-x="dom-crossOriginIsolated">crossOriginIsolated</code> or <code
    data-x="dom-originAgentCluster">originAgentCluster</code> return true, the setter will do
    nothing.</p>
   </dd>
  </dl>

  <div class="critical">
   <p>Avoid using the <code data-x="dom-document-domain">document.domain</code> setter. It
   undermines the security protections provided by the same-origin policy. This is especially acute
   when using shared hosting; for example, if an untrusted third party is able to host an HTTP
   server at the same IP address but on a different port, then the same-origin protection that
   normally protects two different sites on the same host will fail, as the ports are ignored when
   comparing origins after the <code data-x="dom-document-domain">document.domain</code> setter has
   been used.</p>

   <p>Because of these security pitfalls, this feature is in the process of being removed from the
   web platform. (This is a long process that takes many years.)</p>

   <p>Instead, use <code data-x="dom-window-postMessage">postMessage()</code> or
   <code>MessageChannel</code> objects to communicate across origins in a safe manner.</p>
  </div>

  <div w-nodev>

  <p>The <dfn attribute for="Document"><code data-x="dom-document-domain">domain</code></dfn>
  getter steps are:</p>

  <ol>
   <li><p>Let <var>effectiveDomain</var> be <span>this</span>'s <span
   data-x="concept-document-origin">origin</span>'s <span
   data-x="concept-origin-effective-domain">effective domain</span>.

   <li><p>If <var>effectiveDomain</var> is null, then return the empty string.</p></li>

   <li><p>Return <var>effectiveDomain</var>, <span data-x="host
   serializer">serialized</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-document-domain">domain</code> setter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="concept-document-bc">browsing context</span> is
   null, then throw a <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <span>this</span>'s <span>active sandboxing flag set</span> has its <span>sandboxed
   <code data-x="dom-document-domain">document.domain</code> browsing context flag</span> set, then
   throw a <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>effectiveDomain</var> be <span>this</span>'s <span
   data-x="concept-document-origin">origin</span>'s <span
   data-x="concept-origin-effective-domain">effective domain</span>.

   <li><p>If <var>effectiveDomain</var> is null, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If the given value <span data-x="is a registrable domain suffix of or is equal to">is not
   a registrable domain suffix of and is not equal to</span> <var>effectiveDomain</var>, then throw
   a <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If the <span>surrounding agent</span>'s <span>agent cluster</span>'s <span>is
   origin-keyed</span> is true, then return.</p></li>

   <li><p>Set <span>this</span>'s <span data-x="concept-document-origin">origin</span>'s <span
   data-x="concept-origin-domain">domain</span> to the result of <span data-x="host
   parser">parsing</span> the given value.</p></li>
  </ol>

  <p>To determine if a <span>scalar value string</span> <var>hostSuffixString</var> <dfn export
  data-lt="is a registrable domain suffix of or is equal to|is not a registrable domain suffix of
  and is not equal to">is a registrable domain suffix of or is equal to</dfn> a <span
  data-x="concept-host">host</span> <var>originalHost</var>:</p>
  <!-- web platform usage: document.domain, Cookies, web Authentication -->

  <ol>
   <li><p>If <var>hostSuffixString</var> is the empty string, then return false.</p></li>

   <li><p>Let <var>hostSuffix</var> be the result of <span data-x="host parser">parsing</span>
   <var>hostSuffixString</var>.</p></li>

   <li><p>If <var>hostSuffix</var> is failure, then return false.</p></li>

   <li>
    <p>If <var>hostSuffix</var> does not <span data-x="host equals">equal</span>
    <var>originalHost</var>, then:</p>

    <ol>
     <li>
      <p>If <var>hostSuffix</var> or <var>originalHost</var> is not a <span
      data-x="concept-domain">domain</span>, then return false.</p>

      <p class="note">This excludes <span data-x="concept-host">hosts</span> that are <span
      data-x="IP address">IP addresses</span>.</p>
     </li>

     <li><p>If <var>hostSuffix</var>, prefixed by U+002E (.), does not match the end of
     <var>originalHost</var>, then return false.</p></li>
     <!-- It would be cleaner to avoid string manipulation on domains. -->

     <li>
      <p>If any of the following are true:</p>

      <ul>
       <li><p><var>hostSuffix</var> <span data-x="host equals">equals</span>
       <var>hostSuffix</var>'s <span>public suffix</span>; or</p></li>

       <li><p><var>hostSuffix</var>, prefixed by U+002E (.), matches the end of
       <var>originalHost</var>'s <span>public suffix</span>,</p></li>
       <!-- It would be cleaner to avoid string manipulation on domains. -->
      </ul>

      <p>then return false. <ref>URL</ref></p>
     </li>

     <li><p><span>Assert</span>: <var>originalHost</var>'s <span>public suffix</span>, prefixed by
     U+002E (.), matches the end of <var>hostSuffix</var>.</p></li>
     <!-- It would be cleaner to avoid string manipulation on domains. -->
    </ol>
   </li>

   <li><p>Return true.</p></li>
  </ol>

  <div class="example" id="example-registrable-domain-suffix">
   <table>
    <tr>
     <th><var>hostSuffixString</var></th>
     <th><var>originalHost</var></th>
     <th>Outcome of <span>is a registrable domain suffix of or is equal to</span></th>
     <th>Notes</th>
    </tr>
    <tr>
     <td>"<code data-x="">0.0.0.0</code>"</td>
     <td><code data-x="">0.0.0.0</code></td>
     <td>✅</td>
     <td></td>
    </tr>
    <tr>
     <td>"<code data-x="">0x10203</code>"</td>
     <td><code data-x="">0.1.2.3</code></td>
     <td>✅</td>
     <td></td>
    </tr>
    <tr>
     <td>"<code data-x="">[0::1]</code>"</td>
     <td><code data-x="">::1</code></td>
     <td>✅</td>
     <td></td>
    </tr>
    <tr>
     <td>"<code data-x="">example.com</code>"</td>
     <td><code data-x="">example.com</code></td>
     <td>✅</td>
     <td></td>
    </tr>
    <tr>
     <td>"<code data-x="">example.com</code>"</td>
     <td><code data-x="">example.com.</code></td>
     <td>❌</td>
     <td rowspan="2">Trailing dot is significant.</td>
    </tr>
    <tr>
     <td>"<code data-x="">example.com.</code>"</td>
     <td><code data-x="">example.com</code></td>
     <td>❌</td>
    </tr>
    <tr>
     <td>"<code data-x="">example.com</code>"</td>
     <td><code data-x="">www.example.com</code></td>
     <td>✅</td>
     <td></td>
    </tr>
    <tr>
     <td>"<code data-x="">com</code>"</td>
     <td><code data-x="">example.com</code></td>
     <td>❌</td>
     <td>At the time of writing, <code data-x="">com</code> is a public suffix.</td>
    </tr>
    <tr>
     <td>"<code data-x="">example</code>"</td>
     <td><code data-x="">example</code></td>
     <td>✅</td>
     <td></td>
    </tr>
    <tr>
     <td>"<code data-x="">compute.amazonaws.com</code>"</td>
     <td><code data-x="">example.compute.amazonaws.com</code></td>
     <td>❌</td>
     <td rowspan="3">At the time of writing, <code data-x=""><var>*</var>.compute.amazonaws.com</code> is a public suffix.</td>
    </tr>
    <tr>
     <td>"<code data-x="">example.compute.amazonaws.com</code>"</td>
     <td><code data-x="">www.example.compute.amazonaws.com</code></td>
     <td>❌</td>
    </tr>
    <tr>
     <td>"<code data-x="">amazonaws.com</code>"</td>
     <td><code data-x="">www.example.compute.amazonaws.com</code></td>
     <td>❌</td>
    </tr>
    <tr>
     <td>"<code data-x="">amazonaws.com</code>"</td>
     <td><code data-x="">test.amazonaws.com</code></td>
     <td>✅</td>
     <td>At the time of writing, <code data-x="">amazonaws.com</code> is a registrable domain.</td>
    </tr>
   </table>
  </div>

  </div>


  <h4><span id="origin-isolation"></span>Origin-keyed agent clusters</h4>

  <dl class="domintro">
   <dt><code data-x="">window.<span subdfn data-x="dom-originAgentCluster">originAgentCluster</span></code></dt>

   <dd>
    <p>Returns true if this <code>Window</code> belongs to an <span>agent cluster</span> which is
    <span>origin</span>-<span data-x="agent cluster key">keyed</span>, in the manner described in
    this section.</p>
   </dd>
  </dl>

  <p>A <code>Document</code> delivered over a <span>secure context</span> can request that it be
  placed in an <span>origin</span>-<span data-x="agent cluster key">keyed</span> <span>agent
  cluster</span>, by using the `<dfn http-header><code>Origin-Agent-Cluster</code></dfn>` HTTP
  response header. This header is a <span data-x="http-structured-header">structured header</span>
  whose value must be a <span data-x="http-structured-header-boolean">boolean</span>.
  <ref>STRUCTURED-FIELDS</ref></p>

  <p><span w-nodev>Per the processing model in the <span data-x="create-the-document-object">create
  and initialize a new <code>Document</code> object</span>, values</span><span w-dev>Values</span>
  that are not the <span data-x="http-structured-header-boolean">structured header boolean</span>
  true value (i.e., `<code data-x="">?1</code>`) will be ignored.</p>

  <p>The consequences of using this header are that <span w-nodev>the resulting
  <code>Document</code>'s <span>agent cluster key</span> is its <span
  data-x="concept-document-origin">origin</span>, instead of the <span data-x="obtain a
  site">corresponding site</span>. In terms of observable effects, this means that
  </span>attempting to <a href="#relaxing-the-same-origin-restriction">relax the same-origin
  restriction</a> using <code data-x="dom-document-domain">document.domain</code> will instead do
  nothing, and it will not be possible to send <code>WebAssembly.Module</code> objects to
  cross-origin <code>Document</code>s (even if they are <span>same site</span>). Behind the scenes,
  this isolation can allow user agents to allocate implementation-specific resources corresponding
  to <span data-x="agent cluster">agent clusters</span>, such as processes or threads, more
  efficiently.</p>

  <p>Note that within a <span>browsing context group</span>, the
  `<code>Origin-Agent-Cluster</code>` header can never cause same-origin <code>Document</code>
  objects to end up in different <span data-x="agent cluster">agent clusters</span>, even if one
  sends the header and the other doesn't.<span w-nodev> This is prevented by means of the
  <span>historical agent cluster key map</span>.</span></p>

  <p class="note">This means that the <code
  data-x="dom-originAgentCluster">originAgentCluster</code> getter can return false, even if the
  header is set, if the header was omitted on a previously-loaded same-origin page in the same
  <span>browsing context group</span>. Similarly, it can return true even when the header is not
  set.</p>

  <p w-nodev>The <dfn attribute for="Window"><code
  data-x="dom-originAgentCluster">originAgentCluster</code></dfn> getter steps are to return the
  <span>surrounding agent</span>'s <span>agent cluster</span>'s <span>is origin-keyed</span>.</p>

  <p class="note"><code>Document</code>s with an <span data-x="concept-origin-opaque">opaque
  origin</span> can be considered unconditionally origin-keyed; for them the header has no effect,
  and the <code data-x="dom-originAgentCluster">originAgentCluster</code> getter will always return
  true.</p>

  <p class="note">Similarly, <code>Document</code>s whose <span>agent cluster</span>'s <span
  data-x="agent-cluster-cross-origin-isolation">cross-origin isolation mode</span> is not "<code
  data-x="cross-origin-isolation-none">none</code>" are automatically origin-keyed. The
  `<code>Origin-Agent-Cluster</code>` header might be useful as an additional hint to
  implementations about resource allocation, since the `<code>Cross-Origin-Opener-Policy</code>`
  and `<code>Cross-Origin-Embedder-Policy</code>` headers used to achieve cross-origin isolation
  are more about ensuring that everything in the same address space opts in to being there. But
  adding it would have no additional observable effects on author code.</p>


  <h4>Cross-origin opener policies</h4>

  <p>An <dfn id="cross-origin-opener-policy-value">opener policy value</dfn> allows a document which
  is navigated to in a <span>top-level browsing context</span> to force the creation of a new
  <span>top-level browsing context</span>, and a corresponding <span data-x="tlbc
  group">group</span>. The possible values are:</p>

  <dl>
   <dt>"<dfn><code data-x="coop-unsafe-none">unsafe-none</code></dfn>"</dt>
   <dd><p>This is the (current) default and means that the document will occupy the same
   <span>top-level browsing context</span> as its predecessor, unless that document specified a
   different <span>opener policy</span>.</p></dd>

   <dt>"<dfn><code data-x="coop-same-origin-allow-popups">same-origin-allow-popups</code></dfn>"</dt>
   <dd><p>This forces the creation of a new <span>top-level browsing context</span> for the
   document, unless its predecessor specified the same <span>opener policy</span> and they are
   <span>same origin</span>.</p></dd>

   <dt>"<dfn><code data-x="coop-same-origin">same-origin</code></dfn>"</dt>
   <dd><p>This behaves the same as "<code
   data-x="coop-same-origin-allow-popups">same-origin-allow-popups</code>", with the addition that
   any <span>auxiliary browsing context</span> created needs to contain <span>same origin</span>
   documents that also have the same <span>opener policy</span> or it will appear closed to the
   opener.</p></dd>

   <dt>"<dfn><code data-x="coop-same-origin-plus-COEP">same-origin-plus-COEP</code></dfn>"</dt>
   <dd>
    <p>This behaves the same as "<code data-x="coop-same-origin">same-origin</code>", with the
    addition that it sets the (new) <span>top-level browsing context</span>'s <span data-x="tlbc
    group">group</span>'s <span data-x="bcg-cross-origin-isolation">cross-origin isolation
    mode</span> to one of "<code data-x="cross-origin-isolation-logical">logical</code>" or "<code
    data-x="cross-origin-isolation-concrete">concrete</code>".</p>

    <p class="note">"<code data-x="coop-same-origin-plus-COEP">same-origin-plus-COEP</code>" cannot
    be directly set via the `<code>Cross-Origin-Opener-Policy</code>` header, but results from a
    combination of setting both `<code data-x=""><span>Cross-Origin-Opener-Policy</span>: <span
    data-x="coop-same-origin">same-origin</span></code>` and a
    `<code>Cross-Origin-Embedder-Policy</code>` header whose value is <span>compatible with
    cross-origin isolation</span> together.</p>
   </dd>

   <dt>"<dfn><code data-x="coop-noopener-allow-popups">noopener-allow-popups</code></dfn>"</dt>
   <dd>
    <p>This forces the creation of a new <span>top-level browsing context</span> for the document,
    regardless of its predecessor.</p>

    <div class="note">
     <p>While including a <code
     data-x="coop-noopener-allow-popups">noopener-allow-popups</code> value severs the opener
     relationship between the document on which it is applied and its opener, it does not create a
     robust security boundary between those same-origin documents.</p>

     <p>Other risks from same-origin applications include:</p>

     <ul>
      <li><p>Same-origin requests fetching the document's content — could be mitigated through
      Fetch Metadata filtering. <ref>FETCHMETADATA</ref></p></li>
      <li><p>Same-origin framing - could be mitigated through <code>X-Frame-Options</code> or CSP
      <code data-x="frame-ancestors directive">frame-ancestors</code>.</p></li>
      <li><p>JavaScript accessible cookies - can be mitigated by ensuring all cookies are <code
      data-x="">httponly</code>.</p></li>
      <li><p><code data-x="dom-localStorage">localStorage</code> access to sensitive data.</p></li>
      <li><p>Service worker installation.</p></li>
      <li><p><a href="https://w3c.github.io/ServiceWorker/#cache">Cache API</a> manipulation or
      access to sensitive data. <ref>SW</ref></p></li>
      <li><p><code data-x="">postMessage</code> or <code>BroadcastChannel</code> messaging that
      exposes sensitive information.</p></li>
      <li><p>Autofill which may not require user interaction for same-origin documents.</p></li>
     </ul>

     <p>Developers using <code data-x="coop-noopener-allow-popups">noopener-allow-popups</code>
     need to make sure that their sensitive applications don't rely on client-side features
     accessible to other same-origin documents, e.g., <code
     data-x="dom-localStorage">localStorage</code> and other client-side storage APIs,
     <code>BroadcastChannel</code> and related same-origin communication mechanisms. They also need
     to make sure that their server-side endpoints don't return sensitive data to non-navigation
     requests, whose response content is accessible to same-origin
     documents.</p>
    </div>
   </dd>
  </dl>

  <div w-nodev>

  <p>An <dfn id="cross-origin-opener-policy">opener policy</dfn> consists of:</p>

  <ul>
   <li><p>A <dfn data-x="coop-struct-value">value</dfn>, which is an <span data-x="opener policy
   value">opener policy value</span>, initially "<code
   data-x="coop-unsafe-none">unsafe-none</code>".</p></li>

   <li><p>A <dfn data-x="coop-struct-report-endpoint">reporting endpoint</dfn>, which is string or
   null, initially null.</p></li>

   <li><p>A <dfn data-x="coop-struct-report-only-value">report-only value</dfn>, which is an <span
   data-x="opener policy value">opener policy value</span>, initially "<code
   data-x="coop-unsafe-none">unsafe-none</code>".</p></li>

   <li><p>A <dfn data-x="coop-struct-report-only-endpoint">report-only reporting endpoint</dfn>,
   which is a string or null, initially null.</p></li>
  </ul>

  <p>To <dfn data-x="matching-coop">match opener policy values</dfn>, given an <span>opener policy
  value</span> <var>documentCOOP</var>, an <span>origin</span> <var>documentOrigin</var>, an
  <span>opener policy value</span> <var>responseCOOP</var>, and an <span>origin</span>
  <var>responseOrigin</var>:</p>

  <ol>
   <li><p>If <var>documentCOOP</var> is "<code data-x="coop-unsafe-none">unsafe-none</code>" and
   <var>responseCOOP</var> is "<code data-x="coop-unsafe-none">unsafe-none</code>", then return
   true.</p></li>

   <li><p>If <var>documentCOOP</var> is "<code data-x="coop-unsafe-none">unsafe-none</code>" or
   <var>responseCOOP</var> is "<code data-x="coop-unsafe-none">unsafe-none</code>", then return
   false.</p></li>

   <li><p>If <var>documentCOOP</var> is <var>responseCOOP</var> and <var>documentOrigin</var> is
   <span>same origin</span> with <var>responseOrigin</var>, then return true.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  </div>


  <h5 id="the-coop-headers">The headers</h5>

  <p>A <code>Document</code>'s <span data-x="concept-document-coop">cross-origin opener
  policy</span> is derived from the `<dfn http-header
  id="cross-origin-opener-policy-2"><code>Cross-Origin-Opener-Policy</code></dfn>` and `<dfn
  http-header><code>Cross-Origin-Opener-Policy-Report-Only</code></dfn>` HTTP response headers.
  These headers are <span data-x="http-structured-header">structured headers</span> whose value must
  be a <span data-x="http-structured-header-token">token</span>. <ref>STRUCTURED-FIELDS</ref></p>

  <p>The valid <span data-x="http-structured-header-token">token</span> values are the <span
  data-x="opener policy value">opener policy values</span>. The token may also have
  attached <span data-x="http-structured-header-parameters">parameters</span>; of these, the "<dfn
  data-x="coop-report-to"><code>report-to</code></dfn>" parameter can have a <span>valid URL
  string</span> identifying an appropriate reporting endpoint. <ref>REPORTING</ref></p>

  <p class="note"><span w-nodev>Per the processing model described below, user</span><span
  w-dev>User</span> agents will ignore this header if it contains an invalid value. Likewise, user
  agents will ignore this header if the value cannot be parsed as a <span
  data-x="http-structured-header-token">token</span>.</p>

  <div w-nodev>

  <hr>

  <p>To <dfn data-x="obtain-coop">obtain an opener policy</dfn> given a <span
  data-x="concept-response">response</span> <var>response</var> and an <span>environment</span>
  <var>reservedEnvironment</var>:</p>

  <ol>
   <li><p>Let <var>policy</var> be a new <span>opener policy</span>.</p></li>

   <li><p>If <var>reservedEnvironment</var> is a <span>non-secure context</span>, then return
   <var>policy</var>.</p></li>

   <li><p>Let <var>parsedItem</var> be the result of <span>getting a structured field value</span>
   given `<code>Cross-Origin-Opener-Policy</code>` and "<code data-x="">item</code>" from
   <var>response</var>'s <span data-x="concept-response-header-list">header list</span>.</p></li>

   <li>
    <p>If <var>parsedItem</var> is not null, then:</p>

    <ol>
     <li>
      <p>If <var>parsedItem</var>[0] is "<code data-x="coop-same-origin">same-origin</code>",
      then:</p>

      <ol>
       <li><p>Let <var>coep</var> be the result of <span data-x="obtain an embedder
       policy">obtaining a cross-origin embedder policy</span> from <var>response</var> and
       <var>reservedEnvironment</var>.</p></li>

       <li><p>If <var>coep</var>'s <span data-x="embedder-policy-value">value</span> is
       <span>compatible with cross-origin isolation</span>, then set <var>policy</var>'s <span
       data-x="coop-struct-value">value</span> to "<code
       data-x="coop-same-origin-plus-COEP">same-origin-plus-COEP</code>".</p></li>

       <li><p>Otherwise, set <var>policy</var>'s <span data-x="coop-struct-value">value</span> to
       "<code data-x="coop-same-origin">same-origin</code>".</p></li>
      </ol>
     </li>

     <li><p>If <var>parsedItem</var>[0] is "<code
     data-x="coop-same-origin-allow-popups">same-origin-allow-popups</code>", then set
     <var>policy</var>'s <span data-x="coop-struct-value">value</span> to "<code
     data-x="coop-same-origin-allow-popups">same-origin-allow-popups</code>".</p></li>

     <li><p>If <var>parsedItem</var>[0] is "<code
     data-x="coop-noopener-allow-popups">noopener-allow-popups</code>", then set
     <var>policy</var>'s <span data-x="coop-struct-value">value</span> to "<code
     data-x="coop-noopener-allow-popups">noopener-allow-popups</code>".</p></li>

     <li><p>If <var>parsedItem</var>[1]["<code data-x="coop-report-to">report-to</code>"] <span
     data-x="map exists">exists</span> and it is a string, then set <var>policy</var>'s <span
     data-x="coop-struct-report-endpoint">reporting endpoint</span> to
     <var>parsedItem</var>[1]["<code data-x="coop-report-to">report-to</code>"].</p></li>
    </ol>
   </li>

   <li><p>Set <var>parsedItem</var> to the result of <span>getting a structured field value</span>
   given `<code>Cross-Origin-Opener-Policy-Report-Only</code>` and "<code data-x="">item</code>"
   from <var>response</var>'s <span data-x="concept-response-header-list">header
   list</span>.</p></li>

   <li>
    <p>If <var>parsedItem</var> is not null, then:</p>

    <ol>
     <li>
      <p>If <var>parsedItem</var>[0] is "<code data-x="coop-same-origin">same-origin</code>",
      then:</p>

      <ol>
       <li><p>Let <var>coep</var> be the result of <span data-x="obtain an embedder
       policy">obtaining a cross-origin embedder policy</span> from <var>response</var> and
       <var>reservedEnvironment</var>.</p></li>

       <li>
        <p>If <var>coep</var>'s <span data-x="embedder-policy-value">value</span> is
        <span>compatible with cross-origin isolation</span> or <var>coep</var>'s <span
        data-x="embedder-policy-report-only-value">report-only value</span> is <span>compatible
        with cross-origin isolation</span>, then set <var>policy</var>'s <span
        data-x="coop-struct-report-only-value">report-only value</span> to "<code
        data-x="coop-same-origin-plus-COEP">same-origin-plus-COEP</code>".</p>

        <p class="note">Report only COOP also considers report-only COEP to assign the special
        "<code data-x="coop-same-origin-plus-COEP">same-origin-plus-COEP</code>" value. This allows
        developers more freedom in the order of deployment of COOP and COEP.</p>
       </li>

       <li><p>Otherwise, set <var>policy</var>'s <span
       data-x="coop-struct-report-only-value">report-only value</span> to "<code
       data-x="coop-same-origin">same-origin</code>".</p></li>
      </ol>
     </li>

     <li><p>If <var>parsedItem</var>[0] is "<code
     data-x="coop-same-origin-allow-popups">same-origin-allow-popups</code>", then set
     <var>policy</var>'s <span data-x="coop-struct-report-only-value">report-only value</span> to
     "<code data-x="coop-same-origin-allow-popups">same-origin-allow-popups</code>".</p></li>

     <li><p>If <var>parsedItem</var>[1]["<code data-x="coop-report-to">report-to</code>"] <span
     data-x="map exists">exists</span> and it is a string, then set <var>policy</var>'s <span
     data-x="coop-struct-report-only-endpoint">report-only reporting endpoint</span> to
     <var>parsedItem</var>[1]["<code data-x="coop-report-to">report-to</code>"].</p></li>
    </ol>
   </li>

   <li><p>Return <var>policy</var>.</p></li>
  </ol>


  <h5 id="browsing-context-group-switches-due-to-cross-origin-opener-policy">Browsing context group
  switches due to opener policy</h5>

  <p>To <dfn data-x="check-browsing-context-group-switch-coop-value-popup">check if popup COOP
  values require a browsing context group switch</dfn>, given two <span
  data-x="origin">origins</span> <var>responseOrigin</var> and
  <var>activeDocumentNavigationOrigin</var>, and two <span data-x="coop-struct-value">opener policy
  values</span> <var>responseCOOPValue</var> and <var>activeDocumentCOOPValue</var>:</p>
  <ol>
   <li><p><var>responseCOOPValue</var> is "<code
   data-x="coop-noopener-allow-popups">noopener-allow-popups</code>", then return true.</p></li>

   <li>
    <p>If all of the following are true:</p>

    <ul>
     <li><p><var>activeDocumentCOOPValue</var>'s <span data-x="coop-struct-value">value</span> is
     "<code data-x="coop-same-origin-allow-popups">same-origin-allow-popups</code>" or
     "<code data-x="coop-noopener-allow-popups">noopener-allow-popups</code>"; and</p></li>

     <li><p><var>responseCOOPValue</var> is "<code
     data-x="coop-unsafe-none">unsafe-none</code>",</p></li>
    </ul>

    <p>then return false.</p>
   </li>

   <li><p>If the result of <span data-x="matching-coop">matching</span>
   <var>activeDocumentCOOPValue</var>, <var>activeDocumentNavigationOrigin</var>,
   <var>responseCOOPValue</var>, and <var>responseOrigin</var> is true, then return false.</p></li>

   <li><p>Return true.</p>
  </ol>

  <p>To <dfn data-x="check-browsing-context-group-switch-coop-value">check if COOP values require a
  browsing context group switch</dfn>, given a boolean <var>isInitialAboutBlank</var>, two <span
  data-x="origin">origins</span> <var>responseOrigin</var> and
  <var>activeDocumentNavigationOrigin</var>, and two <span data-x="coop-struct-value">opener policy
  values</span> <var>responseCOOPValue</var> and <var>activeDocumentCOOPValue</var>:</p>

  <ol>
   <li><p>If <var>isInitialAboutBlank</var> is true, then return the result of <span
   data-x="check-browsing-context-group-switch-coop-value-popup">checking if popup COOP values
   requires a browsing context group switch</span> with <var>responseOrigin</var>,
   <var>activeDocumentNavigationOrigin</var>, <var>responseCOOPValue</var>, and
   <var>activeDocumentCOOPValue</var>.</p></li>

   <li>
    <p class="note">Here we are dealing with a non-popup navigation.</p>

    <p>If the result of <span data-x="matching-coop">matching</span>
    <var>activeDocumentCOOPValue</var>, <var>activeDocumentNavigationOrigin</var>,
    <var>responseCOOPValue</var>, and <var>responseOrigin</var> is true, then return false.</p>
   </li>

   <li><p>Return true.</p>
  </ol>

  <p>To <dfn data-x="check-bcg-switch-navigation-report-only">check if enforcing report-only COOP
  would require a browsing context group switch</dfn>, given a boolean
  <var>isInitialAboutBlank</var>, two <span data-x="origin">origins</span>
  <var>responseOrigin</var>, <var>activeDocumentNavigationOrigin</var>, and two <span data-x="opener
  policy">opener policies</span> <var>responseCOOP</var> and <var>activeDocumentCOOP</var>:</p>

  <ol>
   <li>
    <p>If the result of <span data-x="check-browsing-context-group-switch-coop-value">checking if
    COOP values require a browsing context group switch</span> given
    <var>isInitialAboutBlank</var>, <var>responseOrigin</var>,
    <var>activeDocumentNavigationOrigin</var>, <var>responseCOOP</var>'s <span
    data-x="coop-struct-report-only-value">report-only value</span> and
    <var>activeDocumentCOOPReportOnly</var>'s <span
    data-x="coop-struct-report-only-value">report-only value</span> is false, then return
    false.</p>

    <p class="note">Matching report-only policies allows a website to specify the same report-only
    opener policy on all its pages and not receive violation reports for navigations between these
    pages.</p>
   </li>

   <li><p>If the result of <span data-x="check-browsing-context-group-switch-coop-value">checking
   if COOP values require a browsing context group switch</span> given
   <var>isInitialAboutBlank</var>, <var>responseOrigin</var>,
   <var>activeDocumentNavigationOrigin</var>, <var>responseCOOP</var>'s <span
   data-x="coop-struct-value">value</span> and <var>activeDocumentCOOPReportOnly</var>'s <span
   data-x="coop-struct-report-only-value">report-only value</span> is true, then return
   true.</p></li>

   <li><p>If the result of <span data-x="check-browsing-context-group-switch-coop-value">checking
   if COOP values require a browsing context group switch</span> given
   <var>isInitialAboutBlank</var>, <var>responseOrigin</var>,
   <var>activeDocumentNavigationOrigin</var>, <var>responseCOOP</var>'s <span
   data-x="coop-struct-report-only-value">report-only value</span> and
   <var>activeDocumentCOOPReportOnly</var>'s <span data-x="coop-struct-value">value</span> is true,
   then return true.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  <p>An <dfn data-x="coop-enforcement-result">opener policy enforcement result</dfn> is a
  <span>struct</span> with the following <span data-x="struct item">items</span>:</p>

  <ul>
   <li><p>A boolean <dfn data-x="coop-enforcement-bcg-switch">needs a browsing context group
   switch</dfn>, initially false.</p></li>

   <li><p>A boolean <dfn data-x="coop-enforcement-bcg-switch-report-only">would need a browsing
   context group switch due to report-only</dfn>, initially false.</p></li>

   <li><p>A <span>URL</span> <dfn data-x="coop-enforcement-url">url</dfn>.</p></li>

   <li><p>An <span>origin</span> <dfn data-x="coop-enforcement-origin">origin</dfn>.</p></li>

   <li><p>An <span>opener policy</span> <dfn data-x="coop-enforcement-coop">opener
   policy</dfn>.</p></li>

   <li><p>A boolean <dfn data-x="coop-enforcement-source">current context is navigation
   source</dfn>, initially false.</p></li>
  </ul>

  <p>To <dfn data-x="coop-enforce">enforce a response's opener policy</dfn>, given a <span>browsing
  context</span> <var>browsingContext</var>, a <span>URL</span> <var>responseURL</var>, an
  <span>origin</span> <var>responseOrigin</var>, an <span>opener policy</span>
  <var>responseCOOP</var>, an <span data-x="coop-enforcement-result">opener policy enforcement
  result</span> <var>currentCOOPEnforcementResult</var>, and a <span
  data-x="concept-request-referrer">referrer</span> <var>referrer</var>:</p>

  <ol>
   <li>
    <p>Let <var>newCOOPEnforcementResult</var> be a new <span
    data-x="coop-enforcement-result">opener policy enforcement result</span> with</p>

    <dl class="props">
     <dt><span data-x="coop-enforcement-bcg-switch">needs a browsing context group switch</span></dt>
     <dd><var>currentCOOPEnforcementResult</var>'s <span data-x="coop-enforcement-bcg-switch">needs
     a browsing context group switch</span></dd>

     <dt><span data-x="coop-enforcement-bcg-switch-report-only">would need a browsing context group switch due to report-only</span></dt>
     <dd><var>currentCOOPEnforcementResult</var>'s <span
     data-x="coop-enforcement-bcg-switch-report-only">would need a browsing context group switch
     due to report-only</span></dd>

     <dt><span data-x="coop-enforcement-url">url</span></dt>
     <dd><var>responseURL</var></dd>

     <dt><span data-x="coop-enforcement-origin">origin</span></dt>
     <dd><var>responseOrigin</var></dd>

     <dt><span data-x="coop-enforcement-coop">opener policy</span></dt>
     <dd><var>responseCOOP</var></dd>

     <dt><span data-x="coop-enforcement-source">current context is navigation source</span></dt>
     <dd>true</dd>
    </dl>
   </li>

   <li><p>Let <var>isInitialAboutBlank</var> be <var>browsingContext</var>'s <span>active
   document</span>'s <span>is initial <code>about:blank</code></span>.</p></li>

   <li><p>If <var>isInitialAboutBlank</var> is true and <var>browsingContext</var>'s <span
   data-x="browsing-context-initial-url">initial URL</span> is null, set
   <var>browsingContext</var>'s <span data-x="browsing-context-initial-url">initial URL</span> to
   <var>responseURL</var>.</p></li>

   <li>
    <p>If the result of <span data-x="check-browsing-context-group-switch-coop-value">checking if
    COOP values require a browsing context group switch</span> given
    <var>isInitialAboutBlank</var>, <var>currentCOOPEnforcementResult</var>'s <span
    data-x="coop-enforcement-coop">opener policy</span>'s <span
    data-x="coop-struct-value">value</span>, <var>currentCOOPEnforcementResult</var>'s <span
    data-x="coop-enforcement-origin">origin</span>, <var>responseCOOP</var>'s <span
    data-x="coop-struct-value">value</span>, and <var>responseOrigin</var> is true, then:</p>

    <ol>
     <li><p>Set <var>newCOOPEnforcementResult</var>'s <span
     data-x="coop-enforcement-bcg-switch">needs a browsing context group switch</span> to
     true.</p></li>

     <li>
      <p>If <var>browsingContext</var>'s <span data-x="tlbc group">group</span>'s <span>browsing
      context set</span>'s <span data-x="list size">size</span> is greater than 1, then:</p>

      <ol>
       <li><p><span data-x="coop-violation-navigation-to">Queue a violation report for browsing
       context group switch when navigating to a COOP response</span> with <var>responseCOOP</var>,
       "<code data-x="">enforce</code>", <var>responseURL</var>,
       <var>currentCOOPEnforcementResult</var>'s <span data-x="coop-enforcement-url">url</span>,
       <var>currentCOOPEnforcementResult</var>'s <span
       data-x="coop-enforcement-origin">origin</span>, <var>responseOrigin</var>, and
       <var>referrer</var>.</p></li>

       <li><p><span data-x="coop-violation-navigation-from">Queue a violation report for browsing
       context group switch when navigating away from a COOP response</span> with
       <var>currentCOOPEnforcementResult</var>'s <span data-x="coop-enforcement-coop">opener
       policy</span>, "<code data-x="">enforce</code>", <var>currentCOOPEnforcementResult</var>'s
       <span data-x="coop-enforcement-url">url</span>, <var>responseURL</var>,
       <var>currentCOOPEnforcementResult</var>'s <span
       data-x="coop-enforcement-origin">origin</span>, <var>responseOrigin</var>, and
       <var>currentCOOPEnforcementResult</var>'s <span data-x="coop-enforcement-source">current
       context is navigation source</span>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li>
    <p>If the result of <span data-x="check-bcg-switch-navigation-report-only">checking if
    enforcing report-only COOP would require a browsing context group switch</span> given
    <var>isInitialAboutBlank</var>, <var>responseOrigin</var>,
    <var>currentCOOPEnforcementResult</var>'s <span data-x="coop-enforcement-origin">origin</span>,
    <var>responseCOOP</var>, and <var>currentCOOPEnforcementResult</var>'s <span
    data-x="coop-enforcement-coop">opener policy</span>, is true, then:</p>

    <ol>
     <li><p>Set <var>newCOOPEnforcementResult</var>'s <span data-x="coop-enforcement-bcg-switch-report-only">would
     need a browsing context group switch due to report-only</span> to true.</p></li>

     <li>
      <p>If <var>browsingContext</var>'s <span data-x="tlbc group">group</span>'s <span>browsing
      context set</span>'s <span data-x="list size">size</span> is greater than 1, then:</p>

      <ol>
       <li><p><span data-x="coop-violation-navigation-to">Queue a violation report for browsing
       context group switch when navigating to a COOP response</span> with <var>responseCOOP</var>,
       "<code data-x="">reporting</code>", <var>responseURL</var>,
       <var>currentCOOPEnforcementResult</var>'s <span data-x="coop-enforcement-url">url</span>,
       <var>currentCOOPEnforcementResult</var>'s <span
       data-x="coop-enforcement-origin">origin</span>, <var>responseOrigin</var>, and
       <var>referrer</var>.</p></li>

       <li><p><span data-x="coop-violation-navigation-from">Queue a violation report for browsing
       context group switch when navigating away from a COOP response</span> with
       <var>currentCOOPEnforcementResult</var>'s <span data-x="coop-enforcement-coop">opener
       policy</span>, "<code data-x="">reporting</code>",
       <var>currentCOOPEnforcementResult</var>'s <span data-x="coop-enforcement-url">url</span>,
       <var>responseURL</var>, <var>currentCOOPEnforcementResult</var>'s <span
       data-x="coop-enforcement-origin">origin</span>, <var>responseOrigin</var>, and
       <var>currentCOOPEnforcementResult</var>'s <span data-x="coop-enforcement-source">current
       context is navigation source</span>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Return <var>newCOOPEnforcementResult</var>.</p></li>
  </ol>

  <p>To <dfn data-x="obtain-browsing-context-navigation">obtain a browsing context to use for a
  navigation response</dfn>, given <span>navigation params</span> <var>navigationParams</var>:</p>

  <ol>
   <li><p>Let <var>browsingContext</var> be <var>navigationParams</var>'s <span
   data-x="navigation-params-navigable">navigable</span>'s <span data-x="nav-bc">active browsing
   context</span>.</p></li>

   <li><p>If <var>browsingContext</var> is not a <span>top-level browsing context</span>, then
   return <var>browsingContext</var>.</p></li>

   <li><p>Let <var>coopEnforcementResult</var> be <var>navigationParams</var>'s <span
   data-x="navigation-params-coop-enforcement-result">COOP enforcement result</span>.</p></li>

   <li><p>Let <var>swapGroup</var> be <var>coopEnforcementResult</var>'s <span
   data-x="coop-enforcement-bcg-switch">needs a browsing context group switch</span>.</p></li>

   <li><p>Let <var>sourceOrigin</var> be <var>browsingContext</var>'s
   <span>active document</span>'s <span data-x="concept-document-origin">origin</span>.</p></li>

   <li><p>Let <var>destinationOrigin</var> be <var>navigationParams</var>'s <span
   data-x="navigation-params-origin">origin</span>.</p></li>

   <li>
    <p>If <var>sourceOrigin</var> is not <span>same site</span> with
    <var>destinationOrigin</var>:</p>

    <ol>
     <li>
      <p>If either of <var>sourceOrigin</var> or <var>destinationOrigin</var> have a <span
      data-x="concept-origin-scheme">scheme</span> that is not an <span>HTTP(S) scheme</span> and
      the user agent considers it necessary for <var>sourceOrigin</var> and
      <var>destinationOrigin</var> to be isolated from each other (for
      <span>implementation-defined</span> reasons), optionally set <var>swapGroup</var> to true.</p>

      <p class="note">For example, if a user navigates from <code data-x="">about:settings</code> to
      <code data-x="">https://example.com</code>, the user agent could force a swap.</p>

      <p class="XXX"><a href="https://github.com/whatwg/html/issues/10842">Issue #10842</a> tracks
      settling on an interoperable behavior here, instead of letting this be optional.</p>
     </li>

     <li>
      <p>If <var>navigationParams</var>'s <span data-x="navigation-params-user-involvement"> user
      involvement</span> is "<code data-x="uni-browser-ui">browser UI</code>", optionally set
      <var>swapGroup</var> to true.</p>

      <p class="XXX"><a href="https://github.com/whatwg/html/issues/6356">Issue #6356</a> tracks
      settling on an interoperable behavior here, instead of letting this be optional.</p>
     </li>
    </ol>
   </li>

   <li>
    <p>If <var>browsingContext</var>'s <span data-x="tlbc group">group</span>'s <span>browsing
    context set</span>'s <span data-x="list size">size</span> is 1, optionally set
    <var>swapGroup</var> to true.</p>

    <p class="note">Some implementations swap browsing context groups here for performance
    reasons.</p>

    <p class="note">The check for other contexts that could script this one is not sufficient to
    prevent differences in behavior that could affect a web page. Even if there are currently no
    other contexts, the destination page could open a window, then if the user navigates back, the
    previous page could expect to be able to script the opened window. Doing a swap here would break
    that use case.</p>
   </li>

   <li>
    <p>If <var>swapGroup</var> is false, then:</p>

    <ol>
     <li><p>If <var>coopEnforcementResult</var>'s <span
     data-x="coop-enforcement-bcg-switch-report-only">would need a browsing context group switch
     due to report-only</span> is true, set <var>browsingContext</var>'s <span
     data-x="virtual-browsing-context-group-id">virtual browsing context group ID</span> to a new
     unique identifier.</p></li>

     <li><p>Return <var>browsingContext</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>Let <var>newBrowsingContext</var> be the first return value of <span>creating a new
    top-level browsing context and document</span>.</p>

    <p class="note">In this case we are going to perform a browsing context group swap.
    <var>browsingContext</var> will not be used by the new <code>Document</code> that we are about
    to <span data-x="create-the-document-object">create</span>. If it is not used by other
    <code>Document</code>s either (such as ones in the back/forward cache), then the user agent
    might <a href="#a-browsing-context-is-discarded">destroy it</a> at this point.</p>
   </li>

   <li><p>Let <var>navigationCOOP</var> be <var>navigationParams</var>'s <span
   data-x="navigation-params-coop">cross-origin opener policy</span>.</p></li>

   <li>
    <p>If <var>navigationCOOP</var>'s <span data-x="coop-struct-value">value</span> is "<code
    data-x="coop-same-origin-plus-COEP">same-origin-plus-COEP</code>", then set
    <var>newBrowsingContext</var>'s <span data-x="tlbc group">group</span>'s <span
    data-x="bcg-cross-origin-isolation">cross-origin isolation mode</span> to either "<code
    data-x="cross-origin-isolation-logical">logical</code>" or "<code
    data-x="cross-origin-isolation-concrete">concrete</code>". The choice of which is
    <span>implementation-defined</span>.</p>

    <p class="note">It is difficult on some platforms to provide the security properties required
    by the <span data-x="concept-settings-object-cross-origin-isolated-capability">cross-origin
    isolated capability</span>. "<code data-x="cross-origin-isolation-concrete">concrete</code>"
    grants access to it and "<code data-x="cross-origin-isolation-logical">logical</code>" does
    not.</p>
   </li>

   <li><p>Let <var>sandboxFlags</var> be a <span data-x="list clone">clone</span> of
   <var>navigationParams</var>'s <span
   data-x="navigation-params-sandboxing">final sandboxing flag set</span>.</p></li>

   <li>
    <p>If <var>sandboxFlags</var> is not empty, then:</p>
    <ol>
     <li><p><span>Assert</span>: <var>navigationCOOP</var>'s <span data-x="coop-struct-value">value</span> is
     "<code data-x="coop-unsafe-none">unsafe-none</code>".</p></li>

     <li><p><span>Assert</span>: <var>newBrowsingContext</var>'s <span>popup sandboxing flag
     set</span> <span data-x="list is empty">is empty</span>.</p></li>

     <li><p>Set <var>newBrowsingContext</var>'s <span>popup sandboxing flag set</span> to
     <var>sandboxFlags</var>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>newBrowsingContext</var>.</p></li>
  </ol>


  <h5 id="coop-reporting"><span id="reporting"></span>Reporting</h5>

  <p>An <dfn>accessor-accessed relationship</dfn> is an enum that describes the relationship
  between two <span data-x="browsing context">browsing contexts</span> between which an access
  happened. It can take the following values:</p>

  <dl>
   <dt><dfn data-x="accessor-accessed-opener">accessor is opener</dfn></dt>
   <dd><p>The accessor <span>browsing context</span> or one of its <span data-x="ancestor browsing
   context">ancestors</span> is the <span>opener browsing context</span> of the accessed
   <span>browsing context</span>'s <span data-x="bc-tlbc">top-level browsing
   context</span>.</p></dd>

   <dt><dfn data-x="accessor-accessed-openee">accessor is openee</dfn></dt>
   <dd><p>The accessed <span>browsing context</span> or one of its <span data-x="ancestor browsing
   context">ancestors</span> is the <span>opener browsing context</span> of the accessor
   <span>browsing context</span>'s <span data-x="bc-tlbc">top-level browsing
   context</span>.</p></dd>

   <dt><dfn data-x="accessor-accessed-none">none</dfn></dt>
   <dd><p>There is no opener relationship between the accessor <span>browsing context</span>, the
   accessor <span>browsing context</span>, or any of their <span data-x="ancestor browsing
   context">ancestors</span>.</p></dd>
  </dl>

  <p>To <dfn data-x="coop-check-access-report">check if an access between two browsing contexts
  should be reported</dfn>, given two <span data-x="browsing context">browsing contexts</span>
  <var>accessor</var> and <var>accessed</var>, a JavaScript property name <var>P</var>, and an
  <span>environment settings object</span> <var>environment</var>:</p>

  <ol>
   <li><p>If <var>P</var> is not a <span>cross-origin accessible window property name</span>, then
   return.</p></li>

   <li><p><span>Assert</span>: <var>accessor</var>'s <span>active document</span> and
   <var>accessed</var>'s <span>active document</span> are both <span>fully active</span>.</p></li>

   <li><p>Let <var>accessorTopDocument</var> be <var>accessor</var>'s <span
   data-x="bc-tlbc">top-level browsing context</span>'s <span>active document</span>.</p></li>

   <li><p>Let <var>accessorInclusiveAncestorOrigins</var> be the list obtained by taking the <span
   data-x="concept-document-origin">origin</span> of the <span data-x="nav-document">active
   document</span> of each of <var>accessor</var>'s <span>active document</span>'s <span>inclusive
   ancestor navigables</span>.</p></li>

   <li><p>Let <var>accessedTopDocument</var> be <var>accessed</var>'s <span
   data-x="bc-tlbc">top-level browsing context</span>'s <span>active document</span>.</p></li>

   <li><p>Let <var>accessedInclusiveAncestorOrigins</var> be the list obtained by taking the <span
   data-x="concept-document-origin">origin</span> of the <span data-x="nav-document">active
   document</span> of each of <var>accessed</var>'s <span>active document</span>'s <span>inclusive
   ancestor navigables</span>.</p></li>

   <li>
    <p>If any of <var>accessorInclusiveAncestorOrigins</var> are not <span>same origin</span> with
    <var>accessorTopDocument</var>'s <span data-x="concept-document-origin">origin</span>, or if
    any of <var>accessedInclusiveAncestorOrigins</var> are not <span>same origin</span> with
    <var>accessedTopDocument</var>'s <span data-x="concept-document-origin">origin</span>, then
    return.</p>

    <p class="note">This avoids leaking information about cross-origin iframes to a top level frame
    with opener policy reporting.</p>
   </li>

   <li><p>If <var>accessor</var>'s <span data-x="bc-tlbc">top-level browsing context</span>'s <span
   data-x="virtual-browsing-context-group-id">virtual browsing context group ID</span> is
   <var>accessed</var>'s <span data-x="bc-tlbc">top-level browsing context</span>'s <span
   data-x="virtual-browsing-context-group-id">virtual browsing context group ID</span>, then
   return.</p></li>

   <li><p>Let <var>accessorAccessedRelationship</var> be a new <span>accessor-accessed
   relationship</span> with value <span data-x="accessor-accessed-none">none</span>.</p></li>

   <li><p>If <var>accessed</var>'s <span data-x="bc-tlbc">top-level browsing context</span>'s
   <span>opener browsing context</span> is <var>accessor</var> or is an <span data-x="ancestor
   browsing context">ancestor</span> of <var>accessor</var>, then set
   <var>accessorAccessedRelationship</var> to <span data-x="accessor-accessed-opener">accessor is
   opener</span>.</p></li>

   <li><p>If <var>accessor</var>'s <span data-x="bc-tlbc">top-level browsing context</span>'s
   <span>opener browsing context</span> is <var>accessed</var> or is an <span data-x="ancestor
   browsing context">ancestor</span> of <var>accessed</var>, then set
   <var>accessorAccessedRelationship</var> to <span data-x="accessor-accessed-openee">accessor is
   openee</span>.</p></li>

   <li><p><span data-x="coop-violation-access">Queue violation reports for accesses</span>, given
   <var>accessorAccessedRelationship</var>, <var>accessorTopDocument</var>'s <span
   data-x="concept-document-coop">opener policy</span>, <var>accessedTopDocument</var>'s <span
   data-x="concept-document-coop">opener policy</span>, <var>accessor</var>'s <span>active
   document</span>'s <span data-x="concept-document-url">URL</span>, <var>accessed</var>'s
   <span>active document</span>'s <span data-x="concept-document-url">URL</span>,
   <var>accessor</var>'s <span data-x="bc-tlbc">top-level browsing context</span>'s <span
   data-x="browsing-context-initial-url">initial URL</span>, <var>accessed</var>'s <span
   data-x="bc-tlbc">top-level browsing context</span>'s <span
   data-x="browsing-context-initial-url">initial URL</span>, <var>accessor</var>'s <span>active
   document</span>'s <span data-x="concept-document-origin">origin</span>, <var>accessed</var>'s
   <span>active document</span>'s <span data-x="concept-document-origin">origin</span>,
   <var>accessor</var>'s <span data-x="bc-tlbc">top-level browsing context</span>'s <span
   data-x="opener-origin-at-creation">opener origin at creation</span>, <var>accessed</var>'s <span
   data-x="bc-tlbc">top-level browsing context</span>'s <span
   data-x="opener-origin-at-creation">opener origin at creation</span>,
   <var>accessorTopDocument</var>'s <span data-x="dom-document-referrer">referrer</span>,
   <var>accessedTopDocument</var>'s <span data-x="dom-document-referrer">referrer</span>,
   <var>P</var>, and <var>environment</var>.</p></li>
  </ol>

  <p>To <dfn data-x="sanitize-url-report">sanitize a URL to send in a report</dfn> given a
  <span>URL</span> <var>url</var>:</p>

  <ol>
   <li><p>Let <var>sanitizedURL</var> be a copy of <var>url</var>.</p></li>

   <li><p><span data-x="set the username">Set the username</span> given <var>sanitizedURL</var> and
   the empty string.

   <li><p><span data-x="set the password">Set the password</span> given <var>sanitizedURL</var> and
   the empty string.

   <li><p>Return the <span data-x="concept-url-serializer">serialization</span> of
   <var>sanitizedURL</var> with <span data-x="url serializer exclude fragment"><i>exclude
   fragment</i></span> set to true.</p></li>
  </ol>

  <p>To <dfn data-x="coop-violation-navigation-to">queue a violation report for browsing context
  group switch when navigating to a COOP response</dfn> given an <span data-x="opener policy">opener
  policy</span> <var>coop</var>, a string <var>disposition</var>, a <span>URL</span>
  <var>coopURL</var>, a <span>URL</span> <var>previousResponseURL</var>, two <span
  data-x="origin">origins</span> <var>coopOrigin</var> and <var>previousResponseOrigin</var>, and a
  <span data-x="concept-request-referrer">referrer</span> <var>referrer</var>:</p>

  <ol>
   <li><p>If <var>coop</var>'s <span data-x="coop-struct-report-endpoint">reporting endpoint</span>
   is null, return.</p></li>

   <li><p>Let <var>coopValue</var> be <var>coop</var>'s <span
   data-x="coop-struct-value">value</span>.</p></li>

   <li><p>If <var>disposition</var> is "<code data-x="">reporting</code>", then set
   <var>coopValue</var> to <var>coop</var>'s <span
   data-x="coop-struct-report-only-value">report-only value</span>.</p></li>

   <li><p>Let <var>serializedReferrer</var> be an empty string.</p></li>

   <li><p>If <var>referrer</var> is a <span>URL</span>, set <var>serializedReferrer</var> to the
   <span data-x="concept-url-serializer">serialization</span> of <var>referrer</var>.

   <li>
    <p>Let <var>body</var> be a new object containing the following properties:</p>

    <table class="data">
     <thead>
      <tr>
       <th>key</th>
       <th>value</th>
      </tr>
     </thead>
     <tbody>
      <tr>
       <td>disposition</td>
       <td><var>disposition</var></td>
      </tr>
      <tr>
       <td>effectivePolicy</td>
       <td><var>coopValue</var></td>
      </tr>
      <tr>
       <td>previousResponseURL</td>
       <td>If <var>coopOrigin</var> and <var>previousResponseOrigin</var> are <span>same origin</span> this is the <span data-x="sanitize-url-report">sanitization</span> of <var>previousResponseURL</var>, null otherwise.</td>
      </tr>
      <tr>
       <td>referrer</td>
       <td><var>serializedReferrer</var></td>
      </tr>
      <tr>
       <td>type</td>
       <td>"<code data-x="">navigation-to-response</code>"</td>
      </tr>
     </tbody>
    </table>
   </li>

   <li><p><span data-x="queue a report">Queue</span> <var>body</var> as "<code
   data-x="">coop</code>" for <var>coop</var>'s <span
   data-x="coop-struct-report-endpoint">reporting endpoint</span> with <var>coopURL</var>.</p></li>
  </ol>

  <p>To <dfn data-x="coop-violation-navigation-from">queue a violation report for browsing context
  group switch when navigating away from a COOP response</dfn> given an <span data-x="opener
  policy">opener policy</span> <var>coop</var>, a string <var>disposition</var>, a <span>URL</span>
  <var>coopURL</var>, a <span>URL</span> <var>nextResponseURL</var>, two <span
  data-x="origin">origins</span> <var>coopOrigin</var> and <var>nextResponseOrigin</var>, and a
  boolean <var>isCOOPResponseNavigationSource</var>:</p>

  <ol>
   <li><p>If <var>coop</var>'s <span data-x="coop-struct-report-endpoint">reporting endpoint</span>
   is null, return.</p></li>

   <li><p>Let <var>coopValue</var> be <var>coop</var>'s <span
   data-x="coop-struct-value">value</span>.</p></li>

   <li><p>If <var>disposition</var> is "<code data-x="">reporting</code>", then set
   <var>coopValue</var> to <var>coop</var>'s <span
   data-x="coop-struct-report-only-value">report-only value</span>.</p></li>

   <li>
    <p>Let <var>body</var> be a new object containing the following properties:</p>

    <table class="data">
     <thead>
      <tr>
       <th>key</th>
       <th>value</th>
      </tr>
     </thead>
     <tbody>
      <tr>
       <td>disposition</td>
       <td><var>disposition</var></td>
      </tr>
      <tr>
       <td>effectivePolicy</td>
       <td><var>coopValue</var></td>
      </tr>
      <tr>
       <td>nextResponseURL</td>
       <td>If <var>coopOrigin</var> and <var>nextResponseOrigin</var> are <span>same origin</span>
       or <var>isCOOPResponseNavigationSource</var> is true, this is the <span
       data-x="sanitize-url-report">sanitization</span> of <var>nextResponseURL</var>, null otherwise.</td>
      </tr>
      <tr>
       <td>type</td>
       <td>"<code data-x="">navigation-from-response</code>"</td>
      </tr>
     </tbody>
    </table>
   </li>

   <li><p><span data-x="queue a report">Queue</span> <var>body</var> as "<code
   data-x="">coop</code>" for <var>coop</var>'s <span
   data-x="coop-struct-report-endpoint">reporting endpoint</span> with <var>coopURL</var>.</p></li>
  </ol>

  <p>To <dfn data-x="coop-violation-access">queue violation reports for accesses</dfn>, given an
  <span>accessor-accessed relationship</span> <var>accessorAccessedRelationship</var>, two <span
  data-x="opener policy">opener policies</span> <var>accessorCOOP</var> and <var>accessedCOOP</var>,
  four <span data-x="url">URLs</span> <var>accessorURL</var>, <var>accessedURL</var>,
  <var>accessorInitialURL</var>, <var>accessedInitialURL</var>, four <span
  data-x="origin">origins</span> <var>accessorOrigin</var>, <var>accessedOrigin</var>,
  <var>accessorCreatorOrigin</var> and <var>accessedCreatorOrigin</var>, two <span
  data-x="dom-document-referrer">referrers</span> <var>accessorReferrer</var> and
  <var>accessedReferrer</var>, a string <var>propertyName</var>, and an <span>environment settings
  object</span> <var>environment</var>:</p>

  <ol>
   <li><p>If <var>coop</var>'s <span data-x="coop-struct-report-endpoint">reporting endpoint</span>
   is null, return.</p></li>

   <li><p>Let <var>coopValue</var> be <var>coop</var>'s <span
   data-x="coop-struct-value">value</span>.</p></li>

   <li><p>If <var>disposition</var> is "<code data-x="">reporting</code>", then set
   <var>coopValue</var> to <var>coop</var>'s <span
   data-x="coop-struct-report-only-value">report-only value</span>.</p></li>

   <li>
    <p>If <var>accessorAccessedRelationship</var> is <span
    data-x="accessor-accessed-opener">accessor is opener</span>:</p>

    <ol>
     <li><p><span data-x="coop-violation-access-to-opened">Queue a violation report
     for access to an opened window</span>, given <var>accessorCOOP</var>, <var>accessorURL</var>,
     <var>accessedURL</var>, <var>accessedInitialURL</var>, <var>accessorOrigin</var>,
     <var>accessedOrigin</var>, <var>accessedCreatorOrigin</var>, <var>propertyName</var>,
     and <var>environment</var>.</p></li>

     <li><p><span data-x="coop-violation-access-from-opener">Queue a violation report for access
     from the opener</span>, given <var>accessedCOOP</var>, <var>accessedURL</var>,
     <var>accessorURL</var>, <var>accessedOrigin</var>, <var>accessorOrigin</var>,
     <var>propertyName</var>, and <var>accessedReferrer</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if <var>accessorAccessedRelationship</var> is <span
    data-x="accessor-accessed-openee">accessor is openee</span>:</p>

    <ol>
     <li><p><span data-x="coop-violation-access-to-opener">Queue a violation report for access to
     the opener</span>, given <var>accessorCOOP</var>, <var>accessorURL</var>,
     <var>accessedURL</var>, <var>accessorOrigin</var>, <var>accessedOrigin</var>,
     <var>propertyName</var>, <var>accessorReferrer</var>, and <var>environment</var>.</p></li>

     <li><p><span data-x="coop-violation-access-from-opened">Queue a violation report for access
     from an opened window</span>, given <var>accessedCOOP</var>, <var>accessedURL</var>,
     <var>accessorURL</var>, <var>accessorInitialURL</var>, <var>accessedOrigin</var>,
     <var>accessorOrigin</var>, <var>accessorCreatorOrigin</var>, and
     <var>propertyName</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p><span data-x="coop-violation-access-to-opened">Queue a violation report for
     access to another window</span>, given <var>accessorCOOP</var>, <var>accessorURL</var>,
     <var>accessedURL</var>, <var>accessorOrigin</var>, <var>accessedOrigin</var>,
     <var>propertyName</var>, and <var>environment</var></p></li>

     <li><p><span data-x="coop-violation-access-from-other">Queue a violation report for
     access from another window</span>, given <var>accessedCOOP</var>, <var>accessedURL</var>,
     <var>accessorURL</var>, <var>accessedOrigin</var>, <var>accessorOrigin</var>, and
     <var>propertyName</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>To <dfn data-x="coop-violation-access-to-opener">queue a violation report for access to the
  opener</dfn>, given an <span data-x="opener policy">opener policy</span> <var>coop</var>, two
  <span data-x="URL">URLs</span> <var>coopURL</var> and <var>openerURL</var>, two <span
  data-x="origin">origins</span> <var>coopOrigin</var> and <var>openerOrigin</var>, a string
  <var>propertyName</var>, a <span data-x="concept-request-referrer">referrer</span>
  <var>referrer</var>, and an <span>environment settings object</span> <var>environment</var>:</p>

  <ol>
   <li><p>Let <var>sourceFile</var>, <var>lineNumber</var> and <var>columnNumber</var> be the
   relevant script URL and problematic position which triggered this report.</p></li>

   <li><p>Let <var>serializedReferrer</var> be an empty string.</p></li>

   <li><p>If <var>referrer</var> is a <span>URL</span>, set <var>serializedReferrer</var> to the
   <span data-x="concept-url-serializer">serialization</span> of <var>referrer</var>.

   <li>
    <p>Let <var>body</var> be a new object containing the following properties:</p>

    <table class="data">
     <thead>
      <tr>
       <th>key</th>
       <th>value</th>
      </tr>
     </thead>
     <tbody>
      <tr>
       <td>disposition</td>
       <td>"<code data-x="">reporting</code>"</td>
      </tr>
      <tr>
       <td>effectivePolicy</td>
       <td><var>coop</var>'s <span data-x="coop-struct-report-only-value">report-only
       value</span></td>
      </tr>
      <tr>
       <td>property</td>
       <td><var>propertyName</var></td>
      </tr>
      <tr>
       <td>openerURL</td>
       <td>If <var>coopOrigin</var> and <var>openerOrigin</var> are <span>same origin</span>, this
       is the <span data-x="sanitize-url-report">sanitization</span> of <var>openerURL</var>, null
       otherwise.</td>
      </tr>
      <tr>
       <td>referrer</td>
       <td><var>serializedReferrer</var></td>
      </tr>
      <tr>
       <td>sourceFile</td>
       <td><var>sourceFile</var></td>
      </tr>
      <tr>
       <td>lineNumber</td>
       <td><var>lineNumber</var></td>
      </tr>
      <tr>
       <td>columnNumber</td>
       <td><var>columnNumber</var></td>
      </tr>
      <tr>
       <td>type</td>
       <td>"<code data-x="">access-to-opener</code>"</td>
      </tr>
     </tbody>
    </table>
   </li>

   <li><p><span data-x="queue a report">Queue</span> <var>body</var> as "<code
   data-x="">coop</code>" for <var>coop</var>'s <span
   data-x="coop-struct-report-endpoint">reporting endpoint</span> with <var>coopURL</var> and
   <var>environment</var>.</p></li>
  </ol>

  <p>To <dfn data-x="coop-violation-access-to-opened">queue a violation report for access to an
  opened window</dfn>, given an <span data-x="opener policy">opener policy</span> <var>coop</var>,
  three <span data-x="URL">URLs</span> <var>coopURL</var>, <var>openedWindowURL</var> and
  <var>initialWindowURL</var>, three <span data-x="origin">origins</span> <var>coopOrigin</var>,
  <var>openedWindowOrigin</var>, and <var>openerInitialOrigin</var>, a string
  <var>propertyName</var>, and an <span>environment settings object</span>
  <var>environment</var>:</p>

  <ol>
   <li><p>Let <var>sourceFile</var>, <var>lineNumber</var> and <var>columnNumber</var> be the
   relevant script URL and problematic position which triggered this report.</p></li>

   <li>
    <p>Let <var>body</var> be a new object containing the following properties:</p>

    <table class="data">
     <thead>
      <tr>
       <th>key</th>
       <th>value</th>
      </tr>
     </thead>
     <tbody>
      <tr>
       <td>disposition</td>
       <td>"<code data-x="">reporting</code>"</td>
      </tr>
      <tr>
       <td>effectivePolicy</td>
       <td><var>coop</var>'s <span data-x="coop-struct-report-only-value">report-only
       value</span></td>
      </tr>
      <tr>
       <td>property</td>
       <td><var>propertyName</var></td>
      </tr>
      <tr>
       <td>openedWindowURL</td>
       <td>If <var>coopOrigin</var> and <var>openedWindowOrigin</var> are <span>same origin</span>, this
       is the <span data-x="sanitize-url-report">sanitization</span> of <var>openedWindowURL</var>,
       null otherwise.</td>
      </tr>
      <tr>
       <td>openedWindowInitialURL</td>
       <td>If <var>coopOrigin</var> and <var>openerInitialOrigin</var> are <span>same origin</span>,
       this is the <span data-x="sanitize-url-report">sanitization</span> of
       <var>initialWindowURL</var>, null otherwise.</td>
      </tr>
      <tr>
       <td>sourceFile</td>
       <td><var>sourceFile</var></td>
      </tr>
      <tr>
       <td>lineNumber</td>
       <td><var>lineNumber</var></td>
      </tr>
      <tr>
       <td>columnNumber</td>
       <td><var>columnNumber</var></td>
      </tr>
      <tr>
       <td>type</td>
       <td>"<code data-x="">access-to-opener</code>"</td>
      </tr>
     </tbody>
    </table>
   </li>

   <li><p><span data-x="queue a report">Queue</span> <var>body</var> as "<code data-x="">coop</code>"
   for <var>coop</var>'s <span data-x="coop-struct-report-endpoint">reporting endpoint</span> with
   <var>coopURL</var> and <var>environment</var>.</p></li>
  </ol>


  <p>To <dfn data-x="coop-violation-access-to-other">queue a violation report for access to another
  window</dfn>, given an <span data-x="opener policy">opener policy</span> <var>coop</var>, two
  <span data-x="URL">URLs</span> <var>coopURL</var> and <var>otherURL</var>, two <span
  data-x="origin">origins</span> <var>coopOrigin</var> and <var>otherOrigin</var>, a string
  <var>propertyName</var>, and an <span>environment settings object</span>
  <var>environment</var>:</p>

  <ol>
   <li><p>Let <var>sourceFile</var>, <var>lineNumber</var> and <var>columnNumber</var> be the
   relevant script URL and problematic position which triggered this report.</p></li>

   <li>
    <p>Let <var>body</var> be a new object containing the following properties:</p>

    <table class="data">
     <thead>
      <tr>
       <th>key</th>
       <th>value</th>
      </tr>
     </thead>
     <tbody>
      <tr>
       <td>disposition</td>
       <td>"<code data-x="">reporting</code>"</td>
      </tr>
      <tr>
       <td>effectivePolicy</td>
       <td><var>coop</var>'s <span data-x="coop-struct-report-only-value">report-only
       value</span></td>
      </tr>
      <tr>
       <td>property</td>
       <td><var>propertyName</var></td>
      </tr>
      <tr>
       <td>otherURL</td>
       <td>If <var>coopOrigin</var> and <var>otherOrigin</var> are <span>same origin</span>, this
       is the <span data-x="sanitize-url-report">sanitization</span> of <var>otherURL</var>, null
       otherwise.</td>
      </tr>
      <tr>
       <td>sourceFile</td>
       <td><var>sourceFile</var></td>
      </tr>
      <tr>
       <td>lineNumber</td>
       <td><var>lineNumber</var></td>
      </tr>
      <tr>
       <td>columnNumber</td>
       <td><var>columnNumber</var></td>
      </tr>
      <tr>
       <td>type</td>
       <td>"<code data-x="">access-to-opener</code>"</td>
      </tr>
     </tbody>
    </table>
   </li>

   <li><p><span data-x="queue a report">Queue</span> <var>body</var> as "<code data-x="">coop</code>"
   for <var>coop</var>'s <span data-x="coop-struct-report-endpoint">reporting endpoint</span> with
   <var>coopURL</var> and <var>environment</var>.</p></li>
  </ol>

  <p>To <dfn data-x="coop-violation-access-from-opener">queue a violation report for access from the
  opener</dfn>, given an <span data-x="opener policy">opener policy</span> <var>coop</var>, two
  <span data-x="URL">URLs</span> <var>coopURL</var> and <var>openerURL</var>, two <span
  data-x="origin">origins</span> <var>coopOrigin</var> and <var>openerOrigin</var>, a string
  <var>propertyName</var>, and a <span data-x="concept-request-referrer">referrer</span>
  <var>referrer</var>:</p>

  <ol>
   <li><p>If <var>coop</var>'s <span data-x="coop-struct-report-endpoint">reporting endpoint</span>
   is null, return.</p></li>

   <li><p>Let <var>serializedReferrer</var> be an empty string.</p></li>

   <li><p>If <var>referrer</var> is a <span>URL</span>, set <var>serializedReferrer</var> to the
   <span data-x="concept-url-serializer">serialization</span> of <var>referrer</var>.

   <li>
    <p>Let <var>body</var> be a new object containing the following properties:</p>

    <table class="data">
     <thead>
      <tr>
       <th>key</th>
       <th>value</th>
      </tr>
     </thead>
     <tbody>
      <tr>
       <td>disposition</td>
       <td>"<code data-x="">reporting</code>"</td>
      </tr>
      <tr>
       <td>effectivePolicy</td>
       <td><var>coop</var>'s <span data-x="coop-struct-report-only-value">report-only
       value</span></td>
      </tr>
      <tr>
       <td>property</td>
       <td><var>propertyName</var></td>
      </tr>
      <tr>
       <td>openerURL</td>
       <td>If <var>coopOrigin</var> and <var>openerOrigin</var> are <span>same origin</span>, this
       is the <span data-x="sanitize-url-report">sanitization</span> of <var>openerURL</var>, null
       otherwise.</td>
      </tr>
      <tr>
       <td>referrer</td>
       <td><var>serializedReferrer</var></td>
      </tr>
      <tr>
       <td>type</td>
       <td>"<code data-x="">access-to-opener</code>"</td>
      </tr>
     </tbody>
    </table>
   </li>

   <li><p><span data-x="queue a report">Queue</span> <var>body</var> as "<code data-x="">coop</code>"
   for <var>coop</var>'s <span data-x="coop-struct-report-endpoint">reporting endpoint</span> with
   <var>coopURL</var>.</p></li>
  </ol>

  <p>To <dfn data-x="coop-violation-access-from-opened">queue a violation report for access from an
  opened window</dfn>, given an <span data-x="opener policy">opener policy</span> <var>coop</var>,
  three <span data-x="URL">URLs</span> <var>coopURL</var>, <var>openedWindowURL</var> and
  <var>initialWindowURL</var>, three <span data-x="origin">origins</span> <var>coopOrigin</var>,
  <var>openedWindowOrigin</var>, and <var>openerInitialOrigin</var>, and a string
  <var>propertyName</var>:</p>

  <ol>
   <li><p>If <var>coop</var>'s <span data-x="coop-struct-report-endpoint">reporting endpoint</span>
   is null, return.</p></li>

   <li>
    <p>Let <var>body</var> be a new object containing the following properties:</p>

    <table class="data">
     <thead>
      <tr>
       <th>key</th>
       <th>value</th>
      </tr>
     </thead>
     <tbody>
      <tr>
       <td>disposition</td>
       <td>"<code data-x="">reporting</code>"</td>
      </tr>
      <tr>
       <td>effectivePolicy</td>
       <td><var>coopValue</var></td>
      </tr>
      <tr>
       <td>property</td>
       <td><var>coop</var>'s <span data-x="coop-struct-report-only-value">report-only
       value</span></td>
      </tr>
      <tr>
       <td>openedWindowURL</td>
       <td>If <var>coopOrigin</var> and <var>openedWindowOrigin</var> are <span>same origin</span>, this
       is the <span data-x="sanitize-url-report">sanitization</span> of <var>openedWindowURL</var>,
       null otherwise.</td>
      </tr>
      <tr>
       <td>openedWindowInitialURL</td>
       <td>If <var>coopOrigin</var> and <var>openerInitialOrigin</var> are <span>same origin</span>,
       this is the <span data-x="sanitize-url-report">sanitization</span> of
       <var>initialWindowURL</var>, null otherwise.</td>
      </tr>
      <tr>
       <td>type</td>
       <td>"<code data-x="">access-to-opener</code>"</td>
      </tr>
     </tbody>
    </table>
   </li>

   <li><p><span data-x="queue a report">Queue</span> <var>body</var> as "<code data-x="">coop</code>"
   for <var>coop</var>'s <span data-x="coop-struct-report-endpoint">reporting endpoint</span> with
   <var>coopURL</var>.</p></li>
  </ol>


  <p>To <dfn data-x="coop-violation-access-from-other">queue a violation report for access from
  another window</dfn>, given an <span data-x="opener policy">opener policy</span> <var>coop</var>,
  two <span data-x="URL">URLs</span> <var>coopURL</var> and <var>otherURL</var>, two <span
  data-x="origin">origins</span> <var>coopOrigin</var> and <var>otherOrigin</var>, and a string
  <var>propertyName</var>:</p>

  <ol>
   <li><p>If <var>coop</var>'s <span data-x="coop-struct-report-endpoint">reporting endpoint</span>
   is null, return.</p></li>

   <li>
    <p>Let <var>body</var> be a new object containing the following properties:</p>

    <table class="data">
     <thead>
      <tr>
       <th>key</th>
       <th>value</th>
      </tr>
     </thead>
     <tbody>
      <tr>
       <td>disposition</td>
       <td>"<code data-x="">reporting</code>"</td>
      </tr>
      <tr>
       <td>effectivePolicy</td>
       <td><var>coop</var>'s <span data-x="coop-struct-report-only-value">report-only
       value</span></td>
      </tr>
      <tr>
       <td>property</td>
       <td><var>propertyName</var></td>
      </tr>
      <tr>
       <td>otherURL</td>
       <td>If <var>coopOrigin</var> and <var>otherOrigin</var> are <span>same origin</span>, this
       is the <span data-x="sanitize-url-report">sanitization</span> of <var>otherURL</var>, null
       otherwise.</td>
      </tr>
      <tr>
       <td>type</td>
       <td><code data-x="">access-to-opener</code></td>
      </tr>
     </tbody>
    </table>
   </li>

   <li><p><span data-x="queue a report">Queue</span> <var>body</var> as "<code data-x="">coop</code>"
   for <var>coop</var>'s <span data-x="coop-struct-report-endpoint">reporting endpoint</span> with
   <var>coopURL</var>.</p></li>
  </ol>

  </div>


  <h4 id="coep">Cross-origin embedder policies</h4>

  <p>An <dfn export>embedder policy value</dfn> is one of three strings that controls the fetching
  of cross-origin resources without explicit permission from resource owners.</p>

  <dl>
   <dt>"<dfn data-x="coep-unsafe-none" export for="embedder policy value"><code
   data-x="">unsafe-none</code></dfn>"</dt>
   <dd><p>This is the default value. When this value is used, cross-origin resources can be fetched
   without giving explicit permission through the <span>CORS protocol</span> or the
   `<code>Cross-Origin-Resource-Policy</code>` header.</p></dd>

   <dt>"<dfn data-x="coep-require-corp" export for="embedder policy value"><code
   data-x="">require-corp</code></dfn>"</dt>
   <dd><p>When this value is used, fetching cross-origin resources requires the server's
   explicit permission through the <span>CORS protocol</span> or the
   `<code>Cross-Origin-Resource-Policy</code>` header.</p></dd>

   <dt>"<dfn data-x="coep-credentialless" export for="embedder policy value"><code
   data-x="">credentialless</code></dfn>"</dt>
   <dd><p>When this value is used, fetching cross-origin no-CORS resources omits credentials. In
   exchange, an explicit `<code>Cross-Origin-Resource-Policy</code>` header is not required. Other
   requests sent with credentials require the server's explicit permission through the <span>CORS
   protocol</span> or the `<code>Cross-Origin-Resource-Policy</code>` header.</p></dd>
  </dl>

  <div class="warning">
   <p>Before supporting "<code data-x="coep-credentialless">credentialless</code>", implementers are
   strongly encouraged to support both:

   <ul class="brief">
    <li><a href="https://wicg.github.io/private-network-access/">Private Network Access</a></li>
    <li><a href="https://github.com/annevk/orb">Opaque Response Blocking</a></li>
   </ul>

   <p>Otherwise, it would allow attackers to leverage the client's network position to read non
   public resources, using the <span
   data-x="concept-settings-object-cross-origin-isolated-capability">cross-origin isolated
   capability</span>.</p>
  </div>

  <p>An <span>embedder policy value</span> is <dfn>compatible with cross-origin isolation</dfn> if
  it is "<code data-x="coep-credentialless">credentialless</code>" or "<code
  data-x="coep-require-corp">require-corp</code>".</p>

  <div w-nodev>

  <p>An <dfn export>embedder policy</dfn> consists of:</p>

  <ul>
   <li><p>A <dfn data-x="embedder-policy-value" for="embedder policy"
   export>value</dfn>, which is an <span>embedder policy value</span>, initially "<code
   data-x="coep-unsafe-none">unsafe-none</code>".</p></li>

   <li><p>A <dfn data-x="embedder-policy-reporting-endpoint" for="embedder policy"
   export>reporting endpoint</dfn> string, initially the empty string.</p></li>

   <li><p>A <dfn data-x="embedder-policy-report-only-value" for="embedder policy"
   export>report only value</dfn>, which is an <span>embedder policy value</span>, initially
   "<code data-x="coep-unsafe-none">unsafe-none</code>".</p></li>

   <li><p>A <dfn data-x="embedder-policy-report-only-reporting-endpoint" for="embedder
   policy" export>report only reporting endpoint</dfn> string, initially the empty
   string.</p></li>
  </ul>

  <p>The <dfn export>"<code>coep</code>" report type</dfn> is a <span>report type</span> whose value
  is "<code data-x="">coep</code>". It is <span>visible to
  <code>ReportingObserver</code>s</span>.</p>

  </div>


  <h5 id="the-coep-headers">The headers</h5>

  <p>The `<dfn http-header><code>Cross-Origin-Embedder-Policy</code></dfn>` and
  `<dfn http-header><code>Cross-Origin-Embedder-Policy-Report-Only</code></dfn>` HTTP response
  headers allow a server to declare an <span>embedder policy</span> for an <span>environment
  settings object</span>. These headers are <span data-x="http-structured-header">structured
  headers</span> whose values must be <span data-x="http-structured-header-token">token</span>.
  <ref>STRUCTURED-FIELDS</ref>

  <p>The valid <span
  data-x="http-structured-header-token">token</span> values are the <span data-x="embedder policy
  value">embedder policy values</span>. The token may also have attached <span
  data-x="http-structured-header-parameters">parameters</span>; of these, the "<dfn
  data-x="coep-report-to"><code>report-to</code></dfn>" parameter can have a <span>valid URL
  string</span> identifying an appropriate reporting endpoint. <ref>REPORTING</ref></p>

  <div class="note">
   <p>The <span data-x="obtain an embedder policy">processing model</span> fails open (by defaulting
   to "<code data-x="coep-unsafe-none">unsafe-none</code>") in the presence of a header that cannot
   be parsed as a token. This includes inadvertent lists created by combining multiple instances of
   the `<code>Cross-Origin-Embedder-Policy</code>` header present in a given response:</p>

   <table class="data">
    <thead>
     <tr>
      <th>`<code>Cross-Origin-Embedder-Policy</code>`</th>
      <th>Final <span>embedder policy value</span></th>
     </tr>
    </thead>
    <tbody>
     <tr>
      <td><em>No header delivered</em></td>
      <td>"<code data-x="coep-unsafe-none">unsafe-none</code>"</td>
     </tr>
     <tr>
      <td>`<code data-x="">require-corp</code>`</td>
      <td>"<code data-x="coep-require-corp">require-corp</code>"</td>
     </tr>
     <tr>
      <td>`<code data-x="">unknown-value</code>`</td>
      <td>"<code data-x="coep-unsafe-none">unsafe-none</code>"</td>
     </tr>
     <tr>
      <td>`<code data-x="">require-corp, unknown-value</code>`</td>
      <td>"<code data-x="coep-unsafe-none">unsafe-none</code>"</td>
     </tr>
     <tr>
      <td>`<code data-x="">unknown-value, unknown-value</code>`</td>
      <td>"<code data-x="coep-unsafe-none">unsafe-none</code>"</td>
     </tr>
     <tr>
      <td>`<code data-x="">unknown-value, require-corp</code>`</td>
      <td>"<code data-x="coep-unsafe-none">unsafe-none</code>"</td>
     </tr>
     <tr>
      <td>`<code data-x="">require-corp, require-corp</code>`</td>
      <td>"<code data-x="coep-unsafe-none">unsafe-none</code>"</td>
     </tr>
    </tbody>
   </table>

   <p>(The same applies to `<code>Cross-Origin-Embedder-Policy-Report-Only</code>`.)</p>
  </div>

  <div w-nodev>

  <hr>

  <p>To <dfn export>obtain an embedder policy</dfn> from a <span
  data-x="concept-response">response</span> <var>response</var> and an <span>environment</span>
  <var>environment</var>:</p>

  <ol>
   <li><p>Let <var>policy</var> be a new <span>embedder policy</span>.</p></li>

   <li><p>If <var>environment</var> is a <span>non-secure context</span>, then return
   <var>policy</var>.</p></li>

   <li><p>Let <var>parsedItem</var> be the result of <span>getting a structured field value</span>
   with `<code>Cross-Origin-Embedder-Policy</code>` and "<code data-x="">item</code>" from
   <var>response</var>'s <span data-x="concept-response-header-list">header list</span>.</p></li>

   <li>
    <p>If <var>parsedItem</var> is non-null and <var>parsedItem</var>[0] is <span>compatible with
    cross-origin isolation</span>:</p>

    <ol>
     <li><p>Set <var>policy</var>'s <span data-x="embedder-policy-value">value</span> to
     <var>parsedItem</var>[0].</p></li>

     <li><p>If <var>parsedItem</var>[1]["<code data-x="coep-report-to">report-to</code>"] <span
     data-x="map exists">exists</span>, then set <var>policy</var>'s <span
     data-x="embedder-policy-reporting-endpoint">endpoint</span> to
     <var>parsedItem</var>[1]["<code data-x="coep-report-to">report-to</code>"].</p></li>
    </ol>
   </li>

   <li><p>Set <var>parsedItem</var> to the result of <span>getting a structured field value</span>
   with `<code>Cross-Origin-Embedder-Policy-Report-Only</code>` and "<code data-x="">item</code>"
   from <var>response</var>'s <span data-x="concept-response-header-list">header
   list</span>.</p></li>

   <li>
    <p>If <var>parsedItem</var> is non-null and <var>parsedItem</var>[0] is <span>compatible with
    cross-origin isolation</span>:</p>

    <ol>
     <li><p>Set <var>policy</var>'s <span
     data-x="embedder-policy-report-only-value">report only value</span> to
     <var>parsedItem</var>[0].</p></li>

     <li><p>If <var>parsedItem</var>[1]["<code data-x="coep-report-to">report-to</code>"] <span
     data-x="map exists">exists</span>, then set <var>policy</var>'s <span
     data-x="embedder-policy-report-only-reporting-endpoint">endpoint</span> to
     <var>parsedItem</var>[1]["<code data-x="coep-report-to">report-to</code>"].</p></li>
    </ol>
   </li>

   <li><p>Return <var>policy</var>.</p></li>
  </ol>


  <h5>Embedder policy checks</h5>

  <p>To <dfn>check a navigation response's adherence to its embedder policy</dfn> given a <span
  data-x="concept-response">response</span> <var>response</var>, a <span>navigable</span>
  <var>navigable</var>, and an <span>embedder policy</span> <var>responsePolicy</var>:</p>

  <ol>
   <li><p>If <var>navigable</var> is not a <span>child navigable</span>, then return
   true.</p></li>

   <li><p>Let <var>parentPolicy</var> be <var>navigable</var>'s <span
   data-x="nav-container-document">container document</span>'s <span
   data-x="concept-document-policy-container">policy container</span>'s <span
   data-x="policy-container-embedder-policy">embedder policy</span>.</p></li>

   <li><p>If <var>parentPolicy</var>'s <span data-x="embedder-policy-report-only-value">report-only
   value</span> is <span>compatible with cross-origin isolation</span> and
   <var>responsePolicy</var>'s <span data-x="embedder-policy-value">value</span> is not, then
   <span>queue a cross-origin embedder policy inheritance violation</span> with <var>response</var>,
   "<code data-x="">navigation</code>", <var>parentPolicy</var>'s <span
   data-x="embedder-policy-report-only-reporting-endpoint">report only reporting endpoint</span>,
   "<code data-x="">reporting</code>", and <var>navigable</var>'s <span
   data-x="nav-container-document">container document</span>'s <span>relevant settings
   object</span>.</p></li>

   <li><p>If <var>parentPolicy</var>'s <span data-x="embedder-policy-value">value</span> is not
   <span>compatible with cross-origin isolation</span> or <var>responsePolicy</var>'s <span
   data-x="embedder-policy-value">value</span> is <span>compatible with cross-origin
   isolation</span>, then return true.</p></li>

   <li><p><span>Queue a cross-origin embedder policy inheritance violation</span> with
   <var>response</var>, "<code data-x="">navigation</code>", <var>parentPolicy</var>'s <span
   data-x="embedder-policy-reporting-endpoint">reporting endpoint</span>,
   "<code data-x="">enforce</code>", and <var>navigable</var>'s
   <span data-x="nav-container-document">container document</span>'s <span>relevant settings
   object</span>.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  <p>To <dfn>check a global object's embedder policy</dfn> given a <code>WorkerGlobalScope</code>
  <var>workerGlobalScope</var>, an <span>environment settings object</span> <var>owner</var>, and
  a <span data-x="concept-response">response</span> <var>response</var>:</p>

  <ol>
   <li><p>If <var>workerGlobalScope</var> is not a <code>DedicatedWorkerGlobalScope</code> object,
   then return true.</p></li>

   <li><p>Let <var>policy</var> be <var>workerGlobalScope</var>'s <span
   data-x="concept-WorkerGlobalScope-embedder-policy">embedder policy</span>.

   <li><p>Let <var>ownerPolicy</var> be <var>owner</var>'s <span
   data-x="concept-settings-object-policy-container">policy container</span>'s <span
   data-x="policy-container-embedder-policy">embedder policy</span>.

   <li><p>If <var>ownerPolicy</var>'s <span data-x="embedder-policy-report-only-value">report-only
   value</span> is <span>compatible with cross-origin isolation</span> and <var>policy</var>'s
   <span data-x="embedder-policy-value">value</span> is not, then <span>queue a cross-origin
   embedder policy inheritance violation</span> with <var>response</var>, "<code data-x="">worker
   initialization</code>", <var>ownerPolicy</var>'s <span
   data-x="embedder-policy-report-only-reporting-endpoint">report only reporting endpoint</span>,
   "<code data-x="">reporting</code>", and <var>owner</var>.</p></li>

   <li><p>If <var>ownerPolicy</var>'s <span data-x="embedder-policy-value">value</span> is not
   <span>compatible with cross-origin isolation</span> or <var>policy</var>'s <span
   data-x="embedder-policy-value">value</span> is <span>compatible with cross-origin
   isolation</span>, then return true.</p></li>

   <li><p><span>Queue a cross-origin embedder policy inheritance violation</span> with
   <var>response</var>, "<code data-x="">worker initialization</code>", <var>ownerPolicy</var>'s
   <span data-x="embedder-policy-reporting-endpoint">reporting endpoint</span>,
   "<code data-x="">enforce</code>", and <var>owner</var>.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  <p>To <dfn>queue a cross-origin embedder policy inheritance violation</dfn> given a <span
  data-x="concept-response">response</span> <var>response</var>, a string <var>type</var>, a string
  <var>endpoint</var>, a string <var>disposition</var>, and an <span>environment settings
  object</span> <var>settings</var>:</p>

  <ol>
   <li><p>Let <var>serialized</var> be the result of <span
   data-x="serialize-a-response-url-for-reporting">serializing a response URL for
   reporting</span> with <var>response</var>.</p></li>

   <li>
    <p>Let <var>body</var> be a new object containing the following properties:</p>

    <table class="data">
     <thead>
      <tr>
       <th>key</th>
       <th>value</th>
      </tr>
     </thead>
     <tbody>
      <tr>
      <td>type</td>
       <td><var>type</var></td>
      </tr>
      <tr>
       <td>blockedURL</td>
       <td><var>serialized</var></td>
      </tr>
      <tr>
       <td>disposition</td>
       <td><var>disposition</var></td>
      </tr>
     </tbody>
    </table>
   </li>

   <li><p><span data-x="queue a report">Queue</span> <var>body</var> as the
   <span>"<code>coep</code>" report type</span> for <var>endpoint</var> on <var>settings</var>.
  </ol>

  </div>


  <h4>Sandboxing</h4>

  <p>A <dfn export>sandboxing flag set</dfn> is a set of zero or more of the following flags, which
  are used to restrict the abilities that potentially untrusted resources have:</p>

  <dl>
   <dt>The <dfn export>sandboxed navigation browsing context flag</dfn></dt>

   <dd>
    <p>This flag <a href="#sandboxLinks">prevents content from navigating browsing contexts other
    than the sandboxed browsing context itself</a> (or browsing contexts further nested inside it),
    <span data-x="auxiliary browsing context">auxiliary browsing contexts</span> (which are protected
    by the <span>sandboxed auxiliary navigation browsing context flag</span> defined next), and the
    <span>top-level browsing context</span> (which is protected by the <span>sandboxed top-level
    navigation without user activation browsing context flag</span> and <span>sandboxed top-level
    navigation with user activation browsing context flag</span> defined below).</p>

    <p>If the <span>sandboxed auxiliary navigation browsing context flag</span> is not set, then in
    certain cases the restrictions nonetheless allow popups (new <span data-x="top-level browsing
    context">top-level browsing contexts</span>) to be opened. These <span data-x="browsing
    context">browsing contexts</span> always have <dfn>one permitted sandboxed navigator</dfn>, set
    when the browsing context is created, which allows the <span>browsing context</span> that
    created them to actually navigate them. (Otherwise, the <span>sandboxed navigation browsing
    context flag</span> would prevent them from being navigated even if they were opened.)</p>
   </dd>


   <dt>The <dfn export>sandboxed auxiliary navigation browsing context flag</dfn></dt>

   <dd>
    <p>This flag <a href="#sandboxWindowOpen">prevents content from creating new auxiliary browsing
    contexts</a>, e.g. using the <code data-x="attr-hyperlink-target">target</code> attribute or
    the <code data-x="dom-open">window.open()</code> method.</p>
   </dd>


   <dt>The <dfn export>sandboxed top-level navigation without user activation browsing
   context flag</dfn></dt>

   <dd>
    <p>This flag <a href="#sandboxLinks">prevents content from navigating their <span>top-level
    browsing context</span></a> and <a href="#sandboxClose">prevents content from closing their
    <span>top-level browsing context</span></a>. It is consulted only when the sandboxed browsing
    context's <span>active window</span> does not have <span>transient activation</span>.</p>

    <p>When the <span>sandboxed top-level navigation without user activation browsing context
    flag</span> is <em>not</em> set, content can navigate its <span>top-level browsing
    context</span>, but other <span data-x="browsing context">browsing contexts</span> are still
    protected by the <span>sandboxed navigation browsing context flag</span> and possibly
    the <span>sandboxed auxiliary navigation browsing context flag</span>.</p>
   </dd>


   <dt>The <dfn export>sandboxed top-level navigation with user activation browsing context
   flag</dfn></dt>

   <dd>
    <p>This flag <a href="#sandboxLinks">prevents content from navigating their <span>top-level
    browsing context</span></a> and <a href="#sandboxClose">prevents content from closing their
    <span>top-level browsing context</span></a>. It is consulted only when the sandboxed browsing
    context's <span>active window</span> has <span>transient activation</span>.</p>

    <p>As with the <span>sandboxed top-level navigation without user activation browsing context
    flag</span>, this flag only affects the <span>top-level browsing context</span>; if it is not
    set, other <span data-x="browsing context">browsing contexts</span> might still be protected by
    other flags.</p>
   </dd>


   <dt>The <dfn export>sandboxed origin browsing context flag</dfn></dt>

   <dd>
    <p>This flag <a href="#sandboxOrigin">forces content into an opaque origin</a>, thus preventing
    it from accessing other content from the same <span>origin</span>.</p>

    <p>This flag also <a href="#sandboxCookies">prevents script from reading from or writing to the
    <code data-x="dom-document-cookie">document.cookie</code> IDL attribute</a>, and blocks access
    to <code data-x="dom-localStorage">localStorage</code>.</p>
   </dd>


   <dt>The <dfn export>sandboxed forms browsing context flag</dfn></dt>

   <dd>
    <p>This flag <a href="#sandboxSubmitBlocked">blocks form submission</a>.</p>
   </dd>


   <dt>The <dfn export>sandboxed pointer lock browsing context flag</dfn></dt>

   <dd>
    <p>This flag disables the Pointer Lock API. <ref>POINTERLOCK</ref></p>
   </dd>


   <dt>The <dfn export>sandboxed scripts browsing context flag</dfn></dt>

   <dd>
    <p>This flag <a href="#sandboxScriptBlocked">blocks script execution</a>.</p>
   </dd>


   <dt>The <dfn export>sandboxed automatic features browsing context flag</dfn></dt>

   <dd>
    <p>This flag blocks features that trigger automatically, such as <span
    data-x="attr-media-autoplay">automatically playing a video</span> or <span
    data-x="attr-fe-autofocus">automatically focusing a form control</span>.</p>
   </dd>


   <dt>The <dfn export>sandboxed <code data-x="dom-document-domain">document.domain</code>
   browsing context flag</dfn></dt>

   <dd>
    <p>This flag prevents content from using the
    <code data-x="dom-document-domain">document.domain</code> setter.</p>
   </dd>


   <dt>The <dfn export>sandbox propagates to auxiliary browsing contexts flag</dfn></dt>

   <dd>
    <p>This flag prevents content from escaping the sandbox by ensuring that any
    <span>auxiliary browsing context</span> it creates inherits the content's
    <span>active sandboxing flag set</span>.</p>
   </dd>

   <dt>The <dfn export>sandboxed modals flag</dfn></dt>

   <dd>
    <p>This flag prevents content from using any of the following features to produce modal
    dialogs:</p>

    <ul>
     <li><code data-x="dom-alert">window.alert()</code></li>
     <li><code data-x="dom-confirm">window.confirm()</code></li>
     <li><code data-x="dom-print">window.print()</code></li>
     <li><code data-x="dom-prompt">window.prompt()</code></li>
     <li>the <code data-x="event-beforeunload">beforeunload</code> event</li>
    </ul>
   </dd>

   <dt>The <dfn export>sandboxed orientation lock browsing context flag</dfn></dt>

   <dd>
    <p>This flag disables the ability to lock the screen orientation.
    <ref>SCREENORIENTATION</ref></p>
   </dd>

   <dt>The <dfn export>sandboxed presentation browsing context flag</dfn></dt>

   <dd>
    <p>This flag disables the Presentation API. <ref>PRESENTATION</ref></p>
   </dd>

   <dt>The <dfn export>sandboxed downloads browsing context flag</dfn></dt>

   <dd>
    <p>This flag prevents content from initiating or instantiating downloads, whether through <span
    data-x="downloading hyperlinks">downloading hyperlinks</span> or through <a
    href="#navigation-as-a-download">navigation</a> that gets <span data-x="handle as a
    download">handled as a download</span>.</p>
   </dd>

   <dt>The <dfn export>sandboxed custom protocols navigation browsing context flag</dfn></dt>

   <dd>
    <p>This flag prevents navigations toward non <span data-x="fetch scheme">fetch schemes</span>
    from being <span data-x="hand-off to external software">handed off to external
    software</span>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>When the user agent is to <dfn export>parse a sandboxing directive</dfn>, given a string
  <var>input</var> and a <span>sandboxing flag set</span> <var>output</var>, it must run the following
  steps:</p>

  <ol>
   <li><p><span data-x="split a string on ASCII whitespace">Split <var>input</var> on ASCII
   whitespace</span>, to obtain <var>tokens</var>.</p></li>

   <li><p>Let <var>output</var> be empty.</p></li>

   <li>
    <p>Add the following flags to <var>output</var>:</p>

    <ul>
     <li><p>The <span>sandboxed navigation browsing context flag</span>.</p></li>

     <li><p>The <span>sandboxed auxiliary navigation browsing context flag</span>, unless
     <var>tokens</var> contains the <dfn attr-value for="iframe/sandbox"><code
     data-x="attr-iframe-sandbox-allow-popups">allow-popups</code></dfn> keyword.</p></li>

     <li><p>The <span>sandboxed top-level navigation without user activation browsing context
     flag</span>, unless <var>tokens</var> contains the <dfn attr-value for="iframe/sandbox"><code
     data-x="attr-iframe-sandbox-allow-top-navigation">allow-top-navigation</code></dfn>
     keyword.</p></li>

     <li>
      <p>The <span>sandboxed top-level navigation with user activation browsing context flag</span>,
      unless <var>tokens</var> contains either the <dfn attr-value for="iframe/sandbox"><code
      data-x="attr-iframe-sandbox-allow-top-navigation-by-user-activation">allow-top-navigation-by-user-activation</code></dfn>
      keyword or the <code
      data-x="attr-iframe-sandbox-allow-top-navigation">allow-top-navigation</code> keyword.</p>

      <p class="note">This means that if the <code
      data-x="attr-iframe-sandbox-allow-top-navigation">allow-top-navigation</code> is present, the <code
      data-x="attr-iframe-sandbox-allow-top-navigation-by-user-activation">allow-top-navigation-by-user-activation</code>
      keyword will have no effect. For this reason, specifying both is a document conformance error.</p>
     </li>

     <li>
      <p>The <span>sandboxed origin browsing context flag</span>, unless the <var>tokens</var>
      contains the <dfn attr-value for="iframe/sandbox"><code
      data-x="attr-iframe-sandbox-allow-same-origin">allow-same-origin</code></dfn> keyword.</p>

      <div class="note">
       <p>The <code data-x="attr-iframe-sandbox-allow-same-origin">allow-same-origin</code> keyword
       is intended for two cases.</p>

       <p>First, it can be used to allow content from the same site to be sandboxed to disable
       scripting, while still allowing access to the DOM of the sandboxed content.</p>

       <p>Second, it can be used to embed content from a third-party site, sandboxed to prevent that
       site from opening popups, etc, without preventing the embedded page from communicating back
       to its originating site, using the database APIs to store data, etc.</p>
      </div>
     </li>

     <li><p>The <span>sandboxed forms browsing context flag</span>, unless <var>tokens</var>
     contains the <dfn attr-value for="iframe/sandbox"><code
     data-x="attr-iframe-sandbox-allow-forms">allow-forms</code></dfn> keyword.</p></li>

     <li><p>The <span>sandboxed pointer lock browsing context flag</span>, unless <var>tokens</var>
     contains the <dfn attr-value for="iframe/sandbox"><code
     data-x="attr-iframe-sandbox-allow-pointer-lock">allow-pointer-lock</code></dfn>
     keyword.</p></li>

     <li><p>The <span>sandboxed scripts browsing context flag</span>, unless <var>tokens</var>
     contains the <dfn attr-value for="iframe/sandbox"><code
     data-x="attr-iframe-sandbox-allow-scripts">allow-scripts</code></dfn> keyword.</p></li>

     <li>
      <p>The <span>sandboxed automatic features browsing context flag</span>, unless
      <var>tokens</var> contains the <code
      data-x="attr-iframe-sandbox-allow-scripts">allow-scripts</code> keyword (defined above).</p>

      <p class="note">This flag is relaxed by the same keyword as scripts, because when scripts are
      enabled these features are trivially possible anyway, and it would be unfortunate to force
      authors to use script to do them when sandboxed rather than allowing them to use the
      declarative features.</p>
     </li>

     <li><p>The <span>sandboxed <code data-x="dom-document-domain">document.domain</code> browsing
     context flag</span>.</p></li>

     <li><p>The <span>sandbox propagates to auxiliary browsing contexts flag</span>, unless
     <var>tokens</var> contains the <dfn attr-value for="iframe/sandbox"><code
     data-x="attr-iframe-sandbox-allow-popups-to-escape-sandbox">allow-popups-to-escape-sandbox</code></dfn>
     keyword.</p></li>

     <li><p>The <span>sandboxed modals flag</span>, unless <var>tokens</var> contains the <dfn
     attr-value for="iframe/sandbox"><code
     data-x="attr-iframe-sandbox-allow-modals">allow-modals</code></dfn> keyword.</p></li>

     <li><p>The <span>sandboxed orientation lock browsing context flag</span>, unless
     <var>tokens</var> contains the <dfn attr-value for="iframe/sandbox"><code
     data-x="attr-iframe-sandbox-allow-orientation-lock">allow-orientation-lock</code></dfn>
     keyword.</p></li>

     <li><p>The <span>sandboxed presentation browsing context flag</span>, unless <var>tokens</var>
     contains the <dfn attr-value for="iframe/sandbox"><code
     data-x="attr-iframe-sandbox-allow-presentation">allow-presentation</code></dfn>
     keyword.</p></li>

     <li><p>The <span>sandboxed downloads browsing context flag</span>, unless <var>tokens</var>
     contains the <dfn attr-value for="iframe/sandbox"><code
     data-x="attr-iframe-sandbox-allow-downloads">allow-downloads</code></dfn> keyword.</p></li>

     <li><p>The <span>sandboxed custom protocols navigation browsing context flag</span>, unless
     <var>tokens</var> contains either the <dfn attr-value for="iframe/sandbox"><code
     data-x="attr-iframe-sandbox-allow-top-navigation-to-custom-protocols">allow-top-navigation-to-custom-protocols</code></dfn>
     keyword, the <code data-x="attr-iframe-sandbox-allow-popups">allow-popups</code> keyword, or
     the <code data-x="attr-iframe-sandbox-allow-top-navigation">allow-top-navigation</code>
     keyword.</p></li>
    </ul>
   </li>
  </ol>

  <hr>

  <p>Every <span>top-level browsing context</span> has a <dfn>popup sandboxing flag set</dfn>, which
  is a <span>sandboxing flag set</span>. When a <span>browsing context</span> is created, its
  <span>popup sandboxing flag set</span> must be empty. It is populated by <span>the rules for
  choosing a navigable</span> and the <span data-x="obtain-browsing-context-navigation">obtain
  a browsing context to use for a navigation response</span> algorithm.</p>

  <p>Every <code>iframe</code> element has an <dfn><code>iframe</code> sandboxing flag set</dfn>,
  which is a <span>sandboxing flag set</span>. Which flags in an <span><code>iframe</code>
  sandboxing flag set</span> are set at any particular time is determined by the <code>iframe</code>
  element's <code data-x="attr-iframe-sandbox">sandbox</code> attribute.</p>

  <p>Every <code>Document</code> has an <dfn for="Document" export>active sandboxing flag set</dfn>,
  which is a <span>sandboxing flag set</span>. When the <code>Document</code> is created, its
  <span>active sandboxing flag set</span> must be empty. It is populated by the <span
  data-x="navigate">navigation algorithm</span>.</p>

  <p id="forced-sandboxing-flag-set">Every <span data-x="concept-csp-list">CSP list</span>
  <var>cspList</var> has <dfn>CSP-derived sandboxing flags</dfn>, which is a <span>sandboxing flag
  set</span>. It is the return value of the following algorithm:</p>

  <ol>
   <li><p>Let <var>directives</var> be an empty <span data-x="set">ordered set</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> policy in <var>cspList</var>:</p>

    <ol>
     <li><p>If <var>policy</var>'s <span data-x="csp-disposition">disposition</span> is not "<code
     data-x="">enforce</code>", then <span>continue</span>.</p></li>

     <li><p>If <var>policy</var>'s <span data-x="csp-directive-set">directive set</span> <span
     data-x="list contains">contains</span> a <span data-x="Content Security Policy
     directive">directive</span> whose name is "<code data-x="sandbox directive">sandbox</code>",
     then <span data-x="list append">append</span> that directive to
     <var>directives</var>.</p></li>
    </ol>
   </li>

   <li><p>If <var>directives</var> is empty, then return an empty <span>sandboxing flag
   set</span>.</p></li>

   <li><p>Let <var>directive</var> be <var>directives</var>[<var>directives</var>'s <span
   data-x="list size">size</span> &minus; 1].</p></li>

   <li><p>Return the result of <span data-x="parse a sandboxing directive">parsing the sandboxing
   directive</span> <var>directive</var>.</p></li>
  </ol>

  <hr>

  <p>To <dfn data-x="determining the creation sandboxing flags">determine the creation sandboxing
  flags</dfn> for a <span data-x="concept-document-bc">browsing context</span> <var>browsing
  context</var>, given null or an element <var>embedder</var>, return the <span data-x="set
  union">union</span> of the flags that are present in the following <span data-x="sandboxing flag
  set">sandboxing flag sets</span>:</p>

  <ul>
   <li><p>If <var>embedder</var> is null, then: the flags set on <var>browsing context</var>'s
   <span>popup sandboxing flag set</span>.</p></li>

   <li><p>If <var>embedder</var> is an element, then: the flags set on <var>embedder</var>'s
   <span><code>iframe</code> sandboxing flag set</span>.</p></li>

   <li><p>If <var>embedder</var> is an element, then: the flags set on <var>embedder</var>'s
   <span>node document</span>'s <span>active sandboxing flag set</span>.</p></li>
  </ul>


  <h4 id="policy-containers">Policy containers</h4>

  <p>A <dfn export>policy container</dfn> is a <span>struct</span> containing policies that apply to
  a <code>Document</code>, a <code>WorkerGlobalScope</code>, or a <code>WorkletGlobalScope</code>.
  It has the following <span data-x="struct item">items</span>:</p>

  <!-- Note: each item has to define a default value for creating a new policy container. -->

  <ul>
   <li><p>A <dfn export for="policy container" data-x="policy-container-csp-list">CSP list</dfn>,
   which is a <span data-x="concept-csp-list">CSP list</span>. It is initially empty.</p></li>

   <li><p>An <dfn export for="policy container" data-x="policy-container-embedder-policy">embedder
   policy</dfn>, which is an <span>embedder policy</span>. It is initially a new <span>embedder
   policy</span>.</p></li>

   <li><p>A <dfn export for="policy container" data-x="policy-container-referrer-policy">referrer
   policy</dfn>, which is a <span>referrer policy</span>. It is initially the <span>default referrer
   policy</span>.</p></li>
  </ul>

  <p class="XXX">Move other policies into the policy container.</p>

  <p>To <dfn export>clone a policy container</dfn> given a <span>policy container</span>
  <var>policyContainer</var>:</p>

  <ol>
   <li><p>Let <var>clone</var> be a new <span>policy container</span>.</p></li>

   <li><p><span data-x="list iterate">For each</span> <var>policy</var> in
   <var>policyContainer</var>'s <span data-x="policy-container-csp-list">CSP list</span>, <span
   data-x="list append">append</span> a copy of <var>policy</var> into <var>clone</var>'s <span
   data-x="policy-container-csp-list">CSP list</span>.</p></li>

   <li><p>Set <var>clone</var>'s <span data-x="policy-container-embedder-policy">embedder
   policy</span> to a copy of <var>policyContainer</var>'s <span
   data-x="policy-container-embedder-policy">embedder policy</span>.</p></li>

   <li><p>Set <var>clone</var>'s <span data-x="policy-container-referrer-policy">referrer
   policy</span> to <var>policyContainer</var>'s <span
   data-x="policy-container-referrer-policy">referrer policy</span>.</p></li>

   <li><p>Return <var>clone</var>.</p></li>
  </ol>

  <p>To determine whether a <span>URL</span> <var>url</var> <dfn>requires storing the policy
  container in history</dfn>:</p>

  <ol>
   <li><p>If <var>url</var>'s <span data-x="concept-url-scheme">scheme</span> is "<code
   data-x="">blob</code>", then return false.</p></li>
   <!-- Note that we are not storing the policy container in history for blob URLs since we can
   always retrieve again a blob URL entry's environment's policy container during a history
   navigation. Indeed, as per https://www.w3.org/TR/FileAPI/#lifeTime, if the environment that
   created the blob URL is dead at the point of the history navigation, then the blob URL itself
   must have been deleted, too. We might want to store the policy container alongside the blob URL
   entry directly (https://github.com/w3c/FileAPI/issues/142), and then reconsider this. -->

   <!-- See about:srcdoc discussion:
   https://github.com/whatwg/html/issues/6809#issuecomment-1170924595 -->
   <li><p>If <var>url</var> <span data-x="is-local">is local</span>, then return true.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  <p>To <dfn export data-lt="creating a policy container from a fetch response" data-x="creating a
  policy container from a fetch response">create a policy container from a fetch response</dfn>
  given a <span data-x="concept-response">response</span> <var>response</var> and an
  <span>environment</span>-or-null <var>environment</var>:</p>

  <ol>
   <li><p>If <var>response</var>'s <span data-x="concept-response-url">URL</span>'s <span
   data-x="concept-url-scheme">scheme</span> is "<code data-x="">blob</code>", then return a <span
   data-x="clone a policy container">clone</span> of <var>response</var>'s <span
   data-x="concept-response-url">URL</span>'s <span data-x="concept-url-blob-entry">blob URL
   entry</span>'s <span data-x="blob-url-entry-environment">environment</span>'s <span>policy
   container</span>.</p></li>

   <li><p>Let <var>result</var> be a new <span>policy container</span>.</p></li>

   <li><p>Set <var>result</var>'s <span data-x="policy-container-csp-list">CSP list</span> to the
   result of <span data-x="parse-response-csp">parsing a response's Content Security Policies</span>
   given <var>response</var>.</p></li>

   <li><p>If <var>environment</var> is non-null, then set <var>result</var>'s <span
   data-x="policy-container-embedder-policy">embedder policy</span> to the result of <span
   data-x="obtain an embedder policy">obtaining an embedder policy</span> given <var>response</var>
   and <var>environment</var>. Otherwise, set it to "<code
   data-x="coep-unsafe-none">unsafe-none</code>".</p></li>

   <li><p>Set <var>result</var>'s <span data-x="policy-container-referrer-policy">referrer
   policy</span> to the result of <span data-x="parse-referrer-policy-header">parsing the
   `<code>Referrer-Policy</code>` header</span> given <var>response</var>.
   <ref>REFERRERPOLICY</ref></p></li>

   <li><p>Return <var>result</var>.</p></li>
  </ol>

  <p>To <dfn data-x="determining navigation params policy container">determine navigation params
  policy container</dfn> given a <span>URL</span> <var>responseURL</var> and four <span
  data-x="policy container">policy container</span>-or-nulls <var>historyPolicyContainer</var>,
  <var>initiatorPolicyContainer</var>, <var>parentPolicyContainer</var>, and
  <var>responsePolicyContainer</var>:</p>

  <ol>
   <li>
    <p>If <var>historyPolicyContainer</var> is not null, then:</p>

    <ol>
     <li><p><span>Assert</span>: <var>responseURL</var> <span>requires storing the policy container
     in history</span>.</p></li>

     <li><p>Return a <span data-x="clone a policy container">clone</span> of
     <var>historyPolicyContainer</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>If <var>responseURL</var> is <code>about:srcdoc</code>, then:</p>

    <ol>
     <li><p><span>Assert</span>: <var>parentPolicyContainer</var> is not null.</p></li>

     <li><p>Return a <span data-x="clone a policy container">clone</span> of
     <var>parentPolicyContainer</var>.</p></li>
    </ol>
   </li>

   <li><p>If <var>responseURL</var> <span data-x="is-local">is local</span> and
   <var>initiatorPolicyContainer</var> is not null, then return a <span data-x="clone a policy
   container">clone</span> of <var>initiatorPolicyContainer</var>.</p></li>

   <li><p>If <var>responsePolicyContainer</var> is not null, then return
   <var>responsePolicyContainer</var>.</p></li>

   <li><p>Return a new <span>policy container</span>.</p></li>
  </ol>

  <p>To <dfn data-x="initialize worker policy container">initialize a worker global scope's policy
  container</dfn> given a <code>WorkerGlobalScope</code> <var>workerGlobalScope</var>, a <span
  data-x="concept-response">response</span> <var>response</var>, and an <span>environment</span>
  <var>environment</var>:</p>

  <ol>
   <li>
    <p>If <var>workerGlobalScope</var>'s <span data-x="concept-WorkerGlobalScope-url">url</span>
    <span data-x="is-local">is local</span> but its <span data-x="concept-url-scheme">scheme</span>
    is not "<code data-x="">blob</code>":</p>

    <ol>
     <li><p><span>Assert</span>: <var>workerGlobalScope</var>'s <span>owner set</span>'s <span
     data-x="list size">size</span> is 1.</p></li>

     <li><p>Set <var>workerGlobalScope</var>'s <span
     data-x="concept-WorkerGlobalScope-policy-container">policy container</span> to a <span
     data-x="clone a policy container">clone</span> of <var>workerGlobalScope</var>'s <span>owner
     set</span>[0]'s <span>relevant settings object</span>'s <span
     data-x="concept-settings-object-policy-container">policy container</span>.</p></li>
    </ol>
   </li>

   <li><p>Otherwise, set <var>workerGlobalScope</var>'s <span
   data-x="concept-WorkerGlobalScope-policy-container">policy container</span> to the result of
   <span>creating a policy container from a fetch response</span> given
   <var>response</var> and <var>environment</var>.</p></li>
  </ol>

  </div>


  <h3 id="nav-traversal-apis" split-filename="nav-history-apis">APIs related to navigation and
  session history</h3>

  <div w-nodev>

  <h4 id="cross-origin-objects">Security infrastructure for <code>Window</code>,
  <code>WindowProxy</code>, and <code>Location</code> objects</h4>

  <p>Although typically objects cannot be accessed across <span data-x="origin">origins</span>, the
  web platform would not be true to itself if it did not have some legacy exceptions to that rule
  that the web depends upon.</p>

  <p>This section uses the terminology and typographic conventions from the JavaScript
  specification. <ref>JAVASCRIPT</ref></p>


  <h5>Integration with IDL</h5>

  <p>When <span>perform a security check</span> is invoked, with a <var>platformObject</var>,
  <var>identifier</var>, and <var>type</var>, run these steps:</p>

  <ol>
   <li><p>If <var>platformObject</var> is not a <code>Window</code> or <code>Location</code> object,
   then return.</p></li>

   <li>
    <p>For each <var>e</var> of <span>CrossOriginProperties</span>(<var>platformObject</var>):</p>

    <ol>
     <li>
      <p>If <span>SameValue</span>(<var>e</var>.[[Property]], <var>identifier</var>) is true,
      then:</p>

      <ol>
       <li><p>If <var>type</var> is "<code data-x="">method</code>" and <var>e</var> has neither
       [[NeedsGet]] nor [[NeedsSet]], then return.</p></li>

       <li><p>Otherwise, if <var>type</var> is "<code data-x="">getter</code>" and
       <var>e</var>.[[NeedsGet]] is true, then return.</p></li>

       <li><p>Otherwise, if <var>type</var> is "<code data-x="">setter</code>" and
       <var>e</var>.[[NeedsSet]] is true, then return.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>If <span>IsPlatformObjectSameOrigin</span>(<var>platformObject</var>) is false, then
   throw a <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>
  </ol>


  <h5>Shared internal slot: [[CrossOriginPropertyDescriptorMap]]</h5>

  <p><code>Window</code> and <code>Location</code> objects both have a
  <dfn>[[CrossOriginPropertyDescriptorMap]]</dfn> internal slot, whose value is initially an empty
  map.</p>

  <p>The <span>[[CrossOriginPropertyDescriptorMap]]</span> internal slot contains a map
  with entries whose keys are (<var>currentGlobal</var>, <var>objectGlobal</var>,
  <var>propertyKey</var>)-tuples and values are property descriptors, as a memoization of what is
  visible to scripts when <var>currentGlobal</var> inspects a <code>Window</code> or
  <code>Location</code> object from <var>objectGlobal</var>. It is filled lazily by
  <span>CrossOriginGetOwnPropertyHelper</span>, which consults it on future lookups.</p>

  <p>User agents should allow a value held in the map to be garbage collected along with its
  corresponding key when nothing holds a reference to any part of the value. That is, as long as
  garbage collection is not observable.</p>

  <p class="example">For example, with <code data-x="">const href =
  Object.getOwnPropertyDescriptor(crossOriginLocation, "href").set</code> the value and its
  corresponding key in the map cannot be garbage collected as that would be observable.</p>

  <p>User agents may have an optimization whereby they remove key-value pairs from the map when
  <code data-x="dom-document-domain">document.domain</code> is set. This is not observable as <code
  data-x="dom-document-domain">document.domain</code> cannot revisit an earlier value.</p>

  <p class="example">For example, setting <code data-x="dom-document-domain">document.domain</code>
  to "<code data-x="">example.com</code>" on www.example.com means user agents can remove all
  key-value pairs from the map where part of the key is www.example.com, as that can never be part
  of the <span>origin</span> again and therefore the corresponding value could never be retrieved
  from the map.</p>


  <h5>Shared abstract operations</h5>

  <h6><dfn>CrossOriginProperties</dfn> ( <var>O</var> )</h6>

  <ol>
   <li><p><span>Assert</span>: <var>O</var> is a <code>Location</code> or <code>Window</code>
   object.</p></li>

   <li><p>If <var>O</var> is a <code>Location</code> object, then return «
   { [[Property]]: "<code data-x="">href</code>", [[NeedsGet]]: false, [[NeedsSet]]: true },
   { [[Property]]: "<code data-x="">replace</code>" } ».</p></li>

   <li><p>Return «
   { [[Property]]: "<code data-x="">window</code>", [[NeedsGet]]: true, [[NeedsSet]]: false },
   { [[Property]]: "<code data-x="">self</code>", [[NeedsGet]]: true, [[NeedsSet]]: false },
   { [[Property]]: "<code data-x="">location</code>", [[NeedsGet]]: true, [[NeedsSet]]: true },
   { [[Property]]: "<code data-x="">close</code>" },
   { [[Property]]: "<code data-x="">closed</code>", [[NeedsGet]]: true, [[NeedsSet]]: false },
   { [[Property]]: "<code data-x="">focus</code>" },
   { [[Property]]: "<code data-x="">blur</code>" },
   { [[Property]]: "<code data-x="">frames</code>", [[NeedsGet]]: true, [[NeedsSet]]: false },
   { [[Property]]: "<code data-x="">length</code>", [[NeedsGet]]: true, [[NeedsSet]]: false },
   { [[Property]]: "<code data-x="">top</code>", [[NeedsGet]]: true, [[NeedsSet]]: false },
   { [[Property]]: "<code data-x="">opener</code>", [[NeedsGet]]: true, [[NeedsSet]]: false },
   { [[Property]]: "<code data-x="">parent</code>", [[NeedsGet]]: true, [[NeedsSet]]: false },
   { [[Property]]: "<code data-x="">postMessage</code>" } ».</p></li>
  </ol>

  <p class="note">This abstract operation does not return a <span>Completion Record</span>.</p>

  <p class="note">Indexed properties do not need to be safelisted in this algorithm, as they are
  handled directly by the <code>WindowProxy</code> object.</p>

  <p>A JavaScript property name <var>P</var> is a <dfn>cross-origin accessible window property
  name</dfn> if it is "<code data-x="">window</code>", "<code data-x="">self</code>", "<code
  data-x="">location</code>", "<code data-x="">close</code>", "<code data-x="">closed</code>",
  "<code data-x="">focus</code>", "<code data-x="">blur</code>", "<code data-x="">frames</code>",
  "<code data-x="">length</code>", "<code data-x="">top</code>", "<code data-x="">opener</code>",
  "<code data-x="">parent</code>", "<code data-x="">postMessage</code>", or an <span>array index
  property name</span>.</p>

  <h6><dfn>CrossOriginPropertyFallback</dfn> ( <var>P</var> )</h6>

  <ol>
   <li><p>If <var>P</var> is "<code data-x="">then</code>", <span>%Symbol.toStringTag%</span>,
   <span>%Symbol.hasInstance%</span>, or <span>%Symbol.isConcatSpreadable%</span>, then return
   <span>PropertyDescriptor</span>{
   [[Value]]: undefined,
   [[Writable]]: false,
   [[Enumerable]]: false,
   [[Configurable]]: true }.</p></li>

   <li><p>Throw a <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>
  </ol>

  <h6><dfn>IsPlatformObjectSameOrigin</dfn> ( <var>O</var> )</h6>

  <ol>
   <li><p>Return true if the <span>current settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span> is <span>same origin-domain</span> with
   <var>O</var>'s <span>relevant settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, and false otherwise.</p></li>
  </ol>

  <p class="note">This abstract operation does not return a <span>Completion Record</span>.</p>

  <p class="note">Here the <span>current settings object</span> roughly corresponds to the "caller",
  because this check occurs before the <span data-x="JavaScript execution context">execution
  context</span> for the getter/setter/method in question makes its way onto the <span>JavaScript
  execution context stack</span>. For example, in the code <code data-x="">w.document</code>, this
  step is invoked before the <code data-x="dom-document">document</code> getter is reached as part
  of the <a href="#windowproxy-get">[[Get]]</a> algorithm for the <code>WindowProxy</code>
  <var>w</var>.</p>

  <h6><dfn>CrossOriginGetOwnPropertyHelper</dfn> ( <var>O</var>, <var>P</var> )</h6>

  <p class="note">If this abstract operation returns undefined and there is no custom behavior, the
  caller needs to throw a <span>"<code>SecurityError</code>"</span> <code>DOMException</code>. In
  practice this is handled by the caller calling <span>CrossOriginPropertyFallback</span>.</p>

  <ol>
   <li><p>Let <var>crossOriginKey</var> be a tuple consisting of the <span>current settings
   object</span>, <var>O</var>'s <span>relevant settings object</span>, and <var>P</var>.</p></li>

   <li>
    <p>For each <var>e</var> of <span>CrossOriginProperties</span>(<var>O</var>):</p>

    <ol>
     <li>
      <p>If <span>SameValue</span>(<var>e</var>.[[Property]], <var>P</var>) is true, then:</p>

      <ol>
       <li><p>If the value of the <span>[[CrossOriginPropertyDescriptorMap]]</span> internal slot of
       <var>O</var> contains an entry whose key is <var>crossOriginKey</var>, then return that
       entry's value.</p></li>

       <li><p>Let <var>originalDesc</var> be <span>OrdinaryGetOwnProperty</span>(<var>O</var>,
       <var>P</var>).</p></li>

       <li><p>Let <var>crossOriginDesc</var> be undefined.</p></li>

       <li>
        <p>If <var>e</var>.[[NeedsGet]] and <var>e</var>.[[NeedsSet]] are absent, then:</p>

        <ol>
         <li><p>Let <var>value</var> be <var>originalDesc</var>.[[Value]].</p></li>

         <li><p>If <span>IsCallable</span>(<var>value</var>) is true, then set <var>value</var> to
         an anonymous built-in function, created in the <span>current realm</span>, that performs
         the same steps as the IDL operation <var>P</var> on object <var>O</var>.</p></li>

         <li><p>Set <var>crossOriginDesc</var> to <span>PropertyDescriptor</span>{
         [[Value]]: <var>value</var>,
         [[Enumerable]]: false,
         [[Writable]]: false,
         [[Configurable]]: true }.</p></li>
        </ol>
       </li>

       <li>
        <p>Otherwise:</p>

        <ol>
         <li><p>Let <var>crossOriginGet</var> be undefined.</p></li>

         <li><p>If <var>e</var>.[[NeedsGet]] is true, then set <var>crossOriginGet</var> to an
         anonymous built-in function, created in the <span>current realm</span>, that performs the
         same steps as the getter of the IDL attribute <var>P</var> on object
         <var>O</var>.</p></li>

         <li><p>Let <var>crossOriginSet</var> be undefined.</p></li>

         <li><p>If <var>e</var>.[[NeedsSet]] is true, then set <var>crossOriginSet</var> to an
         anonymous built-in function, created in the <span>current realm</span>, that performs the
         same steps as the setter of the IDL attribute <var>P</var> on object
         <var>O</var>.</p></li>

         <li><p>Set <var>crossOriginDesc</var> to <span>PropertyDescriptor</span>{
         [[Get]]: <var>crossOriginGet</var>,
         [[Set]]: <var>crossOriginSet</var>,
         [[Enumerable]]: false,
         [[Configurable]]: true }.</p></li>
        </ol>
       </li>

       <li><p>Create an entry in the value of the <span>[[CrossOriginPropertyDescriptorMap]]</span>
       internal slot of <var>O</var> with key <var>crossOriginKey</var> and value
       <var>crossOriginDesc</var>.</p></li>

       <li><p>Return <var>crossOriginDesc</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Return undefined.</p></li>
  </ol>

  <p class="note">This abstract operation does not return a <span>Completion Record</span>.</p>

  <p class="note">The reason that the property descriptors produced here are configurable is to
  preserve the <span>invariants of the essential internal methods</span> required by the JavaScript
  specification. In particular, since the value of the property can change as a consequence of
  navigation, it is required that the property be configurable. (However, see <a
  href="https://github.com/tc39/ecma262/issues/672">tc39/ecma262 issue #672</a> and references to it
  elsewhere in this specification for cases where we are not able to preserve these invariants, for
  compatibility with existing web content.) <ref>JAVASCRIPT</ref></p>

  <p class="note">The reason the property descriptors are non-enumerable, despite this mismatching
  the same-origin behavior, is for compatibility with existing web content. See <a
  href="https://github.com/whatwg/html/issues/3183">issue #3183</a> for details.</p>

  <h6><dfn>CrossOriginGet</dfn> ( <var>O</var>, <var>P</var>, <var>Receiver</var> )</h6>

  <ol>
   <li><p>Let <var>desc</var> be ? <var>O</var>.[[GetOwnProperty]](<var>P</var>).</p></li>

   <li><p><span>Assert</span>: <var>desc</var> is not undefined.</p></li>

   <li><p>If <span>IsDataDescriptor</span>(<var>desc</var>) is true, then return
   <var>desc</var>.[[Value]].</p></li>

   <li><p><span>Assert</span>: <span>IsAccessorDescriptor</span>(<var>desc</var>) is true.</p></li>

   <li><p>Let <var>getter</var> be <var>desc</var>.[[Get]].</p></li>

   <li><p>If <var>getter</var> is undefined, then throw a <span>"<code>SecurityError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Return ? <span>Call</span>(<var>getter</var>, <var>Receiver</var>).</p></li>
  </ol>

  <h6><dfn>CrossOriginSet</dfn> ( <var>O</var>, <var>P</var>, <var>V</var>,
  <var>Receiver</var> )</h6>

  <ol>
   <li><p>Let <var>desc</var> be ? <var>O</var>.[[GetOwnProperty]](<var>P</var>).</p></li>

   <li><p><span>Assert</span>: <var>desc</var> is not undefined.</p></li>

   <li>
    <p>If <var>desc</var>.[[Set]] is present and its value is not undefined, then:

    <ol>
     <li><p>Perform ? <span>Call</span>(<var>desc</var>.[[Set]], <var>Receiver</var>,
     « <var>V</var> »).</p></li>

     <li><p>Return true.</p></li>
    </ol>
   </li>

   <li><p>Throw a <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>
  </ol>

  <h6><dfn>CrossOriginOwnPropertyKeys</dfn> ( <var>O</var> )</h6>

  <ol>
   <li><p>Let <var>keys</var> be a new empty <span data-x="js-List">List</span>.</p></li>

   <li><p>For each <var>e</var> of <span>CrossOriginProperties</span>(<var>O</var>), <span
   data-x="list append">append</span> <var>e</var>.[[Property]] to <var>keys</var>.</p></li>

   <li><p>Return the concatenation of <var>keys</var> and « "<code data-x="">then</code>",
   <span>%Symbol.toStringTag%</span>, <span>%Symbol.hasInstance%</span>, <span>%Symbol.isConcatSpreadable%</span>
   ».</p></li>
  </ol>

  <p class="note">This abstract operation does not return a <span>Completion Record</span>.</p>

  </div>


  <h4>The <code>Window</code> object</h4>

  <pre><code class="idl">[Global=Window,
 Exposed=Window,
 <span>LegacyUnenumerableNamedProperties</span>]
interface <dfn interface>Window</dfn> : <span>EventTarget</span> {
  // the current browsing context
  [<span>LegacyUnforgeable</span>] readonly attribute <span>WindowProxy</span> <span data-x="dom-window">window</span>;
  [Replaceable] readonly attribute <span>WindowProxy</span> <span data-x="dom-self">self</span>;
  [<span>LegacyUnforgeable</span>] readonly attribute <span>Document</span> <span data-x="dom-document">document</span>;
  attribute DOMString <span data-x="dom-name">name</span>; <!-- not [Replaceable] per WebKit and IE8 -->
  [PutForwards=<span data-x="dom-location-href">href</span>, <span>LegacyUnforgeable</span>] readonly attribute <span>Location</span> <span data-x="dom-location">location</span>;
  readonly attribute <span>History</span> <span data-x="dom-history">history</span>;
  readonly attribute <span>Navigation</span> <span data-x="dom-navigation">navigation</span>;
  readonly attribute <span>CustomElementRegistry</span> <span data-x="dom-window-customElements">customElements</span>;
  [Replaceable] readonly attribute <span>BarProp</span> <span data-x="dom-window-locationbar">locationbar</span>;
  [Replaceable] readonly attribute <span>BarProp</span> <span data-x="dom-window-menubar">menubar</span>;
  [Replaceable] readonly attribute <span>BarProp</span> <span data-x="dom-window-personalbar">personalbar</span>;
  [Replaceable] readonly attribute <span>BarProp</span> <span data-x="dom-window-scrollbars">scrollbars</span>;
  [Replaceable] readonly attribute <span>BarProp</span> <span data-x="dom-window-statusbar">statusbar</span>;
  [Replaceable] readonly attribute <span>BarProp</span> <span data-x="dom-window-toolbar">toolbar</span>;
  attribute DOMString <span data-x="dom-window-status">status</span>;
  undefined <span data-x="dom-window-close">close</span>();
  readonly attribute boolean <span data-x="dom-window-closed">closed</span>;
  undefined <span data-x="dom-window-stop">stop</span>();
  undefined <span data-x="dom-window-focus">focus</span>();
  undefined <span data-x="dom-window-blur">blur</span>();

  // other browsing contexts
  [Replaceable] readonly attribute <span>WindowProxy</span> <span data-x="dom-frames">frames</span>;
  [Replaceable] readonly attribute unsigned long <span data-x="dom-length">length</span>;
  [<span>LegacyUnforgeable</span>] readonly attribute <span>WindowProxy</span>? <span data-x="dom-top">top</span>;
  attribute any <span data-x="dom-opener">opener</span>;
  [Replaceable] readonly attribute <span>WindowProxy</span>? <span data-x="dom-parent">parent</span>;
  readonly attribute <span>Element</span>? <span data-x="dom-frameElement">frameElement</span>;
  <span>WindowProxy</span>? <span data-x="dom-open">open</span>(optional USVString url = "", optional DOMString target = "_blank", optional [<span>LegacyNullToEmptyString</span>] DOMString features = "");

  // Since this is the global object, the IDL named getter adds a NamedPropertiesObject exotic
  // object on the prototype chain. Indeed, this does not make the global object an exotic object.
  // Indexed access is taken care of by the <span>WindowProxy</span> exotic object.
  <a href="#dom-window-nameditem">getter</a> <span data-x="idl-object">object</span> (DOMString name);

  // the user agent
  readonly attribute <span>Navigator</span> <span data-x="dom-navigator">navigator</span>;
  [Replaceable] readonly attribute <span>Navigator</span> <span data-x="dom-clientInformation">clientInformation</span>; // legacy alias of .navigator
  readonly attribute boolean <span data-x="dom-originAgentCluster">originAgentCluster</span>;

  // user prompts
  undefined <span data-x="dom-alert-noargs">alert</span>();
  undefined <span data-x="dom-alert">alert</span>(DOMString message);
  boolean <span data-x="dom-confirm">confirm</span>(optional DOMString message = "");
  DOMString? <span data-x="dom-prompt">prompt</span>(optional DOMString message = "", optional DOMString default = "");
  undefined <span data-x="dom-print">print</span>();

  undefined <span data-x="dom-window-postMessage">postMessage</span>(any message, USVString targetOrigin, optional sequence&lt;<span data-x="idl-object">object</span>&gt; transfer = []);
  undefined <span data-x="dom-window-postMessage-options">postMessage</span>(any message, optional <span>WindowPostMessageOptions</span> options = {});

  // <a href="#Window-partial">also has obsolete members</a>
};
<span>Window</span> includes <span>GlobalEventHandlers</span>;
<span>Window</span> includes <span>WindowEventHandlers</span>;

dictionary <dfn dictionary>WindowPostMessageOptions</dfn> : <span>StructuredSerializeOptions</span> {
  USVString <dfn dict-member for="WindowPostMessageOptions" data-x="dom-WindowPostMessageOptions-targetOrigin">targetOrigin</dfn> = "/";
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-window">window</span></code></dt>
   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-frames">frames</span></code></dt>
   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-self">self</span></code></dt>
   <dd><p>These attributes all return <var>window</var>.</p></dd>


   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-document">document</span></code></dt>
   <dd><p>Returns the <code>Document</code> associated with <var>window</var>.</p></dd>


   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-defaultView">defaultView</span></code></dt>
   <dd><p>Returns the <code>Window</code> associated with <var>document</var>, if there is one, or null otherwise.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <code>Window</code> object has an <dfn export data-x="concept-document-window">associated
  <code>Document</code></dfn>, which is a <code>Document</code> object. It is set when the
  <code>Window</code> object is created, and only ever changed during <span
  data-x="navigate">navigation</span> from the <span data-x="is initial about:blank">initial
  <code>about:blank</code></span> <code>Document</code>.</p>

  <p>A <code>Window</code>'s <dfn export for="Window" data-x="window bc">browsing context</dfn> is
  its <span data-x="concept-document-window">associated <code>Document</code></span>'s <span
  data-x="concept-document-bc">browsing context</span>. <span class="note">It is either null or a
  <span>browsing context</span>.</span></p>

  <p>A <code>Window</code>'s <dfn export for="Window" data-x="window navigable">navigable</dfn> is
  the <span>navigable</span> whose <span data-x="nav-document">active document</span> is the
  <code>Window</code>'s <span data-x="concept-document-window">associated
  <code>Document</code></span>'s, or null if there is no such <span>navigable</span>.</p>

  <p>The <dfn attribute for="Window"><code data-x="dom-window">window</code></dfn>, <dfn attribute
  for="Window"><code data-x="dom-frames">frames</code></dfn>, and <dfn attribute for="Window"><code
  data-x="dom-self">self</code></dfn> getter steps are to return <span>this</span>'s <span
  data-x="concept-relevant-realm">relevant realm</span>.[[GlobalEnv]].[[GlobalThisValue]].</p>

  <p>The <dfn attribute for="Window"><code data-x="dom-document">document</code></dfn> getter steps
  are to return <span>this</span>'s <span data-x="concept-document-window">associated
  <code>Document</code></span>.</p>

  <p class="note">The <code>Document</code> object associated with a <code>Window</code> object can
  change in exactly one case: when the <span>navigate</span> algorithm <span
  data-x="create-the-document-object">creates a new <code>Document</code> object</span> for the
  first page loaded in a <span>browsing context</span>. In that specific case, the
  <code>Window</code> object of the <span data-x="is initial about:blank">initial
  <code>about:blank</code></span> page is reused and gets a new <code>Document</code> object.</p>

  <p>The <dfn attribute for="Document"><code
  data-x="dom-document-defaultView">defaultView</code></dfn> getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="concept-document-bc">browsing context</span> is null,
   then return null.</p></li>

   <li><p>Return <span>this</span>'s <span data-x="concept-document-bc">browsing context</span>'s
   <code>WindowProxy</code> object.</p></li>
  </ol>

  <hr>

  <p>For historical reasons, <code>Window</code> objects must also have a writable, configurable,
  non-enumerable property named <dfn><code>HTMLDocument</code></dfn> whose value is the
  <code>Document</code> <span>interface object</span>.</p>

  </div>


  <h5 id="apis-for-creating-and-navigating-browsing-contexts-by-name">Opening and closing windows</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>window</var> = <var>window</var>.<span subdfn data-x="dom-open">open</span>([ <var>url</var> [, <var>target</var> [, <var>features</var> ] ] ])</code></dt>
   <dd>
    <p>Opens a window to show <var>url</var> (defaults to "<code>about:blank</code>"), and returns
    it. <var>target</var> (defaults to "<code data-x="">_blank</code>") gives the name of the new
    window. If a window already exists with that name, it is reused. The <var>features</var>
    argument can contain a <span>set of comma-separated tokens</span>:</p>

    <dl>
     <dt>"<code data-x="">noopener</code>"</dt>
     <dt>"<code data-x="">noreferrer</code>"</dt>
     <dd><p>These behave equivalently to the <code data-x="rel-noopener">noopener</code> and <code
     data-x="rel-noreferrer">noreferrer</code> link types on <span
     data-x="hyperlink">hyperlinks</span>.</p></dd>

     <dt>"<code data-x="">popup</code>"</dt>
     <dd><p>Encourages user agents to provide a minimal web browser user interface for the new
     window. (Impacts the <code data-x="dom-BarProp-visible">visible</code> getter on all
     <code>BarProp</code> objects as well.)</p></dd>
    </dl>

    <pre class="example"><code class="js">globalThis.open("https://email.example/message/CAOOOkFcWW97r8yg=SsWg7GgCmp4suVX9o85y8BvNRqMjuc5PXg", undefined, "noopener,popup");</code></pre>
   </dd>

   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-name">name</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the name of the window.</p>

    <p>Can be set, to change the name.</p>
   </dd>

   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-window-close">close</span>()</code></dt>
   <dd><p>Closes the window.</p></dd>

   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-window-closed">closed</span></code></dt>
   <dd><p>Returns true if the window has been closed, false otherwise.</p></dd>

   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-window-stop">stop</span>()</code></dt>
   <dd><p>Cancels the document load.</p></dd>
  </dl>

  <div w-nodev>

  <p>To <dfn>get noopener for window open</dfn>, given a <code>Document</code>
  <var>sourceDocument</var>, an <span>ordered map</span> <var>tokenizedFeatures</var>, and a
  <span>URL record</span>-or-null <var>url</var>, perform the following steps. They return a
  boolean.</p>

  <ol>
   <li>
    <p>If <var>url</var> is not null and <var>url</var>'s <span
    data-x="concept-url-blob-entry">blob URL entry</span> is not null:</p>

    <ol>
     <li><p>Let <var>blobOrigin</var> be <var>url</var>'s <span
     data-x="concept-url-blob-entry">blob URL entry</span>'s <span
     data-x="blob-url-entry-environment">environment</span>'s <span
     data-x="concept-settings-object-origin">origin</span>.</p></li>

     <li><p>Let <var>topLevelOrigin</var> be <var>sourceDocument</var>'s <span>relevant settings
     object</span>'s <span>top-level origin</span>.</p></li>

     <li><p>If <var>blobOrigin</var> is not <span>same site</span> with <var>topLevelOrigin</var>,
     then return true.</p></li>
    </ol>
   </li>

   <li><p>Let <var>noopener</var> be false.</p></li>

   <li><p>If <var>tokenizedFeatures</var>["<code data-x="">noopener</code>"] <span data-x="map
   exists">exists</span>, then set <var>noopener</var> to the result of <span
   data-x="concept-window-open-features-parse-boolean">parsing <var>tokenizedFeatures</var>["<code
   data-x="">noopener</code>"] as a boolean feature</span>.</p></li>

   <li><p>Return <var>noopener</var>.</p></li>
  </ol>

  <p>The <dfn>window open steps</dfn>, given a string <var>url</var>, a string <var>target</var>,
  and a string <var>features</var>, are as follows:</p>

  <ol>
   <li><p>If the <span>event loop</span>'s <span>termination nesting level</span> is nonzero,
   then return null.</p></li>

   <li><p>Let <var>sourceDocument</var> be the <span>entry global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>Let <var>urlRecord</var> be null.</p></li>

   <li>
    <p>If <var>url</var> is not the empty string, then:</p>

    <ol>
     <li><p>Set <var>urlRecord</var> to the result of <span>encoding-parsing a URL</span> given
     <var>url</var>, relative to <var>sourceDocument</var>.</p></li>

     <li><p>If <var>urlRecord</var> is failure, then throw a <span>"<code>SyntaxError</code>"</span>
     <code>DOMException</code>.</p></li>
    </ol>
   </li>

   <li><p>If <var>target</var> is the empty string, then set <var>target</var> to "<code
   data-x="">_blank</code>".</p></li>

   <li><p>Let <var>tokenizedFeatures</var> be the result of <span
   data-x="concept-window-open-features-tokenize">tokenizing</span> <var>features</var>.</p></li>

   <li><p>Let <var>noreferrer</var> be false.</p></li>

   <li><p>If <var>tokenizedFeatures</var>["<code data-x="">noreferrer</code>"] <span data-x="map
   exists">exists</span>, then set <var>noreferrer</var> to the result of <span
   data-x="concept-window-open-features-parse-boolean">parsing <var>tokenizedFeatures</var>["<code
   data-x="">noreferrer</code>"] as a boolean feature</span>.</p></li>

   <li><p>Let <var>noopener</var> be the result of <span
   data-x="get noopener for window open">getting noopener for window open</span> with
   <var>sourceDocument</var>, <var>tokenizedFeatures</var>, and <var>urlRecord</var>.</p></li>

   <li><p><span data-x="map remove">Remove</span> <var>tokenizedFeatures</var>["<code
   data-x="">noopener</code>"] and <var>tokenizedFeatures</var>["<code
   data-x="">noreferrer</code>"].</p></li>

   <li><p>Let <var>referrerPolicy</var> be the empty string.</p></li>

   <li><p>If <var>noreferrer</var> is true, then set <var>noopener</var> to true and set
   <var>referrerPolicy</var> to "<code data-x="">no-referrer</code>".</p></li>

   <li>
    <p>Let <var>targetNavigable</var> and <var>windowType</var> be the result of applying <span>the
    rules for choosing a navigable</span> given <var>target</var>, <var>sourceDocument</var>'s
    <span>node navigable</span>, and <var>noopener</var>.</p>

    <p class="example">If there is a user agent that supports control-clicking a link to open it in
    a new tab, and the user control-clicks on an element whose <code
    data-x="handler-onclick">onclick</code> handler uses the <code
    data-x="dom-open">window.open()</code> API to open a page in an <code>iframe</code> element,
    the user agent could override the selection of the target browsing context to instead target a
    new tab.</p>
   </li>

   <li><p>If <var>targetNavigable</var> is null, then return null.</p></li>

   <li>
    <p>If <var>windowType</var> is either "<code data-x="">new and unrestricted</code>" or "<code
    data-x="">new with no opener</code>", then:</p>

    <ol>
     <li><p>Set <var>targetNavigable</var>'s <span data-x="nav-bc">active browsing context</span>'s
     <span>is popup</span> to the result of <span data-x="popup-window-is-requested">checking if a
     popup window is requested</span>, given <var>tokenizedFeatures</var>.</p></li>

     <li><p><span>Set up browsing context features</span> for <var>targetNavigable</var>'s <span
     data-x="nav-bc">active browsing context</span> given
     <var>tokenizedFeatures</var>. <ref>CSSOMVIEW</ref></p></li>

     <li><p>If <var>urlRecord</var> is null, then set <var>urlRecord</var> to a <span>URL
     record</span> representing <code>about:blank</code>.</p></li>

     <li>
      <p>If <var>urlRecord</var> <span>matches <code>about:blank</code></span>, then perform the
      <span>URL and history update steps</span> given <var>targetNavigable</var>'s <span
      data-x="nav-document">active document</span> and <var>urlRecord</var>.</p>

      <p class="note">This is necessary in case <var>url</var> is something like <code
      data-x="">about:blank?foo</code>. If <var>url</var> is just plain <code
      data-x="">about:blank</code>, this will do nothing.</p>
     </li>

     <li><p>Otherwise, <span>navigate</span><!--DONAV window.open()--> <var>targetNavigable</var>
     to <var>urlRecord</var> using <var>sourceDocument</var>, with <i
     data-x="navigation-referrer-policy">referrerPolicy</i> set to <var>referrerPolicy</var> and
     <i><span>exceptionsEnabled</span></i> set to true.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><!--DONAV window.open()--><p>If <var>urlRecord</var> is not null, then
     <span>navigate</span> <var>targetNavigable</var> to <var>urlRecord</var> using
     <var>sourceDocument</var>, with <i data-x="navigation-referrer-policy">referrerPolicy</i> set
     to <var>referrerPolicy</var> and <i><span>exceptionsEnabled</span></i> set to true.</p></li>

     <li><p>If <var>noopener</var> is false, then set <var>targetNavigable</var>'s <span
     data-x="nav-bc">active browsing context</span>'s <span>opener browsing context</span> to
     <var>sourceDocument</var>'s <span data-x="concept-document-bc">browsing
     context</span>.</p></li>
    </ol>
   </li>

   <li><p>If <var>noopener</var> is true or <var>windowType</var> is "<code data-x="">new with no
   opener</code>", then return null.</p></li>

   <li><p>Return <var>targetNavigable</var>'s <span data-x="nav-wp">active
   <code>WindowProxy</code></span>.</p></li>
  </ol>

  <p>The <dfn for="Window" method><code data-x="dom-open">open(<var>url</var>, <var>target</var>,
  <var>features</var>)</code></dfn> method steps are to run the <span>window open steps</span> with
  <var>url</var>, <var>target</var>, and <var>features</var>.</p>

  <p class="note">The method provides a mechanism for <span data-x="navigate">navigating</span> an
  existing <span>browsing context</span> or opening and navigating an <span>auxiliary browsing
  context</span>.</p>

  <hr>

  <p>To <dfn data-x="concept-window-open-features-tokenize">tokenize the <var>features</var>
  argument</dfn>:</p>

  <ol>
   <li><p>Let <var>tokenizedFeatures</var> be a new <span>ordered map</span>.</p></li>

   <li><p>Let <var>position</var> point at the first code point of <var>features</var>.</p></li>

   <li>
    <p><span>While</span> <var>position</var> is not past the end of <var>features</var>:</p>

    <ol>
     <li><p>Let <var>name</var> be the empty string.</p></li>

     <li><p>Let <var>value</var> be the empty string.</p></li>

     <li><p><span>Collect a sequence of code points</span> that are <span data-x="feature
     separator">feature separators</span> from <var>features</var> given <var>position</var>. This
     skips past leading separators before the name.</p></li>

     <li><p><span>Collect a sequence of code points</span> that are not <span data-x="feature
     separator">feature separators</span> from <var>features</var> given <var>position</var>. Set
     <var>name</var> to the collected characters, <span>converted to ASCII lowercase</span>.</p></li>

     <li><p>Set <var>name</var> to the result of <span>normalizing the feature name</span>
     <var>name</var>.</p></li>

     <li>
      <p><span>While</span> <var>position</var> is not past the end of <var>features</var> and the
      code point at <var>position</var> in <var>features</var> is not U+003D (=):</p>

      <ol>
       <li><p>If the code point at <var>position</var> in <var>features</var> is U+002C (,), or if
       it is not a <span>feature separator</span>, then <span>break</span>.</p></li>

       <li><p>Advance <var>position</var> by 1.</p></li>
      </ol>

      <p class="note">This skips to the first U+003D (=) but does not skip past a U+002C (,) or a
      non-separator.</p>
     </li>

     <li>
      <p>If the code point at <var>position</var> in <var>features</var> is a <span>feature
      separator</span>:</p>

      <ol>
       <li>
        <p>While <var>position</var> is not past the end of <var>features</var> and the code point
        at <var>position</var> in <var>features</var> is a <span>feature separator</span>:</p>

        <ol>
         <li><p>If the code point at <var>position</var> in <var>features</var> is U+002C (,), then
         <span>break</span>.</p></li>

         <li><p>Advance <var>position</var> by 1.</p></li>
        </ol>

        <p class="note">This skips to the first non-separator but does not skip past a U+002C
        (,).</p>
       </li>

       <li><p><span>Collect a sequence of code points</span> that are not <span data-x="feature
       separator">feature separators</span> code points from <var>features</var> given
       <var>position</var>. Set <var>value</var> to the collected code points, <span>converted to
       ASCII lowercase</span>.</p></li>
      </ol>
     </li>

     <li><p>If <var>name</var> is not the empty string, then set
     <var>tokenizedFeatures</var>[<var>name</var>] to <var>value</var>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>tokenizedFeatures</var>.</p></li>
  </ol>

  <p>To <dfn data-x="window-feature-is-set">check if a window feature is set</dfn>, given
  <var>tokenizedFeatures</var>, <var>featureName</var>, and <var>defaultValue</var>:</p>

  <ol>
   <li><p>If <var>tokenizedFeatures</var>[<var>featureName</var>] <span data-x="map
   exists">exists</span>, then return the result of <span
   data-x="concept-window-open-features-parse-boolean">parsing
   <var>tokenizedFeatures</var>[<var>featureName</var>] as a boolean feature</span>.</p></li>

   <li><p>Return <var>defaultValue</var>.</p></li>
  </ol>

  <p>To <dfn data-x="popup-window-is-requested">check if a popup window is requested</dfn>, given
  <var>tokenizedFeatures</var>:</p>

  <ol>
   <li><p>If <var>tokenizedFeatures</var> is <span data-x="map empty">empty</span>, then return
   false.</p></li>

   <li><p>If <var>tokenizedFeatures</var>["<code data-x="">popup</code>"] <span data-x="map
   exists">exists</span>, then return the result of <span
   data-x="concept-window-open-features-parse-boolean">parsing
   <var>tokenizedFeatures</var>["<code data-x="">popup</code>"] as a boolean
   feature</span>.</p></li>

   <li><p>Let <var>location</var> be the result of <span data-x="window-feature-is-set">checking if
   a window feature is set</span>, given <var>tokenizedFeatures</var>, "<code
   data-x="">location</code>", and false.</p></li>

   <li><p>Let <var>toolbar</var> be the result of <span data-x="window-feature-is-set">checking if
   a window feature is set</span>, given <var>tokenizedFeatures</var>, "<code
   data-x="">toolbar</code>", and false.</p></li>

   <li><p>If <var>location</var> and <var>toolbar</var> are both false, then return true.</p></li>

   <li><p>Let <var>menubar</var> be the result of <span data-x="window-feature-is-set">checking if
   a window feature is set</span>, given <var>tokenizedFeatures</var>, "<code
   data-x="">menubar</code>", and false.</p></li>

   <li><p>If <var>menubar</var> is false, then return true.</p></li>

   <li><p>Let <var>resizable</var> be the result of <span data-x="window-feature-is-set">checking if
   a window feature is set</span>, given <var>tokenizedFeatures</var>, "<code
   data-x="">resizable</code>", and true.</p></li>

   <li><p>If <var>resizable</var> is false, then return true.</p></li>

   <li><p>Let <var>scrollbars</var> be the result of <span data-x="window-feature-is-set">checking
   if a window feature is set</span>, given <var>tokenizedFeatures</var>, "<code
   data-x="">scrollbars</code>", and false.</p></li>

   <li><p>If <var>scrollbars</var> is false, then return true.</p></li>

   <li><p>Let <var>status</var> be the result of <span data-x="window-feature-is-set">checking if
   a window feature is set</span>, given <var>tokenizedFeatures</var>, "<code
   data-x="">status</code>", and false.</p></li>

   <li><p>If <var>status</var> is false, then return true.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  <p>A code point is a <dfn>feature separator</dfn> if it is <span>ASCII whitespace</span>, U+003D
  (=), or U+002C (,).</p>

  <p>For legacy reasons, there are some aliases of some feature names. To <dfn data-x="normalizing
  the feature name">normalize a feature name</dfn> <var>name</var>, switch on <var>name</var>:</p>

  <dl class="switch">
   <dt>"<code data-x="">screenx</code>"
   <dd>Return "<code data-x="">left</code>".
   <dt>"<code data-x="">screeny</code>"
   <dd>Return "<code data-x="">top</code>".
   <dt>"<code data-x="">innerwidth</code>"
   <dd>Return "<code data-x="">width</code>".
   <dt>"<code data-x="">innerheight</code>"
   <dd>Return "<code data-x="">height</code>".
   <dt>Anything else
   <dd>Return <var>name</var>.
  </dl>

  <p>To <dfn data-x="concept-window-open-features-parse-boolean">parse a boolean feature</dfn> given
  a string <var>value</var>:</p>

  <ol>
   <li><p>If <var>value</var> is the empty string, then return true.</p></li>

   <li><p>If <var>value</var> <span>is</span> "<code data-x="">yes</code>", then return
   true.</p></li>

   <li><p>If <var>value</var> <span>is</span> "<code data-x="">true</code>", then return
   true.</p></li>

   <li><p>Let <var>parsed</var> be the result of <span data-x="rules for parsing integers">parsing
   <var>value</var> as an integer</span>.</p></li>

   <li><p>If <var>parsed</var> is an error, then set it to 0.</p></li>

   <li><p>Return false if <var>parsed</var> is 0, and true otherwise.</p></li>
  </ol>

  <hr>

  <p>The <dfn attribute for="Window"><code data-x="dom-name">name</code></dfn> getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="window navigable">navigable</span> is null, then
   return the empty string.</p></li>

   <li><p>Return <span>this</span>'s <span data-x="window navigable">navigable</span>'s <span
   data-x="nav-target">target name</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-name">name</code> setter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="window navigable">navigable</span> is null, then
   return.</p></li>

   <li><p>Set <span>this</span>'s <span data-x="window navigable">navigable</span>'s <span
   data-x="nav-active-history-entry">active session history entry</span>'s <span
   data-x="she-document-state">document state</span>'s <span
   data-x="document-state-nav-target-name">navigable target name</span> to the given value.</p></li>
  </ol>

  <p class="note">The name <a href="#resetBCName">gets reset</a> when the navigable is <span
  data-x="navigate">navigated</span> to another <span>origin</span>.</p>

  <hr>

  <p>The <dfn method for="Window"><code data-x="dom-window-close">close()</code></dfn> method steps
  are:</p>

  <ol>
   <li><p>Let <var>thisTraversable</var> be <span>this</span>'s <span data-x="window
   navigable">navigable</span>.</p></li>

   <li><p>If <var>thisTraversable</var> is not a <span>top-level traversable</span>, then
   return.</p></li>

   <li><p>If <var>thisTraversable</var>'s <span>is closing</span> is true, then return.</p></li>

   <li><p>Let <var>browsingContext</var> be <var>thisTraversable</var>'s <span
   data-x="nav-bc">active browsing context</span>.</p></li>

   <li><p>Let <var>sourceSnapshotParams</var> be the result of <span>snapshotting source snapshot
   params</span> given <var>thisTraversable</var>'s <span data-x="nav-document">active
   document</span>.</p></li>

   <li>
    <p>If all the following are true:</p>

    <ul>
     <li><p><var>thisTraversable</var> is <span>script-closable</span>;</p></li>

     <li><p>the <span data-x="concept-incumbent-global">incumbent global object</span>'s <span
     data-x="window bc">browsing context</span> is <span>familiar with</span>
     <var>browsingContext</var>; and</p></li>

     <li id="sandboxClose"><p>the <span data-x="concept-incumbent-global">incumbent global
     object</span>'s <span data-x="window navigable">navigable</span> is <span>allowed by
     sandboxing to navigate</span> <var>thisTraversable</var>, given
     <var>sourceSnapshotParams</var>,</p></li>
    </ul>

    <p>then:</p>

    <ol>
     <li><p>Set <var>thisTraversable</var>'s <span>is closing</span> to true.</p></li>

     <li><p><span>Queue a task</span> on the <span>DOM manipulation task source</span> to <span
     data-x="definitely close a top-level traversable">definitely close</span>
     <var>thisTraversable</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>A <span>navigable</span> is <dfn>script-closable</dfn> if its <span data-x="nav-bc">active
  browsing context</span> is an <span>auxiliary browsing context</span> that was created by a
  script (as opposed to by an action of the user), or if it is a <span>top-level traversable</span>
  whose <span data-x="tn-session-history-entries">session history entries</span>'s <span
  data-x="list size">size</span> is 1.</p>

  <p>The <dfn attribute for="Window"><code data-x="dom-window-closed">closed</code></dfn> getter
  steps are to return true if <span>this</span>'s <span data-x="window bc">browsing context</span>
  is null or its <span>is closing</span> is true; otherwise false.</p>

  <p>The <dfn method for="Window"><code data-x="dom-window-stop">stop()</code></dfn> method steps
  are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="window navigable">navigable</span> is null, then
   return.</p></li>

   <li><p><span data-x="nav-stop">Stop loading</span> <span>this</span>'s <span data-x="window
   navigable">navigable</span>.</p></li>
  </ol>

  </div>


  <h5 id="accessing-other-browsing-contexts">Indexed access on the <code>Window</code> object</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-length">length</span></code></dt>

   <dd><p>Returns the number of <span>document-tree child navigables</span>.</p></dd>

   <dt><code data-x=""><var>window</var>[<var>index</var>]</code></dt>

   <dd><p>Returns the <code>WindowProxy</code> corresponding to the indicated <span
   data-x="document-tree child navigables">document-tree child navigables</span>.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="Window"><code data-x="dom-length">length</code></dfn> getter steps are
  to return <span>this</span>'s <span data-x="concept-document-window">associated
  <code>Document</code></span>'s <span>document-tree child navigables</span>'s <span data-x="list
  size">size</span>.</p>

  <p class="note">Indexed access to <span>document-tree child navigables</span> is defined through
  the <a href="#windowproxy-getownproperty">[[GetOwnProperty]]</a> internal method of the
  <code>WindowProxy</code> object.</p>

  </div>


  <h5>Named access on the <code>Window</code> object</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>window</var>[<var>name</var>]</code></dt>

   <dd>
    <p>Returns the indicated element or collection of elements.</p>

    <p>As a general rule, relying on this will lead to brittle code. Which IDs end up mapping to
    this API can vary over time, as new features are added to the web platform, for example. Instead
    of this, use <code data-x="">document.getElementById()</code> or <code
    data-x="">document.querySelector()</code>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p id="child-browsing-context-name-property-set"><span
  id="document-tree-child-browsing-context-name-property-set"></span>The <dfn>document-tree child
  navigable target name property set</dfn> of a <code>Window</code> object <var>window</var> is the
  return value of running these steps:</p>

  <ol>
   <li><p>Let <var>children</var> be the <span>document-tree child navigables</span> of
   <var>window</var>'s <span data-x="concept-document-window">associated
   <code>Document</code></span>.</p></li>

   <li><p>Let <var>firstNamedChildren</var> be an empty <span data-x="set">ordered
   set</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>navigable</var> of <var>children</var>:</p>

    <ol>
     <li><p>Let <var>name</var> be <var>navigable</var>'s <span
     data-x="nav-target">target name</span>.</p></li>

     <li><p>If <var>name</var> is the empty string, then <span>continue</span>.</p></li>

     <li><p>If <var>firstNamedChildren</var> <span data-x="list contains">contains</span> a
     <span>navigable</span> whose <span data-x="nav-target">target name</span> is <var>name</var>,
     then <span>continue</span>.</p></li>

     <li><p><span data-x="list append">Append</span> <var>navigable</var> to
     <var>firstNamedChildren</var>.</p></li>
    </ol>
   </li>

   <li><p>Let <var>names</var> be an empty <span data-x="set">ordered set</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>navigable</var> of
    <var>firstNamedChildren</var>:</p>

    <ol>
     <li><p>Let <var>name</var> be <var>navigable</var>'s <span
     data-x="nav-target">target name</span>.</p></li>

     <li><p>If <var>navigable</var>'s <span data-x="nav-document">active document</span>'s <span
     data-x="concept-document-origin">origin</span> is <span>same origin</span> with
     <var>window</var>'s <span>relevant settings object</span>'s <span
     data-x="concept-settings-object-origin">origin</span>, then <span data-x="list
     append">append</span> <var>name</var> to <var>names</var>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>names</var>.</p></li>
  </ol>

  <div class="example">
   <p>The two seperate iterations mean that in the following example, hosted on <code
   data-x="">https://example.org/</code>, assuming <code data-x="">https://elsewhere.example/</code>
   sets <code data-x="dom-name">window.name</code> to "<code data-x="">spices</code>", evaluating
   <code data-x="">window.spices</code> after everything has loaded will yield undefined:</p>

   <pre><code class="html">&lt;iframe src=https://elsewhere.example.com/>&lt;/iframe>
&lt;iframe name=spices>&lt;/iframe></code></pre>
  </div>

  <p>The <code>Window</code> object <span data-x="support named properties">supports named
  properties</span>. The <span>supported property names</span> of a <code>Window</code> object
  <var>window</var> at any moment consist of the following, in <span>tree order</span> according to
  the element that contributed them, ignoring later duplicates:</p>

  <!-- KEEP THIS LIST IN SYNC WITH "NAMED OBJECTS" BELOW -->
  <ul>
   <li><p><var>window</var>'s <span>document-tree child navigable target name property
   set</span>;</p></li>

   <li><p>the value of the <code data-x="">name</code> content attribute for all <code>embed</code>,
   <code>form</code>, <code>img</code>, and <code>object</code> elements that
   have a non-empty <code data-x="">name</code> content attribute and are <span>in a document
   tree</span> with <var>window</var>'s <span data-x="concept-document-window">associated
   <code>Document</code></span> as their <span>root</span>; and</p></li>

   <li><p>the value of the <code data-x="attr-id">id</code> content attribute for all <span>HTML
   elements</span> that have a non-empty <code data-x="attr-id">id</code> content attribute and are
   <span>in a document tree</span> with <var>window</var>'s <span
   data-x="concept-document-window">associated <code>Document</code></span> as their
   <span>root</span>.</p></li>
  </ul>

  <p id="dom-window-nameditem">To <span>determine the value of a named property</span>
  <var>name</var> in a <code>Window</code> object <var>window</var>, the user agent must return the
  value obtained using the following steps:</p>

  <ol>
   <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1716 -->
   <li>
    <p>Let <var>objects</var> be the list of <span data-x="dom-window-namedItem-filter">named
    objects</span> of <var>window</var> with the name <var>name</var>.</p>

    <p class="note">There will be at least one such object, since the algorithm would otherwise not
    have been <span data-x="named-properties-object-getownproperty">invoked by Web IDL</span>.</p>
   </li>

   <li>
    <p>If <var>objects</var> contains a <span>navigable</span>, then:</p>

    <ol>
     <li><p>Let <var>container</var> be the first <span>navigable container</span> in
     <var>window</var>'s <span data-x="concept-document-window">associated
     <code>Document</code></span>'s <span data-x="descendant">descendants</span> whose
     <span>content navigable</span> is in <var>objects</var>.</p></li>

     <li><p>Return <var>container</var>'s <span>content navigable</span>'s <span
     data-x="nav-wp">active <code>WindowProxy</code></span>.</p></li>
    </ol>
   </li>

   <li><p>Otherwise, if <var>objects</var> has only one element, return that element.</p></li>

   <li><p>Otherwise, return an <code>HTMLCollection</code> rooted at <var>window</var>'s <span
   data-x="concept-document-window">associated <code>Document</code></span>, whose filter matches
   only <span data-x="dom-window-namedItem-filter">named objects</span> of <var>window</var> with
   the name <var>name</var>. (By definition, these will all be elements.)</p></li>
   <!-- the same one each time is returned, because of the rule under collections -->
  </ol>

  <p><dfn data-x="dom-window-nameditem-filter">Named objects</dfn> of <code>Window</code> object
  <var>window</var> with the name <var>name</var>, for the purposes of the above algorithm, consist
  of the following:</p>

  <!-- KEEP THIS LIST IN SYNC WITH SUPPORTED PROPERTY NAMES ABOVE -->
  <ul>
   <li><p><span>document-tree child navigables</span> of <var>window</var>'s <span
   data-x="concept-document-window">associated <code>Document</code></span> whose <span
   data-x="nav-target">target name</span> is <var>name</var>;</p></li>

   <li><p><code>embed</code>, <code>form</code>, <code>img</code>, or
   <code>object</code> elements that have a <code data-x="">name</code> content attribute whose
   value is <var>name</var> and are <span>in a document tree</span> with <var>window</var>'s <span
   data-x="concept-document-window">associated <code>Document</code></span> as their
   <span>root</span>; and</p></li>

   <li><p><span>HTML elements</span> that have an <code data-x="attr-id">id</code> content attribute
   whose value is <var>name</var> and are <span>in a document tree</span> with <var>window</var>'s
   <span data-x="concept-document-window">associated <code>Document</code></span> as their
   <span>root</span>.</p></li>
  </ul>

  <p class="note">Since the <code>Window</code> interface has the <code
  data-x="Global">[Global]</code> extended attribute, its named properties follow the rules for
  <span data-x="named properties object">named properties objects</span> rather than <span
  data-x="legacy platform object">legacy platform objects</span>.</p>

  </div>


  <h5 id="navigating-nested-browsing-contexts-in-the-dom">Accessing related windows</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-top">top</span></code></dt>
   <dd><p>Returns the <code>WindowProxy</code> for the <span>top-level traversable</span>.</p></dd>

   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-opener">opener</span> [ = <var>value</var> ]</code></dt>
   <dd>
    <p>Returns the <code>WindowProxy</code> for the <span>opener browsing context</span>.</p>

    <p>Returns null if there isn't one or if it has been set to null.</p>

    <p>Can be set to null.</p>
   </dd>

   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-parent">parent</span></code></dt>
   <dd><p>Returns the <code>WindowProxy</code> for the <span data-x="nav-parent">parent
   navigable</span>.</p></dd>

   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-frameElement">frameElement</span></code></dt>
   <dd>
    <p>Returns the <span>navigable container</span> element.</p>

    <p>Returns null if there isn't one, and in cross-origin situations.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="Window"><code data-x="dom-top">top</code></dfn> getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="window navigable">navigable</span> is null, then
   return null.</p></li>

   <li><p>Return <span>this</span>'s <span data-x="window navigable">navigable</span>'s <span
   data-x="nav-top">top-level traversable</span>'s <span data-x="nav-wp">active
   <code>WindowProxy</code></span>.</p></li>
  </ol>

  <p>The <dfn attribute for="Window"><code data-x="dom-opener">opener</code></dfn> getter steps
  are:</p>

  <ol>
   <li><p>Let <var>current</var> be <span>this</span>'s <span data-x="window bc">browsing
   context</span>.</p></li>

   <li><p>If <var>current</var> is null, then return null.</p></li>

   <li><p>If <var>current</var>'s <span>opener browsing context</span> is null, then return
   null.</p></li>

   <li><p>Return <var>current</var>'s <span>opener browsing context</span>'s
   <code>WindowProxy</code> object.</p></li>
  </ol>

  <p>The <code data-x="dom-opener">opener</code> setter steps are:</p>

  <ol>
   <li><p>If the given value is null and <span>this</span>'s <span data-x="window bc">browsing
   context</span> is non-null, then set <span>this</span>'s <span data-x="window bc">browsing
   context</span>'s <span>opener browsing context</span> to null.</p></li>

   <li><p>If the given value is non-null, then perform ?
   <span>DefinePropertyOrThrow</span>(<span>this</span>, "<code data-x="">opener</code>", {
   [[Value]]: the given value, [[Writable]]: true, [[Enumerable]]: true, [[Configurable]]: true
   }).</p></li>
  </ol>

  <div class="note">
   <p>Setting <code data-x="dom-opener">window.opener</code> to null clears the <span>opener
   browsing context</span> reference. In practice, this prevents future scripts from accessing their
   <span>opener browsing context</span>'s <code>Window</code> object.</p>

   <p>By default, scripts can access their <span>opener browsing context</span>'s
   <code>Window</code> object through the <code data-x="dom-opener">window.opener</code> getter.
   E.g., a script can set <code data-x="">window.opener.location</code>, causing the <span>opener
   browsing context</span> to navigate.</p>
  </div>

  <p>The <dfn attribute for="Window"><code data-x="dom-parent">parent</code></dfn> getter steps
  are:</p>

  <ol>
   <li><p>Let <var>navigable</var> be <span>this</span>'s <span data-x="window
   navigable">navigable</span>.</p></li>

   <li><p>If <var>navigable</var> is null, then return null.</p></li>

   <li><p>If <var>navigable</var>'s <span data-x="nav-parent">parent</span> is not null, then set
   <var>navigable</var> to <var>navigable</var>'s <span data-x="nav-parent">parent</span>.</p></li>

   <li><p>Return <var>navigable</var>'s <span data-x="nav-wp">active
   <code>WindowProxy</code></span>.</p></li>
  </ol>

  <p>The <dfn attribute for="Window"><code data-x="dom-frameElement">frameElement</code></dfn>
  getter steps are:</p>

  <ol>
   <li><p>Let <var>current</var> be <span>this</span>'s <span>node navigable</span>.</p></li>

   <li><p>If <var>current</var> is null, then return null.</p></li>

   <li><p>Let <var>container</var> be <var>current</var>'s <span
   data-x="nav-container">container</span>.</p></li>

   <li><p>If <var>container</var> is null, then return null.</p></li>

   <li><p>If <var>container</var>'s <span>node document</span>'s <span
   data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with the
   <span>current settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then return null.</p></li>

   <li><p>Return <var>container</var>.</p></li>
  </ol>

  <div class="example">
   <p>An example of when these properties can return null is as follows:</p>

   <pre><code class="html">&lt;!DOCTYPE html>
&lt;iframe>&lt;/iframe>

&lt;script>
"use strict";
const element = document.querySelector("iframe");
const iframeWindow = element.contentWindow;
element.remove();

console.assert(iframeWindow.top === null);
console.assert(iframeWindow.parent === null);
console.assert(iframeWindow.frameElement === null);
&lt;/script></code></pre>

   <p>Here the <span>browsing context</span> corresponding to <code data-x="">iframeWindow</code>
   was <span data-x="destroy a document">nulled out</span> when <code data-x="">element</code> was
   removed from the document.</p>
  </div>

  </div>


  <h5 id="browser-interface-elements">Historical browser interface element APIs</h5>

  <p>For historical reasons, the <code>Window</code> interface had some properties that represented
  the visibility of certain web browser interface elements.</p>

  <p w-nodev>For privacy and interoperability reasons, those properties now return values that
  represent whether the <code>Window</code>'s <span data-x="window bc">browsing context</span>'s
  <span>is popup</span> property is true or false.</p>

  <p w-dev>For privacy and interoperability reasons, those properties now return all return the same value:
  whether or not the window represents a popup window.</p>

  <p w-nodev>Each interface element is represented by a <code>BarProp</code> object:</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>BarProp</dfn> {
  readonly attribute boolean <span data-x="dom-BarProp-visible">visible</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-window-locationbar">locationbar</span>.<span subdfn data-x="dom-BarProp-visible">visible</span></code></dt>
   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-window-menubar">menubar</span>.<span data-x="dom-BarProp-visible">visible</span></code></dt>
   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-window-personalbar">personalbar</span>.<span data-x="dom-BarProp-visible">visible</span></code></dt>
   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-window-scrollbars">scrollbars</span>.<span data-x="dom-BarProp-visible">visible</span></code></dt>
   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-window-statusbar">statusbar</span>.<span data-x="dom-BarProp-visible">visible</span></code></dt>
   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-window-toolbar">toolbar</span>.<span data-x="dom-BarProp-visible">visible</span></code></dt>
   <dd><p>Returns true if the <code>Window</code> is not a popup; otherwise, returns false.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="BarProp"><code data-x="dom-BarProp-visible">visible</code></dfn> getter
  steps are:</p>

  <ol>
   <li><p>Let <var>browsingContext</var> be <span>this</span>'s <span>relevant global
   object</span>'s <span data-x="window bc">browsing context</span>.</p></li>

   <li><p>If <var>browsingContext</var> is null, then return true.</p></li>

   <li><p>Return the negation of <var>browsingContext</var>'s <span data-x="bc-tlbc">top-level
   browsing context</span>'s <span>is popup</span>.</p></li>
  </ol>

  <p>The following <code>BarProp</code> objects must exist for each <code>Window</code> object:</p>

  <dl>
   <dt><dfn>The location bar <code>BarProp</code> object</dfn></dt>
   <dd>Historically represented the user interface element that contains a control that displays the
   browser's location bar.</dd>

   <dt><dfn>The menu bar <code>BarProp</code> object</dfn></dt>
   <dd>Historically represented the user interface element that contains a list of commands in menu
   form, or some similar interface concept.</dd>

   <dt><dfn>The personal bar <code>BarProp</code> object</dfn></dt>
   <dd>Historically represented the user interface element that contains links to the user's
   favorite pages, or some similar interface concept.</dd>

   <dt><dfn>The scrollbar <code>BarProp</code> object</dfn></dt>
   <dd>Historically represented the user interface element that contains a scrolling mechanism, or
   some similar interface concept.</dd>

   <dt><dfn>The status bar <code>BarProp</code> object</dfn></dt>
   <dd>Historically represented a user interface element found immediately below or after the
   document, as appropriate for the user's media, which typically provides information about ongoing
   network activity or information about elements that the user's pointing device is currently
   indicating.</dd>

   <dt><dfn>The toolbar <code>BarProp</code> object</dfn></dt>
   <dd>Historically represented the user interface element found immediately above or before the
   document, as appropriate for the user's media, which typically provides <span data-x="apply the
   traverse history step">session history traversal</span> controls (back and forward buttons,
   reload buttons, etc.).</dd>
  </dl>

  <p>The <dfn attribute for="Window"><code data-x="dom-window-locationbar">locationbar</code></dfn>
  attribute must return <span>the location bar <code>BarProp</code> object</span>.</p>

  <p>The <dfn attribute for="Window"><code data-x="dom-window-menubar">menubar</code></dfn>
  attribute must return <span>the menu bar <code>BarProp</code> object</span>.</p>

  <p>The <dfn attribute for="Window"><code data-x="dom-window-personalbar">personalbar</code></dfn>
  attribute must return <span>the personal bar <code>BarProp</code> object</span>.</p>

  <p>The <dfn attribute for="Window"><code data-x="dom-window-scrollbars">scrollbars</code></dfn>
  attribute must return <span>the scrollbar <code>BarProp</code> object</span>.</p>

  <p>The <dfn attribute for="Window"><code data-x="dom-window-statusbar">statusbar</code></dfn>
  attribute must return <span>the status bar <code>BarProp</code> object</span>.</p>

  <p>The <dfn attribute for="Window"><code data-x="dom-window-toolbar">toolbar</code></dfn>
  attribute must return <span>the toolbar <code>BarProp</code> object</span>.</p>

  <hr>

  <p>For historical reasons, the <dfn attribute for="Window"><code
  data-x="dom-window-status">status</code></dfn> attribute on the <code>Window</code> object must,
  on getting, return the last string it was set to, and on setting, must set itself to the new
  value. When the <code>Window</code> object is created, the attribute must be set to the empty
  string. It does not do anything else.</p>


  <h5>Script settings for <code>Window</code> objects</h5>

  <p>To <dfn>set up a window environment settings object</dfn>, given a <span>URL</span>
  <var>creationURL</var>, a <span>JavaScript execution context</span> <var>execution context</var>,
  null or an <span>environment</span> <var>reservedEnvironment</var>, a <span>URL</span>
  <var>topLevelCreationURL</var>, and an <span>origin</span> <var>topLevelOrigin</var>, run these
  steps:</p>

  <ol>
   <li><p>Let <var>realm</var> be the value of <var>execution context</var>'s Realm
   component.</p></li>

   <li><p>Let <var>window</var> be <var>realm</var>'s <span data-x="concept-realm-global">global
   object</span>.</p></li>

   <li>
    <p>Let <var>settings object</var> be a new <span>environment settings object</span> whose
    algorithms are defined as follows:</p>

    <dl>
     <dt>The <span>realm execution context</span></dt>
     <dd>
      <p>Return <var>execution context</var>.</p>
     </dd>

     <dt>The <span data-x="concept-settings-object-module-map">module map</span></dt>
     <dd>
      <p>Return the <span data-x="concept-document-module-map">module map</span> of
      <var>window</var>'s <span data-x="concept-document-window">associated
      <code>Document</code></span>.</p>
     </dd>

     <dt>The <span>API base URL</span></dt>
     <dd>
      <p>Return the current <span data-x="document base URL">base URL</span> of <var>window</var>'s
      <span data-x="concept-document-window">associated <code>Document</code></span>.</p>
     </dd>

     <dt>The <span data-x="concept-settings-object-origin">origin</span></dt>
     <dd>
      <p>Return the <span data-x="concept-document-origin">origin</span> of <var>window</var>'s
      <span data-x="concept-document-window">associated <code>Document</code></span>.</p>
     </dd>

     <dt>The <span data-x="concept-settings-object-policy-container">policy container</span></dt>
     <dd>
      <p>Return the <span data-x="concept-document-policy-container">policy container</span> of
      <var>window</var>'s <span data-x="concept-document-window">associated
      <code>Document</code></span>.</p>
     </dd>

     <dt>The <span data-x="concept-settings-object-cross-origin-isolated-capability">cross-origin
     isolated capability</span></dt>
     <dd>
      <p>Return true if both of the following hold, and false otherwise:</p>
      <ul>
       <li><p><var>realm</var>'s <span>agent cluster</span>'s <span
       data-x="agent-cluster-cross-origin-isolation">cross-origin-isolation mode</span> is "<code
       data-x="cross-origin-isolation-concrete">concrete</code>", and</p></li>

       <li><p><var>window</var>'s <span data-x="concept-document-window">associated
       <code>Document</code></span> is <span>allowed to use</span> the "<code
       data-x="cross-origin-isolated-feature">cross-origin-isolated</code>" feature.</p></li>
      </ul>
     </dd>

     <dt>The <span data-x="concept-settings-object-time-origin">time origin</span></dt>
     <dd>
      <p>Return <var>window</var>'s <span data-x="concept-document-window">associated
      <code>Document</code></span>'s <span>load timing info</span>'s <span>navigation start
      time</span>.</p>
     </dd>
    </dl>
   </li>

   <li>
    <p>If <var>reservedEnvironment</var> is non-null, then:</p>

    <ol>
     <li><p>Set <var>settings object</var>'s <span data-x="concept-environment-id">id</span> to
     <var>reservedEnvironment</var>'s <span data-x="concept-environment-id">id</span>, <span
     data-x="concept-environment-target-browsing-context">target browsing context</span> to
     <var>reservedEnvironment</var>'s <span
     data-x="concept-environment-target-browsing-context">target browsing context</span>, and <span
     data-x="concept-environment-active-service-worker">active service worker</span> to
     <var>reservedEnvironment</var>'s <span
     data-x="concept-environment-active-service-worker">active service worker</span>.</p></li>

     <li>
      <p>Set <var>reservedEnvironment</var>'s <span data-x="concept-environment-id">id</span> to
      the empty string.</p>

      <p class="note">The identity of the reserved environment is considered to be fully
      transferred to the created <span>environment settings object</span>. The reserved environment
      is not searchable by the <span>environment</span>’s <span
      data-x="concept-environment-id">id</span> from this point on.</p>
     </li>
    </ol>
   </li>

   <li><p>Otherwise, set <var>settings object</var>'s <span
   data-x="concept-environment-id">id</span> to a new unique opaque string, <var>settings
   object</var>'s <span data-x="concept-environment-target-browsing-context">target browsing
   context</span> to null, and <var>settings object</var>'s <span
   data-x="concept-environment-active-service-worker">active service worker</span> to null.</p></li>

   <li><p>Set <var>settings object</var>'s <span data-x="concept-environment-creation-url">creation
   URL</span> to <var>creationURL</var>, <var>settings object</var>'s <span>top-level creation
   URL</span> to <var>topLevelCreationURL</var>, and <var>settings object</var>'s <span>top-level
   origin</span> to <var>topLevelOrigin</var>.</p></li>

   <li><p>Set <var>realm</var>'s [[HostDefined]] field to <var>settings object</var>.</p></li>
  </ol>

  </div>


  <h4>The <code>WindowProxy</code> exotic object</h4>

  <p>A <dfn interface><code>WindowProxy</code></dfn> is an exotic object that wraps a
  <code>Window</code> ordinary object, indirecting most operations through to the wrapped object.
  Each <span>browsing context</span> has an associated <code>WindowProxy</code> object. When the
  <span>browsing context</span> is <span data-x="navigate">navigated</span>, the <code>Window</code>
  object wrapped by the <span>browsing context</span>'s associated <code>WindowProxy</code> object
  is changed.</p>

  <div w-nodev>

  <p>The <code>WindowProxy</code> exotic object must use the ordinary internal methods except where
  it is explicitly specified otherwise below.</p>

  <p>There is no <code>WindowProxy</code> <span>interface object</span>.</p>

  <p>Every <code>WindowProxy</code> object has a <dfn data-x="concept-windowproxy-window"
  export>[[Window]]</dfn> internal slot representing the wrapped <code>Window</code> object.</p>

  <p class="note">Although <code>WindowProxy</code> is named as a "proxy", it does not do
  polymorphic dispatch on its target's internal methods as a real proxy would, due to a desire to
  reuse machinery between <code>WindowProxy</code> and <code>Location</code> objects. As long as the
  <code>Window</code> object remains an ordinary object this is unobservable and can be implemented
  either way.</p>

  <h5 id="windowproxy-getprototypeof">[[GetPrototypeOf]] ( )</h5>

  <ol>
   <li><p>Let <var>W</var> be the value of the
   <span data-x="concept-windowproxy-window">[[Window]]</span> internal slot of
   <b>this</b>.</p></li>

   <li><p>If <span>IsPlatformObjectSameOrigin</span>(<var>W</var>) is true, then return !
   <span>OrdinaryGetPrototypeOf</span>(<var>W</var>).</p></li>

   <li><p>Return null.</p></li>
  </ol>

  <h5 id="windowproxy-setprototypeof">[[SetPrototypeOf]] ( <var>V</var> )</h5>

  <ol>
   <li><p>Return ! <span>SetImmutablePrototype</span>(<b>this</b>, <var>V</var>).</p></li>
  </ol>

  <h5 id="windowproxy-isextensible">[[IsExtensible]] ( )</h5>

  <ol>
   <li><p>Return true.</p></li>
  </ol>

  <h5 id="windowproxy-preventextensions">[[PreventExtensions]] ( )</h5>

  <ol>
   <li><p>Return false.</p></li>
  </ol>

  <h5 id="windowproxy-getownproperty">[[GetOwnProperty]] ( <var>P</var> )</h5>

  <ol>
   <li><p>Let <var>W</var> be the value of the
   <span data-x="concept-windowproxy-window">[[Window]]</span> internal slot of
   <b>this</b>.</p></li>

   <li>
    <p>If <var>P</var> is an <span>array index property name</span>, then:</p>

    <ol>
     <li><p>Let <var>index</var> be ! <span>ToUint32</span>(<var>P</var>).</p></li>

     <li><p>Let <var>children</var> be the <span>document-tree child navigables</span> of
     <var>W</var>'s <span data-x="concept-document-window">associated
     <code>Document</code></span>.</p></li>

     <li><p>Let <var>value</var> be undefined.</p></li>

     <li>
      <p>If <var>index</var> is less than <var>children</var>'s <span data-x="list
      size">size</span>, then:</p>

      <ol>
       <li><p><span data-x="list sort">Sort</span> <var>children</var> in ascending order, with
       <var>navigableA</var> being less than <var>navigableB</var> if <var>navigableA</var>'s <span
       data-x="nav-container">container</span> was inserted into <var>W</var>'s <span
       data-x="concept-document-window">associated <code>Document</code></span> earlier than
       <var>navigableB</var>'s <span data-x="nav-container">container</span> was.</p></li>

       <li><p>Set <var>value</var> to <var>children</var>[<var>index</var>]'s <span
       data-x="nav-wp">active <code>WindowProxy</code></span>.</p></li>
      </ol>
     </li>

     <!-- sort order: https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2860 -->

     <li>
      <p>If <var>value</var> is undefined, then:</p>

      <ol>
       <li><p>If <span>IsPlatformObjectSameOrigin</span>(<var>W</var>) is true, then return
       undefined.</p></li>

       <li><p>Throw a <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>
      </ol>
     </li>

     <li><p>Return <span>PropertyDescriptor</span>{
     [[Value]]: <var>value</var>,
     [[Writable]]: false,
     [[Enumerable]]: true,
     [[Configurable]]: true }.</p></li>
    </ol>
   </li>

   <li>
    <p>If <span>IsPlatformObjectSameOrigin</span>(<var>W</var>) is true, then return !
    <span>OrdinaryGetOwnProperty</span>(<var>W</var>, <var>P</var>).</p>

    <p class="note">This is a <span>willful violation</span> of the JavaScript specification's
    <span>invariants of the essential internal methods</span> to maintain compatibility with
    existing web content. See <a href="https://github.com/tc39/ecma262/issues/672">tc39/ecma262
    issue #672</a> for more information. <ref>JAVASCRIPT</ref></p>
   </li>

   <li><p>Let <var>property</var> be <span>CrossOriginGetOwnPropertyHelper</span>(<var>W</var>,
   <var>P</var>).</p></li>

   <li><p>If <var>property</var> is not undefined, then return <var>property</var>.</p></li>

   <li>
    <p>If <var>property</var> is undefined and <var>P</var> is in <var>W</var>'s <span>document-tree
    child navigable target name property set</span>, then:</p>

    <ol>
     <li><p>Let <var>value</var> be the <span data-x="nav-wp">active
     <code>WindowProxy</code></span> of the <span data-x="dom-window-nameditem-filter">named
     object</span> of <var>W</var> with the name <var>P</var>.</p></li>

     <li>
      <p>Return <span>PropertyDescriptor</span>{
      [[Value]]: <var>value</var>,
      [[Enumerable]]: false,
      [[Writable]]: false,
      [[Configurable]]: true }.</p>

      <p class="note">The reason the property descriptors are non-enumerable, despite this
      mismatching the same-origin behavior, is for compatibility with existing web content. See <a
      href="https://github.com/whatwg/html/issues/3183">issue #3183</a> for details.</p>
     </li>
    </ol>

   <li><p>Return ? <span>CrossOriginPropertyFallback</span>(<var>P</var>).</p></li>
  </ol>

  <h5 id="windowproxy-defineownproperty">[[DefineOwnProperty]] ( <var>P</var>, <var>Desc</var>
  )</h5>

  <ol>
   <li><p>Let <var>W</var> be the value of the
   <span data-x="concept-windowproxy-window">[[Window]]</span> internal slot of
   <b>this</b>.</p></li>

   <li>
    <p>If <span>IsPlatformObjectSameOrigin</span>(<var>W</var>) is true, then:

    <ol>
     <li><p>If <var>P</var> is an <span>array index property name</span>, return false.</p></li>

     <li>
      <p>Return ? <span>OrdinaryDefineOwnProperty</span>(<var>W</var>, <var>P</var>,
      <var>Desc</var>).</p>

      <p class="note">This is a <span>willful violation</span> of the JavaScript specification's
      <span>invariants of the essential internal methods</span> to maintain compatibility with
      existing web content. See <a href="https://github.com/tc39/ecma262/issues/672">tc39/ecma262
      issue #672</a> for more information. <ref>JAVASCRIPT</ref></p>
     </li>
    </ol>

   <li><p>Throw a <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>
  </ol>

  <h5 id="windowproxy-get">[[Get]] ( <var>P</var>, <var>Receiver</var> )</h5>

  <ol>
   <li><p>Let <var>W</var> be the value of the
   <span data-x="concept-windowproxy-window">[[Window]]</span> internal slot of
   <b>this</b>.</p></li>

   <li><p><span data-x="coop-check-access-report">Check if an access between two browsing contexts
   should be reported</span>, given the <span>current global object</span>'s
   <span data-x="window bc">browsing context</span>, <var>W</var>'s <span
   data-x="window bc">browsing context</span>, <var>P</var>, and the <span>current settings
   object</span>.</p></li>

   <li><p>If <span>IsPlatformObjectSameOrigin</span>(<var>W</var>) is true, then return ?
   <span>OrdinaryGet</span>(<b>this</b>, <var>P</var>, <var>Receiver</var>).</p></li>

   <li><p>Return ? <span>CrossOriginGet</span>(<b>this</b>, <var>P</var>,
   <var>Receiver</var>).</p></li>
  </ol>

  <p class="note"><b>this</b> is passed rather than <var>W</var> as <span>OrdinaryGet</span> and
  <span>CrossOriginGet</span> will invoke the <a
  href="#windowproxy-getownproperty">[[GetOwnProperty]]</a> internal method.</p>

  <h5 id="windowproxy-set">[[Set]] ( <var>P</var>, <var>V</var>, <var>Receiver</var> )</h5>

  <ol>
   <li><p>Let <var>W</var> be the value of the
   <span data-x="concept-windowproxy-window">[[Window]]</span> internal slot of
   <b>this</b>.</p></li>

   <li><p><span data-x="coop-check-access-report">Check if an access between two browsing contexts
   should be reported</span>, given the <span>current global object</span>'s <span>browsing
   context</span>, <var>W</var>'s <span>browsing context</span>, <var>P</var>,
   and the <span>current settings object</span>.</p></li>

   <li>
    <p>If <span>IsPlatformObjectSameOrigin</span>(<var>W</var>) is true, then:</p>

    <ol>
     <li><p>If <var>P</var> is an <span>array index property name</span>,
     then return false.</p></li>

     <li><p>Return ? <span>OrdinarySet</span>(<var>W</var>, <var>P</var>, <var>V</var>,
     <var>Receiver</var>).</p></li>
    </ol>
   </li>

   <li>
    <p>Return ? <span>CrossOriginSet</span>(<b>this</b>, <var>P</var>, <var>V</var>,
    <var>Receiver</var>).</p>

    <p class="note"><b>this</b> is passed rather than <var>W</var> as <span>CrossOriginSet</span>
    will invoke the <a href="#windowproxy-getownproperty">[[GetOwnProperty]]</a>
    internal method.</p>
   </li>
  </ol>

  <h5 id="windowproxy-delete">[[Delete]] ( <var>P</var> )</h5>

  <ol>
   <li><p>Let <var>W</var> be the value of the
   <span data-x="concept-windowproxy-window">[[Window]]</span> internal slot of
   <b>this</b>.</p></li>

   <li>
    <p>If <span>IsPlatformObjectSameOrigin</span>(<var>W</var>) is true, then:

    <ol>
     <li>
      <p>If <var>P</var> is an <span>array index property name</span>, then:</p>

      <ol>
       <li><p>Let <var>desc</var> be ! <b>this</b>.[[GetOwnProperty]](<var>P</var>).</p></li>

       <li><p>If <var>desc</var> is undefined, then return true.</p></li>

       <li><p>Return false.</p></li>
      </ol>
     </li>

     <li><p>Return ? <span>OrdinaryDelete</span>(<var>W</var>, <var>P</var>).</p></li>
    </ol>

   <li><p>Throw a <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>
  </ol>

  <h5 id="windowproxy-ownpropertykeys">[[OwnPropertyKeys]] ( )</h5>

  <ol>
   <li><p>Let <var>W</var> be the value of the <span
   data-x="concept-windowproxy-window">[[Window]]</span> internal slot of <b>this</b>.</p></li>

   <li><p>Let <var>maxProperties</var> be <var>W</var>'s <span
   data-x="concept-document-window">associated <code>Document</code></span>'s <span>document-tree
   child navigables</span>'s <span data-x="list size">size</span>.</p></li>

   <li><p>Let <var>keys</var> be <span data-x="exclusive range">the range</span> 0 to
   <var>maxProperties</var>, exclusive.</p></li>

   <li><p>If <span>IsPlatformObjectSameOrigin</span>(<var>W</var>) is true, then return the
   concatenation of <var>keys</var> and
   <span>OrdinaryOwnPropertyKeys</span>(<var>W</var>).</p></li>

   <li><p>Return the concatenation of <var>keys</var> and !
   <span>CrossOriginOwnPropertyKeys</span>(<var>W</var>).</p></li>
  </ol>

  </div>


  <h4>The <code>Location</code> interface</h4>

  <p>Each <code>Window</code> object is associated with a unique instance of a <code>Location</code>
  object, allocated when the <code>Window</code> object is created.</p>

  <div w-nodev>

  <p class="warning">The <code>Location</code> exotic object is defined through a mishmash of IDL,
  invocation of JavaScript internal methods post-creation, and overridden JavaScript internal
  methods. Coupled with its scary security policy, please take extra care while implementing
  this excrescence.</p>

  <p>To create a <code>Location</code> object, run these steps:</p>

  <ol>
   <li><p>Let <var>location</var> be a new <code>Location</code> <span>platform
   object</span>.</p></li>

   <li><p>Let <var>valueOf</var> be <var>location</var>'s <span
   data-x="concept-relevant-realm">relevant
   realm</span>.[[Intrinsics]].[[<span>%Object.prototype.valueOf%</span>]].</p></li>

   <li><p>Perform ! <var>location</var>.[[DefineOwnProperty]]("<code data-x="">valueOf</code>", {
   [[Value]]: <var>valueOf</var>,
   [[Writable]]: false,
   [[Enumerable]]: false,
   [[Configurable]]: false }).</p></li>

   <li><p>Perform ! <var>location</var>.[[DefineOwnProperty]](<span>%Symbol.toPrimitive%</span>, {
   [[Value]]: undefined,
   [[Writable]]: false,
   [[Enumerable]]: false,
   [[Configurable]]: false }).</p></li>

   <li><p>Set the value of the <span>[[DefaultProperties]]</span> internal slot of
   <var>location</var> to <var>location</var>.[[OwnPropertyKeys]]().</p></li>

   <li><p>Return <var>location</var>.</p></li>
  </ol>

  <p class="note">The addition of <code data-x="">valueOf</code> and <span>%Symbol.toPrimitive%</span> own
  data properties, as well as the fact that all of <code>Location</code>'s IDL attributes are marked
  <code data-x="">[<span>LegacyUnforgeable</span>]</code>, is required by legacy code that consulted
  the <code>Location</code> interface, or stringified it, to determine the <span
  data-x="concept-document-url">document URL</span>, and then used it in a security-sensitive way.
  In particular, the <code data-x="">valueOf</code>, <span>%Symbol.toPrimitive%</span>, and <code
  data-x="">[<span>LegacyUnforgeable</span>]</code> stringifier mitigations ensure that code such as
  <code data-x="">foo[location] = bar</code> or <code data-x="">location + ""</code> cannot be
  misdirected.</p>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-location">location</span> [ = <var>value</var> ]</code></dt>
   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-location">location</span> [ = <var>value</var> ]</code></dt>

   <dd>
    <p>Returns a <code>Location</code> object with the current page's location.</p>

    <p>Can be set, to navigate to another page.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <code>Document</code> object's <dfn attribute for="Document"><code
  data-x="dom-document-location">location</code></dfn> getter steps are to return
  <span>this</span>'s <span>relevant global object</span>'s <code>Location</code> object, if
  <span>this</span> is <span>fully active</span>, and null otherwise.</p>

  <p>The <code>Window</code> object's <dfn attribute for="Window"><code
  data-x="dom-location">location</code></dfn> getter steps are to return <span>this</span>'s
  <code>Location</code> object.</p>

  </div>

  <p><code>Location</code> objects provide a representation of the <span
  data-x="concept-document-url">URL</span> of their associated <code>Document</code>, as well as
  methods for <span data-x="navigate">navigating</span> and <span data-x="reload">reloading</span>
  the associated <span>navigable</span>.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>Location</dfn> { // but see also <a href="#the-location-interface">additional creation steps</a> and <a href="#location-internal-methods">overridden internal methods</a>
  [<span>LegacyUnforgeable</span>] stringifier attribute USVString <span data-x="dom-location-href">href</span>;
  [<span>LegacyUnforgeable</span>] readonly attribute USVString <span data-x="dom-location-origin">origin</span>;
  [<span>LegacyUnforgeable</span>] attribute USVString <span data-x="dom-location-protocol">protocol</span>;
  [<span>LegacyUnforgeable</span>] attribute USVString <span data-x="dom-location-host">host</span>;
  [<span>LegacyUnforgeable</span>] attribute USVString <span data-x="dom-location-hostname">hostname</span>;
  [<span>LegacyUnforgeable</span>] attribute USVString <span data-x="dom-location-port">port</span>;
  [<span>LegacyUnforgeable</span>] attribute USVString <span data-x="dom-location-pathname">pathname</span>;
  [<span>LegacyUnforgeable</span>] attribute USVString <span data-x="dom-location-search">search</span>;
  [<span>LegacyUnforgeable</span>] attribute USVString <span data-x="dom-location-hash">hash</span>;

  [<span>LegacyUnforgeable</span>] undefined <span data-x="dom-location-assign">assign</span>(USVString url);
  [<span>LegacyUnforgeable</span>] undefined <span data-x="dom-location-replace">replace</span>(USVString url);
  [<span>LegacyUnforgeable</span>] undefined <span data-x="dom-location-reload">reload</span>();

  [<span>LegacyUnforgeable</span>, SameObject] readonly attribute <span>DOMStringList</span> <span data-x="dom-location-ancestorOrigins">ancestorOrigins</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>location</var>.<span data-x="">toString()</span></code></dt>
   <dt><code data-x=""><var>location</var>.<span subdfn data-x="dom-location-href">href</span></code></dt>
   <dd>
    <p>Returns the <code>Location</code> object's URL.</p>

    <p>Can be set, to navigate to the given URL.</p>
   </dd>

   <dt><code data-x=""><var>location</var>.<span subdfn data-x="dom-location-origin">origin</span></code></dt>
   <dd><p>Returns the <code>Location</code> object's URL's origin.</p></dd>

   <dt><code data-x=""><var>location</var>.<span subdfn data-x="dom-location-protocol">protocol</span></code></dt>
   <dd>
    <p>Returns the <code>Location</code> object's URL's scheme.</p>

    <p>Can be set, to navigate to the same URL with a changed scheme.</p>
   </dd>

   <dt><code data-x=""><var>location</var>.<span subdfn data-x="dom-location-host">host</span></code></dt>
   <dd>
    <p>Returns the <code>Location</code> object's URL's host and port (if different from the default
    port for the scheme).</p>

    <p>Can be set, to navigate to the same URL with a changed host and port.</p>
   </dd>

   <dt><code data-x=""><var>location</var>.<span subdfn data-x="dom-location-hostname">hostname</span></code></dt>
   <dd>
    <p>Returns the <code>Location</code> object's URL's host.</p>

    <p>Can be set, to navigate to the same URL with a changed host.</p>
   </dd>

   <dt><code data-x=""><var>location</var>.<span subdfn data-x="dom-location-port">port</span></code></dt>
   <dd>
    <p>Returns the <code>Location</code> object's URL's port.</p>

    <p>Can be set, to navigate to the same URL with a changed port.</p>
   </dd>

   <dt><code data-x=""><var>location</var>.<span subdfn data-x="dom-location-pathname">pathname</span></code></dt>
   <dd>
    <p>Returns the <code>Location</code> object's URL's path.</p>

    <p>Can be set, to navigate to the same URL with a changed path.</p>
   </dd>

   <dt><code data-x=""><var>location</var>.<span subdfn data-x="dom-location-search">search</span></code></dt>
   <dd>
    <p>Returns the <code>Location</code> object's URL's query (includes leading "<code
    data-x="">?</code>" if non-empty).</p>

    <p>Can be set, to navigate to the same URL with a changed query (ignores leading "<code
    data-x="">?</code>").</p>
   </dd>

   <dt><code data-x=""><var>location</var>.<span subdfn data-x="dom-location-hash">hash</span></code></dt>
   <dd>
    <p>Returns the <code>Location</code> object's URL's fragment (includes leading "<code
    data-x="">#</code>" if non-empty).</p>

    <p>Can be set, to navigate to the same URL with a changed fragment (ignores leading "<code
    data-x="">#</code>").</p>
   </dd>

   <dt><code data-x=""><var>location</var>.<span subdfn data-x="dom-location-assign">assign</span>(<var>url</var>)</code></dt>
   <dd><p>Navigates to the given URL.</p></dd>

   <dt><code data-x=""><var>location</var>.<span subdfn data-x="dom-location-replace">replace</span>(<var>url</var>)</code></dt>
   <dd><p>Removes the current page from the session history and navigates to the given URL.</p></dd>

   <dt><code data-x=""><var>location</var>.<span subdfn data-x="dom-location-reload">reload</span>()</code></dt>

   <dd><p>Reloads the current page.</p></dd>

   <dt><code data-x=""><var>location</var>.<span subdfn data-x="dom-location-ancestorOrigins">ancestorOrigins</span></code></dt>
   <dd>
    <p>Returns a <code>DOMStringList</code> object listing the origins of the <span>ancestor navigables</span>' <span
    data-x="nav-document">active documents</span>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>A <code>Location</code> object has an associated <dfn>relevant <code>Document</code></dfn>,
  which is its <span>relevant global object</span>'s <span
  data-x="window bc">browsing context</span>'s <span>active document</span>, if this
  <code>Location</code> object's <span>relevant global object</span>'s <span data-x="window
  bc">browsing context</span> is non-null, and null otherwise.</p>

  <p>A <code>Location</code> object has an associated <dfn data-x="concept-location-url">url</dfn>,
  which is this <code>Location</code> object's <span>relevant <code>Document</code></span>'s <span
  data-x="concept-document-url">URL</span>, if this <code>Location</code> object's <span>relevant
  <code>Document</code></span> is non-null, and <code>about:blank</code> otherwise.</p>

  <p>A <code>Location</code> object has an associated <dfn
  data-x="concept-location-ancestor-origins-list">ancestor origins list</dfn>. When a
  <code>Location</code> object is created, its <span
  data-x="concept-location-ancestor-origins-list">ancestor origins list</span> must be set to a
  <code>DOMStringList</code> object whose associated list is the <span>list</span> of strings that
  the following steps would produce:</p>

  <ol>
   <li><p>Let <var>output</var> be a new <span>list</span> of strings.</p></li>

   <li><p>Let <var>current</var> be the <code>Location</code> object's <span>relevant
   <code>Document</code></span>.</p></li>

   <li>
    <p>While <var>current</var>'s <span data-x="doc-container-document">container document</span>
    is non-null:</p>

    <ol>
     <li><p>Set <var>current</var> to <var>current</var>'s <span
     data-x="doc-container-document">container document</span>.</p></li>

     <li><p><span data-x="list append">Append</span> the <span data-x="serialization of an
     origin">serialization</span> of <var>current</var>'s <span
     data-x="concept-document-origin">origin</span> to <var>output</var>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>output</var>.</p></li>
  </ol>

  <p>To <dfn><code>Location</code>-object navigate</dfn> a <code>Location</code> object
  <var>location</var> to a <span>URL</span> <var>url</var>, optionally given a <code
  data-x="NavigationHistoryBehavior">NavigationHistoryBehavior</code> <var>historyHandling</var>
  (default "<code data-x="NavigationHistoryBehavior-auto">auto</code>"):</p>

  <ol>
   <li><p>Let <var>navigable</var> be <var>location</var>'s <span>relevant global object</span>'s
   <span data-x="window navigable">navigable</span>.</p></li>

   <li><p>Let <var>sourceDocument</var> be the <span data-x="concept-incumbent-global">incumbent
   global object</span>'s <span data-x="concept-document-window">associated
   <code>Document</code></span>.</p></li>

   <li><p>If <var>location</var>'s <span>relevant <code>Document</code></span> is not yet
   <span>completely loaded</span>, and the <span data-x="concept-incumbent-global">incumbent global
   object</span> does not have <span>transient activation</span>, then set
   <var>historyHandling</var> to "<code
   data-x="NavigationHistoryBehavior-replace">replace</code>".</p></li>

   <li><p><span>Navigate</span><!--DONAV location--> <var>navigable</var> to <var>url</var> using
   <var>sourceDocument</var>, with <i><span>exceptionsEnabled</span></i> set to true and <i
   data-x="navigation-hh">historyHandling</i> set to <var>historyHandling</var>.</p></li>
  </ol>

  <p>The <dfn attribute for="Location"><code data-x="dom-location-href">href</code></dfn> getter
  steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is non-null and its
   <span data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with
   the <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Return <span>this</span>'s <span data-x="concept-location-url">url</span>, <span
   data-x="concept-url-serializer">serialized</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-location-href">href</code> setter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is null, then
   return.</p></li>

   <li><p>Let <var>url</var> be the result of <span>encoding-parsing a URL</span> given the given
   value, relative to the <span>entry settings object</span>.</p></li>

   <li><p>If <var>url</var> is failure, then throw a <span>"<code>SyntaxError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p><span><code>Location</code>-object navigate</span> <span>this</span> to
   <var>url</var>.</p></li>
  </ol>

  <p class="note">The <code data-x="dom-location-href">href</code> setter intentionally has no
  security check.</p>

  <p>The <dfn attribute for="Location"><code data-x="dom-location-origin">origin</code></dfn>
  getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is non-null and its
   <span data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with
   the <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Return the <span data-x="serialization of an origin">serialization</span> of
   <span>this</span>'s <span data-x="concept-location-url">url</span>'s <span
   data-x="concept-url-origin">origin</span>.</p></li>
  </ol>

  <p>The <dfn attribute for="Location"><code data-x="dom-location-protocol">protocol</code></dfn>
  getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is non-null and its
   <span data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with
   the <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Return <span>this</span>'s <span data-x="concept-location-url">url</span>'s <span
   data-x="concept-url-scheme">scheme</span>, followed by "<code data-x="">:</code>".</p></li>
  </ol>

  <p>The <code data-x="dom-location-protocol">protocol</code> setter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is null, then
   return.</p></li>

   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span>'s <span
   data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with the
   <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>copyURL</var> be a copy of <span>this</span>'s <span
   data-x="concept-location-url">url</span>.</p></li>

   <li>
    <p>Let <var>possibleFailure</var> be the result of <span data-x="basic url parser">basic URL
    parsing</span> the given value, followed by "<code data-x="">:</code>", with <var>copyURL</var>
    as <span data-x="basic url parser url"><i>url</i></span> and <span>scheme start state</span> as
    <span data-x="basic url parser state override"><i>state override</i></span>.</p>

    <p class="note">Because the URL parser ignores multiple consecutive colons, providing a value
    of "<code data-x="">https:</code>" (or even "<code data-x="">https::::</code>") is the same as
    providing a value of "<code data-x="">https</code>".</p>
   </li>

   <li><p>If <var>possibleFailure</var> is failure, then throw a
   <span>"<code>SyntaxError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <var>copyURL</var>'s <span data-x="concept-url-scheme">scheme</span> is not an
   <span>HTTP(S) scheme</span>, then terminate these steps.</p></li>

   <li><p><span><code>Location</code>-object navigate</span> <span>this</span> to
   <var>copyURL</var>.</p></li>
  </ol>

  <p>The <dfn attribute for="Location"><code data-x="dom-location-host">host</code></dfn> getter
  steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is non-null and its
   <span data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with
   the <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-location-url">url</span>.</p></li>

   <li><p>If <var>url</var>'s <span data-x="concept-url-host">host</span> is null, return the empty
   string.</p></li>

   <li><p>If <var>url</var>'s <span data-x="concept-url-port">port</span> is null, return
   <var>url</var>'s <span data-x="concept-url-host">host</span>, <span data-x="host
   serializer">serialized</span>.</p></li>

   <li><p>Return <var>url</var>'s <span data-x="concept-url-host">host</span>, <span data-x="host
   serializer">serialized</span>, followed by "<code data-x="">:</code>" and <var>url</var>'s <span
   data-x="concept-url-port">port</span>, <span data-x="serialize an
   integer">serialized</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-location-host">host</code> setter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is null, then
   return.</p></li>

   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span>'s <span
   data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with the
   <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>copyURL</var> be a copy of <span>this</span>'s <span
   data-x="concept-location-url">url</span>.</p></li>

   <li><p>If <var>copyURL</var> has an <span>opaque path</span>, then return.</p></li>

   <li><p><span data-x="basic url parser">Basic URL parse</span> the given value, with
   <var>copyURL</var> as <span data-x="basic url parser url"><i>url</i></span> and <span>host
   state</span> as <span data-x="basic url parser state override"><i>state
   override</i></span>.</p></li>

   <li><p><span><code>Location</code>-object navigate</span> <span>this</span> to
   <var>copyURL</var>.</p></li>
  </ol>

  <p>The <dfn attribute for="Location"><code data-x="dom-location-hostname">hostname</code></dfn>
  getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is non-null and its
   <span data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with
   the <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <span>this</span>'s <span data-x="concept-location-url">url</span>'s <span
   data-x="concept-url-host">host</span> is null, return the empty string.</p></li>

   <li><p>Return <span>this</span>'s <span data-x="concept-location-url">url</span>'s <span
   data-x="concept-url-host">host</span>, <span data-x="host
   serializer">serialized</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-location-hostname">hostname</code> setter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is null, then
   return.</p></li>

   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span>'s <span
   data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with the
   <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>copyURL</var> be a copy of <span>this</span>'s <span
   data-x="concept-location-url">url</span>.</p></li>

   <li><p>If <var>copyURL</var> has an <span>opaque path</span>, then return.</p></li>

   <li><p><span data-x="basic url parser">Basic URL parse</span> the given value, with
   <var>copyURL</var> as <span data-x="basic url parser url"><i>url</i></span> and <span>hostname
   state</span> as <span data-x="basic url parser state override"><i>state
   override</i></span>.</p></li>

   <li><p><span><code>Location</code>-object navigate</span> <span>this</span> to
   <var>copyURL</var>.</p></li>
  </ol>

  <p>The <dfn attribute for="Location"><code data-x="dom-location-port">port</code></dfn> getter
  steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is non-null and its
   <span data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with
   the <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <span>this</span>'s <span data-x="concept-location-url">url</span>'s <span
   data-x="concept-url-port">port</span> is null, return the empty string.</p></li>

   <li><p>Return <span>this</span>'s <span data-x="concept-location-url">url</span>'s <span
   data-x="concept-url-port">port</span>, <span data-x="serialize an
   integer">serialized</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-location-port">port</code> setter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is null, then
   return.</p></li>

   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span>'s <span
   data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with the
   <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>copyURL</var> be a copy of <span>this</span>'s <span
   data-x="concept-location-url">url</span>.</p></li>

   <li><p>If <var>copyURL</var> <span>cannot have a username/password/port</span>, then
   return.</p></li>

   <li><p>If the given value is the empty string, then set <var>copyURL</var>'s <span
   data-x="concept-url-port">port</span> to null.</p></li>

   <li><p>Otherwise, <span data-x="basic url parser">basic URL parse</span> the given value, with
   <var>copyURL</var> as <span data-x="basic url parser url"><i>url</i></span> and <span>port
   state</span> as <span data-x="basic url parser state override"><i>state
   override</i></span>.</p></li>

   <li><p><span><code>Location</code>-object navigate</span> <span>this</span> to
   <var>copyURL</var>.</p></li>
  </ol>

  <p>The <dfn attribute for="Location"><code data-x="dom-location-pathname">pathname</code></dfn>
  getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is non-null and its
   <span data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with
   the <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Return the result of <span data-x="URL path serializer">URL path serializing</span> this
   <code>Location</code> object's <span data-x="concept-location-url">url</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-location-pathname">pathname</code> setter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is null, then
   return.</p></li>

   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span>'s <span
   data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with the
   <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>copyURL</var> be a copy of <span>this</span>'s <span
   data-x="concept-location-url">url</span>.</p></li>

   <li><p>If <var>copyURL</var> has an <span>opaque path</span>, then return.</p></li>

   <li><p>Set <var>copyURL</var>'s <span data-x="concept-url-path">path</span> to the empty
   list.</p></li>

   <li><p><span data-x="basic url parser">Basic URL parse</span> the given value, with
   <var>copyURL</var> as <span data-x="basic url parser url"><i>url</i></span> and <span>path start
   state</span> as <span data-x="basic url parser state override"><i>state
   override</i></span>.</p></li>

   <li><p><span><code>Location</code>-object navigate</span> <span>this</span> to
   <var>copyURL</var>.</p></li>
  </ol>

  <p>The <dfn attribute for="Location"><code data-x="dom-location-search">search</code></dfn>
  getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is non-null and its
   <span data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with
   the <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <span>this</span>'s <span data-x="concept-location-url">url</span>'s <span
   data-x="concept-url-query">query</span> is either null or the empty string, return the empty
   string.</p></li>

   <li><p>Return "<code data-x="">?</code>", followed by <span>this</span>'s <span
   data-x="concept-location-url">url</span>'s <span
   data-x="concept-url-query">query</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-location-search">search</code> setter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is null, then
   return.</p></li>

   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span>'s <span
   data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with the
   <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>copyURL</var> be a copy of <span>this</span>'s <span
   data-x="concept-location-url">url</span>.</p></li>

   <li><p>If the given value is the empty string, set <var>copyURL</var>'s <span
   data-x="concept-url-query">query</span> to null.

   <li>
    <p>Otherwise, run these substeps:</p>

    <ol>
     <li><p>Let <var>input</var> be the given value with a single leading "<code
     data-x="">?</code>" removed, if any.</p></li>

     <li><p>Set <var>copyURL</var>'s <span data-x="concept-url-query">query</span> to the empty
     string.</p></li>

     <li><p><span data-x="basic url parser">Basic URL parse</span> <var>input</var>, with null, the
     <span>relevant <code>Document</code></span>'s <span>document's character encoding</span>,
     <var>copyURL</var> as <span data-x="basic url parser url"><i>url</i></span>, and <span>query
     state</span> as <span data-x="basic url parser state override"><i>state
     override</i></span>.</p></li>
    </ol>
   </li>

   <li><p><span><code>Location</code>-object navigate</span> <span>this</span> to
   <var>copyURL</var>.</p></li>
  </ol>

  <p>The <dfn attribute for="Location"><code data-x="dom-location-hash">hash</code></dfn> getter
  steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is non-null and its
   <span data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with
   the <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <span>this</span>'s <span data-x="concept-location-url">url</span>'s <span
   data-x="concept-url-fragment">fragment</span> is either null or the empty string, return the
   empty string.</p></li>

   <li><p>Return "<code data-x="">#</code>", followed by <span>this</span>'s <span
   data-x="concept-location-url">url</span>'s <span
   data-x="concept-url-fragment">fragment</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-location-hash">hash</code> setter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is null, then
   return.</p></li>

   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span>'s <span
   data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with the
   <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>copyURL</var> be a copy of <span>this</span>'s <span
   data-x="concept-location-url">url</span>.</p></li>

   <li><p>Let <var>input</var> be the given value with a single leading "<code data-x="">#</code>"
   removed, if any.</p></li>

   <li><p>Set <var>copyURL</var>'s <span data-x="concept-url-fragment">fragment</span> to the empty
   string.</p></li>

   <li><p><span data-x="basic url parser">Basic URL parse</span> <var>input</var>, with
   <var>copyURL</var> as <span data-x="basic url parser url"><i>url</i></span> and <span>fragment
   state</span> as <span data-x="basic url parser state override"><i>state
   override</i></span>.</p></li>

   <li>
    <p>If <var>copyURL</var>'s <span data-x="concept-url-fragment">fragment</span> is
    <span>this</span>'s <span data-x="concept-location-url">url</span>'s <span
    data-x="concept-url-fragment">fragment</span>, then return.</p>

    <p class="note">This bailout is necessary for compatibility with deployed content, which <a
    href="https://bugzilla.mozilla.org/show_bug.cgi?id=1733797#c2">redundantly sets <code
    data-x="dom-location-hash">location.hash</code> on scroll</a>. It does not apply to other
    mechanisms of fragment navigation, such as the <code
    data-x="dom-location-href">location.href</code> setter or <code
    data-x="dom-location-assign">location.assign()</code>.</p>
   </li>

   <li><p><span><code>Location</code>-object navigate</span> <span>this</span> to
   <var>copyURL</var>.</p></li>
  </ol>

  <p class="note">Unlike the equivalent API for the <code>a</code> and <code>area</code> elements,
  the <code data-x="dom-location-hash">hash</code> setter does not special case the empty string, to
  remain compatible with deployed scripts.</p>

  <hr>

  <p>The <dfn method for="Location"><code
  data-x="dom-location-assign">assign(<var>url</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is null, then
   return.</p></li>

   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span>'s <span
   data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with the
   <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>urlRecord</var> be the result of <span>encoding-parsing a URL</span> given
   <var>url</var>, relative to the <span>entry settings object</span>.</p></li>

   <li><p>If <var>urlRecord</var> is failure, then throw a <span>"<code>SyntaxError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p><span><code>Location</code>-object navigate</span> <span>this</span> to
   <var>urlRecord</var>.</p></li>
  </ol>

  <p>The <dfn method for="Location"><code
  data-x="dom-location-replace">replace(<var>url</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is null, then
   return.</p></li>

   <li><p>Let <var>urlRecord</var> be the result of <span>encoding-parsing a URL</span> given
   <var>url</var>, relative to the <span>entry settings object</span>.</p></li>

   <li><p>If <var>urlRecord</var> is failure, then throw a <span>"<code>SyntaxError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p><span><code>Location</code>-object navigate</span> <span>this</span> to
   <var>urlRecord</var> given "<code
   data-x="NavigationHistoryBehavior-replace">replace</code>".</p></li>
  </ol>

  <p class="note">The <code data-x="dom-location-replace">replace()</code> method intentionally has
  no security check.</p>

  <p>The <dfn method for="Location"><code data-x="dom-location-reload">reload()</code></dfn> method
  steps are:</p>

  <ol>
   <li><p>Let <var>document</var> be <span>this</span>'s <span>relevant
   <code>Document</code></span>.</p></li>

   <li><p>If <var>document</var> is null, then return.</p></li>

   <li><p>If <var>document</var>'s <span data-x="concept-document-origin">origin</span> is not
   <span>same origin-domain</span> with the <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p><span>Reload</span> <var>document</var>'s <span>node navigable</span>.</p></li>
  </ol>

  <hr>

  <p>The <dfn attribute for="Location"><code
  data-x="dom-location-ancestorOrigins">ancestorOrigins</code></dfn> getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span> is null, then return
   an empty <span>list</span>.</p></li>

   <li><p>If <span>this</span>'s <span>relevant <code>Document</code></span>'s <span
   data-x="concept-document-origin">origin</span> is not <span>same origin-domain</span> with the
   <span>entry settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Otherwise, return <span>this</span>'s <span
   data-x="concept-location-ancestor-origins-list">ancestor origins list</span>.</p></li>
  </ol>

  <p class="warning">The details of how the <code
  data-x="dom-location-ancestorOrigins">ancestorOrigins</code> attribute works are still
  controversial and might change. See <a href="https://github.com/whatwg/html/issues/1918">issue
  #1918</a> for more information.</p>

  <hr>

  <p id="location-internal-methods">As explained earlier, the <code>Location</code> exotic object
  requires additional logic beyond IDL for security purposes. The <code>Location</code> object must
  use the ordinary internal methods except where it is explicitly specified otherwise below.</p>

  <p>Also, every <code>Location</code> object has a <dfn>[[DefaultProperties]]</dfn> internal slot
  representing its own properties at time of its creation.</p>

  <h5 id="location-getprototypeof">[[GetPrototypeOf]] ( )</h5>

  <ol>
   <li><p>If <span>IsPlatformObjectSameOrigin</span>(<b>this</b>) is true, then return !
   <span>OrdinaryGetPrototypeOf</span>(<b>this</b>).</p></li>

   <li><p>Return null.</p></li>
  </ol>

  <h5 id="location-setprototypeof">[[SetPrototypeOf]] ( <var>V</var> )</h5>

  <ol>
   <li><p>Return ! <span>SetImmutablePrototype</span>(<b>this</b>, <var>V</var>).</p></li>
  </ol>

  <h5 id="location-isextensible">[[IsExtensible]] ( )</h5>

  <ol>
   <li><p>Return true.</p></li>
  </ol>

  <h5 id="location-preventextensions">[[PreventExtensions]] ( )</h5>

  <ol>
   <li><p>Return false.</p></li>
  </ol>

  <h5 id="location-getownproperty">[[GetOwnProperty]] ( <var>P</var> )</h5>

  <ol>
   <li>
    <p>If <span>IsPlatformObjectSameOrigin</span>(<b>this</b>) is true, then:</p>

    <ol>
     <li><p>Let <var>desc</var> be <span>OrdinaryGetOwnProperty</span>(<b>this</b>,
     <var>P</var>).</p></li>

     <li><p>If the value of the <span>[[DefaultProperties]]</span> internal slot of <b>this</b>
     contains <var>P</var>, then set <var>desc</var>.[[Configurable]] to true.</p></li>

     <li><p>Return <var>desc</var>.</p></li>
    </ol>
   </li>

   <li><p>Let <var>property</var> be <span>CrossOriginGetOwnPropertyHelper</span>(<b>this</b>,
   <var>P</var>).</p></li>

   <li><p>If <var>property</var> is not undefined, then return <var>property</var>.</p></li>

   <li><p>Return ? <span>CrossOriginPropertyFallback</span>(<var>P</var>).</p></li>
  </ol>

  <h5 id="location-defineownproperty">[[DefineOwnProperty]] ( <var>P</var>, <var>Desc</var> )</h5>

  <ol>
   <li>
    <p>If <span>IsPlatformObjectSameOrigin</span>(<b>this</b>) is true, then:</p>

    <ol>
     <li><p>If the value of the <span>[[DefaultProperties]]</span> internal slot of <b>this</b>
     contains <var>P</var>, then return false.</p></li>

     <li><p>Return ? <span>OrdinaryDefineOwnProperty</span>(<b>this</b>, <var>P</var>,
     <var>Desc</var>).</p></li>
    </ol>
   </li>

   <li><p>Throw a <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>
  </ol>

  <h5 id="location-get">[[Get]] ( <var>P</var>, <var>Receiver</var> )</h5>

  <ol>
   <li><p>If <span>IsPlatformObjectSameOrigin</span>(<b>this</b>) is true, then return ?
   <span>OrdinaryGet</span>(<b>this</b>, <var>P</var>, <var>Receiver</var>).</p></li>

   <li><p>Return ? <span>CrossOriginGet</span>(<b>this</b>, <var>P</var>,
   <var>Receiver</var>).</p></li>
  </ol>

  <h5 id="location-set">[[Set]] ( <var>P</var>, <var>V</var>, <var>Receiver</var> )</h5>

  <ol>
   <li><p>If <span>IsPlatformObjectSameOrigin</span>(<b>this</b>) is true, then return ?
   <span>OrdinarySet</span>(<b>this</b>, <var>P</var>, <var>V</var>, <var>Receiver</var>).</p></li>

   <li><p>Return ? <span>CrossOriginSet</span>(<b>this</b>, <var>P</var>, <var>V</var>,
   <var>Receiver</var>).</p></li>
  </ol>

  <h5 id="location-delete">[[Delete]] ( <var>P</var> )</h5>

  <ol>
   <li><p>If <span>IsPlatformObjectSameOrigin</span>(<b>this</b>) is true, then return ?
   <span>OrdinaryDelete</span>(<b>this</b>, <var>P</var>).</p></li>

   <li><p>Throw a <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>
  </ol>

  <h5 id="location-ownpropertykeys">[[OwnPropertyKeys]] ( )</h5>

  <ol>
   <li><p>If <span>IsPlatformObjectSameOrigin</span>(<b>this</b>) is true, then return
   <span>OrdinaryOwnPropertyKeys</span>(<b>this</b>).</p></li>

   <li><p>Return <span>CrossOriginOwnPropertyKeys</span>(<b>this</b>).</p></li>
  </ol>

  </div>


  <h4>The <code>History</code> interface</h4>

  <pre><code class="idl">enum <dfn enum>ScrollRestoration</dfn> { "<span data-x="dom-ScrollRestoration-auto">auto</span>", "<span data-x="dom-ScrollRestoration-manual">manual</span>" };

[Exposed=Window]
interface <dfn interface>History</dfn> {
  readonly attribute unsigned long <span data-x="dom-history-length">length</span>;
  attribute <span>ScrollRestoration</span> <span data-x="dom-history-scroll-restoration">scrollRestoration</span>;
  readonly attribute any <span data-x="dom-history-state">state</span>;
  undefined <span data-x="dom-history-go">go</span>(optional long delta = 0);
  undefined <span data-x="dom-history-back">back</span>();
  undefined <span data-x="dom-history-forward">forward</span>();
  undefined <span data-x="dom-history-pushState">pushState</span>(any data, DOMString unused, optional USVString? url = null);
  undefined <span data-x="dom-history-replaceState">replaceState</span>(any data, DOMString unused, optional USVString? url = null);
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><span subdfn data-x="dom-history">history</span>.<span subdfn data-x="dom-history-length">length</span></code></dt>

   <dd><p>Returns the number of overall <span data-x="tn-session-history-entries">session history
   entries</span> for the current <span>traversable navigable</span>.</p></dd>

   <dt><code data-x=""><span data-x="dom-history">history</span>.<span subdfn data-x="dom-history-scroll-restoration">scrollRestoration</span></code></dt>

   <dd><p>Returns the <span data-x="she-scroll-restoration-mode">scroll restoration mode</span> of
   the <span data-x="nav-active-history-entry">active session history entry</span>.</p></dd>

   <dt><code data-x=""><span data-x="dom-history">history</span>.<span data-x="dom-history-scroll-restoration">scrollRestoration</span> = <var>value</var></code></dt>

   <dd><p>Set the <span data-x="she-scroll-restoration-mode">scroll restoration mode</span> of the
   <span data-x="nav-active-history-entry">active session history entry</span> to
   <var>value</var>.</p></dd>

   <dt><code data-x=""><span data-x="dom-history">history</span>.<span subdfn data-x="dom-history-state">state</span></code></dt>

   <dd><p>Returns the <span data-x="she-classic-history-api-state">classic history API state</span>
   of the <span data-x="nav-active-history-entry">active session history entry</span>, deserialized
   into a JavaScript value.</p></dd>

   <dt><code data-x=""><span data-x="dom-history">history</span>.<span data-x="dom-history-go">go</span>()</code></dt>

   <dd><p>Reloads the current page.</p></dd>

   <dt><code data-x=""><span data-x="dom-history">history</span>.<span subdfn data-x="dom-history-go">go</span>(<var>delta</var>)</code></dt>

   <dd>
    <p>Goes back or forward the specified number of steps in the overall <span
    data-x="tn-session-history-entries">session history entries</span> list for the current
    <span>traversable navigable</span>.</p>

    <p>A zero delta will reload the current page.</p>

    <p>If the delta is out of range, does nothing.</p>
   </dd>

   <dt><code data-x=""><span data-x="dom-history">history</span>.<span subdfn data-x="dom-history-back">back</span>()</code></dt>

   <dd>
    <p>Goes back one step in the overall <span data-x="tn-session-history-entries">session history
    entries</span> list for the current <span>traversable navigable</span>.</p>

    <p>If there is no previous page, does nothing.</p>
   </dd>

   <dt><code data-x=""><span data-x="dom-history">history</span>.<span subdfn data-x="dom-history-forward">forward</span>()</code></dt>

   <dd>
    <p>Goes forward one step in the overall <span data-x="tn-session-history-entries">session
    history entries</span> list for the current <span>traversable navigable</span>.</p>

    <p>If there is no next page, does nothing.</p>
   </dd>

   <dt><code data-x=""><span data-x="dom-history">history</span>.<span subdfn data-x="dom-history-pushState">pushState</span>(<var>data</var>, "")</code></dt>
   <dd>
    <p>Adds a new entry into session history with its <span
    data-x="she-classic-history-api-state">classic history API state</span> set to a serialization
    of <var>data</var>. The <span data-x="nav-active-history-entry">active history entry</span>'s
    <span data-x="she-url">URL</span> will be copied over and used for the new entry's URL.</p>

    <p>(The second parameter exists for historical reasons, and cannot be omitted; passing the
    empty string is traditional.)</p>
   </dd>

   <dt><code data-x=""><span data-x="dom-history">history</span>.<span data-x="dom-history-pushState">pushState</span>(<var>data</var>, "", <var>url</var>)</code></dt>
   <dd>
    <p>Adds a new entry into session history with its <span
    data-x="she-classic-history-api-state">classic history API state</span> set to a serialization
    of <var>data</var>, and with its <span data-x="she-url">URL</span> set to <var>url</var>.</p>

    <p>If the current <code>Document</code> <span data-x="can have its URL rewritten">cannot have
    its URL rewritten</span> to <var>url</var>, a <span>"<code>SecurityError</code>"</span>
    <code>DOMException</code> will be thrown.</p>

    <p>(The second parameter exists for historical reasons, and cannot be omitted; passing the
    empty string is traditional.)</p>
   </dd>

   <dt><code data-x=""><span data-x="dom-history">history</span>.<span subdfn data-x="dom-history-replaceState">replaceState</span>(<var>data</var>, "")</code></dt>
   <dd>
    <p>Updates the <span data-x="she-classic-history-api-state">classic history API state</span> of
    the <span data-x="nav-active-history-entry">active session history entry</span> to a structured
    clone of <var>data</var>.</p>

    <p>(The second parameter exists for historical reasons, and cannot be omitted; passing the
    empty string is traditional.)</p>
   </dd>

   <dt><code data-x=""><span data-x="dom-history">history</span>.<span data-x="dom-history-replaceState">replaceState</span>(<var>data</var>, "", <var>url</var>)</code></dt>
   <dd>
    <p>Updates the <span data-x="she-classic-history-api-state">classic history API state</span> of
    the <span data-x="nav-active-history-entry">active session history entry</span> to a structured
    clone of <var>data</var>, and its <span data-x="she-url">URL</span> to <var>url</var>.</p>

    <p>If the current <code>Document</code> <span data-x="can have its URL rewritten">cannot have
    its URL rewritten</span> to <var>url</var>, a <span>"<code>SecurityError</code>"</span>
    <code>DOMException</code> will be thrown.</p>

    <p>(The second parameter exists for historical reasons, and cannot be omitted; passing the
    empty string is traditional.)</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>A <code>Document</code> has a <dfn data-x="doc-history">history object</dfn>, a
  <code>History</code> object.</p>

  <p>The <dfn attribute for="Window"><code data-x="dom-history">history</code></dfn> getter steps
  are to return <span>this</span>'s <span data-x="concept-document-window">associated
  <code>Document</code></span>'s <span data-x="doc-history">history object</span>.</p>

  <hr>

  <p>Each <code>History</code> object has <dfn data-x="concept-history-state">state</dfn>,
  initially null.</p>

  <p>Each <code>History</code> object has a <dfn data-x="concept-history-length">length</dfn>, a
  non-negative integer, initially 0.</p>

  <p>Each <code>History</code> object has an <dfn data-x="concept-history-index">index</dfn>, a
  non-negative integer, initially 0.</p>

  <p class="note">Although the <span data-x="concept-history-index">index</span> is not directly
  exposed, it can be inferred from changes to the <span
  data-x="concept-history-length">length</span> during synchronous navigations. In fact, that is
  what it's used for.</p>

  <p>The <dfn attribute for="History"><code data-x="dom-history-length">length</code></dfn> getter
  steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span> is not <span>fully
   active</span>, then throw a <span>"<code>SecurityError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Return <span>this</span>'s <span data-x="concept-history-length">length</span>.</p></li>
  </ol>

  <p>The <dfn attribute for="History"><code
  data-x="dom-history-scroll-restoration">scrollRestoration</code></dfn> getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span> is not <span>fully
   active</span>, then throw a <span>"<code>SecurityError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Return <span>this</span>'s <span>node navigable</span>'s <span
   data-x="nav-active-history-entry">active session history entry</span>'s <span
   data-x="she-scroll-restoration-mode">scroll restoration mode</span>.</p></li>
  </ol>

  <p>The <code data-x="dom-history-scroll-restoration">scrollRestoration</code> setter steps
  are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span> is not <span>fully
   active</span>, then throw a <span>"<code>SecurityError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Set <span>this</span>'s <span>node navigable</span>'s <span
   data-x="nav-active-history-entry">active session history entry</span>'s <span
   data-x="she-scroll-restoration-mode">scroll restoration mode</span> to the given value.</p></li>
  </ol>

  <p>The <dfn attribute for="History"><code data-x="dom-history-state">state</code></dfn> getter
  steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span> is not <span>fully
   active</span>, then throw a <span>"<code>SecurityError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Return <span>this</span>'s <span data-x="concept-history-state">state</span>.</p></li>
  </ol>

  <p>The <dfn method for="History"><code data-x="dom-history-go">go(<var>delta</var>)</code></dfn>
  method steps are to <span>delta traverse</span> <span>this</span> given <var>delta</var>.</p>

  <p>The <dfn method for="History"><code data-x="dom-history-back">back()</code></dfn> method steps
  are to <span>delta traverse</span> <span>this</span> given &minus;1.</p>

  <p>The <dfn method for="History"><code data-x="dom-history-forward">forward()</code></dfn> method
  steps are to <span>delta traverse</span> <span>this</span> given +1.</p>

  <p>To <dfn>delta traverse</dfn> a <code>History</code> object <var>history</var> given an integer
  <var>delta</var>:</p>

  <ol>
   <li><p>Let <var>document</var> be <var>history</var>'s <span>relevant global object</span>'s
   <span data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>If <var>document</var> is not <span>fully active</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <var>delta</var> is 0, then <span>reload</span> <var>document</var>'s <span>node
   navigable</span>, and return.</p></li>

   <li><p><span>Traverse the history by a delta</span> given <var>document</var>'s <span>node
   navigable</span>'s <span data-x="nav-traversable">traversable navigable</span>, <var>delta</var>,
   and with <i data-x="traverse-sourceDocument">sourceDocument</i> set to
   <var>document</var>.</p></li>
  </ol>

  <p>The <dfn method for="History"><code data-x="dom-history-pushState">pushState(<var>data</var>,
  <var>unused</var>, <var>url</var>)</code></dfn> method steps are to run the <span>shared history
  push/replace state steps</span> given <span>this</span>, <var>data</var>, <var>url</var>, and
  "<code data-x="NavigationHistoryBehavior-push">push</code>".</p>

  <p>The <dfn method for="History"><code
  data-x="dom-history-replaceState">replaceState(<var>data</var>, <var>unused</var>,
  <var>url</var>)</code></dfn> method steps are to run the <span>shared history push/replace state
  steps</span> given <span>this</span>, <var>data</var>, <var>url</var>, and "<code
  data-x="NavigationHistoryBehavior-replace">replace</code>".</p>

  <p>The <dfn>shared history push/replace state steps</dfn>, given a <code>History</code>
  <var>history</var>, a value <var>data</var>, a <span>scalar value string</span>-or-null
  <var>url</var>, and a <span>history handling behavior</span> <var>historyHandling</var>, are:</p>

  <ol>
   <li><p>Let <var>document</var> be <var>history</var>'s associated <code>Document</code>.</p></li>

   <li><p>If <var>document</var> is not <span>fully active</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Optionally, throw a <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.
   (For example, the user agent might disallow calls to these methods that are invoked on a timer,
   or from event listeners that are not triggered in response to a clear user action, or that are
   invoked in rapid succession.)</p></li>

   <li><p>Let <var>serializedData</var> be
   <span>StructuredSerializeForStorage</span>(<var>data</var>). Rethrow any exceptions.</p></li>

   <li><p>Let <var>newURL</var> be <var>document</var>'s <span
   data-x="concept-document-url">URL</span>.</p></li>

   <li>
    <p>If <var>url</var> is not null or the empty string, then:</p>

    <ol>
     <li><p>Set <var>newURL</var> to the result of <span>encoding-parsing a URL</span> given
     <var>url</var>, relative to the <span>relevant settings object</span> of
     <var>history</var>.</p></li>

     <li><p>If <var>newURL</var> is failure, then throw a <span>"<code>SecurityError</code>"</span>
     <code>DOMException</code>.</p></li>

     <li><p>If <var>document</var> <span data-x="can have its URL rewritten">cannot have its URL
     rewritten</span> to <var>newURL</var>, then throw a <span>"<code>SecurityError</code>"</span>
     <code>DOMException</code>.</p></li>
    </ol>

    <p class="note">The special case for the empty string here is historical, and leads to
    different resulting URLs when comparing code such as <code data-x="">location.href = ""</code>
    (which performs URL parsing on the empty string) versus <code
    data-x="">history.pushState(null, "", "")</code> (which bypasses it).</p>
   </li>

   <li><p>Let <var>navigation</var> be <var>history</var>'s <span>relevant global object</span>'s
   <span data-x="window-navigation-api">navigation API</span>.</p></li>

   <li><p>Let <var>continue</var> be the result of <span data-x="fire a push/replace/reload navigate
   event">firing a push/replace/reload <code data-x="event-navigate">navigate</code> event</span> at
   <var>navigation</var> with <i data-x="fire-navigate-prr-navigationType">navigationType</i> set to
   <var>historyHandling</var>, <i data-x="fire-navigate-prr-isSameDocument">isSameDocument</i> set
   to true, <i data-x="fire-navigate-prr-destinationURL">destinationURL</i> set to
   <var>newURL</var>, and <i
   data-x="fire-navigate-prr-classicHistoryAPIState">classicHistoryAPIState</i> set to
   <var>serializedData</var>.</p></li>

   <li><p>If <var>continue</var> is false, then return.</p>

   <li><p>Run the <span>URL and history update steps</span> given <var>document</var> and
   <var>newURL</var>, with <i data-x="uhus-serializedData">serializedData</i> set to
   <var>serializedData</var> and <i data-x="uhus-historyHandling">historyHandling</i> set to
   <var>historyHandling</var>.</p></li>
  </ol>

  <p>User agents may limit the number of state objects added to the session history per page. If a
  page hits the <span>implementation-defined</span> limit, user agents must remove the entry
  immediately after the first entry for that <code>Document</code> object in the session history
  after having added the new entry. (Thus the state history acts as a FIFO buffer for eviction, but
  as a LIFO buffer for navigation.)</p>

  <p>A <code>Document</code> <var>document</var> <dfn>can have its URL rewritten</dfn> to a
  <span>URL</span> <var>targetURL</var> if the following algorithm returns true:</p>

  <ol>
   <li><p>Let <var>documentURL</var> be <var>document</var>'s <span
   data-x="concept-document-url">URL</span>.</p></li>

   <li><p>If <var>targetURL</var> and <var>documentURL</var> differ in their <span
   data-x="concept-url-scheme">scheme</span>, <span data-x="concept-url-username">username</span>,
   <span data-x="concept-url-password">password</span>, <span
   data-x="concept-url-host">host</span>, or <span data-x="concept-url-port">port</span>
   components, then return false.</p></li>

   <li>
    <p>If <var>targetURL</var>'s <span data-x="concept-url-scheme">scheme</span> is an
    <span>HTTP(S) scheme</span>, then return true.</p>

    <p class="note">Differences in <span data-x="concept-url-path">path</span>, <span
    data-x="concept-url-query">query</span>, and <span
    data-x="concept-url-fragment">fragment</span> are allowed for <code data-x="http
    protocol">http:</code> and <code data-x="https protocol">https:</code> URLs.</p>
   </li>

   <li>
    <p>If <var>targetURL</var>'s <span data-x="concept-url-scheme">scheme</span> is "<code
    data-x="">file</code>", then:</p>

    <ol>
     <li><p>If <var>targetURL</var> and <var>documentURL</var> differ in their <span
     data-x="concept-url-path">path</span> component, then return false.</p></li>

     <li><p>Return true.</p></li>
    </ol>

    <p class="note">Differences in <span data-x="concept-url-query">query</span> and <span
    data-x="concept-url-fragment">fragment</span> are allowed for <code data-x="">file:</code>
    URLs.</p>
   </li>

   <li>
    <p>If <var>targetURL</var> and <var>documentURL</var> differ in their <span
    data-x="concept-url-path">path</span> component or <span
    data-x="concept-url-query">query</span> components, then return false.</p>

    <p class="note">Only differences in <span data-x="concept-url-fragment">fragment</span> are
    allowed for other types of URLs.</p>
   </li>

   <li><p>Return true.</p></li>
  </ol>

  </div>

  <div class="example">
   <table>
    <thead>
     <tr>
      <th><var>document</var>'s <span data-x="concept-document-url">URL</span>
      <th><var>targetURL</var>
      <th><span subdfn>can have its URL rewritten</span>
    <tbody>
     <tr>
      <td><code data-x="">https://example.com/home</code>
      <td><code data-x="">https://example.com/home#about</code>
      <td>✅
     <tr>
      <td><code data-x="">https://example.com/home</code>
      <td><code data-x="">https://example.com/home?page=shop</code>
      <td>✅
     <tr>
      <td><code data-x="">https://example.com/home</code>
      <td><code data-x="">https://example.com/shop</code>
      <td>✅
     <tr>
      <td><code data-x="">https://example.com/home</code>
      <td><code data-x="">https://user:pass@example.com/home</code>
      <td>❌
     <tr>
      <td><code data-x="">https://example.com/home</code>
      <td><code data-x="">http://example.com/home</code>
      <td>❌
     <tr>
      <td><code data-x="">file:///path/to/x</code>
      <td><code data-x="">file:///path/to/x#hash</code>
      <td>✅
     <tr>
      <td><code data-x="">file:///path/to/x</code>
      <td><code data-x="">file:///path/to/x?search</code>
      <td>✅
     <tr>
      <td><code data-x="">file:///path/to/x</code>
      <td><code data-x="">file:///path/to/y</code>
      <td>❌
     <tr>
      <td><code data-x="">about:blank</code>
      <td><code data-x="">about:blank#hash</code>
      <td>✅
     <tr>
      <td><code data-x="">about:blank</code>
      <td><code data-x="">about:blank?search</code>
      <td>❌
     <tr>
      <td><code data-x="">about:blank</code>
      <td><code data-x="">about:srcdoc</code>
      <td>❌
     <tr>
      <td><code data-x="">data:text/html,foo</code>
      <td><code data-x="">data:text/html,foo#hash</code>
      <td>✅
     <tr>
      <td><code data-x="">data:text/html,foo</code>
      <td><code data-x="">data:text/html,foo?search</code>
      <td>❌
     <tr>
      <td><code data-x="">data:text/html,foo</code>
      <td><code data-x="">data:text/html,bar</code>
      <td>❌
     <tr>
      <td><code data-x="">data:text/html,foo</code>
      <td><code data-x="">data:bar</code>
      <td>❌
     <tr>
      <td><code data-x="">blob:https://example.com/77becafe-657b-4fdc-8bd3-e83aaa5e8f43</code>
      <td><code data-x="">blob:https://example.com/77becafe-657b-4fdc-8bd3-e83aaa5e8f43#hash</code>
      <td>✅
     <tr>
      <td><code data-x="">blob:https://example.com/77becafe-657b-4fdc-8bd3-e83aaa5e8f43</code>
      <td><code data-x="">blob:https://example.com/77becafe-657b-4fdc-8bd3-e83aaa5e8f43?search</code>
      <td>❌
     <tr>
      <td><code data-x="">blob:https://example.com/77becafe-657b-4fdc-8bd3-e83aaa5e8f43</code>
      <td><code data-x="">blob:https://example.com/anything</code>
      <td>❌
     <tr>
      <td><code data-x="">blob:https://example.com/77becafe-657b-4fdc-8bd3-e83aaa5e8f43</code>
      <td><code data-x="">blob:path</code>
      <td>❌
   </table>

   <p>Note how only the <span data-x="concept-document-url">URL</span> of the <code>Document</code>
   matters, and not its <span data-x="concept-document-origin">origin</span>. They can mismatch in
   cases like <code>about:blank</code> <code>Document</code>s with inherited origins, in sandboxed
   <code>iframe</code>s, or when the <code data-x="dom-document-domain">document.domain</code>
   setter has been used.</p>
  </div>

  <div class="example">

   <p>Consider a game where the user can navigate along a line, such that the user is always at some
   coordinate, and such that the user can bookmark the page corresponding to a particular
   coordinate, to return to it later.</p>

   <p>A static page implementing the x=5 position in such a game could look like the following:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;!-- this is https://example.com/line?x=5 -->
&lt;html lang="en">
&lt;title>Line Game - 5&lt;/title>
&lt;p>You are at coordinate 5 on the line.&lt;/p>
&lt;p>
 &lt;a href="?x=6">Advance to 6&lt;/a> or
 &lt;a href="?x=4">retreat to 4&lt;/a>?
&lt;/p></code></pre>

   <p>The problem with such a system is that each time the user clicks, the whole page has to be
   reloaded. Here instead is another way of doing it, using script:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;!-- this starts off as https://example.com/line?x=5 -->
&lt;html lang="en">
&lt;title>Line Game - 5&lt;/title>
&lt;p>You are at coordinate &lt;span id="coord">5&lt;/span> on the line.&lt;/p>
&lt;p>
 &lt;a href="?x=6" onclick="go(1); return false;">Advance to 6&lt;/a> or
 &lt;a href="?x=4" onclick="go(-1); return false;">retreat to 4&lt;/a>?
&lt;/p>
&lt;script>
 var currentPage = 5; // prefilled by server
 function go(d) {
   setupPage(currentPage + d);
   history.pushState(currentPage, "", '?x=' + currentPage);
 }
 onpopstate = function(event) {
   setupPage(event.state);
 }
 function setupPage(page) {
   currentPage = page;
   document.title = 'Line Game - ' + currentPage;
   document.getElementById('coord').textContent = currentPage;
   document.links[0].href = '?x=' + (currentPage+1);
   document.links[0].textContent = 'Advance to ' + (currentPage+1);
   document.links[1].href = '?x=' + (currentPage-1);
   document.links[1].textContent = 'retreat to ' + (currentPage-1);
 }
&lt;/script></code></pre>

   <p>In systems without script, this still works like the previous example. However, users that
   <em>do</em> have script support can now navigate much faster, since there is no network access
   for the same experience. Furthermore, contrary to the experience the user would have with just a
   na&iuml;ve script-based approach, bookmarking and navigating the session history still work.</p>

   <p>In the example above, the <var>data</var> argument to the <code
   data-x="dom-history-pushState">pushState()</code> method is the same information as would be sent
   to the server, but in a more convenient form, so that the script doesn't have to parse the URL
   each time the user navigates.</p>

  </div>

  <div class="example">
   <p>Most applications want to use the same <span>scroll restoration mode</span> value for all of
   their history entries. To achieve this they can set the <code
   data-x="dom-history-scroll-restoration">scrollRestoration</code> attribute as soon as possible
   (e.g., in the first <code>script</code> element in the document's <code>head</code> element) to
   ensure that any entry added to the history session gets the desired scroll restoration mode.</p>

   <pre><code class="html">&lt;head&gt;
  &lt;script&gt;
       if ('scrollRestoration' in history)
            history.scrollRestoration = 'manual';
  &lt;/script&gt;
&lt;/head&gt;
   </code></pre>
  </div>


  <h4 id="navigation-api">The navigation API</h4>

  <h5 id="navigation-api-intro">Introduction</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>The navigation API, provided by the global <code data-x="dom-navigation">navigation</code>
  property, provides a modern and web application-focused way of managing navigations and history entries. It is a successor to the classic <code data-x="dom-location">location</code> and <code data-x="dom-history">history</code> APIs.</p>

  <p>One ability the API provides is inspecting <span data-x="session history entry">session history entries</span>. For example, the following will display the entries' URLs in an ordered list:</p>

  <pre><code class="js">const ol = document.createElement("ol");
ol.start = 0; // so that the list items' ordinal values match up with the entry indices

for (const entry of navigation.entries()) {
  const li = document.createElement("li");

  if (entry.index &lt; navigation.currentEntry.index) {
    li.className = "backward";
  } else if (entry.index > navigation.currentEntry.index) {
    li.className = "forward";
  } else {
    li.className = "current";
  }

  li.textContent = entry.url;
  ol.append(li);
}</code></pre>

  <p>The <code data-x="dom-Navigation-entries">navigation.entries()</code> array contains <code>NavigationHistoryEntry</code> instances, which have other useful properties in addition to the <code data-x="dom-NavigationHistoryEntry-url">url</code> and <code data-x="dom-NavigationHistoryEntry-index">index</code> properties shown here. Note that the array only contains <code>NavigationHistoryEntry</code> objects that represent the current <span>navigable</span>, and thus its contents are not impacted by navigations inside <span data-x="navigable container">navigable containers</span> such as <code>iframe</code>s, or by navigations of the <span data-x="nav-parent">parent navigable</span> in cases where the navigation API is itself being used inside an <code>iframe</code>. Additionally, it only contains <code>NavigationHistoryEntry</code> objects representing same-<span>origin</span> <span data-x="session history entry">session history entries</span>, meaning that if the user has visited other origins before or after the current one, there will not be corresponding <code>NavigationHistoryEntry</code>s.</p>

  <hr>

  <p>The navigation API can also be used to navigate, reload, or traverse through the history:</p>

  <pre><code class="html">&lt;button onclick="navigation.reload()">Reload&lt;/button>

&lt;input type="url" id="navigationURL">
&lt;button onclick="navigation.navigate(navigationURL.value)">Navigate&lt;/button>

&lt;button id="backButton" onclick="navigation.back()">Back&lt;/button>
&lt;button id="forwardButton" onclick="navigation.forward()">Forward&lt;/button>

&lt;select id="traversalDestinations">&lt;/select>
&lt;button id="goButton" onclick="navigation.traverseTo(traversalDestinations.value)">Traverse To&lt;/button>

&lt;script>
backButton.disabled = !navigation.canGoBack;
forwardButton.disabled = !navigation.canGoForward;

for (const entry of navigation.entries()) {
  traversalDestinations.append(new Option(entry.url, entry.key));
}
&lt;/script></code></pre>

  <p>Note that traversals are again limited to same-<span>origin</span> destinations, meaning that, for example, <code data-x="dom-Navigation-canGoBack">navigation.canGoBack</code> will be false if the previous <span>session history entry</span> is for a page from another origin.</p>

  <hr>

  <p>The most powerful part of the navigation API is the <code data-x="event-navigate">navigate</code> event, which fires whenever almost any navigation or traversal occurs in the current <span>navigable</span>:</p>

  <pre><code class="js">navigation.onnavigate = event => {
  console.log(event.navigationType); // "push", "replace", "reload", or "traverse"
  console.log(event.destination.url);
  console.log(event.userInitiated);
  // ... and <a href="#the-navigateevent-interface">other useful properties</a>
};</code></pre>

  <p>(The event will not fire for <a href="#nav-traversal-ui">location bar-initiated navigations</a>, or navigations initiated from other windows, when the destination of the navigation is a new document.)</p>

  <p>Much of the time, the event's <code data-x="dom-Event-cancelable">cancelable</code> property will be true, meaning this event can be canceled using <code data-x="dom-Event-preventDefault">preventDefault()</code>:</p>

  <pre><code class="js">navigation.onnavigate = event => {
  if (event.cancelable && isDisallowedURL(event.destination.url)) {
    alert(`Please don't go to ${event.destination.url}!`);
    event.preventDefault();
  }
};
</code></pre>

  <p>The <code data-x="dom-Event-cancelable">cancelable</code> property will be false for some "<code data-x="dom-NavigationType-traverse">traverse</code>" navigations, such as those taking place inside <span data-x="child navigable">child navigables</span>, those crossing to new origins, or when the user attempts to traverse again shortly after a previous call to <code data-x="dom-Event-preventDefault">preventDefault()</code> prevented them from doing so.</p>

  <p>The <code>NavigateEvent</code>'s <code data-x="dom-NavigateEvent-intercept">intercept()</code> method allows intercepting a navigation and converting it into a same-document navigation:</p>

  <pre><code class="js">navigation.addEventListener("navigate", e => {
  // Some navigations, e.g. cross-origin navigations, we cannot intercept.
  // Let the browser handle those normally.
  if (!e.canIntercept) {
    return;
  }

  // Similarly, don't intercept fragment navigations or downloads.
  if (e.hashChange || e.downloadRequest !== null) {
    return;
  }

  const url = new URL(event.destination.url);

  if (url.pathname.startsWith("/articles/")) {
    e.intercept({
      async handler() {
        // The URL has already changed, so show a placeholder while
        // fetching the new content, such as a spinner or loading page.
        renderArticlePagePlaceholder();

        // Fetch the new content and display when ready.
        const articleContent = await getArticleContent(url.pathname, { signal: e.signal });
        renderArticlePage(articleContent);
      }
    });
  }
});</code></pre>

  <p>Note that the <code data-x="dom-NavigationInterceptOptions-handler">handler</code> function can return a promise to represent the asynchronous progress, and success or failure, of the navigation. While the promise is still pending, browser UI can treat the navigation as ongoing (e.g., by presenting a loading spinner). Other parts of the navigation API are also sensitive to these promises, such as the return value of <code data-x="dom-Navigation-navigate">navigation.navigate()</code>:

  <pre><code class="js">const { committed, finished } = await navigation.navigate("/articles/the-navigation-api-is-cool");

// The committed promise will fulfill once the URL has changed, which happens
// immediately (as long as the NavigateEvent wasn't canceled).
await committed;

// The finished promise will fulfill once the Promise returned by handler() has
// fulfilled, which happens once the article is downloaded and rendered. (Or,
// it will reject, if handler() fails along the way).
await finished;</code></pre>

  <h5 id="navigation-interface">The <code>Navigation</code> interface</h5>

  <div w-nodev>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>Navigation</dfn> : <span>EventTarget</span> {
  sequence&lt;<span>NavigationHistoryEntry</span>> <span data-x="dom-Navigation-entries">entries</span>();
  readonly attribute <span>NavigationHistoryEntry</span>? <span data-x="dom-Navigation-currentEntry">currentEntry</span>;
  undefined <span data-x="dom-Navigation-updateCurrentEntry">updateCurrentEntry</span>(<span>NavigationUpdateCurrentEntryOptions</span> options);
  readonly attribute <span>NavigationTransition</span>? <span data-x="dom-Navigation-transition">transition</span>;
  readonly attribute <span>NavigationActivation</span>? <span data-x="dom-Navigation-activation">activation</span>;

  readonly attribute boolean <span data-x="dom-Navigation-canGoBack">canGoBack</span>;
  readonly attribute boolean <span data-x="dom-Navigation-canGoForward">canGoForward</span>;

  <span>NavigationResult</span> <span data-x="dom-Navigation-navigate">navigate</span>(USVString url, optional <span>NavigationNavigateOptions</span> options = {});
  <span>NavigationResult</span> <span data-x="dom-Navigation-reload">reload</span>(optional <span>NavigationReloadOptions</span> options = {});

  <span>NavigationResult</span> <span data-x="dom-Navigation-traverseTo">traverseTo</span>(DOMString key, optional <span>NavigationOptions</span> options = {});
  <span>NavigationResult</span> <span data-x="dom-Navigation-back">back</span>(optional <span>NavigationOptions</span> options = {});
  <span>NavigationResult</span> <span data-x="dom-Navigation-forward">forward</span>(optional <span>NavigationOptions</span> options = {});

  attribute <span>EventHandler</span> <span data-x="handler-Navigation-onnavigate">onnavigate</span>;
  attribute <span>EventHandler</span> <span data-x="handler-Navigation-onnavigatesuccess">onnavigatesuccess</span>;
  attribute <span>EventHandler</span> <span data-x="handler-Navigation-onnavigateerror">onnavigateerror</span>;
  attribute <span>EventHandler</span> <span data-x="handler-Navigation-oncurrententrychange">oncurrententrychange</span>;
};

dictionary <dfn dictionary>NavigationUpdateCurrentEntryOptions</dfn> {
  required any <dfn dict-member for="NavigationUpdateCurrentEntryOptions" data-x="dom-NavigationUpdateCurrentEntryOptions-state">state</dfn>;
};

dictionary <dfn dictionary>NavigationOptions</dfn> {
  any <dfn dict-member for="NavigationOptions" data-x="dom-NavigationOptions-info">info</dfn>;
};

dictionary <dfn dictionary>NavigationNavigateOptions</dfn> : <span>NavigationOptions</span> {
  any <dfn dict-member for="NavigationNavigateOptions" data-x="dom-NavigationNavigateOptions-state">state</dfn>;
  <span>NavigationHistoryBehavior</span> <dfn dict-member for="NavigationNavigateOptions" data-x="dom-NavigationNavigateOptions-history">history</dfn> = "<span data-x="NavigationHistoryBehavior-auto">auto</span>";
};

dictionary <dfn dictionary>NavigationReloadOptions</dfn> : <span>NavigationOptions</span> {
  any <dfn dict-member for="NavigationNavigateOptions" data-x="dom-NavigationReloadOptions-state">state</dfn>;
};

dictionary <dfn dictionary>NavigationResult</dfn> {
  <span data-x="idl-Promise">Promise</span>&lt;<span>NavigationHistoryEntry</span>> <dfn dict-member for="NavigationResult" data-x="dom-NavigationResult-committed">committed</dfn>;
  <span data-x="idl-Promise">Promise</span>&lt;<span>NavigationHistoryEntry</span>> <dfn dict-member for="NavigationResult" data-x="dom-NavigationResult-finished">finished</dfn>;
};

enum <dfn enum>NavigationHistoryBehavior</dfn> {
  "<span data-x="NavigationHistoryBehavior-auto">auto</span>",
  "<span data-x="NavigationHistoryBehavior-push">push</span>",
  "<span data-x="NavigationHistoryBehavior-replace">replace</span>"
};</code></pre>

  <p>Each <code>Window</code> has an associated <dfn data-x="window-navigation-api">navigation
  API</dfn>, which is a <code>Navigation</code> object. Upon creation of the <code>Window</code>
  object, its <span data-x="window-navigation-api">navigation API</span> must be set to a
  <span>new</span> <code>Navigation</code> object created in the <code>Window</code> object's <span
  data-x="concept-relevant-realm">relevant realm</span>.</p>

  <p>The <dfn attribute for="Window"><code data-x="dom-navigation">navigation</code></dfn> getter
  steps are to return <span>this</span>'s <span data-x="window-navigation-api">navigation
  API</span>.</p>

  </div>

  <p>The following are the <span>event handlers</span> (and their corresponding <span data-x="event
  handler event type">event handler event types</span>) <span w-nodev>that must be</span> supported,
  as <span>event handler IDL attributes</span>, by all objects implementing the
  <code>Navigation</code> interface:</p>

  <table>
   <thead>
    <tr>
     <th><span data-x="event handlers">Event handler</span>
     <th><span>Event handler event type</span>
   <tbody>
    <tr>
     <td><dfn attribute for="Navigation"><code data-x="handler-Navigation-onnavigate">onnavigate</code></dfn>
     <td><code data-x="event-navigate">navigate</code>
    <tr>
     <td><dfn attribute for="Navigation"><code data-x="handler-Navigation-onnavigatesuccess">onnavigatesuccess</code></dfn>
     <td><code data-x="event-navigatesuccess">navigatesuccess</code>
    <tr>
     <td><dfn attribute for="Navigation"><code data-x="handler-Navigation-onnavigateerror">onnavigateerror</code></dfn>
     <td><code data-x="event-navigateerror">navigateerror</code>
    <tr>
     <td><dfn attribute for="Navigation"><code data-x="handler-Navigation-oncurrententrychange">oncurrententrychange</code></dfn>
     <td><code data-x="event-currententrychange">currententrychange</code>
  </table>

  <h5 id="navigation-api-core">Core infrastructure</h5>

  <div w-nodev>

  <p>Each <code>Navigation</code> has an associated <dfn data-x="navigation-entry-list">entry
  list</dfn>, a <span>list</span> of <code>NavigationHistoryEntry</code> objects, initially
  empty.</p>

  <p>Each <code>Navigation</code> has an associated <dfn
  data-x="navigation-current-entry-index">current entry index</dfn>, an integer, initially
  &minus;1.</p>

  <p>The <dfn data-x="navigation-current-entry">current entry</dfn> of a <code>Navigation</code>
  <var>navigation</var> is the result of running the following steps:</p>

  <ol>
   <li><p>If <var>navigation</var> <span>has entries and events disabled</span>, then return
   null.</p></li>

   <li><p><span>Assert</span>: <var>navigation</var>'s <span
   data-x="navigation-current-entry-index">current entry index</span> is not &minus;1.</p></li>

   <li><p>Return <var>navigation</var>'s <span data-x="navigation-entry-list">entry
   list</span>[<var>navigation</var>'s <span data-x="navigation-current-entry-index">current entry
   index</span>].</p></li>
  </ol>

  <p>A <code>Navigation</code> <var>navigation</var> <dfn>has entries and events disabled</dfn> if
  the following steps return true:</p>

  <ol>
   <li><p>Let <var>document</var> be <var>navigation</var>'s <span>relevant global object</span>'s
   <span data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>If <var>document</var> is not <span>fully active</span>, then return true.</p></li>

   <li><p>If <var>document</var>'s <span>is initial <code>about:blank</code></span> is true, then
   return true.</p></li>

   <li><p>If <var>document</var>'s <span data-x="concept-document-origin">origin</span> is <span
   data-x="concept-origin-opaque">opaque</span>, then return true.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  <p>To <dfn data-x="getting the navigation API entry index">get the navigation API entry
  index</dfn> of a <span>session history entry</span> <var>she</var> within a
  <code>Navigation</code> <var>navigation</var>:</p>

  <ol>
   <li><p>Let <var>index</var> be 0.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>nhe</var> of <var>navigation</var>'s <span
    data-x="navigation-entry-list">entry list</span>:</p>

    <ol>
     <li><p>If <var>nhe</var>'s <span data-x="nhe-she">session history entry</span> is equal to
     <var>she</var>, then return <var>index</var>.</p></li>

     <li><p>Increment <var>index</var> by 1.</p></li>
    </ol>
   </li>

   <li><p>Return &minus;1.</p></li>
  </ol>

  <hr>

  </div>

  <p>A key type used throughout the navigation API is the <code>NavigationType</code>
  enumeration:</p>

  <pre><code class="idl">enum <dfn enum>NavigationType</dfn> {
 "<span data-x="dom-NavigationType-push">push</span>",
 "<span data-x="dom-NavigationType-replace">replace</span>",
 "<span data-x="dom-NavigationType-reload">reload</span>",
 "<span data-x="dom-NavigationType-traverse">traverse</span>"
};</code></pre>

  <p>This captures the main web developer-visible types of "navigations", which (as <a
  href="#note-meaning-of-navigate">noted elsewhere</a>) do not exactly correspond to this standard's
  singular <span>navigate</span> algorithm. The meaning of each value is the following:</p>

  <dl>
   <dt>"<dfn enum-value for="NavigationType"><code data-x="dom-NavigationType-push">push</code></dfn>"</dt>
   <dd>Corresponds to calls to <span>navigate</span> where the <span>history handling
   behavior</span> ends up as "<code data-x="NavigationHistoryBehavior-push">push</code>", or to
   <code data-x="dom-History-pushState">history.pushState()</code>.</dd>

   <dt>"<dfn enum-value for="NavigationType"><code data-x="dom-NavigationType-replace">replace</code></dfn>"</dt>
   <dd>Corresponds to calls to <span>navigate</span> where the <span>history handling
   behavior</span> ends up as "<code data-x="NavigationHistoryBehavior-replace">replace</code>", or
   to <code data-x="dom-History-pushState">history.replaceState()</code>.</dd>

   <dt>"<dfn enum-value for="NavigationType"><code data-x="dom-NavigationType-reload">reload</code></dfn>"</dt>
   <dd>Corresponds to calls to <span>reload</span>.</dd>

   <dt>"<dfn enum-value for="NavigationType"><code data-x="dom-NavigationType-traverse">traverse</code></dfn>"</dt>
   <dd>Corresponds to calls to <span>traverse the history by a delta</span>.</dd>
  </dl>

  <div w-nodev>

  <p class="note">The value space of the <code>NavigationType</code> enumeration is a superset of
  the value space of the specification-internal <span>history handling behavior</span> type. Several
  parts of this standard make use of this overlap, by passing in a <span>history handling
  behavior</span> to an algorithm that expects a <code>NavigationType</code>.</p>


  <h5 id="navigation-api-entry-updates">Initializing and updating the entry list</h5>

  <p>To <dfn>initialize the navigation API entries for a new document</dfn> given a
  <code>Navigation</code> <var>navigation</var>, a <span>list</span> of <span data-x="session
  history entry">session history entries</span> <var>newSHEs</var>, and a <span>session history
  entry</span> <var>initialSHE</var>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>navigation</var>'s <span data-x="navigation-entry-list">entry
   list</span> <span data-x="list is empty">is empty</span>.</p></li>

   <li><p><span>Assert</span>: <var>navigation</var>'s <span
   data-x="navigation-current-entry-index">current entry index</span> is &minus;1.</p></li>

   <li><p>If <var>navigation</var> <span>has entries and events disabled</span>, then
   return.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>newSHE</var> of <var>newSHEs</var>:</p>

    <ol>
     <li><p>Let <var>newNHE</var> be a <span>new</span> <code>NavigationHistoryEntry</code> created
     in the <span data-x="concept-relevant-realm">relevant realm</span> of
     <var>navigation</var>.</p></li>

     <li><p>Set <var>newNHE</var>'s <span data-x="nhe-she">session history entry</span> to
     <var>newSHE</var>.</p></li>

     <li><p><span data-x="list append">Append</span> <var>newNHE</var> to <var>navigation</var>'s
     <span data-x="navigation-entry-list">entry list</span>.</p></li>
    </ol>

    <p class="note"><var>newSHEs</var> will have originally come from <span>getting session history
    entries for the navigation API</span>, and thus each <var>newSHE</var> will be contiguous <span
    data-x="same origin">same</span> <span data-x="document-state-origin">origin</span> with
    <var>initialSHE</var>.</p>
   </li>

   <li><p>Set <var>navigation</var>'s <span data-x="navigation-current-entry-index">current entry
   index</span> to the result of <span>getting the navigation API entry index</span> of
   <var>initialSHE</var> within <var>navigation</var>.</p></li>
  </ol>

  <p>To <dfn>update the navigation API entries for reactivation</dfn> given a
  <code>Navigation</code> <var>navigation</var>, a <span>list</span> of <span data-x="session
  history entry">session history entries</span> <var>newSHEs</var>, and a <span>session history
  entry</span> <var>reactivatedSHE</var>:</p>

  <ol>
   <li><p>If <var>navigation</var> <span>has entries and events disabled</span>, then
   return.</p></li>

   <li><p>Let <var>newNHEs</var> be a new empty <span>list</span>.</p></li>

   <li><p>Let <var>oldNHEs</var> be a <span data-x="list clone">clone</span> of
   <var>navigation</var>'s <span data-x="navigation-entry-list">entry list</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>newSHE</var> of <var>newSHEs</var>:</p>

    <ol>
     <li><p>Let <var>newNHE</var> be null.</p></li>

     <li>
      <p>If <var>oldNHEs</var> <span data-x="list contains">contains</span> a
      <code>NavigationHistoryEntry</code> <var>matchingOldNHE</var> whose <span
      data-x="nhe-she">session history entry</span> is <var>newSHE</var>, then:</p>

      <ol>
       <li><p>Set <var>newNHE</var> to <var>matchingOldNHE</var>.</p></li>

       <li><p><span data-x="list remove">Remove</span> <var>matchingOldNHE</var> from
       <var>oldNHEs</var>.</p></li>
      </ol>
     </li>

     <li>
      <p>Otherwise:</p>

      <ol>
       <li><p>Set <var>newNHE</var> to a <span>new</span> <code>NavigationHistoryEntry</code>
       created in the <span data-x="concept-relevant-realm">relevant realm</span> of
       <var>navigation</var>.</p></li>

       <li><p>Set <var>newNHE</var>'s <span data-x="nhe-she">session history entry</span> to
       <var>newSHE</var>.</p></li>
      </ol>
     </li>

     <li><p><span data-x="list append">Append</span> <var>newNHE</var> to
     <var>newNHEs</var>.</p></li>
    </ol>

    <p class="note"><var>newSHEs</var> will have originally come from <span>getting session history
    entries for the navigation API</span>, and thus each <var>newSHE</var> will be contiguous <span
    data-x="same origin">same</span> <span data-x="document-state-origin">origin</span> with
    <var>reactivatedSHE</var>.</p>

    <p class="note">By the end of this loop, all <code>NavigationHistoryEntry</code>s that remain in
    <var>oldNHEs</var> represent <span data-x="session history entry">session history entries</span>
    which have been disposed while the <code>Document</code> was in <a
    href="#note-bfcache">bfcache</a>.</p>
   </li>

   <li><p>Set <var>navigation</var>'s <span>entry list</span> to <var>newNHEs</var>.</p></li>

   <li><p>Set <var>navigation</var>'s <span data-x="navigation-current-entry-index">current entry
   index</span> to the result of <span>getting the navigation API entry index</span> of
   <var>reactivatedSHE</var> within <var>navigation</var>.</p></li>

   <li>
    <p><span>Queue a global task</span> on the <span>navigation and traversal task source</span>
    given <var>navigation</var>'s <span>relevant global object</span> to run the following
    steps:</p>

    <ol>
     <li>
      <p><span data-x="list iterate">For each</span> <var>disposedNHE</var> of
      <var>oldNHEs</var>:</p>

      <ol>
       <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
       data-x="event-dispose">dispose</code> at <var>disposedNHE</var>.</p></li>
      </ol>
     </li>
    </ol>

    <div class="note">
     <p>We delay these steps by a task to ensure that <code data-x="event-dispose">dispose</code>
     events will fire after the <code data-x="event-pageshow">pageshow</code> event. This ensures
     that <code data-x="event-pageshow">pageshow</code> is the first event a page receives upon
     <span data-x="reactivate a document">reactivation</span>.</p>

     <p>(However, the rest of this algorithm runs before the <code
     data-x="event-pageshow">pageshow</code> event fires. This ensures that <code
     data-x="dom-Navigation-entries">navigation.entries()</code> and <code
     data-x="dom-Navigation-currentEntry">navigation.currentEntry</code> will have correctly-updated
     values during any <code data-x="event-pageshow">pageshow</code> event handlers.)</p>
    </div>
   </li>
  </ol>

  <p>To <dfn>update the navigation API entries for a same-document navigation</dfn> given a
  <code>Navigation</code> <var>navigation</var>, a <span>session history entry</span>
  <var>destinationSHE</var>, and a <code>NavigationType</code> <var>navigationType</var>:</p>

  <ol>
   <li><p>If <var>navigation</var> <span>has entries and events disabled</span>, then
   return.</p></li>

   <li><p>Let <var>oldCurrentNHE</var> be the <span data-x="navigation-current-entry">current
   entry</span> of <var>navigation</var>.</p></li>

   <li><p>Let <var>disposedNHEs</var> be a new empty <span>list</span>.</p></li>

   <li>
    <p>If <var>navigationType</var> is "<code data-x="dom-NavigationType-traverse">traverse</code>",
    then:</p>

    <ol>
     <li><p>Set <var>navigation</var>'s <span data-x="navigation-current-entry-index">current entry
     index</span> to the result of <span>getting the navigation API entry index</span> of
     <var>destinationSHE</var> within <var>navigation</var>.</p></li>

     <li><p><span>Assert</span>: <var>navigation</var>'s <span
     data-x="navigation-current-entry-index">current entry index</span> is not &minus;1.</p></li>
    </ol>

    <p class="note">This algorithm is only called for same-document traversals. Cross-document
    traversals will instead call either <span>initialize the navigation API entries for a new
    document</span> or <span>update the navigation API entries for reactivation</span>.</p>
   </li>

   <li>
    <p>Otherwise, if <var>navigationType</var> is "<code
    data-x="dom-NavigationType-push">push</code>", then:</p>

    <ol>
     <li><p>Set <var>navigation</var>'s <span data-x="navigation-current-entry-index">current entry
     index</span> to <var>navigation</var>'s <span data-x="navigation-current-entry-index">current
     entry index</span> + 1.</p></li>

     <li><p>Let <var>i</var> be <var>navigation</var>'s <span
     data-x="navigation-current-entry-index">current entry index</span>.</p></li>

     <li>
      <p><span>While</span> <var>i</var> &lt; <var>navigation</var>'s <span
      data-x="navigation-entry-list">entry list</span>'s <span data-x="list size">size</span>:</p>

      <ol>
       <li><p><span data-x="list append">Append</span> <var>navigation</var>'s <span
       data-x="navigation-entry-list">entry list</span>[<var>i</var>] to
       <var>disposedNHEs</var>.</p></li>

       <li><p>Set <var>i</var> to <var>i</var> + 1.</p></li>
      </ol>
     </li>

     <li><p><span data-x="list remove">Remove</span> all items in <var>disposedNHEs</var> from
     <var>navigation</var>'s <span data-x="navigation-entry-list">entry list</span>.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if <var>navigationType</var> is "<code
    data-x="dom-NavigationType-replace">replace</code>", then:</p>

    <ol>
     <li><p><span data-x="list append">Append</span> <var>oldCurrentNHE</var> to
     <var>disposedNHEs</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>If <var>navigationType</var> is "<code data-x="dom-NavigationType-push">push</code>" or
    "<code data-x="dom-NavigationType-replace">replace</code>", then:</p>

    <ol>
     <li><p>Let <var>newNHE</var> be a <span>new</span> <code>NavigationHistoryEntry</code> created
     in the <span data-x="concept-relevant-realm">relevant realm</span> of
     <var>navigation</var>.</p></li>

     <li><p>Set <var>newNHE</var>'s <span data-x="nhe-she">session history entry</span> to
     <var>destinationSHE</var>.</p></li>

     <li><p>Set <var>navigation</var>'s <span data-x="navigation-entry-list">entry
     list</span>[<var>navigation</var>'s <span data-x="navigation-current-entry-index">current entry
     index</span>] to <var>newNHE</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>If <var>navigation</var>'s <span>ongoing API method tracker</span> is non-null, then
    <span>notify about the committed-to entry</span> given <var>navigation</var>'s <span>ongoing API
    method tracker</span> and the <span data-x="navigation-current-entry">current entry</span> of
    <var>navigation</var>.</p>

    <p class="note">It is important to do this before firing the <code
    data-x="event-dispose">dispose</code> or <code
    data-x="event-currententrychange">currententrychange</code> events, since event handlers could
    start another navigation, or otherwise change the value of <var>navigation</var>'s <span>ongoing
    API method tracker</span>.</p>
   </li>

   <li>
    <p><span>Prepare to run script</span> given <var>navigation</var>'s <span>relevant settings
    object</span>.</p>

    <p class="note">See <a href="#note-suppress-microtasks-during-navigation-events">the discussion
    for other navigation API events</a> to understand why we do this.</p>
   </li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
   data-x="event-currententrychange">currententrychange</code> at <var>navigation</var> using
   <code>NavigationCurrentEntryChangeEvent</code>, with its <code
   data-x="dom-NavigationCurrentEntryChangeEvent-navigationType">navigationType</code> attribute
   initialized to <var>navigationType</var> and its <code
   data-x="dom-NavigationCurrentEntryChangeEvent-from">from</code> initialized to
   <var>oldCurrentNHE</var>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>disposedNHE</var> of
    <var>disposedNHEs</var>:</p>

    <ol>
     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-dispose">dispose</code> at <var>disposedNHE</var>.</p></li>
    </ol>
   </li>

   <li><p><span>Clean up after running script</span> given <var>navigation</var>'s <span>relevant
   settings object</span>.</p></li>
  </ol>

  <p class="XXX">In implementations, same-document navigations can cause <span data-x="session
  history entry">session history entries</span> to be disposed by falling off the back of the
  session history entry list. This is not yet handled by the above algorithm (or by any other part
  of this standard). See <a href="https://github.com/whatwg/html/issues/8620">issue #8620</a> to
  track progress on defining the correct behavior in such cases.</p>

  </div>


  <h5>The <code>NavigationHistoryEntry</code> interface</h5>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>NavigationHistoryEntry</dfn> : <span>EventTarget</span> {
  readonly attribute USVString? <span data-x="dom-NavigationHistoryEntry-url">url</span>;
  readonly attribute DOMString <span data-x="dom-NavigationHistoryEntry-key">key</span>;
  readonly attribute DOMString <span data-x="dom-NavigationHistoryEntry-id">id</span>;
  readonly attribute long long <span data-x="dom-NavigationHistoryEntry-index">index</span>;
  readonly attribute boolean <span data-x="dom-NavigationHistoryEntry-sameDocument">sameDocument</span>;

  any <span data-x="dom-NavigationHistoryEntry-getState">getState</span>();

  attribute <span>EventHandler</span> <span data-x="handler-NavigationHistoryEntry-ondispose">ondispose</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>entry</var>.<span subdfn data-x="dom-NavigationHistoryEntry-url">url</span></code></dt>
   <dd>
    <p>The URL of this navigation history entry.</p>

    <p>This can return null if the entry corresponds to a different <code>Document</code> than the
    current one (i.e., if <code data-x="dom-NavigationHistoryEntry-sameDocument">sameDocument</code>
    is false), and that <code>Document</code> was fetched with a <span>referrer policy</span> of
    "<code data-x="">no-referrer</code>" or "<code data-x="">origin</code>", since that indicates
    the <code>Document</code> in question is hiding its URL even from other same-origin pages.</p>
   </dd>

   <dt><code data-x=""><var>entry</var>.<span subdfn data-x="dom-NavigationHistoryEntry-key">key</span></code></dt>
   <dd>
    <p>A user agent-generated random UUID string representing this navigation history entry's place
    in the navigation history list. This value will be reused by other
    <code>NavigationHistoryEntry</code> instances that replace this one due to "<code
    data-x="dom-NavigationType-replace">replace</code>" navigations, and will survive reloads and
    session restores.</p>

    <p>This is useful for navigating back to this entry in the navigation history list, using <code
    data-x="dom-Navigation-traverseTo">navigation.traverseTo(key)</code>.</p>
   </dd>

   <dt><code data-x=""><var>entry</var>.<span subdfn data-x="dom-NavigationHistoryEntry-id">id</span></code></dt>
   <dd>
    <p>A user agent-generated random UUID string representing this specific navigation history
    entry. This value will <em>not</em> be reused by other <code>NavigationHistoryEntry</code>
    instances. This value will survive reloads and session restores.</p>

    <p>This is useful for associating data with this navigation history entry using other storage
    APIs.</p>
   </dd>

   <dt><code data-x=""><var>entry</var>.<span subdfn data-x="dom-NavigationHistoryEntry-index">index</span></code></dt>
   <dd><p>The index of this <code>NavigationHistoryEntry</code> within <code
   data-x="dom-Navigation-entries">navigation.entries()</code>, or &minus;1 if the entry is not in
   the navigation history entry list.</p></dd>

   <dt><code data-x=""><var>entry</var>.<span subdfn data-x="dom-NavigationHistoryEntry-sameDocument">sameDocument</span></code></dt>
   <dd><p>Indicates whether or not this navigation history entry is for the same
   <code>Document</code> as the current one, or not. This will be true, for example, when the entry
   represents a fragment navigation or single-page app navigation.</p></dd>

   <dt><code data-x=""><var>entry</var>.<span subdfn data-x="dom-NavigationHistoryEntry-getState">getState</span>()</code></dt>
   <dd>
    <p>Returns the <span data-x="StructuredDeserialize">deserialization</span> of the state stored
    in this entry, which was added to the entry using <code
    data-x="dom-Navigation-navigate">navigation.navigate()</code> or <code
    data-x="dom-Navigation-updateCurrentEntry">navigation.updateCurrentEntry()</code>. This state
    survives reloads and session restores.</p>

    <p>Note that in general, unless the state value is a primitive, <code class="js"
    data-x="">entry.getState() !== entry.getState()</code>, since a fresh deserialization is
    returned each time.</p>

    <p>This state is unrelated to the classic history API's <code
    data-x="dom-history-state">history.state</code>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>Each <code>NavigationHistoryEntry</code> has an associated <dfn data-x="nhe-she">session
  history entry</dfn>, which is a <span>session history entry</span>.</p>

  <p>The <dfn data-x="concept-NavigationHistoryEntry-key">key</dfn> of a
  <code>NavigationHistoryEntry</code> <var>nhe</var> is given by the return value of the following
  algorithm:</p>

  <ol>
   <li><p>If <var>nhe</var>'s <span>relevant global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span> is not <span>fully
   active</span>, then return the empty string.</p></li>

   <li><p>Return <var>nhe</var>'s <span data-x="nhe-she">session history entry</span>'s <span
   data-x="she-navigation-api-key">navigation API key</span>.</p></li>
  </ol>

  <p>The <dfn data-x="concept-NavigationHistoryEntry-id">ID</dfn> of a
  <code>NavigationHistoryEntry</code> <var>nhe</var> is given by the return value of the following
  algorithm:</p>

  <ol>
   <li><p>If <var>nhe</var>'s <span>relevant global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span> is not <span>fully
   active</span>, then return the empty string.</p></li>

   <li><p>Return <var>nhe</var>'s <span data-x="nhe-she">session history entry</span>'s <span
   data-x="she-navigation-api-id">navigation API ID</span>.</p></li>
  </ol>

  <p>The <dfn data-x="concept-NavigationHistoryEntry-index">index</dfn> of a
  <code>NavigationHistoryEntry</code> <var>nhe</var> is given by the return value of the following
  algorithm:</p>

  <ol>
   <li><p>If <var>nhe</var>'s <span>relevant global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span> is not <span>fully
   active</span>, then return &minus;1.</p></li>

   <li><p>Return the result of <span>getting the navigation API entry index</span> of
   <span>this</span>'s <span data-x="nhe-she">session history entry</span> within
   <span>this</span>'s <span>relevant global object</span>'s <span
   data-x="window-navigation-api">navigation API</span>.</p></li>
  </ol>

  <p>The <dfn attribute for="NavigationHistoryEntry"><code
  data-x="dom-NavigationHistoryEntry-url">url</code></dfn> getter steps are:</p>

  <ol>
   <li><p>Let <var>document</var> be <span>this</span>'s <span>relevant global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>If <var>document</var> is not <span>fully active</span>, then return the empty
   string.</p></li>

   <li><p>Let <var>she</var> be <span>this</span>'s <span data-x="nhe-she">session history
   entry</span>.</p></li>

   <li><p>If <var>she</var>'s <span data-x="she-document">document</span> does not equal
   <var>document</var>, and <var>she</var>'s <span data-x="she-document-state">document
   state</span>'s <span data-x="document-state-request-referrer-policy">request referrer
   policy</span> is "<code data-x="">no-referrer</code>" or "<code data-x="">origin</code>", then
   return null.</p></li>

   <li><p>Return <var>she</var>'s <span data-x="she-url">URL</span>, <span
   data-x="concept-url-serializer">serialized</span>.</p></li>
  </ol>

  <p>The <dfn attribute for="NavigationHistoryEntry"><code
  data-x="dom-NavigationHistoryEntry-key">key</code></dfn> getter steps are to return
  <span>this</span>'s <span data-x="concept-NavigationHistoryEntry-key">key</span>.</p>

  <p>The <dfn attribute for="NavigationHistoryEntry"><code
  data-x="dom-NavigationHistoryEntry-id">id</code></dfn> getter steps are to return
  <span>this</span>'s <span data-x="concept-NavigationHistoryEntry-id">ID</span>.</p>

  <p>The <dfn attribute for="NavigationHistoryEntry"><code
  data-x="dom-NavigationHistoryEntry-index">index</code></dfn> getter steps are to return
  <span>this</span>'s <span data-x="concept-NavigationHistoryEntry-index">index</span>.</p>

  <p>The <dfn attribute for="NavigationHistoryEntry"><code
  data-x="dom-NavigationHistoryEntry-sameDocument">sameDocument</code></dfn> getter steps are:</p>

  <ol>
   <li><p>Let <var>document</var> be <span>this</span>'s <span>relevant global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>If <var>document</var> is not <span>fully active</span>, then return false.</p></li>

   <li><p>Return true if <span>this</span>'s <span data-x="nhe-she">session history entry</span>'s
   <span data-x="she-document">document</span> equals <var>document</var>, and false
   otherwise.</p></li>
  </ol>

  <p>The <dfn method for="NavigationHistoryEntry"><code
  data-x="dom-NavigationHistoryEntry-getState">getState()</code></dfn> method steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>relevant global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span> is not <span>fully
   active</span>, then return undefined.</p></li>

   <li>
    <p>Return <span>StructuredDeserialize</span>(<span>this</span>'s <span data-x="nhe-she">session
    history entry</span>'s <span data-x="she-navigation-api-state">navigation API state</span>).
    Rethrow any exceptions.</p>

    <p class="note">This can in theory throw an exception, if attempting to deserialize a large
    <code data-x="idl-ArrayBuffer">ArrayBuffer</code> when not enough memory is available.</p>
   </li>
  </ol>

  </div>

  <p>The following are the <span>event handlers</span> (and their corresponding <span data-x="event
  handler event type">event handler event types</span>) <span w-nodev>that must be</span> supported,
  as <span>event handler IDL attributes</span>, by all objects implementing the
  <code>NavigationHistoryEntry</code> interface:</p>

  <table>
   <thead>
    <tr>
     <th><span data-x="event handlers">Event handler</span>
     <th><span>Event handler event type</span>
   <tbody>
    <tr>
     <td><dfn attribute for="NavigationHistoryEntry"><code data-x="handler-NavigationHistoryEntry-ondispose">ondispose</code></dfn>
     <td><code data-x="event-dispose">dispose</code>
  </table>


  <h5>The history entry list</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>entries</var> = <span data-x="dom-navigation">navigation</span>.<span subdfn data-x="dom-Navigation-entries">entries()</span></code></dt>
   <dd><p>Returns an array of <code>NavigationHistoryEntry</code> instances represent the current
   navigation history entry list, i.e., all <span data-x="session history entry">session history
   entries</span> for this <span>navigable</span> that are <span>same origin</span> and contiguous
   to the <span data-x="nav-current-history-entry">current session history entry</span>.</p>

   <dt><code data-x=""><span data-x="dom-navigation">navigation</span>.<span subdfn data-x="dom-Navigation-currentEntry">currentEntry</span></code></dt>
   <dd><p>Returns the <code>NavigationHistoryEntry</code> corresponding to the <span
   data-x="nav-current-history-entry">current session history entry</span>.</p>

   <dt><code data-x=""><span data-x="dom-navigation">navigation</span>.<span subdfn data-x="dom-Navigation-updateCurrentEntry">updateCurrentEntry</span>({ <span data-x="dom-NavigationUpdateCurrentEntryOptions-state">state</span> })</code></dt>
   <dd>
    <p>Updates the <span data-x="she-navigation-api-state">navigation API state</span> of the <span
    data-x="nav-current-history-entry">current session history entry</span>, without performing a
    navigation like <code data-x="dom-navigation-reload">navigation.reload()</code> would do.</p>

    <p>This method is best used to capture updates to the page that have already happened, and need
    to be reflected into the navigation API state. For cases where the state update is meant to
    drive a page update, instead use <code
    data-x="dom-Navigation-navigate">navigation.navigate()</code> or <code
    data-x="dom-Navigation-reload">navigation.reload()</code>, which will trigger a <code
    data-x="event-navigate">navigate</code> event.</p>
   </dd>

   <dt><code data-x=""><span data-x="dom-navigation">navigation</span>.<span subdfn data-x="dom-Navigation-canGoBack">canGoBack</span></code></dt>
   <dd><p>Returns true if the current <span data-x="nav-current-history-entry">current session
   history entry</span> (i.e., <code data-x="dom-Navigation-currentEntry">currentEntry</code>) is
   not the first one in the navigation history entry list (i.e., in <code
   data-x="dom-Navigation-entries">entries()</code>). This means that there is a previous
   <span>session history entry</span> for this <span>navigable</span>, and its <span
   data-x="she-document-state">document state</span>'s <span
   data-x="document-state-origin">origin</span> is <span>same origin</span> with the current
   <code>Document</code>'s <span data-x="concept-document-origin">origin</span>.</p>

   <dt><code data-x=""><span data-x="dom-navigation">navigation</span>.<span subdfn data-x="dom-Navigation-canGoForward">canGoForward</span></code></dt>
   <dd><p>Returns true if the current <span data-x="nav-current-history-entry">current session
   history entry</span> (i.e., <code data-x="dom-Navigation-currentEntry">currentEntry</code>) is
   not the last one in the navigation history entry list (i.e., in <code
   data-x="dom-Navigation-entries">entries()</code>). This means that there is a next <span>session
   history entry</span> for this <span>navigable</span>, and its <span
   data-x="she-document-state">document state</span>'s <span
   data-x="document-state-origin">origin</span> is <span>same origin</span> with the current
   <code>Document</code>'s <span data-x="concept-document-origin">origin</span>.</p>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="Navigation"><code data-x="dom-Navigation-entries">entries()</code></dfn>
  method steps are:</p>

  <ol>
   <li><p>If <span>this</span> <span>has entries and events disabled</span>, then return the empty
   list.</p></li>

   <li>
    <p>Return <span>this</span>'s <span data-x="navigation-entry-list">entry list</span>.</p>

    <p class="note">Recall that because of Web IDL's sequence type conversion rules, this will
    create a new JavaScript array object on each call. That is, <code data-x=""><span
    data-x="dom-Navigation-entries">navigation.entries()</span> !== <span
    data-x="dom-Navigation-entries">navigation.entries()</span></code>.</p>
   </li>
  </ol>

  <p>The <dfn attribute for="Navigation"><code
  data-x="dom-Navigation-currentEntry">currentEntry</code></dfn> getter steps are to return the
  <span data-x="navigation-current-entry">current entry</span> of <span>this</span>.</p>

  <p>The <dfn method for="Navigation"><code
  data-x="dom-Navigation-updateCurrentEntry">updateCurrentEntry(<var>options</var>)</code></dfn>
  method steps are:</p>

  <ol>
   <li><p>Let <var>current</var> be the <span data-x="navigation-current-entry">current entry</span>
   of <span>this</span>.</p></li>

   <li><p>If <var>current</var> is null, then throw an <span>"<code>InvalidStateError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Let <var>serializedState</var> be
   <span>StructuredSerializeForStorage</span>(<var>options</var>["<code
   data-x="dom-NavigationUpdateCurrentEntryOptions-state">state</code>"]), rethrowing any
   exceptions.</p></li>

   <li><p>Set <var>current</var>'s <span data-x="nhe-she">session history entry</span>'s <span
   data-x="she-navigation-api-state">navigation API state</span> to
   <var>serializedState</var>.</p></li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
   data-x="event-currententrychange">currententrychange</code> at <span>this</span> using
   <code>NavigationCurrentEntryChangeEvent</code>, with its <code
   data-x="dom-NavigationCurrentEntryChangeEvent-navigationType">navigationType</code> attribute
   initialized to null and its <code data-x="dom-NavigationCurrentEntryChangeEvent-from">from</code>
   initialized to <var>current</var>.</p></li>
  </ol>

  <p>The <dfn attribute for="Navigation"><code
  data-x="dom-Navigation-canGoBack">canGoBack</code></dfn> getter steps are:</p>

  <ol>
   <li><p>If <span>this</span> <span>has entries and events disabled</span>, then return
   false.</p></li>

   <li><p><span>Assert</span>: <span>this</span>'s <span
   data-x="navigation-current-entry-index">current entry index</span> is not &minus;1.</p></li>

   <li><p>If <span>this</span>'s <span data-x="navigation-current-entry-index">current entry
   index</span> is 0, then return false.</p></li>

   <li><p>Return true.</p></li>
  </ol>

  <p>The <dfn attribute for="Navigation"><code
  data-x="dom-Navigation-canGoForward">canGoForward</code></dfn> getter steps are:</p>

  <ol>
   <li><p>If <span>this</span> <span>has entries and events disabled</span>, then return
   false.</p></li>

   <li><p><span>Assert</span>: <span>this</span>'s <span
   data-x="navigation-current-entry-index">current entry index</span> is not &minus;1.</p></li>

   <li><p>If <span>this</span>'s <span data-x="navigation-current-entry-index">current entry
   index</span> is equal to <span>this</span>'s <span data-x="navigation-entry-list">entry
   list</span>'s <span data-x="list size">size</span> &minus; 1, then return false.</p></li>

   <li><p>Return true.</p></li>
  </ol>

  </div>


  <h5 id="navigation-api-initiating-navigations">Initiating navigations</h5>

  <dl class="domintro">
   <dt><code data-x="">{ <span data-x="dom-NavigationResult-committed">committed</span>, <span data-x="dom-NavigationResult-finished">finished</span> } = <span data-x="dom-navigation">navigation</span>.<span subdfn data-x="dom-Navigation-navigate">navigate</span>(<var>url</var>)</code></dt>
   <dt><code data-x="">{ <span data-x="dom-NavigationResult-committed">committed</span>, <span data-x="dom-NavigationResult-finished">finished</span> } = <span data-x="dom-navigation">navigation</span>.<span data-x="dom-Navigation-navigate">navigate</span>(<var>url</var>, <var>options</var>)</code></dt>
   <dd>
    <p><span data-x="navigate">Navigates</span> the current page to the given <var>url</var>.
    <var>options</var> can contain the following values:</p>

    <ul>
     <li><p><code subdfn data-x="dom-NavigationNavigateOptions-history">history</code> can be set to
     "<code subdfn data-x="NavigationHistoryBehavior-replace">replace</code>" to replace the current
     session history entry, instead of pushing a new one.</p></li>

     <li><p><code subdfn data-x="dom-NavigationOptions-info">info</code> can be set to any value; it
     will populate the <code data-x="dom-NavigateEvent-info">info</code> property of the
     corresponding <code>NavigateEvent</code>.</p></li>

     <li><p><code subdfn data-x="dom-NavigationNavigateOptions-state">state</code> can be set to any
     <span data-x="serializable objects">serializable</span> value; it will populate the state
     retrieved by <code
     data-x="dom-NavigationHistoryEntry-getState">navigation.currentEntry.getState()</code> once the
     navigation completes, for same-document navigations. (It will be ignored for navigations that
     end up cross-document.)</p></li>
    </ul>

    <p>By default this will perform a full navigation (i.e., a cross-document navigation, unless the
    given URL differs only in a <span data-x="concept-url-fragment">fragment</span> from the current
    one). The <code data-x="dom-NavigateEvent-intercept">navigateEvent.intercept()</code> method can
    be used to convert it into a same-document navigation.</p>

    <p>The returned promises will behave as follows:</p>

    <ul>
     <li><p>For navigations that get aborted, both promises will reject with an
     <span>"<code>AbortError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>For same-document navigations created by using the <code
     data-x="dom-NavigateEvent-intercept">navigateEvent.intercept()</code> method, <code
     data-x="dom-NavigationResult-committed">committed</code> will fulfill immediately, and <code
     data-x="dom-NavigationResult-finished">finished</code> will fulfill or reject according to any
     promsies returned by handlers passed to <code
     data-x="dom-NavigateEvent-intercept">intercept()</code>.</p></li>

     <li><p>For other same-document navigations (e.g., non-intercepted <span
     data-x="navigate-fragid">fragment navigations</span>), both promises will fulfill
     immediately.</p></li>

     <li><p>For cross-document navigations, or navigations that result in 204 or 205 <span
     data-x="concept-response-status">statuses</span> or `<code
     data-x="http-content-disposition">Content-Disposition: attachment</code>` header fields from
     the server (and thus do not actually navigate), both promises will never settle.</p></li>
    </ul>

    <p>In all cases, when the returned promises fulfill, it will be with the
    <code>NavigationHistoryEntry</code> that was navigated to.</p>
   </dd>

   <dt><code data-x="">{ <span data-x="dom-NavigationResult-committed">committed</span>, <span data-x="dom-NavigationResult-finished">finished</span> } = <span data-x="dom-navigation">navigation</span>.<span subdfn data-x="dom-Navigation-reload">reload</span>(<var>options</var>)</code></dt>
   <dd>
    <p><span data-x="reload">Reloads</span> the current page. <var>options</var> can contain <code
    data-x="dom-NavigationOptions-info">info</code> and <code subdfn
    data-x="dom-NavigationReloadOptions-state">state</code>, which behave as described above.</p>

    <p>The default behavior of performing a from-network-or-cache reload of the current page can be
    overriden by the using the <code
    data-x="dom-NavigateEvent-intercept">navigateEvent.intercept()</code> method. Doing so will mean
    this call only updates state or passes along the appropriate <code
    data-x="dom-NavigationOptions-info">info</code>, plus performing whater actions the <code
    data-x="event-navigate">navigate</code> event handlers see fit to carry out.</p>

    <p>The returned promises will behave as follows:</p>

    <ul>
     <li><p>If the reload is intercepted by using the <code
     data-x="dom-NavigateEvent-intercept">navigateEvent.intercept()</code> method, <code
     data-x="dom-NavigationResult-committed">committed</code> will fulfill immediately, and <code
     data-x="dom-NavigationResult-finished">finished</code> will fulfill or reject according to any
     promsies returned by handlers passed to <code
     data-x="dom-NavigateEvent-intercept">intercept()</code>.</p></li>

     <li><p>Otherwise, both promises will never settle.</p></li>
    </ul>
   </dd>

   <dt><code data-x="">{ <span data-x="dom-NavigationResult-committed">committed</span>, <span data-x="dom-NavigationResult-finished">finished</span> } = <span data-x="dom-navigation">navigation</span>.<span subdfn data-x="dom-Navigation-traverseTo">traverseTo</span>(<var>key</var>)</code></dt>
   <dt><code data-x="">{ <span data-x="dom-NavigationResult-committed">committed</span>, <span data-x="dom-NavigationResult-finished">finished</span> } = <span data-x="dom-navigation">navigation</span>.<span data-x="dom-Navigation-traverseTo">traverseTo</span>(<var>key</var>, { <span data-x="dom-NavigationOptions-info">info</span> })</code></dt>
   <dd>
    <p><span data-x="apply the traverse history step">Traverses</span> to the closest <span>session
    history entry</span> that matches the <code>NavigationHistoryEntry</code> with the given
    <var>key</var>. <code data-x="dom-NavigationOptions-info">info</code> can be set to any value;
    it will populate the <code data-x="dom-NavigateEvent-info">info</code> property of the
    corresponding <code>NavigateEvent</code>.</p>

    <p>If a traversal to that <span>session history entry</span> is already in progress, then this
    will return the promises for that original traversal, and <code
    data-x="dom-NavigateEvent-info">info</code> will be ignored.</p>

    <p>The returned promises will behave as follows:</p>

    <ul>
     <li><p>If there is no <code>NavigationHistoryEntry</code> in <code
     data-x="dom-Navigation-entries">navigation.entries()</code> whose <code
     data-x="dom-NavigationHistoryEntry-key">key</code> matches <var>key</var>, both promises will
     reject with an <span>"<code>InvalidStateError</code>"</span>
     <code>DOMException</code>.</p></li>

     <li><p>For same-document traversals intercepted by the <code
     data-x="dom-NavigateEvent-intercept">navigateEvent.intercept()</code> method, <code
     data-x="dom-NavigationResult-committed">committed</code> will fulfill as soon as the traversal
     is processed and <code data-x="dom-Navigation-currentEntry">navigation.currentEntry</code> is
     updated, and <code data-x="dom-NavigationResult-finished">finished</code> will fulfill or
     reject according to any promsies returned by the handlers passed to <code
     data-x="dom-NavigateEvent-intercept">intercept()</code>.</p></li>

     <li><p>For non-intercepted same-document travesals, both promises will fulfill as soon as the
     traversal is processed and <code
     data-x="dom-Navigation-currentEntry">navigation.currentEntry</code> is updated.</p></li>

     <li><p>For cross-document traversals, including attempted cross-document traversals that end up
     resulting in a 204 or 205 <span data-x="concept-response-status">statuses</span> or `<code
     data-x="http-content-disposition">Content-Disposition: attachment</code>` header fields from
     the server (and thus do not actually traverse), both promises will never settle.</p></li>
    </ul>
   </dd>

   <dt><code data-x="">{ <span data-x="dom-NavigationResult-committed">committed</span>, <span data-x="dom-NavigationResult-finished">finished</span> } = <span data-x="dom-navigation">navigation</span>.<span subdfn data-x="dom-Navigation-back">back</span>(<var>key</var>)</code></dt>
   <dt><code data-x="">{ <span data-x="dom-NavigationResult-committed">committed</span>, <span data-x="dom-NavigationResult-finished">finished</span> } = <span data-x="dom-navigation">navigation</span>.<span data-x="dom-Navigation-back">back</span>(<var>key</var>, { <span data-x="dom-NavigationOptions-info">info</span> })</code></dt>
   <dd>
    <p>Traverses to the closest previous <span>session history entry</span> which results in this
    <span>navigable</span> traversing, i.e., which corresponds to a different
    <code>NavigationHistoryEntry</code> and thus will cause <code
    data-x="dom-Navigation-currentEntry">navigation.currentEntry</code> to change. <code
    data-x="dom-NavigationOptions-info">info</code> can be set to any value; it will populate the
    <code data-x="dom-NavigateEvent-info">info</code> property of the corresponding
    <code>NavigateEvent</code>.</p>

    <p>If a traversal to that <span>session history entry</span> is already in progress, then this
    will return the promises for that original traversal, and <code
    data-x="dom-NavigateEvent-info">info</code> will be ignored.</p>

    <p>The returned promises behave equivalently to those returned by <code
    data-x="dom-Navigation-traverseTo">traverseTo()</code>.</p>
   </dd>

   <dt><code data-x="">{ <span data-x="dom-NavigationResult-committed">committed</span>, <span data-x="dom-NavigationResult-finished">finished</span> } = <span data-x="dom-navigation">navigation</span>.<span subdfn data-x="dom-Navigation-forward">forward</span>(<var>key</var>)</code></dt>
   <dt><code data-x="">{ <span data-x="dom-NavigationResult-committed">committed</span>, <span data-x="dom-NavigationResult-finished">finished</span> } = <span data-x="dom-navigation">navigation</span>.<span data-x="dom-Navigation-forward">forward</span>(<var>key</var>, { <span data-x="dom-NavigationOptions-info">info</span> })</code></dt>
   <dd>
    <p>Traverses to the closest forward <span>session history entry</span> which results in this
    <span>navigable</span> traversing, i.e., which corresponds to a different
    <code>NavigationHistoryEntry</code> and thus will cause <code
    data-x="dom-Navigation-currentEntry">navigation.currentEntry</code> to change. <code
    data-x="dom-NavigationOptions-info">info</code> can be set to any value; it will populate the
    <code data-x="dom-NavigateEvent-info">info</code> property of the corresponding
    <code>NavigateEvent</code>.</p>

    <p>If a traversal to that <span>session history entry</span> is already in progress, then this
    will return the promises for that original traversal, and <code
    data-x="dom-NavigateEvent-info">info</code> will be ignored.</p>

    <p>The returned promises behave equivalently to those returned by <code
    data-x="dom-Navigation-traverseTo">traverseTo()</code>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="Navigation"><code
  data-x="dom-Navigation-navigate">navigate(<var>url</var>, <var>options</var>)</code></dfn> method
  steps are:</p>

  <ol>
   <li><p>Let <var>urlRecord</var> be the result of <span data-x="parse a URL">parsing a URL</span>
   given <var>url</var>, relative to <span>this</span>'s <span>relevant settings
   object</span>.</p></li>

   <li><p>If <var>urlRecord</var> is failure, then return an <span
   data-x="navigation-api-early-error-result">early error result</span> for a
   <span>"<code>SyntaxError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>document</var> be <span>this</span>'s <span>relevant global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>If <var>options</var>["<code
   data-x="dom-NavigationNavigateOptions-history">history</code>"] is "<code
   data-x="NavigationHistoryBehavior-push">push</code>", and <span>the navigation must be a
   replace</span> given <var>urlRecord</var> and <var>document</var>, then return an <span
   data-x="navigation-api-early-error-result">early error result</span> for a
   <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p>

   <li><p>Let <var>state</var> be <var>options</var>["<code
   data-x="dom-NavigationNavigateOptions-state">state</code>"], if it <span data-x="map
   exists">exists</span>; otherwise, undefined.</p></li>

   <li>
    <p>Let <var>serializedState</var> be
    <span>StructuredSerializeForStorage</span>(<var>state</var>). If this throws an exception, then
    return an <span data-x="navigation-api-early-error-result">early error result</span> for that
    exception.</p>

    <p class="note">It is important to perform this step early, since serialization can invoke web
    developer code, which in turn might change various things we check in later steps.</p>
   </li>

   <li><p>If <var>document</var> is not <span>fully active</span>, then return an <span
   data-x="navigation-api-early-error-result">early error result</span> for an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p>

   <li><p>If <var>document</var>'s <span>unload counter</span> is greater than 0, then return an
   <span data-x="navigation-api-early-error-result">early error result</span> for an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>info</var> be <var>options</var>["<code
   data-x="dom-NavigationOptions-info">info</code>"], if it <span data-x="map exists">exists</span>;
   otherwise, undefined.</p></li>

   <li><p>Let <var>apiMethodTracker</var> be the result of <span data-x="maybe set the upcoming
   non-traverse API method tracker">maybe setting the upcoming non-traverse API method
   tracker</span> for <span>this</span> given <var>info</var> and
   <var>serializedState</var>.</p></li>

   <li>
    <p><!-- DONAV navigation API --><span>Navigate</span> <var>document</var>'s <span>node
    navigable</span> to <var>urlRecord</var> using <var>document</var>, with <i
    data-x="navigation-hh">historyHandling</i> set to <var>options</var>["<code
    data-x="dom-NavigationNavigateOptions-history">history</code>"] and <i
    data-x="navigation-navigation-api-state">navigationAPIState</i> set to
    <var>serializedState</var>.</p>

    <p class="note">Unlike <code data-x="dom-location-assign">location.assign()</code> and friends,
    which are exposed across <span data-x="same origin-domain">origin-domain</span> boundaries,
    <code data-x="dom-Navigation-navigate">navigation.navigate()</code> can only be accessed by code
    with direct synchronous access to the <code data-x="dom-navigation">window.navigation</code>
    property. Thus, we avoid the complications about attributing the source document of the
    navigation, and we don't need to deal with the <span>allowed by sandboxing to navigate</span>
    check and its acccompanying <i data-x="exceptionsEnabled">exceptionsEnabled</i> flag. We just
    treat all navigations as if they come from the <code>Document</code> corresponding to this
    <code>Navigation</code> object itself (i.e., <var>document</var>).</p>
   </li>

   <li>
    <p>If <span>this</span>'s <span>upcoming non-traverse API method tracker</span> is
    <var>apiMethodTracker</var>, then:</p>

    <p class="note">If the <span>upcoming non-traverse API method tracker</span> is still
    <var>apiMethodTracker</var>, this means that the <span>navigate</span> algorithm bailed out
    before ever getting to the <span>inner <code data-x="event-navigate">navigate</code> event
    firing algorithm</span> which would <span data-x="promote an upcoming API method tracker to
    ongoing">promote that upcoming API method tracker to ongoing</span>.</p>

    <ol>
     <li><p>Set <span>this</span>'s <span>upcoming non-traverse API method tracker</span> to
     null.</p></li>

     <li><p>Return an <span data-x="navigation-api-early-error-result">early error result</span> for
     an <span>"<code>AbortError</code>"</span> <code>DOMException</code>.</p></li>
    </ol>
   </li>

   <li><p>Return a <span>navigation API method tracker-derived result</span> for
   <var>apiMethodTracker</var>.</p></li>
  </ol>

  <p>The <dfn method for="Navigation"><code
  data-x="dom-Navigation-reload">reload(<var>options</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>document</var> be <span>this</span>'s <span>relevant global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>Let <var>serializedState</var> be
   <span>StructuredSerializeForStorage</span>(undefined).</p></li>

   <li>
    <p>If <var>options</var>["<code data-x="dom-NavigationReloadOptions-state">state</code>"] <span
    data-x="map exists">exists</span>, then set <var>serializedState</var> to
    <span>StructuredSerializeForStorage</span>(<var>options</var>["<code
    data-x="dom-NavigationReloadOptions-state">state</code>"]). If this throws an exception, then
    return an <span data-x="navigation-api-early-error-result">early error result</span> for that
    exception.</p>

    <p class="note">It is important to perform this step early, since serialization can invoke web
    developer code, which in turn might change various things we check in later steps.</p>
   </li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p>Let <var>current</var> be the <span data-x="navigation-current-entry">current
     entry</span> of <span>this</span>.</p></li>

     <li><p>If <var>current</var> is not null, then set <var>serializedState</var> to
     <var>current</var>'s <span data-x="nhe-she">session history entry</span>'s <span
     data-x="she-navigation-api-state">navigation API state</span>.</p></li>
    </ol>
   </li>

   <li><p>If <var>document</var> is not <span>fully active</span>, then return an <span
   data-x="navigation-api-early-error-result">early error result</span> for an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p>

   <li><p>If <var>document</var>'s <span>unload counter</span> is greater than 0, then return an
   <span data-x="navigation-api-early-error-result">early error result</span> for an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>info</var> be <var>options</var>["<code
   data-x="dom-NavigationOptions-info">info</code>"], if it <span data-x="map exists">exists</span>;
   otherwise, undefined.</p></li>

   <li><p>Let <var>apiMethodTracker</var> be the result of <span data-x="maybe set the upcoming
   non-traverse API method tracker">maybe setting the upcoming non-traverse API method
   tracker</span> for <span>this</span> given <var>info</var> and
   <var>serializedState</var>.</p></li>

   <li><p><span>Reload</span> <var>document</var>'s <span>node navigable</span> with <i
   data-x="reload-navigation-api-state">navigationAPIState</i> set to
   <var>serializedState</var>.</p></li>

   <li><p>Return a <span>navigation API method tracker-derived result</span> for
   <var>apiMethodTracker</var>.</p></li>
  </ol>

  <p>The <dfn method for="Navigation"><code
  data-x="dom-Navigation-traverseTo">traverseTo(<var>key</var>, <var>options</var>)</code></dfn>
  method steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="navigation-current-entry-index">current entry
   index</span> is &minus;1, then return an <span data-x="navigation-api-early-error-result">early
   error result</span> for an <span>"<code>InvalidStateError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>If <span>this</span>'s <span data-x="navigation-entry-list">entry list</span> does not
   <span data-x="list contains">contain</span> a <code>NavigationHistoryEntry</code> whose <span
   data-x="nhe-she">session history entry</span>'s <span data-x="she-navigation-api-key">navigation
   API key</span> equals <var>key</var>, then return an <span
   data-x="navigation-api-early-error-result">early error result</span> for an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Return the result of <span>performing a navigation API traversal</span> given
   <span>this</span>, <var>key</var>, and <var>options</var>.</p></li>
  </ol>

  <p>The <dfn method for="Navigation"><code
  data-x="dom-Navigation-back">back(<var>options</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="navigation-current-entry-index">current entry
   index</span> is &minus;1 or 0, then return an <span
   data-x="navigation-api-early-error-result">early error result</span> for an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>key</var> be <span>this</span>'s <span data-x="navigation-entry-list">entry
   list</span>[<span>this</span>'s <span data-x="navigation-current-entry-index">current entry
   index</span> &minus; 1]'s <span data-x="nhe-she">session history entry</span>'s <span
   data-x="she-navigation-api-key">navigation API key</span>.</p></li>

   <li><p>Return the result of <span>performing a navigation API traversal</span> given
   <span>this</span>, <var>key</var>, and <var>options</var>.</p></li>
  </ol>

  <p>The <dfn method for="Navigation"><code
  data-x="dom-Navigation-forward">forward(<var>options</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="navigation-current-entry-index">current entry
   index</span> is &minus;1 or is equal to <span>this</span>'s <span
   data-x="navigation-entry-list">entry list</span>'s <span data-x="list size">size</span> &minus;
   1, then return an <span data-x="navigation-api-early-error-result">early error result</span> for
   an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>key</var> be <span>this</span>'s <span data-x="navigation-entry-list">entry
   list</span>[<span>this</span>'s <span data-x="navigation-current-entry-index">current entry
   index</span> + 1]'s <span data-x="nhe-she">session history entry</span>'s <span
   data-x="she-navigation-api-key">navigation API key</span>.</p></li>

   <li><p>Return the result of <span>performing a navigation API traversal</span> given
   <span>this</span>, <var>key</var>, and <var>options</var>.</p></li>
  </ol>

  <p>To <dfn data-x="performing a navigation API traversal">perform a navigation API traversal</dfn>
  given a <code>Navigation</code> <var>navigation</var>, a string <var>key</var>, and a
  <code>NavigationOptions</code> <var>options</var>:</p>

  <ol>
   <li><p>Let <var>document</var> be <var>navigation</var>'s <span>relevant global object</span>'s
   <span data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>If <var>document</var> is not <span>fully active</span>, then return an <span
   data-x="navigation-api-early-error-result">early error result</span> for an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p>

   <li><p>If <var>document</var>'s <span>unload counter</span> is greater than 0, then return an
   <span data-x="navigation-api-early-error-result">early error result</span> for an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>current</var> be the <span data-x="navigation-current-entry">current entry</span>
   of <var>navigation</var>.</p></li>

   <li><p>If <var>key</var> equals <var>current</var>'s <span data-x="nhe-she">session history
   entry</span>'s <span data-x="she-navigation-api-key">navigation API key</span>, then return «[
   "<code data-x="dom-NavigationResult-committed">committed</code>" → <span>a promise resolved
   with</span> <var>current</var>, "<code data-x="dom-NavigationResult-finished">finished</code>"
   → <span>a promise resolved with</span> <var>current</var> ]».</p></li>

   <li><p>If <var>navigation</var>'s <span>upcoming traverse API method
   trackers</span>[<var>key</var>] <span data-x="map exists">exists</span>, then return a
   <span>navigation API method tracker-derived result</span> for <var>navigation</var>'s
   <span>upcoming traverse API method trackers</span>[<var>key</var>].</p></li>

   <li><p>Let <var>info</var> be <var>options</var>["<code
   data-x="dom-NavigationOptions-info">info</code>"], if it <span data-x="map exists">exists</span>;
   otherwise, undefined.</p></li>

   <li><p>Let <var>apiMethodTracker</var> be the result of <span data-x="add an upcoming traverse
   API method tracker">adding an upcoming traverse API method tracker</span> for
   <var>navigation</var> given <var>key</var> and <var>info</var>.</p></li>

   <li><p>Let <var>navigable</var> be <var>document</var>'s <span>node navigable</span>.</p></li>

   <li><p>Let <var>traversable</var> be <var>navigable</var>'s <span
   data-x="nav-traversable">traversable navigable</span>.</p></li>

   <li><p>Let <var>sourceSnapshotParams</var> be the result of <span>snapshotting source snapshot
   params</span> given <var>document</var>.</p></li>

   <li>
    <p><span data-x="tn-append-session-history-traversal-steps">Append the following session history
    traversal steps</span> to <var>traversable</var>:</p>

    <ol>
     <li><p>Let <var>navigableSHEs</var> be the result of <span>getting session history
     entries</span> given <var>navigable</var>.</p></li>

     <li>
      <p>Let <var>targetSHE</var> be the <span>session history entry</span> in
      <var>navigableSHEs</var> whose <span data-x="she-navigation-api-key">navigation API key</span>
      is <var>key</var>. If no such entry exists, then:</p>

      <ol>
       <li><p><span>Queue a global task</span> on the <span>navigation and traversal task
       source</span> given <var>navigation</var>'s <span>relevant global object</span> to
       <span>reject the finished promise</span> for <var>apiMethodTracker</var> with an
       <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

       <li><p>Abort these steps.</p></li>
      </ol>

      <p class="note">This path is taken if <var>navigation</var>'s <span
      data-x="navigation-entry-list">entry list</span> was outdated compared to
      <var>navigableSHEs</var>, which can occur for brief periods while all the relevant threads and
      processes are being synchronized in reaction to a history change.</p>
     </li>

     <li>
      <p>If <var>targetSHE</var> is <var>navigable</var>'s <span
      data-x="nav-active-history-entry">active session history entry</span>, then abort these
      steps.</p>

      <p class="note">This can occur if a previously <span
      data-x="tn-append-session-history-traversal-steps">queued</span> traversal already took us to
      this <span>session history entry</span>. In that case the previous traversal will have dealt
      with <var>apiMethodTracker</var> already.</p>
     </li>

     <li><p>Let <var>result</var> be the result of <span data-x="apply the traverse history
     step">applying the traverse history step</span> given by <var>targetSHE</var>'s <span
     data-x="she-step">step</span> to <var>traversable</var>, given <var>sourceSnapshotParams</var>,
     <var>navigable</var>, and "<code data-x="uni-none">none</code>".</p></li>

     <li><p>If <var>result</var> is "<code data-x="">canceled-by-beforeunload</code>", then
     <span>queue a global task</span> on the <span>navigation and traversal task source</span> given
     <var>navigation</var>'s <span>relevant global object</span> to <span>reject the finished
     promise</span> for <var>apiMethodTracker</var> with a new
     <span>"<code>AbortError</code>"</span> <code>DOMException</code> created in
     <var>navigation</var>'s <span data-x="concept-relevant-realm">relevant realm</span>.</p></li>

     <li>
      <p>If <var>result</var> is "<code data-x="">initiator-disallowed</code>", then <span>queue a
      global task</span> on the <span>navigation and traversal task source</span> given
      <var>navigation</var>'s <span>relevant global object</span> to <span>reject the finished
      promise</span> for <var>apiMethodTracker</var> with a new
      <span>"<code>SecurityError</code>"</span> <code>DOMException</code> created in
      <var>navigation</var>'s <span data-x="concept-relevant-realm">relevant realm</span>.</p>

      <div class="note">
       <p>When <var>result</var> is "<code data-x="">canceled-by-beforeunload</code>" or "<code
       data-x="">initiator-disallowed</code>", the <code data-x="event-navigate">navigate</code>
       event was never fired, <span data-x="abort the ongoing navigation">aborting the ongoing
       navigation</span> would not be correct; it would result in a <code
       data-x="event-navigateerror">navigateerror</code> event without a preceding <code
       data-x="event-navigate">navigate</code> event.</p>

       <p>In the "<code data-x="">canceled-by-navigate</code>" case, <code
       data-x="event-navigate">navigate</code> <em>is</em> fired, but the <span>inner <code
       data-x="event-navigate">navigate</code> event firing algorithm</span> will take care of
       <span data-x="abort the ongoing navigation">aborting the ongoing navigation</span>.</p>
      </div>
     </li>
    </ol>
   </li>

   <li><p>Return a <span>navigation API method tracker-derived result</span> for
   <var>apiMethodTracker</var>.</p></li>
  </ol>

  <p>An <dfn data-x="navigation-api-early-error-result">early error result</dfn> for an exception
  <var>e</var> is a <code>NavigationResult</code> dictionary instance given by «[ "<code
  data-x="dom-NavigationResult-committed">committed</code>" → <span>a promise rejected with</span>
  <var>e</var>, "<code data-x="dom-NavigationResult-finished">finished</code>" → <span>a promise
  rejected with</span> <var>e</var> ]».</p>

  <p>A <dfn>navigation API method tracker-derived result</dfn> for a <span>navigation API method
  tracker</span> is a <code>NavigationResult</code> dictionary instance given by «[ "<code
  data-x="dom-NavigationResult-committed">committed</code>" → <var>apiMethodTracker</var>'s <span
  data-x="navigation-api-method-tracker-committed">committed promise</span>, "<code
  data-x="dom-NavigationResult-finished">finished</code>" → <var>apiMethodTracker</var>'s <span
  data-x="navigation-api-method-tracker-finished">finished promise</span> ]».</p>

  </div>


  <h5>Ongoing navigation tracking</h5>

  <div w-nodev>

  <p>During any given navigation (in the <span data-x="NavigationType">broad sense of the
  word</span>), the <code>Navigation</code> object needs to keep track of the following:</p>

  <table>
   <caption>For all navigations</caption>
   <thead>
    <tr>
     <th>State
     <th>Duration
     <th>Explanation
   <tbody>
    <tr>
     <td>The <code>NavigateEvent</code>
     <td>For the duration of event firing
     <td>So that if the navigation is canceled while the event is firing, we can <span data-x="canceled flag">cancel</span> the event
    <tr>
     <td>The event's <span data-x="concept-NavigateEvent-abort-controller">abort controller</span>
     <td>Until all promises returned from handlers passed to <code data-x="dom-NavigateEvent-intercept">intercept()</code> have settled
     <td>So that if the navigation is canceled, we can <span>signal abort</span>
    <tr>
     <td>Whether a new element was <span data-x="focusing steps">focused</span>
     <td>Until all promises returned from handlers passed to <code data-x="dom-NavigateEvent-intercept">intercept()</code> have settled
     <td>So that if one was, focus is not <span data-x="potentially reset the focus">reset</span>
    <tr>
     <td>The <code>NavigationHistoryEntry</code> being navigated to
     <td>From when it is determined, until all promises returned from handlers passed to <code data-x="dom-NavigateEvent-intercept">intercept()</code> have settled
     <td>So that we know what to resolve any <code data-x="dom-NavigationResult-committed">committed</code> and <code data-x="dom-NavigationResult-finished">finished</code> promises with
    <tr>
     <td>Any <code data-x="dom-NavigationResult-finished">finished</code> promise that was returned
     <td>Until all promises returned from handlers passed to <code data-x="dom-NavigateEvent-intercept">intercept()</code> have settled
     <td>So that we can resolve or reject it appropriately
  </table>

  <table>
   <caption>For non-"<code data-x="dom-NavigationType-traverse">traverse</code>" navigations</caption>
   <thead>
    <tr>
     <th>State
     <th>Duration
     <th>Explanation
   <tbody>
    <tr>
     <td>Any <code data-x="dom-NavigationNavigateOptions-state">state</code>
     <td>For the duration of event firing
     <td>So that we can update the current entry's <span data-x="she-navigation-api-state">navigation API state</span> if the event finishes firing without being <span data-x="canceled flag">canceled</span>
  </table>

  <table>
   <caption>For "<code data-x="dom-NavigationType-traverse">traverse</code>" navigations</caption>
   <thead>
    <tr>
     <th>State
     <th>Duration
     <th>Explanation
   <tbody>
    <tr>
     <td>Any <code data-x="dom-NavigationOptions-info">info</code>
     <td>Until the task is queued to fire the <code data-x="event-navigate">navigate</code> event
     <td>So that we can use it to fire the <code data-x="event-navigate">navigate</code> after the trip through the <span data-x="tn-session-history-traversal-queue">session history traversal queue</span>.
    <tr>
     <td>Any <code data-x="dom-NavigationResult-committed">committed</code> promise that was returned
     <td>Until the session history is updated (inside that same task)
     <td>So that we can resolve or reject it appropriately
    <tr>
     <td>Whether <code data-x="dom-NavigateEvent-intercept">intercept()</code> was called
     <td>Until the session history is updated (inside that same task)
     <td>So that we can suppress the normal scroll restoration logic in favor of the behavior given by the <code data-x="dom-NavigationInterceptOptions-scroll">scroll</code> option
  </table>

  <p>We also cannot assume there is only a single navigation requested at any given time, due to web
  developer code such as:</p>

  <pre><code class="js">const p1 = navigation.navigate(url1).finished;
const p2 = navigation.navigate(url2).finished;</code></pre>

  <p>That is, in this scenario, we need to ensure that while navigating to <code
  data-x="">url2</code>, we still have the promise <code data-x="">p1</code> around so that we can
  reject it. We can't just get rid of any ongoing navigation promises the moment the second call to
  <code data-x="dom-Navigation-navigate">navigate()</code> happens.</p>

  <p>We end up accomplishing all this by associating the following with each
  <code>Navigation</code>:</p>

  <ul>
   <li><p><dfn>Ongoing <code data-x="event-navigate">navigate</code> event</dfn>, a
   <code>NavigateEvent</code> or null, initially null.</p></li>

   <li><p><dfn>Focus changed during ongoing navigation</dfn>, a boolean, initially false.</p></li>

   <li><p><dfn>Suppress normal scroll restoration during ongoing navigation</dfn>, a boolean,
   initially false.</p></li>

   <li><p><dfn>Ongoing API method tracker</dfn>, a <span>navigation API method tracker</span> or
   null, initially null.</p></li>

   <li><p><dfn>Upcoming non-traverse API method tracker</dfn>, a <span>navigation API method
   tracker</span> or null, initially null.</p></li>

   <li><p><dfn>Upcoming traverse API method trackers</dfn>, an <span>ordered map</span> from strings
   to <span data-x="navigation API method tracker">navigation API method trackers</span>, initially
   empty.</p></li>
  </ul>

  <p class="note">The state here that is not stored in <span data-x="navigation API method
  tracker">navigation API method trackers</span> is state which needs to be tracked even for
  navigations that are not initiated via navigation API methods.</p>

  <p>A <dfn>navigation API method tracker</dfn> is a <span>struct</span> with the following <span
  data-x="struct item">items</span>:</p>

  <ul>
   <li><p>A <dfn data-x="navigation-api-method-tracker-navigation">navigation object</dfn>, a
   <code>Navigation</code></p></li>

   <li><p>A <dfn data-x="navigation-api-method-tracker-key">key</dfn>, a string or null</p></li>

   <li><p>An <dfn data-x="navigation-api-method-tracker-info">info</dfn>, a JavaScript
   value</p></li>

   <li><p>A <dfn data-x="navigation-api-method-tracker-state">serialized state</dfn>, a
   <span>serialized state</span> or null</p></li>

   <li><p>A <dfn data-x="navigation-api-method-tracker-committed-to-entry">committed-to entry</dfn>,
   a <code>NavigationHistoryEntry</code> or null</p></li>

   <li><p>A <dfn data-x="navigation-api-method-tracker-committed">committed promise</dfn>, a
   promise</p></li>

   <li><p>A <dfn data-x="navigation-api-method-tracker-finished">finished promise</dfn>, a
   promise</p></li>
  </ul>

  <p>All this state is then managed via the following algorithms.</p>

  <p>To <dfn>maybe set the upcoming non-traverse API method tracker</dfn> given a
  <code>Navigation</code> <var>navigation</var>, a JavaScript value <var>info</var>, and a
  <span>serialized state</span>-or-null <var>serializedState</var>:</p>

  <ol>
   <li><p>Let <var>committedPromise</var> and <var>finishedPromise</var> be new promises created in
   <var>navigation</var>'s <span data-x="concept-relevant-realm">relevant realm</span>.</p></li>

   <li>
    <p><span>Mark as handled</span> <var>finishedPromise</var>.</p>

    <div class="note" id="note-mark-as-handled-navigation-api-finished">
     <p>The web developer doesn’t necessarily care about <var>finishedPromise</var> being
     rejected:</p>

     <ul>
      <li><p>They might only care about <var>committedPromise</var>.</p></li>

      <li><p>They could be doing multiple synchronous navigations within the same task, in which
      case all but the last will be aborted (causing their <var>finishedPromise</var> to reject).
      This could be an application bug, but also could just be an emergent feature of disparate
      parts of the application overriding each others' actions.</p></li>

      <li><p>They might prefer to listen to other transition-failure signals instead of
      <var>finishedPromise</var>, e.g., the <code data-x="event-navigateerror">navigateerror</code>
      event, or the <code
      data-x="dom-NavigationTransition-finished">navigation.transition.finished</code>
      promise.</p></li>
     </ul>

     <p>As such, we mark it as handled to ensure that it never triggers <code
     data-x="event-unhandledrejection">unhandledrejection</code> events.</p>
    </div>
   </li>

   <li>
    <p>Let <var>apiMethodTracker</var> be a new <span>navigation API method tracker</span> with:</p>

    <dl class="props">
     <dt><span data-x="navigation-api-method-tracker-navigation">navigation object</span></dt>
     <dd><var>navigation</var></dd>

     <dt><span data-x="navigation-api-method-tracker-key">key</span></dt>
     <dd>null</dd>

     <dt><span data-x="navigation-api-method-tracker-info">info</span></dt>
     <dd><var>info</var></dd>

     <dt><span data-x="navigation-api-method-tracker-state">serialized state</span></dt>
     <dd><var>serializedState</var></dd>

     <dt><span data-x="navigation-api-method-tracker-committed-to-entry">committed-to entry</span></dt>
     <dd>null</dd>

     <dt><span data-x="navigation-api-method-tracker-committed">committed promise</span></dt>
     <dd><var>committedPromise</var></dd>

     <dt><span data-x="navigation-api-method-tracker-finished">finished promise</span></dt>
     <dd><var>finishedPromise</var></dd>
    </dl>
   </li>

   <li><p><span>Assert</span>: <var>navigation</var>'s <span>upcoming non-traverse API method
   tracker</span> is null.</p></li>

   <li id="dont-always-set-upcoming-non-traverse-api-method-tracker">
    <p>If <var>navigation</var> does not <span data-x="has entries and events disabled">have
    entries and events disabled</span>, then set <var>navigation</var>'s <span>upcoming
    non-traverse API method tracker</span> to <var>apiMethodTracker</var>.</p>

    <p class="note">If <var>navigation</var> <span>has entries and events disabled</span>, then
    <var>committedPromise</var> and <var>finishedPromise</var> will never fulfill (since we never
    create a <code>NavigationHistoryEntry</code> object for such <code>Document</code>s, and so we
    have nothing to resolve them with); there is no <code>NavigationHistoryEntry</code> to apply
    <var>serializedState</var> to; and there is no <code data-x="event-navigate">navigate</code>
    event to include <var>info</var> with. So, we don't need to track this API method call after
    all.</p>
   </li>

   <li><p>Return <var>apiMethodTracker</var>.</p></li>
  </ol>

  <p>To <dfn>add an upcoming traverse API method tracker</dfn> given a <code>Navigation</code>
  <var>navigation</var>, a string <var>destinationKey</var>, and a JavaScript value
  <var>info</var>:</p>

  <ol>
   <li><p>Let <var>committedPromise</var> and <var>finishedPromise</var> be new promises created in
   <var>navigation</var>'s <span data-x="concept-relevant-realm">relevant realm</span>.</p></li>

   <li>
    <p><span>Mark as handled</span> <var>finishedPromise</var>.</p>

    <p class="note">See the <a href="#note-mark-as-handled-navigation-api-finished">previous
    discussion</a> about why this is done.</p>
   </li>

   <li>
    <p>Let <var>apiMethodTracker</var> be a new <span>navigation API method tracker</span> with:</p>

    <dl class="props">
     <dt><span data-x="navigation-api-method-tracker-navigation">navigation object</span></dt>
     <dd><var>navigation</var></dd>

     <dt><span data-x="navigation-api-method-tracker-key">key</span></dt>
     <dd><var>destinationKey</var></dd>

     <dt><span data-x="navigation-api-method-tracker-info">info</span></dt>
     <dd><var>info</var></dd>

     <dt><span data-x="navigation-api-method-tracker-state">serialized state</span></dt>
     <dd>null</dd>

     <dt><span data-x="navigation-api-method-tracker-committed-to-entry">committed-to entry</span></dt>
     <dd>null</dd>

     <dt><span data-x="navigation-api-method-tracker-committed">committed promise</span></dt>
     <dd><var>committedPromise</var></dd>

     <dt><span data-x="navigation-api-method-tracker-finished">finished promise</span></dt>
     <dd><var>finishedPromise</var></dd>
    </dl>
   </li>

   <li><p>Set <var>navigation</var>'s <span>upcoming traverse API method
   trackers</span>[<var>destinationKey</var>] to <var>apiMethodTracker</var>.</p></li>

   <li><p>Return <var>apiMethodTracker</var>.</p></li>
  </ol>

  <p>To <dfn>promote an upcoming API method tracker to ongoing</dfn> given a <code>Navigation</code>
  <var>navigation</var> and a string-or-null <var>destinationKey</var>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>navigation</var>'s <span>ongoing API method tracker</span> is
   null.</p></li>

   <li>
    <p>If <var>destinationKey</var> is not null, then:</p>

    <ol>
     <li><p><span>Assert</span>: <var>navigation</var>'s <span>upcoming non-traverse API method
     tracker</span> is null.</p></li>

     <li>
      <p>If <var>navigation</var>'s <span>upcoming traverse API method
      trackers</span>[<var>destinationKey</var>] <span data-x="map exists">exists</span>, then:</p>

      <ol>
       <li><p>Set <var>navigation</var>'s <span>ongoing API method tracker</span> to
       <var>navigation</var>'s <span>upcoming traverse API method
       trackers</span>[<var>destinationKey</var>].</p></li>

       <li><p><span data-x="map remove">Remove</span> <var>navigation</var>'s <span>upcoming
       traverse API method trackers</span>[<var>destinationKey</var>].</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p>Set <var>navigation</var>'s <span>ongoing API method tracker</span> to
     <var>navigation</var>'s <span>upcoming non-traverse API method tracker</span>.</p></li>

     <li><p>Set <var>navigation</var>'s <span>upcoming non-traverse API method tracker</span> to
     null.</p></li>
    </ol>
   </li>
  </ol>

  <p>To <dfn data-x="navigation-api-method-tracker-clean-up">clean up</dfn> a <span>navigation API
  method tracker</span> <var>apiMethodTracker</var>:</p>

  <ol>
   <li><p>Let <var>navigation</var> be <var>apiMethodTracker</var>'s <span
   data-x="navigation-api-method-tracker-navigation">navigation object</span>.</p></li>

   <li><p>If <var>navigation</var>'s <span>ongoing API method tracker</span> is
   <var>apiMethodTracker</var>, then set <var>navigation</var>'s <span>ongoing API method
   tracker</span> to null.</p></li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p>Let <var>key</var> be <var>apiMethodTracker</var>'s <span
     data-x="navigation-api-method-tracker-key">key</span>.</p></li>

     <li><p><span>Assert</span>: <var>key</var> is not null.</p></li>

     <li><p><span>Assert</span>: <var>navigation</var>'s <span>upcoming traverse API method
     trackers</span>[<var>key</var>] <span data-x="map exists">exists</span>.</p></li>

     <li><p><span data-x="map remove">Remove</span> <var>navigation</var>'s <span>upcoming traverse
     API method trackers</span>[<var>key</var>].</p></li>
    </ol>
   </li>
  </ol>

  <p>To <dfn>notify about the committed-to entry</dfn> given a <span>navigation API method
  tracker</span> <var>apiMethodTracker</var> and a <code>NavigationHistoryEntry</code>
  <var>nhe</var>:</p>

  <ol>
   <li><p>Set <var>apiMethodTracker</var>'s <span
   data-x="navigation-api-method-tracker-committed-to-entry">committed-to entry</span> to
   <var>nhe</var>.</p></li>

   <li>
    <p>If <var>apiMethodTracker</var>'s <span
    data-x="navigation-api-method-tracker-state">serialized state</span> is not null, then set
    <var>nhe</var>'s <span data-x="nhe-she">session history entry</span>'s <span
    data-x="she-navigation-api-state">navigation API state</span> to <var>apiMethodTracker</var>'s
    <span data-x="navigation-api-method-tracker-state">serialized state</span>.</p>

    <p class="note">If it's null, then we're traversing to <var>nhe</var> via <code
    data-x="dom-Navigation-traverseTo">navigation.traverseTo()</code>, which does not allow changing
    the state.</p>

    <p class="note">At this point, <var>apiMethodTracker</var>'s <span
    data-x="navigation-api-method-tracker-state">serialized state</span> is no longer needed.
    Implementations might want to clear it out to avoid keeping it alive for the lifetime of the
    <span>navigation API method tracker</span>.</p>
   </li>

   <li>
    <p>Resolve <var>apiMethodTracker</var>'s <span
    data-x="navigation-api-method-tracker-committed">committed promise</span> with
    <var>nhe</var>.</p>

    <p class="note">At this point, <var>apiMethodTracker</var>'s <span
    data-x="navigation-api-method-tracker-committed">committed promise</span> is only needed in
    cases where it has not yet been returned to author code. Implementations might want to clear it
    out to avoid keeping it alive for the lifetime of the <span>navigation API method
    tracker</span>.</p>
   </li>
  </ol>

  <p>To <dfn>resolve the finished promise</dfn> for a <span>navigation API method tracker</span>
  <var>apiMethodTracker</var>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>apiMethodTracker</var>'s <span
   data-x="navigation-api-method-tracker-committed-to-entry">committed-to entry</span> is not
   null.</p></li>

   <li><p>Resolve <var>apiMethodTracker</var>'s <span
   data-x="navigation-api-method-tracker-finished">finished promise</span> with its <span
   data-x="navigation-api-method-tracker-committed-to-entry">committed-to entry</span>.</p></li>

   <li><p><span data-x="navigation-api-method-tracker-clean-up">Clean up</span>
   <var>apiMethodTracker</var>.</p></li>
  </ol>

  <p>To <dfn>reject the finished promise</dfn> for a <span>navigation API method tracker</span>
  <var>apiMethodTracker</var> with a JavaScript value <var>exception</var>:</p>

  <ol>
   <li>
    <p>Reject <var>apiMethodTracker</var>'s <span
    data-x="navigation-api-method-tracker-committed">committed promise</span> with
    <var>exception</var>.</p>

    <p class="note">This will do nothing if <var>apiMethodTracker</var>'s <span
    data-x="navigation-api-method-tracker-committed">committed promise</span> was previously
    resolved via <span>notify about the committed-to entry</span>.</p>
   </li>

   <li><p>Reject <var>apiMethodTracker</var>'s <span
   data-x="navigation-api-method-tracker-finished">finished promise</span> with
   <var>exception</var>.</p></li>

   <li><p><span data-x="navigation-api-method-tracker-clean-up">Clean up</span>
   <var>apiMethodTracker</var>.</p></li>
  </ol>

  <p>To <dfn>abort the ongoing navigation</dfn> given a <code>Navigation</code>
  <var>navigation</var> and an optional <code>DOMException</code> <var>error</var>:</p>

  <ol>
   <li><p>Let <var>event</var> be <var>navigation</var>'s <span>ongoing <code
   data-x="event-navigate">navigate</code> event</span>.</p></li>

   <li><p><span>Assert</span>: <var>event</var> is not null.</p></li>

   <li><p>Set <var>navigation</var>'s <span>focus changed during ongoing navigation</span> to
   false.</p></li>

   <li><p>Set <var>navigation</var>'s <span>suppress normal scroll restoration during ongoing
   navigation</span> to false.</p></li>

   <li><p>If <var>error</var> was not given, then let <var>error</var> be a new
   <span>"<code>AbortError</code>"</span> <code>DOMException</code> created in
   <var>navigation</var>'s <span data-x="concept-relevant-realm">relevant realm</span>.</p></li>

   <li><p>If <var>event</var>'s <span>dispatch flag</span> is set, then set <var>event</var>'s
   <span>canceled flag</span> to true.</p></li>

   <li><p><span>Signal abort</span> on <var>event</var>'s <span
   data-x="concept-NavigateEvent-abort-controller">abort controller</span> given
   <var>error</var>.</p></li>

   <li><p>Set <var>navigation</var>'s <span>ongoing <code data-x="event-navigate">navigate</code>
   event</span> to null.</p></li>

   <li>
    <p>Let <var>errorInfo</var> be the result of <span data-x="extract-error">extracting error
    information</span> from <var>error</var>.</p>

    <p class="note">For example, if this algorithm is reached because of a call to <code
    data-x="dom-window-stop">window.stop()</code>, these properties would probably end up
    initialized based on the line of script that called <code
    data-x="dom-window-stop">window.stop()</code>. But if it's because the user clicked the stop
    button, these properties would probably end up with default values like the empty string or
    0.</p>
   </li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
   data-x="event-navigateerror">navigateerror</code> at <var>navigation</var> using
   <code>ErrorEvent</code>, with additional attributes initialized according to
   <var>errorInfo</var>.</p></li>

   <li><p>If <var>navigation</var>'s <span>ongoing API method tracker</span> is non-null, then
   <span>reject the finished promise</span> for <var>apiMethodTracker</var> with
   <var>error</var>.</p></li>

   <li>
    <p>If <var>navigation</var>'s <span data-x="concept-Navigation-transition">transition</span> is
    not null, then:</p>

    <ol>
     <li><p>Reject <var>navigation</var>'s <span
     data-x="concept-Navigation-transition">transition</span>'s <span
     data-x="concept-NavigationTransition-finished">finished promise</span> with
     <var>error</var>.</p></li>

     <li><p>Set <var>navigation</var>'s <span
     data-x="concept-Navigation-transition">transition</span> to null.</p></li>
    </ol>
   </li>
  </ol>

  <p>To <dfn>inform the navigation API about aborting navigation</dfn> in a <span>navigable</span>
  <var>navigable</var>:</p>

  <ol>
   <li><p>If this algorithm is running on <var>navigable</var>'s <span data-x="nav-window">active
   window</span>'s <span>relevant agent</span>'s <span data-x="concept-agent-event-loop">event
   loop</span>, then continue on to the following steps. Otherwise, <span>queue a global task</span>
   on the <span>navigation and traversal task source</span> given <var>navigable</var>'s <span
   data-x="nav-window">active window</span> to run the following steps.</p></li>

   <li><p>Let <var>navigation</var> be <var>navigable</var>'s <span data-x="nav-window">active
   window</span>'s <span data-x="window-navigation-api">navigation API</span>.</p></li>

   <li><p>If <var>navigation</var>'s <span>ongoing <code data-x="event-navigate">navigate</code>
   event</span> is null, then return.</p></li>

   <li><p><span>Abort the ongoing navigation</span> given <var>navigation</var>.</p></li>
  </ol>

  <p>To <dfn>inform the navigation API about child navigable destruction</dfn> given a
  <span>navigable</span> <var>navigable</var>:</p>

  <ol>
   <li><p><span>Inform the navigation API about aborting navigation</span> in
   <var>navigable</var>.</p></li>

   <li><p>Let <var>navigation</var> be <var>navigable</var>'s <span data-x="nav-window">active
   window</span>'s <span data-x="window-navigation-api">navigation API</span>.</p></li>

   <li><p>Let <var>traversalAPIMethodTrackers</var> be a <span data-x="list clone">clone</span> of
   <var>navigation</var>'s <span>upcoming traverse API method trackers</span>.</p></li>

   <li><p><span data-x="list iterate">For each</span> <var>apiMethodTracker</var> of
   <var>traversalAPIMethodTrackers</var>: <span>reject the finished promise</span> for
   <var>apiMethodTracker</var> with a new <span>"<code>AbortError</code>"</span>
   <code>DOMException</code> created in <var>navigation</var>'s <span
   data-x="concept-relevant-realm">relevant realm</span>.</p></li>
  </ol>

  <hr>

  <p>The ongoing navigation concept is most-directly exposed to web developers through the <code
  data-x="dom-Navigation-transition">navigation.transition</code> property, which is an instance of
  the <code>NavigationTransition</code> interface:</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>NavigationTransition</dfn> {
  readonly attribute <span>NavigationType</span> <span data-x="dom-NavigationTransition-navigationType">navigationType</span>;
  readonly attribute <span>NavigationHistoryEntry</span> <span data-x="dom-NavigationTransition-from">from</span>;
  readonly attribute <span data-x="idl-Promise">Promise</span>&lt;undefined> <span data-x="dom-NavigationTransition-finished">finished</span>;
};</code></pre>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><span data-x="dom-navigation">navigation</span>.<span subdfn data-x="dom-Navigation-transition">transition</span></code></dt>
   <dd>
    <p>A <code>NavigationTransition</code> representing any ongoing navigation that hasn't yet
    reached the <code data-x="event-navigatesuccess">navigatesuccess</code> or <code
    data-x="event-navigateerror">navigateerror</code> stage, if one exists; or null, if there is no
    such transition ongoing.</p>

    <p>Since <code data-x="dom-Navigation-currentEntry">navigation.currentEntry</code> (and other
    properties like <code data-x="dom-location-href">location.href</code>) are updated immediately
    upon navigation, this <code data-x="dom-Navigation-transition">navigation.transition</code>
    property is useful for determining when such navigations are not yet fully settled, according to
    any handlers passed to <code
    data-x="dom-NavigateEvent-intercept">navigateEvent.intercept()</code>.</p>
   </dd>

   <dt><code data-x=""><span data-x="dom-navigation">navigation</span>.<span data-x="dom-Navigation-transition">transition</span>.<span subdfn data-x="dom-NavigationTransition-navigationType">navigationType</span></code></dt>
   <dd><p>One of "<code data-x="dom-NavigationType-push">push</code>", "<code
   data-x="dom-NavigationType-replace">replace</code>", "<code
   data-x="dom-NavigationType-reload">reload</code>", or "<code
   data-x="dom-NavigationType-traverse">traverse</code>", indicating what type of navigation this
   transition is for.</p></dd>

   <dt><code data-x=""><span data-x="dom-navigation">navigation</span>.<span data-x="dom-Navigation-transition">transition</span>.<span subdfn data-x="dom-NavigationTransition-from">from</span></code></dt>
   <dd><p>The <code>NavigationHistoryEntry</code> from which the transition is coming. This can be
   useful to compare against <code
   data-x="dom-Navigation-currentEntry">navigation.currentEntry</code>.</p></dd>

   <dt><code data-x=""><span data-x="dom-navigation">navigation</span>.<span data-x="dom-Navigation-transition">transition</span>.<span subdfn data-x="dom-NavigationTransition-finished">finished</span></code></dt>
   <dd><p>A promise which fulfills at the same time as the <code
   data-x="event-navigatesuccess">navigatesuccess</code> fires, or rejects at the same time the
   <code data-x="event-navigateerror">navigateerror</code> event fires.</p></dd>
  </dl>

  <div w-nodev>

  <p>Each <code>Navigation</code> has a <dfn
  data-x="concept-Navigation-transition">transition</dfn>, which is a
  <code>NavigationTransition</code> or null, initially null.</p>

  <p>The <dfn attribute for="Navigation"><code
  data-x="dom-Navigation-transition">transition</code></dfn> getter steps are to return
  <span>this</span>'s <span data-x="concept-Navigation-transition">transition</span>.</p>

  <p>Each <code>NavigationTransition</code> has an associated <dfn
  data-x="concept-NavigationTransition-navigationType">navigation type</dfn>, which is a
  <code>NavigationType</code>.</p>

  <p>Each <code>NavigationTransition</code> has an associated <dfn
  data-x="concept-NavigationTransition-from">from entry</dfn>, which is a
  <code>NavigationHistoryEntry</code>.</p>

  <p>Each <code>NavigationTransition</code> has an associated <dfn
  data-x="concept-NavigationTransition-finished">finished promise</dfn>, which is a promise.</p>

  <p>The <dfn attribute for="NavigationTransition"><code
  data-x="dom-NavigationTransition-navigationType">navigationType</code></dfn> getter steps are to
  return <span>this</span>'s <span data-x="concept-NavigationTransition-navigationType">navigation
  type</span>.</p>

  <p>The <dfn attribute for="NavigationTransition"><code
  data-x="dom-NavigationTransition-from">from</code></dfn> getter steps are to return
  <span>this</span>'s <span data-x="concept-NavigationTransition-from">from entry</span>.</p>

  <p>The <dfn attribute for="NavigationTransition"><code
  data-x="dom-NavigationTransition-finished">finished</code></dfn> getter steps are to return
  <span>this</span>'s <span data-x="concept-NavigationTransition-finished">finished
  promise</span>.</p>

  </div>


  <h5 id="navigation-activation-interface">The <code>NavigationActivation</code> interface</h5>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>NavigationActivation</dfn> {
  readonly attribute <span>NavigationHistoryEntry</span>? <span data-x="dom-NavigationActivation-from">from</span>;
  readonly attribute <span>NavigationHistoryEntry</span> <span data-x="dom-NavigationActivation-entry">entry</span>;
  readonly attribute <span>NavigationType</span> <span data-x="dom-NavigationActivation-navigationType">navigationType</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><span data-x="dom-navigation">navigation</span>.<span subdfn data-x="dom-Navigation-activation">activation</span></code></dt>
   <dd>
    <p>A <code>NavigationActivation</code> containing information about the most recent
    cross-document navigation, the navigation that "activated" this <code>Document</code>.</p>

    <p>While <code data-x="dom-Navigation-currentEntry">navigation.currentEntry</code> and the
    <code>Document</code>'s <span data-x="concept-document-url">URL</span> can be updated regularly
    due to same-document navigations, <code
    data-x="dom-Navigation-activation">navigation.activation</code> stays constant, and its properties
    are only updated if the <code>Document</code> is <span
    data-x="reactivate a document">reactivated</span> from history.</p>
   </dd>

   <dt><code data-x=""><span data-x="dom-navigation">navigation</span>.<span data-x="dom-Navigation-activation">activation</span>.<span subdfn data-x="dom-NavigationActivation-entry">entry</span></code></dt>
   <dd><p>A <code>NavigationHistoryEntry</code>, equivalent to the value of the <code
   data-x="dom-Navigation-currentEntry">navigation.currentEntry</code> property at the moment the
   <code>Document</code> was activated.</p></dd>

   <dt><code data-x=""><span data-x="dom-navigation">navigation</span>.<span data-x="dom-Navigation-activation">activation</span>.<span subdfn data-x="dom-NavigationActivation-from">from</span></code></dt>
   <dd>
    <p>A <code>NavigationHistoryEntry</code>, representing the <code>Document</code> that was active
    right before the current <code>Document</code>. This will have a value null in case the
    previous <code>Document</code> was not <span>same origin</span> with this one or if it was the
    <span data-x="is initial about:blank">initial <code>about:blank</code></span>
    <code>Document</code>.</p>

    <p>There are some cases in which either the <code
    data-x="dom-NavigationActivation-from">from</code> or <code
    data-x="dom-NavigationActivation-entry">entry</code> <code>NavigationHistoryEntry</code> objects
    would not be viable targets for the <code data-x="dom-Navigation-traverseTo">traverseTo()</code>
    method, as they might not be retained in history. For example, the <code>Document</code> can be
    activated using <code data-x="dom-location-replace">location.replace()</code> or its initial
    entry could be replaced by <code
    data-x="dom-history-replaceState">history.replaceState()</code>. However, those entries' <code
    data-x="dom-NavigationHistoryEntry-url">url</code> property and <code
    data-x="dom-NavigationHistoryEntry-getState">getState()</code> method are still accessible.</p>
   </dd>

   <dt><code data-x=""><span data-x="dom-navigation">navigation</span>.<span data-x="dom-Navigation-activation">activation</span>.<span subdfn data-x="dom-NavigationActivation-navigationType">navigationType</span></code></dt>
   <dd><p>One of "<code data-x="dom-NavigationType-push">push</code>", "<code
   data-x="dom-NavigationType-replace">replace</code>", "<code
   data-x="dom-NavigationType-reload">reload</code>", or "<code
   data-x="dom-NavigationType-traverse">traverse</code>", indicating what type of navigation
   activated this <code>Document</code>.</p></dd>
  </dl>

  <p>Each <code>Navigation</code> has an associated <dfn
  data-x="navigation-activation">activation</dfn>, which is null or a
  <code>NavigationActivation</code> object, initially null.</p>

  <p>Each <code>NavigationActivation</code> has:</p>

  <ul>
   <li><p><dfn data-x="nav-activation-old-entry">old entry</dfn>, null or a
   <code>NavigationHistoryEntry</code>.</p></li>

   <li><p><dfn data-x="nav-activation-new-entry">new entry</dfn>, null or a
   <code>NavigationHistoryEntry</code>.</p></li>

   <li><p><dfn data-x="nav-activation-navigation-type">navigation type</dfn>, a
   <code>NavigationType</code>.</p></li>
  </ul>

  <p>The <dfn attribute for="Navigation"><code
  data-x="dom-Navigation-activation">activation</code></dfn> getter steps are to return
  <span>this</span>'s <span data-x="navigation-activation">activation</span>.</p>

  <p>The <dfn attribute for="NavigationActivation"><code
  data-x="dom-NavigationActivation-from">from</code></dfn> getter steps are to return
  <span>this</span>'s <span data-x="nav-activation-old-entry">old entry</span>.</p>

  <p>The <dfn attribute for="NavigationActivation"><code
  data-x="dom-NavigationActivation-entry">entry</code></dfn> getter steps are to return
  <span>this</span>'s <span data-x="nav-activation-new-entry">new entry</span>.</p>

  <p>The <dfn attribute for="NavigationActivation"><code
  data-x="dom-NavigationActivation-navigationType">navigationType</code></dfn> getter steps are to
  return <span>this</span>'s <span
  data-x="nav-activation-navigation-type">navigation type</span>.</p>


  <h5>The <code data-x="event-navigate">navigate</code> event</h5>

  <p>A major feature of the navigation API is the <code data-x="event-navigate">navigate</code>
  event. This event is fired on any navigation (in the <span data-x="NavigationType">broad sense of
  the word</span>), allowing web developers to monitor such outgoing navigations. In many cases, the
  event is <code data-x="dom-Event-cancelable">cancelable</code>, which allows preventing the
  navigation from happening. And in others, the navigation can be intercepted and replaced with a
  same-document navigation by using the <code
  data-x="dom-NavigateEvent-intercept">intercept()</code> method of the <code>NavigateEvent</code>
  class.</p>

  <h6>The <code>NavigateEvent</code> interface</h6>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>NavigateEvent</dfn> : <span>Event</span> {
  constructor(DOMString type, <span>NavigateEventInit</span> eventInitDict);

  readonly attribute <span>NavigationType</span> <span data-x="dom-NavigateEvent-navigationType">navigationType</span>;
  readonly attribute <span>NavigationDestination</span> <span data-x="dom-NavigateEvent-destination">destination</span>;
  readonly attribute boolean <span data-x="dom-NavigateEvent-canIntercept">canIntercept</span>;
  readonly attribute boolean <span data-x="dom-NavigateEvent-userInitiated">userInitiated</span>;
  readonly attribute boolean <span data-x="dom-NavigateEvent-hashChange">hashChange</span>;
  readonly attribute <span>AbortSignal</span> <span data-x="dom-NavigateEvent-signal">signal</span>;
  readonly attribute <code>FormData</code>? <span data-x="dom-NavigateEvent-formData">formData</span>;
  readonly attribute DOMString? <span data-x="dom-NavigateEvent-downloadRequest">downloadRequest</span>;
  readonly attribute any <span data-x="dom-NavigateEvent-info">info</span>;
  readonly attribute boolean <span data-x="dom-NavigateEvent-hasUAVisualTransition">hasUAVisualTransition</span>;
  readonly attribute Element? <span data-x="dom-NavigateEvent-sourceElement">sourceElement</span>;

  undefined <span data-x="dom-NavigateEvent-intercept">intercept</span>(optional <span>NavigationInterceptOptions</span> options = {});
  undefined <span data-x="dom-NavigateEvent-scroll">scroll</span>();
};

dictionary <dfn dictionary>NavigateEventInit</dfn> : <span>EventInit</span> {
  <span>NavigationType</span> <dfn dict-member for="NavigateEventInit" data-x="dom-NavigateEventInit-navigationType">navigationType</dfn> = "<span data-x="dom-NavigationType-push">push</span>";
  required <span>NavigationDestination</span> <dfn dict-member for="NavigateEventInit" data-x="dom-NavigateEventInit-destination">destination</dfn>;
  boolean <dfn dict-member for="NavigateEventInit" data-x="dom-NavigateEventInit-canIntercept">canIntercept</dfn> = false;
  boolean <dfn dict-member for="NavigateEventInit" data-x="dom-NavigateEventInit-userInitiated">userInitiated</dfn> = false;
  boolean <dfn dict-member for="NavigateEventInit" data-x="dom-NavigateEventInit-hashChange">hashChange</dfn> = false;
  required <span>AbortSignal</span> <dfn dict-member for="NavigateEventInit" data-x="dom-NavigateEventInit-signal">signal</dfn>;
  <span>FormData</span>? <dfn dict-member for="NavigateEventInit" data-x="dom-NavigateEventInit-formData">formData</dfn> = null;
  DOMString? <dfn dict-member for="NavigateEventInit" data-x="dom-NavigateEventInit-downloadRequest">downloadRequest</dfn> = null;
  any <dfn dict-member for="NavigateEventInit" data-x="dom-NavigateEventInit-info">info</dfn>;
  boolean <dfn dict-member for="NavigateEventInit" data-x="dom-NavigateEventInit-hasUAVisualTransition">hasUAVisualTransition</dfn> = false;
  Element? <dfn dict-member for="NavigateEventInit" data-x="dom-NavigateEventInit-sourceElement">sourceElement</dfn> = null;
};

dictionary <dfn dictionary>NavigationInterceptOptions</dfn> {
  <span>NavigationInterceptHandler</span> <dfn dict-member for="NavigationInterceptOptions" data-x="dom-NavigationInterceptOptions-handler">handler</dfn>;
  <span>NavigationFocusReset</span> <dfn dict-member for="NavigationInterceptOptions" data-x="dom-NavigationInterceptOptions-focusReset">focusReset</dfn>;
  <span>NavigationScrollBehavior</span> <dfn dict-member for="NavigationInterceptOptions" data-x="dom-NavigationInterceptOptions-scroll">scroll</dfn>;
};

enum <dfn enum>NavigationFocusReset</dfn> {
  "<dfn enum-value for="NavigationFocusReset" data-x="dom-NavigationFocusReset-after-transition">after-transition</dfn>",
  "<dfn enum-value for="NavigationFocusReset" data-x="dom-NavigationFocusReset-manual">manual</dfn>"
};

enum <dfn enum>NavigationScrollBehavior</dfn> {
  "<dfn enum-value for="NavigationScrollBehavior" data-x="dom-NavigationScrollBehavior-after-transition">after-transition</dfn>",
  "<dfn enum-value for="NavigationScrollBehavior" data-x="dom-NavigationScrollBehavior-manual">manual</dfn>"
};

callback <dfn callback>NavigationInterceptHandler</dfn> = <span data-x="idl-Promise">Promise</span>&lt;undefined> ();</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-NavigateEvent-navigationType">navigationType</span></code></dt>
   <dd><p>One of "<code data-x="dom-NavigationType-push">push</code>", "<code
   data-x="dom-NavigationType-replace">replace</code>", "<code
   data-x="dom-NavigationType-reload">reload</code>", or "<code
   data-x="dom-NavigationType-traverse">traverse</code>", indicating what type of navigation this
   is.</p></dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-NavigateEvent-destination">destination</span></code></dt>
   <dd><p>A <code>NavigationDestination</code> representing the destination of the
   navigation.</p></dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-NavigateEvent-canIntercept">canIntercept</span></code></dt>
   <dd>
    <p>True if <code data-x="dom-NavigateEvent-intercept">intercept()</code> can be called to
    intercept this navigation and convert it into a same-document navigation, replacing its usual
    behavior; false otherwise.</p>

    <p>Generally speaking, this will be true whenever the current <code>Document</code> <span>can
    have its URL rewritten</span> to the destination URL, except for in the case of cross-document
    "<code data-x="dom-NavigationType-traverse">traverse</code>" navigations, where it will always
    be false.</p>
   </dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-NavigateEvent-userInitiated">userInitiated</span></code></dt>
   <dd><p>True if this navigation was due to a user clicking on an <code>a</code> element,
   submitting a <code>form</code> element, or using the <a href="#nav-traversal-ui">browser UI</a>
   to navigate; false otherwise.</p></dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-NavigateEvent-hashChange">hashChange</span></code></dt>
   <dd><p>True for a <span data-x="navigate-fragid">fragment navigation</span>; false
   otherwise.</p></dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-NavigateEvent-signal">signal</span></code></dt>
   <dd>
    <p>An <code>AbortSignal</code> which will become aborted if the navigation gets canceled, e.g.,
    by the user pressing their browser's "Stop" button, or by another navigation interrupting this
    one.</p>

    <p>The expected pattern is for developers to pass this along to any async operations, such as
    <code>fetch()</code>, which they perform as part of handling this navigation.</p>
   </dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-NavigateEvent-formData">formData</span></code></dt>
   <dd>
    <p>The <code>FormData</code> representing the submitted form entries for this navigation, if
    this navigation is a "<code data-x="dom-NavigationType-push">push</code>" or "<code
    data-x="dom-NavigationType-replace">replace</code>" navigation representing a POST <span
    data-x="concept-form-submit">form submission</span>; null otherwise.</p>

    <p>(Notably, this will be null even for "<code data-x="dom-NavigationType-reload">reload</code>"
    or "<code data-x="dom-NavigationType-traverse">traverse</code>" navigations that are revisiting
    a <span>session history entry</span> that was originally created from a form submission.)</p>
   </dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-NavigateEvent-downloadRequest">downloadRequest</span></code></dt>
   <dd>
    <p>Represents whether or not this navigation was requested to be a download, by using an
    <code>a</code> or <code>area</code> element's <code
    data-x="attr-hyperlink-download">download</code> attribute:</p>

    <ul>
     <li><p>If a download was not requested, then this property is null.</p></li>

     <li><p>If a download was requested, returns the filename that was supplied as the <code
     data-x="attr-hyperlink-download">download</code> attribute's value. (This could be the empty
     string.)</p></li>
    </ul>

    <p>Note that a download being requested does not always mean that a download will happen: for
    example, a download might be blocked by browser security policies, or end up being treated as a
    "<code data-x="NavigationHistoryBehavior-push">push</code>" navigation for <a
    href="https://github.com/whatwg/html/issues/7718" class="XXX">unspecified reasons</a>.</p>

    <p>Similarly, a navigation might end up being a download even if it was not requested to be one,
    due to the destination server responding with a `<code
    data-x="http-content-disposition">Content-Disposition: attachment</code>` header.</p>

    <p>Finally, note that the <code data-x="event-navigate">navigate</code> event will not fire at
    all for downloads initiated using browser UI affordances, e.g., those created by right-clicking
    and choosing to save the target of a link.</p>
   </dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-NavigateEvent-info">info</span></code></dt>
   <dd><p>An arbitrary JavaScript value passed via one of the <a
   href="#navigation-api-initiating-navigations">navigation API methods</a> which initiated this
   navigation, or undefined if the navigation was initiated by the user or by a different
   API.</p></dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-NavigateEvent-hasUAVisualTransition">hasUAVisualTransition</span></code></dt>
   <dd><p>Returns true if the user agent performed a visual transition for this navigation before
   dispatching this event. If true, the best user experience will be given if the author
   synchronously updates the DOM to the post-navigation state.</p></dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-NavigateEvent-sourceElement">sourceElement</span></code></dt>
   <dd><p>Returns the <code>Element</code> responsible for this navigation. This can be an
   <code>a</code>or <code>area</code> element, a <span
   data-x="concept-submit-button">submit button</span>, or a submitted <code>form</code>
   element.</p></dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-NavigateEvent-intercept">intercept</span>({ <span subdfn data-x="dom-NavigationInterceptOptions-handler">handler</span>, <span subdfn data-x="dom-NavigationInterceptOptions-focusReset">focusReset</span>, <span subdfn data-x="dom-NavigationInterceptOptions-scroll">scroll</span> })</code></dt>
   <dd>
    <p>Intercepts this navigation, preventing its normal handling and instead converting it into a
    same-document navigation of the same type to the destination URL.</p>

    <p>The <code data-x="dom-NavigationInterceptOptions-handler">handler</code> option can be a
    function that returns a promise. The handler function will run after the <code
    data-x="event-navigate">navigate</code> event has finished firing, and the <code
    data-x="dom-Navigation-currentEntry">navigation.currentEntry</code> property has been
    synchronously updated. This returned promise is used to signal the duration, and success or
    failure, of the navigation. After it settles, the browser signals to the user (e.g., via a
    loading spinner UI, or assistive technology) that the navigation is finished. Additionally, it
    fires <code data-x="event-navigatesuccess">navigatesuccess</code> or <code
    data-x="event-navigateerror">navigateerror</code> events as appropriate, which other parts of
    the web application can respond to.</p>

    <p>By default, using this method will cause focus to reset when any handlers' returned promises
    settle. Focus will be reset to the first element with the <code
    data-x="attr-fe-autofocus">autofocus</code> attribute set, or <span>the body element</span> if
    the attribute isn't present. The <code
    data-x="dom-NavigationInterceptOptions-focusReset">focusReset</code> option can be set to "<code
    data-x="dom-NavigationFocusReset-manual">manual</code>" to avoid this behavior.</p>

    <p>By default, using this method will delay the browser's scroll restoration logic for "<code
    data-x="dom-NavigationType-traverse">traverse</code>" or "<code
    data-x="dom-NavigationType-reload">reload</code>" navigations, or its
    scroll-reset/scroll-to-a-fragment logic for "<code data-x="dom-NavigationType-push">push</code>"
    or "<code data-x="dom-NavigationType-replace">replace</code>" navigations, until any handlers'
    returned promises settle. The <code data-x="dom-NavigationInterceptOptions-scroll">scroll</code>
    option can be set to "<code data-x="dom-NavigationScrollBehavior-manual">manual</code>" to turn
    off any browser-driven scroll behavior entirely for this navigation, or <code
    data-x="dom-NavigateEvent-scroll">scroll()</code> can be called before the promise settles to
    trigger this behavior early.</p>

    <p>This method will throw a <span>"<code>SecurityError</code>"</span> <code>DOMException</code>
    if <code data-x="dom-NavigateEvent-canIntercept">canIntercept</code> is false, or if <code
    data-x="dom-Event-isTrusted">isTrusted</code> is false. It will throw an
    <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> if not called
    synchronously, during event dispatch.</p>
   </dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-NavigateEvent-scroll">scroll</span>()</code></dt>
   <dd>
    <p>For "<code data-x="dom-NavigationType-traverse">traverse</code>" or "<code
    data-x="dom-NavigationType-reload">reload</code>" navigations, restores the scroll position
    using the browser's usual scroll restoration logic.</p>

    <p>For "<code data-x="dom-NavigationType-push">push</code>" or "<code
    data-x="dom-NavigationType-replace">replace</code>" navigations, either resets the scroll
    position to the top of the document or scrolls to the <span
    data-x="concept-url-fragment">fragment</span> specified by <code
    data-x="dom-NavigationDestination-url">destination.url</code> if there is one.</p>

    <p>If called more than once, or called after automatic post-transition scroll processing has
    happened due to the <code data-x="dom-NavigationInterceptOptions-scroll">scroll</code> option
    being left as "<code
    data-x="dom-NavigationScrollBehavior-after-transition">after-transition</code>", or called
    before the navigation has committed, this method will throw an
    <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>Each <code>NavigateEvent</code> has an <dfn
  data-x="concept-NavigateEvent-interception-state">interception state</dfn>, which is either "<code
  data-x="">none</code>", "<code data-x="">intercepted</code>", "<code data-x="">committed</code>",
  "<code data-x="">scrolled</code>", or "<code data-x="">finished</code>", initially "<code
  data-x="">none</code>".</p>

  <p>Each <code>NavigateEvent</code> has a <dfn
  data-x="concept-NavigateEvent-navigation-handler-list">navigation handler list</dfn>, a
  <span>list</span> of <code>NavigationInterceptHandler</code> callbacks, initially empty.</p>

  <p>Each <code>NavigateEvent</code> has a <dfn data-x="concept-NavigateEvent-focusReset">focus
  reset behavior</dfn>, a <code>NavigationFocusReset</code>-or-null, initially null.</p>

  <p>Each <code>NavigateEvent</code> has a <dfn data-x="concept-NavigateEvent-scroll">scroll
  behavior</dfn>, a <code>NavigationScrollBehavior</code>-or-null, initially null.</p>

  <p>Each <code>NavigateEvent</code> has an <dfn
  data-x="concept-NavigateEvent-abort-controller">abort controller</dfn>, an
  <code>AbortController</code>-or-null, initially null.</p>

  <p>Each <code>NavigateEvent</code> has a <dfn
  data-x="concept-NavigateEvent-classic-history-API-state">classic history API state</dfn>, a
  <span>serialized state</span> or null. It is only used in some cases where the event's <code
  data-x="dom-NavigateEvent-navigationType">navigationType</code> is "<code
  data-x="dom-NavigationType-push">push</code>" or "<code
  data-x="dom-NavigationType-replace">replace</code>", and is set appropriately when the event is
  <span data-x="concept-event-fire">fired</span>.</p>

  <p>The <dfn attribute for="NavigateEvent"><code
  data-x="dom-NavigateEvent-navigationType">navigationType</code></dfn>, <dfn attribute
  for="NavigateEvent"><code data-x="dom-NavigateEvent-destination">destination</code></dfn>, <dfn
  attribute for="NavigateEvent"><code
  data-x="dom-NavigateEvent-canIntercept">canIntercept</code></dfn>, <dfn attribute
  for="NavigateEvent"><code data-x="dom-NavigateEvent-userInitiated">userInitiated</code></dfn>,
  <dfn attribute for="NavigateEvent"><code
  data-x="dom-NavigateEvent-hashChange">hashChange</code></dfn>, <dfn attribute
  for="NavigateEvent"><code data-x="dom-NavigateEvent-signal">signal</code></dfn>, <dfn attribute
  for="NavigateEvent"><code data-x="dom-NavigateEvent-formData">formData</code></dfn>, <dfn
  attribute for="NavigateEvent"><code
  data-x="dom-NavigateEvent-downloadRequest">downloadRequest</code></dfn>, <dfn attribute
  for="NavigateEvent"><code data-x="dom-NavigateEvent-info">info</code></dfn>, <dfn attribute
  for="NavigateEvent"><code
  data-x="dom-NavigateEvent-hasUAVisualTransition">hasUAVisualTransition</code></dfn>, and
  <dfn attribute for="NavigateEvent"><code
  data-x="dom-NavigateEvent-sourceElement">sourceElement</code></dfn> attributes
  must return the values they are initialized to.</p>

  <p>The <dfn method for="NavigateEvent"><code
  data-x="dom-NavigateEvent-intercept">intercept(<var>options</var>)</code></dfn> method steps
  are:</p>

  <ol>
   <li><p><span data-x="NavigateEvent-perform-shared-checks">Perform shared checks</span> given
   <span>this</span>.</p></li>

   <li><p>If <span>this</span>'s <code data-x="dom-NavigateEvent-canIntercept">canIntercept</code>
   attribute was initialized to false, then throw a <span>"<code>SecurityError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>If <span>this</span>'s <span>dispatch flag</span> is unset, then throw an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p><span>Assert</span>: <span>this</span>'s <span
   data-x="concept-NavigateEvent-interception-state">interception state</span> is either "<code
   data-x="">none</code>" or "<code data-x="">intercepted</code>".</p></li>

   <li><p>Set <span>this</span>'s <span
   data-x="concept-NavigateEvent-interception-state">interception state</span> to "<code
   data-x="">intercepted</code>".</p></li>

   <li><p>If <var>options</var>["<code
   data-x="dom-NavigationInterceptOptions-handler">handler</code>"] <span data-x="map
   exists">exists</span>, then <span data-x="list append">append</span> it to <span>this</span>'s
   <span data-x="concept-NavigateEvent-navigation-handler-list">navigation handler
   list</span>.</p></li>

   <li>
    <p>If <var>options</var>["<code
    data-x="dom-NavigationInterceptOptions-focusReset">focusReset</code>"] <span data-x="map
    exists">exists</span>, then:</p>

    <ol>
     <li><p>If <span>this</span>'s <span data-x="concept-NavigateEvent-focusReset">focus reset
     behavior</span> is not null, and it is not equal to <var>options</var>["<code
     data-x="dom-NavigationInterceptOptions-focusReset">focusReset</code>"], then the user agent may
     <span>report a warning to the console</span> indicating that the <code
     data-x="dom-NavigationInterceptOptions-focusReset">focusReset</code> option for a previous call
     to <code data-x="dom-NavigateEvent-intercept">intercept()</code> was overridden by this new
     value, and the previous value will be ignored.</p></li>

     <li><p>Set <span>this</span>'s <span data-x="concept-NavigateEvent-focusReset">focus reset
     behavior</span> to <var>options</var>["<code
     data-x="dom-NavigationInterceptOptions-focusReset">focusReset</code>"].</p></li>
    </ol>
   </li>

   <li>
    <p>If <var>options</var>["<code data-x="dom-NavigationInterceptOptions-scroll">scroll</code>"]
    <span data-x="map exists">exists</span>, then:</p>

    <ol>
     <li><p>If <span>this</span>'s <span data-x="concept-NavigateEvent-scroll">scroll
     behavior</span> is not null, and it is not equal to <var>options</var>["<code
     data-x="dom-NavigationInterceptOptions-scroll">scroll</code>"], then the user agent may
     <span>report a warning to the console</span> indicating that the <code
     data-x="dom-NavigationInterceptOptions-scroll">scroll</code> option for a previous call to
     <code data-x="dom-NavigateEvent-intercept">intercept()</code> was overridden by this new value,
     and the previous value will be ignored.</p></li>

     <li><p>Set <span>this</span>'s <span data-x="concept-NavigateEvent-scroll">scroll
     behavior</span> to <var>options</var>["<code
     data-x="dom-NavigationInterceptOptions-scroll">scroll</code>"].</p></li>
    </ol>
   </li>
  </ol>

  <p>The <dfn method for="NavigateEvent"><code
  data-x="dom-NavigateEvent-scroll">scroll()</code></dfn> method steps are:</p>

  <ol>
   <li><p><span data-x="NavigateEvent-perform-shared-checks">Perform shared checks</span> given
   <span>this</span>.</p></li>

   <li><p>If <span>this</span>'s <span
   data-x="concept-NavigateEvent-interception-state">interception state</span> is not "<code
   data-x="">committed</code>", then throw an <span>"<code>InvalidStateError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p><span>Process scroll behavior</span> given <span>this</span>.</p></li>
  </ol>

  <p>To <dfn data-x="NavigateEvent-perform-shared-checks">perform shared checks</dfn> for a
  <code>NavigateEvent</code> <var>event</var>:</p>

  <ol>
   <li><p>If <var>event</var>'s <span>relevant global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span> is not <span>fully
   active</span>, then throw an <span>"<code>InvalidStateError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>If <var>event</var>'s <code data-x="dom-Event-isTrusted">isTrusted</code> attribute was
   initialized to false, then throw a <span>"<code>SecurityError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>If <var>event</var>'s <span>canceled flag</span> is set, then throw an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>
  </ol>

  </div>

  <h6>The <code>NavigationDestination</code> interface</h6>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>NavigationDestination</dfn> {
  readonly attribute USVString <span data-x="dom-NavigationDestination-url">url</span>;
  readonly attribute DOMString <span data-x="dom-NavigationDestination-key">key</span>;
  readonly attribute DOMString <span data-x="dom-NavigationDestination-id">id</span>;
  readonly attribute long long <span data-x="dom-NavigationDestination-index">index</span>;
  readonly attribute boolean <span data-x="dom-NavigationDestination-sameDocument">sameDocument</span>;

  any <span data-x="dom-NavigationDestination-getState">getState</span>();
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>event</var>.<span data-x="dom-NavigateEvent-destination">destination</span>.<span data-x="dom-NavigationDestination-url">url</span></code></dt>
   <dd><p>The URL being navigated to.</p></dd>

   <dt><code data-x=""><var>event</var>.<span data-x="dom-NavigateEvent-destination">destination</span>.<span data-x="dom-NavigationDestination-key">key</span></code></dt>
   <dd><p>The value of the <code data-x="dom-NavigationHistoryEntry-key">key</code> property of the
   destination <code>NavigationHistoryEntry</code>, if this is a "<code
   data-x="dom-NavigationType-traverse">traverse</code>" navigation, or the empty string otherwise. </p></dd>

   <dt><code data-x=""><var>event</var>.<span data-x="dom-NavigateEvent-destination">destination</span>.<span data-x="dom-NavigationDestination-id">id</span></code></dt>
   <dd><p>The value of the <code data-x="dom-NavigationHistoryEntry-id">id</code> property of the
   destination <code>NavigationHistoryEntry</code>, if this is a "<code
   data-x="dom-NavigationType-traverse">traverse</code>" navigation, or the empty string otherwise. </p></dd>

   <dt><code data-x=""><var>event</var>.<span data-x="dom-NavigateEvent-destination">destination</span>.<span data-x="dom-NavigationDestination-index">index</span></code></dt>
   <dd><p>The value of the <code data-x="dom-NavigationHistoryEntry-index">index</code> property of
   the destination <code>NavigationHistoryEntry</code>, if this is a "<code
   data-x="dom-NavigationType-traverse">traverse</code>" navigation, or &minus;1 otherwise.
   </p></dd>

   <dt><code data-x=""><var>event</var>.<span data-x="dom-NavigateEvent-destination">destination</span>.<span data-x="dom-NavigationDestination-sameDocument">sameDocument</span></code></dt>
   <dd>
    <p>Indicates whether or not this navigation is to the same <code>Document</code> as the current
    one, or not. This will be true, for example, in the case of fragment navigations or <code
    data-x="dom-history-pushState">history.pushState()</code> navigations.</p>

    <p>Note that this property indicates the original nature of the navigation. If a cross-document
    navigation is converted into a same-document navigation using <code
    data-x="dom-NavigateEvent-intercept">navigateEvent.intercept()</code>, that will not change the
    value of this property.</p>
   </dd>

   <dt><code data-x=""><var>event</var>.<span data-x="dom-NavigateEvent-destination">destination</span>.<span data-x="dom-NavigationDestination-getState">getState</span>()</code></dt>
   <dd>
    <p>For "<code data-x="dom-NavigationType-traverse">traverse</code>" navigations, returns the
    <span data-x="StructuredDeserialize">deserialization</span> of the state stored in the
    destination <span>session history entry</span>.</p>

    <p>For "<code data-x="dom-NavigationType-push">push</code>" or "<code
    data-x="dom-NavigationType-replace">replace</code>" navigations, returns the <span
    data-x="StructuredDeserialize">deserialization</span> of the state passed to <code
    data-x="dom-Navigation-navigate">navigation.navigate()</code>, if the navigation was initiated
    by that method, or undefined it if it wasn't.</p>

    <p>For "<code data-x="dom-NavigationType-reload">reload</code>" navigations, returns the <span
    data-x="StructuredDeserialize">deserialization</span> of the state passed to <code
    data-x="dom-Navigation-reload">navigation.reload()</code>, if the reload was initiated by that
    method, or undefined it if it wasn't.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>Each <code>NavigationDestination</code> has a <dfn
  data-x="concept-NavigationDestination-url">URL</dfn>, which is a <span>URL</span>.</p>

  <p>Each <code>NavigationDestination</code> has an <dfn
  data-x="concept-NavigationDestination-entry">entry</dfn>, which is a
  <code>NavigationHistoryEntry</code> or null.</p>

  <p class="note">It will be non-null if and only if the <code>NavigationDestination</code>
  corresponds to a "<code data-x="dom-NavigationType-traverse">traverse</code>" navigation.</p>

  <p>Each <code>NavigationDestination</code> has a <dfn
  data-x="concept-NavigationDestination-state">state</dfn>, which is a <span>serialized
  state</span>.</p>

  <p>Each <code>NavigationDestination</code> has an <dfn
  data-x="concept-NavigationDestination-sameDocument">is same document</dfn>, which is a
  boolean.</p>

  <hr>

  <p>The <dfn attribute for="NavigationDestination"><code
  data-x="dom-NavigationDestination-url">url</code></dfn> getter steps are to return
  <span>this</span>'s <span data-x="concept-NavigationDestination-url">URL</span>, <span
  data-x="concept-url-serializer">serialized</span>.</p>

  <p>The <dfn attribute for="NavigationDestination"><code
  data-x="dom-NavigationDestination-key">key</code></dfn> getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="concept-NavigationDestination-entry">entry</span> is
   null, then return the empty string.</p></li>

   <li><p>Return <span>this</span>'s <span
   data-x="concept-NavigationDestination-entry">entry</span>'s <span
   data-x="concept-NavigationHistoryEntry-key">key</span>.</p></li>
  </ol>

  <p>The <dfn attribute for="NavigationDestination"><code
  data-x="dom-NavigationDestination-id">id</code></dfn> getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="concept-NavigationDestination-entry">entry</span> is
   null, then return the empty string.</p></li>

   <li><p>Return <span>this</span>'s <span
   data-x="concept-NavigationDestination-entry">entry</span>'s <span
   data-x="concept-NavigationHistoryEntry-id">ID</span>.</p></li>
  </ol>

  <p>The <dfn attribute for="NavigationDestination"><code
  data-x="dom-NavigationDestination-index">index</code></dfn> getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="concept-NavigationDestination-entry">entry</span> is
   null, then return &minus;1.</p></li>

   <li><p>Return <span>this</span>'s <span
   data-x="concept-NavigationDestination-entry">entry</span>'s <span
   data-x="concept-NavigationHistoryEntry-index">index</span>.</p></li>
  </ol>

  <p>The <dfn attribute for="NavigationDestination"><code
  data-x="dom-NavigationDestination-sameDocument">sameDocument</code></dfn> getter steps are to
  return <span>this</span>'s <span data-x="concept-NavigationDestination-sameDocument">is same
  document</span>.</p>

  <p>The <dfn method for="NavigationDestination"><code
  data-x="dom-NavigationDestination-getState">getState()</code></dfn> method steps are to return
  <span>StructuredDeserialize</span>(<span>this</span>'s <span
  data-x="concept-NavigationDestination-state">state</span>).</p>


  <h6 id="navigate-event-firing">Firing the event</h6>

  <p>Other parts of the standard fire the <code data-x="event-navigate">navigate</code> event,
  through a series of wrapper algorithms given in this section.</p>

  <p>To <dfn>fire a traverse <code data-x="event-navigate">navigate</code> event</dfn> at a
  <code>Navigation</code> <var>navigation</var> given a <span>session history entry</span> <dfn
  data-x="fire-navigate-traverse-destinationSHE"><var>destinationSHE</var></dfn> and an optional
  <span>user navigation involvement</span> <dfn
  data-x="fire-navigate-traverse-userInvolvement"><var>userInvolvement</var></dfn> (default "<code
  data-x="uni-none">none</code>"):</p>

  <ol>
   <li><p>Let <var>event</var> be the result of <span>creating an event</span> given
   <code>NavigateEvent</code>, in <var>navigation</var>'s <span
   data-x="concept-relevant-realm">relevant realm</span>.</p></li>

   <li><p>Set <var>event</var>'s <span
   data-x="concept-NavigateEvent-classic-history-API-state">classic history API state</span> to
   null.</p></li>

   <li><p>Let <var>destination</var> be a <span>new</span> <code>NavigationDestination</code>
   created in <var>navigation</var>'s <span data-x="concept-relevant-realm">relevant
   realm</span>.</p></li>

   <li><p>Set <var>destination</var>'s <span data-x="concept-NavigationDestination-url">URL</span>
   to <var>destinationSHE</var>'s <span data-x="she-url">URL</span>.</p></li>

   <li><p>Let <var>destinationNHE</var> be the <code>NavigationHistoryEntry</code> in
   <var>navigation</var>'s <span data-x="navigation-entry-list">entry list</span> whose <span
   data-x="nhe-she">session history entry</span> is <var>destinationSHE</var>, or null if no such
   <code>NavigationHistoryEntry</code> exists.</p></li>

   <li>
    <p>If <var>destinationNHE</var> is non-null, then:</p>

    <ol>
     <li><p>Set <var>destination</var>'s <span
     data-x="concept-NavigationDestination-entry">entry</span> to
     <var>destinationNHE</var>.</p></li>

     <li><p>Set <var>destination</var>'s <span
     data-x="concept-NavigationDestination-state">state</span> to <var>destinationSHE</var>'s <span
     data-x="she-navigation-api-state">navigation API state</span>.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise,</p>

    <ol>
     <li><p>Set <var>destination</var>'s <span
     data-x="concept-NavigationDestination-entry">entry</span> to null.</p></li>

     <li><p>Set <var>destination</var>'s <span
     data-x="concept-NavigationDestination-state">state</span> to
     <span>StructuredSerializeForStorage</span>(null).</p></li>
    </ol>
   </li>

   <li><p>Set <var>destination</var>'s <span data-x="concept-NavigationDestination-sameDocument">is
   same document</span> to true if <var>destinationSHE</var>'s <span
   data-x="she-document">document</span> is equal to <var>navigation</var>'s <span>relevant global
   object</span>'s <span data-x="concept-document-window">associated <code>Document</code></span>;
   otherwise false.</p></li>

   <li><p>Return the result of performing the <span>inner <code
   data-x="event-navigate">navigate</code> event firing algorithm</span> given
   <var>navigation</var>, "<code data-x="dom-NavigationType-traverse">traverse</code>",
   <var>event</var>, <var>destination</var>, <var>userInvolvement</var>, null, null, and
   null.</p></li>
  </ol>

  <p>To <dfn>fire a push/replace/reload <code data-x="event-navigate">navigate</code> event</dfn> at
  a <code>Navigation</code> <var>navigation</var> given a <code>NavigationType</code> <dfn
  data-x="fire-navigate-prr-navigationType"><var>navigationType</var></dfn>, a <span>URL</span> <dfn
  data-x="fire-navigate-prr-destinationURL"><var>destinationURL</var></dfn>, a boolean <dfn
  data-x="fire-navigate-prr-isSameDocument"><var>isSameDocument</var></dfn>, an optional <span>user
  navigation involvement</span> <dfn
  data-x="fire-navigate-prr-userInvolvement"><var>userInvolvement</var></dfn> (default "<code
  data-x="uni-none">none</code>"), an optional <code>Element</code>-or-null <dfn
  data-x="fire-navigate-prr-sourceElement"><var>sourceElement</var></dfn> (default null), an
  optional <span>entry list</span>-or-null <dfn
  data-x="fire-navigate-prr-formDataEntryList"><var>formDataEntryList</var></dfn> (default null), an
  optional <span>serialized state</span> <dfn
  data-x="fire-navigate-prr-navigationAPIState"><var>navigationAPIState</var></dfn> (default
  <span>StructuredSerializeForStorage</span>(null)), and an optional <span>serialized
  state</span>-or-null <dfn
  data-x="fire-navigate-prr-classicHistoryAPIState"><var>classicHistoryAPIState</var></dfn> (default
  null):</p>

  <ol>
   <li><p>Let <var>event</var> be the result of <span>creating an event</span> given
   <code>NavigateEvent</code>, in <var>navigation</var>'s <span
   data-x="concept-relevant-realm">relevant realm</span>.</p></li>

   <li><p>Set <var>event</var>'s <span
   data-x="concept-NavigateEvent-classic-history-API-state">classic history API state</span> to
   <var>classicHistoryAPIState</var>.</p></li>

   <li><p>Let <var>destination</var> be a <span>new</span> <code>NavigationDestination</code>
   created in <var>navigation</var>'s <span data-x="concept-relevant-realm">relevant
   realm</span>.</p></li>

   <li><p>Set <var>destination</var>'s <span data-x="concept-NavigationDestination-url">URL</span>
   to <var>destinationURL</var>.</p></li>

   <li><p>Set <var>destination</var>'s <span
   data-x="concept-NavigationDestination-entry">entry</span> to null.</p></li>

   <li><p>Set <var>destination</var>'s <span
   data-x="concept-NavigationDestination-state">state</span> to
   <var>navigationAPIState</var>.</p></li>

   <li><p>Set <var>destination</var>'s <span data-x="concept-NavigationDestination-sameDocument">is
   same document</span> to <var>isSameDocument</var>.</p></li>

   <li><p>Return the result of performing the <span>inner <code
   data-x="event-navigate">navigate</code> event firing algorithm</span> given
   <var>navigation</var>, <var>navigationType</var>, <var>event</var>, <var>destination</var>,
   <var>userInvolvement</var>, <var>sourceElement</var>, <var>formDataEntryList</var>, and
   null.</p></li>
  </ol>

  <p>To <dfn>fire a download request <code data-x="event-navigate">navigate</code> event</dfn> at a
  <code>Navigation</code> <var>navigation</var> given a <span>URL</span> <dfn
  data-x="fire-navigate-download-destinationURL"><var>destinationURL</var></dfn>, a <span>user
  navigation involvement</span> <dfn
  data-x="fire-navigate-download-userInvolvement"><var>userInvolvement</var></dfn>, an
  <code>Element</code>-or-null <dfn
  data-x="fire-navigate-download-sourceElement"><var>sourceElement</var></dfn>, and a string
  <dfn data-x="fire-navigate-download-filename"><var>filename</var></dfn>:</p>

  <ol>
   <li><p>Let <var>event</var> be the result of <span>creating an event</span> given
   <code>NavigateEvent</code>, in <var>navigation</var>'s <span
   data-x="concept-relevant-realm">relevant realm</span>.</p></li>

   <li><p>Set <var>event</var>'s <span
   data-x="concept-NavigateEvent-classic-history-API-state">classic history API state</span> to
   null.</p></li>

   <li><p>Let <var>destination</var> be a <span>new</span> <code>NavigationDestination</code>
   created in <var>navigation</var>'s <span data-x="concept-relevant-realm">relevant
   realm</span>.</p></li>

   <li><p>Set <var>destination</var>'s <span data-x="concept-NavigationDestination-url">URL</span>
   to <var>destinationURL</var>.</p></li>

   <li><p>Set <var>destination</var>'s <span
   data-x="concept-NavigationDestination-entry">entry</span> to null.</p></li>

   <li><p>Set <var>destination</var>'s <span
   data-x="concept-NavigationDestination-state">state</span> to
   <span>StructuredSerializeForStorage</span>(null).</p></li>

   <li><p>Set <var>destination</var>'s <span data-x="concept-NavigationDestination-sameDocument">is
   same document</span> to false.</p></li>

   <li><p>Return the result of performing the <span>inner <code
   data-x="event-navigate">navigate</code> event firing algorithm</span> given
   <var>navigation</var>, "<code data-x="dom-NavigationType-push">push</code>", <var>event</var>,
   <var>destination</var>, <var>userInvolvement</var>, <var>sourceElement</var>, null, and
   <var>filename</var>.</p></li>
  </ol>

  <p>The <dfn>inner <code data-x="event-navigate">navigate</code> event firing algorithm</dfn>
  consists of the following steps, given a <code>Navigation</code> <var>navigation</var>, a
  <code>NavigationType</code> <var>navigationType</var>, a <code>NavigateEvent</code>
  <var>event</var>, a <code>NavigationDestination</code> <var>destination</var>, a <span>user
  navigation involvement</span> <var>userInvolvement</var>, an <code>Element</code>-or-null
  <var>sourceElement</var>, an <span>entry list</span>-or-null <var>formDataEntryList</var>, and a
  string-or-null <var>downloadRequestFilename</var>:</p>

  <ol>
   <li>
    <p>If <var>navigation</var> <span>has entries and events disabled</span>, then:</p>

    <ol>
     <li><p><span>Assert</span>: <var>navigation</var>'s <span>ongoing API method tracker</span> is
     null.</p></li>

     <li><p><span>Assert</span>: <var>navigation</var>'s <span>upcoming non-traverse API method
     tracker</span> is null.</p></li>

     <li><p><span>Assert</span>: <var>navigation</var>'s <span>upcoming traverse API method
     trackers</span> <span data-x="list is empty">is empty</span>.</p></li>

     <li><p>Return true.</p></li>
    </ol>

    <p class="note">These assertions holds because <code
    data-x="dom-Navigation-traverseTo">traverseTo()</code>, <code
    data-x="dom-Navigation-back">back()</code>, and <code
    data-x="dom-Navigation-forward">forward()</code> will immediately fail when entries and events
    are disabled (since there are no entries to traverse to), and if our starting point is instead
    <code data-x="dom-Navigation-navigate">navigate()</code> or <code
    data-x="dom-Navigation-reload">reload()</code>, then we <a
    href="#dont-always-set-upcoming-non-traverse-api-method-tracker">avoided</a> setting the
    <span>upcoming non-traverse API method tracker</span> in the first place.</p>
   </li>

   <li><p>Let <var>destinationKey</var> be null.</p></li>

   <li><p>If <var>destination</var>'s <span
   data-x="concept-NavigationDestination-entry">entry</span> is non-null, then set
   <var>destinationKey</var> to <var>destination</var>'s <span
   data-x="concept-NavigationDestination-entry">entry</span>'s <span
   data-x="concept-NavigationHistoryEntry-key">key</span>.</p></li>

   <li><p><span>Assert</span>: <var>destinationKey</var> is not the empty string.</p></li>

   <li><p><span>Promote an upcoming API method tracker to ongoing</span> given
   <var>navigation</var> and <var>destinationKey</var>.</p></li>

   <li><p>Let <var>apiMethodTracker</var> be <var>navigation</var>'s <span>ongoing API method
   tracker</span>.</p></li>

   <li><p>Let <var>navigable</var> be <var>navigation</var>'s <span>relevant global object</span>'s
   <span data-x="window navigable">navigable</span>.</p></li>

   <li><p>Let <var>document</var> be <var>navigation</var>'s <span>relevant global object</span>'s
   <span data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>If <var>document</var> <span>can have its URL rewritten</span> to <var>destination</var>'s
   <span data-x="concept-NavigationDestination-url">URL</span>, and either <var>destination</var>'s
   <span data-x="concept-NavigationDestination-sameDocument">is same document</span> is true or
   <var>navigationType</var> is not "<code data-x="dom-NavigationType-traverse">traverse</code>",
   then initialize <var>event</var>'s <code
   data-x="dom-NavigateEvent-canIntercept">canIntercept</code> to true. Otherwise, initialize it to
   false.</p></li>

   <li id="navigate-event-traverse-can-be-canceled">
    <p>Let <var>traverseCanBeCanceled</var> be true if all of the following are true:</p>

    <ul>
     <li><p><var>navigable</var> is a <span>top-level traversable</span>;</p></li>

     <li><p><var>destination</var>'s <span data-x="concept-NavigationDestination-sameDocument">is
     same document</span> is true; and</p></li>

     <li><p>either <var>userInvolvement</var> is not "<code data-x="uni-browser-ui">browser
     UI</code>", or <var>navigation</var>'s <span>relevant global object</span> has
     <span>history-action activation</span>.</p></li>
    </ul>

    <p>Otherwise, let it be false.</p>
   </li>

   <li>
    <p>If either:</p>

    <ul>
     <li><p><var>navigationType</var> is not "<code
     data-x="dom-NavigationType-traverse">traverse</code>"; or</p></li>

     <li><p><var>traverseCanBeCanceled</var> is true,</p></li>
    </ul>

    <p>then initialize <var>event</var>'s <code data-x="dom-Event-cancelable">cancelable</code> to
    true. Otherwise, initialize it to false.</p>
   </li>

   <li><p>Initialize <var>event</var>'s <code data-x="dom-Event-type">type</code> to "<code
   data-x="event-navigate">navigate</code>".</p></li>

   <li><p>Initialize <var>event</var>'s <code
   data-x="dom-NavigateEvent-navigationType">navigationType</code> to
   <var>navigationType</var>.</p></li>

   <li><p>Initialize <var>event</var>'s <code
   data-x="dom-NavigateEvent-destination">destination</code> to <var>destination</var>.</p></li>

   <li><p>Initialize <var>event</var>'s <code
   data-x="dom-NavigateEvent-downloadRequest">downloadRequest</code> to
   <var>downloadRequestFilename</var>.</p></li>

   <li>
    <p>If <var>apiMethodTracker</var> is not null, then initialize <var>event</var>'s <code
    data-x="dom-NavigateEvent-info">info</code> to <var>apiMethodTracker</var>'s <span
    data-x="navigation-api-method-tracker-info">info</span>. Otherwise, initialize it to
    undefined.</p>

    <p class="note">At this point <var>apiMethodTracker</var>'s <span
    data-x="navigation-api-method-tracker-info">info</span> is no longer needed and can be nulled
    out instead of keeping it alive for the lifetime of the <span>navigation API method
    tracker</span>.</p>
   </li>

   <li><p>Initialize <var>event</var>'s <code
   data-x="dom-NavigateEvent-hasUAVisualTransition">hasUAVisualTransition</code> to true if a visual
   transition, to display a cached rendered state of the <var>document</var>'s <span>latest
   entry</span>, was done by the user agent. Otherwise, initialize it to false.</p></li>

   <li><p>Initialize <var>event</var>'s <code
   data-x="dom-NavigateEvent-sourceElement">sourceElement</code> to
   <var>sourceElement</var>.</p></li>

   <li><p>Set <var>event</var>'s <span data-x="concept-NavigateEvent-abort-controller">abort
   controller</span> to a <span>new</span> <code>AbortController</code> created in
   <var>navigation</var>'s <span data-x="concept-relevant-realm">relevant realm</span>.</p></li>

   <li><p>Initialize <var>event</var>'s <code data-x="dom-NavigateEvent-signal">signal</code> to
   <var>event</var>'s <span data-x="concept-NavigateEvent-abort-controller">abort
   controller</span>'s <span data-x="concept-AbortController-signal">signal</span>.</p></li>

   <li><p>Let <var>currentURL</var> be <var>document</var>'s <span
   data-x="concept-document-url">URL</span>.</p></li>

   <li>
    <p>If all of the following are true:</p>

    <ul>
     <li><p><var>event</var>'s <span
     data-x="concept-NavigateEvent-classic-history-API-state">classic history API state</span> is
     null;</p></li>

     <li><p><var>destination</var>'s <span data-x="concept-NavigationDestination-sameDocument">is
     same document</span> is true;</p></li>

     <li><p><var>destination</var>'s <span data-x="concept-NavigationDestination-url">URL</span>
     <span data-x="concept-url-equals">equals</span> <var>currentURL</var> with <span data-x="url
     equals exclude fragments"><i>exclude fragments</i></span> set to true; and</p></li>

     <li><p><var>destination</var>'s <span data-x="concept-NavigationDestination-url">URL</span>'s
     <span data-x="concept-url-fragment">fragment</span> is not <span>identical to</span>
     <var>currentURL</var>'s <span data-x="concept-url-fragment">fragment</span>,</p></li>
    </ul>

    <p>then initialize <var>event</var>'s <code
    data-x="dom-NavigateEvent-hashChange">hashChange</code> to true. Otherwise, initialize it to
    false.</p>

    <p class="note">The first condition here means that <code
    data-x="dom-NavigateEvent-hashChange">hashChange</code> will be true for <span
    data-x="navigate-fragid">fragment navigations</span>, but false for cases like <code
    data-x="">history.pushState(undefined, "", "#fragment")</code>.</p>
   </li>

   <li><p>If <var>userInvolvement</var> is not "<code data-x="uni-none">none</code>", then
   initialize <var>event</var>'s <code data-x="dom-NavigateEvent-userInitiated">userInitiated</code>
   to true. Otherwise, initialize it to false.</p></li>

   <li><p>If <var>formDataEntryList</var> is not null, then initialize <var>event</var>'s <code
   data-x="dom-NavigateEvent-formData">formData</code> to a <span>new</span> <code>FormData</code>
   created in <var>navigation</var>'s <span data-x="concept-relevant-realm">relevant realm</span>,
   associated to <var>formDataEntryList</var>. Otherwise, initialize it to null.</p></li>

   <li><p><span>Assert</span>: <var>navigation</var>'s <span>ongoing <code
   data-x="event-navigate">navigate</code> event</span> is null.</p></li>

   <li><p>Set <var>navigation</var>'s <span>ongoing <code data-x="event-navigate">navigate</code>
   event</span> to <var>event</var>.</p></li>

   <li><p>Set <var>navigation</var>'s <span>focus changed during ongoing navigation</span> to
   false.</p></li>

   <li><p>Set <var>navigation</var>'s <span>suppress normal scroll restoration during ongoing
   navigation</span> to false.</p></li>

   <li><p>Let <var>dispatchResult</var> be the result of <span
   data-x="concept-event-dispatch">dispatching</span> <var>event</var> at
   <var>navigation</var>.</p></li>

   <li>
    <p>If <var>dispatchResult</var> is false:</p>

    <ol>
     <li><p>If <var>navigationType</var> is "<code
     data-x="dom-NavigationType-traverse">traverse</code>", then <span>consume history-action user
     activation</span> given <var>navigation</var>'s <span>relevant global object</span>.</p></li>

     <li><p>If <var>event</var>'s <span data-x="concept-NavigateEvent-abort-controller">abort
     controller</span>'s <span data-x="concept-AbortController-signal">signal</span> is not <span
     data-x="AbortSignal-aborted">aborted</span>, then <span>abort the ongoing navigation</span>
     given <var>navigation</var>.</p></li>

     <li><p>Return false.</p></li>
    </ol>
   </li>

   <li><p>Let <var>endResultIsSameDocument</var> be true if <var>event</var>'s <span
   data-x="concept-NavigateEvent-interception-state">interception state</span> is not "<code
   data-x="">none</code>" or <var>event</var>'s <code
   data-x="dom-NavigateEvent-destination">destination</code>'s <span
   data-x="concept-NavigationDestination-sameDocument">is same document</span> is true.</p></li>

   <li>
    <p><span>Prepare to run script</span> given <var>navigation</var>'s <span>relevant settings
    object</span>.</p>

    <div class="note" id="note-suppress-microtasks-during-navigation-events">
     <p>This is done to avoid the <span>JavaScript execution context stack</span> becoming empty
     right after any <code data-x="event-currententrychange">currententrychange</code> event
     handlers run as a result of the <span>URL and history update steps</span> that could soon
     happen. If the stack were to become empty at that time, then it would immediately <span>perform
     a microtask checkpoint</span>, causing various promise fulfillment handlers to run interleaved
     with the event handlers and before any handlers passed to <code
     data-x="dom-NavigateEvent-intercept">navigateEvent.intercept()</code>. This is undesirable
     since it means promise handler ordering vs. <code
     data-x="event-currententrychange">currententrychange</code> event handler ordering vs. <code
     data-x="dom-NavigateEvent-intercept">intercept()</code> handler ordering would be dependent on
     whether the navigation is happening with an empty <span>JavaScript execution context
     stack</span> (e.g., because the navigation was user-initiated) or with a nonempty one (e.g.,
     because the navigation was caused by a JavaScript API call).</p>

     <p>By inserting an otherwise-unnecessary <span>JavaScript execution context</span> onto the
     stack in this step, we essentially suppress the <span>perform a microtask checkpoint</span>
     algorithm until later, thus ensuring that the sequence is always: <code
     data-x="event-currententrychange">currententrychange</code> event handlers, then <code
     data-x="dom-NavigateEvent-intercept">intercept()</code> handlers, then promise handlers.</p>
    </div>
   </li>

   <li>
    <p>If <var>event</var>'s <span data-x="concept-NavigateEvent-interception-state">interception
    state</span> is not "<code data-x="">none</code>":</p>

    <ol>
     <li><p>Set <var>event</var>'s <span
     data-x="concept-NavigateEvent-interception-state">interception state</span> to "<code
     data-x="">committed</code>".</p></li>

     <li><p>Let <var>fromNHE</var> be the <span data-x="navigation-current-entry">current
     entry</span> of <var>navigation</var>.</p></li>

     <li><p><span>Assert</span>: <var>fromNHE</var> is not null.</p></li>

     <li><p>Set <var>navigation</var>'s <span
     data-x="concept-Navigation-transition">transition</span> to a <span>new</span>
     <code>NavigationTransition</code> created in <var>navigation</var>'s <span
     data-x="concept-relevant-realm">relevant realm</span>, whose <span
     data-x="concept-NavigationTransition-navigationType">navigation type</span> is
     <var>navigationType</var>, whose <span data-x="concept-NavigationTransition-from">from entry</span>
     is <var>fromNHE</var>, and whose <span data-x="concept-NavigationTransition-finished">finished
     promise</span> is a new promise created in <var>navigation</var>'s <span
     data-x="concept-relevant-realm">relevant realm</span>.</p></li>

     <li>
      <p><span>Mark as handled</span> <var>navigation</var>'s <span
      data-x="concept-Navigation-transition">transition</span>'s <span
      data-x="concept-NavigationTransition-finished">finished promise</span>.</p>

      <p class="note">See the <a href="#note-mark-as-handled-navigation-api-finished">discussion
      about other finished promises</a> to understand why this is done.</p>
     </li>

     <li>
      <p>If <var>navigationType</var> is "<code
      data-x="dom-NavigationType-traverse">traverse</code>", then set <var>navigation</var>'s
      <span>suppress normal scroll restoration during ongoing navigation</span> to true.</p>

      <p class="note">If <var>event</var>'s <span data-x="concept-NavigateEvent-scroll">scroll
      behavior</span> was set to "<code
      data-x="dom-NavigationScrollBehavior-after-transition">after-transition</code>", then scroll
      restoration will happen as part of <span data-x="NavigateEvent-finish">finishing</span> the
      relevant <code>NavigateEvent</code>. Otherwise, there will be no scroll restoration. That is,
      no navigation which is intercepted by <code
      data-x="dom-NavigateEvent-intercept">intercept()</code> goes through the normal scroll
      restoration process; scroll restoration for such navigations is either done manually, by the
      web developer, or is done after the transition.</p>
     </li>

     <li>
      <p>If <var>navigationType</var> is "<code data-x="dom-NavigationType-push">push</code>" or
      "<code data-x="dom-NavigationType-replace">replace</code>", then run the <span>URL and history
      update steps</span> given <var>document</var> and <var>event</var>'s <code
      data-x="dom-NavigateEvent-destination">destination</code>'s <span
      data-x="dom-NavigationDestination-url">URL</span>, with <i
      data-x="uhus-serializedData">serializedData</i> set to <var>event</var>'s <span
      data-x="concept-NavigateEvent-classic-history-API-state">classic history API state</span> and
      <i data-x="uhus-historyHandling">historyHandling</i> set to <var>navigationType</var>.</p>

     <li>
      <p>Otherwise, if <var>navigationType</var> is "<code
      data-x="dom-NavigationType-reload">reload</code>", then
      <span>update the navigation API entries for a same-document navigation</span> given
      <var>navigation</var>, <var>navigable</var>'s <span
      data-x="nav-active-history-entry">active session history entry</span>, and "<code
      data-x="dom-NavigationType-reload">reload</code>".</p>

      <p class="note">If <var>navigationType</var> is "<code
      data-x="dom-NavigationType-traverse">traverse</code>", then this event firing is happening as
      part of <span data-x="apply the traverse history step">the traversal process</span>, and that
      process will take care of performing the appropriate session history entry updates.</p>
     </li>
    </ol>
   </li>

   <li>
    <p>If <var>endResultIsSameDocument</var> is true:</p>

    <ol>
     <li><p>Let <var>promisesList</var> be an empty <span>list</span>.</p></li>

     <li>
      <p><span data-x="list iterate">For each</span> <var>handler</var> of <var>event</var>'s <span
      data-x="concept-NavigateEvent-navigation-handler-list">navigation handler list</span>:</p>

      <ol>
       <li><p><span data-x="list append">Append</span> the result of <span
       data-x="es-invoking-callback-functions">invoking</span> <var>handler</var> with an empty
       arguments list to <var>promisesList</var>.</p></li>
      </ol>
     </li>

     <li>
      <p>If <var>promisesList</var>'s <span data-x="list size">size</span> is 0, then set
      <var>promisesList</var> to « <span>a promise resolved with</span> undefined ».</p>

      <p class="note">There is a subtle timing difference between how <span data-x="wait for
      all">waiting for all</span> schedules its success and failure steps when given zero promises
      versus ≥1 promises. For most uses of <span data-x="wait for all">waiting for all</span>,
      this does not matter. However, with this API, there are so many events and promise handlers
      which could fire around the same time that the difference is pretty easily observable: it can
      cause the event/promise handler sequence to vary. (Some of the events and promises involved
      include: <code data-x="event-navigatesuccess">navigatesuccess</code> / <code
      data-x="event-navigateerror">navigateerror</code>, <code
      data-x="event-currententrychange">currententrychange</code>, <code
      data-x="event-dispose">dispose</code>, <var>apiMethodTracker</var>'s promises, and the <code
      data-x="dom-NavigationTransition-finished">navigation.transition.finished</code> promise.)
     </li>

     <li>
      <p><span>Wait for all</span> of <var>promisesList</var>, with the following success steps:</p>

      <ol>
       <li><p>If <var>event</var>'s <span>relevant global object</span> is not <span>fully
       active</span>, then abort these steps.</p></li>

       <li><p>If <var>event</var>'s <span data-x="concept-NavigateEvent-abort-controller">abort
       controller</span>'s <span data-x="concept-AbortController-signal">signal</span> is <span
       data-x="AbortSignal-aborted">aborted</span>, then abort these steps.</p></li>

       <li><p><span>Assert</span>: <var>event</var> equals <var>navigation</var>'s <span>ongoing
       <code data-x="event-navigate">navigate</code> event</span>.</p></li>

       <li><p>Set <var>navigation</var>'s <span>ongoing <code
       data-x="event-navigate">navigate</code> event</span> to null.</p></li>

       <li><p><span data-x="NavigateEvent-finish">Finish</span> <var>event</var> given
       true.</p></li>

       <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
       data-x="event-navigatesuccess">navigatesuccess</code> at <var>navigation</var>.</p></li>

       <li><p>If <var>apiMethodTracker</var> is non-null, then <span>resolve the finished
       promise</span> for <var>apiMethodTracker</var>.</p></li>

       <li><p>If <var>navigation</var>'s <span
       data-x="concept-Navigation-transition">transition</span> is not null, then resolve
       <var>navigation</var>'s <span data-x="concept-Navigation-transition">transition</span>'s
       <span data-x="concept-NavigationTransition-finished">finished promise</span> with
       undefined.</p></li>

       <li><p>Set <var>navigation</var>'s <span
       data-x="concept-Navigation-transition">transition</span> to null.</p></li>
      </ol>

      <p>and the following failure steps given reason <var>rejectionReason</var>:</p>

      <ol>
       <li><p>If <var>event</var>'s <span>relevant global object</span> is not <span>fully
       active</span>, then abort these steps.</p></li>

       <li><p>If <var>event</var>'s <span data-x="concept-NavigateEvent-abort-controller">abort
       controller</span>'s <span data-x="concept-AbortController-signal">signal</span> is <span
       data-x="AbortSignal-aborted">aborted</span>, then abort these steps.</p></li>

       <li><p><span>Assert</span>: <var>event</var> equals <var>navigation</var>'s <span>ongoing
       <code data-x="event-navigate">navigate</code> event</span>.</p></li>

       <li><p>Set <var>navigation</var>'s <span>ongoing <code
       data-x="event-navigate">navigate</code> event</span> to null.</p></li>

       <li><p><span data-x="NavigateEvent-finish">Finish</span> <var>event</var> given
       false.</p></li>

       <li><p>Let <var>errorInfo</var> be the result of <span data-x="extract-error">extracting error
       information</span> from <var>rejectionReason</var>.</p></li>

       <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
       data-x="event-navigateerror">navigateerror</code> at <var>navigation</var> using
       <code>ErrorEvent</code>, with additional attributes initialized according to
       <var>errorInfo</var>.</p></li>

       <li><p>If <var>apiMethodTracker</var> is non-null, then <span>reject the finished
       promise</span> for <var>apiMethodTracker</var> with <var>rejectionReason</var>.</p></li>

       <li><p>If <var>navigation</var>'s <span
       data-x="concept-Navigation-transition">transition</span> is not null, then reject
       <var>navigation</var>'s <span data-x="concept-Navigation-transition">transition</span>'s
       <span data-x="concept-NavigationTransition-finished">finished promise</span> with
       <var>rejectionReason</var>.</p></li>

       <li><p>Set <var>navigation</var>'s <span
       data-x="concept-Navigation-transition">transition</span> to null.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Otherwise, if <var>apiMethodTracker</var> is non-null, then <span
   data-x="navigation-api-method-tracker-clean-up">clean up</span>
   <var>apiMethodTracker</var>.</p></li>

   <li>
    <p><span>Clean up after running script</span> given <var>navigation</var>'s <span>relevant
    settings object</span>.</p>

    <p class="note">Per the <a href="#note-suppress-microtasks-during-navigation-events">previous
    note</a>, this stops suppressing any potential promise handler microtasks, causing them to run
    at this point or later.</p>
   </li>

   <li><p>If <var>event</var>'s <span data-x="concept-NavigateEvent-interception-state">interception
   state</span> is "<code data-x="">none</code>", then return true.</p></li>

   <li><p>Return false.</p></li>
  </ol>


  <h6 id="navigate-event-scroll-focus">Scroll and focus behavior</h6>

  <p>By calling <code data-x="dom-NavigateEvent-intercept">navigateEvent.intercept()</code>, web
  developers can suppress the normal scroll and focus behavior for same-document navigations,
  instead invoking cross-document navigation-like behavior at a later time. The algorithms in this
  section are called at those appropriate later points.</p>

  <p>To <dfn data-x="NavigateEvent-finish">finish</dfn> a <code>NavigateEvent</code>
  <var>event</var>, given a boolean <var>didFulfill</var>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>event</var>'s <span
   data-x="concept-NavigateEvent-interception-state">interception state</span> is not "<code
   data-x="">intercepted</code>" or "<code data-x="">finished</code>".</p></li>

   <li><p>If <var>event</var>'s <span data-x="concept-NavigateEvent-interception-state">interception
   state</span> is "<code data-x="">none</code>", then return.</p></li>

   <li><p><span>Potentially reset the focus</span> given <var>event</var>.</p></li>

   <li><p>If <var>didFulfill</var> is true, then <span>potentially process scroll behavior</span>
   given <var>event</var>.</p></li>

   <li><p>Set <var>event</var>'s <span
   data-x="concept-NavigateEvent-interception-state">interception state</span> to "<code
   data-x="">finished</code>".</p></li>
  </ol>

  <p>To <dfn>potentially reset the focus</dfn> given a <code>NavigateEvent</code>
  <var>event</var>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>event</var>'s <span
   data-x="concept-NavigateEvent-interception-state">interception state</span> is "<code
   data-x="">committed</code>" or "<code data-x="">scrolled</code>".</p></li>

   <li><p>Let <var>navigation</var> be <var>event</var>'s <span>relevant global object</span>'s
   <span data-x="window-navigation-api">navigation API</span>.</p></li>

   <li><p>Let <var>focusChanged</var> be <var>navigation</var>'s <span>focus changed during ongoing
   navigation</span>.</p></li>

   <li><p>Set <var>navigation</var>'s <span>focus changed during ongoing navigation</span> to
   false.</p></li>

   <li><p>If <var>focusChanged</var> is true, then return.</p></li>

   <li>
    <p>If <var>event</var>'s <span data-x="concept-NavigateEvent-focusReset">focus reset
    behavior</span> is "<code data-x="dom-NavigationFocusReset-manual">manual</code>", then
    return.</p>

    <p class="note">If it was left as null, then we treat that as "<code
    data-x="dom-NavigationFocusReset-after-transition">after-transition</code>", and continue
    onward.</p>
   </li>

   <li><p>Let <var>document</var> be <var>event</var>'s <span>relevant global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>Let <var>focusTarget</var> be the <span>autofocus delegate</span> for
   <var>document</var>.</p></li>

   <li><p>If <var>focusTarget</var> is null, then set <var>focusTarget</var> to
   <var>document</var>'s <span data-x="the body element">body element</span>.</p></li>

   <li><p>If <var>focusTarget</var> is null, then set <var>focusTarget</var> to
   <var>document</var>'s <span>document element</span>.</p></li>

   <li><p>Run the <span>focusing steps</span> for <var>focusTarget</var>, with <var>document</var>'s
   <span>viewport</span> as the <var>fallback target</var>.</p></li>

   <li><p>Move the <span>sequential focus navigation starting point</span> to
   <var>focusTarget</var>.</p></li>
  </ol>

  <p>To <dfn>potentially process scroll behavior</dfn> given a <code>NavigateEvent</code>
  <var>event</var>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>event</var>'s <span
   data-x="concept-NavigateEvent-interception-state">interception state</span> is "<code
   data-x="">committed</code>" or "<code data-x="">scrolled</code>".</p></li>

   <li><p>If <var>event</var>'s <span data-x="concept-NavigateEvent-interception-state">interception
   state</span> is "<code data-x="">scrolled</code>", then return.</p></li>

   <li>
    <p>If <var>event</var>'s <span data-x="concept-NavigateEvent-scroll">scroll behavior</span> is
    "<code data-x="dom-NavigationScrollBehavior-manual">manual</code>", then return.</p>

    <p class="note">If it was left as null, then we treat that as "<code
    data-x="dom-NavigationScrollBehavior-after-transition">after-transition</code>", and continue
    onward.</p>
   </li>

   <li><p><span>Process scroll behavior</span> given <var>event</var>.</p>
  </ol>

  <p>To <dfn>process scroll behavior</dfn> given a <code>NavigateEvent</code> <var>event</var>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>event</var>'s <span
   data-x="concept-NavigateEvent-interception-state">interception state</span> is "<code
   data-x="">committed</code>".</p></li>

   <li><p>Set <var>event</var>'s <span
   data-x="concept-NavigateEvent-interception-state">interception state</span> to "<code
   data-x="">scrolled</code>".</p></li>

   <li><p>If <var>event</var>'s <code
   data-x="dom-NavigateEvent-navigationType">navigationType</code> was initialized to "<code
   data-x="dom-NavigationType-traverse">traverse</code>" or "<code
   data-x="dom-NavigationType-reload">reload</code>", then <span>restore scroll position data</span>
   given <var>event</var>'s <span>relevant global object</span>'s <span data-x="window
   navigable">navigable</span>'s <span data-x="nav-active-history-entry">active session history
   entry</span>.</p></li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p>Let <var>document</var> be <var>event</var>'s <span>relevant global object</span>'s
     <span data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

     <li><p>If <var>document</var>'s <span>indicated part</span> is null, then <span>scroll to the
     beginning of the document</span> given <var>document</var>. <ref>CSSOMVIEW</ref></p></li>

     <li><p>Otherwise, <span>scroll to the fragment</span> given <var>document</var>.</p></li>
    </ol>
   </li>
  </ol>

  </div>

  <h4 id="nav-traversal-event-interfaces">Event interfaces</h4>

  <p class="note">The <code>NavigateEvent</code> interface has <a href="#the-navigate-event">its own
  dedicated section</a>, due to its complexity.</p>


  <h5>The <code>NavigationCurrentEntryChangeEvent</code> interface</h5>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>NavigationCurrentEntryChangeEvent</dfn> : <span>Event</span> {
  constructor(DOMString type, <span>NavigationCurrentEntryChangeEventInit</span> eventInitDict);

  readonly attribute <span>NavigationType</span>? <span data-x="dom-NavigationCurrentEntryChangeEvent-navigationType">navigationType</span>;
  readonly attribute <span>NavigationHistoryEntry</span> <span data-x="dom-NavigationCurrentEntryChangeEvent-from">from</span>;
};

dictionary <dfn dictionary>NavigationCurrentEntryChangeEventInit</dfn> : <span>EventInit</span> {
  <span>NavigationType</span>? <dfn dict-member for="NavigationCurrentEntryChangeEventInit" data-x="dom-NavigationCurrentEntryChangeEventInit-navigationType">navigationType</dfn> = null;
  required <span>NavigationHistoryEntry</span> <dfn dict-member for="NavigationCurrentEntryChangeEventInit" data-x="dom-NavigationCurrentEntryChangeEventInit-from">from</dfn>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-NavigationCurrentEntryChangeEvent-navigationType">navigationType</span></code></dt>
   <dd><p>Returns the type of navigation which caused the current entry to change, or null if the
   change is due to <code
   data-x="dom-Navigation-updateCurrentEntry">navigation.updateCurrentEntry()</code>.</p></dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-NavigationCurrentEntryChangeEvent-from">from</span></code></dt>
   <dd>
    <p>Returns the previous value of <code
    data-x="dom-Navigation-currentEntry">navigation.currentEntry</code>, before the current entry
    changed.</p>

    <p>If <code data-x="dom-NavigationCurrentEntryChangeEvent-navigationType">navigationType</code>
    is null or "<code data-x="dom-NavigationType-reload">reload</code>", then this value will be the
    same as <code data-x="dom-Navigation-currentEntry">navigation.currentEntry</code>. In that case,
    the event signifies that the contents of the entry changed, even if we did not move to a new
    entry or replace the current one.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="NavigationCurrentEntryChangeEvent"><code
  data-x="dom-NavigationCurrentEntryChangeEvent-navigationType">navigationType</code></dfn> and <dfn
  attribute for="NavigationCurrentEntryChangeEvent"><code
  data-x="dom-NavigationCurrentEntryChangeEvent-from">from</code></dfn> attributes must return the
  values they were initialized to.</p>

  </div>


  <h5>The <code>PopStateEvent</code> interface</h5>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>PopStateEvent</dfn> : <span>Event</span> {
  constructor(DOMString type, optional <span>PopStateEventInit</span> eventInitDict = {});

  readonly attribute any <span data-x="dom-PopStateEvent-state">state</span>;
  readonly attribute boolean <span data-x="dom-PopStateEvent-hasUAVisualTransition">hasUAVisualTransition</span>;
};

dictionary <dfn dictionary>PopStateEventInit</dfn> : <span>EventInit</span> {
  any <dfn dict-member for="PopStateEventInit" data-x="dom-PopStateEventInit-state">state</dfn> = null;
  boolean <dfn dict-member for="PopStateEventInit" data-x="dom-PopStateEventInit-hasUAVisualTransition">hasUAVisualTransition</dfn> = false;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-PopStateEvent-state">state</span></code></dt>

   <dd><p>Returns a copy of the information that was provided to <code
   data-x="dom-history-pushState">pushState()</code> or <code
   data-x="dom-history-replaceState">replaceState()</code>.</p></dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-PopStateEvent-hasUAVisualTransition">hasUAVisualTransition</span></code></dt>

   <dd><p>Returns true if the user agent performed a visual transition for this navigation before
   dispatching this event. If true, the best user experience will be given if the author
   synchronously updates the DOM to the post-navigation state.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="PopStateEvent"><code
  data-x="dom-PopStateEvent-state">state</code></dfn> attribute must return the value it was
  initialized to. It represents the context information for the event, or null, if the state
  represented is the initial state of the <code>Document</code>.</p>

  <p>The <dfn attribute for="PopStateEvent"><code data-x="dom-PopStateEvent-hasUAVisualTransition">
  hasUAVisualTransition</code></dfn> attribute must return the value it was initialized to.</p>

  </div>


  <h5>The <code>HashChangeEvent</code> interface</h5>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HashChangeEvent</dfn> : <span>Event</span> {
  constructor(DOMString type, optional <span>HashChangeEventInit</span> eventInitDict = {});

  readonly attribute USVString <span data-x="dom-HashChangeEvent-oldURL">oldURL</span>;
  readonly attribute USVString <span data-x="dom-HashChangeEvent-newURL">newURL</span>;
};

dictionary <dfn dictionary>HashChangeEventInit</dfn> : <span>EventInit</span> {
  USVString oldURL = "";
  USVString newURL = "";
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-HashChangeEvent-oldURL">oldURL</span></code></dt>

   <dd>
    <p>Returns the <span>URL</span> of the <span>session history entry</span> that was previously
    current.</p>
   </dd>


   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-HashChangeEvent-newURL">newURL</span></code></dt>

   <dd>
    <p>Returns the <span>URL</span> of the <span>session history entry</span> that is now
    current.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="HashChangeEvent"><code
  data-x="dom-HashChangeEvent-oldURL">oldURL</code></dfn> attribute must return the value it was
  initialized to. It represents context information for the event, specifically the URL of the
  <span>session history entry</span> that was traversed from.</p>

  <p>The <dfn attribute for="HashChangeEvent"><code
  data-x="dom-HashChangeEvent-newURL">newURL</code></dfn> attribute must return the value it was
  initialized to. It represents context information for the event, specifically the URL of the
  <span>session history entry</span> that was traversed to.</p>

  </div>


  <h5>The <code>PageSwapEvent</code> interface</h5>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>PageSwapEvent</dfn> : <span>Event</span> {
  constructor(DOMString type, optional <span>PageSwapEventInit</span> eventInitDict = {});
  readonly attribute <code>NavigationActivation</code>? <span data-x="dom-PageSwapEvent-activation">activation</span>;
  readonly attribute <code>ViewTransition</code>? <span data-x="dom-PageSwapEvent-viewTransition">viewTransition</span>;
};

dictionary <dfn dictionary>PageSwapEventInit</dfn> : <span>EventInit</span> {
  <span>NavigationActivation</span>? <dfn dict-member for="PageSwapEventInit" data-x="dom-PageSwapEventInit-activation">activation</dfn> = null;
  <code>ViewTransition</code>? <dfn dict-member for="PageSwapEventInit" data-x="dom-PageSwapEventInit-viewTransition">viewTransition</dfn> = null;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-PageSwapEvent-activation">activation</span></code></dt>
   <dd><p>A <code>NavigationActivation</code> object representing the destination and type of the
   cross-document navigation. This would be null for cross-origin navigations.</p></dd>

   <dt><code data-x=""><var>event</var>.<span data-x="dom-PageSwapEvent-activation">activation</span>.<span data-x="dom-NavigationActivation-entry">entry</span></code></dt>
   <dd><p>A <code>NavigationHistoryEntry</code>, representing the <code>Document</code> that is
   about to become active.</p></dd>

   <dt><code data-x=""><var>event</var>.<span data-x="dom-PageSwapEvent-activation">activation</span>.<span data-x="dom-NavigationActivation-from">from</span></code></dt>
   <dd><p>A <code>NavigationHistoryEntry</code>, equivalent to the value of the <code
   data-x="dom-Navigation-currentEntry">navigation.currentEntry</code> property at the moment the
   event is fired.</p></dd>

   <dt><code data-x=""><var>event</var>.<span data-x="dom-PageSwapEvent-activation">activation</span>.<span data-x="dom-NavigationActivation-navigationType">navigationType</span></code></dt>
   <dd><p>One of "<code data-x="dom-NavigationType-push">push</code>", "<code
   data-x="dom-NavigationType-replace">replace</code>", "<code
   data-x="dom-NavigationType-reload">reload</code>", or "<code
   data-x="dom-NavigationType-traverse">traverse</code>", indicating what type of navigation
   that is about to result in a page swap.</p></dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-PageSwapEvent-viewTransition">viewTransition</span></code></dt>
   <dd>
    <p>Returns the <code>ViewTransition</code> object that represents an outbound cross-document view
    transition, if such transition is active when the event is fired. Otherwise, returns null.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="PageSwapEvent"><code
  data-x="dom-PageSwapEvent-activation">activation</code></dfn> and <dfn attribute
  for="PageSwapEvent"><code
  data-x="dom-PageSwapEvent-viewTransition">viewTransition</code></dfn> attributes must return
  the values they were initialized to.</p>

  </div>


  <h5>The <code>PageRevealEvent</code> interface</h5>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>PageRevealEvent</dfn> : <span>Event</span> {
  constructor(DOMString type, optional <span>PageRevealEventInit</span> eventInitDict = {});
  readonly attribute <code>ViewTransition</code>? <span data-x="dom-PageRevealEvent-viewTransition">viewTransition</span>;
};

dictionary <dfn dictionary>PageRevealEventInit</dfn> : <span>EventInit</span> {
  <code>ViewTransition</code>? <dfn dict-member for="PageRevealEventInit" data-x="dom-PageRevealEventInit-viewTransition">viewTransition</dfn> = null;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-PageRevealEvent-viewTransition">viewTransition</span></code></dt>
   <dd>
    <p>Returns the <code>ViewTransition</code> object that represents an inbound cross-document view
    transition, if such transition is active when the event is fired. Otherwise, returns null.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="PageRevealEvent"><code
  data-x="dom-PageRevealEvent-viewTransition">viewTransition</code></dfn> attribute must return the
  value it was initialized to.</p>

  </div>


  <h5>The <code>PageTransitionEvent</code> interface</h5>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>PageTransitionEvent</dfn> : <span>Event</span> {
  constructor(DOMString type, optional <span>PageTransitionEventInit</span> eventInitDict = {});

  readonly attribute boolean <span data-x="dom-PageTransitionEvent-persisted">persisted</span>;
};

dictionary <dfn dictionary>PageTransitionEventInit</dfn> : <span>EventInit</span> {
  boolean persisted = false;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-PageTransitionEvent-persisted">persisted</span></code></dt>

   <dd>
    <p>For the <code data-x="event-pageshow">pageshow</code> event, returns false if the page is
    newly being loaded (and the <code data-x="event-load">load</code> event will fire). Otherwise,
    returns true.</p>

    <p>For the <code data-x="event-pagehide">pagehide</code> event, returns false if the page is
    going away for the last time. Otherwise, returns true, meaning that the page might be reused if
    the user navigates back to this page (if the <code>Document</code>'s <i
    data-x="concept-document-salvageable">salvageable</i> state stays true). </p>

    <p>Things that can cause the page to be unsalvageable include:</p>

    <ul>
     <li><p>The user agent decided to not keep the <code>Document</code> alive in a <span>session
     history entry</span> after <span data-x="unload a document">unload</span></p></li>

     <li><p>Having <code>iframe</code>s that are not <i
     data-x="concept-document-salvageable">salvageable</i></p></li>

     <li><p>Active <code>WebSocket</code> objects</p></li>

     <li><p><span data-x="abort a document">Aborting a <code>Document</code></span></p></li>
    </ul>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="PageTransitionEvent"><code
  data-x="dom-PageTransitionEvent-persisted">persisted</code></dfn> attribute must return the value
  it was initialized to. It represents the context information for the event.</p>

  <p>To <dfn>fire a page transition event</dfn> named <var>eventName</var> at a <code>Window</code>
  <var>window</var> with a boolean <var>persisted</var>, <span data-x="concept-event-fire">fire
  an event</span> named <var>eventName</var> at <var>window</var>, using
  <code>PageTransitionEvent</code>, with the <code
  data-x="dom-PageTransitionEvent-persisted">persisted</code> attribute initialized to
  <var>persisted</var>, the <code data-x="dom-Event-cancelable">cancelable</code> attribute
  initialized to true, the <code data-x="dom-Event-bubbles">bubbles</code> attribute initialized to
  true, and <var>legacy target override flag</var> set.</p>

  <p class="note">The values for <code data-x="dom-Event-cancelable">cancelable</code> and <code
  data-x="dom-Event-bubbles">bubbles</code> don't make any sense, since canceling the event does
  nothing and it's not possible to bubble past the <code>Window</code> object. They are set to true
  for historical reasons.</p>

  </div>


  <h5>The <code>BeforeUnloadEvent</code> interface</h5>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>BeforeUnloadEvent</dfn> : <span>Event</span> {
  attribute DOMString <span data-x="dom-BeforeUnloadEvent-returnValue">returnValue</span>;
};</code></pre>

  <p class="note">There are no <code>BeforeUnloadEvent</code>-specific initialization methods.</p>

  <p>The <code>BeforeUnloadEvent</code> interface is a legacy interface which allows <span>checking
  if unloading is canceled</span> to be controlled not only by canceling the event, but by setting
  the <code data-x="dom-BeforeUnloadEvent-returnValue">returnValue</code> attribute to a value
  besides the empty string. Authors should use the <code
  data-x="dom-Event-preventDefault">preventDefault()</code> method, or other means of canceling
  events, instead of using <code data-x="dom-BeforeUnloadEvent-returnValue">returnValue</code>.</p>

  <div w-nodev>

  <p>The <dfn attribute for="BeforeUnloadEvent"><code
  data-x="dom-BeforeUnloadEvent-returnValue">returnValue</code></dfn> attribute controls the process
  of <span>checking if unloading is canceled</span>. When the event is created, the
  attribute must be set to the empty string. On getting, it must return the last value it was set
  to. On setting, the attribute must be set to the new value.</p>

  <p class="note">This attribute is a <code data-x="">DOMString</code> only for historical reasons.
  Any value besides the empty string will be treated as a request to ask the user for
  confirmation.</p>

  </div>

  <h4>The <code>NotRestoredReasons</code> interface</h4>
  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>NotRestoredReasonDetails</dfn> {
  readonly attribute DOMString <span data-x="dom-not-restored-reason-details-reason">reason</span>;
  [Default] object toJSON();
};

[Exposed=Window]
interface <dfn interface>NotRestoredReasons</dfn> {
  readonly attribute USVString? <span data-x="dom-not-restored-reasons-src">src</span>;
  readonly attribute DOMString? <span data-x="dom-not-restored-reasons-id">id</span>;
  readonly attribute DOMString? <span data-x="dom-not-restored-reasons-name">name</span>;
  readonly attribute USVString? <span data-x="dom-not-restored-reasons-url">url</span>;
  readonly attribute FrozenArray&lt;<span>NotRestoredReasonDetails</span>&gt;? <span data-x="dom-not-restored-reasons-reasons">reasons</span>;
  readonly attribute FrozenArray&lt;<span>NotRestoredReasons</span>&gt;? <span data-x="dom-not-restored-reasons-children">children</span>;
  [Default] object toJSON();
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>notRestoredReasonDetails</var>.<span subdfn
   data-x="dom-not-restored-reason-details-reason">reason</span></code></dt>
   <dd><p>Returns a string that explains the reason that prevented the document from <a
   href="#note-bfcache">being served from back/forward cache</a>. See the definition of <span
   data-x="concept-document-bfcache-blocking-details">bfcache blocking details</span> for the
   possible string values.</p></dd>

   <dt><code data-x=""><var>notRestoredReasons</var>.<span subdfn
   data-x="dom-not-restored-reasons-src">src</span></code></dt>
   <dd><p>Returns the <code data-x="attr-iframe-src">src</code> attribute of the document's <span>node
   navigable</span>'s <span data-x="nav-container">container</span> if it is an <code>iframe</code>
   element. This can be null if not set or if it is not an <code>iframe</code> element.</p></dd>

   <dt><code data-x=""><var>notRestoredReasons</var>.<span subdfn
   data-x="dom-not-restored-reasons-id">id</span></code></dt>
   <dd><p>Returns the <code data-x="attr-id">id</code> attribute of the document's <span>node
   navigable</span>'s <span data-x="nav-container">container</span> if it is an <code>iframe</code>
   element. This can be null if not set or if it is not an <code>iframe</code> element.</p></dd>

   <dt><code data-x=""><var>notRestoredReasons</var>.<span subdfn
   data-x="dom-not-restored-reasons-name">name</span></code></dt>
   <dd><p>Returns the <code data-x="attr-iframe-name">name</code> attribute of the document's
   <span>node navigable</span>'s <span data-x="nav-container">container</span> if it is an
   <code>iframe</code> element. This can be null if not set or if it is not an <code>iframe</code>
   element.</p></dd>

   <dt><code data-x=""><var>notRestoredReasons</var>.<span subdfn
   data-x="dom-not-restored-reasons-url">url</span></code></dt>
   <dd><p>Returns the document's <span data-x="concept-document-url">URL</span>, or null if the
   document is in a cross-origin <code>iframe</code>. This is reported in addition to <code
   data-x="dom-not-restored-reasons-src">src</code> because it is possible <code>iframe</code>
   navigated since the original <code data-x="attr-iframe-src">src</code> was set.</p></dd>

   <dt><code data-x=""><var>notRestoredReasons</var>.<span subdfn
   data-x="dom-not-restored-reasons-reasons">reasons</span></code></dt>
   <dd><p>Returns an array of <code>NotRestoredReasonDetails</code> for the document. This is null
   if the document is in a cross-origin <code>iframe</code>.</p></dd>

   <dt><code data-x=""><var>notRestoredReasons</var>.<span subdfn
   data-x="dom-not-restored-reasons-children">children</span></code></dt>
   <dd><p>Returns an array of <code>NotRestoredReasons</code> that are for the document’s
   children. This is null if the document is in a cross-origin <code>iframe</code>.</p></dd>
  </dl>

  <p>A <code>NotRestoredReasonDetails</code> object has a <dfn
  data-x="concept-not-restored-reason-details-backing-struct">backing struct</dfn>, a <span
  data-x="nrr-details-struct">not restored reason details</span> or null, initially null.</p>

  <p>The <dfn attribute for="NotRestoredReasonDetails"><code
  data-x="dom-not-restored-reason-details-reason">reason</code></dfn> getter steps are to return
  <span>this</span>'s <span
  data-x="concept-not-restored-reason-details-backing-struct">backing struct</span>'s <span
  data-x="nrr-details-reason">reason</span>.</p>

  <p>To <dfn>create a <code>NotRestoredReasonDetails</code> object</dfn> given a <span
  data-x="nrr-details-struct">not restored reason details</span> <var>backingStruct</var> and a
  <span>realm</span> <var>realm</var>:</p>

  <ol>
   <li><p>Let <var>notRestoredReasonDetails</var> be a new <code>NotRestoredReasonDetails</code>
   object created in <var>realm</var>.</p></li>

   <li><p>Set <var>notRestoredReasonDetails</var>'s <span
   data-x="concept-not-restored-reason-details-backing-struct">backing struct</span> to
   <var>backingStruct</var>.</p></li>

   <li><p>Return <var>notRestoredReasonDetails</var>.</p></li>
  </ol>

  <p>A <dfn data-x="nrr-details-struct" export>not restored reason details</dfn> is a
  <span>struct</span> with the following <span data-x="struct item">items</span>:</p>

  <ul>
   <li><p><dfn data-x="nrr-details-reason">reason</dfn>, a string, initially empty.</p></li>
  </ul>

  <p>The <span data-x="nrr-details-reason">reason</span> is a string that represents the reason that
  prevented the page from being restored from <a href="#note-bfcache">back/forward cache</a>.
  The string is one of the following:</p>

  <dl>
   <dt>"<dfn data-x="blocking-fetch"><code>fetch</code></dfn>"</dt>
   <dd>While <span data-x="unload a document">unloading</span>, a fetch initiated by this
   <code>Document</code> was still ongoing and was canceled, so the page was not in a state that
   could be stored in the <a href="#note-bfcache">back/forward cache</a>.</dd>

   <dt>"<dfn data-x="blocking-navigation-failure"><code>navigation-failure</code></dfn>"</dt>
   <dd>The original navigation that created this <code>Document</code> errored, so storing the
   resulting error document in the <a href="#note-bfcache">back/forward cache</a> was
   prevented.</dd>

   <dt>"<dfn data-x="blocking-parser-aborted"><code>parser-aborted</code></dfn>"</dt>
   <dd>The <code>Document</code> never finished its initial HTML parsing, so storing the unfinished
   document in the <a href="#note-bfcache">back/forward cache</a> was prevented.</dd>

   <dt>"<dfn data-x="blocking-websocket"><code>websocket</code></dfn>"</dt>
   <dd>While <span data-x="unload a document">unloading</span>, an open <code>WebSocket</code>
   connection was shut down, so the page was not in a state that could be stored in the <a
   href="#note-bfcache">back/forward cache</a>. <ref>WEBSOCKETS</ref></dd>

   <dt>"<dfn data-x="blocking-weblock" export><code>lock</code></dfn>"</dt>
   <dd>While <span data-x="unload a document">unloading</span>, held <span>locks</span> and
   <span>lock requests</span> were terminated, so the page was not in a state that could be stored
   in the <a href="#note-bfcache">back/forward cache</a>. <ref>WEBLOCKS</ref></dd>

   <dt>"<dfn data-x="blocking-masked" export><code>masked</code></dfn>"</dt>
   <dd>This <code>Document</code> has children that are in a cross-origin <code>iframe</code>, and
   they prevented <a href="#note-bfcache">back/forward cache</a>; or this <code>Document</code>
   could not be <a href="#note-bfcache">back/forward cached</a> for user agent-specific reasons, and
   the user agent has chosen not to use one of the more specific reasons from the list of <span
   data-x="ua-specific-blocking-reasons">user-agent specific blocking reasons</span>.
  </dl>

  <p>In addition to the list above, a user agent might choose to expose a reason that prevented the
  page from being restored from <a href="note-bfcache">back/forward cache</a> for <dfn
  data-x="ua-specific-blocking-reasons">user-agent specific blocking reasons</dfn>. These are one of
  the following strings:</p>

  <dl>
   <dt>"<dfn data-x="blocking-audio-capture" export><code>audio-capture</code></dfn>"</dt>
   <dd>The <code>Document</code> requested audio capture permission by using <cite>Media Capture and
   Streams</cite>'s <code>getUserMedia()</code> with audio. <ref>MEDIASTREAM</ref></dd>

   <dt>"<dfn data-x="blocking-background-work" export><code>background-work</code></dfn>"</dt>
   <dd>The <code>Document</code> requested background work by calling <code>SyncManager</code>'s
   <code data-x="dom-SyncManager-register">register()</code> method,
   <code>PeriodicSyncManager</code>'s <code
   data-x="dom-PeriodicSyncManager-register">register()</code> method, or
   <code>BackgroundFetchManager</code>'s <code
   data-x="dom-BackgroundFetchManager-fetch">fetch()</code> method.</dd>

   <dt>"<dfn data-x="blocking-broadcastchannel-message" export><code>broadcastchannel-message</code></dfn>"</dt>
   <dd>While the page was stored in <a href="#note-bfcache">back/forward cache</a>, a
   <code>BroadcastChannel</code> connection on the page received a message and <code
   data-x="event-message">message</code> event was fired.</dd>

   <dt>"<dfn data-x="blocking-idb-event" export><code>idbversionchangeevent</code></dfn>"</dt>
   <dd>The <code>Document</code> had a pending <code>IDBVersionChangeEvent</code> while <span
   data-x="unload a document">unloading</span>. <ref>INDEXEDDB</ref></dd>

   <dt>"<dfn data-x="blocking-idledetector" export><code>idledetector</code></dfn>"</dt>
   <dd>The <code>Document</code> had an active <code>IdleDetector</code> while <span data-x="unload
   a document">unloading</span>.</dd>

   <dt>"<dfn data-x="blocking-keyboard-lock" export><code>keyboardlock</code></dfn>"</dt>
   <dd>While <span data-x="unload a document">unloading</span>, keyboard lock was still active
   because <code>Keyboard</code>'s <code data-x="dom-Keyboard-lock">lock()</code> method was
   called.</dd>

   <dt>"<dfn data-x="blocking-mediastream" export><code>mediastream</code></dfn>"</dt>
   <dd>A <code>MediaStreamTrack</code> was in the <span>live state</span> upon <span data-x="unload
   a document">unloading</span>. <ref>MEDIASTREAM</ref></dd>

   <dt>"<dfn data-x="blocking-midi" export><code>midi</code></dfn>"</dt>
   <dd>The <code>Document</code> requested a MIDI permission by calling <code
   data-x="dom-Navigator-requestMIDIAccess">navigator.requestMIDIAccess()</code>.</dd>

   <dt>"<dfn data-x="blocking-modals" export><code>modals</code></dfn>"</dt>
   <dd><a href="#user-prompts">User prompts</a> were shown while <span data-x="unload a document">
   unloading</span>.</dd>

   <dt>"<dfn data-x="blocking-navigating" export><code>navigating</code></dfn>"</dt>
   <dd>While <span data-x="unload a document">unloading</span>, loading was still ongoing, and so
   the <code>Document</code> was not in a state that could be stored in <a
   href="#note-bfcache">back/forward cache</a>.</dd>

   <dt>"<dfn data-x="blocking-navigation-canceled" export><code>navigation-canceled</code></dfn>"</dt>
   <dd>The navigation request was canceled by calling <code
   data-x="dom-window-stop">window.stop()</code> and the page was not in a state to be stored in <a
   href="note-bfcache">back/forward cache</a>.</dd>

   <dt>"<dfn data-x="blocking-non-trivial-bcg" export><code>non-trivial-browsing-context-group</code></dfn>"</dt>
   <dd>The <span>browsing context group</span> of this <code>Document</code> had more than one
   <span>top-level browsing context</span>.</dd>

   <dt>"<dfn data-x="blocking-otp" export><code>otpcredential</code></dfn>"</dt>
   <dd>The <code>Document</code> created an <code>OTPCredential</code>.</dd>

   <dt>"<dfn data-x="blocking-outstanding-network-request" export><code>outstanding-network-request</code></dfn>"</dt>
   <dd>While <span data-x="unload a document">unloading</span>, the <code>Document</code> had
   outstanding network requests and was not in a state that could be stored in <a
   href="#note-bfcache">back/forward cache</a>.</dd>

   <dt>"<dfn data-x="blocking-payment" export><code>paymentrequest</code></dfn>"</dt>
   <dd>The <code>Document</code> had an active <code>PaymentRequest</code> while <span
   data-x="unload a document">unloading</span>. <ref>PAYMENTREQUEST</ref></dd>

   <dt>"<dfn data-x="blocking-picture-in-picture" export><code>pictureinpicturewindow</code></dfn>"</dt>
   <dd>The <code>Document</code> had an active <code>PictureInPictureWindow</code> while <span
   data-x="unload a document">unloading</span>. <ref>PICTUREINPICTURE</ref></dd>

   <dt>"<dfn data-x="blocking-plugins" export><code>plugins</code></dfn>"</dt>
   <dd>The <code>Document</code> contained <span data-x="plugin">plugins</span>.</dd>

   <dt>"<dfn data-x="blocking-method-not-get" export><code>request-method-not-get</code></dfn>"</dt>
   <dd>The <code>Document</code> was created from an HTTP request whose <span
   data-x="concept-request-method">method</span> was not `<code data-x="">GET</code>`.
   <ref>FETCH</ref></dd>

   <dt>"<dfn data-x="blocking-auth-required" export><code>response-auth-required</code></dfn>"</dt>
   <dd>The <code>Document</code> was created from an HTTP response that required HTTP
   authentication.</dd>

   <dt>"<dfn data-x="blocking-ccns" export><code>response-cache-control-no-store</code></dfn>"</dt>
   <dd>The <code>Document</code> was created from an HTTP response whose `<code
   data-x="http-cache-control">Cache-Control</code>` header included the "<code
   data-x="">no-store</code>" token. <ref>HTTP</ref></dd>

   <dt>"<dfn data-x="blocking-ccnc" export><code>response-cache-control-no-cache</code></dfn>"</dt>
   <dd>The <code>Document</code> was created from an HTTP response whose `<code
   data-x="http-cache-control">Cache-Control</code>` header included the "<code
   data-x="">no-cache</code>" token. <ref>HTTP</ref></dd>

   <dt>"<dfn data-x="blocking-keepalive" export><code>response-keep-alive</code></dfn>"</dt>
   <dd>The <code>Document</code> was created from an HTTP response that contained a `<code
   data-x="">Keep-Alive</code>` header.</dd>

   <dt>"<dfn data-x="blocking-not-http-https" export><code>response-scheme-not-http-or-https</code></dfn>"</dt>
   <dd>The <code>Document</code> was created from a response whose <span
   data-x="concept-response-url">URL</span>'s <span data-x="concept-url-scheme">scheme</span> was
   not an <span>HTTP(S) scheme</span>. <ref>FETCH</ref></dd>

   <dt>"<dfn data-x="blocking-response-not-ok" export><code>response-status-not-ok</code></dfn>"</dt>
   <dd>The <code>Document</code> was created from an HTTP response whose <span
   data-x="concept-response-status">status</span> was not an <span>ok status</span>.
   <ref>FETCH</ref></dd>

   <dt>"<dfn data-x="blocking-rtc" export><code>rtc</code></dfn>"</dt>
   <dd>While <span data-x="unload a document">unloading</span>, a <code>RTCPeerConnection</code> or
   <code>RTCDataChannel</code> was shut down, so the page was not in a state that could be stored in
   the <a href="#note-bfcache">back/forward cache</a>. <ref>WEBRTC</ref></dd>

   <dt>"<dfn data-x="blocking-sensors" export><code>sensors</code></dfn>"</dt>
   <dd>The <code>Document</code> <span data-x="request-sensor-access">requested sensor
   access</span>.</dd>

   <dt>"<dfn data-x="sw-added" export><code>serviceworker-added</code></dfn>"</dt>
   <dd>The <code>Document</code>'s <span
   data-x="serviceworkercontainer-service-worker-client">service worker client</span> started to be
   <span data-x="dfn-control">controlled</span> by a <code>ServiceWorker</code> while the page was
   in <a href="#note-bfcache">back/forward cache</a>. <ref>SW</ref></dd>

   <dt>"<dfn data-x="sw-claimed" export><code>serviceworker-claimed</code></dfn>"</dt>
   <dd>The <code>Document</code>'s <span
   data-x="serviceworkercontainer-service-worker-client">service worker client</span>'s <span
   data-x="concept-environment-active-service-worker">active service worker</span> was claimed while
   the page was in <a href="#note-bfcache">back/forward cache</a>. <ref>SW</ref></dd>

   <dt>"<dfn data-x="blocking-sw-message" export><code>serviceworker-postmessage</code></dfn>"</dt>
   <dd>The <code>Document</code>'s <span
   data-x="serviceworkercontainer-service-worker-client">service worker client</span>'s <span
   data-x="concept-environment-active-service-worker">active service worker</span> received a
   message while the page was in <a href="#note-bfcache">back/forward cache</a>. <ref>SW</ref></dd>

   <dt>"<dfn data-x="blocking-sw-activation" export><code>serviceworker-version-activated</code></dfn>"</dt>
   <dd>The <code>Document</code>'s <span
   data-x="serviceworkercontainer-service-worker-client">service worker client</span>'s <span
   data-x="concept-environment-active-service-worker">active service worker</span>'s version was
   activated while the page was in <a href="#note-bfcache">back/forward cache</a>.
   <ref>SW</ref></dd>

   <dt>"<dfn data-x="blocking-sw-unregistered" export><code>serviceworker-unregistered</code></dfn>"</dt>
   <dd>The <code>Document</code>'s <span
   data-x="serviceworkercontainer-service-worker-client">service worker client</span>'s <span
   data-x="concept-environment-active-service-worker">active service worker</span>'s <span>service
   worker registration</span> was <span data-x="service-worker-unregister">unregistered</span> while
   the page was in <a href="#note-bfcache">back/forward cache</a>. <ref>SW</ref></dd>

   <dt>"<dfn data-x="blocking-sharedworker" export><code>sharedworker</code></dfn>"</dt>
   <dd>This <code>Document</code> was in the <span>owner set</span> of a
   <code>SharedWorkerGlobalScope</code>.</dd>

   <dt>"<dfn data-x="blocking-smartcard" export><code>smartcardconnection</code></dfn>"</dt>
   <dd>The <code>Document</code> had an active <code>SmartCardConnection</code> while <span
   data-x="unload a document">unloading</span>.</dd>

   <dt>"<dfn data-x="blocking-speech-reco" export><code>speechrecognition</code></dfn>"</dt>
   <dd>The <code>Document</code> had an active <code>SpeechRecognition</code> while <span
   data-x="unload a document">unloading</span>.</dd>

   <dt>"<dfn data-x="blocking-storage" export><code>storageaccess</code></dfn>"</dt>
   <dd>The <code>Document</code> requested storage access permission by using the Storage Access
   API.</dd>

   <dt>"<dfn data-x="blocking-unload" export><code>unload-listener</code></dfn>"</dt>
   <dd>The <code>Document</code> registered an <span>event listener</span> for the <code
   data-x="event-unload">unload</code> event.</dd>

   <dt>"<dfn data-x="blocking-video-capture" export><code>video-capture</code></dfn>"</dt>
   <dd>The <code>Document</code> requested video capture permission by using <cite>Media Capture and
   Streams</cite>'s <code>getUserMedia()</code> with video. <ref>MEDIASTREAM</ref></dd>

   <dt>"<dfn data-x="blocking-webhid" export><code>webhid</code></dfn>"</dt>
   <dd>The <code>Document</code> called the WebHID API's <code
   data-x="request-device">requestDevice()</code> method.</dd>

   <dt>"<dfn data-x="blocking-webshare" export><code>webshare</code></dfn>"</dt>
   <dd>The <code>Document</code> used the Web Share API's <code
   data-x="dom-Navigator-share">navigator.share()</code> method.</dd>

   <dt>"<dfn data-x="blocking-webtransport" export><code>webtransport</code></dfn>"</dt>
   <dd>While <span data-x="unload a document">unloading</span>, an open <code>WebTransport</code>
   connection was shut down, so the page was not in a state that could be stored in the <a
   href="#note-bfcache">back/forward cache</a>. <ref>WEBTRANSPORT</ref></dd>

   <dt>"<dfn data-x="blocking-webxr" export><code>webxrdevice</code></dfn>"</dt>
   <dd>The <code>Document</code> created a <code>XRSystem</code>.</dd>
  </dl>

  <hr>

  <p>A <code>NotRestoredReasons</code> object has a <dfn
  data-x="concept-not-restored-reasons-backing-struct">backing struct</dfn>,
  a <span data-x="nrr-struct">not restored reasons</span> or null, initially null.</p>

  <p>A <code>NotRestoredReasons</code> object has a <dfn
  data-x="concept-not-restored-reasons-reasons">reasons array</dfn>, a
  <code data-x="">FrozenArray&lt;<code>NotRestoredReasonDetails</code>></code> or null,
  initially null.</p>

  <p>A <code>NotRestoredReasons</code> object has a <dfn
  data-x="concept-not-restored-reasons-children">children array</dfn>, a
  <code data-x="">FrozenArray&lt;<span>NotRestoredReasons</span>></code> or null, initially
  null.</p>

  <p>The <dfn attribute for="NotRestoredReasons"><code
  data-x="dom-not-restored-reasons-src">src</code></dfn> getter steps are to return
  <span>this</span>'s <span
  data-x="concept-not-restored-reasons-backing-struct">backing struct</span>'s <span
  data-x="nrr-src">src</span>.</p>

  <p>The <dfn attribute for="NotRestoredReasons"><code
  data-x="dom-not-restored-reasons-id">id</code></dfn> getter steps are to return
  <span>this</span>'s <span
  data-x="concept-not-restored-reasons-backing-struct">backing struct</span>'s <span
  data-x="nrr-id">id</span>.</p>

  <p>The <dfn attribute for="NotRestoredReasons"><code
  data-x="dom-not-restored-reasons-name">name</code></dfn> getter steps are to return
  <span>this</span>'s <span
  data-x="concept-not-restored-reasons-backing-struct">backing struct</span>'s <span
  data-x="nrr-name">name</span>.</p>

  <p>The <dfn attribute for="NotRestoredReasons"><code
  data-x="dom-not-restored-reasons-url">url</code></dfn> getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span
   data-x="concept-not-restored-reasons-backing-struct">backing struct</span>'s <span
   data-x="nrr-url">URL</span> is null, then return null.</p></li>

   <li><p>Return <span>this</span>'s <span
   data-x="concept-not-restored-reasons-backing-struct">backing struct</span>'s <span
   data-x="nrr-url">URL</span>, <span data-x="concept-url-serializer">serialized</span>.</p></li>
  </ol>

  <p>The <dfn attribute for="NotRestoredReasons"><code
  data-x="dom-not-restored-reasons-reasons">reasons</code></dfn> getter steps are to return
  <span>this</span>'s <span
  data-x="concept-not-restored-reasons-reasons">reasons array</span>.</p>

  <p>The <dfn attribute for="NotRestoredReasons"><code
  data-x="dom-not-restored-reasons-children">children</code></dfn> getter steps are to return
  <span>this</span>'s <span
  data-x="concept-not-restored-reasons-children">children array</span>.</p>

  <p>To <dfn export>create a <code>NotRestoredReasons</code> object</dfn> given a <span
  data-x="nrr-struct">not restored reasons</span> <var>backingStruct</var> and a
  <span>realm</span> <var>realm</var>:</p>

  <ol>
   <li><p>Let <var>notRestoredReasons</var> be a new <code>NotRestoredReasons</code> object created
   in <var>realm</var>.</p></li>

   <li><p>Set <var>notRestoredReasons</var>'s <span
   data-x="concept-not-restored-reasons-backing-struct">backing struct</span> to
   <var>backingStruct</var>.</p></li>

   <li><p>If <var>backingStruct</var>'s <span data-x="nrr-reasons">reasons</span> is null, set
   <var>notRestoredReasons</var>'s <span
   data-x="concept-not-restored-reasons-reasons">reasons array</span> to null.</p></li>

   <li>
    <p>Otherwise:</p>
    <ol>
     <li><p>Let <var>reasonsArray</var> be an empty <span>list</span>.</p></li>

     <li>
      <p><span data-x="list iterate">For each</span> <var>reason</var> of
      <var>backingStruct</var>'s <span data-x="nrr-reasons">reasons</span>:</p>
      <ol>
       <li><p><span>Create a <code>NotRestoredReasonDetails</code> object</span> given
       <var>reason</var> and <var>realm</var>, and <span data-x="list append">append</span>
       it to <var>reasonsArray</var>.</p></li>
      </ol>
     </li>

     <li><p>Set <var>notRestoredReasons</var>'s <span
     data-x="concept-not-restored-reasons-reasons">reasons array</span> to the result of
     <span>creating a frozen array</span> given <var>reasonsArray</var>.</p></li>
    </ol>
   </li>

   <li><p>If <var>backingStruct</var>'s <span data-x="nrr-children">children</span> is null, set
   <var>notRestoredReasons</var>'s <span
   data-x="concept-not-restored-reasons-children">children array</span> to null.</p></li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p>Let <var>childrenArray</var> be an empty <span>list</span>.</p></li>

     <li>
      <p><span data-x="list iterate">For each</span> <var>child</var> of
      <var>backingStruct</var>'s <span data-x="nrr-children">children</span>:</p>

      <ol>
       <li><p><span>Create a <code>NotRestoredReasons</code> object</span> given <var>child</var>
       and <var>realm</var> and <span data-x="list append">append</span> it to
       <var>childrenArray</var>.</p></li>
      </ol>
     </li>

     <li><p>Set <var>notRestoredReasons</var>'s <span
     data-x="concept-not-restored-reasons-children">children array</span> to the result of
     <span>creating a frozen array</span> given <var>childrenArray</var>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>notRestoredReasons</var>.</p></li>
  </ol>

  <p>A <dfn data-x="nrr-struct" export>not restored reasons</dfn> is a <span>struct</span> with
  the following <span data-x="struct item">items</span>:</p>

  <ul>
   <li><p><dfn data-x="nrr-src">src</dfn>, a <span>scalar value string</span> or null, initially
   null.</p></li>

   <li><p><dfn data-x="nrr-id">id</dfn>, a string or null, initially null.</p></li>

   <li><p><dfn data-x="nrr-name">name</dfn>, a string or null, initially null.</p></li>

   <li><p><dfn data-x="nrr-url">url</dfn>, a <span>URL</span> or null, initially null.</p></li>

   <li><p><dfn data-x="nrr-reasons">reasons</dfn>, null or a <span>list</span> of
   <span data-x="nrr-details-struct">not restored reason details</span>, initially null.</p></li>

   <li><p><dfn data-x="nrr-children">children</dfn>, null or a <span>list</span> of <span
   data-x="nrr-struct">not restored reasons</span>, initially null.</p></li>
  </ul>

  <p>A <dfn data-x="concept-document-nrr" data-lt="not restored reasons" for="Document"
  export><code>Document</code>'s not restored reasons</dfn> is its <span>node navigable</span>'s
  <span data-x="nav-active-history-entry">active session history entry</span>'s <span
  data-x="she-document-state">document state</span>'s <span
  data-x="document-state-not-restored-reasons">not restored reasons</span>, if
  <code>Document</code>'s <span>node navigable</span> is a <span>top-level traversable</span>;
  otherwise null.</p>

  <h3 split-filename="document-sequences">Infrastructure for sequences of documents</h3>

  <p>This standard contains several related concepts for grouping sequences of documents. As a
  brief, non-normative summary:</p>

  <ul>
   <li><p><span data-x="navigable">Navigables</span> are a user-facing representation of a sequence
   of documents, i.e., they represent something that can be navigated between documents. Typical
   examples are tabs or windows in a web browser, or <code>iframe</code>s<span w-nodev>, or
   <code>frame</code>s in a <code>frameset</code></span>.</p></li>

   <li><p><span data-x="traversable navigable">Traversable navigables</span> are a special type of
   navigable which control the session history of themselves and of their descendant navigables.
   That is, in addition to their own series of documents, they represent a tree of further series
   of documents, plus the ability to linearly traverse back and forward through a flattened view of
   this tree.</p></li>

   <li><p><span data-x="browsing context">Browsing contexts</span> are a developer-facing
   representation of a series of documents. They correspond 1:1 with <code>WindowProxy</code>
   objects. Each navigable can present a series of browsing contexts, with <a
   href="#browsing-context-group-switches-due-to-cross-origin-opener-policy">switches</a> between
   those browsing contexts occuring under certain well-defined circumstances.</p></li>
  </ul>

  <p>Most of this standard works in the language of navigables, but certain APIs expose the
  existence of browsing context switches, and so some parts of the standard need to work in terms
  of browsing contexts.</p>

  <h4>Navigables</h4>

  <p>A <dfn export>navigable</dfn> presents a <code>Document</code> to the user via its <span
  data-x="nav-active-history-entry">active session history entry</span>. Each navigable has:</p>

  <ul>
   <li><p>An <dfn data-x="nav-id">id</dfn>, a <span>new unique internal value</span>.</p></li>

   <li><p id="parent-browsing-context">A <dfn data-x="nav-parent" export for=navigable>parent</dfn>,
   a <span>navigable</span> or null.</p></li>

   <li>
    <p id="current-entry">A <dfn data-x="nav-current-history-entry">current session history
    entry</dfn>, a <span>session history entry</span>.</p>

    <p>This can only be modified within the <span
    data-x="tn-session-history-traversal-queue">session history traversal queue</span> of the
    parent <span>traversable navigable</span>.</p>
   </li>

   <li>
    <p>An <dfn data-x="nav-active-history-entry">active session history entry</dfn>, a
    <span>session history entry</span>.</p>

    <p>This can only be modified from the event loop of the <span
    data-x="nav-active-history-entry">active session history entry</span>'s <span
    data-x="she-document">document</span>.</p>
   </li>

   <li>
    <p>An <dfn>is closing</dfn> boolean, initially false.</p>

    <p class="note">This is only ever set to true for <span data-x="top-level
    traversable">top-level traversable navigables</span>.</p>
   </li>

   <li>
    <p>An <dfn id="delaying-load-events-mode">is delaying <code data-x="event-load">load</code>
    events</dfn> boolean, initially false.</p>

    <p class="note">This is only ever set to true in cases where the navigable's <span
    data-x="nav-parent">parent</span> is non-null.</p>
  </ul>

  <p>The <span data-x="nav-current-history-entry">current session history entry</span> and the
  <span data-x="nav-active-history-entry">active session history entry</span> are usually the same,
  but they get out of sync when:</p>

  <ul>
   <li><p>Synchronous navigations are performed. This causes the <span
   data-x="nav-active-history-entry">active session history entry</span> to temporarily step ahead
   of the <span data-x="nav-current-history-entry">current session history entry</span>.</p></li>

   <li><p>A non-displayable, non-error response is received when <span data-x="apply the history
   step">applying the history step</span>. This updates the <span
   data-x="nav-current-history-entry">current session history entry</span> but leaves the <span
   data-x="nav-active-history-entry">active session history entry</span> as-is.</p></li>
  </ul>

  <hr>

  <p>A <span>navigable</span>'s <dfn export for="navigable" data-x="nav-document">active
  document</dfn> is its <span data-x="nav-active-history-entry">active session history
  entry</span>'s <span data-x="she-document">document</span>.</p>

  <p class="note">This can be safely read from within the <span
  data-x="tn-session-history-traversal-queue">session history traversal queue</span> of the
  navigable's <span data-x="nav-top">top-level traversable</span>. Although a
  <span>navigable</span>'s <span data-x="nav-active-history-entry">active history entry</span> can
  change synchronously, the new entry will always have the same <code>Document</code>.</p>

  <p>A <span>navigable</span>'s <dfn export for="navigable" data-x="nav-bc">active browsing
  context</dfn> is its <span data-x="nav-document">active document</span>'s <span
  data-x="concept-document-bc">browsing context</span>. If this <span>navigable</span> is a
  <span>traversable navigable</span>, then its <span data-x="nav-bc">active browsing context</span>
  will be a <span>top-level browsing context</span>.</p>

  <p>A <span>navigable</span>'s <dfn data-x="nav-wp">active <code>WindowProxy</code></dfn> is its
  <span data-x="nav-bc">active browsing context</span>'s associated <code>WindowProxy</code>.</p>

  <p>A <span>navigable</span>'s <dfn export for="navigable" data-x="nav-window">active window</dfn>
  is its <span data-x="nav-wp">active <code>WindowProxy</code></span>'s <span
  data-x="concept-windowproxy-window">[[Window]]</span>.</p>

  <p class="note">This will always equal the navigable's <span data-x="nav-document">active
  document</span>'s <span>relevant global object</span>; this is kept in sync by the <span>make
  active</span> algorithm.</p>

  <p id="browsing-context-name">A <span>navigable</span>'s <dfn data-x="nav-target">target
  name</dfn> is its <span data-x="nav-active-history-entry">active session history entry</span>'s
  <span data-x="she-document-state">document state</span>'s <span
  data-x="document-state-nav-target-name">navigable target name</span>.</p>

  <hr>

  <p>To get the <dfn export>node navigable</dfn> of a <span>node</span> <var>node</var>, return the
  <span>navigable</span> whose <span data-x="nav-document">active document</span> is
  <var>node</var>'s <span>node document</span>, or null if there is no such
  <span>navigable</span>.</p>

  <hr>

  <p>To <dfn>initialize the navigable</dfn> <span>navigable</span> <var>navigable</var>, given a
  <span>document state</span> <var>documentState</var> and an optional
  <span>navigable</span>-or-null <var>parent</var> (default null):</p>

  <ol>
   <li><p><span>Assert</span>: <var>documentState</var>'s <span
   data-x="document-state-document">document</span> is non-null.</p></li>

   <li>
    <p>Let <var>entry</var> be a new <span>session history entry</span>, with</p>

    <dl class="props">
     <dt><span data-x="she-url">URL</span></dt>
     <dd><var>documentState</var>'s <span data-x="document-state-document">document</span>'s <span
     data-x="concept-document-url">URL</span></dd>

     <dt><span data-x="she-document-state">document state</span></dt>
     <dd><var>documentState</var></dd>
    </dl>

    <p class="note">The caller of this algorithm is responsible for initializing <var>entry</var>'s
    <span data-x="she-step">step</span>; it will be left as "<code data-x="">pending</code>" until
    that is complete.</p>
   </li>

   <li><p>Set <var>navigable</var>'s <span data-x="nav-current-history-entry">current session
   history entry</span> to <var>entry</var>.</p></li>

   <li><p>Set <var>navigable</var>'s <span data-x="nav-active-history-entry">active session history
   entry</span> to <var>entry</var>.</p></li>

   <li><p>Set <var>navigable</var>'s <span data-x="nav-parent">parent</span> to
   <var>parent</var>.</p></li>
  </ol>


  <h5>Traversable navigables</h5>

  <p>A <dfn export>traversable navigable</dfn> is a <span>navigable</span> that also controls which
  <span>session history entry</span> should be the <span data-x="nav-current-history-entry">current
  session history entry</span> and <span data-x="nav-active-history-entry">active session history
  entry</span> for itself and its descendant <span data-x="navigable">navigables</span>.</p>

  <p>In addition to the properties of a <span>navigable</span>, a <span>traversable
  navigable</span> has:</p>

  <ul>
   <li><p>A <dfn export data-x="tn-current-session-history-step">current session history
   step</dfn>, a number, initially 0.</p></li>

   <li><p id="session-history"><dfn data-x="tn-session-history-entries">Session history
   entries</dfn>, a <span>list</span> of <span data-x="session history entry">session history
   entries</span>, initially a new <span>list</span>.</p></li>

   <li><p>A <dfn data-x="tn-session-history-traversal-queue">session history traversal queue</dfn>,
   a <span>session history traversal parallel queue</span>, the result of <span>starting a new
   session history traversal parallel queue</span>.</p></li>

   <li><p>A <dfn data-x="tn-running-nested-apply-history-step">running nested apply history
   step</dfn> boolean, initially false.</p></li>

   <li>
    <p>A <dfn>system visibility state</dfn>, which is either "<code data-x="">hidden</code>" or
    "<code data-x="">visible</code>".</p>

    <p>See the <a href="#page-visibility">page visibility</a> section for the requirements on this
    item.</p>
   </li>
  </ul>

  <p>To get the <dfn data-x="nav-traversable" export for="navigable">traversable navigable</dfn> of
  a <span>navigable</span> <var>inputNavigable</var>:</p>

  <ol>
   <li><p>Let <var>navigable</var> be <var>inputNavigable</var>.</p></li>

   <li><p>While <var>navigable</var> is not a <span>traversable navigable</span>, set
   <var>navigable</var> to <var>navigable</var>'s <span data-x="nav-parent">parent</span>.</p></li>

   <li><p>Return <var>navigable</var>.</p></li>
  </ol>

  <h5>Top-level traversables</h5>

  <p>A <dfn export>top-level traversable</dfn> is a <span>traversable navigable</span> with a null
  <span data-x="nav-parent">parent</span>.</p>

  <p class="note">Currently, all <span data-x="traversable navigable">traversable navigables</span>
  are <span data-x="top-level traversable">top-level traversables</span>. Future proposals envision
  introducing non-top-level traversables.</p>

  <p>A user agent holds a <dfn export for="user agent">top-level traversable set</dfn> (a
  <span>set</span> of <span data-x="top-level traversable">top-level traversables</span>). These are
  typically presented to the user in the form of browser windows or browser tabs.</p>

  <p>To get the <dfn export for="navigable" data-x="nav-top">top-level traversable</dfn> of a
  <span>navigable</span> <var>inputNavigable</var>:</p>

  <ol>
   <li><p>Let <var>navigable</var> be <var>inputNavigable</var>.</p></li>

   <li><p>While <var>navigable</var>'s <span data-x="nav-parent">parent</span> is not null, set
   <var>navigable</var> to <var>navigable</var>'s <span data-x="nav-parent">parent</span>.</p></li>

   <li><p>Return <var>navigable</var>.</p></li>
  </ol>

  <p>To <dfn data-x="creating a new top-level traversable">create a new top-level traversable</dfn>
  given a <span>browsing context</span>-or-null <var>opener</var>, a string
  <var>targetName</var>, and an optional <span>navigable</span>
  <var>openerNavigableForWebDriver</var>:</p>

  <ol>
   <li><p>Let <var>document</var> be null.</p></li>

   <li><p>If <var>opener</var> is null, then set <var>document</var> to the second return value of
   <span>creating a new top-level browsing context and document</span>.</p></li>

   <li><p>Otherwise, set <var>document</var> to the second return value of <span>creating a new
   auxiliary browsing context and document</span> given <var>opener</var>.</p></li>

   <li>
    <p>Let <var>documentState</var> be a new <span>document state</span>, with</p>

    <dl class="props">
     <dt><span data-x="document-state-document">document</span></dt>
     <dd><var>document</var></dd>

     <dt><span data-x="document-state-initiator-origin">initiator origin</span></dt>
     <dd>null if <var>opener</var> is null; otherwise, <var>document</var>'s <span
     data-x="concept-document-origin">origin</span></dd>

     <dt><span data-x="document-state-origin">origin</span></dt>
     <dd><var>document</var>'s <span data-x="concept-document-origin">origin</span></dd>

     <dt><span data-x="document-state-nav-target-name">navigable target name</span></dt>
     <dd><var>targetName</var></dd>

     <dt><span data-x="document-state-about-base-url">about base URL</span></dt>
     <dd><var>document</var>'s <span data-x="concept-document-about-base-url">about base
     URL</span></dd>
    </dl>
   </li>

   <li><p>Let <var>traversable</var> be a new <span>traversable navigable</span>.</p></li>

   <li><p><span>Initialize the navigable</span> <var>traversable</var> given
   <var>documentState</var>.</p></li>

   <li><p>Let <var>initialHistoryEntry</var> be <var>traversable</var>'s <span
   data-x="nav-active-history-entry">active session history entry</span>.</p></li>

   <li><p>Set <var>initialHistoryEntry</var>'s <span data-x="she-step">step</span> to 0.</p></li>

   <li><p><span data-x="list append">Append</span> <var>initialHistoryEntry</var> to
   <var>traversable</var>'s <span data-x="tn-session-history-entries">session history
   entries</span>.</p></li>

   <li id="copy-session-storage"><p>If <var>opener</var> is non-null, then <span>legacy-clone a
   traversable storage shed</span> given <var>opener</var>'s <span data-x="bc-traversable">top-level
   traversable</span> and <var>traversable</var>. <ref>STORAGE</ref></p></li>

   <li><p><span data-x="list append">Append</span> <var>traversable</var> to the user agent's
   <span>top-level traversable set</span>.</p></li>

   <li><p>Invoke <span>WebDriver BiDi navigable created</span> with <var>traversable</var>
   and <var>openerNavigableForWebDriver</var>.</p></li>

   <li><p>Return <var>traversable</var>.</p></li>
  </ol>

  <p>To <dfn export>create a fresh top-level traversable</dfn> given a <span>URL</span>
  <var>initialNavigationURL</var> and an optional <span>POST resource</span>-or-null
  <var>initialNavigationPostResource</var> (default null):</p>

  <ol>
   <li><p>Let <var>traversable</var> be the result of <span>creating a new top-level
   traversable</span> given null and the empty string.</p></li>

   <li>
    <p><span>Navigate</span> <var>traversable</var> to <var>initialNavigationURL</var> using
    <var>traversable</var>'s <span data-x="nav-document">active document</span>, with <i
    data-x="navigation-resource">documentResource</i> set to
    <var>initialNavigationPostResource</var>.</p>

    <p class="note">We treat these initial navigations as <var>traversable</var> navigating itself,
    which will ensure all relevant security checks pass.</p>
   </li>

   <li><p>Return <var>traversable</var>.</p></li>
  </ol>


  <h5>Child navigables</h5>

  <p id="browsing-context-container">Certain elements (for example, <code>iframe</code> elements)
  can present a <span>navigable</span> to the user. These elements are called <dfn export
  data-lt="navigable container" data-x="navigable container">navigable containers</dfn>.</p>

  <p id="nested-browsing-context">Each <span>navigable container</span> has a <dfn export
  for="navigable container">content navigable</dfn>, which is either a <span>navigable</span> or
  null. It is initially null.</p>

  <p id="bc-container">The <dfn export for="navigable" data-x="nav-container">container</dfn> of a
  <span>navigable</span> <var>navigable</var> is the <span>navigable container</span> whose
  <span>content navigable</span> is <var>navigable</var>, or null if there is no such element.</p>

  <p id="bc-container-document">The <dfn export for="navigable"
  data-x="nav-container-document">container document</dfn> of a <span>navigable</span>
  <var>navigable</var> is the result of running these steps:</p>

  <ol>
   <li><p>If <var>navigable</var>'s <span data-x="nav-container">container</span> is null, then
   return null.</p></li>

   <li>
    <p>Return <var>navigable</var>'s <span data-x="nav-container">container</span>'s <span>node
    document</span>.</p>

    <p class="note">This is equal to <var>navigable</var>'s <span
    data-x="nav-container">container</span>'s <span>shadow-including root</span> as
    <var>navigable</var>'s <span data-x="nav-container">container</span> has to be
    <span>connected</span>.</p>
   </li>
  </ol>

  <p>The <dfn data-x="doc-container-document">container document</dfn> of a <code>Document</code>
  <var>document</var> is the result of running these steps:</p>

  <ol>
   <li><p>If <var>document</var>'s <span>node navigable</span> is null, then return null.</p></li>

   <li><p>Return <var>document</var>'s <span>node navigable</span>'s <span
   data-x="nav-container-document">container document</span>.</p></li>
  </ol>

  <p id="child-browsing-context">A <span>navigable</span> <var>navigable</var> is a <dfn
  export>child navigable</dfn> of another navigable <var>potentialParent</var> when
  <var>navigable</var>'s <span data-x="nav-parent">parent</span> is <var>potentialParent</var>. We
  can also just say that a <span>navigable</span> "is a <span>child navigable</span>", which means
  that its <span data-x="nav-parent">parent</span> is non-null.</p>

  <p class="note">All <span data-x="child navigable">child navigables</span> are the <span>content
  navigable</span> of their <span data-x="nav-container">container</span>.</p>

  <p>The <dfn id="concept-bcc-content-document">content document</dfn> of a <span>navigable
  container</span> <var>container</var> is the result of running these steps:</p>

  <ol>
   <li><p>If <var>container</var>'s <span>content navigable</span> is null, then return
   null.</p></li>

   <li><p>Let <var>document</var> be <var>container</var>'s <span>content navigable</span>'s <span
   data-x="nav-document">active document</span>.</p></li>

   <li><p>If <var>document</var>'s <span data-x="concept-document-origin">origin</span> and
   <var>container</var>'s <span>node document</span>'s <span
   data-x="concept-document-origin">origin</span> are not <span>same origin-domain</span>, then
   return null.</p></li>

   <li><p>Return <var>document</var>.</p></li>
  </ol>

  <p>The <dfn>content window</dfn> of a <span>navigable container</span> <var>container</var> is the
  result of running these steps:</p>

  <ol>
   <li><p>If <var>container</var>'s <span>content navigable</span> is null, then return
   null.</p></li>

   <li><p>Return <var>container</var>'s <span>content navigable</span>'s <span
   data-x="nav-wp">active <code>WindowProxy</code></span>'s object.</p></li>
  </ol>

  <hr>

  <p id="creating-a-new-nested-browsing-context">To <dfn>create a new child navigable</dfn>, given
  an element <var>element</var>:</p>

  <ol>
   <li><p>Let <var>parentNavigable</var> be <var>element</var>'s <span>node
   navigable</span>.</p></li>

   <li><p>Let <var>group</var> be <var>element</var>'s <span>node document</span>'s <span
   data-x="concept-document-bc">browsing context</span>'s <span data-x="bc-tlbc">top-level browsing
   context</span>'s <span data-x="tlbc group">group</span>.</p></li>

   <li><p>Let <var>browsingContext</var> and <var>document</var> be the result of <span>creating a
   new browsing context and document</span> given <var>element</var>'s <span>node document</span>,
   <var>element</var>, and <var>group</var>.</p></li>

   <li><p>Let <var>targetName</var> be null.</p></li>

   <li><p>If <var>element</var> has a <code data-x="">name</code> content attribute, then set
   <var>targetName</var> to the value of that attribute.</p></li>

   <li>
    <p>Let <var>documentState</var> be a new <span>document state</span>, with</p>

    <dl class="props">
     <dt><span data-x="document-state-document">document</span></dt>
     <dd><var>document</var></dd>

     <dt><span data-x="document-state-initiator-origin">initiator origin</span></dt>
     <dd><var>document</var>'s <span data-x="concept-document-origin">origin</span></dd>

     <dt><span data-x="document-state-origin">origin</span></dt>
     <dd><var>document</var>'s <span data-x="concept-document-origin">origin</span></dd>

     <dt><span data-x="document-state-nav-target-name">navigable target name</span></dt>
     <dd><var>targetName</var></dd>

     <dt><span data-x="document-state-about-base-url">about base URL</span></dt>
     <dd><var>document</var>'s <span data-x="concept-document-about-base-url">about base
     URL</span></dd>
    </dl>
   </li>

   <li><p>Let <var>navigable</var> be a new <span>navigable</span>.</p></li>

   <li><p><span>Initialize the navigable</span> <var>navigable</var> given <var>documentState</var>
   and <var>parentNavigable</var>.</p></li>

   <li><p>Set <var>element</var>'s <span>content navigable</span> to <var>navigable</var>.</p></li>

   <li><p>Let <var>historyEntry</var> be <var>navigable</var>'s <span
   data-x="nav-active-history-entry">active session history entry</span>.</p></li>

   <li><p>Let <var>traversable</var> be <var>parentNavigable</var>'s <span
   data-x="nav-traversable">traversable navigable</span>.</p></li>

   <li>
    <p><span data-x="tn-append-session-history-traversal-steps">Append the following session
    history traversal steps</span> to <var>traversable</var>:</p>

    <ol>
     <li><p>Let <var>parentDocState</var> be <var>parentNavigable</var>'s <span
     data-x="nav-active-history-entry">active session history entry</span>'s <span
     data-x="she-document-state">document state</span>.</p></li>

     <li><p>Let <var>parentNavigableEntries</var> be the result of <span>getting session history
     entries</span> for <var>parentNavigable</var>.</p></li>

     <li><p>Let <var>targetStepSHE</var> be the first <span>session history entry</span> in
     <var>parentNavigableEntries</var> whose <span data-x="she-document-state">document
     state</span> equals <var>parentDocState</var>.</p></li>

     <li><p>Set <var>historyEntry</var>'s <span data-x="she-step">step</span> to
     <var>targetStepSHE</var>'s <span data-x="she-step">step</span>.</p></li>

     <li><p>Let <var>nestedHistory</var> be a new <span>nested history</span> whose <span
     data-x="nested-history-id">id</span> is <var>navigable</var>'s <span data-x="nav-id">id</span>
     and <span data-x="nested-history-entries">entries list</span> is « <var>historyEntry</var>
     ».</p></li>

     <li><p><span data-x="list append">Append</span> <var>nestedHistory</var> to
     <var>parentDocState</var>'s <span data-x="document-state-nested-histories">nested
     histories</span>.</p></li>

     <li><p><span>Update for navigable creation/destruction</span> given
     <var>traversable</var>.</p></li>
    </ol>
   </li>

   <li><p>Invoke <span>WebDriver BiDi navigable created</span> with <var>traversable</var>.</p></li>
  </ol>


  <h5>Jake diagrams</h5>

  <p>A useful method for visualizing sequences of documents, and in particular <span
  data-x="navigable">navigables</span> and their <span data-x="session history entry">session
  history entries</span>, is the <dfn>Jake diagram</dfn>. A typical Jake diagram is the
  following:</p>

  <!--https://domenic.github.io/jake-diagram-generator/#IWN1cnJlbnQgPSAxCnRvcAogIDAtMjogL3QtYSB8IGRvYzEKICAzOiAvdC1hI2ZvbyB8IGRvYzEKICA0OiAvdC1iCmZyYW1lc1swXQogIDA6IC9pLTAtYQogIDEtMzogL2ktMC1iCmZyYW1lc1sxXQogIDAtMTogL2ktMS1hCiAgMi0zOiAvaS0xLWIK -->
  <table class="jake-diagram"><thead><tr><td></td><th class="step">0</th><th class="step current">1</th><th class="step">2</th><th class="step">3</th><th class="step">4</th></tr></thead><tbody><tr><th><code data-x="">top</code></th><td colspan="3" class="doc-0 current next-is-same-doc">/t-a</td><td colspan="1" class="doc-0 prev-is-same-doc">/t-a#foo</td><td colspan="1" class="doc-1">/t-b</td></tr><tr><th><code data-x="">frames[0]</code></th><td colspan="1" class="doc-2">/i-0-a</td><td colspan="3" class="doc-3 current">/i-0-b</td></tr><tr><th><code data-x="">frames[1]</code></th><td colspan="2" class="doc-4 current">/i-1-a</td><td colspan="2" class="doc-5">/i-1-b</td></tr></tbody></table>

  <p>Here, each numbered column denotes a possible value for the traversable's <span
  data-x="tn-current-session-history-step">session history step</span>. Each labeled row depicts a
  <span>navigable</span>, as it transitions between different URLs and documents. The first,
  labeled <code data-x="">top</code>, being the <span>top-level traversable</span>, and the others
  being <span data-x="child navigable">child navigables</span>. The documents are given by the
  background color of each cell, with a new background color indicating a new document in that
  <span>navigable</span>. The URLs are given by the text content of the cells; usually they are
  given as <span data-x="relative URL">relative URLs</span> for brevity, unless a cross-origin case
  is specifically under investigation. A given navigable might not exist at a given step, in which
  case the corresponding cells are empty. The bold-italic step number depicts the <span
  data-x="tn-current-session-history-step">current session history step</span> of the traversable,
  and all cells with bold-italic URLs represent the <span
  data-x="nav-current-history-entry">current session history entry</span> for that row's
  navigable.</p>

  <p>Thus, the above Jake diagram depicts the following sequence of events:</p>

  <ol start="0">
   <li><p>A <span>top-level traversable</span> is created, starting a the URL <code
   data-x="">/t-a</code>, with two <span data-x="child navigable">child navigables</span>
   starting at <code data-x="">/i-0-a</code> and <code data-x="">/i-1-a</code>
   respectively.</p></li>

   <li><p>The first child navigable is <span data-x="navigate">navigated</span> to another
   document, with URL <code data-x="">/i-0-b</code>.</p></li>

   <li><p>The second child navigable is <span data-x="navigate">navigated</span> to another
   document, with URL <code data-x="">/i-1-b</code>.</p></li>

   <li><p>The top-level traversable is <span data-x="navigate">navigated</span> to the
   <em>same</em> document, updating its URL to <code data-x="">/t-a#foo</code>.</p></li>

   <li><p>The top-level traversable is <span data-x="navigate">navigated</span> to another
   document, with URL <code data-x="">/t-b</code>. (Notice how this document, of course, does not
   carry over the old document's child navigables.)</p></li>

   <li><p>The traversable was <span data-x="traverse the history by a delta">traversed by a
   delta</span> of &minus;3, back to step 1.</p></li>
  </ol>

  <p><span data-x="Jake diagram">Jake diagrams</span> are a powerful tool for visualizing the
  interactions of multiple navigables, navigations, and traversals. They cannot capture every
  possible interaction &mdash; for example, they only work with a single level of nesting &mdash;
  but we will have ocassion to use them to illustrate several complex situations throughout this
  standard.</p>

  <p class="note"><span data-x="Jake diagram">Jake diagrams</span> are named after their creator,
  the inimitable Jake Archibald.</p>


  <h5>Related navigable collections</h5>

  <p>It is often helpful in this standard's algorithms to look at collections of <span
  data-x="navigable">navigables</span> starting at a given <code>Document</code>. This section
  contains a curated set of algorithms for collecting those navigables.</p>

  <p class="note">The return values of these algorithms are ordered so that parents appears before
  their children. Callers rely on this ordering.</p>

  <p class="note">Starting with a <code>Document</code>, rather than a <span>navigable</span>, is
  generally better because it makes the caller cognizant of whether they are starting with a
  <span>fully active</span> <code>Document</code> or not. Although non-<span>fully active</span>
  <code>Document</code>s do have ancestor and descendant navigables, they often behave as if they
  don't (e.g., in the <code data-x="dom-parent">window.parent</code> getter).</p>

  <p>The <dfn export for="Document">ancestor navigables</dfn> of a <code>Document</code>
  <var>document</var> are given by these steps:</p>

  <ol>
   <li><p>Let <var>navigable</var> be <var>document</var>'s <span>node navigable</span>'s <span
   data-x="nav-parent">parent</span>.</p></li>

   <li><p>Let <var>ancestors</var> be an empty list.</p>

   <li>
    <p>While <var>navigable</var> is not null:</p>

    <ol>
     <li><p><span data-x="list prepend">Prepend</span> <var>navigable</var> to
     <var>ancestors</var>.</p></li>

     <li><p>Set <var>navigable</var> to <var>navigable</var>'s <span
     data-x="nav-parent">parent</span>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>ancestors</var>.</p></li>
  </ol>

  <p>The <dfn export for="Document">inclusive ancestor navigables</dfn> of a <code>Document</code>
  <var>document</var> are given by these steps:</p>

  <ol>
   <li><p>Let <var>navigables</var> be <var>document</var>'s <span>ancestor
   navigables</span>.</p></li>

   <li><p><span data-x="list append">Append</span> <var>document</var>'s <span>node
   navigable</span> to <var>navigables</var>.</p></li>

   <li><p>Return <var>navigables</var>.</p></li>
  </ol>

  <p id="list-of-the-descendant-browsing-contexts">The <dfn export for="Document">descendant
  navigables</dfn> of a <code>Document</code> <var>document</var> are given by these steps:</p>

  <ol>
   <li><p>Let <var>navigables</var> be new <span>list</span>.</p></li>

   <li><p>Let <var>navigableContainers</var> be a <span>list</span> of all <span
   data-x="shadow-including descendant">shadow-including descendants</span> of <var>document</var>
   that are <span data-x="navigable container">navigable containers</span>, in
   <span>shadow-including tree order</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>navigableContainer</var> of
    <var>navigableContainers</var>:</p>

    <ol>
     <li><p>If <var>navigableContainer</var>'s <span>content navigable</span> is null, then
     continue.</p></li>

     <li><p><span data-x="list extend">Extend</span> <var>navigables</var> with
     <var>navigableContainer</var>'s <span>content navigable</span>'s <span
     data-x="nav-document">active document</span>'s <span>inclusive descendant
     navigables</span>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>navigables</var>.</p></li>
  </ol>

  <p>The <dfn export for="Document">inclusive descendant navigables</dfn> of a
  <code>Document</code> <var>document</var> are given by these steps:</p>

  <ol>
   <li><p>Let <var>navigables</var> be « <var>document</var>'s <span>node navigable</span>
   ».</p></li>

   <li><p><span data-x="list extend">Extend</span> <var>navigables</var> with <var>document</var>'s
   <span>descendant navigables</span>.</p></li>

   <li><p>Return <var>navigables</var>.</p></li>
  </ol>

  <p class="note">These descendant-collecting algorithms are described as looking at the DOM tree
  of descendant <code>Document</code> objects. In reality, this is often not feasible since the DOM
  tree can be in another process from the caller of the algorithm. Instead, implementations
  generally replicate the appropriate trees across processes.</p>

  <p id="document-tree-child-browsing-context">The <dfn>document-tree child navigables</dfn> of a
  <code>Document</code> <var>document</var> are given by these steps:</p>

  <ol>
   <li><p>If <var>document</var>'s <span>node navigable</span> is null, then return the empty
   list.</p></li>

   <li><p>Let <var>navigables</var> be new <span>list</span>.</p></li>

   <li><p>Let <var>navigableContainers</var> be a <span>list</span> of all <span
   data-x="descendant">descendants</span> of <var>document</var> that are <span data-x="navigable
   container">navigable containers</span>, in <span>tree order</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>navigableContainer</var> of
    <var>navigableContainers</var>:</p>

    <ol>
     <li><p>If <var>navigableContainer</var>'s <span>content navigable</span> is null, then
     <span>continue</span>.</p></li>

     <li><p><span data-x="list append">Append</span> <var>navigableContainer</var>'s <span>content
     navigable</span> to <var>navigables</var>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>navigables</var>.</p></li>
  </ol>


  <h5 id="garbage-collection-and-browsing-contexts">Navigable destruction</h5>

  <p>To <dfn>destroy a child navigable</dfn> given a <span>navigable container</span>
  <var>container</var>:</p>

  <ol>
   <li><p>Let <var>navigable</var> be <var>container</var>'s <span>content
   navigable</span>.</p></li>

   <li><p>If <var>navigable</var> is null, then return.</p></li>

   <li><p>Set <var>container</var>'s <span>content navigable</span> to null.</p></li>

   <li><p><span>Inform the navigation API about child navigable destruction</span> given
   <var>navigable</var>.</p></li>

   <li><p><span>Destroy a document and its descendants</span> given <var>navigable</var>'s <span
   data-x="nav-document">active document</span>.</p></li>

   <li><p>Let <var>parentDocState</var> be <var>container</var>'s <span>node navigable</span>'s
   <span data-x="nav-active-history-entry">active session history entry</span>'s <span
   data-x="she-document-state">document state</span>.</p></li>

   <li><p><span data-x="list remove">Remove</span> the <span>nested history</span> from
   <var>parentDocState</var>'s <span data-x="document-state-nested-histories">nested
   histories</span> whose <span data-x="nested-history-id">id</span> equals
   <var>navigable</var>'s <span data-x="nav-id">id</span>.</p></li>

   <li><p>Let <var>traversable</var> be <var>container</var>'s <span>node navigable</span>'s <span
   data-x="nav-traversable">traversable navigable</span>.</p></li>

   <li>
    <p><span data-x="tn-append-session-history-traversal-steps">Append the following session
    history traversal steps</span> to <var>traversable</var>:</p>

    <ol>
     <li><p><span>Update for navigable creation/destruction</span> given
     <var>traversable</var>.</p></li>
    </ol>
   </li>

   <li><p>Invoke <span>WebDriver BiDi navigable destroyed</span> with <var>navigable</var>.</p></li>
  </ol>

  <p>To <dfn data-x="destroy a top-level traversable">destroy</dfn> a <span>top-level
  traversable</span> <var>traversable</var>:</p>

  <ol>
   <li><p>Let <var>browsingContext</var> be <var>traversable</var>'s <span
   data-x="nav-bc">active browsing context</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>historyEntry</var> in
    <var>traversable</var>'s <span data-x="tn-session-history-entries">session history
    entries</span> <span class="XXX" data-x="">in what order?</span>:</p>

    <ol>
     <li><p>Let <var>document</var> be <var>historyEntry</var>'s <span
     data-x="she-document">document</span>.</p></li>

     <li><p>If <var>document</var> is not null, then <span>destroy a document and its
     descendants</span> given <var>document</var>.</p></li>
    </ol>
   </li>

   <li><p><span data-x="bcg remove">Remove</span> <var>browsingContext</var>.</p></li>

   <li><p>Remove <var>traversable</var> from the user interface (e.g., close or hide its tab in
   a tabbed browser).</p></li>

   <li><p><span data-x="list remove">Remove</span> <var>traversable</var> from the user agent's
   <span>top-level traversable set</span>.</p></li>

   <li><p>Invoke <span>WebDriver BiDi navigable destroyed</span> with <var>traversable</var>.</p></li>
  </ol>

  <p>User agents may <span>destroy a top-level traversable</span> at any time (typically, <a
  href="#nav-traversal-ui">in response to user requests</a>).</p>

  <p id="closing-browsing-contexts"><span id="close-a-browsing-context"></span>To <dfn export
  id="close-a-top-level-traversable" for="top-level traversable" data-x="close a top-level
  traversable">close</dfn> a <span>top-level traversable</span> <var>traversable</var>:</p>

  <ol>
   <li><p>If <var>traversable</var>'s <span>is closing</span> is true, then return.</p></li>

   <li><p><span data-x="definitely close a top-level traversable">Definitely close</span>
   <var>traversable</var>.</p></li>
  </ol>

  <p>To <dfn data-x="definitely close a top-level traversable">definitely close</dfn> a
  <span>top-level traversable</span> <var>traversable</var>:</p>

  <ol>
   <li><p>Let <var>toUnload</var> be <var>traversable</var>'s <span data-x="nav-document">active
   document</span>'s <span>inclusive descendant navigables</span>.</p></li>

   <li><p>If the result of <span>checking if unloading is canceled</span> for <var>toUnload</var> is
   true, then return.</p></li>

   <li>
    <p><span data-x="tn-append-session-history-traversal-steps">Append the following session history
    traversal steps</span> to <var>traversable</var>:</p>

    <ol>
     <li><p>Let <var>afterAllUnloads</var> be an algorithm step which <span data-x="destroy a
     top-level traversable">destroys</span> <var>traversable</var>.</p></li>

     <li><p><span>Unload a document and its descendants</span> given <var>traversable</var>'s <span
     data-x="nav-document">active document</span>, null, and <var>afterAllUnloads</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p class="note">The <span data-x="close a top-level traversable">close</span> vs. <span
  data-x="definitely close a top-level traversable">definitely close</span> separation allows other
  specifications to call <span data-x="close a top-level traversable">close</span> and have it be a
  no-op if the top-level traversable is already closing due to JavaScript code calling <code
  data-x="dom-window-close">window.close()</code>.</p>


  <h5><span id="browsing-context-names"></span>Navigable target names</h5>

  <p><span data-x="navigable">Navigables</span> can be given <span data-x="nav-target">target
  names</span>, which are strings allowing certain APIs (such as <code
  data-x="dom-open">window.open()</code> or the <code>a</code> element's <code
  data-x="attr-hyperlink-target">target</code> attribute) to target <span
  data-x="navigate">navigations</span> at that navigable.</p>

  <p id="valid-browsing-context-name">A <dfn>valid navigable target name</dfn> is any string with at
  least one character that does not contain both an <span>ASCII tab or newline</span> and a U+003C
  (&lt;), and it does not start with a U+005F (_). (Names starting with a U+005F (_) are reserved
  for special keywords.)</p>

  <p id="valid-browsing-context-name-or-keyword">A <dfn>valid navigable target name or
  keyword</dfn> is any string that is either a <span>valid navigable target name</span> or that is
  an <span>ASCII case-insensitive</span> match for one of: <code data-x="">_blank</code>, <code
  data-x="">_self</code>, <code data-x="">_parent</code>, or <code data-x="">_top</code>.</p>

  <p>These values have different meanings based on whether the page is sandboxed or not, as
  summarized in the following (non-normative) table. In this table, "current" means the
  <span>navigable</span> that the link or script is in, "parent" means the <span
  data-x="nav-parent">parent</span> of the <span>navigable</span> that the link or script is in,
  "top" means the <span data-x="nav-top">top-level traversable</span> of the <span>navigable</span>
  that the link or script is in, "new" means a new <span>traversable navigable</span> with a null
  <span data-x="nav-parent">parent</span> (which may use an <span>auxiliary browsing
  context</span>, subject to various user preferences and user agent policies), "none" means that
  nothing will happen, and "maybe new" means the same as "new" if the "<code
  data-x="attr-iframe-sandbox-allow-popups">allow-popups</code>" keyword is also specified on the
  <code data-x="attr-iframe-sandbox">sandbox</code> attribute (or if the user overrode the
  sandboxing), and the same as "none" otherwise.</p>

  <table>
   <thead>
    <tr>
     <th rowspan=2>Keyword
     <th rowspan=2>Ordinary effect
     <th colspan=2>Effect in an <code>iframe</code> with...
    <tr>           <!-- nothing -->
     <th><code data-x="">sandbox=""</code>
     <th><code data-x="">sandbox="allow-top-navigation"</code>

   <tbody>
    <tr>
     <td>none specified, for links and form submissions <!-- same as empty string -->
     <td>current
     <td>current
     <td>current

    <tr>
     <td>empty string
     <td>current
     <td>current
     <td>current

    <tr>
     <td><code data-x="">_blank</code>
     <td>new
     <td>maybe new
     <td>maybe new

    <tr>
     <td><code data-x="">_self</code>
     <td>current
     <td>current
     <td>current

    <tr>
     <td><code data-x="">_parent</code> if there isn't a parent
     <td>current
     <td>current
     <td>current

    <tr>
     <td><code data-x="">_parent</code> if parent is also top
     <td>parent/top
     <td>none
     <td>parent/top

    <tr>
     <td><code data-x="">_parent</code> if there is one and it's not top
     <td>parent
     <td>none
     <td>none

    <tr>
     <td><code data-x="">_top</code> if top is current
     <td>current
     <td>current
     <td>current

    <tr>
     <td><code data-x="">_top</code> if top is not current
     <td>top
     <td>none
     <td>top

    <tr>
     <td>name that doesn't exist
     <td>new
     <td>maybe new
     <td>maybe new

    <tr>
     <td>name that exists and is a descendant
     <td>specified descendant
     <td>specified descendant
     <td>specified descendant

    <tr>
     <td>name that exists and is current
     <td>current
     <td>current
     <td>current

    <tr>
     <td>name that exists and is an ancestor that is top
     <td>specified ancestor
     <td>none
     <td>specified ancestor/top

    <tr>
     <td>name that exists and is an ancestor that is not top
     <td>specified ancestor
     <td>none
     <td>none

    <tr>
     <td>other name that exists with common top
     <td>specified
     <td>none
     <td>none

    <tr>
     <td>name that exists with different top, if <span data-x="familiar with">familiar</span> and <span>one permitted sandboxed navigator</span>
     <td>specified
     <td>specified
     <td>specified

    <tr>
     <td>name that exists with different top, if <span data-x="familiar with">familiar</span> but not <span>one permitted sandboxed navigator</span>
     <td>specified
     <td>none
     <td>none

    <tr>
     <td>name that exists with different top, not <span data-x="familiar with">familiar</span>
     <td>new
     <td>maybe new
     <td>maybe new

  </table>

  <p class="tablenote"><small>Most of the restrictions on sandboxed browsing contexts are applied by
  other algorithms, e.g. the <span data-x="navigate">navigation</span> algorithm, not <span>the
  rules for choosing a navigable</span> given below.</small></p>

  <hr>

  <p>To <dfn>find a navigable by target name</dfn> given a string <var>name</var> and a
  <span>navigable</span> <var>currentNavigable</var>:</p>

  <ol>
   <li><p>Let <var>currentDocument</var> be <var>currentNavigable</var>'s <span
   data-x="nav-document">active document</span>.</p></li>

   <li><p>Let <var>sourceSnapshotParams</var> be the result of <span>snapshotting source snapshot
   params</span> given <var>currentDocument</var>.</p></li>

   <li>
    <p>Let <var>subtreesToSearch</var> be an <span>implementation-defined</span> choice of one of
    the following:</p>

    <ul>
     <li><p>« <var>currentNavigable</var>'s <span data-x="nav-traversable">traversable
     navigable</span>, <var>currentNavigable</var> »</p></li>

     <li><p>the <span>inclusive ancestor navigables</span> of <var>currentDocument</var></p></li>
    </ul>

    <p class="XXX"><a href="https://github.com/whatwg/html/issues/10848">Issue #10848</a> tracks
    settling on one of these two possibilities, to achieve interoperability.</p>
   </li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>subtreeToSearch</var> of
    <var>subtreesToSearch</var>, in reverse order:</p>

    <ol>
     <li><p>Let <var>documentToSearch</var> be <var>subtreeToSearch</var>'s <span
     data-x="nav-document">active document</span>.</p></li>

     <li>
      <p><span data-x="list iterate">For each</span> <var>navigable</var> of the <span>inclusive
      descendant navigables</span> of <var>documentToSearch</var>:</p>

      <ol>
       <li>
        <p>If <var>currentNavigable</var> is not <span>allowed by sandboxing to navigate</span>
        <var>navigable</var> given <var>sourceSnapshotParams</var>, then optionally
        <span>continue</span>.</p>

        <p class="XXX"><a href="https://github.com/whatwg/html/issues/10849">Issue #10849</a> tracks
        making this check required, to achieve interoperability.</p>
       </li>

       <li><p>If <var>navigable</var>'s <span data-x="nav-target">target name</span> is
       <var>name</var>, then return <var>navigable</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Let <var>currentTopLevelBrowsingContext</var> be <var>currentNavigable</var>'s <span
   data-x="nav-bc">active browsing context</span>'s <span data-x="bc-tlbc">top-level browsing
   context</span>.</p></li>

   <li><p>Let <var>group</var> be <var>currentTopLevelBrowsingContext</var>'s <span data-x="tlbc
   group">group</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>topLevelBrowsingContext</var> of
    <var>group</var>'s <span>browsing context set</span>, in an <span>implementation-defined</span>
    order (the user agent should pick a consistent ordering, such as the most recently opened, most
    recently focused, or more closely related):</p>

    <p class="XXX"><a href="https://github.com/whatwg/html/issues/10850">Issue #10850</a> tracks
    picking a specific ordering, to achieve interoperability.</p>

    <ol>
     <li><p>If <var>currentTopLevelBrowsingContext</var> is <var>topLevelBrowsingContext</var>, then
     <span>continue</span>.</p></li>

     <li><p>Let <var>documentToSearch</var> be <var>topLevelBrowsingContext</var>'s <span>active
     document</span>.</p></li>

     <li>
      <p><span data-x="list iterate">For each</span> <var>navigable</var> of the <span>inclusive
      descendant navigables</span> of <var>documentToSearch</var>:</p>

      <ol>
       <li><p>If <var>currentNavigable</var>'s <span data-x="nav-bc">active browsing context</span>
       is not <span>familiar with</span> <var>navigable</var>'s <span data-x="nav-bc">active
       browsing context</span>, then <span>continue</span>.</p></li>

       <li>
        <p>If <var>currentNavigable</var> is not <span>allowed by sandboxing to navigate</span>
        <var>navigable</var> given <var>sourceSnapshotParams</var>, then optionally
        <span>continue</span>.</p>


        <p class="XXX"><a href="https://github.com/whatwg/html/issues/10849">Issue #10849</a> tracks
        making this check required, to achieve interoperability.</p>
       </li>

       <li><p>If <var>navigable</var>'s <span data-x="nav-target">target name</span> is
       <var>name</var>, then return <var>navigable</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Return null.</p></li>
  </ol>

  <p id="the-rules-for-choosing-a-browsing-context-given-a-browsing-context-name"><dfn>The rules
  for choosing a navigable</dfn>, given a string <var>name</var>, a <span>navigable</span>
  <var>currentNavigable</var>, and a boolean <var>noopener</var> are as follows:</p>

  <ol>
   <li><p>Let <var>chosen</var> be null.</p></li>

   <li><p>Let <var>windowType</var> be "<code data-x="">existing or none</code>".</p></li>

   <li><p>Let <var>sandboxingFlagSet</var> be <var>currentNavigable</var>'s <span
   data-x="nav-document">active document</span>'s <span>active sandboxing flag set</span>.</p></li>

   <li><p>If <var>name</var> is the empty string or an <span>ASCII case-insensitive</span> match
   for "<code data-x="">_self</code>", then set <var>chosen</var> to
   <var>currentNavigable</var>.</p></li>

   <li><p>Otherwise, if <var>name</var> is an <span>ASCII case-insensitive</span> match for "<code
   data-x="">_parent</code>", set <var>chosen</var> to <var>currentNavigable</var>'s <span
   data-x="nav-parent">parent</span>, if any, and <var>currentNavigable</var> otherwise.</p></li>

   <li><p>Otherwise, if <var>name</var> is an <span>ASCII case-insensitive</span> match for "<code
   data-x="">_top</code>", set <var>chosen</var> to <var>currentNavigable</var>'s <span
   data-x="nav-traversable">traversable navigable</span>.</p></li>

   <li><p>Otherwise, if <var>name</var> is not an <span>ASCII case-insensitive</span> match for
   "<code data-x="">_blank</code>", and there exists a <span>navigable</span> that is the result of
   <span data-x="find a navigable by target name">finding a navigable by target name</span> given
   <var>name</var> and <var>currentNavigable</var>, set <var>chosen</var> to that
   navigable.</p></li>

   <li>
    <p>Otherwise, a new <span>top-level traversable</span> is being requested, and what happens
    depends on the user agent's configuration and abilities &mdash; it is determined by the rules
    given for the first applicable option from the following list:</p>

    <dl class="switch">
     <dt id="popup-blocker">If <var>currentNavigable</var>'s <span data-x="nav-window">active
     window</span> does not have <span>transient activation</span> and the user agent has been
     configured to not show popups (i.e., the user agent has a "popup blocker" enabled)</dt>

     <dd><p>The user agent may inform the user that a popup has been blocked.</p></dd>

     <dt id="sandboxWindowOpen">If <var>sandboxingFlagSet</var> has the <span>sandboxed auxiliary
     navigation browsing context flag</span> set</dt>

     <dd><p>The user agent may report to a developer console that a popup has been blocked.</p></dd>

     <dt>If the user agent has been configured such that in this instance it will create a new <span>top-level traversable</span></dt>

     <dd>
      <ol>
       <li><p><span>Consume user activation</span> of <var>currentNavigable</var>'s <span
       data-x="nav-window">active window</span>.</p></li>

       <li><p>Set <var>windowType</var> to "<code data-x="">new and unrestricted</code>".</p></li>

       <li><p>Let <var>currentDocument</var> be <var>currentNavigable</var>'s <span
       data-x="nav-document">active document</span>.</p></li>

       <li>
        <p>If <var>currentDocument</var>'s <span data-x="concept-document-coop">opener
        policy</span>'s <span data-x="coop-struct-value">value</span> is "<code
        data-x="coop-same-origin">same-origin</code>" or "<code
        data-x="coop-same-origin-plus-coep">same-origin-plus-COEP</code>", and
        <var>currentDocument</var>'s <span data-x="concept-document-origin">origin</span> is not
        <span>same origin</span> with <var>currentDocument</var>'s <span>relevant settings
        object</span>'s <span>top-level origin</span>, then:</p>

        <ol>
         <li><p>Set <var>noopener</var> to true.</p></li>

         <li><p>Set <var>name</var> to "<code data-x="">_blank</code>".</p></li>

         <li><p>Set <var>windowType</var> to "<code data-x="">new with no opener</code>".</p>
        </ol>

        <p class="note">In the presence of an <span>opener policy</span>, nested documents that are
        cross-origin with their top-level browsing context's active document always set
        <var>noopener</var> to true.</p>
       </li>

       <li><p>Let <var>chosen</var> be null.</p></li>

       <li><p>Let <var>targetName</var> be the empty string.</p></li>

       <li><p>If <var>name</var> is not an <span>ASCII case-insensitive</span> match for "<code
       data-x="">_blank</code>", then set <var>targetName</var> to <var>name</var>.</p></li>

       <li id="noopener"><p>If <var>noopener</var> is true, then set <var>chosen</var> to the
       result of <span>creating a new top-level traversable</span> given null,
       <var>targetName</var>, and <var>currentNavigable</var>.</p></li>

       <li>
        <p>Otherwise:</p>

        <ol>
         <li><p>Set <var>chosen</var> to the result of <span>creating a new top-level
         traversable</span> given <var>currentNavigable</var>'s <span data-x="nav-bc">active
         browsing context</span>, <var>targetName</var>, and <var>currentNavigable</var>.</p></li>

         <li><p>If <var>sandboxingFlagSet</var>'s <span>sandboxed navigation browsing context
         flag</span> is set, then set <var>chosen</var>'s <span data-x="nav-bc">active browsing
         context</span>'s <span>one permitted sandboxed navigator</span> to
         <var>currentNavigable</var>'s <span data-x="nav-bc">active browsing
         context</span>.</p></li>
        </ol>
       </li>

       <li><p>If <var>sandboxingFlagSet</var>'s <span>sandbox propagates to auxiliary browsing
       contexts flag</span> is set, then all the flags that are set in <var>sandboxingFlagSet</var>
       must be set in <var>chosen</var>'s <span data-x="nav-bc">active browsing context</span>'s
       <span>popup sandboxing flag set</span>.</p></li>
      </ol>

      <p class="note">If the newly created <span>navigable</span> <var>chosen</var> is immediately
      <span data-x="navigate">navigated</span>, then the navigation will be done as a "<code
      data-x="NavigationHistoryBehavior-replace">replace</code>" navigation.</p>
     </dd>

     <dt>If the user agent has been configured such that in this instance it will choose <var>currentNavigable</var></dt>

     <dd><p>Set <var>chosen</var> to <var>currentNavigable</var>.</p></dd>

     <dt>If the user agent has been configured such that in this instance it will not find a navigable</dt>

     <dd><p>Do nothing.</p></dd>
    </dl>

    <p class="note">User agents are encouraged to provide a way for users to configure the user
    agent to always choose <var>currentNavigable</var>.</p>
   </li>

   <li><p>Return <var>chosen</var> and <var>windowType</var>.</p></li>
  </ol>


  <h4 id="windows">Browsing contexts</h4>

  <p>A <dfn export>browsing context</dfn> is a programmatic representation of a series of documents,
  multiple of which can live within a single <span>navigable</span>. Each <span>browsing
  context</span> has a corresponding <code>WindowProxy</code> object, as well as the following:</p>

  <ul>
   <li><p>An <dfn export>opener browsing context</dfn>, a <span>browsing context</span> or null,
   initially null.</p></li>

   <li><p>An <dfn data-x="opener-origin-at-creation">opener origin at creation</dfn>, an
   <span>origin</span> or null, initially null.</p>

   <li>
    <p>An <dfn>is popup</dfn> boolean, initially false.</p>

    <p class="note">The only mandatory impact in this specification of <span>is popup</span> is on
    the <code data-x="dom-BarProp-visible">visible</code> getter of the relevant
    <code>BarProp</code> objects. However, user agents might also use it for <a
    href="#nav-traversal-ui">user interface considerations</a>.</p>
   </li>

   <li><p>An <dfn>is auxiliary</dfn> boolean, initially false.</p></li>

   <li><p>An <dfn data-x="browsing-context-initial-url">initial URL</dfn>, a <span>URL</span> or
   null, initially null.</p></li>

   <li><p>A <dfn data-x="virtual-browsing-context-group-id">virtual browsing context group ID</dfn>
   integer, initially 0. This is used by <span data-x="coop-struct-report-only-value">opener policy
   reporting</span>, to keep track of the browsing context group switches that would have happened
   if the report-only policy had been enforced.</p></li>
  </ul>

  <p>A <span>browsing context</span>'s <dfn>active window</dfn> is its <code>WindowProxy</code>
  object's <span data-x="concept-windowproxy-window">[[Window]]</span> internal slot value. A
  <span>browsing context</span>'s <dfn>active document</dfn> is its <span>active window</span>'s
  <span data-x="concept-document-window">associated <code>Document</code></span>.</p>

  <p>A <span>browsing context</span>'s <dfn export for="browsing context"
  data-x="bc-traversable">top-level traversable</dfn> is its <span>active document</span>'s
  <span>node navigable</span>'s <span data-x="nav-top">top-level traversable</span>.</p>

  <p>A <span>browsing context</span> whose <span>is auxiliary</span> is true is known as an <dfn
  export>auxiliary browsing context</dfn>. Auxiliary browsing contexts are always <span
  data-x="top-level browsing context">top-level browsing contexts</span>.</p>

  <p class="XXX">It's unclear whether a separate <span>is auxiliary</span> concept is necessary. In
  <a href="https://github.com/whatwg/html/issues/5680">issue #5680</a>, it is indicated that we may
  be able to simplify this by using whether or not the <span>opener browsing context</span> is
  null.</p>

  <p class="warning" id="warning-avoid-using-bcs">Modern specifications should avoid using the
  <span>browsing context</span> concept in most cases, unless they are dealing with the subtleties
  of <a href="#browsing-context-group-switches-due-to-cross-origin-opener-policy">browsing context
  group switches</a> and <span data-x="agent cluster map">agent cluster allocation</span>. Instead,
  the <code>Document</code> and <span>navigable</span> concepts are usually more appropriate.</p>

  <hr>

  <p>A <dfn data-x="concept-document-bc" data-lt="browsing context" for="Document"
  export><code>Document</code>'s browsing context</dfn> is a <span>browsing context</span> or null,
  initially null.</p>

  <p class="note">A <code>Document</code> does not necessarily have a non-null <span
  data-x="concept-document-bc">browsing context</span>. In particular, data mining tools are likely
  to never instantiate browsing contexts. A <code>Document</code> created using an API such as <code
  data-x="dom-DOMImplementation-createDocument">createDocument()</code> never has a non-null <span
  data-x="concept-document-bc">browsing context</span>. And the <code>Document</code> originally
  created for an <code>iframe</code> element, which has since been <span data-x="node is removed
  from a document">removed from the document</span>, has no associated browsing context, since that
  browsing context was <span data-x="destroy a document">nulled out</span>.</p>

  <p class="note">In general, there is a 1-to-1 mapping from the <code>Window</code> object to the
  <code>Document</code> object, as long as the <code>Document</code> object has a non-null <span
  data-x="concept-document-bc">browsing context</span>. There is one exception. A
  <code>Window</code> can be reused for the presentation of a second <code>Document</code> in the
  same <span>browsing context</span>, such that the mapping is then 1-to-2. This occurs when a
  <span>browsing context</span> is <span data-x="navigate">navigated</span> from the <span
  data-x="is initial about:blank">initial <code>about:blank</code></span> <code>Document</code> to
  another, which will be done with <span
  data-x="NavigationHistoryBehavior-replace">replacement</span>.</p>

  <h5>Creating browsing contexts</h5>

  <p>To <dfn id="creating-a-new-browsing-context" data-x="creating a new browsing context and
  document">create a new browsing context and document</dfn>, given null or a <code>Document</code>
  object <var>creator</var>, null or an element <var>embedder</var>, and a <span>browsing context
  group</span> <var>group</var>:</p>

  <!-- Note: Do not append the new browsing context to group as we might be creating a nested
       browsing context and not a top level browsing context -->

  <ol>
   <li><p>Let <var>browsingContext</var> be a new <span>browsing context</span>.</p></li>

   <li><p>Let <var>unsafeContextCreationTime</var> be the
   <span>unsafe shared current time</span>.</p></li>

   <li><p>Let <var>creatorOrigin</var> be null.</p>

   <li><p>Let <var>creatorBaseURL</var> be null.</p></li>

   <li id="creator-browsing-context">
    <p>If <var>creator</var> is non-null, then:</p>

    <ol>
     <li><p>Set <var>creatorOrigin</var> to <var>creator</var>'s <span
     data-x="concept-document-origin">origin</span>.</p></li>

     <li><p>Set <var>creatorBaseURL</var> to <var>creator</var>'s <span>document base
     URL</span>.</p></li>

     <li><p>Set <var>browsingContext</var>'s <span
     data-x="virtual-browsing-context-group-id">virtual browsing context group ID</span> to
     <var>creator</var>'s <span data-x="concept-document-bc">browsing context</span>'s <span
     data-x="bc-tlbc">top-level browsing context</span>'s <span
     data-x="virtual-browsing-context-group-id">virtual browsing context group ID</span>.</p></li>
    </ol>
   </li>

   <li><p>Let <var>sandboxFlags</var> be the result of <span>determining the creation sandboxing
   flags</span> given <var>browsingContext</var> and <var>embedder</var>.</p></li>

   <li id="about-blank-origin"><p>Let <var>origin</var> be the result of <span>determining the
   origin</span> given <code>about:blank</code>, <var>sandboxFlags</var>, and
   <var>creatorOrigin</var>.</p></li>

   <li>
    <p>Let <var>permissionsPolicy</var> be the result of <span>creating a permissions policy</span>
    given <var>embedder</var> and <var>origin</var>. <ref>PERMISSIONSPOLICY</ref></p>
   </li>

   <li><p>Let <var>agent</var> be the result of <span
   data-x="obtain-similar-origin-window-agent">obtaining a similar-origin window agent</span> given
   <var>origin</var>, <var>group</var>, and false.</p></li>

   <li>
    <p>Let <var>realm execution context</var> be the result of <span>creating a new realm</span>
    given <var>agent</var> and the following customizations:</p>

    <ul>
     <li><p>For the global object, create a new <code>Window</code> object.</p></li>

     <li><p>For the global <b>this</b> binding, use <var>browsingContext</var>'s
     <code>WindowProxy</code> object.</li>
    </ul>
   </li>

   <li><p>Let <var>topLevelCreationURL</var> be <code>about:blank</code> if <var>embedder</var> is
   null; otherwise <var>embedder</var>'s <span>relevant settings object</span>'s <span>top-level
   creation URL</span>.</p></li>

   <li><p>Let <var>topLevelOrigin</var> be <var>origin</var> if <var>embedder</var> is null;
   otherwise <var>embedder</var>'s <span>relevant settings object</span>'s <span>top-level
   origin</span>.</p></li>

   <li><p><span>Set up a window environment settings object</span> with <code>about:blank</code>,
   <var>realm execution context</var>, null, <var>topLevelCreationURL</var>, and
   <var>topLevelOrigin</var>.</p></li>

   <li><p>Let <var>loadTimingInfo</var> be a new <span>document load timing info</span> with
   its <span>navigation start time</span> set to the result of calling <span>coarsen time</span>
   with <var>unsafeContextCreationTime</var> and the new <span>environment settings object</span>'s
   <span data-x="concept-settings-object-cross-origin-isolated-capability">cross-origin isolated
   capability</span>.</p></li>

   <li>
    <p>Let <var>document</var> be a new <code>Document</code>, with:</p>

    <dl class="props">
     <dt><span data-x="concept-document-type">type</span></dt>
     <dd>"<code data-x="">html</code>"</dd>

     <dt><span data-x="concept-document-content-type">content type</span></dt>
     <dd>"<code>text/html</code>"</dd>

     <dt><span data-x="concept-document-mode">mode</span></dt>
     <dd>"<code data-x="">quirks</code>"</dd>

     <dt><span data-x="concept-document-origin">origin</span></dt>
     <dd><var>origin</var></dd>

     <dt><span data-x="concept-document-bc">browsing context</span></dt>
     <dd><var>browsingContext</var>

     <dt><span data-x="concept-document-permissions-policy">permissions policy</span></dt>
     <dd><var>permissionsPolicy</var></dd>

     <dt><span>active sandboxing flag set</span></dt>
     <dd><var>sandboxFlags</var></dd>

     <dt><span>load timing info</span></dt>
     <dd><var>loadTimingInfo</var></dd>

     <dt><span>is initial <code>about:blank</code></span></dt>
     <dd>true</dd>

     <dt><span data-x="concept-document-about-base-URL">about base URL</span></dt>
     <dd><var>creatorBaseURL</var></dd>

     <dt><span data-x="concept-document-allow-declarative-shadow-roots">allow declarative shadow
     roots</span></dt>
     <dd>true</dd>
    </dl>
   </li>

   <li>
    <p>If <var>creator</var> is non-null, then:</p>

    <ol>
     <li><p>Set <var>document</var>'s <span data-x="the document's referrer">referrer</span> to the
     <span data-x="concept-url-serializer">serialization</span> of <var>creator</var>'s <span
     data-x="concept-document-url">URL</span>.</p></li>

     <li><p>Set <var>document</var>'s <span data-x="concept-document-policy-container">policy
     container</span> to a <span data-x="clone a policy container">clone</span> of
     <var>creator</var>'s <span data-x="concept-document-policy-container">policy
     container</span>.</p></li>

     <li><p>If <var>creator</var>'s <span data-x="concept-document-origin">origin</span> is
     <span>same origin</span> with <var>creator</var>'s <span>relevant settings object</span>'s
     <span>top-level origin</span>, then set <var>document</var>'s <span
     data-x="concept-document-coop">opener policy</span> to <var>creator</var>'s <span
     data-x="concept-document-bc">browsing context</span>'s <span data-x="bc-tlbc">top-level
     browsing context</span>'s <span>active document</span>'s <span
     data-x="concept-document-coop">opener policy</span>.</p></li>
    </ol>
   </li>

   <li><p><span>Assert</span>: <var>document</var>'s <span data-x="concept-document-url">URL</span>
   and <var>document</var>'s <span>relevant settings object</span>'s <span
   data-x="concept-environment-creation-url">creation URL</span> are
   <code>about:blank</code>.</p></li>

   <li><p>Mark <var>document</var> as <span>ready for post-load tasks</span>.</p></li>

   <li><p><span>Populate with <code>html</code>/<code>head</code>/<code>body</code></span> given
   <var>document</var>.</p></li>

   <li><p><span>Make active</span> <var>document</var>.</p></li>

   <li><p><span>Completely finish loading</span> <var>document</var>.</p></li>

   <li><p>Return <var>browsingContext</var> and <var>document</var>.</p></li>
  </ol>

  <p>To <dfn id="creating-a-new-top-level-browsing-context" data-x="creating a new top-level
  browsing context and document">create a new top-level browsing context and document</dfn>:</p>

  <ol>
   <li><p>Let <var>group</var> and <var>document</var> be the result of <span>creating a new
   browsing context group and document</span>.</p></li>

   <li><p>Return <var>group</var>'s <span>browsing context set</span>[0] and
   <var>document</var>.</p></li>
  </ol>

  <p>To <dfn id="creating-a-new-auxiliary-browsing-context" data-x="creating a new auxiliary
  browsing context and document">create a new auxiliary browsing context and document</dfn>, given
  a <span>browsing context</span> <var>opener</var>:</p>

  <ol>
   <li><p>Let <var>openerTopLevelBrowsingContext</var> be <var>opener</var>'s <span
   data-x="bc-traversable">top-level traversable</span>'s <span data-x="nav-bc">active browsing
   context</span>.</p></li>

   <li><p>Let <var>group</var> be <var>openerTopLevelBrowsingContext</var>'s <span data-x="tlbc
   group">group</span>.</p></li>

   <li><p><span>Assert</span>: <var>group</var> is non-null, as <span
   data-x="navigate">navigating</span> invokes this directly.</p></li>

   <li><p>Let <var>browsingContext</var> and <var>document</var> be the result of <span>creating a
   new browsing context and document</span> with <var>opener</var>'s <span
   data-x="nav-document">active document</span>, null, and <var>group</var>.</p></li>

   <li><p>Set <var>browsingContext</var>'s <span>is auxiliary</span> to true.</p></li>

   <li><p><span data-x="bcg append">Append</span> <var>browsingContext</var> to
   <var>group</var>.</p></li>

   <li><p>Set <var>browsingContext</var>'s <span>opener browsing context</span> to
   <var>opener</var>.</p></li>

   <li><p>Set <var>browsingContext</var>'s <span data-x="virtual-browsing-context-group-id">virtual
   browsing context group ID</span> to <var>openerTopLevelBrowsingContext</var>'s <span
   data-x="virtual-browsing-context-group-id">virtual browsing context group ID</span>.</p></li>

   <li><p>Set <var>browsingContext</var>'s <span data-x="opener-origin-at-creation">opener origin
   at creation</span> to <var>opener</var>'s <span data-x="nav-document">active document</span>'s
   <span data-x="concept-document-origin">origin</span>.</p></li>

   <li><p>Return <var>browsingContext</var> and <var>document</var>.</p></li>
  </ol>

  <p>To <dfn data-x="determining the origin">determine the origin</dfn>, given a <span>URL</span>
  <var>url</var>, a <span>sandboxing flag set</span> <var>sandboxFlags</var>, and an
  <span>origin</span>-or-null <var>sourceOrigin</var>:</p>

  <ol>
   <li id="sandboxOrigin"><p>If <var>sandboxFlags</var> has its <span>sandboxed origin browsing
   context flag</span> set, then return a new <span data-x="concept-origin-opaque">opaque
   origin</span>.</p></li>

   <li><p>If <var>url</var> is null, then return a new <span data-x="concept-origin-opaque">opaque
   origin</span>.</p></li>

   <li>
    <p>If <var>url</var> is <code>about:srcdoc</code>, then:</p>

    <ol>
     <li><p><span>Assert</span>: <var>sourceOrigin</var> is non-null.</p></li>

     <li><p>Return <var>sourceOrigin</var>.</p></li>
    </ol>
   </li>

   <li><p>If <var>url</var> <span>matches <code>about:blank</code></span> and
   <var>sourceOrigin</var> is non-null, then return <var>sourceOrigin</var>.</p></li>

   <li><p>Return <var>url</var>'s <span data-x="concept-url-origin">origin</span>.</p></li>
  </ol>

  <p class="note">The cases that return <var>sourceOrigin</var> result in two
  <code>Document</code>s that end up with the same underlying <span
  data-x="concept-document-origin">origin</span>, meaning that <code
  data-x="dom-document-domain">document.domain</code> affects both.</p>


  <h5 id="nested-browsing-contexts">Related browsing contexts</h5>

  <p>A <span>browsing context</span> <var>potentialDescendant</var> is said to be an <dfn
  data-x="ancestor browsing context">ancestor</dfn> of a browsing context
  <var>potentialAncestor</var> if the following algorithm returns true:</p>

  <ol>
   <li><p>Let <var>potentialDescendantDocument</var> be <var>potentialDescendant</var>'s
   <span>active document</span>.</p></li>

   <li><p>If <var>potentialDescendantDocument</var> is not <span>fully active</span>, then return
   false.</p></li>

   <li><p>Let <var>ancestorBCs</var> be the list obtained by taking the <span
   data-x="concept-document-bc">browsing context</span> of the <span data-x="nav-document">active
   document</span> of each member of <var>potentialDescendantDocument</var>'s <span>ancestor
   navigables</span>.</p></li>

   <li><p>If <var>ancestorBCs</var> <span data-x="list contains">contains</span>
   <var>potentialAncestor</var>, then return true.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  <p>A <dfn export>top-level browsing context</dfn> is a <span>browsing context</span> whose
  <span>active document</span>'s <span>node navigable</span> is a <span>traversable
  navigable</span>.</p>

  <p class="note">It is <em>not</em> required to be a <span>top-level traversable</span>.</p>

  <p>The <dfn data-x="bc-tlbc">top-level browsing context</dfn> of a <span>browsing context</span>
  <var>start</var> is the result of the following algorithm:</p>

  <ol>
   <li><p>If <var>start</var>'s <span>active document</span> is not <span>fully active</span>, then
   return null.</p></li>

   <li><p>Let <var>navigable</var> be <var>start</var>'s <span>active document</span>'s <span>node
   navigable</span>.</p></li>

   <li><p>While <var>navigable</var>'s <span data-x="nav-parent">parent</span> is not null, set
   <var>navigable</var> to <var>navigable</var>'s <span data-x="nav-parent">parent</span>.</p></li>

   <li><p>Return <var>navigable</var>'s <span data-x="nav-bc">active browsing
   context</span>.</p></li>
  </ol>

  <div class="warning" id="warning-avoid-related-bc-terms">
   <p>The terms <span>ancestor browsing context</span> and <span>top-level browsing context</span>
   are rarely useful, since <span data-x="browsing context">browsing contexts</span> in general are
   <a href="#warning-avoid-using-bcs">usually the inappropriate specification concept to use</a>.
   Note in particular that when a <span>browsing context</span>'s <span>active document</span> is
   not <span>fully active</span>, it never counts as an ancestor or top-level browsing context, and
   as such these concepts are not useful when <a href="#note-bfcache">bfcache</a> is in play.</p>

   <p>Instead, use concepts such as the <span>ancestor navigables</span> collection, the <span
   data-x="nav-parent">parent navigable</span>, or a navigable's <span
   data-x="nav-traversable">top-level traversable</span>.</p>
  </div>

  <hr>

  <p id="security-1"><span id="security-nav"></span>A <span>browsing context</span> <var>A</var> is
  <dfn>familiar with</dfn> a second <span>browsing context</span> <var>B</var> if the following
  algorithm returns true:</p>

  <ol>
   <li><p>If <var>A</var>'s <span>active document</span>'s <span
   data-x="concept-document-origin">origin</span> is <span>same origin</span> with <var>B</var>'s
   <span>active document</span>'s <span data-x="concept-document-origin">origin</span>, then return
   true.</p></li>

   <li><p>If <var>A</var>'s <span data-x="bc-tlbc">top-level browsing context</span> is
   <var>B</var>, then return true.</p></li>

   <li><p>If <var>B</var> is an <span>auxiliary browsing context</span> and <var>A</var> is
   <span>familiar with</span> <var>B</var>'s <span>opener browsing context</span>, then return
   true.</p></li>

   <li>
    <p>If there exists an <span>ancestor browsing context</span> of <var>B</var> whose <span>active
    document</span> has the <span data-x="same origin">same</span> <span
    data-x="concept-document-origin">origin</span> as the <span>active document</span> of
    <var>A</var>, then return true.</p>

    <p class="note">This includes the case where <var>A</var> is an <span>ancestor browsing
    context</span> of <var>B</var>.</p>
   </li>

   <li><p>Return false.</p></li>
  </ol>

  <h5>Groupings of browsing contexts</h5>

  <p>A <span>top-level browsing context</span> has an associated <dfn data-x="tlbc
  group">group</dfn> (null or a <span>browsing context group</span>). It is initially null.</p>

  <p>A user agent holds a <dfn>browsing context group set</dfn> (a <span>set</span> of <span
  data-x="browsing context group">browsing context groups</span>).</p>

  <p>A <dfn>browsing context group</dfn> holds a <dfn>browsing context set</dfn> (a <span>set</span>
  of <span data-x="top-level browsing context">top-level browsing contexts</span>).</p>

  <p class="note">A <span>top-level browsing context</span> is added to the <span data-x="browsing
  context group">group</span> when the group is <span data-x="creating a new browsing context group
  and document">created</span>. All subsequent <span data-x="top-level browsing context">top-level
  browsing contexts</span> added to the <span data-x="browsing context group">group</span> will be
  <span data-x="auxiliary browsing context">auxiliary browsing contexts</span>.</p>

  <p>A <span>browsing context group</span> has an associated <dfn>agent cluster map</dfn> (a weak
  <span data-x="ordered map">map</span> of <span data-x="agent cluster key">agent cluster
  keys</span> to <span data-x="agent cluster">agent clusters</span>). User agents are responsible
  for collecting agent clusters when it is deemed that nothing can access them anymore.</p>

  <p>A <span>browsing context group</span> has an associated <dfn>historical agent cluster key
  map</dfn>, which is a <span data-x="ordered map">map</span> of <span
  data-x="origin">origins</span> to <span data-x="agent cluster key">agent cluster keys</span>. This
  map is used to ensure the consistency of the <a href="#origin-keyed-agent-clusters">origin-keyed
  agent clusters</a> feature by recording what agent cluster keys were previously used for a given
  origin.</p>

  <p class="note">The <span>historical agent cluster key map</span> only ever gains entries over the
  lifetime of the browsing context group.</p>

  <p>A <span>browsing context group</span> has a <dfn
  data-x="bcg-cross-origin-isolation">cross-origin isolation mode</dfn>, which is a
  <span>cross-origin isolation mode</span>. It is initially "<code
  data-x="cross-origin-isolation-none">none</code>".</p>

  <p>A <dfn>cross-origin isolation mode</dfn> is one of three possible values: "<dfn><code
  data-x="cross-origin-isolation-none">none</code></dfn>", "<dfn><code
  data-x="cross-origin-isolation-logical">logical</code></dfn>", or "<dfn><code
  data-x="cross-origin-isolation-concrete">concrete</code></dfn>".</p>

  <div class="note">
   <p>"<code data-x="cross-origin-isolation-logical">logical</code>" and "<code
   data-x="cross-origin-isolation-concrete">concrete</code>" are similar. They are both used for
   <span data-x="browsing context group">browsing context groups</span> where:</p>

   <ul>
    <li><p>every top-level <code>Document</code> has `<code
    data-x=""><span>Cross-Origin-Opener-Policy</span>: <span
    data-x="coop-same-origin">same-origin</span></code>`, and</p></li>

    <li><p>every <code>Document</code> has a `<code>Cross-Origin-Embedder-Policy</code>` header
    whose value is <span>compatible with cross-origin isolation</span>.</p></li>
   </ul>

   <p>On some platforms, it is difficult to provide the security properties required to grant safe
   access to the APIs gated by the <span
   data-x="concept-settings-object-cross-origin-isolated-capability">cross-origin isolated
   capability</span>. As a result, only "<code
   data-x="cross-origin-isolation-concrete">concrete</code>" can grant access that capability.
   "<code data-x="cross-origin-isolation-logical">logical</code>" is used on platform not supporting
   this capability, where various restrictions imposed by cross-origin isolation will still apply,
   but the capability is not granted.</p>
  </div>

  <p id="creating-a-new-browsing-context-group">To <dfn data-x="creating a new browsing context
  group and document">create a new browsing context group and document</dfn>:</p>

  <ol>
   <li><p>Let <var>group</var> be a new <span>browsing context group</span>.</p></li>

   <li><p><span data-x="set append">Append</span> <var>group</var> to the user agent's
   <span>browsing context group set</span>.</p></li>

   <li><p>Let <var>browsingContext</var> and <var>document</var> be the result of <span>creating a
   new browsing context and document</span> with null, null, and <var>group</var>.</p></li>

   <li><p><span data-x="bcg append">Append</span> <var>browsingContext</var> to
   <var>group</var>.</p></li>

   <li><p>Return <var>group</var> and <var>document</var>.</p></li>
  </ol>

  <p>To <dfn data-x="bcg append">append</dfn> a <span>top-level browsing context</span>
  <var>browsingContext</var> to a <span>browsing context group</span> <var>group</var>:</p>

  <ol>
   <li><p><span data-x="set append">Append</span> <var>browsingContext</var> to <var>group</var>'s
   <span>browsing context set</span>.</p></li>

   <li><p>Set <var>browsingContext</var>'s <span data-x="tlbc group">group</span> to
   <var>group</var>.</p></li>
  </ol>

  <p>To <dfn data-x="bcg remove">remove</dfn> a <span>top-level browsing context</span>
  <var>browsingContext</var>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>browsingContext</var>'s <span data-x="tlbc group">group</span>
   is non-null.</p></li>

   <li><p>Let <var>group</var> be <var>browsingContext</var>'s <span data-x="tlbc
   group">group</span>.</p></li>

   <li><p>Set <var>browsingContext</var>'s <span data-x="tlbc group">group</span> to null.</p></li>

   <li><p><span data-x="list remove">Remove</span> <var>browsingContext</var> from
   <var>group</var>'s <span>browsing context set</span>.</p></li>

   <li><p>If <var>group</var>'s <span>browsing context set</span> <span data-x="list is empty">is
   empty</span>, then <span data-x="list remove">remove</span> <var>group</var> from the user
   agent's <span>browsing context group set</span>.</p></li>
  </ol>

  <p class="note"><span data-x="bcg append">Append</span> and <span data-x="bcg
  remove">remove</span> are primitive operations that help define the lifetime of a <span>browsing
  context group</span>. They are called by higher-level creation and destruction operations for
  <code>Document</code>s and <span data-x="browsing context">browsing contexts</span>.</p>

  <p id="a-browsing-context-is-discarded">When there are no <code>Document</code> objects whose
  <span data-x="concept-document-bc">browsing context</span> equals a given <span>browsing
  context</span> (i.e., all such <code>Document</code>s have been <span data-x="destroy a
  document">destroyed</span>), and that <span>browsing context</span>'s <code>WindowProxy</code> is
  eligible for garbage collection, then the <span>browsing context</span> will never be accessed
  again. If it is a <span>top-level browsing context</span>, then at this point the user agent must
  <span data-x="bcg remove">remove</span> it.</p>


  <h4>Fully active documents</h4>

  <p>A <code>Document</code> <var>d</var> is said to be <dfn export for="Document">fully
  active</dfn> when <var>d</var> is the <span data-x="nav-document">active document</span> of a
  <span>navigable</span> <var>navigable</var>, and either <var>navigable</var> is a <span>top-level
  traversable</span> or <var>navigable</var>'s <span data-x="nav-container-document">container
  document</span> is <span>fully active</span>.</p>

  <p>Because they are associated with an element, <span data-x="child navigable">child
  navigables</span> are always tied to a specific <code>Document</code>, their <span
  data-x="nav-container-document">container document</span>, in their <span
  data-x="nav-parent">parent navigable</span>. User agents must not allow the user to interact with
  <span data-x="child navigable">child navigables</span> whose <span
  data-x="nav-container-document">container documents</span> are not themselves <span>fully
  active</span>.</p>

  <div class="example">
   <p>The following example illustrates how a <code>Document</code> can be the <span
   data-x="nav-document">active document</span> of its <span>node navigable</span>, while not being
   <span>fully active</span>. Here <code data-x="">a.html</code> is loaded into a browser window,
   <code data-x="">b-1.html</code> starts out loaded into an <code>iframe</code> as shown, and
   <code data-x="">b-2.html</code> and <code data-x="">c.html</code> are omitted (they can simply
   be an empty document).</p>

   <pre><code data-x="" class="html">&lt;!-- a.html -->
&lt;!DOCTYPE html>
&lt;html lang="en">
&lt;title>Navigable A&lt;/title>

&lt;iframe src="b-1.html">&lt;/iframe>
&lt;button onclick="frames[0].location.href = 'b-2.html'">Click me&lt;/button>

&lt;!-- b-1.html -->
&lt;!DOCTYPE html>
&lt;html lang="en">
&lt;title>Navigable B&lt;/title>

&lt;iframe src="c.html">&lt;/iframe></code></pre>

   <p>At this point, the documents given by <code data-x="">a.html</code>, <code
   data-x="">b-1.html</code>, and <code data-x="">c.html</code> are all the <span
   data-x="nav-document">active documents</span> of their respective <span data-x="node
   navigable">node navigables</span>. They are also all <span>fully active</span>.</p>

   <p>After clicking on the <code>button</code>, and thus loading a new <code>Document</code> from
   <code data-x="">b-2.html</code> into navigable B, we have the following results:</p>

   <ul>
    <li><p>The <code data-x="">a.html</code> <code>Document</code> remains both the <span
    data-x="nav-document">active document</span> of navigable A, and <span>fully
    active</span>.</p></li>

    <li><p>The <code data-x="">b-1.html</code> <code>Document</code> is now <em>not</em> the <span
    data-x="nav-document">active document</span> of navigable B. As such it is also not <span>fully
    active</span>.</p></li>

    <li><p>The new <code data-x="">b-2.html</code> <code>Document</code> is now the <span
    data-x="nav-document">active document</span> of navigable B, and is also <span>fully
    active</span>.</p></li>

    <li><p>The <code data-x="">c.html</code> <code>Document</code> is still the <span
    data-x="nav-document">active document</span> of navigable C. However, since C's <span
    data-x="nav-container-document">container document</span> is the <code
    data-x="">b-1.html</code> <code>Document</code>, which is itself not <span>fully active</span>,
    this means the <code data-x="">c.html</code> <code>Document</code> is now not <span>fully
    active</span>.</p></li>
   </ul>
  </div>


  <h3 split-filename="browsing-the-web"><span id="history"></span>Navigation and session
  history</h3>

  <p>Welcome to the dragon's maw. Navigation, session history, and the traversal through that
  session history are some of the most complex parts of this standard.</p>

  <p>The basic concept may not seem so difficult:</p>

  <ul>
   <li><p>The user is looking at a <span>navigable</span> that is presenting its <span
   data-x="nav-document">active document</span>. They <span>navigate</span> it to another
   <span>URL</span>.</p></li>

   <li><p>The browser fetches the given URL from the network, using it to <span data-x="attempt to
   populate the history entry's document">populate</span> a new <span>session history entry</span>
   with a newly-<span data-x="create-the-document-object">created</span>
   <code>Document</code>.</p></li>

   <li><p>The browser updates the <span>navigable</span>'s <span
   data-x="nav-active-history-entry">active session history entry</span> to the newly-populated
   one, and thus updates the <span data-x="nav-document">active document</span> that it is showing
   to the user.</p></li>

   <li><p>At some point later, the user <span data-x="traverse the history by a delta">presses the
   browser back button</span> to go back to the previous <span>session history
   entry</span>.</p></li>

   <li><p>The browser looks at the <span data-x="she-url">URL</span> stored in that <span>session
   history entry</span>, and uses it to re-fetch and <span data-x="attempt to populate the history
   entry's document">populate</span> that entry's <span
   data-x="she-document">document</span>.</p></li>

   <li><p>The browser again updates the <span>navigable</span>'s <span
   data-x="nav-active-history-entry">active session history entry</span>.</p></li>
  </ul>

  <p>You can see some of the intertwined complexity peeking through here, in how traversal can
  cause a navigation (i.e., a network fetch to a stored URL), and how a navigation necessarily
  needs to interface with the session history list to ensure that when it finishes the user is
  looking at the right thing. But the real problems come in with the various edge cases and
  interacting web platform features:</p>

  <ul>
   <li><p><span data-x="child navigable">Child navigables</span> (e.g., those contained in
   <code>iframe</code>s) can also navigate and traverse, but those navigations need to be
   linearized into <span data-x="tn-session-history-entries">a single session history list</span>
   since the user only has a single back/forward interface for the entire <span>traversable
   navigable</span> (e.g., browser tab).</p></li>

   <li><p>Since the user can traverse back more than a single step in the session history (e.g., by
   holding down their back button), they can end up traversing multiple <span
   data-x="navigable">navigables</span> at the same time when <span data-x="child
   navigable">child navigables</span> are involved. This needs to be synchronized across all of
   the involved navigables, which might involve multiple <span data-x="event loop">event
   loops</span> or even <span data-x="agent cluster">agent clusters</span>.</p></li>

   <li><p>During navigation, servers can respond with 204 or 205 status codes or with `<code
   data-x="http-content-disposition">Content-Disposition: attachment</code>` headers, which cause
   navigation to abort and the <span>navigable</span> to stay on its original <span data-x="active
   document">active document</span>. (This is much worse if it happens during a traversal-initiated
   navigation!)</p></li>

   <li><p>Various other HTTP headers, such as `<code data-x="">Location</code>`,
   `<code>Refresh</code>`, `<code>X-Frame-Options</code>`, and those for Content Security Policy,
   contribute to either the <span data-x="create navigation params by fetching">fetching
   process</span>, or the <span data-x="create-the-document-object"><code>Document</code>-creation
   process</span>, or both. The `<code>Cross-Origin-Opener-Policy</code>` header even contributes
   to the <a href="#browsing-context-group-switches-due-to-cross-origin-opener-policy">browsing
   context selection and creation</a> process!</p></li>

   <li><p>Some navigations (namely <a href="#scroll-to-fragid">fragment navigations</a> and <a
   href="#navigate-non-frag-sync">single-page app navigations</a>) are synchronous, meaning that
   JavaScript code expects to observe the navigation's results instantly. This then needs to be
   synchronized with the view of the session history that all other <span
   data-x="navigable">navigables</span> in the tree see, which can be subject to race conditions
   and necessitate resolving conflicting views of the session history.</p></li>

   <li><p>The platform has accumulated various exciting navigation-related features that need
   special-casing, such as <code data-x="javascript protocol">javascript:</code> URLs, <code
   data-x="attr-iframe-srcdoc">srcdoc</code> <code>iframe</code>s, and the <code
   data-x="event-beforeunload">beforeunload</code> event.</p></li>
  </ul>

  <p>In what follows, we have attempted to guide the reader through these complexities by
  appropriately cordoning them off into labeled sections and algorithms, and giving appropriate
  words of introduction where possible. Nevertheless, if you wish to truly understand navigation
  and session history, <a href="#how-to-read-this-specification">the usual advice</a> will be
  invaluable.</p>


  <h4 id="session-history-infrastructure"><span id="the-session-history-of-browsing-contexts"></span>Session history</h4>

  <h5>Session history entries</h5>

  <p>A <dfn>session history entry</dfn> is a <span>struct</span> with the following <span
  data-x="struct item">items</span>:</p>

  <ul>
   <li><p><dfn data-x="she-step">step</dfn>, a non-negative integer or "<code
   data-x="">pending</code>", initially "<code data-x="">pending</code>".</p></li>

   <li><p><dfn data-x="she-url">URL</dfn>, a <span>URL</span></p></li>

   <li><p><dfn data-x="she-document-state">document state</dfn>, a <span>document
   state</span>.</p></li>

   <li><p id="she-serialized-state"><dfn data-x="she-classic-history-api-state">classic history API
   state</dfn>, which is <span>serialized state</span>, initially
   <span>StructuredSerializeForStorage</span>(null).</p></li>

   <li><p><dfn data-x="she-navigation-api-state">navigation API state</dfn>, which is a
   <span>serialized state</span>, initially
   <span>StructuredSerializeForStorage</span>(undefined).</p></li>

   <li><p><dfn data-x="she-navigation-api-key">navigation API key</dfn>, which is a string,
   initially set to the result of <span>generating a random UUID</span>.</p></li>

   <li><p><dfn data-x="she-navigation-api-id">navigation API ID</dfn>, which is a string, initially
   set to the result of <span>generating a random UUID</span>.</p></li>

   <li><p><dfn data-x="she-scroll-restoration-mode">scroll restoration mode</dfn>, a <span>scroll
   restoration mode</span>, initially "<code
   data-x="dom-ScrollRestoration-auto">auto</code>".</p></li>

   <li><p><dfn data-x="she-scroll-position">scroll position data</dfn>, which is scroll position
   data for the <span data-x="she-document">document</span>'s <span>restorable scrollable
   regions</span>.</p></li>

   <li>
    <p id="an-entry-with-persisted-user-state"><dfn data-x="she-other">persisted user state</dfn>,
    which is <span>implementation-defined</span>, initially null</p>

    <p class="example">For example, some user agents might want to persist the values of form
    controls.</p>

    <p class="note">User agents that persist the value of form controls are encouraged to also
    persist their directionality (the value of the element's <code data-x="attr-dir">dir</code>
    attribute). This prevents values from being displayed incorrectly after a history traversal
    when the user had originally entered the values with an explicit, non-default
    directionality.</p>
   </li>
  </ul>

  <p>To get a <span>session history entry</span>'s <dfn data-x="she-document">document</dfn>,
  return its <span data-x="she-document-state">document state</span>'s <span
  data-x="document-state-document">document</span>.</p>

  <hr>

  <p id="state-object"><dfn>Serialized state</dfn> is a serialization (via
  <span>StructuredSerializeForStorage</span>) of an object representing a user interface state. We
  sometimes informally refer to "state objects", which are the objects representing user interface
  state supplied by the author, or alternately the objects created by deserializing (via
  <span>StructuredDeserialize</span>) serialized state.</p>

  <p>Pages can <span data-x="dom-history-pushState">add</span> <span>serialized state</span> to the
  session history. These are then <span data-x="StructuredDeserialize">deserialized</span> and <span
  data-x="event-popstate">returned to the script</span> when the user (or script) goes back in the
  history, thus enabling authors to use the "navigation" metaphor even in one-page applications.</p>

  <div class="note">
   <p><span>Serialized state</span> is intended to be used for two main purposes: first, storing a
   preparsed description of the state in the <span>URL</span> so that in the simple case an author
   doesn't have to do the parsing (though one would still need the parsing for handling <span
   data-x="URL">URLs</span> passed around by users, so it's only a minor optimization). Second, so
   that the author can store state that one wouldn't store in the URL because it only applies to the
   current <code>Document</code> instance and it would have to be reconstructed if a new
   <code>Document</code> were opened.</p>

   <p>An example of the latter would be something like keeping track of the precise coordinate from
   which a popup <code>div</code> was made to animate, so that if the user goes back, it can be made
   to animate to the same location. Or alternatively, it could be used to keep a pointer into a
   cache of data that would be fetched from the server based on the information in the
   <span>URL</span>, so that when going back and forward, the information doesn't have to be fetched
   again.</p>
  </div>

  <hr>

  <p>A <dfn>scroll restoration mode</dfn> indicates whether the user agent should restore the
  persisted scroll position (if any) when traversing to an <span data-x="session history
  entry">entry</span>. A scroll restoration mode is one of the following:</p>

  <dl>
   <dt>"<dfn enum-value for="ScrollRestoration"><code subdfn
   data-x="dom-ScrollRestoration-auto">auto</code></dfn>"</dt>
   <dd>The user agent is responsible for restoring the scroll position upon navigation.</dd>
   <dt>"<dfn enum-value for="ScrollRestoration"><code subdfn
   data-x="dom-ScrollRestoration-manual">manual</code></dfn>"</dt>
   <dd>The page is responsible for restoring the scroll position and the user agent does not
   attempt to do so automatically</dd>
  </dl>


  <h5>Document state</h5>

  <p><dfn>Document state</dfn> holds state inside a <span>session history entry</span> regarding
  how to present and, if necessary, recreate, a <code>Document</code>. It has:</p>

  <ul>
   <li>
    <p>A <dfn data-x="document-state-document">document</dfn>, a <code>Document</code> or null,
    initially null.</p>

    <div class="note" id="note-bfcache">
     <p>When a history entry is <span data-x="nav-active-history-entry">active</span>, it has a
     <code>Document</code> in its <span data-x="she-document-state">document state</span>. However,
     when a <code>Document</code> is not <span>fully active</span>, it's possible for it to be
     <span data-x="destroy a Document">destroyed</span> to free resources. In such cases, this
     <span data-x="document-state-document">document</span> item will be nulled out. The <span
     data-x="she-url">URL</span> and other data in the <span>session history entry</span> and <span
     data-x="she-document-state">document state</span> is then used to bring a new
     <code>Document</code> into being to take the place of the original, in the case where the user
     agent finds itself having to traverse to the entry.</p>

     <p>If the <code>Document</code> is <em>not</em> <span data-x="destroy a
     Document">destroyed</span>, then during <span data-x="traverse the history by a delta">history
     traversal</span>, it can be <span data-x="reactivate a document">reactivated</span>. The cache
     in which browsers store such <code>Document</code>s is often called a <em>back-forward
     cache</em>, or <em>bfcache</em> (or perhaps <a
     href="https://bugzilla.mozilla.org/show_bug.cgi?id=274784">"blazingly fast"</a> cache).</p>
    </div>
   </li>

   <li><p id="she-policy-container">A <dfn data-x="document-state-history-policy-container">history
   policy container</dfn>, a <span>policy container</span> or null, initially null.</p></li>

   <li><p>A <dfn data-x="document-state-request-referrer">request referrer</dfn>, which is "<code
   data-x="">no-referrer</code>", "<code data-x="">client</code>", or a <span>URL</span>, initially
   "<code data-x="">client</code>".</p></li>

   <li>
    <p>A <dfn data-x="document-state-request-referrer-policy">request referrer policy</dfn>, which
    is a <span>referrer policy</span>, initially the <span>default referrer policy</span>.</p>

    <p class="note">The <span data-x="document-state-request-referrer-policy">request referrer
    policy</span> is distinct from the <span
    data-x="document-state-history-policy-container">history policy container</span>'s <span
    data-x="policy-container-referrer-policy">referrer policy</span>. The former is used for
    fetches <em>of</em> this document, whereas the latter controls fetches <em>by</em> this
    document.</p>
   </li>

   <li>
    <p>An <dfn data-x="document-state-initiator-origin">initiator origin</dfn>, which is an
    <span>origin</span> or null, initially null.</p>
   </li>

   <li>
    <p>An <dfn data-x="document-state-origin">origin</dfn>, which is an <span>origin</span> or
    null, initially null.</p>

    <p class="note">This is the origin that we set "<code data-x="">about:</code>"-schemed
    <code>Document</code>s' <span data-x="concept-document-origin">origin</span> to. We store it
    here because it is also used when restoring these <code>Document</code>s during traversal,
    since they are reconstructed locally without visiting the network. It is also used to compare
    the origin before and after the <span>session history entry</span> is <span data-x="attempt to
    populate the history entry's document">repopulated</span>. If the origins change, the <span
    data-x="document-state-nav-target-name">navigable target name</span> is cleared.</p>
   </li>

   <li>
    <p>An <dfn data-x="document-state-about-base-url">about base URL</dfn>, which is a
    <span>URL</span> or null, initially null.</p>

    <p class="note">This will be populated only for "<code data-x="">about:</code>"-schemed
    <code>Document</code>s and will be the <span>fallback base URL</span> for those
    <code>Document</code>s. It is a snapshot of the initiator <code>Document</code>'s <span>document
    base URL</span>.</p>
   </li>

   <li><p><dfn data-x="document-state-nested-histories">Nested histories</dfn>, a <span>list</span>
   of <span data-x="nested history">nested histories</span>, initially an empty
   <span>list</span>.</p></li>

   <li>
    <p>A <dfn data-x="document-state-resource">resource</dfn>, a string, <span>POST resource</span>
    or null, initially null.</p>

    <p class="note">A string is treated as HTML. It's used to store the source of an <span
    data-x="an iframe srcdoc document"><code>iframe</code> <code
    data-x="attr-iframe-srcdoc">srcdoc</code> document</span>.</p>
   </li>

   <li><p>A <dfn data-x="document-state-reload-pending">reload pending</dfn> boolean, initially
   false.</p></li>

   <li><p>An <dfn data-x="document-state-ever-populated">ever populated</dfn> boolean, initially
   false.</p></li>

   <li><p id="she-bc-name">A <dfn data-x="document-state-nav-target-name">navigable target
   name</dfn> string, initially the empty string.</p></li>

   <li><p>A <dfn data-x="document-state-not-restored-reasons">not restored reasons</dfn>, a
   <span data-x="nrr-struct">not restored reasons</span> or null, initially null.</p></li>
  </ul>

  <p>User agents may <span>destroy a document and its descendants</span> given the <span
  data-x="document-state-document">documents</span> of <span data-x="document state">document
  states</span> with non-null <span data-x="document-state-document">documents</span>, as long as
  the <code>Document</code> is not <span>fully active</span>.</p>

  <p>Apart from that restriction, this standard does not specify when user agents should destroy
  the <span data-x="document-state-document">document</span> stored in a <span>document
  state</span>, versus keeping it cached.</p>

  <hr>

  <p>A <dfn export>POST resource</dfn> has:</p>

  <ul>
   <li>
    <p>A <dfn export for="POST resource" data-x="post-resource-request-body">request body</dfn>, a
    <span>byte sequence</span> or failure.</p>

    <p>This is only ever accessed <span>in parallel</span>, so it doesn't need to be stored in
    memory. However, it must return the same <span>byte sequence</span> each time. If this isn't
    possible due to resources changing on disk, or if resources can no longer be accessed, then
    this must be set to failure.</p>
   </li>

   <li><p>A <dfn export for="POST resource" data-x="post-resource-request-content-type">request
   content-type</dfn>, which is `<code>application/x-www-form-urlencoded</code>`,
   `<code>multipart/form-data</code>`, or `<code>text/plain</code>`.</p></li>
  </ul>

  <hr>

  <p>A <dfn>nested history</dfn> has:</p>

  <ul>
   <li>
    <p>An <dfn data-x="nested-history-id">id</dfn>, a <span>unique internal value</span>.</p>

    <p class="note">This is used to associate the <span>nested history</span> with a
    <span>navigable</span>.</p>
   </li>

   <li><p><dfn data-x="nested-history-entries">Entries</dfn>, a <span>list</span> of <span
   data-x="session history entry">session history entries</span>.</p></li>
  </ul>

  <p class="XXX">This will later contain ways to identify a child navigable across reloads.</p>

  <hr>

  <hr>

  <p>Several contiguous entries in a session history can share the same <span
  data-x="she-document-state">document state</span>. This can occur when the initial entry is
  reached via normal <span data-x="navigate">navigation</span>, and the following entry is added
  via <code data-x="dom-history-pushState">history.pushState()</code>. Or it can occur via <span
  data-x="navigate-fragid">navigation to a fragment</span>.</p>

  <p class="note">All entries that share the same <span data-x="she-document-state">document
  state</span> (and that are therefore merely different states of one particular document) are
  contiguous by construction.</p>

  <hr>

  <p>A <code>Document</code> has a <dfn>latest entry</dfn>, a <span>session history entry</span> or
  null.</p>

  <p class="note">This is the entry that was most recently represented by a given
  <code>Document</code>. A single <code>Document</code> can represent many <span data-x="session
  history entry">session history entries</span> over time, as many contiguous <span data-x="session
  history entry">session history entries</span> can share the same <span
  data-x="she-document-state">document state</span> as explained above.</p>


  <h5>Centralized modifications of session history</h5>

  <p>To maintain a single source of truth, all modifications to a <span>traversable
  navigable</span>'s <span data-x="tn-session-history-entries">session history entries</span> need
  to be synchronized. This is especially important due to how session history is influenced by all
  of the descendant <span data-x="navigable">navigables</span>, and thus by multiple <span
  data-x="event loop">event loops</span>. To accomplish this, we use the <span>session history
  traversal parallel queue</span> structure.</p>

  <p id="session-history-traversal-queue">A <dfn>session history traversal parallel queue</dfn> is
  very similar to a <span>parallel queue</span>. It has an <dfn
  data-x="session-history-traversal-parallel-queue-algorithm-set">algorithm set</dfn>, an <span
  data-x="set">ordered set</span>.</p>

  <p>The <span data-x="list item">items</span> in a <span>session history traversal parallel
  queue</span>'s <span data-x="session-history-traversal-parallel-queue-algorithm-set">algorithm
  set</span> are either algorithm steps, or <dfn
  data-x="session-history-traversal-parallel-queue-sync-nav-steps">synchronous navigation
  steps</dfn>, which are a particular brand of algorithm steps involving a <dfn
  data-x="session-history-traversal-parallel-queue-sync-nav-steps-target-nav">target
  navigable</dfn> (a <span>navigable</span>).</p>

  <p>To <dfn data-x="tn-append-session-history-traversal-steps">append session history traversal
  steps</dfn> to a <span>traversable navigable</span> <var>traversable</var> given algorithm steps
  <var>steps</var>, <span data-x="list append">append</span> <var>steps</var> to
  <var>traversable</var>'s <span data-x="tn-session-history-traversal-queue">session history
  traversal queue</span>'s <span
  data-x="session-history-traversal-parallel-queue-algorithm-set">algorithm set</span>.</p>

  <p>To <dfn data-x="tn-append-session-history-sync-nav-steps">append session history synchronous
  navigation steps</dfn> involving a <span>navigable</span> <var>targetNavigable</var> to a
  <span>traversable navigable</span> <var>traversable</var> given algorithm steps <var>steps</var>,
  <span data-x="list append">append</span> <var>steps</var> as <span
  data-x="session-history-traversal-parallel-queue-sync-nav-steps">synchronous navigation
  steps</span> targeting <span
  data-x="session-history-traversal-parallel-queue-sync-nav-steps-target-nav">target
  navigable</span> <var>targetNavigable</var> to <var>traversable</var>'s <span
  data-x="tn-session-history-traversal-queue">session history traversal queue</span>'s <span
  data-x="session-history-traversal-parallel-queue-algorithm-set">algorithm set</span>.</p>

  <p id="session-history-event-loop">To <dfn data-x="starting a new session history traversal
  parallel queue">start a new session history traversal parallel queue</dfn>:</p>

  <ol>
   <li><p>Let <var>sessionHistoryTraversalQueue</var> be a new <span>session history traversal
   parallel queue</span>.</p></li>

   <li>
    <p>Run the following steps <span>in parallel</span>:</p>

    <ol>
     <li>
      <p>While true:</p>

      <ol>
       <li><p>If <var>sessionHistoryTraversalQueue</var>'s <span
       data-x="session-history-traversal-parallel-queue-algorithm-set">algorithm set</span> is
       empty, then <span>continue</span>.</p></li>

       <li><p>Let <var>steps</var> be the result of <span data-x="dequeue">dequeuing</span> from
       <var>sessionHistoryTraversalQueue</var>'s <span
       data-x="session-history-traversal-parallel-queue-algorithm-set">algorithm
       set</span>.</p></li>

       <li><p>Run <var>steps</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Return <var>sessionHistoryTraversalQueue</var>.</p></li>
  </ol>

  <p id="sync-navigation-steps-queue-jumping-examples"><span
  data-x="session-history-traversal-parallel-queue-sync-nav-steps">Synchronous navigation
  steps</span> are tagged in the <span
  data-x="session-history-traversal-parallel-queue-algorithm-set">algorithm set</span> to allow
  them to conditionally "jump the queue". This is handled <a
  href="#sync-navigations-jump-queue">within apply the history step</a>.</p>

  <div class="example" id="example-sync-navigation-steps-queue-jumping-basic">
   <p>Imagine the joint session history depicted by this <span>Jake diagram</span>:</p>

   <!--https://domenic.github.io/jake-diagram-generator/#IWN1cnJlbnQgPSAxCnRvcAogIDA6IC9hCiAgMTogL2I= -->
   <table class="jake-diagram"><thead><tr><td></td><th class="step">0</th><th class="step current">1</th></tr></thead><tbody><tr><th><code data-x="">top</code></th><td colspan="1" class="doc-0">/a</td><td colspan="1" class="doc-1 current">/b</td></tr></tbody></table>

   <p>And the following code runs at the top level:</p>

   <pre><code class="js">history.back();
location.href = '#foo';</code></pre>

   <p>The desired result is:</p>

   <!--https://domenic.github.io/jake-diagram-generator/#IWN1cnJlbnQgPSAwCnRvcAogIDA6IC9hCiAgMTogL2IgfCBkb2MxCiAgMjogL2IjZm9vIHwgZG9jMQ== -->
   <table class="jake-diagram"><thead><tr><td></td><th class="step current">0</th><th class="step">1</th><th class="step">2</th></tr></thead><tbody><tr><th><code data-x="">top</code></th><td colspan="1" class="doc-0 current">/a</td><td colspan="1" class="doc-1 next-is-same-doc">/b</td><td colspan="1" class="doc-1 prev-is-same-doc">/b#foo</td></tr></tbody></table>

   <p>This isn't straightforward, as the sync navigation wins the race in terms of being
   observable, whereas the traversal wins the race in terms of queuing steps on the <span>session
   history traversal parallel queue</span>. To achieve this result, the following happens:</p>

   <ol>
    <li><p><code data-x="dom-history-back">history.back()</code> <span
    data-x="tn-append-session-history-traversal-steps">appends steps</span> intended to traverse by
    a delta of &minus;1.</p></li>

    <li><p><code data-x="dom-location-href">location.href = '#foo'</code> synchronously changes the
    <span data-x="nav-active-history-entry">active session history entry</span> entry to a
    newly-created one, with the URL <code data-x="">/b#foo</code>, and <span
    data-x="tn-append-session-history-sync-nav-steps">appends synchronous steps</span> to notify
    the central source of truth about that new entry. Note that this does <em>not</em> yet update
    the <span data-x="nav-current-history-entry">current session history entry</span>, <span
    data-x="tn-current-session-history-step">current session history step</span>, or the <span
    data-x="tn-session-history-entries">session history entries</span> list; those updates cannot
    be done synchronously, and instead must be done as part of the queued steps.</p></li>

    <li>
     <p>On the <span>session history traversal parallel queue</span>, the steps queued by <code
     data-x="dom-history-back">history.back()</code> run:</p>

     <ol>
      <li><p>The target history step is determined to be 0: the <span
      data-x="tn-current-session-history-step">current session history step</span> (i.e., 1) plus
      the intended delta of &minus;1.</p></li>

      <li>
       <p>We enter the main <span>apply the history step</span> algorithm.</p>

       <p>The entry at step 0, for the <code data-x="">/a</code> URL, has its <span
       data-x="she-document">document</span> <span data-x="attempt to populate the history entry's
       document">populated</span>.</p>

       <p>Meanwhile, the queue is checked for <span
       data-x="session-history-traversal-parallel-queue-sync-nav-steps">synchronous navigation
       steps</span>. The steps queued by the <code data-x="dom-location-href">location.href</code>
       setter now run, and block the traversal from performing effects beyond document population
       (such as, unloading documents and switching active history entries) until they are finished.
       Those steps cause the following to happen:</p>

       <ol>
        <li><p>The entry with URL <code data-x="">/b#foo</code> is added, with its <span
        data-x="she-step">step</span> determined to be 2: the <span
        data-x="tn-current-session-history-step">current session history step</span> (i.e., 1) plus
        1.</p></li>

        <li><p>We fully switch to that newly added entry, including a nested call to <span>apply
        the history step</span>. This ultimately results in <span data-x="update document for
        history step application">updating the document</span> by dispatching events like <code
        data-x="event-hashchange">hashchange</code>.</p></li>
       </ol>

       <p>Only once that is all complete, and the <code data-x="">/a</code> history entry has been
       fully populated with a <span data-x="she-document">document</span>, do we move on with
       applying the history step given the target step of 0.</p>

       <p>At this point, the <code>Document</code> with URL <code data-x="">/b#foo</code> <span
       data-x="unload a document">unloads</span>, and we finish moving to our target history step
       0, which makes the entry with URL <code data-x="">/a</code> become the <span
       data-x="nav-active-history-entry">active session history entry</span> and 0 become the <span
       data-x="tn-current-session-history-step">current session history step</span>.</p>
      </li>
     </ol>
    </li>
   </ol>
  </div>

  <div class="example" id="example-sync-navigation-steps-queue-jumping-complex">
   <p>Here is another more complex example, involving races between populating two different
   <code>iframe</code>s, and a synchronous navigation once one of those iframes loads. We start
   with this setup:</p>

   <!--https://domenic.github.io/jake-diagram-generator/#IWN1cnJlbnQgPSAyCnRvcAogIDAtMjogL3QKZnJhbWVzWzBdCiAgMDogL2ktMC1hCiAgMS0yOiAvaS0wLWIKZnJhbWVzWzFdCiAgMC0xOiAvaS0xLWEKICAyOiAvaS0xLWIK -->
   <table class="jake-diagram"><thead><tr><td></td><th class="step">0</th><th class="step">1</th><th class="step current">2</th></tr></thead><tbody><tr><th><code data-x="">top</code></th><td colspan="3" class="doc-0 current">/t</td></tr><tr><th><code data-x="">frames[0]</code></th><td colspan="1" class="doc-1">/i-0-a</td><td colspan="2" class="doc-2 current">/i-0-b</td></tr><tr><th><code data-x="">frames[1]</code></th><td colspan="2" class="doc-3">/i-1-a</td><td colspan="1" class="doc-4 current">/i-1-b</td></tr></tbody></table>

   <p>and then call <code data-x="dom-history-go">history.go(-2)</code>. The following then
   occurs:</p>

   <ol>
    <li>
     <p><code data-x="dom-history-go">history.go(-2)</code> <span
     data-x="tn-append-session-history-traversal-steps">appends steps</span> intended to traverse
     by a delta of &minus;2. Once those steps run:</p>

     <ol>
      <li><p>The target step is determined to be 2 + (&minus;2) = 0.</p></li>

      <li>
       <p>In parallel, the fetches are made to <span data-x="attempt to populate the history
       entry's document">populate</span> the two iframes, fetching <code data-x="">/i-0-a</code>
       and <code data-x="">/i-1-a</code> respectively.</p>

       <p>Meanwhile, the queue is checked for <span
       data-x="session-history-traversal-parallel-queue-sync-nav-steps">synchronous navigation
       steps</span>. There aren't any right now.</p>
      </li>

      <li><p>In the fetch race, the fetch for <code data-x="">/i-0-a</code> wins. We proceed onward
      to finish all of <span>apply the history step</span>'s work for how the traversal impacts the
      <code data-x="">frames[0]</code> <span>navigable</span>, including updating its <span
      data-x="nav-active-history-entry">active session history entry</span> to the entry with URL
      <code data-x="">/i-0-a</code>.</p></li>

      <li>
       <p>Before the fetch for <code data-x="">/i-1-a</code> finishes, we reach the point where
       <span>scripts may run for the newly-created document</span> in the <code
       data-x="">frames[0]</code> <span>navigable</span>'s <span data-x="nav-document">active
       document</span>. Some such script does run:</p>

       <pre><code class="js">location.href = '#foo'</code></pre>

       <p>This synchronously changes the <code data-x="">frames[0]</code> navigable's <span
       data-x="nav-active-history-entry">active session history entry</span> entry to a
       newly-created one, with the URL <code data-x="">/i-0-a#foo</code>, and <span
       data-x="tn-append-session-history-sync-nav-steps">appends synchronous steps</span> to notify
       the central source of truth about that new entry.</p>

       <p>Unlike in the <a href="#example-sync-navigation-steps-queue-jumping-basic">previous
       example</a>, these synchronous steps do <em>not</em> "jump the queue" and update the <span
       data-x="traversable navigable">traversable</span> before we finish the fetch for <code
       data-x="">/i-1-a</code>. This is because the navigable in question, <code
       data-x="">frames[0]</code>, has already been altered as part of the traversal, so we know
       that with the <span data-x="tn-current-session-history-step">current session history
       step</span> being 2, adding the new entry as a step 3 doesn't make sense.</p>
      </li>

      <li><p>Once the fetch for <code data-x="">/i-1-a</code> finally finishes, we proceed to finish
      updating the <code data-x="">frames[1]</code> <span>navigable</span> for the traversal,
      including updating its <span data-x="nav-active-history-entry">active session history
      entry</span> to the entry with URL <code data-x="">/i-1-a</code>.</p></li>

      <li><p>Now that both navigables have finished processing the traversal, we update the <span
      data-x="tn-current-session-history-step">current session history step</span> to the target
      step of 0.</p></li>
     </ol>
    </li>

    <li>
     <p>Now we can process the steps that were queued for the synchronous navigation:</p>

     <ol>
      <li><p>The <code data-x="">/i-0-a#foo</code> entry is added, with its <span
      data-x="she-step">step</span> determined to be 1: the <span
      data-x="tn-current-session-history-step">current session history step</span> (i.e., 0) plus
      1. This also <span data-x="clear the forward session history">clears existing forward
      history</span>.</p></li>

      <li><p>We fully switch to that newly added entry, including calling <span>apply the history
      step</span>. This ultimately results in <span data-x="update document for history step
      application">updating the document</span> by dispatching events like <code
      data-x="event-hashchange">hashchange</code>, as well as updating the <span
      data-x="tn-current-session-history-step">current session history step</span> to the target
      step of 1.</p></li>
     </ol>
    </li>
   </ol>

   <p>The end result is:</p>

   <!--https://domenic.github.io/jake-diagram-generator/#IWN1cnJlbnQgPSAxCiFza2lwRG9jQ29sb3JzID0gMgoKdG9wCiAgMC0xOiAvdApmcmFtZXNbMF0KICAwOiAvaS0wLWEgfCBkb2MxCiAgMTogL2ktMC1hI2ZvbyB8IGRvYzEKZnJhbWVzWzFdCiAgMC0xOiAvaS0xLWEK -->
   <table class="jake-diagram"><thead><tr><td></td><th class="step">0</th><th class="step current">1</th></tr></thead><tbody><tr><th><code data-x="">top</code></th><td colspan="2" class="doc-0 current">/t</td></tr><tr><th><code data-x="">frames[0]</code></th><td colspan="1" class="doc-1 next-is-same-doc">/i-0-a</td><td colspan="1" class="doc-1 current prev-is-same-doc">/i-0-a#foo</td></tr><tr><th><code data-x="">frames[1]</code></th><td colspan="2" class="doc-3 current">/i-1-a</td></tr></tbody></table>
  </div>


  <h5>Low-level operations on session history</h5>

  <p>This section contains a miscellaneous grab-bag of operations that we perform throughout the
  standard when manipulating session history. The best way to get a sense of what they do is to
  look at their call sites.</p>

  <p>To <dfn data-x="getting session history entries">get session history entries</dfn> of a
  <span>navigable</span> <var>navigable</var>:</p>

  <ol>
   <li><p>Let <var>traversable</var> be <var>navigable</var>'s <span
   data-x="nav-traversable">traversable navigable</span>.</p></li>

   <li><p><span>Assert</span>: this is running within <var>traversable</var>'s <span
   data-x="tn-session-history-traversal-queue">session history traversal queue</span>.</p></li>

   <li><p>If <var>navigable</var> is <var>traversable</var>, return <var>traversable</var>'s <span
   data-x="tn-session-history-entries">session history entries</span>.</p></li>

   <li><p>Let <var>docStates</var> be an empty <span data-x="set">ordered set</span> of <span
   data-x="document state">document states</span>.</p></li>

   <li><p>For each <var>entry</var> of <var>traversable</var>'s <span
   data-x="tn-session-history-entries">session history entries</span>, <span data-x="set
   append">append</span> <var>entry</var>'s <span data-x="she-document-state">document state</span>
   to <var>docStates</var>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>docState</var> of <var>docStates</var>:</p>

    <ol>
     <li>
      <p><span data-x="list iterate">For each</span> <var>nestedHistory</var> of
      <var>docState</var>'s <span data-x="document-state-nested-histories">nested
      histories</span>:</p>

      <ol>
       <li><p>If <var>nestedHistory</var>'s <span data-x="nested-history-id">id</span> equals
       <var>navigable</var>'s <span data-x="nav-id">id</span>, return <var>nestedHistory</var>'s
       <span data-x="nested-history-entries">entries</span>.</p></li>

       <li><p>For each <var>entry</var> of <var>nestedHistory</var>'s <span
       data-x="nested-history-entries">entries</span>, <span data-x="set append">append</span>
       <var>entry</var>'s <span data-x="she-document-state">document state</span> to
       <var>docStates</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p><span>Assert</span>: this step is not reached.</p></li>
  </ol>

  <p>To <dfn data-x="getting session history entries for the navigation API">get session history
  entries for the navigation API</dfn> of a <span>navigable</span> <var>navigable</var> given an
  integer <var>targetStep</var>:</p>

  <ol>
   <li><p>Let <var>rawEntries</var> be the result of <span>getting session history entries</span>
   for <var>navigable</var>.</p></li>

   <li><p>Let <var>entriesForNavigationAPI</var> be a new empty <span>list</span>.</p></li>

   <li>
    <p>Let <var>startingIndex</var> be the index of the <span>session history entry</span> in
    <var>rawEntries</var> who has the greatest <span data-x="she-step">step</span> less than or
    equal to <var>targetStep</var>.</p>

    <p class="note">See <a href="#example-getting-the-target-history-entry">this example</a> to
    understand why it's the greatest step less than or equal to <var>targetStep</var>.</p>
   </li>

   <li><p><span data-x="list append">Append</span> <var>rawEntries</var>[<var>startingIndex</var>]
   to <var>entriesForNavigationAPI</var>.</p></li>

   <li><p>Let <var>startingOrigin</var> be <var>rawEntries</var>[<var>startingIndex</var>]'s <span
   data-x="she-document-state">document state</span>'s <span
   data-x="document-state-origin">origin</span>.</p></li>

   <li><p>Let <var>i</var> be <var>startingIndex</var> &minus; 1.</p></li>

   <li>
    <p>While <var>i</var> &gt; 0:</p>

    <ol>
     <li><p>If <var>rawEntries</var>[<var>i</var>]'s <span data-x="she-document-state">document
     state</span>'s <span data-x="document-state-origin">origin</span> is not <span>same
     origin</span> with <var>startingOrigin</var>, then <span>break</span>.</p></li>

     <li><p><span data-x="list prepend">Prepend</span> <var>rawEntries</var>[<var>i</var>] to
     <var>entriesForNavigationAPI</var>.</p></li>

     <li><p>Set <var>i</var> to <var>i</var> &minus; 1.</p></li>
    </ol>
   </li>

   <li><p>Set <var>i</var> to <var>startingIndex</var> + 1.</p></li>

   <li>
    <p>While <var>i</var> &lt; <var>rawEntries</var>'s <span data-x="list size">size</span>:</p>

    <ol>
     <li><p>If <var>rawEntries</var>[<var>i</var>]'s <span data-x="she-document-state">document
     state</span>'s <span data-x="document-state-origin">origin</span> is not <span>same
     origin</span> with <var>startingOrigin</var>, then <span>break</span>.</p></li>

     <li><p><span data-x="list append">Append</span> <var>rawEntries</var>[<var>i</var>] to
     <var>entriesForNavigationAPI</var>.</p></li>

     <li><p>Set <var>i</var> to <var>i</var> + 1.</p></li>
    </ol>
   </li>

   <li><p>Return <var>entriesForNavigationAPI</var>.</p></li>
  </ol>

  <p>To <dfn>clear the forward session history</dfn> of a <span>traversable navigable</span>
  <var>navigable</var>:</p>

  <ol>
   <li><p><span>Assert</span>: this is running within <var>navigable</var>'s <span
   data-x="tn-session-history-traversal-queue">session history traversal queue</span>.</p></li>

   <li><p>Let <var>step</var> be the <var>navigable</var>'s <span
   data-x="tn-current-session-history-step">current session history step</span>.</p></li>

   <li><p>Let <var>entryLists</var> be the <span data-x="set">ordered set</span> «
   <var>navigable</var>'s <span data-x="tn-session-history-entries">session history entries</span>
   ».</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>entryList</var> of
    <var>entryLists</var>:</p>

    <ol>
     <li><p><span data-x="list remove">Remove</span> every <span>session history entry</span> from
     <var>entryList</var> that has a <span data-x="she-step">step</span> greater than
     <var>step</var>.</p></li>

     <li>
      <p><span data-x="list iterate">For each</span> <var>entry</var> of <var>entryList</var>:</p>

      <ol>
       <li><p><span data-x="list iterate">For each</span> <var>nestedHistory</var> of
       <var>entry</var>'s <span data-x="she-document-state">document state</span>'s <span
       data-x="document-state-nested-histories">nested histories</span>, <span data-x="set
       append">append</span> <var>nestedHistory</var>'s <span
       data-x="nested-history-entries">entries list</span> to <var>entryLists</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>
  </ol>

  <p>To <dfn data-x="getting all used history steps">get all used history steps</dfn> that are part
  of <span>traversable navigable</span> <var>traversable</var>:</p>

  <ol>
   <li><p><span>Assert</span>: this is running within <var>traversable</var>'s <span
   data-x="tn-session-history-traversal-queue">session history traversal queue</span>.</p></li>

   <li><p>Let <var>steps</var> be an empty <span data-x="set">ordered set</span> of non-negative
   integers.</p></li>

   <li><p>Let <var>entryLists</var> be the <span data-x="set">ordered set</span> «
   <var>traversable</var>'s <span data-x="tn-session-history-entries">session history
   entries</span> ».</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>entryList</var> of
    <var>entryLists</var>:</p>

    <ol>
     <li>
      <p><span data-x="list iterate">For each</span> <var>entry</var> of <var>entryList</var>:</p>

      <ol>
       <li><p><span data-x="set append">Append</span> <var>entry</var>'s <span
       data-x="she-step">step</span> to <var>steps</var>.</p></li>

       <li><p><span data-x="list iterate">For each</span> <var>nestedHistory</var> of
       <var>entry</var>'s <span data-x="she-document-state">document state</span>'s <span
       data-x="document-state-nested-histories">nested histories</span>, <span data-x="set
       append">append</span> <var>nestedHistory</var>'s <span
       data-x="nested-history-entries">entries list</span> to <var>entryLists</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Return <var>steps</var>, <span data-x="list sort">sorted</span>.</p></li>
  </ol>


  <h4 id="navigating-across-documents"><span id="browsing-the-web"></span>Navigation</h4>

  <p>Certain actions cause a <span>navigable</span> to <i data-x="navigate">navigate</i> to
  a new resource.</p>

  <p class="example">For example, <span data-x="following hyperlinks">following a hyperlink</span>,
  <span data-x="concept-form-submit">form submission</span>, and the <code
  data-x="dom-open">window.open()</code> and <code
  data-x="dom-location-assign">location.assign()</code> methods can all cause navigation.</p>

  <div class="note" id="note-meaning-of-navigate">
   <p>Although in this standard the word "navigation" refers specifically to the
   <span>navigate</span> algorithm, this doesn't always line up with web developer or user
   perceptions. For example:</p>

   <ul>
    <li><p>The <span>URL and history update steps</span> are often used during so-called
    "single-page app navigations" or "same-document navigations", but they do not trigger the
    <span>navigate</span> algorithm.</p></li>

    <li><p><span data-x="reload">Reloads</span> and <span data-x="traverse the history by a
    delta">traversals</span> are sometimes talked about as a type of navigation, since all three
    will often <span>attempt to populate the history entry's document</span> and thus could perform
    navigational fetches. See, e.g., the APIs exposed <cite>Navigation Timing</cite>. But they have
    their own entry point algorithms, separate from the <span>navigate</span> algorithm.
    <ref>NAVIGATIONTIMING</ref></p></li>

    <li><p>Although <span data-x="navigate-fragid">fragment navigations</span> are always done
    through the <span>navigate</span> algorithm, a user might perceive them as more like jumping
    around a single page, than as a true navigation.</p></li>
   </ul>
  </div>


  <h5 id="navigation-supporting-concepts">Supporting concepts</h5>

  <p>Before we can jump into the <span data-x="navigate">navigation algorithm</span> itself, we
  need to establish several important structures that it uses.</p>

  <p>The <dfn>source snapshot params</dfn> <span>struct</span> is used to capture data from a
  <code>Document</code> initiating a navigation. It is snapshotted at the beginning of a navigation
  and used throughout the navigation's lifetime. It has the following <span data-x="struct
  item">items</span>:</p>

  <dl>
   <dt><dfn data-x="source-snapshot-params-activation">has transient activation</dfn></dt>
   <dd>a boolean</dd>

   <dt><dfn data-x="source-snapshot-params-sandbox">sandboxing flags</dfn></dt>
   <dd>a <span>sandboxing flag set</span></dd>

   <dt><dfn data-x="source-snapshot-params-download">allows downloading</dfn></dt>
   <dd>a boolean</dd>

   <dt><dfn data-x="source-snapshot-params-client">fetch client</dfn></dt>
   <dd>an <span>environment settings object</span>, only to be used as a <span
   data-x="concept-request-client">request client</span></dd>

   <dt><dfn data-x="source-snapshot-params-policy-container">source policy container</dfn></dt>
   <dd>a <span>policy container</span></dd>
  </dl>

  <p>To <dfn data-x="snapshotting source snapshot params">snapshot source snapshot params</dfn>
  given a <code>Document</code> <var>sourceDocument</var>, return a new <span>source snapshot
  params</span> with</p>

  <dl class="props">
   <dt><span data-x="source-snapshot-params-activation">has transient activation</span></dt>
   <dd>true if <var>sourceDocument</var>'s <span>relevant global object</span> has <span>transient
   activation</span>; otherwise false</dd>

   <dt><span data-x="source-snapshot-params-sandbox">sandboxing flags</span></dt>
   <dd><var>sourceDocument</var>'s <span>active sandboxing flag set</span></dd>

   <dt><span data-x="source-snapshot-params-download">allows downloading</span></dt>
   <dd>false if <var>sourceDocument</var>'s <span>active sandboxing flag set</span> has the
   <span>sandboxed downloads browsing context flag</span> set; otherwise true</dd>

   <dt><span data-x="source-snapshot-params-client">fetch client</span></dt>
   <dd><var>sourceDocument</var>'s <span>relevant settings object</span></dd>

   <dt><span data-x="source-snapshot-params-policy-container">source policy container</span></dt>
   <dd>a <span data-x="clone a policy container">clone</span> of <var>sourceDocument</var>'s <span
   data-x="concept-document-policy-container">policy container</span></dd>
  </dl>

  <hr>

  <p>The <dfn>target snapshot params</dfn> <span>struct</span> is used to capture data from a
  <span>navigable</span> being navigated. Like <span>source snapshot params</span>, it is
  snapshotted at the beginning of a navigation and used throughout the navigation's lifetime. It
  has the following <span data-x="struct item">items</span>:</p>

  <dl>
   <dt><dfn data-x="target-snapshot-params-sandbox">sandboxing flags</dfn></dt>
   <dd>a <span>sandboxing flag set</span></dd>
  </dl>

  <p>To <dfn data-x="snapshotting target snapshot params">snapshot target snapshot params</dfn>
  given a <span>navigable</span> <var>targetNavigable</var>, return a new <span>target snapshot
  params</span> with <span data-x="target-snapshot-params-sandbox">sandboxing flags</span> set to
  the result of <span>determining the creation sandboxing flags</span> given
  <var>targetNavigable</var>'s <span data-x="nav-bc">active browsing context</span> and
  <var>targetNavigable</var>'s <span data-x="nav-container">container</span>.</p>

  <hr>

  <p>Much of the navigation process is concerned with determining how to create a new
  <code>Document</code>, which ultimately happens in the <span
  data-x="create-the-document-object">create and initialize a <code>Document</code> object</span>
  algorithm. The parameters to that algorithm are tracked via a <dfn>navigation params</dfn>
  <span>struct</span>, which has the following <span data-x="struct item">items</span>:</p>

  <!--
   These struct items are roughly ordered/grouped like so:

   * Very basic concepts, shared with non-fetch scheme navigation params
   * Fetch-related stuff
   * Other stuff that affects the process but isn't directly used during new Document creation
   * Stuff used directly during new Document creation
     * Core stuff first
     * Then policy stuff
     * Then other stuff

   Try to keep this sort of grouping in mind when adding to this struct.
  -->
  <dl>
   <dt><dfn data-x="navigation-params-id">id</dfn></dt>
   <dd>null or a <span>navigation ID</span></dd>

   <dt><dfn data-x="navigation-params-navigable">navigable</dfn></dt>
   <dd>the <span>navigable</span> to be navigated</dd>

   <dt><dfn data-x="navigation-params-request">request</dfn></dt>
   <dd>null or a <span data-x="concept-request">request</span> that started the navigation</dd>

   <dt><dfn data-x="navigation-params-response">response</dfn></dt>
   <dd>a <span data-x="concept-response">response</span> that ultimately was navigated to
   (potentially a <span>network error</span>)</dd>

   <dt><dfn data-x="navigation-params-fetch-controller">fetch controller</dfn></dt>
   <dd>null or a <span>fetch controller</span></dd>

   <dt><dfn data-x="navigation-params-commit-early-hints">commit early hints</dfn></dt>
   <dd>null or an algorithm accepting a <code>Document</code>, once it has been created</dd>

   <dt><dfn data-x="navigation-params-coop-enforcement-result">COOP enforcement result</dfn></dt>
   <dd>an <span data-x="coop-enforcement-result">opener policy enforcement result</span>, used for
   reporting and potentially for causing a <a
   href="#browsing-context-group-switches-due-to-cross-origin-opener-policy">browsing context
   group switch</a></dd>

   <dt><dfn data-x="navigation-params-reserved-environment">reserved environment</dfn></dt>
   <dd>null or an <span>environment</span> reserved for the new <code>Document</code></dd>

   <dt><dfn data-x="navigation-params-origin">origin</dfn></dt>
   <dd>an <span>origin</span> to use for the new <code>Document</code></dd>

   <dt><dfn data-x="navigation-params-policy-container">policy container</dfn></dt>
   <dd>a <span>policy container</span> to use for the new <code>Document</code></dd>

   <dt><dfn data-x="navigation-params-sandboxing">final sandboxing flag set</dfn></dt>
   <dd>a <span>sandboxing flag set</span> to impose on the new <code>Document</code></dd>

   <dt><dfn data-x="navigation-params-coop">opener policy</dfn></dt>
   <dd>an <span>opener policy</span> to use for the new <code>Document</code></dd>

   <dt><dfn data-x="navigation-params-nav-timing-type">navigation timing type</dfn></dt>
   <dd>a <code>NavigationTimingType</code> used for <span data-x="create the navigation timing
   entry">creating the navigation timing entry</span> for the new <code>Document</code></dd>

   <dt><dfn data-x="navigation-params-about-base-url">about base URL</dfn></dt>
   <dd>a <span>URL</span> or null used to populate the new <code>Document</code>'s <span
   data-x="concept-document-about-base-url">about base URL</span></dd>

   <dt><dfn data-x="navigation-params-user-involvement">user involvement</dfn></dt>
   <dd>a <span>user navigation involvement</span> used when <span
   data-x="obtain-browsing-context-navigation">obtaining a browsing context</span> for the new
   <code>Document</code></dd>
  </dl>

  <p class="note">Once a <span>navigation params</span> struct is created, this standard does not
  mutate any of its <span data-x="struct item">items</span>. They are only passed onward to other
  algorithms.</p>

  <hr>

  <p>A <dfn>navigation ID</dfn> is a UUID string generated during navigation. It is used to
  interface with the <cite>WebDriver BiDi</cite> specification as well as to track the <span>ongoing
  navigation</span>. <ref>WEBDRIVERBIDI</ref></p>

  <hr>

  <p>After <code>Document</code> creation, the relevant <span>traversable navigable</span>'s <span
  data-x="tn-session-history-entries">session history</span> gets updated. The
  <code>NavigationHistoryBehavior</code> enumeration is used to indicate the desired type of session
  history update to the <span>navigate</span> algorithm. It is one of the following:</p>

  <dl>
   <dt id="hh-default"><span id="hh-push"></span>"<dfn enum-value for="NavigationHistoryBehavior" data-x="NavigationHistoryBehavior-push"><code>push</code></dfn>"</dt>
   <dd>A regular navigation which adds a new <span>session history entry</span>, and will
   <span>clear the forward session history</span>.</dd>

   <dt id="hh-replace">"<dfn enum-value for="NavigationHistoryBehavior" data-x="NavigationHistoryBehavior-replace"><code>replace</code></dfn>"</dt>
   <dd>A navigation that will replace the <span data-x="nav-active-history-entry">active session
   history entry</span>.</dd>

   <dt>"<dfn enum-value for="NavigationHistoryBehavior" data-x="NavigationHistoryBehavior-auto"><code>auto</code></dfn>"</dt>
   <dd>The default value, which will be converted very early in the <span>navigate</span> algorithm
   into "<code data-x="NavigationHistoryBehavior-push">push</code>" or "<code
   data-x="NavigationHistoryBehavior-replace">replace</code>". Usually it becomes "<code
   data-x="NavigationHistoryBehavior-push">push</code>", but under <a
   href="#navigate-convert-to-replace">certain circumstances</a> it becomes "<code
   data-x="NavigationHistoryBehavior-replace">replace</code>" instead.</dd>
  </dl>

  <p>A <dfn>history handling behavior</dfn> is a <code>NavigationHistoryBehavior</code> that is
  either "<code data-x="NavigationHistoryBehavior-push">push</code>" or "<code
  data-x="NavigationHistoryBehavior-replace">replace</code>", i.e., that has been resolved away from
  any initial "<code data-x="NavigationHistoryBehavior-auto">auto</code>" value.</p>

  <p><dfn>The navigation must be a replace</dfn>, given a <span>URL</span> <var>url</var> and a
  <code>Document</code> <var>document</var>, if any of the following are true:</p>

  <ul>
   <li><p><var>url</var>'s <span data-x="concept-url-scheme">scheme</span> is "<code
   data-x="javascript protocol">javascript</code>"; or</p></li>

   <li><p><var>document</var>'s <span>is initial <code>about:blank</code></span> is true.</p></li>
  </ul>

  <div class="note">
   <p>Other cases that often, but not always, force a "<code
   data-x="NavigationHistoryBehavior-replace">replace</code>" navigation are:</p>

   <ul>
    <li><p>if the <code>Document</code> is not <span>completely loaded</span>; or</p></li>

    <li><p>if the target <span>URL</span> equals the <code>Document</code>'s <span
    data-x="concept-document-url">URL</span>.</p></li>
   </ul>
  </div>

  <hr>

  <p>Various parts of the platform track whether a user is involved in a navigation. A <dfn
  export>user navigation involvement</dfn> is one of the following:</p>

  <dl>
   <dt>"<dfn export for="user navigation involvement"
   data-x="uni-browser-ui"><code>browser UI</code></dfn>"</dt>
   <dd>The navigation was initiated by the user via browser UI mechanisms.</dd>

   <dt>"<dfn export for="user navigation involvement"
   data-x="uni-activation"><code>activation</code></dfn>"</dt>
   <dd>The navigation was initiated by the user via the <span>activation behavior</span> of an
   element.</dd>

   <dt>"<dfn export for="user navigation involvement"
   data-x="uni-none"><code>none</code></dfn>"</dt>
   <dd>The navigation was not initiated by the user.</dd>
  </dl>

  <p>For convenience at certain call sites, the <dfn for="Event" data-x="event-uni">user navigation
  involvement</dfn> for an <code>Event</code> <var>event</var> is defined as follows:</p>

  <ol>
   <li><p><span>Assert</span>: this algorithm is being called as part of an <span>activation
   behavior</span> definition.</p></li>

   <li><p><span>Assert</span>: <var>event</var>'s <code data-x="dom-Event-type">type</code> is
   "<code data-x="event-click">click</code>".</p></li>

   <li><p>If <var>event</var>'s <code data-x="dom-Event-isTrusted">isTrusted</code> is initialized
   to true, then return "<code data-x="uni-activation">activation</code>".</p></li>

   <li><p>Return "<code data-x="uni-none">none</code>".</p></li>
  </ol>


  <h5>Beginning navigation</h5>

  <!-- NAVIGATE <dfn>navigate</dfn> -->
  <!-- For places that _call_ this, as opposed to just referring to
  it, search for "DONAV" -->

  <p>To <dfn export>navigate</dfn> a <span>navigable</span> <var>navigable</var> to a
  <span>URL</span> <var>url</var> using a <code>Document</code> <var
  id="source-browsing-context">sourceDocument</var>, with an optional <span>POST resource</span>,
  string, or null <dfn for="navigate" data-x="navigation-resource"
  export><var>documentResource</var></dfn> (default null), an optional <span
  data-x="concept-response">response</span>-or-null <dfn for="navigate"
  data-x="navigation-response" export><var>response</var></dfn> (default null), an optional boolean
  <dfn for="navigate" id="exceptions-enabled" export><var>exceptionsEnabled</var></dfn> (default
  false), an optional <code>NavigationHistoryBehavior</code> <dfn for="navigate"
  data-x="navigation-hh" export><var>historyHandling</var></dfn> (default "<code
  data-x="NavigationHistoryBehavior-auto">auto</code>"), an optional <span>serialized
  state</span>-or-null <dfn for="navigate" data-x="navigation-navigation-api-state"
  export><var>navigationAPIState</var></dfn> (default null), an optional <span>entry list</span> or
  null <dfn for="navigate" data-x="navigation-form-data-entry-list"
  export><var>formDataEntryList</var></dfn> (default null), an optional <span>referrer
  policy</span> <dfn data-x="navigation-referrer-policy"><var>referrerPolicy</var></dfn> (default
  the empty string), an optional <span>user navigation involvement</span> <dfn
  data-x="navigation-user-involvement"><var>userInvolvement</var></dfn> (default "<code
  data-x="uni-none">none</code>"), and an optional <code>Element</code> <dfn
  data-x="navigation-source-element"><var>sourceElement</var></dfn> (default null):</p>

  <ol>
   <li><p>Let <var>cspNavigationType</var> be "<code data-x="">form-submission</code>" if
   <var>formDataEntryList</var> is non-null; otherwise "<code data-x="">other</code>".</p></li>

   <li><p>Let <var>sourceSnapshotParams</var> be the result of <span>snapshotting source snapshot
   params</span> given <var>sourceDocument</var>.</p></li>

   <li><p>Let <var>initiatorOriginSnapshot</var> be <var>sourceDocument</var>'s <span
   data-x="concept-document-origin">origin</span>.</p></li>

   <li><p>Let <var>initiatorBaseURLSnapshot</var> be <var>sourceDocument</var>'s <span>document base
   URL</span>.</p></li>

   <li id="sandboxLinks">
    <p>If <var>sourceDocument</var>'s <span>node navigable</span> is not <span>allowed by
    sandboxing to navigate</span> <var>navigable</var> given <var>sourceSnapshotParams</var>,
    then:</p>

    <ol>
     <li><p>If <var>exceptionsEnabled</var> is true, then throw a
     <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li><p>Let <var>navigationId</var> be the result of <span>generating a random UUID</span>.
   <ref>WEBCRYPTO</ref></p></li>

   <li>
    <p>If the <span>surrounding agent</span> is equal to <var>navigable</var>'s <span
    data-x="nav-document">active document</span>'s <span>relevant agent</span>, then continue these
    steps. Otherwise, <span>queue a global task</span> on the <span>navigation and traversal task
    source</span> given <var>navigable</var>'s <span data-x="nav-window">active window</span> to
    continue these steps.</p>

    <div class="note">
     <p>We do this because we are about to look at a lot of properties of <var>navigable</var>'s
     <span data-x="nav-document">active document</span>, which are in theory only accessible over
     in the appropriate <span>event loop</span>. (But, we do not want to unconditionally queue a
     task, since &mdash; for example &mdash; same-event-loop <span
     data-x="navigate-fragid">fragment navigations</span> need to take effect synchronously.)</p>

     <p>Another implementation strategy would be to replicate the relevant information across event
     loops, or into a canonical "browser process", so that it can be consulted without queueing a
     task. This could give different results than what we specify here in edge cases, where the
     relevant properties have changed over in the target event loop but not yet been replicated.
     Further testing is needed to determine which of these strategies best matches browser
     behavior, in such racy edge cases.</p>
    </div>
   </li>

   <li><p>If <var>navigable</var>'s <span data-x="nav-document">active document</span>'s
   <span>unload counter</span> is greater than 0, then invoke <span>WebDriver BiDi navigation
   failed</span> with <var>navigable</var> and a <span>WebDriver BiDi navigation status</span> whose
   <span data-x="navigation-status-id">id</span> is <var>navigationId</var>, <span
   data-x="navigation-status-status">status</span> is "<code
   data-x="navigation-status-canceled">canceled</code>", and <span
   data-x="navigation-status-url">url</span> is <var>url</var>, and return.</p></li>
   <!--
     this stops pages from hijacking the back/forward button, among other things
     https://www.hixie.ch/tests/adhoc/html/navigation/unload/
     https://github.com/whatwg/html/issues/1213
     https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1946 to 1955 -->

   <li><p>Let <var>container</var> be <var>navigable</var>'s <span
   data-x="nav-container">container</span>.</p></li>

   <li><p>If <var>container</var> is an <code>iframe</code> element and <span>will lazy load
   element steps</span> given <var>container</var> returns true, then <span>stop
   intersection-observing a lazy loading element</span> <var>container</var> and set
   <var>container</var>'s <span>lazy load resumption steps</span> to null.</p></li>

   <li id="navigate-convert-to-replace">
    <p>If <var>historyHandling</var> is "<code data-x="NavigationHistoryBehavior-auto">auto</code>",
    then:</p>

    <ol>
     <li><p>If <var>url</var> <span data-x="concept-url-equals">equals</span>
     <var>navigable</var>'s <span data-x="nav-document">active document</span>'s <span
     data-x="concept-document-url">URL</span>, and <var>initiatorOriginSnapshot</var> is <span>same
     origin</span> with <var>targetNavigable</var>'s <span data-x="nav-document">active
     document</span>'s <span data-x="concept-document-origin">origin</span>, then set
     <var>historyHandling</var> to "<code
     data-x="NavigationHistoryBehavior-replace">replace</code>".</p></li>

     <li><p>Otherwise, set <var>historyHandling</var> to "<code
     data-x="NavigationHistoryBehavior-push">push</code>".</p></li>
    </ol>
   </li>

   <li><p>If <span>the navigation must be a replace</span> given <var>url</var> and
   <var>navigable</var>'s <span data-x="nav-document">active document</span>, then set
   <var>historyHandling</var> to "<code
   data-x="NavigationHistoryBehavior-replace">replace</code>".</p></li>

   <li id="navigate-fragid-step">
    <p>If all of the following are true:</p>

    <ul>
     <li><p><var>documentResource</var> is null;</p></li>

     <li><p><var>response</var> is null;</p></li>

     <li><p><var>url</var> <span data-x="concept-url-equals">equals</span> <var>navigable</var>'s
     <span data-x="nav-active-history-entry">active session history entry</span>'s <span
     data-x="she-url">URL</span> with <span data-x="url equals exclude fragments"><i>exclude
     fragments</i></span> set to true; and</p></li>

     <li><p><var>url</var>'s <span data-x="concept-url-fragment">fragment</span> is
     non-null,</p></li>
    </ul>

    <p>then:</p>

    <ol>
     <li><p><span data-x="navigate-fragid">Navigate to a fragment</span> given <var>navigable</var>,
     <var>url</var>, <var>historyHandling</var>, <var>userInvolvement</var>,
     <var>sourceElement</var>, <var>navigationAPIState</var>, and <var>navigationId</var>.</p></li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li><p>If <var>navigable</var>'s <span data-x="nav-parent">parent</span> is non-null, then set
   <var>navigable</var>'s <span>is delaying <code data-x="event-load">load</code> events</span> to
   true.</p></li>

   <li><p>Let <var>targetSnapshotParams</var> be the result of <span>snapshotting target snapshot
   params</span> given <var>navigable</var>.</p></li>

   <li><p>Invoke <span>WebDriver BiDi navigation started</span> with <var>navigable</var> and a new
   <span>WebDriver BiDi navigation status</span> whose <span data-x="navigation-status-id">id</span>
   is <var>navigationId</var>, <span data-x="navigation-status-status">status</span> is "<code
   data-x="navigation-status-pending">pending</code>", and <span
   data-x="navigation-status-url">url</span> is <var>url</var>.</p></li>

   <li>
    <p>If <var>navigable</var>'s <span>ongoing navigation</span> is "<code
    data-x="">traversal</code>", then:</p>

    <ol>
     <li><p>Invoke <span>WebDriver BiDi navigation failed</span> with <var>navigable</var> and a new
     <span>WebDriver BiDi navigation status</span> whose <span
     data-x="navigation-status-id">id</span> is <var>navigationId</var>, <span
     data-x="navigation-status-status">status</span> is "<code
     data-x="navigation-status-canceled">canceled</code>", and <span
     data-x="navigation-status-url">url</span> is <var>url</var>.</p></li>

     <li><p>Return.</p></li>
    </ol>

    <p class="note">Any attempts to navigate a <span>navigable</span> that is currently <span
    data-x="apply the traverse history step">traversing</span> are ignored.</p>
   </li>

   <li>
    <p><span>Set the ongoing navigation</span> for <var>navigable</var> to
    <var>navigationId</var>.</p>

    <p class="note">This will have the effect of aborting other ongoing navigations of
    <var>navigable</var>, since at certain points during navigation changes to the <span>ongoing
    navigation</span> will cause further work to be abandoned.</p>
   </li>

   <li>
    <p>If <var>url</var>'s <span data-x="concept-url-scheme">scheme</span> is "<code
    data-x="javascript protocol">javascript</code>", then:</p>

    <ol>
     <li><p><span>Queue a global task</span> on the <span>navigation and traversal task
     source</span> given <var>navigable</var>'s <span data-x="nav-window">active window</span> to
     <span>navigate to a <code>javascript:</code> URL</span> given <var>navigable</var>,
     <var>url</var>, <var>historyHandling</var>, <var>sourceSnapshotParams</var>,
     <var>initiatorOriginSnapshot</var>, <var>userInvolvement</var>, and
     <var>cspNavigationType</var>.</p></li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li>
    <p>If all of the following are true:</p>

    <ul>
     <li><p><var>userInvolvement</var> is not "<code data-x="uni-browser-ui">browser
     UI</code>";</p></li>

     <li><p><var>navigable</var>'s <span data-x="nav-document">active document</span>'s <span
     data-x="concept-document-origin">origin</span> is <span>same origin-domain</span> with
     <var>sourceDocument</var>'s <span data-x="concept-document-origin">origin</span>;</p></li>

     <li><p><var>navigable</var>'s <span data-x="nav-document">active document</span>'s <span>is
     initial <code>about:blank</code></span> is false; and</p></li>

     <li><p><var>url</var>'s <span data-x="concept-url-scheme">scheme</span> is a <span>fetch
     scheme</span>,</p></li>
    </ul>

    <p>then:</p>

    <ol>
     <li><p>Let <var>navigation</var> be <var>navigable</var>'s <span data-x="nav-window">active
     window</span>'s <span data-x="window-navigation-api">navigation API</span>.</p></li>

     <li><p>Let <var>entryListForFiring</var> be <var>formDataEntryList</var> if
     <var>documentResource</var> is a <span>POST resource</span>; otherwise, null.</p></li>

     <li><p>Let <var>navigationAPIStateForFiring</var> be <var>navigationAPIState</var> if
     <var>navigationAPIState</var> is not null; otherwise,
     <span>StructuredSerializeForStorage</span>(undefined).</p></li>

     <li><p>Let <var>continue</var> be the result of <span data-x="fire a push/replace/reload
     navigate event">firing a push/replace/reload <code data-x="event-navigate">navigate</code>
     event</span> at <var>navigation</var> with <i
     data-x="fire-navigate-prr-navigationType">navigationType</i> set to <var>historyHandling</var>,
     <i data-x="fire-navigate-prr-isSameDocument">isSameDocument</i> set to false, <i
     data-x="fire-navigate-prr-userInvolvement">userInvolvement</i> set to
     <var>userInvolvement</var>, <i data-x="fire-navigate-prr-sourceElement">sourceElement</i> set
     to <var>sourceElement</var>, <i
     data-x="fire-navigate-prr-formDataEntryList">formDataEntryList</i> set to
     <var>entryListForFiring</var>, <i data-x="fire-navigate-prr-destinationURL">destinationURL</i>
     set to <var>url</var>, and <i
     data-x="fire-navigate-prr-navigationAPIState">navigationAPIState</i> set to
     <var>navigationAPIStateForFiring</var>.</p></li>

     <li><p>If <var>continue</var> is false, then return.</p>
    </ol>

    <p class="note">It is possible for navigations with <var>userInvolvement</var> of "<code
    data-x="uni-browser-ui">browser UI</code>" or initiated by a <span data-x="same
    origin-domain">cross origin-domain</span> <var>sourceDocument</var> to fire <code
    data-x="event-navigate">navigate</code> events, if they go through the earlier <span
    data-x="navigate-fragid">navigate to a fragment</span> path.</p>
   </li>

   <li>
    <p><span>In parallel</span>, run these steps:</p>

    <ol>
     <li><p>Let <var>unloadPromptCanceled</var> be the result of <span>checking if unloading is
     canceled</span> for <var>navigable</var>'s <span data-x="nav-document">active
     document</span>'s <span>inclusive descendant navigables</span>.</p>

     <li>
      <p>If <var>unloadPromptCanceled</var> is true, or <var>navigable</var>'s <span>ongoing
      navigation</span> is no longer <var>navigationId</var>, then:</p>

      <ol>
       <li><p>Invoke <span>WebDriver BiDi navigation failed</span> with <var>navigable</var> and a
       new <span>WebDriver BiDi navigation status</span> whose <span
       data-x="navigation-status-id">id</span> is <var>navigationId</var>, <span
       data-x="navigation-status-status">status</span> is "<code
       data-x="navigation-status-canceled">canceled</code>", and <span
       data-x="navigation-status-url">url</span> is <var>url</var>.</p></li>

       <li><p>Abort these steps.</p></li>
      </ol>
     </li>

     <li><p><span>Queue a global task</span> on the <span>navigation and traversal task
     source</span> given <var>navigable</var>'s <span data-x="nav-window">active window</span> to
     <span>abort a document and its descendants</span> given <var>navigable</var>'s <span
     data-x="nav-document">active document</span>.</p></li>

     <li id="navigation-create-document-state">
      <p>Let <var>documentState</var> be a new <span>document state</span> with</p>

      <dl class="props">
       <dt><span data-x="document-state-request-referrer-policy">request referrer policy</span></dt>
       <dd><var>referrerPolicy</var></dd>

       <dt><span data-x="document-state-initiator-origin">initiator origin</span></dt>
       <dd><var>initiatorOriginSnapshot</var></dd>

       <dt><span data-x="document-state-resource">resource</span></dt>
       <dd><var>documentResource</var></dd>

       <dt><span data-x="document-state-nav-target-name">navigable target name</span></dt>
       <dd><var>navigable</var>'s <span data-x="nav-target">target name</span></dd>
      </dl>

      <p class="note">The <span data-x="document-state-nav-target-name">navigable target
      name</span> can get cleared under various conditions later in the navigation process, before
      the document state is finalized.</p>
     </li>

     <li>
      <p>If <var>url</var> <span>matches <code>about:blank</code></span> or
      is <code>about:srcdoc</code>, then:</p>

      <ol>
       <li><p>Set <var>documentState</var>'s <span data-x="document-state-origin">origin</span> to
       <var>initiatorOriginSnapshot</var>.</p></li>

       <li><p>Set <var>documentState</var>'s <span data-x="document-state-about-base-url">about base
       URL</span> to <var>initiatorBaseURLSnapshot</var>.</p></li>
      </ol>
     </li>

     <li><p>Let <var>historyEntry</var> be a new <span>session history entry</span>, with its <span
     data-x="she-url">URL</span> set to <var>url</var> and its <span
     data-x="she-document-state">document state</span> set to
     <var>documentState</var>.</p></li>

     <li><p>Let <var>navigationParams</var> be null.</p></li>

     <li>
      <p>If <var>response</var> is non-null:</p>

      <p class="note" id="note-navigate-called-with-response">The <span>navigate</span> algorithm
      is only supplied with a <span data-x="concept-response">response</span> as part of the
      <code>object</code> and <code>embed</code> processing models, or for processing parts of
      <span data-x="navigate-multipart-x-mixed-replace"><code>multipart/x-mixed-replace</code>
      responses</span> after the initial response.</p>

      <ol>
       <li><p>Let <var>policyContainer</var> be the result of <span data-x="determining navigation
       params policy container">determining navigation params policy container</span> given <span
       data-x="concept-response">response</span>'s <span data-x="concept-response-url">URL</span>,
       null, a <span data-x="clone a policy container">clone</span> of the
       <var>sourceDocument</var>'s <span data-x="concept-document-policy-container">policy
       container</span>, <var>navigable</var>'s <span data-x="nav-container-document">container
       document</span>'s <span data-x="concept-document-policy-container">policy container</span>,
       and null.</p></li>

       <li><p>Let <var>finalSandboxFlags</var> be the <span data-x="set union">union</span> of
       <var>targetSnapshotParams</var>'s <span data-x="target-snapshot-params-sandbox">sandboxing
       flags</span> and <var>policyContainer</var>'s <span data-x="policy-container-csp-list">CSP
       list</span>'s <span>CSP-derived sandboxing flags</span>.</p></li>

       <li><p>Let <var>responseOrigin</var> be the result of <span>determining the origin</span>
       given <var>response</var>'s <span data-x="concept-response-url">URL</span>,
       <var>finalSandboxFlags</var>, and <var>documentState</var>'s <span
       data-x="document-state-initiator-origin">initiator origin</span>.</p></li>

       <li><p>Let <var>coop</var> be a new <span>opener policy</span>.</p></li>

       <li>
        <p>Let <var>coopEnforcementResult</var> be a new <span
        data-x="coop-enforcement-result">opener policy enforcement result</span>
        with</p>

        <dl class="props">
         <dt><span data-x="coop-enforcement-url">url</span></dt>
         <dd><var>response</var>'s <span data-x="concept-response-url">URL</span></dd>

         <dt><span data-x="coop-enforcement-origin">origin</span></dt>
         <dd><var>responseOrigin</var></dd>

         <dt><span data-x="coop-enforcement-coop">opener policy</span></dt>
         <dd><var>coop</var></dd>
        </dl>
       </li>

       <li>
        <p>Set <var>navigationParams</var> to a new <span>navigation params</span>, with</p>

        <dl class="props">
         <dt><span data-x="navigation-params-id">id</span></dt>
         <dd><var>navigationId</var></dd>

         <dt><span data-x="navigation-params-navigable">navigable</span></dt>
         <dd><var>navigable</var></dd>

         <dt><span data-x="navigation-params-request">request</span></dt>
         <dd>null</dd>

         <dt><span data-x="navigation-params-response">response</span></dt>
         <dd><var>response</var></dd>

         <dt><span data-x="navigation-params-fetch-controller">fetch controller</span></dt>
         <dd>null</dd>

         <dt><span data-x="navigation-params-commit-early-hints">commit early hints</span></dt>
         <dd>null</dd>

         <dt><span data-x="navigation-params-coop-enforcement-result">COOP enforcement result</span></dt>
         <dd><var>coopEnforcementResult</var></dd>

         <dt><span data-x="navigation-params-reserved-environment">reserved environment</span></dt>
         <dd>null</dd>

         <dt><span data-x="navigation-params-origin">origin</span></dt>
         <dd><var>responseOrigin</var></dd>

         <dt><span data-x="navigation-params-policy-container">policy container</span></dt>
         <dd><var>policyContainer</var></dd>

         <dt><span data-x="navigation-params-sandboxing">final sandboxing flag set</span></dt>
         <dd><var>finalSandboxFlags</var></dd>

         <dt><span data-x="navigation-params-coop">opener policy</span></dt>
         <dd><var>coop</var></dd>

         <dt><span data-x="navigation-params-nav-timing-type">navigation timing type</span></dt>
         <dd>"<code data-x="dom-navigationtimingtype-navigate">navigate</code>"</dd>

         <dt><span data-x="navigation-params-about-base-url">about base URL</span></dt>
         <dd><var>documentState</var>'s <span data-x="document-state-about-base-url">about base
         URL</span></dd>

         <dt><span data-x="navigation-params-user-involvement">user involvement</span></dt>
         <dd><var>userInvolvement</var></dd>
        </dl>
       </li>
      </ol>
     </li>

     <li>
      <p><span>Attempt to populate the history entry's document</span> for <var>historyEntry</var>,
      given <var>navigable</var>, "<code
      data-x="dom-navigationtimingtype-navigate">navigate</code>", <var>sourceSnapshotParams</var>,
      <var>targetSnapshotParams</var>, <var>userInvolvement</var>, <var>navigationId</var>,
      <var>navigationParams</var>, <var>cspNavigationType</var>, with <i
      data-x="attempt-to-populate-allow-post">allowPOST</i> set to true and <i
      data-x="attempt-to-populate-completion-steps">completionSteps</i> set to the following
      step:</p>

      <ol>
       <li><p><span data-x="tn-append-session-history-traversal-steps">Append session history
       traversal steps</span> to <var>navigable</var>'s <span
       data-x="nav-traversable">traversable</span> to <span>finalize a cross-document
       navigation</span> given <var>navigable</var>, <var>historyHandling</var>,
       <var>userInvolvement</var>, and <var>historyEntry</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>
  </ol>


  <h5>Ending navigation</h5>

  <p>Although the usual cross-document navigation case will first foray into <a
  href="#populating-a-session-history-entry">populating a session history entry</a> with a
  <code>Document</code>, all navigations that don't get aborted will ultimately end up calling into
  one of the below algorithms.</p>


  <h6>The usual cross-document navigation case</h6>

  <p>To <dfn>finalize a cross-document navigation</dfn> given a <span>navigable</span>
  <var>navigable</var>, a <span>history handling behavior</span> <var>historyHandling</var>, a
  <span>user navigation involvement</span> <var>userInvolvement</var>, and a <span>session history
  entry</span> <var>historyEntry</var>:</p>

  <ol>
   <li><p><span>Assert</span>: this is running on <var>navigable</var>'s <span
   data-x="nav-traversable">traversable navigable's</span> <span
   data-x="tn-session-history-traversal-queue">session history traversal queue</span>.</p></li>

   <li><p>Set <var>navigable</var>'s <span>is delaying <code data-x="event-load">load</code>
   events</span> to false.</p></li>

   <li>
    <p>If <var>historyEntry</var>'s <span data-x="she-document">document</span> is null, then
    return.</p>

    <p class="note">This means that <span data-x="attempt to populate the history entry's
    document">attempting to populate the history entry's document</span> ended up not creating a
    document, as a result of e.g., the navigation being canceled by a subsequent navigation, a 204
    No Content response, etc.</p>
   </li>

   <li id="resetBCName">
    <p>If all of the following are true:</p>

    <ul>
     <li><p><var>navigable</var>'s <span data-x="nav-parent">parent</span> is null;</p></li>

     <li><p><var>historyEntry</var>'s <span data-x="she-document">document</span>'s <span
     data-x="concept-document-bc">browsing context</span> is not an <span>auxiliary browsing
     context</span> whose <span>opener browsing context</span> is non-null; and</p></li>

     <li><p><var>historyEntry</var>'s <span data-x="she-document">document</span>'s <span
     data-x="concept-document-origin">origin</span> is not <var>navigable</var>'s <span
     data-x="nav-document">active document</span>'s <span
     data-x="concept-document-origin">origin</span>,</p></li>
    </ul>

    <p>then set <var>historyEntry</var>'s <span data-x="she-document-state">document state</span>'s
    <span data-x="document-state-nav-target-name">navigable target name</span> to the empty
    string.</p>
   </li>

   <li><p>Let <var>entryToReplace</var> be <var>navigable</var>'s <span
   data-x="nav-active-history-entry">active session history entry</span> if
   <var>historyHandling</var> is "<code data-x="NavigationHistoryBehavior-replace">replace</code>",
   otherwise null.</p></li>

   <li><p>Let <var>traversable</var> be <var>navigable</var>'s <span
   data-x="nav-traversable">traversable navigable</span>.</p></li>

   <li><p>Let <var>targetStep</var> be null.</p></li>

   <li><p>Let <var>targetEntries</var> be the result of <span>getting session history
   entries</span> for <var>navigable</var>.</p></li>

   <li>
    <p>If <var>entryToReplace</var> is null, then:</p>

    <ol>
     <li><p><span>Clear the forward session history</span> of <var>traversable</var>.</p></li>

     <li><p>Set <var>targetStep</var> to <var>traversable</var>'s <span
     data-x="tn-current-session-history-step">current session history step</span> + 1.</p></li>

     <li><p>Set <var>historyEntry</var>'s <span data-x="she-step">step</span> to
     <var>targetStep</var>.</p></li>

     <li><p><span data-x="list append">Append</span> <var>historyEntry</var> to
     <var>targetEntries</var>.</p></li>
    </ol>

    <p>Otherwise:</p>

    <ol>
     <li><p><span data-x="list replace">Replace</span> <var>entryToReplace</var> with
     <var>historyEntry</var> in <var>targetEntries</var>.</p></li>

     <li><p>Set <var>historyEntry</var>'s <span data-x="she-step">step</span> to
     <var>entryToReplace</var>'s <span data-x="she-step">step</span>.</p></li>

     <li><p>If <var>historyEntry</var>'s <span data-x="she-document-state">document state</span>'s
     <span data-x="document-state-origin">origin</span> is <span>same origin</span> with
     <var>entryToReplace</var>'s <span data-x="she-document-state">document state</span>'s <span
     data-x="document-state-origin">origin</span>, then set <var>historyEntry</var>'s <span
     data-x="she-navigation-api-key">navigation API key</span> to <var>entryToReplace</var>'s <span
     data-x="she-navigation-api-key">navigation API key</span>.</p></li>

     <li><p>Set <var>targetStep</var> to <var>traversable</var>'s <span
     data-x="tn-current-session-history-step">current session history step</span>.</p></li>
    </ol>
   </li>

   <li><p><span>Apply the push/replace history step</span> <var>targetStep</var> to
   <var>traversable</var> given <var>historyHandling</var> and <var>userInvolvement</var>.</p></li>
  </ol>


  <h6><dfn data-x="javascript protocol">The <code>javascript:</code> URL special case</dfn></h6>

  <p class="XXX"><code data-x="javascript protocol">javascript:</code> URLs have a <a
  href="https://github.com/whatwg/html/labels/topic%3A%20javascript%3A%20URLs">dedicated label</a>
  on the issue tracker documenting various problems with their specification.</p>

  <p>To <dfn>navigate to a <code>javascript:</code> URL</dfn>, given a <span>navigable</span>
  <var>targetNavigable</var>, a <span>URL</span> <var>url</var>, a <span>history handling
  behavior</span> <var>historyHandling</var>, a <span>source snapshot params</span>
  <var>sourceSnapshotParams</var>, an <span>origin</span> <var>initiatorOrigin</var>, a <span>user
  navigation involvement</span> <var>userInvolvement</var>, and a string
  <var>cspNavigationType</var>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>historyHandling</var> is "<code
   data-x="NavigationHistoryBehavior-replace">replace</code>".</p></li>

   <li><p><span>Set the ongoing navigation</span> for <var>targetNavigable</var> to null.</p></li>

   <li><p>If <var>initiatorOrigin</var> is not <span>same origin-domain</span> with
   <var>targetNavigable</var>'s <span data-x="nav-document">active document</span>'s <span
   data-x="concept-document-origin">origin</span>, then return.</p></li>

   <li>
    <p>Let <var>request</var> be a new <span data-x="concept-request">request</span> whose <span
    data-x="concept-request-url">URL</span> is <var>url</var> and whose <span
    data-x="concept-request-policy-container">policy container</span> is
    <var>sourceSnapshotParams</var>'s <span data-x="source-snapshot-params-policy-container">source
    policy container</span>.</p>

    <p class="note">This is a synthetic <span data-x="concept-request">request</span> solely for
    plumbing into the next step. It will never hit the network.</p>
   </li>

   <li><p>If the result of <span>should navigation request of type be blocked by Content Security
   Policy?</span> given <var>request</var> and <var>cspNavigationType</var> is "<code
   data-x="">Blocked</code>", then return. <ref>CSP</ref></p></li>

   <li><p>Let <var>newDocument</var> be the result of <span data-x="evaluate a javascript:
   URL">evaluating a <code>javascript:</code> URL</span> given <var>targetNavigable</var>,
   <var>url</var>, <var>initiatorOrigin</var>, and <var>userInvolvement</var>.</p></li>

   <li>
    <p>If <var>newDocument</var> is null, then return.</p>

    <p class="note">In this case, some JavaScript code was executed, but no new
    <code>Document</code> was created, so we will not perform a navigation.</p>
   </li>

   <li><p><span>Assert</span>: <var>initiatorOrigin</var> is <var>newDocument</var>'s <span
   data-x="concept-document-origin">origin</span>.</p></li>

   <li><p>Let <var>entryToReplace</var> be <var>targetNavigable</var>'s <span
   data-x="nav-active-history-entry">active session history entry</span>.</p></li>

   <li><p>Let <var>oldDocState</var> be <var>entryToReplace</var>'s <span
   data-x="she-document-state">document state</span>.</p></li>

   <li>
    <p>Let <var>documentState</var> be a new <span>document state</span> with</p>

    <dl class="props">
     <dt><span data-x="document-state-document">document</span></dt>
     <dd><var>newDocument</var></dd>

     <dt><span data-x="document-state-history-policy-container">history policy container</span></dt>
     <dd>a <span data-x="clone a policy container">clone</span> of the <var>oldDocState</var>'s
     <span data-x="document-state-history-policy-container">history policy container</span> if it is
     non-null; null otherwise</dd>

     <dt><span data-x="document-state-request-referrer">request referrer</span></dt>
     <dd><var>oldDocState</var>'s <span data-x="document-state-request-referrer">request
     referrer</span></dd>

     <dt><span data-x="document-state-request-referrer-policy">request referrer policy</span></dt>
     <dd><var>oldDocState</var>'s <span data-x="document-state-request-referrer-policy">request
     referrer policy</span> <span class="XXX">or should this be the <var>referrerPolicy</var> that
     was passed to <span>navigate</span>?</span></dd>

     <dt><span data-x="document-state-initiator-origin">initiator origin</span></dt>
     <dd><var>initiatorOrigin</var></dd>

     <dt><span data-x="document-state-origin">origin</span></dt>
     <dd><var>initiatorOrigin</var></dd>

     <dt><span data-x="document-state-about-base-url">about base URL</span></dt>
     <dd><var>oldDocState</var>'s <span data-x="document-state-about-base-url">about base
     URL</span></dd>

     <dt><span data-x="document-state-resource">resource</span></dt>
     <dd>null</dd>

     <dt><span data-x="document-state-ever-populated">ever populated</span></dt>
     <dd>true</dd>

     <dt><span data-x="document-state-nav-target-name">navigable target name</span></dt>
     <dd><var>oldDocState</var>'s <span data-x="document-state-nav-target-name">navigable target
     name</span></dd>
    </dl>
   </li>

   <li>
    <p>Let <var>historyEntry</var> be a new <span>session history entry</span>, with</p>

    <dl class="props">
     <dt><span data-x="she-url">URL</span></dt>
     <dd><var>entryToReplace</var>'s <span data-x="she-url">URL</span></dd>

     <dt><span data-x="she-document-state">document state</span></dt>
     <dd><var>documentState</var></dd>
    </dl>

    <p class="note">For the <span data-x="she-url">URL</span>, we do <em>not</em> use
    <var>url</var>, i.e. the actual <code data-x="javascript protocol">javascript:</code> URL that
    the <span>navigate</span> algorithm was called with. This means <code data-x="javascript
    protocol">javascript:</code> URLs are never stored in session history, and so can never be
    traversed to.</p>
   </li>

   <li><p><span data-x="tn-append-session-history-traversal-steps">Append session history traversal
   steps</span> to <var>targetNavigable</var>'s <span data-x="nav-traversable">traversable</span>
   to <span>finalize a cross-document navigation</span> with <var>targetNavigable</var>,
   <var>historyHandling</var>, <var>userInvolvement</var>, and <var>historyEntry</var>.</p></li>
  </ol>

  <p>To <dfn>evaluate a <code>javascript:</code> URL</dfn> given a <span>navigable</span>
  <var>targetNavigable</var>, a <span>URL</span> <var>url</var>, an <span>origin</span>
  <var>newDocumentOrigin</var>, and a <span>user navigation involvement</span>
  <var>userInvolvement</var>:</p>

  <ol>
   <li><p>Let <var>urlString</var> be the result of running the <span
   data-x="concept-url-serializer">URL serializer</span> on <var>url</var>.</p></li>

   <li><p>Let <var>encodedScriptSource</var> be the result of removing the leading "<code
   data-x="">javascript:</code>" from <var>urlString</var>.</p></li>

   <li><p>Let <var>scriptSource</var> be the <span data-x="UTF-8 decode">UTF-8 decoding</span> of
   the <span data-x="percent-decode">percent-decoding</span> of
   <var>encodedScriptSource</var>.</p></li>

   <!-- <body onload="&#xFFFE;w('test')"> and javascript:%EF%BB%BF'test' both work, so we assume
   that JavaScript is stripping the BOM, and we don't have to -->

   <!-- https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2639 -> about:blank
   https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2640 -> javascript:'test' in
   Firefox, about:blank otherwise -->

   <li><p>Let <var>settings</var> be <var>targetNavigable</var>'s <span
   data-x="nav-document">active document</span>'s <span>relevant settings object</span>.</p></li>

   <li><p>Let <var>baseURL</var> be <var>settings</var>'s <span>API base URL</span>.</p></li>

   <li><p>Let <var>script</var> be the result of <span>creating a classic script</span> given
   <var>scriptSource</var>, <var>settings</var>, <var>baseURL</var>, and the <span>default script
   fetch options</span>.</p></li>

   <li><p>Let <var>evaluationStatus</var> be the result of <span data-x="run a classic
   script">running the classic script</span> <var>script</var>.</p></li>

   <li><p>Let <var>result</var> be null.</p></li>

   <li><p>If <var>evaluationStatus</var> is a normal completion, and
   <var>evaluationStatus</var>.[[Value]] is a String, then set <var>result</var> to
   <var>evaluationStatus</var>.[[Value]].</p></li>

   <li><p>Otherwise, return null.</p></li>

   <li>
    <p>Let <var>response</var> be a new <span data-x="concept-response">response</span> with</p>

    <dl class="props">
     <dt><span data-x="concept-response-url">URL</span></dt>
     <dd><var>targetNavigable</var>'s <span data-x="nav-document">active document</span>'s <span
     data-x="concept-document-url">URL</span></dd>

     <dt><span data-x="concept-response-header-list">header list</span></dt>
     <dd>« (`<code>Content-Type</code>`, `<code data-x="">text/html;charset=utf-8</code>`) »</dd>

     <dt><span data-x="concept-response-body">body</span></dt>
     <dd>the <span data-x="UTF-8 encode">UTF-8 encoding</span> of <var>result</var>, <span>as a
     body</span></dd>
    </dl>

    <p class="note">The encoding to UTF-8 means that unpaired <span
    data-x="surrogate">surrogates</span> will not roundtrip, once the HTML parser decodes the
    response body.</p>
   </li>

   <li><p>Let <var>policyContainer</var> be <var>targetNavigable</var>'s <span
   data-x="nav-document">active document</span>'s <span
   data-x="concept-document-policy-container">policy container</span>.</p>

   <li><p>Let <var>finalSandboxFlags</var> be <var>policyContainer</var>'s <span
   data-x="policy-container-csp-list">CSP list</span>'s <span>CSP-derived sandboxing
   flags</span>.</p></li>

   <li><p>Let <var>coop</var> be <var>targetNavigable</var>'s <span data-x="nav-document">active
   document</span>'s <span data-x="concept-document-coop">opener policy</span>.</p></li>

   <li>
    <p>Let <var>coopEnforcementResult</var> be a new <span data-x="coop-enforcement-result">opener
    policy enforcement result</span> with</p>

    <dl class="props">
     <dt><span data-x="coop-enforcement-url">url</span></dt>
     <dd><var>url</var></dd>

     <dt><span data-x="coop-enforcement-origin">origin</span></dt>
     <dd><var>newDocumentOrigin</var></dd>

     <dt><span data-x="coop-enforcement-coop">opener policy</span></dt>
     <dd><var>coop</var></dd>
    </dl>
   </li>

   <li>
    <p>Let <var>navigationParams</var> be a new <span>navigation params</span>, with</p>

    <dl class="props">
     <dt><span data-x="navigation-params-id">id</span></dt>
     <dd><var>navigationId</var></dd>

     <dt><span data-x="navigation-params-navigable">navigable</span></dt>
     <dd><var>targetNavigable</var></dd>

     <dt><span data-x="navigation-params-request">request</span></dt>
     <dd>null <span class="XXX">this will cause the referrer of the resulting <code>Document</code>
     to be null; is that correct?</span></dd>

     <dt><span data-x="navigation-params-response">response</span></dt>
     <dd><var>response</var></dd>

     <dt><span data-x="navigation-params-fetch-controller">fetch controller</span></dt>
     <dd>null</dd>

     <dt><span data-x="navigation-params-commit-early-hints">commit early hints</span></dt>
     <dd>null</dd>

     <dt><span data-x="navigation-params-coop-enforcement-result">COOP enforcement result</span></dt>
     <dd><var>coopEnforcementResult</var></dd>

     <dt><span data-x="navigation-params-reserved-environment">reserved environment</span></dt>
     <dd>null</dd>

     <dt><span data-x="navigation-params-origin">origin</span></dt>
     <dd><var>newDocumentOrigin</var></dd>

     <dt><span data-x="navigation-params-policy-container">policy container</span></dt>
     <dd><var>policyContainer</var></dd>

     <dt><span data-x="navigation-params-sandboxing">final sandboxing flag set</span></dt>
     <dd><var>finalSandboxFlags</var></dd>

     <dt><span data-x="navigation-params-coop">opener policy</span></dt>
     <dd><var>coop</var></dd>

     <dt><span data-x="navigation-params-nav-timing-type">navigation timing type</span></dt>
     <dd>"<code data-x="dom-navigationtimingtype-navigate">navigate</code>"</dd>

     <dt><span data-x="navigation-params-about-base-url">about base URL</span></dt>
     <dd><var>targetNavigable</var>'s <span data-x="nav-document">active document</span>'s <span
     data-x="concept-document-about-base-url">about base URL</span></dd>

     <dt><span data-x="navigation-params-user-involvement">user involvement</span></dt>
     <dd><var>userInvolvement</var></dd>
    </dl>
   </li>

   <li><p>Return the result of <span data-x="navigate-html">loading an HTML document</span> given
   <var>navigationParams</var>.</p></li>
  </ol>


  <h6 id="scroll-to-fragid">Fragment navigations</h6>

  <p>To <dfn data-x="navigate-fragid">navigate to a fragment</dfn> given a <span>navigable</span>
  <var>navigable</var>, a <span>URL</span> <var>url</var>, a <span>history handling behavior</span>
  <var>historyHandling</var>, a <span>user navigation involvement</span> <var>userInvolvement</var>,
  an <code>Element</code>-or-null <var>sourceElement</var>, a <span>serialized state</span>-or-null
  <var>navigationAPIState</var>, and a <span>navigation ID</span> <var>navigationId</var>:</p>

  <ol>
   <li><p>Let <var>navigation</var> be <var>navigable</var>'s <span data-x="nav-window">active
   window</span>'s <span data-x="window-navigation-api">navigation API</span>.</p></li>

   <li><p>Let <var>destinationNavigationAPIState</var> be <var>navigable</var>'s <span
   data-x="nav-active-history-entry">active session history entry</span>'s <span
   data-x="she-navigation-api-state">navigation API state</span>.</p></li>

   <li><p>If <var>navigationAPIState</var> is not null, then set
   <var>destinationNavigationAPIState</var> to <var>navigationAPIState</var>.</p></li>

   <li><p>Let <var>continue</var> be the result of <span data-x="fire a push/replace/reload navigate
   event">firing a push/replace/reload <code data-x="event-navigate">navigate</code> event</span> at
   <var>navigation</var> with <i data-x="fire-navigate-prr-navigationType">navigationType</i> set to
   <var>historyHandling</var>, <i data-x="fire-navigate-prr-isSameDocument">isSameDocument</i> set
   to true, <i data-x="fire-navigate-prr-userInvolvement">userInvolvement</i> set to
   <var>userInvolvement</var>, <i data-x="fire-navigate-prr-sourceElement">sourceElement</i> set
   to <var>sourceElement</var>, <i data-x="fire-navigate-prr-destinationURL">destinationURL</i>
   set to <var>url</var>, and <i
   data-x="fire-navigate-prr-navigationAPIState">navigationAPIState</i> set to
   <var>destinationNavigationAPIState</var>.</p></li>

   <li><p>If <var>continue</var> is false, then return.</p>

   <li>
    <p>Let <var>historyEntry</var> be a new <span>session history entry</span>, with</p>

    <dl class="props">
     <dt><span data-x="she-url">URL</span></dt>
     <dd><var>url</var></dd>

     <dt><span data-x="she-document-state">document state</span></dt>
     <dd><var>navigable</var>'s <span data-x="nav-active-history-entry">active session history
     entry</span>'s <span data-x="she-document-state">document state</span></dd>

     <dt><span data-x="she-navigation-api-state">navigation API state</span></dt>
     <dd><var>destinationNavigationAPIState</var></dd>

     <dt><span data-x="she-scroll-restoration-mode">scroll restoration mode</span></dt>
     <dd><var>navigable</var>'s <span data-x="nav-active-history-entry">active session history
     entry</span>'s <span data-x="she-scroll-restoration-mode">scroll restoration mode</span></dd>
    </dl>

    <div class="note">
     <p>For navigations peformed with <code
     data-x="dom-Navigation-navigate">navigation.navigate()</code>, the value provided by the <code
     data-x="dom-NavigationNavigateOptions-state">state</code> option is used for the new <span
     data-x="she-navigation-api-state">navigation API state</span>. (This will set it to the
     serialization of undefined, if no value is provided for that option.) For other fragment
     navigations, including user-initiated ones, the <span
     data-x="she-navigation-api-state">navigation API state</span> is carried over from the previous
     entry.</p>

     <p>The <span data-x="she-classic-history-api-state">classic history API state</span> is never
     carried over.</p>
    </div>
   </li>

   <li><p>Let <var>entryToReplace</var> be <var>navigable</var>'s <span
   data-x="nav-active-history-entry">active session history entry</span> if
   <var>historyHandling</var> is "<code data-x="NavigationHistoryBehavior-replace">replace</code>",
   otherwise null.</p></li>

   <li><p>Let <var>history</var> be <var>navigable</var>'s <span data-x="nav-document">active
   document</span>'s <span data-x="doc-history">history object</span>.</p></li>

   <li><p>Let <var>scriptHistoryIndex</var> be <var>history</var>'s <span
   data-x="concept-history-index">index</span>.</p></li>

   <li><p>Let <var>scriptHistoryLength</var> be <var>history</var>'s <span
   data-x="concept-history-length">length</span>.</p></li>

   <li>
    <p>If <var>historyHandling</var> is "<code data-x="NavigationHistoryBehavior-push">push</code>",
    then:</p>

    <ol>
     <li><p>Set <var>history</var>'s <span data-x="concept-history-state">state</span> to
     null.</p></li>

     <li><p>Increment <var>scriptHistoryIndex</var>.</p></li>

     <li><p>Set <var>scriptHistoryLength</var> to <var>scriptHistoryIndex</var> + 1.</p></li>
    </ol>
   </li>

   <li><p>Set <var>navigable</var>'s <span data-x="nav-document">active document</span>'s <span
   data-x="concept-document-url">URL</span> to <var>url</var>.</p></li>

   <li><p>Set <var>navigable</var>'s <span data-x="nav-active-history-entry">active session history
   entry</span> to <var>historyEntry</var>.</p></li>

   <li>
    <p><span>Update document for history step application</span> given <var>navigable</var>'s
    <span data-x="nav-document">active document</span>, <var>historyEntry</var>, true,
    <var>scriptHistoryIndex</var>, <var>scriptHistoryLength</var>, and
    <var>historyHandling</var>.</p>

    <p class="note">This algorithm will be called twice as a result of a single fragment
    navigation: once synchronously, where best-guess values <var>scriptHistoryIndex</var> and
    <var>scriptHistoryLength</var> are set, <code data-x="dom-history-state">history.state</code>
    is nulled out, and various events are fired; and once asynchronously, where the final values for
    index and length are set, <code data-x="dom-history-state">history.state</code> remains
    untouched, and no events are fired.</p>
   </li>

   <li>
    <p><span>Scroll to the fragment</span> given <var>navigable</var>'s <span
    data-x="nav-document">active document</span>.</p>

    <p class="note">If the scrolling fails because the <code>Document</code> is new and the
    relevant <span data-x="concept-id">ID</span> has not yet been parsed, then the second
    asynchronous call to <span>update document for history step application</span> will take
    care of scrolling.</p>
   </li>

   <li><p>Let <var>traversable</var> be <var>navigable</var>'s <span
   data-x="nav-traversable">traversable navigable</span>.</p></li>

   <li>
    <p><span data-x="tn-append-session-history-sync-nav-steps">Append the following session history
    synchronous navigation steps</span> involving <var>navigable</var> to
    <var>traversable</var>:</p>

    <ol>
     <li><p><span>Finalize a same-document navigation</span> given <var>traversable</var>,
     <var>navigable</var>, <var>historyEntry</var>, <var>entryToReplace</var>,
     <var>historyHandling</var>, and <var>userInvolvement</var>.</p></li>

     <li><p>Invoke <span>WebDriver BiDi fragment navigated</span> with <var>navigable</var> and a
     new <span>WebDriver BiDi navigation status</span> whose <span
     data-x="navigation-status-id">id</span> is <var>navigationId</var>, <span
     data-x="navigation-status-url">url</span> is <var>url</var>, and <span
     data-x="navigation-status-status">status</span> is "<code
     data-x="navigation-status-complete">complete</code>".</p></li>
    </ol>
   </li>
  </ol>

  <p>To <dfn>finalize a same-document navigation</dfn> given a <span>traversable navigable</span>
  <var>traversable</var>, a <span>navigable</span> <var>targetNavigable</var>, a <span>session
  history entry</span> <var>targetEntry</var>, a <span>session history entry</span>-or-null
  <var>entryToReplace</var>, a <span>history handling behavior</span> <var>historyHandling</var>,
  and a <span>user navigation involvement</span> <var>userInvolvement</var>:</p>

  <p class="note">This is used by both <span data-x="navigate-fragid">fragment navigations</span>
  and by the <span>URL and history update steps</span>, which are the only synchronous updates to
  session history. By virtue of being synchronous, those algorithms are performed outside of the
  <span>top-level traversable</span>'s <span data-x="tn-session-history-traversal-queue">session
  history traversal queue</span>. This puts them out of sync with the <span>top-level
  traversable</span>'s <span data-x="tn-current-session-history-step">current session history
  step</span>, so this algorithm is used to resolve conflicts due to race conditions.</p>

  <ol>
   <li><p><span>Assert</span>: this is running on <var>traversable</var>'s <span
   data-x="tn-session-history-traversal-queue">session history traversal queue</span>.</p></li>

   <li><p>If <var>targetNavigable</var>'s <span data-x="nav-active-history-entry">active session
   history entry</span> is not <var>targetEntry</var>, then return.</p></li>

   <li><p>Let <var>targetStep</var> be null.</p></li>

   <li><p>Let <var>targetEntries</var> be the result of <span>getting session history
   entries</span> for <var>targetNavigable</var>.</p></li>

   <li>
    <p>If <var>entryToReplace</var> is null, then:</p>

    <ol>
     <li><p><span>Clear the forward session history</span> of <var>traversable</var>.</p></li>

     <li><p>Set <var>targetStep</var> to <var>traversable</var>'s <span
     data-x="tn-current-session-history-step">current session history step</span> + 1.</p></li>

     <li><p>Set <var>targetEntry</var>'s <span data-x="she-step">step</span> to
     <var>targetStep</var>.</p></li>

     <li><p><span data-x="list append">Append</span> <var>targetEntry</var> to
     <var>targetEntries</var>.</p></li>
    </ol>

    <p>Otherwise:</p>

    <ol>
     <li><p><span data-x="list replace">Replace</span> <var>entryToReplace</var> with
     <var>targetEntry</var> in <var>targetEntries</var>.</p></li>

     <li><p>Set <var>targetEntry</var>'s <span data-x="she-step">step</span> to
     <var>entryToReplace</var>'s <span data-x="she-step">step</span>.</p></li>

     <li><p>Set <var>targetStep</var> to <var>traversable</var>'s <span
     data-x="tn-current-session-history-step">current session history step</span>.</p></li>
    </ol>
   </li>

   <li>
    <p><span>Apply the push/replace history step</span> <var>targetStep</var> to
    <var>traversable</var> given <var>historyHandling</var> and <var>userInvolvement</var>.</p>

    <p class="note">This is done even for "<code
    data-x="NavigationHistoryBehavior-replace">replace</code>" navigations, as it resolves race
    conditions across multiple synchronous navigations.</p>
   </li>
  </ol>


  <h6>Non-fetch schemes and external software</h6>

  <p>The input to <span>attempt to create a non-fetch scheme document</span> is the <dfn>non-fetch
  scheme navigation params</dfn> <span>struct</span>. It is a lightweight version of
  <span>navigation params</span> which only carries parameters relevant to the non-<span>fetch
  scheme</span> navigation case. It has the following <span data-x="struct item">items</span>:</p>

  <!--
   If modifying this struct, try to keep it somewhat parallel to the normal navigation params
   struct, where that makes sense. See the comment there.
  -->
  <dl>
   <dt><dfn data-x="non-fetch-scheme-params-id">id</dfn></dt>
   <dd>null or a <span>navigation ID</span></dd>

   <dt><dfn data-x="non-fetch-scheme-params-navigable">navigable</dfn></dt>
   <dd>the <span>navigable</span> experiencing the navigation</dd>

   <dt><dfn data-x="non-fetch-scheme-params-url">URL</dfn></dt>
   <dd>a <span>URL</span></dd>

   <dt><dfn data-x="non-fetch-scheme-params-target-sandbox">target snapshot sandboxing flags</dfn></dt>
   <dd>the <span>target snapshot params</span>'s <span
   data-x="target-snapshot-params-sandbox">sandboxing flags</span> present during navigation</dd>

   <dt><dfn data-x="non-fetch-scheme-params-source-activation">source snapshot has transient activation</dfn></dt>
   <dd>a copy of the <span>source snapshot params</span>'s <span
   data-x="source-snapshot-params-activation">has transient activation</span> boolean present during
   activation</dd>

   <dt><dfn data-x="non-fetch-scheme-params-initiator-origin">initiator origin</dfn></dt>
   <dd>
    <p>an <span>origin</span> possibly for use in a user-facing prompt to confirm the invocation of
    an external software package</p>
    <p class="note">This differs slightly from a <span data-x="she-document-state">document
    state</span>'s <span data-x="document-state-initiator-origin">initiator origin</span> in that a
    <span>non-fetch scheme navigation params</span>'s <span
    data-x="non-fetch-scheme-params-initiator-origin">initiator origin</span> follows redirects up
    to the last <span>fetch scheme</span> URL in a redirect chain that ends in a non-<span>fetch
    scheme</span> URL.</p>
   </dd>

   <dt><dfn data-x="non-fetch-scheme-params-nav-timing-type">navigation timing type</dfn></dt>
   <dd>a <code>NavigationTimingType</code> used for <span data-x="create the navigation timing
   entry">creating the navigation timing entry</span> for the new <code>Document</code> (if one is
   created)</dd>

   <dt><dfn data-x="non-fetch-scheme-params-user-involvement">user involvement</dfn></dt>
   <dd>a <span>user navigation involvement</span> used when <span
   data-x="obtain-browsing-context-navigation">obtaining a browsing context</span> for the new
   <code>Document</code> (if one is created)</dd>
  </dl>

  <p id="process-a-navigate-url-scheme">To <dfn>attempt to create a non-fetch scheme document</dfn>,
  given a <span>non-fetch scheme navigation params</span> <var>navigationParams</var>:</p>

  <ol>
   <li>Let <var>url</var> be <var>navigationParams</var>'s <span
   data-x="non-fetch-scheme-params-url">URL</span>.</li>

   <li>Let <var>navigable</var> be <var>navigationParams</var>'s <span
   data-x="non-fetch-scheme-params-navigable">navigable</span>.</li>

   <li>
    <p>If <var>url</var> is to be handled using a mechanism that does not affect
    <var>navigable</var>, e.g., because <var>url</var>'s <span
    data-x="concept-url-scheme">scheme</span> is handled externally, then:</p>

    <ol>
     <li><p><span>Hand-off to external software</span> given <var>url</var>, <var>navigable</var>,
     <var>navigationParams</var>'s <span data-x="non-fetch-scheme-params-target-sandbox">target
     snapshot sandboxing flags</span>, <var>navigationParams</var>'s <span
     data-x="non-fetch-scheme-params-source-activation">source snapshot has transient
     activation</span>, and <var>navigationParams</var>'s <span
     data-x="non-fetch-scheme-params-initiator-origin">initiator origin</span>.</p></li>

     <li><p>Return null.</p></li>
    </ol>
   </li>

   <li>
    <p>Handle <var>url</var> by displaying some sort of inline content, e.g., an error message
    because the specified scheme is not one of the supported protocols, or an inline prompt to allow
    the user to select <span data-x="dom-navigator-registerProtocolHandler">a registered
    handler</span> for the given scheme. Return the result of <span
    data-x="navigate-ua-inline">displaying the inline content</span> given <var>navigable</var>,
    <var>navigationParams</var>'s <span data-x="non-fetch-scheme-params-id">id</span>,
    <var>navigationParams</var>'s <span data-x="non-fetch-scheme-params-nav-timing-type">navigation
    timing type</span>, and <var>navigationParams</var>'s <span
    data-x="non-fetch-scheme-params-user-involvement">user involvement</span>.</p>

    <p class="note">In the case of a registered handler being used, <span>navigate</span> will be
    invoked with a new URL.</p>
   </li>
  </ol>

  <p>To <dfn>hand-off to external software</dfn> given a <span>URL</span> or <span
  data-x="concept-response">response</span> <var>resource</var>, a <span>navigable</span>
  <var>navigable</var>, a <span>sandboxing flag set</span> <var>sandboxFlags</var>, a boolean
  <var>hasTransientActivation</var>, and an <span>origin</span> <var>initiatorOrigin</var>, user
  agents should:</p>

  <ol>
   <li>
    <p>If all of the following are true:</p>

    <ul>
     <li><p><var>navigable</var> is not a <span>top-level traversable</span>;</p></li>

     <li><p><var>sandboxFlags</var> has its <span>sandboxed custom protocols navigation browsing
     context flag</span> set; and</p></li>

     <li><p><var>sandboxFlags</var> has its <span>sandboxed top-level navigation with user
     activation browsing context flag</span> set, or <var>hasTransientActivation</var> is
     false,</p></li>
    </ul>

    <p>then return without invoking the external software package.</p>

    <p class="note">Navigation inside an iframe toward external software can be seen by users as a
    new popup or a new top-level navigation. That's why its is allowed in sandboxed
    <code>iframe</code> only when one of <code
    data-x="attr-iframe-sandbox-allow-popups">allow-popups</code>, <code
    data-x="attr-iframe-sandbox-allow-top-navigation">allow-top-navigation</code>, <code
    data-x="attr-iframe-sandbox-allow-top-navigation-by-user-activation">allow-top-navigation-by-user-activation</code>,
    or <code
    data-x="attr-iframe-sandbox-allow-top-navigation-to-custom-protocols">allow-top-navigation-to-custom-protocols</code>
    is specified.</p>
   </li>

   <li>
    <p>Perform the appropriate handoff of <var>resource</var> while attempting to mitigate the risk
    that this is an attempt to exploit the target software. For example, user agents could prompt
    the user to confirm that <var>initiatorOrigin</var> is to be allowed to invoke the external
    software in question. In particular, if <var>hasTransientActivation</var> is false, then the
    user agent should not invoke the external software package without prior user confirmation.</p>

    <p class="example">For example, there could be a vulnerability in the target software's URL
    handler which a hostile page would attempt to exploit by tricking a user into clicking a
    link.</p>
   </li>
  </ol>


  <h5>Preventing navigation</h5>

  <p>A couple of scenarios can intervene early in the navigation process and put the whole thing to
  a halt. This can be especially exciting when multiple <span data-x="navigable">navigables</span>
  are navigating at the same time, due to a session history traversal.</p>

  <p>A <span>navigable</span> <var>source</var> is <dfn id="allowed-to-navigate">allowed by
  sandboxing to navigate</dfn> a second <span>navigable</span> <var>target</var>, given a
  <span>source snapshot params</span> <var>sourceSnapshotParams</var>, if the following steps
  return true:</p>

  <ol>
   <li><p>If <var>source</var> is <var>target</var>, then return true.</p></li>

   <li><p>If <var>source</var> is an ancestor of <var>target</var>, then return true.</p></li>

   <li>
    <p>If <var>target</var> is an ancestor of <var>source</var>, then:</p>

    <ol>
     <li><p>If <var>target</var> is not a <span>top-level traversable</span>, then return
     true.</p></li>

     <li><p>If <var>sourceSnapshotParams</var>'s <span
     data-x="source-snapshot-params-activation">has transient activation</span> is true, and
     <var>sourceSnapshotParams</var>'s <span data-x="source-snapshot-params-sandbox">sandboxing
     flags</span>'s <span>sandboxed top-level navigation with user activation browsing context
     flag</span> is set, then return false.</p></li>

     <li><p>If <var>sourceSnapshotParams</var>'s <span
     data-x="source-snapshot-params-activation">has transient activation</span> is false, and
     <var>sourceSnapshotParams</var>'s <span data-x="source-snapshot-params-sandbox">sandboxing
     flags</span>'s <span>sandboxed top-level navigation without user activation browsing context
     flag</span> is set, then return false.</p></li>

     <li><p>Return true.</p></li>
    </ol>
   </li>

   <li>
    <p>If <var>target</var> is a <span>top-level traversable</span>:</p>

    <ol>
     <li><p>If <var>source</var> is the <span>one permitted sandboxed navigator</span> of
     <var>target</var>, then return true.</p></li>

     <li><p>If <var>sourceSnapshotParams</var>'s <span
     data-x="source-snapshot-params-sandbox">sandboxing flags</span>'s <span>sandboxed navigation
     browsing context flag</span> is set, then return false.</p></li>

     <li><p>Return true.</p></li>
    </ol>
   </li>

   <li><p>If <var>sourceSnapshotParams</var>'s <span
   data-x="source-snapshot-params-sandbox">sandboxing flags</span>'s <span>sandboxed navigation
   browsing context flag</span> is set, then return false.</p></li>

   <li><p>Return true.</p></li>
  </ol>

  <p id="prompt-to-unload-a-document"><span id="prompt-to-unload"></span><span
  id="checking-if-unloading-is-user-canceled"></span>To <dfn data-x="checking if unloading is
  canceled">check if unloading is canceled</dfn> for a <span>list</span> of <span
  data-x="navigable">navigables</span> <var>navigablesThatNeedBeforeUnload</var>, given an optional
  <span>traversable navigable</span> <var>traversable</var>, an optional integer
  <var>targetStep</var>, and an optional <span>user navigation involvement</span>
  <var>userInvolvementForNavigateEvent</var>, run these steps. They return "<code
  data-x="">canceled-by-beforeunload</code>", "<code data-x="">canceled-by-navigate</code>", or
  "<code data-x="">continue</code>".</p>

  <ol>
   <li><p>Let <var>documentsToFireBeforeunload</var> be the <span data-x="nav-document">active
   document</span> of each <span data-x="list item">item</span> in
   <var>navigablesThatNeedBeforeUnload</var>.</p></li>

   <li><p>Let <var>unloadPromptShown</var> be false.</p></li>

   <li><p>Let <var>finalStatus</var> be "<code data-x="">continue</code>".</p></li>

   <li>
    <p>If <var>traversable</var> was given, then:</p>

    <ol>
     <li><p><span>Assert</span>: <var>targetStep</var> and
     <var>userInvolvementForNavigateEvent</var> were given.</p></li>

     <li><p>Let <var>targetEntry</var> be the result of <span>getting the target history
     entry</span> given <var>traversable</var> and <var>targetStep</var>.</p></li>

     <li>
      <p>If <var>targetEntry</var> is not <var>traversable</var>'s <span
      data-x="nav-current-history-entry">current session history entry</span>, and
      <var>targetEntry</var>'s <span data-x="she-document-state">document state</span>'s <span
      data-x="document-state-origin">origin</span> is the <span data-x="same origin">same</span> as
      <var>traversable</var>'s <span data-x="nav-current-history-entry">current session history
      entry</span>'s <span data-x="she-document-state">document state</span>'s <span
      data-x="document-state-origin">origin</span>, then:</p>

      <div class="note">
       <p>In this case, we're going to fire the <code data-x="event-navigate">navigate</code> event
       for <var>traversable</var> here. Because <a
       href="#navigate-event-traverse-can-be-canceled">under some circumstances</a> it might be
       canceled, we need to do this separately from <a
       href="#descendant-navigable-traversal-navigate-events">other traversal <code
       data-x="event-navigate">navigate</code> events</a>, which happen later.</p>

       <p>Additionally, because we want <code data-x="event-beforeunload">beforeunload</code> events
       to fire before <code data-x="event-navigate">navigate</code> events, this means we need to
       fire <code data-x="event-beforeunload">beforeunload</code> for <var>traversable</var> here
       (if applicable), instead of doing it as part of the below loop over
       <var>documentsToFireBeforeunload</var>.</p>
      </div>

      <ol>
       <li><p>Let <var>eventsFired</var> be false.</p></li>

       <li><p>Let <var>needsBeforeunload</var> be true if <var>navigablesThatNeedBeforeUnload</var>
       <span data-x="list contains">contains</span> <var>traversable</var>; otherwise
       false.</p></li>

       <li><p>If <var>needsBeforeunload</var> is true, then <span data-x="list remove">remove</span>
       <var>traversable</var>'s <span data-x="nav-document">active document</span> from
       <var>documentsToFireBeforeunload</var>.</p></li>

       <li>
        <p><span>Queue a global task</span> on the <span>navigation and traversal task source</span>
        given <var>traversable</var>'s <span data-x="nav-window">active window</span> to perform the
        following steps:</p>

        <ol>
         <li>
          <p>If <var>needsBeforeunload</var> is true, then:</p>

          <ol>
           <li><p>Let (<var>unloadPromptShownForThisDocument</var>,
           <var>unloadPromptCanceledByThisDocument</var>) be the result of running the <span>steps
           to fire <code data-x="event-beforeunload">beforeunload</code></span> given
           <var>traversable</var>'s <span data-x="nav-document">active document</span> and
           false.</p></li>

           <li><p>If <var>unloadPromptShownForThisDocument</var> is true, then set
           <var>unloadPromptShown</var> to true.</p></li>

           <li><p>If <var>unloadPromptCanceledByThisDocument</var> is true, then set
           <var>finalStatus</var> to "<code data-x="">canceled-by-beforeunload</code>".</p></li>
          </ol>
         </li>

         <li><p>If <var>finalStatus</var> is "<code data-x="">canceled-by-beforeunload</code>", then
         abort these steps.</p></li>

         <li><p>Let <var>navigation</var> be <var>traversable</var>'s <span
         data-x="nav-window">active window</span>'s <span data-x="window-navigation-api">navigation
         API</span>.</p></li>

         <li><p>Let <var>navigateEventResult</var> be the result of <span data-x="fire a traverse
         navigate event">firing a traverse <code data-x="event-navigate">navigate</code>
         event</span> at <var>navigation</var> given <var>targetEntry</var> and
         <var>userInvolvementForNavigateEvent</var>.</p></li>

         <li><p>If <var>navigateEventResult</var> is false, then set <var>finalStatus</var> to
         "<code data-x="">canceled-by-navigate</code>".</p></li>

         <li><p>Set <var>eventsFired</var> to true.</p></li>
        </ol>
       </li>

       <li><p>Wait until <var>eventsFired</var> is true.</p></li>

       <li><p>If <var>finalStatus</var> is not "<code data-x="">continue</code>", then return
       <var>finalStatus</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Let <var>totalTasks</var> be the <span data-x="list size">size</span> of
   <var>documentsToFireBeforeunload</var>.</p></li>

   <li><p>Let <var>completedTasks</var> be 0.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>document</var> of <var>documentsToFireBeforeunload</var>,
    <span>queue a global task</span> on the <span>navigation and traversal task source</span> given
    <var>document</var>'s <span>relevant global object</span> to run the steps:</p>

    <ol>
     <li><p>Let (<var>unloadPromptShownForThisDocument</var>,
     <var>unloadPromptCanceledByThisDocument</var>) be the result of running the <span>steps to fire
     <code data-x="event-beforeunload">beforeunload</code></span> given <var>document</var> and
     <var>unloadPromptShown</var>.</p></li>

     <li><p>If <var>unloadPromptShownForThisDocument</var> is true, then set
     <var>unloadPromptShown</var> to true.</p></li>

     <li><p>If <var>unloadPromptCanceledByThisDocument</var> is true, then set
     <var>finalStatus</var> to "<code data-x="">canceled-by-beforeunload</code>".</p></li>

     <li><p>Increment <var>completedTasks</var>.</p></li>
    </ol>
   </li>

   <li><p>Wait for <var>completedTasks</var> to be <var>totalTasks</var>.</p></li>

   <li><p>Return <var>finalStatus</var>.</p></li>
  </ol>

  <p>The <dfn>steps to fire <code data-x="event-beforeunload">beforeunload</code></dfn> given a
  <code>Document</code> <var>document</var> and a boolean <var>unloadPromptShown</var> are:</p>

  <ol>
   <li><p>Let <var>unloadPromptCanceled</var> be false.</p></li>

   <li><p>Increase the <var>document</var>'s <span>unload counter</span> by 1.</p></li>

   <li><p>Increase <var>document</var>'s <span>relevant agent</span>'s <span
   data-x="concept-agent-event-loop">event loop</span>'s <span>termination nesting level</span> by
   1.</p></li>

   <li><p>Let <var>eventFiringResult</var> be the result of <span data-x="concept-event-fire">firing
   an event</span> named <code data-x="event-beforeunload">beforeunload</code> at
   <var>document</var>'s <span>relevant global object</span>, using <code>BeforeUnloadEvent</code>,
   with the <code data-x="dom-Event-cancelable">cancelable</code> attribute initialized to
   true.</p></li>

   <li><p>Decrease <var>document</var>'s <span>relevant agent</span>'s <span
   data-x="concept-agent-event-loop">event loop</span>'s <span>termination nesting level</span> by
   1.</p></li>

   <li>
    <p>If all of the following are true:</p>

    <ul>
     <li><p><var>unloadPromptShown</var> is false;</p></li>

     <li><p><var>document</var>'s <span>active sandboxing flag set</span> does not have its
     <span>sandboxed modals flag</span> set;</p></li>

     <li><p><var>document</var>'s <span>relevant global object</span> has <span>sticky
     activation</span>;</p></li>

     <li><p><var>eventFiringResult</var> is false, or the <code
     data-x="dom-BeforeUnloadEvent-returnValue">returnValue</code> attribute of <var>event</var> is
     not the empty string; and</p></li>

     <li><p>showing an unload prompt is unlikely to be annoying, deceptive, or pointless,</p></li>
    </ul>

    <p>then:</p>

    <ol>
     <li><p>Set <var>unloadPromptShown</var> to true.</p></li>

     <li><p>Let <var>userPromptHandler</var> be the result of <span>WebDriver BiDi user prompt
     opened</span> with <var>document</var>'s <span>relevant global object</span>,
     "<code data-x="">beforeunload</code>", and "".</p></li>

     <li><p>If <var>userPromptHandler</var> is "<code data-x="">dismiss</code>", then set
     <var>unloadPromptCanceled</var> to true.</p></li>

     <li>
      <p>If <var>userPromptHandler</var> is "<code data-x="">none</code>", then:</p>

      <ul>
       <li>
        <p>Ask the user to confirm that they wish to unload the document, and <span>pause</span>
        while waiting for the user's response.</p>

        <p class="note">The message shown to the user is not customizable, but instead determined by
        the user agent. In particular, the actual value of the <code
        data-x="dom-BeforeUnloadEvent-returnValue">returnValue</code> attribute is ignored.</p>
       </li>

       <li><p>If the user did not confirm the page navigation, then set
       <var>unloadPromptCanceled</var> to true.</p></li>
     </ul>

     <li><p>Invoke <span>WebDriver BiDi user prompt closed</span> with <var>document</var>'s
     <span>relevant global object</span>, "<code data-x="">beforeunload</code>", and true
     if <var>unloadPromptCanceled</var> is false or false otherwise.</p></li>
    </ol>
   </li>

   <li><p>Decrease <var>document</var>'s <span>unload counter</span> by 1.</p></li>

   <li><p>Return (<var>unloadPromptShown</var>, <var>unloadPromptCanceled</var>).</p></li>
  </ol>


  <h5>Aborting navigation</h5>

  <p id="concept-navigate-mature">Each <span>navigable</span> has an <dfn>ongoing navigation</dfn>,
  which is a <span>navigation ID</span>, "<code data-x="">traversal</code>", or null, initially
  null. It is used to track navigation aborting and to prevent any navigations from taking place
  during <span data-x="apply the traverse history step">traversal</span>.</p>

  <p>To <dfn>set the ongoing navigation</dfn> for a <span>navigable</span> <var>navigable</var> to
  <var>newValue</var>:</p>

  <ol>
   <li><p>If <var>navigable</var>'s <span>ongoing navigation</span> is equal to <var>newValue</var>,
   then return.</p></li>

   <li><p><span>Inform the navigation API about aborting navigation</span> given
   <var>navigable</var>.</p></li>

   <li><p>Set <var>navigable</var>'s <span>ongoing navigation</span> to
   <var>newValue</var>.</p></li>
  </ol>


  <h4>Reloading and traversing</h4>

  <p>To <dfn>reload</dfn> a <span>navigable</span> <var>navigable</var> given an optional
  <span>serialized state</span>-or-null <dfn
  data-x="reload-navigation-api-state"><var>navigationAPIState</var></dfn> (default null) and an
  optional <span>user navigation involvement</span> <dfn
  data-x="reload-user-involvement"><var>userInvolvement</var></dfn> (default "<code
  data-x="uni-none">none</code>"):</p>

  <ol>
   <li>
    <p>If <var>userInvolvement</var> is not "<code data-x="uni-browser-ui">browser UI</code>",
    then:</p>

    <ol>
     <li><p>Let <var>navigation</var> be <var>navigable</var>'s <span data-x="nav-window">active
     window</span>'s <span data-x="window-navigation-api">navigation API</span>.</p></li>

     <li><p>Let <var>destinationNavigationAPIState</var> be <var>navigable</var>'s <span
     data-x="nav-active-history-entry">active session history entry</span>'s <span
     data-x="she-navigation-api-state">navigation API state</span>.</p></li>

     <li><p>If <var>navigationAPIState</var> is not null, then set
     <var>destinationNavigationAPIState</var> to <var>navigationAPIState</var>.</p></li>

     <li><p>Let <var>continue</var> be the result of <span data-x="fire a push/replace/reload
     navigate event">firing a push/replace/reload <code data-x="event-navigate">navigate</code>
     event</span> at <var>navigation</var> with <i
     data-x="fire-navigate-prr-navigationType">navigationType</i> set to "<code
     data-x="dom-NavigationType-reload">reload</code>", <i
     data-x="fire-navigate-prr-isSameDocument">isSameDocument</i> set to false, <i
     data-x="fire-navigate-prr-userInvolvement">userInvolvement</i> set to
     <var>userInvolvement</var>, <i data-x="fire-navigate-prr-destinationURL">destinationURL</i>
     set to <var>navigable</var>'s <span data-x="nav-active-history-entry">active session history
     entry</span>'s <span data-x="she-url">URL</span>, and <i
     data-x="fire-navigate-prr-navigationAPIState">navigationAPIState</i> set to
     <var>destinationNavigationAPIState</var>.</p></li>

     <li><p>If <var>continue</var> is false, then return.</p></li>
    </ol>
   </li>

   <li><p>Set <var>navigable</var>'s <span data-x="nav-active-history-entry">active session history
   entry</span>'s <span data-x="she-document-state">document state</span>'s <span
   data-x="document-state-reload-pending">reload pending</span> to true.</p></li>

   <li><p>Let <var>traversable</var> be <var>navigable</var>'s <span
   data-x="nav-traversable">traversable navigable</span>.</p></li>

   <li>
    <p><span data-x="tn-append-session-history-traversal-steps">Append the following session
    history traversal steps</span> to <var>traversable</var>:</p>

    <ol>
     <li><p><span>Apply the reload history step</span> to <var>traversable</var> given
     <var>userInvolvement</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>To <dfn>traverse the history by a delta</dfn> given a <span>traversable navigable</span>
  <var>traversable</var>, an integer <var>delta</var>, and an optional <code>Document</code> <dfn
  data-x="traverse-sourceDocument"><var>sourceDocument</var></dfn>:</p>

  <ol>
   <li><p>Let <var>sourceSnapshotParams</var> and <var>initiatorToCheck</var> be null.</p></li>

   <li><p>Let <var>userInvolvement</var> be "<code data-x="uni-browser-ui">browser
   UI</code>".</p></li>

   <li>
    <p>If <var>sourceDocument</var> is given, then:</p>

    <ol>
     <li><p>Set <var>sourceSnapshotParams</var> to the result of <span>snapshotting source snapshot
     params</span> given <var>sourceDocument</var>.</p></li>

     <li><p>Set <var>initiatorToCheck</var> to <var>sourceDocument</var>'s <span>node
     navigable</span>.</p></li>

     <li><p>Set <var>userInvolvement</var> to "<code data-x="uni-none">none</code>".</p></li>
    </ol>
   </li>

   <li>
    <p><span data-x="tn-append-session-history-traversal-steps">Append the following session
    history traversal steps</span> to <var>traversable</var>:</p>

    <ol>
     <li><p>Let <var>allSteps</var> be the result of <span>getting all used history steps</span>
     for <var>traversable</var>.</p></li>

     <li><p>Let <var>currentStepIndex</var> be the index of <var>traversable</var>'s <span
     data-x="tn-current-session-history-step">current session history step</span> within
     <var>allSteps</var>.</p></li>

     <li><p>Let <var>targetStepIndex</var> be <var>currentStepIndex</var> plus
     <var>delta</var>.</p></li>

     <li><p>If <var>allSteps</var>[<var>targetStepIndex</var>] does not <span data-x="list
     contains">exist</span>, then abort these steps.</p></li>

     <li><p><span>Apply the traverse history step</span>
     <var>allSteps</var>[<var>targetStepIndex</var>] to <var>traversable</var>, given
     <var>sourceSnapshotParams</var>, <var>initiatorToCheck</var>, and
     <var>userInvolvement</var>.</p></li>
    </ol>
   </li>
  </ol>


  <h4 id="navigate-non-frag-sync">Non-fragment synchronous "navigations"</h4>

  <p>Apart from the <span>navigate</span> algorithm, <span data-x="session history entry">session
  history entries</span> can be pushed or replaced via one more mechanism, the <span>URL and
  history update steps</span>. The most well-known callers of these steps are the <code
  data-x="dom-history-replaceState">history.replaceState()</code> and <code
  data-x="dom-history-pushState">history.pushState()</code> APIs, but various other parts of the
  standard also need to perform updates to the <span data-x="nav-active-history-entry">active
  history entry</span>, and they use these steps to do so.</p>

  <p id="history-1">The <dfn>URL and history update steps</dfn>, given a <code>Document</code>
  <var>document</var>, a <span>URL</span> <var>newURL</var>, an optional <span>serialized
  state</span>-or-null <dfn data-x="uhus-serializedData"><var>serializedData</var></dfn> (default
  null), and an optional <span>history handling behavior</span> <dfn
  data-x="uhus-historyHandling"><var id="uhus-ispush">historyHandling</var></dfn> (default "<code
  data-x="NavigationHistoryBehavior-replace">replace</code>"), are:</p>

  <ol>
   <li><p>Let <var>navigable</var> be <var>document</var>'s <span>node navigable</span>.</p></li>

   <li><p>Let <var>activeEntry</var> be <var>navigable</var>'s <span
   data-x="nav-active-history-entry">active session history entry</span>.</p></li>

   <li>
    <p>Let <var>newEntry</var> be a new <span>session history entry</span>, with</p>

    <dl class="props">
     <dt><span data-x="she-url">URL</span></dt>
     <dd><var>newURL</var></dd>

     <dt><span data-x="she-classic-history-api-state">serialized state</span></dt>
     <dd>if <var>serializedData</var> is not null, <var>serializedData</var>; otherwise
     <var>activeEntry</var>'s <span data-x="she-classic-history-api-state">classic history API
     state</span></dd>

     <dt><span data-x="she-document-state">document state</span></dt>
     <dd><var>activeEntry</var>'s <span data-x="she-document-state">document state</span></dd>

     <dt><span data-x="she-scroll-restoration-mode">scroll restoration mode</span></dt>
     <dd><var>activeEntry</var>'s <span data-x="she-scroll-restoration-mode">scroll restoration
     mode</span></dd>

     <dt><span data-x="she-other">persisted user state</span></dt>
     <dd><var>activeEntry</var>'s <span data-x="she-other">persisted user state</span></dd>
    </dl>
   </li>

   <li>
    <p>If <var>document</var>'s <span>is initial <code>about:blank</code></span> is true, then set
    <var>historyHandling</var> to "<code
    data-x="NavigationHistoryBehavior-replace">replace</code>".</p>

    <p class="note">This means that <code data-x="dom-history-pushState">pushState()</code> on an
    <span data-x="is initial about:blank">initial <code>about:blank</code></span>
    <code>Document</code> behaves as a <code
    data-x="dom-history-replaceState">replaceState()</code> call.</p>
   </li>

   <li><p>Let <var>entryToReplace</var> be <var>activeEntry</var> if <var>historyHandling</var> is
   "<code data-x="NavigationHistoryBehavior-replace">replace</code>", otherwise null.</p></li>

   <li>
    <p>If <var>historyHandling</var> is "<code data-x="NavigationHistoryBehavior-push">push</code>",
    then:</p>

    <ol>
     <li><p>Increment <var>document</var>'s <span data-x="doc-history">history object</span>'s
     <span data-x="concept-history-index">index</span>.</p></li>

     <li><p>Set <var>document</var>'s <span data-x="doc-history">history object</span>'s <span
     data-x="concept-history-length">length</span> to its <span
     data-x="concept-history-index">index</span> + 1.</p></li>
    </ol>

    <p class="note">These are temporary best-guess values for immediate synchronous access.</p>
   </li>

   <li><p>If <var>serializedData</var> is not null, then <span>restore the history object
   state</span> given <var>document</var> and <var>newEntry</var>.</p></li>

   <li>
    <p>Set <var>document</var>'s <span data-x="concept-document-url">URL</span> to
    <var>newURL</var>.</p>

    <p class="note">Since this is neither a <span data-x="navigate">navigation</span> nor a <span
    data-x="traverse the history by a delta">history traversal</span>, it does not cause a <code
    data-x="event-hashchange">hashchange</code> event to be fired.</p>
   </li>

   <li><p>Set <var>document</var>'s <span>latest entry</span> to <var>newEntry</var>.</p></li>

   <li><p>Set <var>navigable</var>'s <span data-x="nav-active-history-entry">active session history
   entry</span> to <var>newEntry</var>.</p></li>

   <li><p><span>Update the navigation API entries for a same-document navigation</span> given
   <var>document</var>'s <span>relevant global object</span>'s <span
   data-x="window-navigation-api">navigation API</span>, <var>newEntry</var>, and
   <var>historyHandling</var>.</p></li>

   <li><p>Let <var>traversable</var> be <var>navigable</var>'s <span
   data-x="nav-traversable">traversable navigable</span>.</p></li>

   <li>
    <p><span data-x="tn-append-session-history-sync-nav-steps">Append the following session history
    synchronous navigation steps</span> involving <var>navigable</var> to
    <var>traversable</var>:</p>

    <ol>
     <li><p><span>Finalize a same-document navigation</span> given <var>traversable</var>,
     <var>navigable</var>, <var>newEntry</var>, <var>entryToReplace</var>,
     <var>historyHandling</var>, and "<code data-x="uni-none">none</code>".</p></li>

     <li><p>Invoke <span>WebDriver BiDi history updated</span> with <var>navigable</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p class="note">Although both <span data-x="navigate-fragid">fragment navigation</span> and the
  <span>URL and history update steps</span> perform synchronous history updates, only fragment
  navigation contains a synchronous call to <span>update document for history step
  application</span>. The <span>URL and history update steps</span> instead perform a few select
  updates inside the above algorithm, omitting others. This is somewhat of an unfortunate
  historical accident, and generally leads to <a
  href="https://github.com/whatwg/html/issues/5562">web-developer sadness</a> about the
  inconsistency. For example, this means that <code data-x="event-popstate">popstate</code> events
  fire for fragment navigations, but not for <code
  data-x="dom-history-pushState">history.pushState()</code> calls.</p>


  <h4><span id="history-traversal"></span><span id="traverse-the-history"></span>Populating a session history entry</h4>

  <p>As explained in <a href="#history">the overview</a>, both <a
  href="#navigating-across-documents">navigation</a> and <a
  href="#reloading-and-traversing">traversal</a> involve creating a <span>session history
  entry</span> and then attempting to populate its <span data-x="she-document">document</span>
  member, so that it can be presented inside the <span>navigable</span>.</p>

  <p>This involves either: using <a href="#note-navigate-called-with-response">an already-given
  response</a>; using the <span data-x="document-state-resource">srcdoc resource</span> stored in
  the <span>session history entry</span>; or <span data-x="create navigation params by
  fetching">fetching</span>. The process has several failure modes, which can either result in
  doing nothing (leaving the <span>navigable</span> on its currently-<span
  data-x="nav-document">active</span> <code>Document</code>) or can result in populating the
  <span>session history entry</span> with an <span data-x="navigate-ua-inline">error
  document</span>.</p>

  <p>To <dfn>attempt to populate the history entry's document</dfn> for a <span>session history
  entry</span> <var>entry</var>, given a <span>navigable</span> <var>navigable</var>, a
  <code>NavigationTimingType</code> <var>navTimingType</var>, a <span>source snapshot params</span>
  <var>sourceSnapshotParams</var>, a <span>target snapshot params</span>
  <var>targetSnapshotParams</var>, a <span>user navigation involvement</span>
  <var>userInvolvement</var>, an optional <span>navigation ID</span>-or-null <var>navigationId</var>
  (default null), an optional <span>navigation params</span>-or-null <var>navigationParams</var>
  (default null), an optional string <var>cspNavigationType</var> (default "<code
  data-x="">other</code>"), an optional boolean <dfn
  data-x="attempt-to-populate-allow-post"><var>allowPOST</var></dfn> (default false), and optional
  algorithm steps <dfn
  data-x="attempt-to-populate-completion-steps"><var>completionSteps</var></dfn> (default an empty
  algorithm):</p>

  <ol>
   <li><p><span>Assert</span>: this is running <span>in parallel</span>.</p></li>

   <li><p><span>Assert</span>: if <var>navigationParams</var> is non-null, then
   <var>navigationParams</var>'s <span data-x="navigation-params-response">response</span> is
   non-null.</p></li>

   <li><p>Let <var>documentResource</var> be <var>entry</var>'s <span
   data-x="she-document-state">document state</span>'s <span
   data-x="document-state-resource">resource</span>.</p></li>

   <li>
    <p>If <var>navigationParams</var> is null, then:</p>

    <ol>
     <li><p>If <var>documentResource</var> is a string, then set <var>navigationParams</var> to the
     result of <span data-x="create navigation params from a srcdoc resource">creating navigation
     params from a srcdoc resource</span> given <var>entry</var>, <var>navigable</var>,
     <var>targetSnapshotParams</var>, <var>userInvolvement</var>, <var>navigationId</var>, and
     <var>navTimingType</var>.</p></li>

     <li>
      <p>Otherwise, if all of the following are true:

      <ul>
       <li><p><var>entry</var>'s <span data-x="she-url">URL</span>'s <span
       data-x="concept-url-scheme">scheme</span> is a <span>fetch scheme</span>; and</p></li>

       <li><p><var>documentResource</var> is null, or <var>allowPOST</var> is true and
       <var>documentResource</var>'s <span data-x="post-resource-request-body">request body</span>
       is not failure,</p></li>
      </ul>

      <p>then set <var>navigationParams</var> to the result of <span data-x="create navigation
      params by fetching">creating navigation params by fetching</span> given <var>entry</var>,
      <var>navigable</var>, <var>sourceSnapshotParams</var>, <var>targetSnapshotParams</var>,
      <var>cspNavigationType</var>, <var>userInvolvement</var>, <var>navigationId</var>, and
      <var>navTimingType</var>.</p>
     </li>

     <li>
      <p>Otherwise, if <var>entry</var>'s <span data-x="she-url">URL</span>'s <span
      data-x="concept-url-scheme">scheme</span> is not a <span>fetch scheme</span>, then set
      <var>navigationParams</var> to a new <span>non-fetch scheme navigation params</span>,
      with</p>

      <dl class="props">
       <dt><span data-x="non-fetch-scheme-params-id">id</span></dt>
       <dd><var>navigationId</var></dd>

       <dt><span data-x="non-fetch-scheme-params-navigable">navigable</span></dt>
       <dd><var>navigable</var></dd>

       <dt><span data-x="non-fetch-scheme-params-url">URL</span></dt>
       <dd><var>entry</var>'s <span data-x="she-url">URL</span></dd>

       <dt><span data-x="non-fetch-scheme-params-target-sandbox">target snapshot sandboxing flags</span></dt>
       <dd><var>targetSnapshotParams</var>'s <span
       data-x="target-snapshot-params-sandbox">sandboxing flags</span></dd>

       <dt><span data-x="non-fetch-scheme-params-source-activation">source snapshot has transient activation</span></dt>
       <dd><var>sourceSnapshotParams</var>'s <span data-x="source-snapshot-params-activation">has
       transient activation</span></dd>

       <dt><span data-x="non-fetch-scheme-params-initiator-origin">initiator origin</span></dt>
       <dd><var>entry</var>'s <span data-x="she-document-state">document state</span>'s <span
       data-x="document-state-initiator-origin">initiator origin</span></dd>

       <dt><span data-x="non-fetch-scheme-params-nav-timing-type">navigation timing type</span></dt>
       <dd><var>navTimingType</var></dd>

       <dt><span data-x="non-fetch-scheme-params-user-involvement">user involvement</span></dt>
       <dd><var>userInvolvement</var></dd>
      </dl>
     </li>
    </ol>
   </li>

   <li id="process-a-navigate-response">
    <p><span>Queue a global task</span> on the <span>navigation and traversal task source</span>,
    given <var>navigable</var>'s <span>active window</span>, to run these steps:</p>

    <ol>
     <li><p>If <var>navigable</var>'s <span>ongoing navigation</span> no longer equals
     <var>navigationId</var>, then run <var>completionSteps</var> and abort these steps.</p></li>

     <li>
      <p>Let <var>saveExtraDocumentState</var> be true.</p>

      <div class="note">
       <p>Usually, in the cases where we end up populating <var>entry</var>'s <span
       data-x="she-document-state">document state</span>'s <span
       data-x="document-state-document">document</span>, we then want to save some of the state
       from that <code>Document</code> into <var>entry</var>. This ensures that if there are future
       traversals to <var>entry</var> where its <span
       data-x="document-state-document">document</span> <a href="#note-bfcache">has been
       destroyed</a>, we can use that state when creating a new <code>Document</code>.</p>

       <p>However, in some specific cases, saving the state would be unhelpful. For those, we set
       <var>saveExtraDocumentState</var> to false later in this algorithm.</p>
      </div>
     </li>

     <li>
      <p>If <var>navigationParams</var> is a <span>non-fetch scheme navigation params</span>,
      then:</p>

      <ol>
       <li>
        <p>Set <var>entry</var>'s <span data-x="she-document-state">document state</span>'s <span
        data-x="document-state-document">document</span> to the result of running <span>attempt to
        create a non-fetch scheme document</span> given <var>navigationParams</var>.</p>

        <p class="note">This can result in setting <var>entry</var>'s <span
        data-x="she-document-state">document state</span>'s <span
        data-x="document-state-document">document</span> to null, e.g., when <span data-x="hand-off
        to external software">handing-off to external software</span>.</p>
       </li>

       <li><p>Set <var>saveExtraDocumentState</var> to false.</p></li>
     </ol>
     </li>

     <li>
      <p>Otherwise, if any of the following are true:</p>

      <ul>
       <li><p><var>navigationParams</var> is null;</p></li>

       <li><p>the result of <span>should navigation response to navigation request of type in
       target be blocked by Content Security Policy?</span> given <var>navigationParams</var>'s
       <span data-x="navigation-params-request">request</span>, <var>navigationParams</var>'s <span
       data-x="navigation-params-response">response</span>, <var>navigationParams</var>'s <span
       data-x="navigation-params-policy-container">policy container</span>'s <span
       data-x="policy-container-csp-list">CSP list</span>, <var>cspNavigationType</var>, and
       <var>navigable</var> is "<code data-x="">Blocked</code>";</p></li>

       <li><p><var>navigationParams</var>'s <span
       data-x="navigation-params-reserved-environment">reserved environment</span> is non-null and
       the result of <span data-x="check a navigation response's adherence to its embedder
       policy">checking a navigation response's adherence to its embedder policy</span> given
       <var>navigationParams</var>'s <span data-x="navigation-params-response">response</span>,
       <var>navigable</var>, and <var>navigationParams</var>'s <span
       data-x="navigation-params-policy-container">policy container</span>'s <span
       data-x="policy-container-embedder-policy">embedder policy</span> is false; or </p></li>

       <li><p>the result of <span data-x="check a navigation response's adherence to
       `X-Frame-Options`">checking a navigation response's adherence to
       `<code>X-Frame-Options</code>`</span> given <var>navigationParams</var>'s <span
       data-x="navigation-params-response">response</span>, <var>navigable</var>,
       <var>navigationParams</var>'s <span data-x="navigation-params-policy-container">policy
       container</span>'s <span data-x="policy-container-csp-list">CSP list</span>, and
       <var>navigationParams</var>'s <span data-x="navigation-params-origin">origin</span> is
       false,</p></li>
      </ul>

      <p>then:</p>

      <ol>
       <li><p>Set <var>entry</var>'s <span data-x="she-document-state">document state</span>'s
       <span data-x="document-state-document">document</span> to the result of <span
       data-x="navigate-ua-inline">creating a document for inline content that doesn't have a
       DOM</span>, given <var>navigable</var>, null, <var>navTimingType</var>, and
       <var>userInvolvement</var>. The inline content should indicate to the user the sort of error
       that occurred.</p></li>

       <li><p><span>Make document unsalvageable</span> given <var>entry</var>'s <span
       data-x="she-document-state">document state</span>'s <span
       data-x="document-state-document">document</span> and "<code
       data-x="blocking-navigation-failure">navigation-failure</code>".</p></li>

       <li><p>Set <var>saveExtraDocumentState</var> to false.</p></li>

       <li>
        <p>If <var>navigationParams</var> is not null, then:</p>

        <ol>
         <li><p>Run the <span data-x="environment discarding steps">environment discarding
         steps</span> for <var>navigationParams</var>'s <span
         data-x="navigation-params-reserved-environment">reserved environment</span>.</p></li>

         <li><p>Invoke <span>WebDriver BiDi navigation failed</span> with <var>navigable</var> and a
         new <span>WebDriver BiDi navigation status</span> whose <span
         data-x="navigation-status-id">id</span> is <var>navigationId</var>, <span
         data-x="navigation-status-status">status</span> is "<code
         data-x="navigation-status-canceled">canceled</code>", and <span
         data-x="navigation-status-url">url</span> is <var>navigationParams</var>'s <span
         data-x="navigation-params-response">response</span>'s <span
         data-x="concept-response-url">URL</span>.</p></li>
        </ol>
       </li>
      </ol>
     </li>

     <li id="navigation-as-a-download">
      <p>Otherwise, if <var>navigationParams</var>'s <span
      data-x="navigation-params-response">response</span> has a `<code
      data-x="http-content-disposition">Content-Disposition</code>` header specifying the <code
      data-x="">attachment</code> disposition type, then:</p>

      <ol>
       <li><p>Let <var>sourceAllowsDownloading</var> be <var>sourceSnapshotParams</var>'s <span
       data-x="source-snapshot-params-download">allows downloading</span>.</p></li>

       <li><p>Let <var>targetAllowsDownloading</var> be false if <var>navigationParams</var>'s
       <span data-x="navigation-params-sandboxing">final sandboxing flag set</span> has the
       <span>sandboxed downloads browsing context flag</span> set; otherwise true.</p></li>

       <li><p>Let <var>uaAllowsDownloading</var> be true.</p></li>

       <li><p>Optionally, the user agent may set <var>uaAllowsDownloading</var> to false, if it
       believes doing so would safeguard the user from a potentially hostile download.</p></li>

       <li>
        <p>If <var>sourceAllowsDownloading</var>, <var>targetAllowsDownloading</var>, and
        <var>uaAllowsDownloading</var> are true, then:</p>

        <ol>
         <li><p>Let <var>suggestedFilename</var> be the result of <span data-x="handle as a
         download">handling as a download</span> <var>navigationParams</var>'s <span
         data-x="navigation-params-response">response</span>.</p></li>

         <li><p>Invoke <span>WebDriver BiDi download started</span> with <var>navigable</var> and a
         new <span>WebDriver BiDi navigation status</span> whose <span
         data-x="navigation-status-id">id</span> is <var>navigationId</var>, <span
         data-x="navigation-status-status">status</span> is "<code
         data-x="navigation-status-complete">complete</code>", <span
         data-x="navigation-status-url">url</span> is <var>navigationParams</var>'s <span
         data-x="navigation-params-response">response</span>'s <span
         data-x="concept-response-url">URL</span>, and <span
         data-x="navigation-status-suggested-filename">suggestedFilename</span> is
         <var>suggestedFilename</var>.</p></li>
        </ol>
       </li>
      </ol>

      <p class="note">This branch leaves <var>entry</var>'s <span
      data-x="she-document-state">document state</span>'s <span
      data-x="document-state-document">document</span> as null.</p>
     </li>

     <li>
      <p>Otherwise, if <var>navigationParams</var>'s <span
      data-x="navigation-params-response">response</span>'s <span
      data-x="concept-response-status">status</span> is not 204 and is not 205, then set
      <var>entry</var>'s <span data-x="she-document-state">document state</span>'s <span
      data-x="document-state-document">document</span> to the result of <span>loading a
      document</span> given <var>navigationParams</var>, <var>sourceSnapshotParams</var>, and
      <var>entry</var>'s <span data-x="she-document-state">document state</span>'s <span
      data-x="document-state-initiator-origin">initiator origin</span>.</p>

      <p class="note">This can result in setting <var>entry</var>'s <span
      data-x="she-document-state">document state</span>'s <span
      data-x="document-state-document">document</span> to null, e.g., when <span data-x="hand-off
      to external software">handing-off to external software</span>.</p>
     </li>

     <li>
      <p>If <var>entry</var>'s <span data-x="she-document-state">document state</span>'s <span
      data-x="document-state-document">document</span> is not null, then:</p>

      <ol>
       <li><p>Set <var>entry</var>'s <span data-x="she-document-state">document state</span>'s
       <span data-x="document-state-ever-populated">ever populated</span> to true.</p></li>

       <li>
        <p>If <var>saveExtraDocumentState</var> is true:</p>

        <ol>
         <li><p>Let <var>document</var> be <var>entry</var>'s <span
         data-x="she-document-state">document state</span>'s <span
         data-x="document-state-document">document</span>.</p></li>

         <li><p>Set <var>entry</var>'s <span data-x="she-document-state">document state</span>'s
         <span data-x="document-state-origin">origin</span> to <var>document</var>'s <span
         data-x="concept-document-origin">origin</span>.</p></li>

         <li>
          <p>If <var>document</var>'s <span data-x="concept-document-url">URL</span>
          <span>requires storing the policy container in history</span>, then:</p>

          <ol>
           <li><p><span>Assert</span>: <var>navigationParams</var> is a <span>navigation
           params</span> (i.e., neither null nor a <span>non-fetch scheme navigation
           params</span>).</p></li>

           <li><p>Set <var>entry</var>'s <span data-x="she-document-state">document state</span>'s
           <span data-x="document-state-history-policy-container">history policy container</span>
           to <var>navigationParams</var>'s <span
           data-x="navigation-params-policy-container">policy container</span>.</p></li>
          </ol>
         </li>
        </ol>
       </li>

       <li>
        <p>If <var>entry</var>'s <span data-x="she-document-state">document state</span>'s <span
        data-x="document-state-request-referrer">request referrer</span> is "<code
        data-x="">client</code>", and <var>navigationParams</var> is a <span>navigation
        params</span> (i.e., neither null nor a <span>non-fetch scheme navigation params</span>),
        then:</p>

        <ol>
         <li><p><span>Assert</span>: <var>navigationParams</var>'s <span
         data-x="navigation-params-request">request</span> is not null.</p></li>

         <li><p>Set <var>entry</var>'s <span data-x="she-document-state">document state</span>'s
         <span data-x="document-state-request-referrer">request referrer</span> to
         <var>navigationParams</var>'s <span data-x="navigation-params-request">request</span>'s
         <span data-x="concept-request-referrer">referrer</span>.</p></li>
        </ol>
       </li>
      </ol>
     </li>

     <li><p>Run <var>completionSteps</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>To <dfn>create navigation params from a srcdoc resource</dfn> given a <span>session history
  entry</span> <var>entry</var>, a <span>navigable</span> <var>navigable</var>, a <span>target
  snapshot params</span> <var>targetSnapshotParams</var>, a <span>user navigation involvement</span>
  <var>userInvolvement</var>, a <span>navigation ID</span>-or-null <var>navigationId</var>, and a
  <code>NavigationTimingType</code> <var>navTimingType</var>:</p>

  <ol>
   <li><p>Let <var>documentResource</var> be <var>entry</var>'s <span
   data-x="she-document-state">document state</span>'s <span
   data-x="document-state-resource">resource</span>.</p></li>

   <li>
    <p>Let <var>response</var> be a new <span data-x="concept-response">response</span> with</p>

    <dl class="props">
     <dt><span data-x="concept-response-url">URL</span></dt>
     <dd><code>about:srcdoc</code></dd>

     <dt><span data-x="concept-response-header-list">header list</span></dt>
     <dd>« (`<code>Content-Type</code>`, `<code data-x="">text/html</code>`) »</dd>

     <dt><span data-x="concept-response-body">body</span></dt>
     <dd>the <span data-x="UTF-8 encode">UTF-8 encoding</span> of <var>documentResource</var>,
     <span>as a body</span></dd>
    </dl>
   </li>

   <li><p>Let <var>responseOrigin</var> be the result of <span>determining the origin</span> given
   <var>response</var>'s <span data-x="concept-response-url">URL</span>,
   <var>targetSnapshotParams</var>'s <span data-x="target-snapshot-params-sandbox">sandboxing
   flags</span>, and <var>entry</var>'s <span data-x="she-document-state">document state</span>'s
   <span data-x="document-state-origin">origin</span>.</p></li>

   <li><p>Let <var>coop</var> be a new <span>opener policy</span>.</p></li>

   <li>
    <p>Let <var>coopEnforcementResult</var> be a new <span data-x="coop-enforcement-result">opener
    policy enforcement result</span> with</p>

    <dl class="props">
     <dt><span data-x="coop-enforcement-url">url</span></dt>
     <dd><var>response</var>'s <span data-x="concept-response-url">URL</span></dd>

     <dt><span data-x="coop-enforcement-origin">origin</span></dt>
     <dd><var>responseOrigin</var></dd>

     <dt><span data-x="coop-enforcement-coop">opener policy</span></dt>
     <dd><var>coop</var></dd>
    </dl>
   </li>

   <li><p>Let <var>policyContainer</var> be the result of <span data-x="determining navigation
   params policy container">determining navigation params policy container</span> given
   <var>response</var>'s <span data-x="concept-response-url">URL</span>, <var>entry</var>'s <span
   data-x="she-document-state">document state</span>'s <span
   data-x="document-state-history-policy-container">history policy container</span>, null,
   <var>navigable</var>'s <span data-x="nav-container-document">container document</span>'s <span
   data-x="concept-document-policy-container">policy container</span>, and null.</p></li>

   <li>
    <p>Return a new <span>navigation params</span>, with</p>

    <dl class="props">
     <dt><span data-x="navigation-params-id">id</span></dt>
     <dd><var>navigationId</var></dd>

     <dt><span data-x="navigation-params-navigable">navigable</span></dt>
     <dd><var>navigable</var></dd>

     <dt><span data-x="navigation-params-request">request</span></dt>
     <dd>null</dd>

     <dt><span data-x="navigation-params-response">response</span></dt>
     <dd><var>response</var></dd>

     <dt><span data-x="navigation-params-fetch-controller">fetch controller</span></dt>
     <dd>null</dd>

     <dt><span data-x="navigation-params-commit-early-hints">commit early hints</span></dt>
     <dd>null</dd>

     <dt><span data-x="navigation-params-coop-enforcement-result">COOP enforcement result</span></dt>
     <dd><var>coopEnforcementResult</var></dd>

     <dt><span data-x="navigation-params-reserved-environment">reserved environment</span></dt>
     <dd>null</dd>

     <dt><span data-x="navigation-params-origin">origin</span></dt>
     <dd><var>responseOrigin</var></dd>

     <dt><span data-x="navigation-params-policy-container">policy container</span></dt>
     <dd><var>policyContainer</var></dd>

     <dt><span data-x="navigation-params-sandboxing">final sandboxing flag set</span></dt>
     <dd><var>targetSnapshotParams</var>'s <span data-x="target-snapshot-params-sandbox">sandboxing
     flags</span></dd>

     <dt><span data-x="navigation-params-coop">opener policy</span></dt>
     <dd><var>coop</var></dd>

     <dt><span data-x="navigation-params-nav-timing-type">navigation timing type</span></dt>
     <dd><var>navTimingType</var></dd>

     <dt><span data-x="navigation-params-about-base-url">about base URL</span></dt>
     <dd><var>entry</var>'s <span data-x="she-document-state">document state</span>'s <span
     data-x="document-state-about-base-url">about base URL</span></dd>

     <dt><span data-x="navigation-params-user-involvement">user involvement</span></dt>
     <dd><var>userInvolvement</var></dd>
    </dl>
   </li>
  </ol>

  <p id="process-a-navigate-fetch">To <dfn>create navigation params by fetching</dfn> given a
  <span>session history entry</span> <var>entry</var>, a <span>navigable</span>
  <var>navigable</var>, a <span>source snapshot params</span> <var>sourceSnapshotParams</var>, a
  <span>target snapshot params</span> <var>targetSnapshotParams</var>, a string
  <var>cspNavigationType</var>, a <span>user navigation involvement</span>
  <var>userInvolvement</var>, a <span>navigation ID</span>-or-null <var>navigationId</var>, and a
  <code>NavigationTimingType</code> <var>navTimingType</var>, perform the following steps. They
  return a <span>navigation params</span>, a <span>non-fetch scheme navigation params</span>, or
  null.</p>

  <p class="note">This algorithm mutates <var>entry</var>.</p>

  <ol>
   <li><p><span>Assert</span>: this is running <span>in parallel</span>.</p></li>

   <li><p>Let <var>documentResource</var> be <var>entry</var>'s <span
   data-x="she-document-state">document state</span>'s <span
   data-x="document-state-resource">resource</span>.</p></li>

   <li>
    <p>Let <var>request</var> be a new <span data-x="concept-request">request</span>, with</p>

    <dl class="props">
     <dt><span data-x="concept-request-url">url</span></dt>
     <dd><var>entry</var>'s <span data-x="she-url">URL</span></dd>

     <dt><span data-x="concept-request-client">client</span></dt>
     <dd><var>sourceSnapshotParams</var>'s <span data-x="source-snapshot-params-client">fetch
     client</span></dd>

     <dt><span data-x="concept-request-destination">destination</span></dt>
     <dd>"<code data-x="">document</code>"</dd>

     <dt><span data-x="concept-request-credentials-mode">credentials mode</span></dt>
     <dd>"<code data-x="">include</code>"</dd>

     <dt><span>use-URL-credentials flag</span></dt>
     <dd>set</dd>

     <dt><span data-x="concept-request-redirect-mode">redirect mode</span></dt>
     <dd>"<code data-x="">manual</code>"</dd>

     <dt><span data-x="concept-request-replaces-client-id">replaces client id</span></dt>
     <dd><var>navigable</var>'s <span data-x="nav-document">active document</span>'s <span>relevant
     settings object</span>'s <span data-x="concept-environment-id">id</span></dd>

     <dt><span data-x="concept-request-mode">mode</span></dt>
     <dd>"<code data-x="">navigate</code>"</dd>

     <dt><span data-x="concept-request-referrer">referrer</span></dt>
     <dd><var>entry</var>'s <span data-x="she-document-state">document state</span>'s <span
     data-x="document-state-request-referrer">request referrer</span></dd>

     <dt><span data-x="concept-request-referrer-policy">referrer policy</span></dt>
     <dd><var>entry</var>'s <span data-x="she-document-state">document state</span>'s <span
     data-x="document-state-request-referrer-policy">request referrer policy</span></dd>

     <dt><span data-x="concept-request-policy-container">policy container</span></dt>
     <dd><var>sourceSnapshotParams</var>'s <span
     data-x="source-snapshot-params-policy-container">source policy container</span></dd>
    </dl>
   </li>

   <li>
    <p>If <var>documentResource</var> is a <span>POST resource</span>, then:</p>

    <ol>
     <li><p>Set <var>request</var>'s <span data-x="concept-request-method">method</span> to `<code
     data-x="">POST</code>`.</p></li>

     <li><p>Set <var>request</var>'s <span data-x="concept-request-body">body</span> to
     <var>documentResource</var>'s <span data-x="post-resource-request-body">request
     body</span>.</p></li>

     <li><p><span data-x="concept-header-list-set">Set</span> `<code data-x="">Content-Type</code>`
     to <var>documentResource</var>'s <span data-x="post-resource-request-content-type">request
     content-type</span> in <var>request</var>'s <span data-x="concept-request-header-list">header
     list</span>.</p></li>
    </ol>
   </li>

   <li><p>If <var>entry</var>'s <span data-x="she-document-state">document state</span>'s <span
   data-x="document-state-reload-pending">reload pending</span> is true, then set
   <var>request</var>'s <span data-x="concept-request-reload-navigation-flag">reload-navigation
   flag</span>.</p></li>

   <li><p>Otherwise, if <var>entry</var>'s <span data-x="she-document-state">document
   state</span>'s <span data-x="document-state-ever-populated">ever populated</span> is true, then
   set <var>request</var>'s <span
   data-x="concept-request-history-navigation-flag">history-navigation flag</span>.</p></li>

   <li><p>If <var>sourceSnapshotParams</var>'s <span data-x="source-snapshot-params-activation">has
   transient activation</span> is true, then set <var>request</var>'s <span
   data-x="concept-request-user-activation">user-activation</span> to true.</p></li>

   <li>
    <p>If <var>navigable</var>'s <span data-x="nav-container">container</span> is non-null:</p>

    <ol>
     <li><p>If the <var>navigable</var>'s <span data-x="nav-container">container</span> has a
     <span>browsing context scope origin</span>, then set <var>request</var>'s <span
     data-x="concept-request-origin">origin</span> to that <span>browsing context scope
     origin</span>.</p></li>

     <li><p>Set <var>request</var>'s <span data-x="concept-request-destination">destination</span>
     to <var>navigable</var>'s <span data-x="nav-container">container</span>'s <span
     data-x="concept-element-local-name">local name</span>.</p></li>

     <li>
      <p>If <var>sourceSnapshotParams</var>'s <span data-x="source-snapshot-params-client">fetch
      client</span> is <var>navigable</var>'s <span data-x="nav-container-document">container
      document</span>'s <span>relevant settings object</span>, then set <var>request</var>'s <span
      data-x="concept-request-initiator-type">initiator type</span> to <var>navigable</var>'s <span
      data-x="nav-container">container</span>'s <span data-x="concept-element-local-name">local
      name</span>.</p>

      <p class="note">This ensure that only container-initiated navigations are reported to
      resource timing.</p>
     </li>
    </ol>
   </li>

   <li><p>Let <var>response</var> be null.</p></li>

   <li><p>Let <var>responseOrigin</var> be null.</p></li>

   <li><p>Let <var>fetchController</var> be null.</p></li>

   <li>
    <p>Let <var>coopEnforcementResult</var> be a new <span data-x="coop-enforcement-result">opener
    policy enforcement result</span>, with</p>

    <dl class="props">
     <dt><span data-x="coop-enforcement-url">url</span></dt>
     <dd><var>navigable</var>'s <span data-x="nav-document">active document</span>'s <span
     data-x="concept-document-url">URL</span></dd>

     <dt><span data-x="coop-enforcement-origin">origin</span></dt>
     <dd><var>navigable</var>'s <span data-x="nav-document">active document</span>'s <span
     data-x="concept-document-origin">origin</span></dd>

     <dt><span data-x="coop-enforcement-coop">opener policy</span></dt>
     <dd><var>navigable</var>'s <span data-x="nav-document">active document</span>'s <span
     data-x="concept-document-coop">opener policy</span></dd>

     <dt><span data-x="coop-enforcement-source">current context is navigation source</span></dt>
     <dd>true if <var>navigable</var>'s <span data-x="nav-document">active document</span>'s <span
     data-x="concept-document-origin">origin</span> is <span>same origin</span> with
     <var>entry</var>'s <span data-x="she-document-state">document state</span>'s <span
     data-x="document-state-initiator-origin">initiator origin</span> otherwise false</dd>
    </dl>
   </li>

   <li><p>Let <var>finalSandboxFlags</var> be an empty <span>sandboxing flag set</span>.</p></li>

   <li><p>Let <var>responsePolicyContainer</var> be null.</p></li>

   <li><p>Let <var>responseCOOP</var> be a new <span>opener policy</span>.</p></li>

   <li><p>Let <var>locationURL</var> be null.</p></li>

   <li><p>Let <var>currentURL</var> be <var>request</var>'s <span
   data-x="concept-request-current-url">current URL</span>.</p></li>

   <li><p>Let <var>commitEarlyHints</var> be null.</p></li>

   <li>
    <p>While true:</p>

    <ol>
     <li>
      <p>If <var>request</var>'s <span data-x="concept-request-reserved-client">reserved
      client</span> is not null and <var>currentURL</var>'s <span
      data-x="concept-url-origin">origin</span> is not the <span data-x="same origin">same</span>
      as <var>request</var>'s <span data-x="concept-request-reserved-client">reserved
      client</span>'s <span data-x="concept-environment-creation-url">creation URL</span>'s <span
      data-x="concept-url-origin">origin</span>, then:</p>

      <ol>
       <li><p>Run the <span>environment discarding steps</span> for <var>request</var>'s <span
       data-x="concept-request-reserved-client">reserved client</span>.</p></li>

       <li><p>Set <var>request</var>'s <span data-x="concept-request-reserved-client">reserved
       client</span> to null.</p></li>

       <li>
        <p>Set <var>commitEarlyHints</var> to null.</p>

        <p class="note">Preloaded links from <span data-x="early hints">early hint headers</span>
        remain in the preload cache after a <span>same origin</span> redirect, but get discarded
        when the redirect is cross-origin.</p>
       </li>
      </ol>
     </li>

     <li>
      <p>If <var>request</var>'s <span data-x="concept-request-reserved-client">reserved
      client</span> is null, then:</p>

      <ol>
       <li><p>Let <var>topLevelCreationURL</var> be <var>currentURL</var>.</p></li>

       <li><p>Let <var>topLevelOrigin</var> be null.</p></li>

       <li>
        <p>If <var>navigable</var> is not a <span>top-level traversable</span>, then:</p>

        <ol>
         <li><p>Let <var>parentEnvironment</var> be <var>navigable</var>'s <span
         data-x="nav-parent">parent</span>'s <span data-x="nav-document">active document</span>'s
         <span>relevant settings object</span>.</p></li>

         <li><p>Set <var>topLevelCreationURL</var> to <var>parentEnvironment</var>'s
         <span>top-level creation URL</span>.</p></li>

         <li><p>Set <var>topLevelOrigin</var> to <var>parentEnvironment</var>'s <span>top-level
         origin</span>.</p></li>
        </ol>
       </li>

       <li>
        <p>Set <var>request</var>'s <span data-x="concept-request-reserved-client">reserved
        client</span> to a new <span>environment</span> whose <span
        data-x="concept-environment-id">id</span> is a unique opaque string, <span
        data-x="concept-environment-target-browsing-context">target browsing context</span> is
        <var>navigable</var>'s <span data-x="nav-bc">active browsing context</span>, <span
        data-x="concept-environment-creation-url">creation URL</span> is <var>currentURL</var>,
        <span>top-level creation URL</span> is <var>topLevelCreationURL</var>, and <span>top-level
        origin</span> is <var>topLevelOrigin</var>.</p>

        <p class="note">The created environment's <span
        data-x="concept-environment-active-service-worker">active service worker</span> is set in
        the <span data-x="on-fetch-request-algorithm">Handle Fetch</span> algorithm during the fetch
        if the request URL matches a service worker registration. <ref>SW</ref></p>
       </li>
      </ol>
     </li>

     <li><p>If the result of <span>should navigation request of type be blocked by Content Security
     Policy?</span> given <var>request</var> and <var>cspNavigationType</var> is "<code
     data-x="">Blocked</code>", then set <var>response</var> to a <span>network error</span> and
     <span>break</span>. <ref>CSP</ref></p></li>

     <li><p>Set <var>response</var> to null.</p></li>

     <li>
      <p>If <var>fetchController</var> is null, then set <var>fetchController</var> to the result
      of <!--FETCH--><span data-x="concept-fetch">fetching</span> <var>request</var>, with <i
      data-x="processEarlyHintsResponse">processEarlyHintsResponse</i> set to
      <var>processEarlyHintsResponse</var> as defined below, <i
      data-x="processResponse">processResponse</i> set to <var>processResponse</var> as defined
      below, and <i data-x="useParallelQueue">useParallelQueue</i> set to true.</p>

      <p>Let <var>processEarlyHintsResponse</var> be the following algorithm given a <span
      data-x="concept-response">response</span> <var>earlyResponse</var>:</p>

      <ol>
       <li><p>If <var>commitEarlyHints</var> is null, then set <var>commitEarlyHints</var> to the
       result of <span data-x="process early hint headers">processing early hint headers</span>
       given <var>earlyResponse</var> and <var>request</var>'s <span
       data-x="concept-request-reserved-client">reserved client</span>.</p></li>
      </ol>

      <p>Let <var>processResponse</var> be the following algorithm given a <span
      data-x="concept-response">response</span> <var>fetchedResponse</var>:</p>

      <ol>
       <li><p>Set <var>response</var> to <var>fetchedResponse</var>.</p></li>
      </ol>
     </li>

     <li>
      <p>Otherwise, <span>process the next manual redirect</span> for
      <var>fetchController</var>.</p>

      <p class="note">This will result in calling the <i
      data-x="processResponse">processResponse</i> we supplied above, during our first iteration
      through the loop, and thus setting <var>response</var>.</p>

      <p class="note">Navigation handles redirects manually as navigation is the only place in the
      web platform that cares for redirects to <code data-x="mailto protocol">mailto:</code> URLs
      and such.</p>
     </li>

     <li>
      <p>Wait until either <var>response</var> is non-null, or <var>navigable</var>'s <span>ongoing
      navigation</span> changes to no longer equal <var>navigationId</var>.</p>

      <p>If the latter condition occurs, then <span data-x="fetch-controller-abort">abort</span>
      <var>fetchController</var>, and return.</p>

      <p>Otherwise, proceed onward.</p>
     </li>

     <li>
      <p>If <var>request</var>'s <span data-x="concept-request-body">body</span> is null, then set
      <var>entry</var>'s <span data-x="she-document-state">document state</span>'s <span
      data-x="document-state-resource">resource</span> to null.</p>

      <p class="note">Fetch unsets the <span data-x="concept-request-body">body</span> for
      particular redirects.</p>
     </li>

     <li><p>Set <var>responsePolicyContainer</var> to the result of <span>creating a policy
     container from a fetch response</span> given <var>response</var> and <var>request</var>'s
     <span data-x="concept-request-reserved-client">reserved client</span>.</p></li>

     <li><p>Set <var>finalSandboxFlags</var> to the <span data-x="set union">union</span> of
     <var>targetSnapshotParams</var>'s <span data-x="target-snapshot-params-sandbox">sandboxing
     flags</span> and <var>responsePolicyContainer</var>'s <span
     data-x="policy-container-csp-list">CSP list</span>'s <span>CSP-derived sandboxing
     flags</span>.</p></li>

     <li>
      <p>Set <var>responseOrigin</var> to the result of <span>determining the origin</span> given
      <var>response</var>'s <span data-x="concept-response-url">URL</span>,
      <var>finalSandboxFlags</var>, and <var>entry</var>'s <span
      data-x="she-document-state">document state</span>'s <span
      data-x="document-state-initiator-origin">initiator origin</span>.</p>

      <p class="note">If <var>response</var> is a redirect, then <var>response</var>'s <span
      data-x="concept-response-url">URL</span> will be the URL that led to the redirect to
      <var>response</var>'s <span data-x="concept-response-location-url">location URL</span>; it
      will not be the <span data-x="concept-response-location-url">location URL</span> itself.</p>
     </li>

     <li>
      <p>If <var>navigable</var> is a <span>top-level traversable</span>, then:</p>

      <ol>
       <li><p>Set <var>responseCOOP</var> to the result of <span data-x="obtain-coop">obtaining an
       opener policy</span> given <var>response</var> and <var>request</var>'s <span
       data-x="concept-request-reserved-client">reserved client</span>.</p></li>

       <li><p>Set <var>coopEnforcementResult</var> to the result of <span
       data-x="coop-enforce">enforcing the response's opener policy</span> given
       <var>navigable</var>'s <span data-x="nav-bc">active browsing context</span>,
       <var>response</var>'s <span data-x="concept-response-url">URL</span>,
       <var>responseOrigin</var>, <var>responseCOOP</var>, <var>coopEnforcementResult</var> and
       <var>request</var>'s <span data-x="concept-request-referrer">referrer</span>.</p></li>

       <li>
        <p>If <var>finalSandboxFlags</var> is not empty and <var>responseCOOP</var>'s <span
        data-x="coop-struct-value">value</span> is not "<code
        data-x="coop-unsafe-none">unsafe-none</code>", then set <var>response</var> to an
        appropriate <span>network error</span> and <span>break</span>.</p>

        <p class="note">This results in a network error as one cannot simultaneously provide a clean
        slate to a response using opener policy and sandbox the result of
        navigating to that response.</p>
       </li>
      </ol>
     </li>

     <li>
      <p>If <var>response</var> is not a <span>network error</span>, <var>navigable</var> is a
      <span>child navigable</span>, and the result of performing a <span>cross-origin resource
      policy check</span> with <var>navigable</var>'s <span
      data-x="nav-container-document">container document</span>'s <span
      data-x="concept-document-origin">origin</span>, <var>navigable</var>'s <span
      data-x="nav-container-document">container document</span>'s <span>relevant settings
      object</span>, <var>request</var>'s <span
      data-x="concept-request-destination">destination</span>, <var>response</var>, and true is
      <b>blocked</b>, then set <var>response</var> to a <span>network error</span> and
      <span>break</span>.</p>

      <p class="note">Here we're running the <span>cross-origin resource policy check</span>
      against the <span data-x="nav-parent">parent navigable</span> rather than
      <var>navigable</var> itself. This is because we care about the same-originness of the
      embedded content against the parent context, not the navigation source.</p>
     </li>

     <li><p>Set <var>locationURL</var> to <var>response</var>'s <span
     data-x="concept-response-location-url">location URL</span> given <var>currentURL</var>'s <span
     data-x="concept-url-fragment">fragment</span>.</p></li>

     <li><p>If <var>locationURL</var> is failure or null, then <span>break</span>.</p></li>

     <li><p><span>Assert</span>: <var>locationURL</var> is a <span>URL</span>.</p></li>

     <li><p>Set <var>entry</var>'s <span data-x="she-classic-history-api-state">classic history API
     state</span> to <span>StructuredSerializeForStorage</span>(null).</p></li>

     <li><p>Let <var>oldDocState</var> be <var>entry</var>'s <span
     data-x="she-document-state">document state</span>.</p></li>

     <li>
      <p>Set <var>entry</var>'s <span data-x="she-document-state">document state</span> to a new
      <span>document state</span>, with

      <dl class="props">
       <dt><span data-x="document-state-history-policy-container">history policy container</span></dt>
       <dd>a <span data-x="clone a policy container">clone</span> of the <var>oldDocState</var>'s
       <span data-x="document-state-history-policy-container">history policy container</span> if it
       is non-null; null otherwise</dd>

       <dt><span data-x="document-state-request-referrer">request referrer</span></dt>
       <dd><var>oldDocState</var>'s <span data-x="document-state-request-referrer">request
       referrer</span></dd>

       <dt><span data-x="document-state-request-referrer-policy">request referrer policy</span></dt>
       <dd><var>oldDocState</var>'s <span data-x="document-state-request-referrer-policy">request
       referrer policy</span></dd>

       <dt><span data-x="document-state-initiator-origin">initiator origin</span></dt>
       <dd><var>oldDocState</var>'s <span data-x="document-state-initiator-origin">initiator
       origin</span></dd>

       <dt><span data-x="document-state-origin">origin</span></dt>
       <dd><var>oldDocState</var>'s <span data-x="document-state-origin">origin</span></dd>

       <dt><span data-x="document-state-about-base-url">about base URL</span></dt>
       <dd><var>oldDocState</var>'s <span data-x="document-state-about-base-url">about base URL</span></dd>

       <dt><span data-x="document-state-resource">resource</span></dt>
       <dd><var>oldDocState</var>'s <span data-x="document-state-resource">resource</span></dd>

       <dt><span data-x="document-state-ever-populated">ever populated</span></dt>
       <dd><var>oldDocState</var>'s <span data-x="document-state-ever-populated">ever
       populated</span></dd>

       <dt><span data-x="document-state-nav-target-name">navigable target name</span></dt>
       <dd><var>oldDocState</var>'s <span data-x="document-state-nav-target-name">navigable target
       name</span></dd>
      </dl>

      <!-- https://github.com/whatwg/html/issues/6680 -->
      <p class="note">For the navigation case, only <var>entry</var> referenced
      <var>oldDocState</var>, which was created <a href="#navigation-create-document-state">early
      in the navigate algorithm</a>. So for navigations, this is functionally just an update to
      <var>entry</var>'s <span data-x="she-document-state">document state</span>. For the traversal
      case, it's possible adjacent <span data-x="session history entry">session history
      entries</span> also reference <var>oldDocState</var>, in which case they will continue doing
      so even after we've updated <var>entry</var>'s <span data-x="she-document-state">document
      state</span>.</p>

      <p class="note"><var>oldDocState</var>'s <span
      data-x="document-state-history-policy-container">history policy container</span> is only ever
      non-null here in the traversal case, after we've populated it during a navigation to a URL
      that <span>requires storing the policy container in history</span>.</p>

      <div class="example">
       <p>The setup is given by the following <span>Jake diagram</span>:</p>

       <!--https://domenic.github.io/jake-diagram-generator/#IWN1cnJlbnQgPSAzCiFza2lwRG9jQ29sb3JzID0gMQp0b3AKICAwOiAvYSB8IGRvYzEKICAxOiAvYSNmb28gfCBkb2MxCiAgMjogL2EjYmFyIHwgZG9jMQogIDM6IC9i -->
<table class="jake-diagram"><thead><tr><td></td><th class="step">0</th><th class="step">1</th><th class="step">2</th><th class="step current">3</th></tr></thead><tbody><tr><th><code data-x="">top</code></th><td colspan="1" class="doc-0 next-is-same-doc">/a</td><td colspan="1" class="doc-0 prev-is-same-doc next-is-same-doc">/a#foo</td><td colspan="1" class="doc-0 prev-is-same-doc">/a#bar</td><td colspan="1" class="doc-2 current">/b</td></tr></tbody></table>

       <p>Also assume that the <span data-x="she-document-state">document state</span> shared by
       the entries in steps 0, 1, and 2 has a null <span
       data-x="document-state-document">document</span>, i.e., <a href="#note-bfcache">bfcache</a>
       is not in play.</p>

       <p>Now consider the scenario where we traverse back to step 2, but this time when fetching
       <code data-x="">/a</code>, the server responds with a `<code data-x="">Location</code>`
       header pointing to <code data-x="">/c</code>. That is, <var>locationURL</var> points to
       <code data-x="">/c</code> and so we have reached this step instead of <span
       data-x="break">breaking</span> out of the loop.</p>

       <p>In this case, we replace the <span data-x="she-document-state">document state</span> of
       the <span>session history entry</span> occupying step 2, but we do <em>not</em> replace the
       document state of the entries occupying steps 0 and 1. The resulting <span>Jake
       diagram</span> looks like this:</p>

       <!--https://domenic.github.io/jake-diagram-generator/#IWN1cnJlbnQgPSAyCnRvcAogIDA6IC9hIHwgZG9jMQogIDE6IC9hI2ZvbyB8IGRvYzEKICAyOiAvYyNiYXIKICAzOiAvYg== -->
<table class="jake-diagram"><thead><tr><td></td><th class="step">0</th><th class="step">1</th><th class="step current">2</th><th class="step">3</th></tr></thead><tbody><tr><th><code data-x="">top</code></th><td colspan="1" class="doc-0 next-is-same-doc">/a</td><td colspan="1" class="doc-0 prev-is-same-doc">/a#foo</td><td colspan="1" class="doc-1 current">/c#bar</td><td colspan="1" class="doc-2">/b</td></tr></tbody></table>

       <p>Note that we perform this replacement even if we end up in a redirect chain back to the
       original URL, for example if <code data-x="">/c</code> itself had a `<code
       data-x="">Location</code>` header pointing to <code data-x="">/a</code>. Such a case would
       end up like so:</p>

       <!--https://domenic.github.io/jake-diagram-generator/#IWN1cnJlbnQgPSAyCnRvcAogIDA6IC9hIHwgZG9jMQogIDE6IC9hI2ZvbyB8IGRvYzEKICAyOiAvYSNiYXIKICAzOiAvYg== -->
<table class="jake-diagram"><thead><tr><td></td><th class="step">0</th><th class="step">1</th><th class="step current">2</th><th class="step">3</th></tr></thead><tbody><tr><th><code data-x="">top</code></th><td colspan="1" class="doc-0 next-is-same-doc">/a</td><td colspan="1" class="doc-0 prev-is-same-doc">/a#foo</td><td colspan="1" class="doc-1 current">/a#bar</td><td colspan="1" class="doc-2">/b</td></tr></tbody></table>
      </div>
     </li>

     <li>
      <p>If <var>locationURL</var>'s <span data-x="concept-url-scheme">scheme</span> is not an
      <span>HTTP(S) scheme</span>, then:</p>

      <ol>
       <li><p>Set <var>entry</var>'s <span data-x="she-document-state">document state</span>'s
       <span data-x="document-state-resource">resource</span> to null.</p></li>

       <li><p><span>Break</span>.</p></li>
      </ol>
     </li>

     <li><p>Set <var>currentURL</var> to <var>locationURL</var>.</p></li>

     <li><p>Set <var>entry</var>'s <span data-x="she-url">URL</span> to
     <var>currentURL</var>.</p></li>
    </ol>

    <div class="note">
     <p>By the end of this loop we will be in one of these scenarios:</p>

     <ul>
      <li><p><var>locationURL</var> is failure, because of an unparseable `<code
      data-x="">Location</code>` header.</p></li>

      <li><p><var>locationURL</var> is null, either because <var>response</var> is a <span>network
      error</span> or because we successfully fetched a non-<span>network error</span> HTTP(S)
      response with no `<code data-x="">Location</code>` header.</p></li>

      <li><p><var>locationURL</var> is a <span>URL</span> with a non-<span data-x="HTTP(S)
      scheme">HTTP(S)</span> <span data-x="concept-url-scheme">scheme</span>.</p></li>
     </ul>
    </div>
   </li>

   <li>
    <p>If <var>locationURL</var> is a <span>URL</span> whose <span
    data-x="concept-url-scheme">scheme</span> is not a <span>fetch scheme</span>, then return a new
    <span>non-fetch scheme navigation params</span>, with</p>

    <dl class="props">
     <dt><span data-x="non-fetch-scheme-params-id">id</span></dt>
     <dd><var>navigationId</var></dd>

     <dt><span data-x="non-fetch-scheme-params-navigable">navigable</span></dt>
     <dd><var>navigable</var></dd>

     <dt><span data-x="non-fetch-scheme-params-url">URL</span></dt>
     <dd><var>locationURL</var></dd>

     <dt><span data-x="non-fetch-scheme-params-target-sandbox">target snapshot sandboxing flags</span></dt>
     <dd><var>targetSnapshotParams</var>'s <span data-x="target-snapshot-params-sandbox">sandboxing
     flags</span></dd>

     <dt><span data-x="non-fetch-scheme-params-source-activation">source snapshot has transient activation</span></dt>
     <dd><var>sourceSnapshotParams</var>'s <span data-x="source-snapshot-params-activation">has
     transient activation</span></dd>

     <dt><span data-x="non-fetch-scheme-params-initiator-origin">initiator origin</span></dt>
     <dd><var>responseOrigin</var></dd>

     <dt><span data-x="non-fetch-scheme-params-nav-timing-type">navigation timing type</span></dt>
     <dd><var>navTimingType</var></dd>

     <dt><span data-x="non-fetch-scheme-params-user-involvement">user involvement</span></dt>
     <dd><var>userInvolvement</var></dd>
    </dl>

    <p class="note">At this point, <var>request</var>'s <span
    data-x="concept-request-current-url">current URL</span> is the last <span>URL</span> in the
    redirect chain with a <span data-x="fetch scheme">fetch</span> <span
    data-x="concept-url-scheme">scheme</span> before redirecting to a non-<span>fetch scheme</span>
    <span>URL</span>. It is this <span>URL</span>'s <span data-x="concept-url-origin">origin</span>
    that will be used as the initiator origin for navigations to non-<span>fetch scheme</span>
    <span data-x="url">URLs</span>.</p>
   </li>

   <li>
    <p>If any of the following are true:</p>

    <ul>
     <li><p><var>response</var> is a <span>network error</span>;</p></li>

     <li><p><var>locationURL</var> is failure; or</p></li>

     <li><p><var>locationURL</var> is a <span>URL</span> whose <span
     data-x="concept-url-scheme">scheme</span> is a <span>fetch scheme</span>,</p></li>
    </ul>

    <p>then return null.</p>

    <p class="note">We allow redirects to non-<span>fetch scheme</span> <span
    data-x="URL">URLs</span>, but redirects to <span>fetch scheme</span> <span
    data-x="URL">URLs</span> that aren't <span data-x="HTTP(S) scheme">HTTP(S)</span> are treated
    like network errors.</p>
   </li>

   <li><p><span>Assert</span>: <var>locationURL</var> is null and <var>response</var> is not a
   <span>network error</span>.</li>

   <li><p>Let <var>resultPolicyContainer</var> be the result of <span>determining navigation params
   policy container</span> given <var>response</var>'s <span
   data-x="concept-response-url">URL</span>, <var>entry</var>'s <span
   data-x="she-document-state">document state</span>'s <span
   data-x="document-state-history-policy-container">history policy container</span>,
   <var>sourceSnapshotParams</var>'s <span data-x="source-snapshot-params-policy-container">source
   policy container</span>, null, and <var>responsePolicyContainer</var>.</p></li>

   <li>
    <p>If <var>navigable</var>'s <span data-x="nav-container">container</span> is an
    <code>iframe</code>, and <var>response</var>'s <span
    data-x="concept-response-timing-allow-passed">timing allow passed flag</span> is set, then
    set <span data-x="nav-container">container</span>'s <span
    data-x="iframe-pending-resource-timing-start-time">pending resource-timing start time</span> to
    null.</p>

    <p class="note">If the <code>iframe</code> is allowed to report to resource timing,
    we don't need to run its fallback steps as the normal reporting would happen.</p>
   </li>

   <li>
    <p>Return a new <span>navigation params</span>, with</p>

    <dl class="props">
     <dt><span data-x="navigation-params-id">id</span></dt>
     <dd><var>navigationId</var></dd>

     <dt><span data-x="navigation-params-navigable">navigable</span></dt>
     <dd><var>navigable</var></dd>

     <dt><span data-x="navigation-params-request">request</span></dt>
     <dd><var>request</var></dd>

     <dt><span data-x="navigation-params-response">response</span></dt>
     <dd><var>response</var></dd>

     <dt><span data-x="navigation-params-fetch-controller">fetch controller</span></dt>
     <dd><var>fetchController</var></dd>

     <dt><span data-x="navigation-params-commit-early-hints">commit early hints</span></dt>
     <dd><var>commitEarlyHints</var></dd>

     <dt><span data-x="navigation-params-coop">opener policy</span></dt>
     <dd><var>responseCOOP</var></dd>

     <dt><span data-x="navigation-params-reserved-environment">reserved environment</span></dt>
     <dd><var>request</var>'s <span data-x="concept-request-reserved-client">reserved client</span></dd>

     <dt><span data-x="navigation-params-origin">origin</span></dt>
     <dd><var>responseOrigin</var></dd>

     <dt><span data-x="navigation-params-policy-container">policy container</span></dt>
     <dd><var>resultPolicyContainer</var></dd>

     <dt><span data-x="navigation-params-sandboxing">final sandboxing flag set</span></dt>
     <dd><var>finalSandboxFlags</var></dd>

     <dt><span data-x="navigation-params-coop-enforcement-result">COOP enforcement result</span></dt>
     <dd><var>coopEnforcementResult</var></dd>

     <dt><span data-x="navigation-params-nav-timing-type">navigation timing type</span></dt>
     <dd><var>navTimingType</var></dd>

     <dt><span data-x="navigation-params-about-base-url">about base URL</span></dt>
     <dd><var>entry</var>'s <span data-x="she-document-state">document state</span>'s <span
     data-x="document-state-about-base-url">about base URL</span></dd>

     <dt><span data-x="navigation-params-user-involvement">user involvement</span></dt>
     <dd><var>userInvolvement</var></dd>
    </dl>
   </li>
  </ol>

  <p>An element has a <dfn>browsing context scope origin</dfn> if its <code>Document</code>'s
  <span>node navigable</span> is a <span>top-level traversable</span> or if all of its
  <code>Document</code>'s <span>ancestor navigables</span> all have <span
  data-x="nav-document">active documents</span> whose <span
  data-x="concept-document-origin">origins</span> are the <span>same origin</span> as the element's
  <span>node document</span>'s <span data-x="concept-document-origin">origin</span>. If an element
  has a <span>browsing context scope origin</span>, then its value is the <span
  data-x="concept-document-origin">origin</span> of the element's <span>node document</span>.</p>

  <p class="XXX">This definition is broken and needs investigation to see what it was intended to
  express: see <a href="https://github.com/whatwg/html/issues/4703">issue #4703</a>.</p>

  <p>To <dfn data-x="loading a document">load a document</dfn> given <span>navigation params</span>
  <var>navigationParams</var>, <span>source snapshot params</span> <var>sourceSnapshotParams</var>,
  and <span>origin</span> <var>initiatorOrigin</var>, perform the following steps. They return a
  <code>Document</code> or null.</p>

  <ol>
   <li><p>Let <var>type</var> be the <span data-x="Content-Type sniffing">computed type</span> of
   <var>navigationParams</var>'s <span
   data-x="navigation-params-response">response</span>.</p></li>

   <li>
    <p>If the user agent has been configured to process resources of the given <var>type</var>
    using some mechanism other than rendering the content in a <span>navigable</span>, then
    skip this step. Otherwise, if the <var>type</var> is one of the following types:</p>

    <dl class="switch">
     <dt>an <span>HTML MIME type</span></dt>
     <dd>Return the result of <span data-x="navigate-html">loading an HTML document</span>, given
     <var>navigationParams</var>.</dd>

     <dt>an <span>XML MIME type</span> that is not an <span>explicitly supported XML MIME
     type</span></dt>
     <dd>Return the result of <span data-x="navigate-xml">loading an XML document</span> given
     <var>navigationParams</var> and <var>type</var>.</dd>

     <dt>a <span>JavaScript MIME type</span></dt>
     <dt>a <span>JSON MIME type</span> that is not an <span>explicitly supported JSON MIME
     type</span></dt>
     <dt>"<code>text/css</code>"</dt>
     <dt>"<code>text/plain</code>"</dt>
     <dt>"<code>text/vtt</code>"</dt>
     <dd>Return the result of <span data-x="navigate-text">loading a text document</span> given
     <var>navigationParams</var> and <var>type</var>.</dd>

     <dt>"<code>multipart/x-mixed-replace</code>"</dt>
     <dd>Return the result of <span data-x="navigate-multipart-x-mixed-replace">loading a
     <code>multipart/x-mixed-replace</code> document</span>, given <var>navigationParams</var>,
     <var>sourceSnapshotParams</var>, and <var>initiatorOrigin</var>.</dd>

     <dt>A supported image, video, or audio type</dt>
     <dd>Return the result of <span data-x="navigate-media">loading a media document</span> given
     <var>navigationParams</var> and <var>type</var>.</dd>

     <dt>"<code data-x="">application/pdf</code>"</dt>
     <dt>"<code data-x="">text/pdf</code>"</dt>
     <dd>If the user agent's <span>PDF viewer supported</span> is true, return the result of <span
     data-x="navigate-ua-inline">creating a document for inline content that doesn't have a
     DOM</span> given <var>navigationParams</var>'s <span
     data-x="navigation-params-navigable">navigable</span>, <var>navigationParams</var>'s <span
     data-x="navigation-params-id">id</span>, <var>navigationParams</var>'s <span
     data-x="navigation-params-nav-timing-type">navigation timing type</span>, and
     <var>navigationParams</var>'s <span data-x="navigation-params-user-involvement">user
     involvement</span>.</dd>
    </dl>

    <p>Otherwise, proceed onward.</p>

    <p>An <dfn>explicitly supported XML MIME type</dfn> is an <span>XML MIME type</span> for which
    the user agent is configured to use an external application to render the content, or for which
    the user agent has dedicated processing rules. For example, a web browser with a built-in Atom
    feed viewer would be said to explicitly support the <code>application/atom+xml</code> MIME
    type.</p>

    <p>An <dfn>explicitly supported JSON MIME type</dfn> is a <span>JSON MIME type</span> for which
    the user agent is configured to use an external application to render the content, or for which
    the user agent has dedicated processing rules.</p>

    <p class="note">In both cases, the external application or user agent will either <span
    data-x="navigate-ua-inline">display the content inline</span> directly in
    <var>navigationParams</var>'s <span data-x="navigation-params-navigable">navigable</span>, or
    <span data-x="hand-off to external software">hand it off to external software</span>. Both
    happen in the steps below.</p>
   </li>

   <li id="navigate-non-Document"><p>If, given <var>type</var>, the new resource is to be handled
   by displaying some sort of inline content, e.g., a native rendering of the content or an error
   message because the specified type is not supported, then return the result of <span
   data-x="navigate-ua-inline">creating a document for inline content that doesn't have a
   DOM</span> given <var>navigationParams</var>'s <span
   data-x="navigation-params-navigable">navigable</span>, <var>navigationParams</var>'s <span
   data-x="navigation-params-id">id</span>, <var>navigationParams</var>'s <span
   data-x="navigation-params-nav-timing-type">navigation timing type</span>, and
   <var>navigationParams</var>'s <span data-x="navigation-params-user-involvement">user
   involvement</span>.</p></li>

   <li><p>Otherwise, the document's <var>type</var> is such that the resource will not affect
   <var>navigationParams</var>'s <span data-x="navigation-params-navigable">navigable</span>, e.g.,
   because the resource is to be handed to an external application or because it is an unknown type
   that will be processed by <span>handle as a download</span>. <span>Hand-off to external
   software</span> given <var>navigationParams</var>'s <span
   data-x="navigation-params-response">response</span>, <var>navigationParams</var>'s <span
   data-x="navigation-params-navigable">navigable</span>, <var>navigationParams</var>'s <span
   data-x="navigation-params-sandboxing">final sandboxing flag set</span>,
   <var>sourceSnapshotParams</var>'s <span data-x="source-snapshot-params-activation">has transient
   activation</span>, and <var>initiatorOrigin</var>.</p></li>

   <li><p>Return null.</p></li>
  </ol>


  <h4>Applying the history step</h4>

  <p>For both navigation and traversal, once we have an idea of where we want to head to in the
  session history, much of the work comes about in applying that notion to the <span>traversable
  navigable</span> and the relevant <code>Document</code>. For navigations, this work generally
  occurs toward the end of the process; for traversals, it is the beginning.</p>

  <h5>Updating the traversable</h5>

  <p>Ensuring a <span data-x="traversable navigable">traversable</span> ends up at the right
  session history step is particularly complex, as it can involve coordinating across multiple
  <span>navigable</span> descendants of the traversable, <a
  href="#populating-a-session-history-entry">populating</a> them in parallel, and then
  synchronizing back up to ensure everyone has the same view of the result. This is further
  complicated by the existence of synchronous same-document navigations being mixed together with
  cross-document navigations, and how web pages have come to have certain relative timing
  expectations.</p>

  <p>A <dfn>changing navigable continuation state</dfn> is used to store information during the
  <span>apply the history step</span> algorithm, allowing parts of the algorithm to continue only
  after other parts have finished. It is a <span>struct</span> with:</p>

  <dl>
   <dt><dfn data-x="changing-nav-continuation-displayed-document">displayed document</dfn></dt>
   <dd>A <code>Document</code></dd>

   <dt><dfn data-x="changing-nav-continuation-target-entry">target entry</dfn></dt>
   <dd>A <span>session history entry</span></dd>

   <dt><dfn data-x="changing-nav-continuation-navigable">navigable</dfn></dt>
   <dd>A <span>navigable</span></dd>

   <dt><dfn data-x="changing-nav-continuation-update-only">update only</dfn></dt>
   <dd>A boolean</dd>
  </dl>

  <hr>

  <p>Although all updates to the <span>traversable navigable</span> end up in the same <span>apply
  the history step</span> algorithm, each possible entry point comes along with some minor
  customizations:</p>

  <p>To <dfn>update for navigable creation/destruction</dfn> given a <span>traversable
  navigable</span> <var>traversable</var>:</p>

  <ol>
   <li><p>Let <var>step</var> be <var>traversable</var>'s <span
   data-x="tn-current-session-history-step">current session history step</span>.</p></li>

   <li><p>Return the result of <span data-x="apply the history step">applying the history
   step</span> <var>step</var> to <var>traversable</var> given false, null, null, "<code
   data-x="uni-none">none</code>", and null.</p></li>
  </ol>

  <p>To <dfn>apply the push/replace history step</dfn> given a non-negative integer <var>step</var>
  to a <span>traversable navigable</span> <var>traversable</var>, given a <span>history handling
  behavior</span> <var>historyHandling</var> and a <span>user navigation involvement</span>
  <var>userInvolvement</var>:</p>

  <ol>
   <li><p>Return the result of <span data-x="apply the history step">applying the history
   step</span> <var>step</var> to <var>traversable</var> given false, null, null,
   <var>userInvolvement</var>, and <var>historyHandling</var>.</p></li>
  </ol>

  <p class="note"><span>Apply the push/replace history step</span> never passes <span>source
  snapshot params</span> or an initiator <span>navigable</span> to <span>apply the history
  step</span>. This is because those checks are done earlier in the <span
  data-x="navigate">navigation</span> algorithm.</p>

  <p>To <dfn>apply the reload history step</dfn> to a <span>traversable navigable</span>
  <var>traversable</var> given <span>user navigation involvement</span>
  <var>userInvolvement</var>:</p>

  <ol>
   <li><p>Let <var>step</var> be <var>traversable</var>'s <span
   data-x="tn-current-session-history-step">current session history step</span>.</p></li>

   <li><p>Return the result of <span data-x="apply the history step">applying the history
   step</span> <var>step</var> to <var>traversable</var> given true, null, null,
   <var>userInvolvement</var>, and "<code
   data-x="dom-NavigationType-reload">reload</code>".</p></li>
  </ol>

  <p class="note"><span>Apply the reload history step</span> never passes <span>source snapshot
  params</span> or an initiator <span>navigable</span> to <span>apply the history step</span>. This
  is because reloading is always treated as if it were done by the <span>navigable</span> itself,
  even in cases like <code data-x="">parent.location.reload()</code>.</p>

  <p>To <dfn>apply the traverse history step</dfn> given a non-negative integer <var>step</var> to a
  <span>traversable navigable</span> <var>traversable</var>, with <span>source snapshot
  params</span> <var>sourceSnapshotParams</var>, <span>navigable</span> <var>initiatorToCheck</var>,
  and <span>user navigation involvement</span> <var>userInvolvement</var>:</p>

  <ol>
   <li><p>Return the result of <span data-x="apply the history step">applying the history
   step</span> <var>step</var> to <var>traversable</var> given true,
   <var>sourceSnapshotParams</var>, <var>initiatorToCheck</var>,
   <var>userInvolvement</var>, and "<code
   data-x="dom-NavigationType-traverse">traverse</code>".</p></li>
  </ol>

  <hr>

  <p>Now for the algorithm itself.</p>

  <p id="update-the-session-history-with-the-new-page">To <dfn>apply the history step</dfn> given a
  non-negative integer <var>step</var> to a <span>traversable navigable</span>
  <var>traversable</var>, with boolean <var>checkForCancelation</var>, <span>source snapshot
  params</span>-or-null <var>sourceSnapshotParams</var>, <span>navigable</span>-or-null
  <var>initiatorToCheck</var>, <span>user navigation involvement</span> <var>userInvolvement</var>,
  and <code>NavigationType</code>-or-null <var>navigationType</var>, perform the following steps.
  They return "<code data-x="">initiator-disallowed</code>", "<code
  data-x="">canceled-by-beforeunload</code>", "<code data-x="">canceled-by-navigate</code>", or
  "<code data-x="">applied</code>".</p>

  <ol>
   <li><p><span>Assert</span>: This is running within <var>traversable</var>'s <span
   data-x="tn-session-history-traversal-queue">session history traversal queue</span>.</p></li>

   <li><p>Let <var>targetStep</var> be the result of <span>getting the used step</span> given
   <var>traversable</var> and <var>step</var>.</p></li>

   <li>
    <p>If <var>initiatorToCheck</var> is not null, then:</p>

    <ol>
     <li><p><span>Assert</span>: <var>sourceSnapshotParams</var> is not null.</p></li>

     <li><p><span data-x="list iterate">For each</span> <var>navigable</var> of <span>get all
     navigables whose current session history entry will change or reload</span>: if
     <var>initiatorToCheck</var> is not <span>allowed by sandboxing to navigate</span>
     <var>navigable</var> given <var>sourceSnapshotParams</var>, then return "<code
     data-x="">initiator-disallowed</code>".</p></li>
    </ol>
   </li>

   <li><p>Let <var>navigablesCrossingDocuments</var> be the result of <span>getting all navigables
   that might experience a cross-document traversal</span> given <var>traversable</var> and
   <var>targetStep</var>.</p></li>

   <li><p>If <var>checkForCancelation</var> is true, and the result of <span>checking if unloading
   is canceled</span> given <var>navigablesCrossingDocuments</var>, <var>traversable</var>,
   <var>targetStep</var>, and <var>userInvolvement</var> is not "<code data-x="">continue</code>",
   then return that result.</p></li>

   <li><p>Let <var>changingNavigables</var> be the result of <span>get all navigables whose current
   session history entry will change or reload</span> given <var>traversable</var> and
   <var>targetStep</var>.</p></li>

   <li><p>Let <var>nonchangingNavigablesThatStillNeedUpdates</var> be the result of <span>getting
   all navigables that only need history object length/index update</span> given
   <var>traversable</var> and <var>targetStep</var>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>navigable</var> of
    <var>changingNavigables</var>:</p>

    <ol>
     <li><p>Let <var>targetEntry</var> be the result of <span>getting the target history
     entry</span> given <var>navigable</var> and <var>targetStep</var>.</p></li>

     <li><p>Set <var>navigable</var>'s <span data-x="nav-current-history-entry">current session
     history entry</span> to <var>targetEntry</var>.</p></li>

     <li><p><span>Set the ongoing navigation</span> for <var>navigable</var> to "<code
     data-x="">traversal</code>".</p></li>
    </ol>
   </li>

   <li><p>Let <var>totalChangeJobs</var> be the <span data-x="list size">size</span> of
   <var>changingNavigables</var>.</p></li>

   <li><p>Let <var>completedChangeJobs</var> be 0.</p></li>

   <li>
    <p>Let <var>changingNavigableContinuations</var> be an empty <span>queue</span> of <span
    data-x="changing navigable continuation state">changing navigable continuation
    states</span>.</p>

    <p class="note">This queue is used to split the operations on <var>changingNavigables</var>
    into two parts. Specifically, <var>changingNavigableContinuations</var> holds data for the <a
    href="#continuations-dequeue">second part</a>.</p>
   </li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>navigable</var> of
    <var>changingNavigables</var>, <span>queue a global task</span> on the <span>navigation and
    traversal task source</span> of <var>navigable</var>'s <span data-x="nav-window">active
    window</span> to run the steps:</p>

    <p class="note">This set of steps are split into two parts to allow synchronous navigations to
    be processed before documents unload. State is stored in
    <var>changingNavigableContinuations</var> for the <a href="#continuations-dequeue">second
    part</a>.</p>

    <ol>
     <li><p>Let <var>displayedEntry</var> be <var>navigable</var>'s <span
     data-x="nav-active-history-entry">active session history entry</span>.</p></li>

     <li><p>Let <var>targetEntry</var> be <var>navigable</var>'s <span
     data-x="nav-current-history-entry">current session history entry</span>.</p></li>

     <li>
      <p>Let <var>changingNavigableContinuation</var> be a <span>changing navigable continuation
      state</span> with:</p>

      <dl class="props">
       <dt><span data-x="changing-nav-continuation-displayed-document">displayed document</span></dt>
       <dd><var>displayedEntry</var>'s <span data-x="she-document">document</span></dd>

       <dt><span data-x="changing-nav-continuation-target-entry">target entry</span></dt>
       <dd><var>targetEntry</var></dd>

       <dt><span data-x="changing-nav-continuation-navigable">navigable</span></dt>
       <dd><var>navigable</var></dd>

       <dt><span data-x="changing-nav-continuation-update-only">update-only</span></dt>
       <dd>false</dd>
      </dl>
     </li>

     <li>
      <p>If <var>displayedEntry</var> is <var>targetEntry</var> and <var>targetEntry</var>'s <span
      data-x="she-document-state">document state</span>'s <span
      data-x="document-state-reload-pending">reload pending</span> is false, then:</p>

      <ol>
       <li><p>Set <var>changingNavigableContinuation</var>'s <span
       data-x="changing-nav-continuation-update-only">update-only</span> to true.</p></li>

       <li><p><span>Enqueue</span> <var>changingNavigableContinuation</var> on
       <var>changingNavigableContinuations</var>.</p></li>

       <li><p>Abort these steps.</p></li>
      </ol>

      <p class="note">This case occurs due to a <span data-x="finalize a same-document
      navigation">synchronous navigation</span> which already updated the <span
      data-x="nav-active-history-entry">active session history entry</span>.</p>
     </li>

     <li>
      <p>Switch on <var>navigationType</var>:</p>

      <dl class=switch>
       <dt>"<code data-x="dom-NavigationType-reload">reload</code>"</dt>
       <dd><p><span>Assert</span>: <var>targetEntry</var>'s <span
       data-x="she-document-state">document state</span>'s <span
       data-x="document-state-reload-pending">reload pending</span> is true.</p></dd>

       <dt>"<code data-x="dom-NavigationType-traverse">traverse</code>"</dt>
       <dd><p><span>Assert</span>: <var>targetEntry</var>'s
       <span data-x="she-document-state">document state</span>'s <span
       data-x="document-state-ever-populated">ever populated</span> is true.</p></dd>

       <dt>"<code data-x="dom-NavigationType-replace">replace</code>"</dt>
       <dd><p><span>Assert</span>: <var>targetEntry</var>'s <span data-x="she-step">step</span>
       is <var>displayedEntry</var>'s <span data-x="she-step">step</span> and
       <var>targetEntry</var>'s <span data-x="she-document-state">document state</span>'s <span
       data-x="document-state-ever-populated">ever populated</span> is false.</p></dd>

       <dt>"<code data-x="dom-NavigationType-push">push</code>"</dt>
       <dd><p><span>Assert</span>: <var>targetEntry</var>'s <span data-x="she-step">step</span> is
       <var>displayedEntry</var>'s <span data-x="she-step">step</span> + 1 and
       <var>targetEntry</var>'s <span data-x="she-document-state">document state</span>'s <span
       data-x="document-state-ever-populated">ever populated</span> is false.</p></dd>
      </dl>
     </li>

     <li><p>Let <var>oldOrigin</var> be <var>targetEntry</var>'s <span
     data-x="she-document-state">document state</span>'s <span
     data-x="document-state-origin">origin</span>.</p></li>

     <li id="descendant-navigable-traversal-navigate-events">
      <p>If all of the following are true:</p>

      <ul>
       <li><p><var>navigable</var> is not <var>traversable</var>;</p></li>

       <li><p><var>targetEntry</var> is not <var>navigable</var>'s <span
       data-x="nav-current-history-entry">current session history entry</span>; and</p></li>

       <li><p><var>oldOrigin</var> is the <span data-x="same origin">same</span> as
       <var>navigable</var>'s <span
       data-x="nav-current-history-entry">current session history entry</span>'s <span
       data-x="she-document-state">document state</span>'s <span
       data-x="document-state-origin">origin</span>,</p></li>
      </ul>

      <p>then:</p>

      <ol>
       <li><p>Let <var>navigation</var> be <var>navigable</var>'s <span
       data-x="nav-window">active window</span>'s <span
       data-x="window-navigation-api">navigation API.</span></p></li>

       <li><p><span>Fire a traverse <code data-x="event-navigate">navigate</code> event</span> at
       <var>navigation</var> given <var>targetEntry</var> and <var>userInvolvement</var>.</p></li>
      </ol>
     </li>

     <li>
      <p>If <var>targetEntry</var>'s <span data-x="she-document">document</span> is null, or
      <var>targetEntry</var>'s <span data-x="she-document-state">document state</span>'s <span
      data-x="document-state-reload-pending">reload pending</span> is true, then:</p>

      <ol>
       <li><p>Let <var>navTimingType</var> be "<code
       data-x="dom-navigationtimingtype-back_forward">back_forward</code>" if
       <var>targetEntry</var>'s <span data-x="she-document">document</span> is null; otherwise
       "<code data-x="dom-navigationtimingtype-back_forward">reload</code>".</p></li>

       <li><p>Let <var>targetSnapshotParams</var> be the result of <span>snapshotting target
       snapshot params</span> given <var>navigable</var>.</p></li>

       <li><p>Let <var>potentiallyTargetSpecificSourceSnapshotParams</var> be
       <var>sourceSnapshotParams</var>.</p></li>

       <li>
        <p>If <var>potentiallyTargetSpecificSourceSnapshotParams</var> is null, then set it to the
        result of <span>snapshotting source snapshot params</span> given <var>navigable</var>'s
        <span data-x="nav-document">active document</span>.</p>

        <p class="note">In this case there is no clear source of the traversal/reload. We treat this
        situation as if <var>navigable</var> navigated itself, but note that some properties of
        <var>targetEntry</var>'s original initiator are preserved in <var>targetEntry</var>'s <span
        data-x="she-document-state">document state</span>, such as the <span
        data-x="document-state-initiator-origin">initiator origin</span> and <span
        data-x="document-state-request-referrer">referrer</span>, which will appropriately influence
        the navigation.</p>
       </li>

       <li><p>Set <var>targetEntry</var>'s <span data-x="she-document-state">document
       state</span>'s <span data-x="document-state-reload-pending">reload pending</span> to
       false.</p></li>

       <li><p>Let <var>allowPOST</var> be <var>targetEntry</var>'s <span
       data-x="she-document-state">document state</span>'s <span
       data-x="document-state-reload-pending">reload pending</span>.</p>

       <li><p><span>In parallel</span>, <span>attempt to populate the history entry's
       document</span> for <var>targetEntry</var>, given <var>navigable</var>,
       <var>potentiallyTargetSpecificSourceSnapshotParams</var>, <var>targetSnapshotParams</var>,
       <var>userInvolvement</var>, with <i data-x="attempt-to-populate-allow-post">allowPOST</i> set
       to <var>allowPOST</var> and <span
       data-x="attempt-to-populate-completion-steps"><i>completionSteps</i></span> set to
       <span>queue a global task</span> on the <span>navigation and traversal task source</span>
       given <var>navigable</var>'s <span data-x="nav-window">active window</span> to run
       <var>afterDocumentPopulated</var>.</p></li>
      </ol>

      <p>Otherwise, run <var>afterDocumentPopulated</var> <span>immediately</span>.</p>

      <p>In both cases, let <var>afterDocumentPopulated</var> be the following steps:</p>

      <ol>
       <li>
        <p>If <var>targetEntry</var>'s <span data-x="she-document">document</span> is null, then
        set <var>changingNavigableContinuation</var>'s <span
        data-x="changing-nav-continuation-update-only">update-only</span> to true.</p>

        <p class="note">This means we tried to populate the document, but were unable to do so,
        e.g. because of the server returning a 204.</p>

        <div class="note">
         <p>These kinds of failed navigations or traversals will not be signaled to the <a
         href="#navigation-api">navigation API</a> (e.g., through the promises of any
         <span>navigation API method tracker</span>, or the <code
         data-x="event-navigateerror">navigateerror</code> event). Doing so would leak information
         about the timing of responses from other origins, in the cross-origin case, and providing
         different results in the cross-origin vs. same-origin cases was deemed too confusing.</p>

         <p>However, implementations could use this opportunity to clear any promise handlers for
         the <code data-x="dom-NavigationTransition-finished">navigation.transition.finished</code>
         promise, as they are guaranteed at this point to never run. And, they might wish to
         <span>report a warning to the console</span> if any part of the navigation API initiated
         these navigations, to make it clear to the web developer why their promises will never
         settle and events will never fire.</p>
        </div>
       </li>

       <li>
        <p>If <var>targetEntry</var>'s <span data-x="she-document">document</span>'s <span
        data-x="concept-document-origin">origin</span> is not <var>oldOrigin</var>, then set
        <var>targetEntry</var>'s <span data-x="she-classic-history-api-state">classic history API
        state</span> to <span>StructuredSerializeForStorage</span>(null).</p>

        <p class="note">This clears history state when the origin changed vs a previous load of
        <var>targetEntry</var> without a redirect occuring. This can happen due to a change in CSP
        sandbox headers.</p>
       </li>

       <li>
        <p>If all of the following are true:</p>

        <ul>
         <li><p><var>navigable</var>'s <span data-x="nav-parent">parent</span> is null;</p></li>

         <li><p><var>targetEntry</var>'s <span data-x="she-document">document</span>'s <span
         data-x="concept-document-bc">browsing context</span> is not an <span>auxiliary browsing
         context</span> whose <span>opener browsing context</span> is non-null; and</p></li>

         <li><p><var>targetEntry</var>'s <span data-x="she-document">document</span>'s <span
         data-x="concept-document-origin">origin</span> is not <var>oldOrigin</var>,</p></li>
        </ul>

        <p>then set <var>targetEntry</var>'s <span data-x="she-document-state">document
        state</span>'s <span data-x="document-state-nav-target-name">navigable target name</span>
        to the empty string.</p>
       </li>

       <li>
        <p><span>Enqueue</span> <var>changingNavigableContinuation</var> on
        <var>changingNavigableContinuations</var>.</p>

        <p class="note">The rest of this job <a href="#continuations-dequeue">runs later</a> in
        this algorithm.</p>
       </li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Let <var>navigablesThatMustWaitBeforeHandlingSyncNavigation</var> be an empty
   <span>set</span>.</p></li>

   <li>
    <p>While <var>completedChangeJobs</var> does not equal <var>totalChangeJobs</var>:</p>

    <ol>
     <li>
      <p>If <var>traversable</var>'s <span data-x="tn-running-nested-apply-history-step">running
      nested apply history step</span> is false, then:</p>

      <ol>
       <li id="sync-navigations-jump-queue">
        <p>While <var>traversable</var>'s <span data-x="tn-session-history-traversal-queue">session
        history traversal queue</span>'s <span
        data-x="session-history-traversal-parallel-queue-algorithm-set">algorithm set</span> <span
        data-x="list contains">contains</span> one or more <span
        data-x="session-history-traversal-parallel-queue-sync-nav-steps">synchronous navigation
        steps</span> with a <span
        data-x="session-history-traversal-parallel-queue-sync-nav-steps-target-nav">target
        navigable</span> not <span data-x="list contains">contained</span> in
        <var>navigablesThatMustWaitBeforeHandlingSyncNavigation</var>:</p>

        <ol>
         <li><p>Let <var>steps</var> be the first <span data-x="list item">item</span> in
         <var>traversable</var>'s <span data-x="tn-session-history-traversal-queue">session history
         traversal queue</span>'s <span
         data-x="session-history-traversal-parallel-queue-algorithm-set">algorithm set</span> that
         is <span data-x="session-history-traversal-parallel-queue-sync-nav-steps">synchronous
         navigation steps</span> with a <span
         data-x="session-history-traversal-parallel-queue-sync-nav-steps-target-nav">target
         navigable</span> not <span data-x="list contains">contained</span> in
         <var>navigablesThatMustWaitBeforeHandlingSyncNavigation</var>.</p></li>

         <li><p><span data-x="list remove">Remove</span> <var>steps</var> from
         <var>traversable</var>'s <span data-x="tn-session-history-traversal-queue">session history
         traversal queue</span>'s <span
         data-x="session-history-traversal-parallel-queue-algorithm-set">algorithm
         set</span>.</p></li>

         <li><p>Set <var>traversable</var>'s <span
         data-x="tn-running-nested-apply-history-step">running nested apply history step</span> to
         true.</p></li>

         <li><p>Run <var>steps</var>.</p></li>

         <li><p>Set <var>traversable</var>'s <span
         data-x="tn-running-nested-apply-history-step">running nested apply history step</span> to
         false.</p></li>
        </ol>

        <p class="note">Synchronous navigations that are intended to take place before this
        traversal jump the queue at this point, so they can be added to the correct place in
        <var>traversable</var>'s <span data-x="tn-session-history-entries">session history
        entries</span> before this traversal potentially unloads their document. <a
        href="#sync-navigation-steps-queue-jumping-examples">More details can be found
        here</a>.</p>
       </li>
      </ol>
     </li>

     <li id="continuations-dequeue"><p>Let <var>changingNavigableContinuation</var> be the result
     of <span data-x="dequeue">dequeuing</span> from
     <var>changingNavigableContinuations</var>.</p></li>

     <li><p>If <var>changingNavigableContinuation</var> is nothing, then
     <span>continue</span>.</p></li>

     <li><p>Let <var>displayedDocument</var> be <var>changingNavigableContinuation</var>'s <span
     data-x="changing-nav-continuation-displayed-document">displayed document</span>.</p></li>

     <li><p>Let <var>targetEntry</var> be <var>changingNavigableContinuation</var>'s <span
     data-x="changing-nav-continuation-target-entry">target entry</span>.</p></li>

     <li><p>Let <var>navigable</var> be <var>changingNavigableContinuation</var>'s <span
     data-x="changing-nav-continuation-navigable">navigable</span>.</p></li>

     <li>
      <p>Let (<var>scriptHistoryLength</var>, <var>scriptHistoryIndex</var>) be the result of
      <span>getting the history object length and index</span> given <var>traversable</var> and
      <var>targetStep</var>.</p>

      <p class="note">These values might have changed since they were last calculated.</p>
     </li>

     <li>
      <p><span data-x="list append">Append</span> <var>navigable</var> to
      <var>navigablesThatMustWaitBeforeHandlingSyncNavigation</var>.</p>

      <p class="note">Once a navigable has reached this point in traversal, additionally queued
      synchronous navigation steps are likely to be intended to occur after this traversal rather
      than before it, so they no longer jump the queue. <a
      href="#sync-navigation-steps-queue-jumping-examples">More details can be found here</a>.</p>
     </li>

     <li><p>Let <var>entriesForNavigationAPI</var> be the result of <span>getting session history
     entries for the navigation API</span> given <var>navigable</var> and
     <var>targetStep</var>.</p></li>

     <li>
      <p>If <var>changingNavigableContinuation</var>'s <span
      data-x="changing-nav-continuation-update-only">update-only</span> is true, or
      <var>targetEntry</var>'s <span data-x="she-document">document</span> is
      <var>displayedDocument</var>, then:</p>

      <p class="note">This is a same-document navigation: we proceed without unloading.</p>

      <ol>
       <li>
        <p><span>Set the ongoing navigation</span> for <var>navigable</var> to null.</p>

        <p class="note">This allows new <span data-x="navigate">navigations</span> of
        <var>navigable</var> to start, whereas during the traversal they were blocked.</p>
       </li>

       <li><p><span>Queue a global task</span> on the <span>navigation and traversal task
       source</span> given <var>navigable</var>'s <span data-x="nav-window">active window</span> to
       perform <var>afterPotentialUnloads</var>.</p></li>
      </ol>
     </li>

     <li>
      <p>Otherwise:</p>

      <ol>
       <li><p><span>Assert</span>: <var>navigationType</var> is not null.</p></li>

       <li><p><span data-x="deactivate a document for a cross-document navigation">Deactivate</span>
       <var>displayedDocument</var>, given <var>userInvolvement</var>, <var>targetEntry</var>,
       <var>navigationType</var>, and <var>afterPotentialUnloads</var>.</p></li>
      </ol>
     </li>

     <li>
      <p>In both cases, let <var>afterPotentialUnloads</var> be the following steps:</p>

      <ol>
       <li><p>Let <var>previousEntry</var> be <var>navigable</var>'s <span
       data-x="nav-active-history-entry">active session history entry</span>.</p></li>

       <li><p>If <var>changingNavigableContinuation</var>'s <span
       data-x="changing-nav-continuation-update-only">update-only</span> is false, then
       <span>activate history entry</span> <var>targetEntry</var> for
       <var>navigable</var>.</p></li>

       <li><p>Let <var>updateDocument</var> be an algorithm step which performs <span>update
       document for history step application</span> given <var>targetEntry</var>'s <span
       data-x="she-document">document</span>, <var>targetEntry</var>,
       <var>changingNavigableContinuation</var>'s <span
       data-x="changing-nav-continuation-update-only">update-only</span>,
       <var>scriptHistoryLength</var>, <var>scriptHistoryIndex</var>, <var>navigationType</var>,
       <var>entriesForNavigationAPI</var>, and <var>previousEntry</var>.</p></li>

       <li><p>If <var>targetEntry</var>'s <span data-x="she-document">document</span> is equal to
       <var>displayedDocument</var>, then perform <var>updateDocument</var>.</p></li>

       <li><p>Otherwise, <span>queue a global task</span> on the <span>navigation and traversal task
       source</span> given <var>targetEntry</var>'s <span data-x="she-document">document</span>'s
       <span>relevant global object</span> to perform <var>updateDocument</var>.</p></li>

       <li><p>Increment <var>completedChangeJobs</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li>
    <p>Let <var>totalNonchangingJobs</var> be the <span data-x="list size">size</span> of
    <var>nonchangingNavigablesThatStillNeedUpdates</var>.</p>

    <p class="note">This step onwards deliberately waits for all the previous operations to
    complete, as they include <a href="#sync-navigations-jump-queue">processing synchronous
    navigations</a> which will also post tasks to update history length and index.</p>
   </li>

   <li><p>Let <var>completedNonchangingJobs</var> be 0.</p></li>

   <li><p>Let (<var>scriptHistoryLength</var>, <var>scriptHistoryIndex</var>) be the result of
   <span>getting the history object length and index</span> given <var>traversable</var> and
   <var>targetStep</var>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>navigable</var> of
    <var>nonchangingNavigablesThatStillNeedUpdates</var>, <span>queue a global task</span> on the
    <span>navigation and traversal task source</span> given <var>navigable</var>'s <span
    data-x="nav-window">active window</span> to run the steps:</p>

    <ol>
     <li><p>Let <var>document</var> be <var>navigable</var>'s <span data-x="nav-document">active
     document</span>.</p></li>

     <li><p>Set <var>document</var>'s <span data-x="doc-history">history object</span>'s <span
     data-x="concept-history-index">index</span> to <var>scriptHistoryIndex</var>.</p></li>

     <li><p>Set <var>document</var>'s <span data-x="doc-history">history object</span>'s <span
     data-x="concept-history-length">length</span> to <var>scriptHistoryLength</var>.</p></li>

     <li><p>Increment <var>completedNonchangingJobs</var>.</p></li>
    </ol>
   </li>

   <li><p>Wait for <var>completedNonchangingJobs</var> to equal
   <var>totalNonchangingJobs</var>.</p></li>

   <li><p>Set <var>traversable</var>'s <span data-x="tn-current-session-history-step">current
   session history step</span> to <var>targetStep</var>.</p></li>

   <li><p>Return "<code data-x="">applied</code>".</p></li>
  </ol>

  <p>To <dfn>deactivate a document for a cross-document navigation</dfn> given a
  <code>Document</code> <var>displayedDocument</var>, a <span>user navigation involvement</span>
  <var>userNavigationInvolvement</var>, a <span>session history entry</span> <var>targetEntry</var>,
  a <code>NavigationType</code> <var>navigationType</var>, and <var>afterPotentialUnloads</var>,
  which is an algorithm that receives no arguments:</p>

  <ol>
   <li><p>Let <var>navigable</var> be <var>displayedDocument</var>'s
   <span>node navigable</span>.</p></li>

   <li><p>Let <var>potentiallyTriggerViewTransition</var> be false.</p></li>

   <li><p>Let <var>isBrowserUINavigation</var> be true if
   <var>userNavigationInvolvement</var> is "<code
   data-x="uni-browser-ui">browser UI</code>"; otherwise false.</p></li>

   <li><p>Set <var>potentiallyTriggerViewTransition</var> to the result of calling
   <span>can navigation trigger a cross-document view-transition?</span> given
   <var>displayedDocument</var>, <var>targetEntry</var>'s <span
   data-x="she-document">document</span>, <var>navigationType</var>, and
   <var>isBrowserUINavigation</var>.</p></li>

   <li>
    <p>If <var>potentiallyTriggerViewTransition</var> is false, then:</p>

    <ol>
     <li>
      <p>Let <var>firePageSwapBeforeUnload</var> be the following step:</p>

      <ol>
       <li><p><span>Fire the <code data-x="event-pageswap">pageswap</code> event</span> given
       <var>displayedDocument</var>, <var>targetEntry</var>, <var>navigationType</var>, and
       null.</p></li>
      </ol>
     </li>

     <li>
      <p><span>Set the ongoing navigation</span> for <var>navigable</var> to null.</p>

      <p class="note">This allows new <span data-x="navigate">navigations</span> of
      <var>navigable</var> to start, whereas during the traversal they were blocked.</p>
     </li>

     <li><p><span>Unload a document and its descendants</span> given
     <var>displayedDocument</var>, <var>targetEntry</var>'s <span
     data-x="she-document">document</span>, <var>afterPotentialUnloads</var>, and
     <var>firePageSwapBeforeUnload</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise, <span>queue a global task</span> on the
    <span>navigation and traversal task source</span> given <var>navigable</var>'s <span
    data-x="nav-window">active window</span> to run the steps:</p>

    <ol>
     <li>
      <p>Let <var>proceedWithNavigationAfterViewTransitionCapture</var> be the following
      step:</p>

      <ol>
       <li>
        <p><span data-x="tn-append-session-history-traversal-steps">Append the following session
        history traversal steps</span> to <var>navigable</var>'s <span
        data-x="nav-traversable">traversable navigable</span>:</p>

        <ol>
         <li>
          <p><span>Set the ongoing navigation</span> for <var>navigable</var> to null.</p>

          <p class="note">This allows new <span data-x="navigate">navigations</span> of
          <var>navigable</var> to start, whereas during the traversal they were blocked.</p>
         </li>

         <li><p><span>Unload a document and its descendants</span> given
         <var>displayedDocument</var>, <var>targetEntry</var>'s <span
         data-x="she-document">document</span>, and <var>afterPotentialUnloads</var>.</p></li>
        </ol>
       </li>
      </ol>
     </li>

     <li><p>Let <var>viewTransition</var> be the result of
     <span>setting up a cross-document view-transition</span> given
     <var>displayedDocument</var>, <var>targetEntry</var>'s <span
     data-x="she-document">document</span>, <var>navigationType</var>, and
     <var>proceedWithNavigationAfterViewTransitionCapture</var>.</p></li>

     <li><p><span>Fire the <code data-x="event-pageswap">pageswap</code> event</span> given
     <var>displayedDocument</var>, <var>targetEntry</var>, <var>navigationType</var>, and
     <var>viewTransition</var>.</p>

     <li>
      <p>If <var>viewTransition</var> is null, then run
      <var>proceedWithNavigationAfterViewTransitionCapture</var>.</p>

      <p class="note">In the case where a view transition started, the view transitions
      algorithms are responsible for calling
      <var>proceedWithNavigationAfterViewTransitionCapture</var>.</p>
     </li>
    </ol>
   </li>
  </ol>

  <p>To <dfn>fire the <code data-x="event-pageswap">pageswap</code> event</dfn> given a
  <code>Document</code> <var>displayedDocument</var>, a <span>session history entry</span>
  <var>targetEntry</var>, a <code>NavigationType</code> <var>navigationType</var>, and a
  <code>ViewTransition</code>-or-null <var>viewTransition</var>:</p>

  <ol>
   <li><p><span>Assert</span>: this is running as part of a <span
   data-x="concept-task">task</span> queued on <var>displayedDocument</var>'s
   <span>relevant agent</span>'s <span
   data-x="concept-agent-event-loop">event loop</span>.</p></li>

   <li><p>Let <var>navigation</var> be <var>displayedDocument</var>'s
   <span>relevant global object</span>'s <span
   data-x="window-navigation-api">navigation API</span>.</p></li>

   <li><p>Let <var>activation</var> be null.</p></li>

   <li>
    <p>If all of the following are true:</p>

    <ul>
     <li><p><var>targetEntry</var>'s <span data-x="she-document">document</span>'s <span
     data-x="concept-document-origin">origin</span> is <span>same origin</span> with
     <var>displayedDocument</var>'s <span data-x="concept-document-origin">origin</span>;
     and</p></li>

     <li><p><var>targetEntry</var>'s <span data-x="she-document">document</span>'s
     <span>was created via cross-origin redirects</span> is false, or <var>targetEntry</var>'s
     <span data-x="she-document">document</span>'s <span>latest entry</span> is not null,</p></li>
    </ul>

    <p>then:</p>

    <ol>
     <li>
      <p>Let <var>destinationEntry</var> be determined by switching on
      <var>navigationType</var>:</p>

      <dl class="switch">
       <dt>"<code data-x="dom-NavigationType-reload">reload</code>"</dt>
       <dd><p>The <span data-x="navigation-current-entry">current entry</span> of
       <var>navigation</var></p></dd>

       <dt>"<code data-x="dom-NavigationType-traverse">traverse</code>"</dt>
       <dd><p>The <code>NavigationHistoryEntry</code> in <var>navigation</var>'s <span
       data-x="navigation-entry-list">entry list</span> whose <span
       data-x="nhe-she">session history entry</span> is <var>targetEntry</var></p></dd>

       <dt>"<code data-x="dom-NavigationType-push">push</code>"</dt>
       <dt>"<code data-x="dom-NavigationType-replace">replace</code>"</dt>
       <dd><p>A new <code>NavigationHistoryEntry</code> in <var>displayedDocument</var>'s <span
       data-x="concept-relevant-realm">relevant realm</span> with its <span
       data-x="nhe-she">session history entry</span> set to <var>targetEntry</var>.</p></dd>
      </dl>
     </li>
     <li>
      <p>Set <var>activation</var> to a <span>new</span> <code>NavigationActivation</code>
      created in <var>displayedDocument</var>'s <span data-x="concept-relevant-realm">relevant
      realm</span>, with</p>

      <dl class="props">
       <dt><span data-x="nav-activation-old-entry">old entry</span></dt>
       <dd>the <span data-x="navigation-current-entry">current entry</span> of
       <var>navigation</var></dd>

       <dt><span data-x="nav-activation-new-entry">new entry</span></dt>
       <dd><var>destinationEntry</var></dd>

       <dt><span data-x="nav-activation-navigation-type">navigation type</span></dt>
       <dd><var>navigationType</var></dd>
      </dl>
     </li>
    </ol>

    <p class="note">This means that a cross-origin redirect during a navigation would result in a
    null <code data-x="dom-PageSwapEvent-activation">activation</code> in the old document's
    <code>PageSwapEvent</code>, unless the new document is being restored from <a
    href="#note-bfcache">bfcache</a>.</p>
   </li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named
   <code data-x="event-pageswap">pageswap</code> at <var>displayedDocument</var>'s
   <span>relevant global object</span>, using <code>PageSwapEvent</code> with its <code
   data-x="dom-PageSwapEvent-activation">activation</code> set to <var>activation</var>,
   and its <code data-x="dom-PageSwapEvent-viewTransition">viewTransition</code> set to
   <var>viewTransition</var>.</p></li>
  </ol>

  <p>To <dfn>activate history entry</dfn> <span>session history entry</span> <var>entry</var> for
  <span>navigable</span> <var>navigable</var>:</p>

  <ol>
   <li><p><span>Save persisted state</span> to the <span>navigable</span>'s <span
   data-x="nav-active-history-entry">active session history entry</span>.</p></li>

   <li><p>Let <var>newDocument</var> be <var>entry</var>'s <span
   data-x="she-document">document</span>.</p></li>

   <li><p><span>Assert</span>: <var>newDocument</var>'s <span>is initial
   <code>about:blank</code></span> is false, i.e., we never traverse back to the <span data-x="is
   initial about:blank">initial <code>about:blank</code></span> <code>Document</code> because it
   always gets <a href="#navigate-convert-to-replace">replaced</a> when we navigate away from
   it.</p></li>

   <li><p>Set <var>navigable</var>'s <span data-x="nav-active-history-entry">active session history
   entry</span> to <var>entry</var>.</p></li>

   <li><p><span>Make active</span> <var>newDocument</var>.</p></li>
  </ol>

  <p>To <dfn data-x="getting the used step">get the used step</dfn> given a <span>traversable
  navigable</span> <var>traversable</var>, and a non-negative integer <var>step</var>, perform the
  following steps. They return a non-negative integer.</p>

  <ol>
   <li><p>Let <var>steps</var> be the result of <span>getting all used history steps</span> within
   <var>traversable</var>.</p></li>

   <li>
    <p>Return the greatest <span data-x="list item">item</span> in <var>steps</var> that is less
    than or equal to <var>step</var>.</p>

    <p class="note">This caters for situations where there's no <span>session history entry</span>
    with <span data-x="she-step">step</span> <var>step</var>, due to the removal of a
    <span>navigable</span>.</p>
   </li>
  </ol>

  <p>To <dfn data-x="getting the history object length and index">get the history object length and
  index</dfn> given a <span>traversable navigable</span> <var>traversable</var>, and a non-negative
  integer <var>step</var>, perform the following steps. They return a <span>tuple</span> of two
  non-negative integers.</p>

  <ol>
   <li><p>Let <var>steps</var> be the result of <span>getting all used history steps</span> within
   <var>traversable</var>.</p></li>

   <li><p>Let <var>scriptHistoryLength</var> be the <span data-x="list size">size</span> of
   <var>steps</var>.</p></li>

   <li>
    <p><span>Assert</span>: <var>steps</var> <span data-x="list contains">contains</span>
    <var>step</var>.</p>

    <p class="note">It is assumed that <var>step</var> has been adjusted by <span>getting the used
    step</span>.</p>
   </li>

   <li><p>Let <var>scriptHistoryIndex</var> be the index of <var>step</var> in
   <var>steps</var>.</p></li>

   <li><p>Return (<var>scriptHistoryLength</var>, <var>scriptHistoryIndex</var>).</p></li>
  </ol>

  <p>To <dfn>get all navigables whose current session history entry will change or reload</dfn>
  given a <span>traversable navigable</span> <var>traversable</var>, and a non-negative integer
  <var>targetStep</var>, perform the following steps. They return a <span>list</span> of <span
  data-x="navigable">navigables</span>.</p>

  <ol>
   <li><p>Let <var>results</var> be an empty <span>list</span>.</p></li>

   <li>
    <p>Let <var>navigablesToCheck</var> be « <var>traversable</var> ».</p>

    <p class="note">This list is extended in the loop below.</p>
   </li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>navigable</var> of
    <var>navigablesToCheck</var>:</p>

    <ol>
     <li><p>Let <var>targetEntry</var> be the result of <span>getting the target history
     entry</span> given <var>navigable</var> and <var>targetStep</var>.</p></li>

     <li><p>If <var>targetEntry</var> is not <var>navigable</var>'s <span
     data-x="nav-current-history-entry">current session history entry</span> or
     <var>targetEntry</var>'s <span data-x="she-document-state">document state</span>'s <span
     data-x="document-state-reload-pending">reload pending</span> is true, then <span data-x="list
     append">append</span> <var>navigable</var> to <var>results</var>.</p></li>

     <li>
      <p>If <var>targetEntry</var>'s <span data-x="she-document">document</span> is
      <var>navigable</var>'s <span data-x="nav-document">document</span>, and
      <var>targetEntry</var>'s <span data-x="she-document-state">document state</span>'s <span
      data-x="document-state-reload-pending">reload pending</span> is false, then <span
      data-x="list extend">extend</span> <var>navigablesToCheck</var> with the <span data-x="child
      navigable">child navigables</span> of <var>navigable</var>.</p>

      <p class="note">Adding <span data-x="child navigable">child navigables</span> to
      <var>navigablesToCheck</var> means those navigables will also be checked by this loop. <span
      data-x="child navigable">Child navigables</span> are only checked if the
      <var>navigable</var>'s <span data-x="nav-document">active document</span> will not change as
      part of this traversal.</p>
     </li>
    </ol>
   </li>

   <li><p>Return <var>results</var>.</p></li>
  </ol>

  <p>To <dfn data-x="getting all navigables that only need history object length/index update">get
  all navigables that only need history object length/index update</dfn> given a <span>traversable
  navigable</span> <var>traversable</var>, and a non-negative integer <var>targetStep</var>,
  perform the following steps. They return a <span>list</span> of <span
  data-x="navigable">navigables</span>.</p>

  <p class="note">Other <span data-x="navigable">navigables</span> might not be impacted by the
  traversal. For example, if the response is a 204, the currently active document will remain.
  Additionally, going 'back' after a 204 will change the <span
  data-x="nav-current-history-entry">current session history entry</span>, but the <span
  data-x="nav-active-history-entry">active session history entry</span> will already be
  correct.</p>

  <ol>
   <li><p>Let <var>results</var> be an empty <span>list</span>.</p></li>

   <li>
    <p>Let <var>navigablesToCheck</var> be « <var>traversable</var> ».</p>

    <p class="note">This list is extended in the loop below.</p>
   </li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>navigable</var> of
    <var>navigablesToCheck</var>:</p>

    <ol>
     <li><p>Let <var>targetEntry</var> be the result of <span>getting the target history
     entry</span> given <var>navigable</var> and <var>targetStep</var>.</p></li>

     <li>
      <p>If <var>targetEntry</var> is <var>navigable</var>'s <span
      data-x="nav-current-history-entry">current session history entry</span> and
      <var>targetEntry</var>'s <span data-x="she-document-state">document state</span>'s <span
      data-x="document-state-reload-pending">reload pending</span> is false, then:</p>

      <ol>
       <li><p><span data-x="list append">Append</span> <var>navigable</var> to
       <var>results</var>.</p></li>

       <li>
        <p><span data-x="list extend">Extend</span> <var>navigablesToCheck</var> with
        <var>navigable</var>'s <span data-x="child navigable">child navigables</span>.</p>

        <p class="note">Adding <span data-x="child navigable">child navigables</span> to
        <var>navigablesToCheck</var> means those navigables will also be checked by this loop.
        <span data-x="child navigable">child navigables</span> are only checked if the
        <var>navigable</var>'s <span data-x="nav-document">active document</span> will not change
        as part of this traversal.</p>
       </li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Return <var>results</var>.</p></li>
  </ol>

  <p>To <dfn data-x="getting the target history entry">get the target history entry</dfn> given a
  <span>navigable</span> <var>navigable</var>, and a non-negative integer <var>step</var>, perform
  the following steps. They return a <span>session history entry</span>.</p>

  <ol>
   <li><p>Let <var>entries</var> be the result of <span>getting session history entries</span> for
   <var>navigable</var>.</p></li>

   <li><p>Return the <span data-x="list item">item</span> in <var>entries</var> that has the
   greatest <span data-x="she-step">step</span> less than or equal to <var>step</var>.</p></li>
  </ol>

  <div class="example" id="example-getting-the-target-history-entry">
   <p>To see why <span>getting the target history entry</span> returns the entry with the greatest
   <span data-x="she-step">step</span> less than or equal to the input step, consider the following
   <span>Jake diagram</span>:</p>

   <!--https://domenic.github.io/jake-diagram-generator/#dG9wCiAgMC0yOiAvdCB8IGRvYzEKICAzOiAvdCNmb28gfCBkb2MxCmZyYW1lc1swXQogIDA6IC9pLTAtYQogIDEtMzogL2ktMC1i -->
   <table class="jake-diagram"><thead><tr><td></td><th class="step">0</th><th class="step">1</th><th class="step">2</th><th class="step">3</th></tr></thead><tbody><tr><th><code data-x="">top</code></th><td colspan="3" class="doc-0">/t</td><td colspan="1" class="doc-0">/t#foo</td></tr><tr><th><code data-x="">frames[0]</code></th><td colspan="1" class="doc-1">/i-0-a</td><td colspan="3" class="doc-2">/i-0-b</td></tr></tbody></table>

   <p>For the input step 1, the target history entry for the <code data-x="">top</code> navigable
   is the <code data-x="">/t</code> entry, whose <span data-x="she-step">step</span> is 0, while the target history entry for the <code data-x="">frames[0]</code> navigable is the <code
   data-x="">/i-0-b</code> entry, whose <span data-x="she-step">step</span> is 1:</p>

   <!--https://domenic.github.io/jake-diagram-generator/#IWN1cnJlbnQgPSAxCnRvcAogIDAtMjogL3QgfCBkb2MxCiAgMzogL3QjZm9vIHwgZG9jMQpmcmFtZXNbMF0KICAwOiAvaS0wLWEKICAxLTM6IC9pLTAtYg== -->
   <table class="jake-diagram"><thead><tr><td></td><th class="step">0</th><th class="step current">1</th><th class="step">2</th><th class="step">3</th></tr></thead><tbody><tr><th><code data-x="">top</code></th><td colspan="3" class="doc-0 current next-is-same-doc">/t</td><td colspan="1" class="doc-0 prev-is-same-doc">/t#foo</td></tr><tr><th><code data-x="">frames[0]</code></th><td colspan="1" class="doc-1">/i-0-a</td><td colspan="3" class="doc-2 current">/i-0-b</td></tr></tbody></table>

   <p>Similarly, given the input step 3 we get the <code data-x="">top</code> entry whose <span
   data-x="she-step">step</span> is 3, and the <code data-x="">frames[0]</code> entry whose <span
   data-x="she-step">step</span> is 1:</p>

   <!--https://domenic.github.io/jake-diagram-generator/#IWN1cnJlbnQgPSAzCnRvcAogIDAtMjogL3QgfCBkb2MxCiAgMzogL3QjZm9vIHwgZG9jMQpmcmFtZXNbMF0KICAwOiAvaS0wLWEKICAxLTM6IC9pLTAtYg== -->
   <table class="jake-diagram"><thead><tr><td></td><th class="step">0</th><th class="step">1</th><th class="step">2</th><th class="step current">3</th></tr></thead><tbody><tr><th><code data-x="">top</code></th><td colspan="3" class="doc-0 next-is-same-doc">/t</td><td colspan="1" class="doc-0 current prev-is-same-doc">/t#foo</td></tr><tr><th><code data-x="">frames[0]</code></th><td colspan="1" class="doc-1">/i-0-a</td><td colspan="3" class="doc-2 current">/i-0-b</td></tr></tbody></table>
  </div>

  <p>To <dfn data-x="getting all navigables that might experience a cross-document traversal">get
  all navigables that might experience a cross-document traversal</dfn> given a <span>traversable
  navigable</span> <var>traversable</var>, and a non-negative integer <var>targetStep</var>,
  perform the following steps. They return a <span>list</span> of <span
  data-x="navigable">navigables</span>.</p>

  <div class="note">
   <p>From <var>traversable</var>'s <span data-x="tn-session-history-traversal-queue">session
   history traversal queue</span>'s perspective, these documents are candidates for going
   cross-document during the traversal described by <var>targetStep</var>. They will not experience
   a cross-document traversal if the status code for their target document is HTTP 204 No Content.</p>

   <p>Note that if a given <span>navigable</span> might experience a cross-document traversal, this
   algorithm will return <span>navigable</span> but not its <span data-x="child navigable">child
   navigables</span>. Those would end up <span data-x="unload a document">unloaded</span>, not
   traversed.</p>
  </div>

  <ol>
   <li><p>Let <var>results</var> be an empty <span>list</span>.</p></li>

   <li>
    <p>Let <var>navigablesToCheck</var> be « <var>traversable</var> ».</p>

    <p class="note">This list is extended in the loop below.</p>
   </li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>navigable</var> of
    <var>navigablesToCheck</var>:</p>

    <ol>
     <li><p>Let <var>targetEntry</var> be the result of <span>getting the target history
     entry</span> given <var>navigable</var> and <var>targetStep</var>.</p></li>

     <li>
      <p>If <var>targetEntry</var>'s <span data-x="she-document">document</span> is not
      <var>navigable</var>'s <span data-x="nav-document">document</span> or
      <var>targetEntry</var>'s <span data-x="she-document-state">document state</span>'s <span
      data-x="document-state-reload-pending">reload pending</span> is true, then <span data-x="list
      append">append</span> <var>navigable</var> to <var>results</var>.</p>

      <p class="note">Although <var>navigable</var>'s <span
      data-x="nav-active-history-entry">active history entry</span> can change synchronously, the
      new entry will always have the same <code>Document</code>, so accessing
      <var>navigable</var>'s <span data-x="nav-document">document</span> is reliable.</p>
     </li>

     <li>
      <p>Otherwise, <span data-x="list extend">extend</span> <var>navigablesToCheck</var> with
      <var>navigable</var>'s <span data-x="child navigable">child navigables</span>.</p>

      <p class="note">Adding <span data-x="child navigable">child navigables</span> to
      <var>navigablesToCheck</var> means those navigables will also be checked by this loop. <span
      data-x="child navigable">Child navigables</span> are only checked if the
      <var>navigable</var>'s <span data-x="nav-document">active document</span> will not change as
      part of this traversal.</p>
     </li>
    </ol>
   </li>

   <li><p>Return <var>results</var>.</p></li>
  </ol>


  <h5>Updating the document</h5>

  <p>To <dfn>update document for history step application</dfn> given a <code>Document</code>
  <var>document</var>, a <span>session history entry</span> <var>entry</var>, a boolean
  <var>doNotReactivate</var>, integers <var>scriptHistoryLength</var> and
  <var>scriptHistoryIndex</var>, <code>NavigationType</code>-or-null <var>navigationType</var>, an
  optional <span>list</span> of <span data-x="session history entry">session history entries</span>
  <var>entriesForNavigationAPI</var>, and an optional <span>session history entry</span>
  <var>previousEntryForActivation</var>:</p>

  <ol>
   <li><p>Let <var>documentIsNew</var> be true if <var>document</var>'s <span>latest entry</span>
   is null; otherwise false.</p></li>

   <li><p>Let <var>documentsEntryChanged</var> be true if <var>document</var>'s <span>latest
   entry</span> is not <var>entry</var>; otherwise false.</p></li>

   <li><p>Set <var>document</var>'s <span data-x="doc-history">history object</span>'s <span
   data-x="concept-history-index">index</span> to <var>scriptHistoryIndex</var>.</p></li>

   <li><p>Set <var>document</var>'s <span data-x="doc-history">history object</span>'s <span
   data-x="concept-history-length">length</span> to <var>scriptHistoryLength</var>.</p></li>

   <li><p>Let <var>navigation</var> be <var>history</var>'s <span>relevant global object</span>'s
   <span data-x="window-navigation-api">navigation API</span>.</p></li>

   <li>
    <p>If <var>documentsEntryChanged</var> is true, then:</p>

    <ol>
     <li><p>Let <var>oldURL</var> be <var>document</var>'s <span>latest entry</span>'s <span
     data-x="she-url">URL</span>.</p></li>

     <li><p>Set <var>document</var>'s <span>latest entry</span> to <var>entry</var>.</p></li>

     <li><p><span>Restore the history object state</span> given <var>document</var> and
     <var>entry</var>.</p></li>

     <li>
      <p>If <var>documentIsNew</var> is false, then:</p>

      <ol>
       <li><p><span>Assert</span>: <var>navigationType</var> is not null.</p></li>

       <li><p><span>Update the navigation API entries for a same-document navigation</span> given
       <var>navigation</var>, <var>entry</var>, and <var>navigationType</var>.</p></li>

       <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
       data-x="event-popstate">popstate</code> at <var>document</var>'s <span>relevant global
       object</span>, using <code>PopStateEvent</code>, with the <code
       data-x="dom-PopStateEvent-state">state</code> attribute initialized to <var>document</var>'s
       <span data-x="doc-history">history object</span>'s <span
       data-x="concept-history-state">state</span> and <code
       data-x="dom-PopStateEvent-hasUAVisualTransition">hasUAVisualTransition</code> initialized to
       true if a visual transition, to display a cached rendered state of the <span>latest
       entry</span>, was done by the user agent.</p>

       <li><p><span>Restore persisted state</span> given <var>entry</var>.</p></li>

       <li><p>If <var>oldURL</var>'s <span data-x="concept-url-fragment">fragment</span> is not
       equal to <var>entry</var>'s <span data-x="she-url">URL</span>'s <span
       data-x="concept-url-fragment">fragment</span>, then <span>queue a global task</span> on the
       <span>DOM manipulation task source</span> given <var>document</var>'s <span>relevant global
       object</span> to <span data-x="concept-event-fire">fire an event</span> named <code
       data-x="event-hashchange">hashchange</code> at <var>document</var>'s <span>relevant global
       object</span>, using <code>HashChangeEvent</code>, with the <code
       data-x="dom-HashChangeEvent-oldURL">oldURL</code> attribute initialized to the <span
       data-x="concept-url-serializer">serialization</span> of <var>oldURL</var> and the <code
       data-x="dom-HashChangeEvent-newURL">newURL</code> attribute initialized to the <span
       data-x="concept-url-serializer">serialization</span> of <var>entry</var>'s <span
       data-x="she-url">URL</span>.</p></li>
      </ol>
     </li>

     <li>
      <p>Otherwise:</p>

      <ol>
       <li><p><span>Assert</span>: <var>entriesForNavigationAPI</var> is given.</p></li>

       <li><p><span>Restore persisted state</span> given <var>entry</var>.</p></li>

       <li><p><span>Initialize the navigation API entries for a new document</span> given
       <var>navigation</var>, <var>entriesForNavigationAPI</var>, and <var>entry</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li>
    <p>If all the following are true:</p>

    <ul>
     <li><p><var>previousEntryForActivation</var> is given;</p></li>

     <li><p><var>navigationType</var> is non-null; and</p></li>

     <li><p><var>navigationType</var> is "<code data-x="dom-NavigationType-reload">reload</code>" or
     <var>previousEntryForActivation</var>'s <span data-x="she-document">document</span> is not
     <var>document</var>,</p></li>
    </ul>

    <p>then:</p>

    <ol>
     <li><p>If <var>navigation</var>'s <span
     data-x="navigation-activation">activation</span> is null, then set <var>navigation</var>'s
     <span data-x="navigation-activation">activation</span> to a new
     <code>NavigationActivation</code> object in <var>navigation</var>'s
     <span data-x="concept-relevant-realm">relevant realm</span>.</p></li>

     <li><p>Let <var>previousEntryIndex</var> be the result of
     <span>getting the navigation API entry index</span> of <var>previousEntryForActivation</var>
     within <var>navigation</var>.</p></li>

     <li><p>If <var>previousEntryIndex</var> is non-negative, then set <var>activation</var>'s
     <span data-x="nav-activation-old-entry">old entry</span> to <var>navigation</var>'s <span
     data-x="navigation-entry-list">entry list</span>[<var>previousEntryIndex</var>].</p></li>

     <li>
      <p>Otherwise, if all the following are true:</p>

      <ul>
       <li><p><var>navigationType</var> is "<code
       data-x="dom-NavigationType-replace">replace</code>";</p></li>

       <li><p><var>previousEntryForActivation</var>'s
       <span data-x="she-document-state">document state</span>'s <span
       data-x="document-state-origin">origin</span> is <span>same origin</span> with
       <var>document</var>'s <span data-x="concept-document-origin">origin</span>; and</p></li>

       <li><p><var>previousEntryForActivation</var>'s <span data-x="she-document">document</span>'s
       <span data-x="is initial about:blank">initial <code>about:blank</code></span> is
       false,</p></li>
      </ul>

      <p>then set <var>activation</var>'s <span data-x="nav-activation-old-entry">old entry</span>
      to a new <code>NavigationHistoryEntry</code> in <var>navigation</var>'s <span
      data-x="concept-relevant-realm">relevant realm</span>, whose <span
      data-x="nhe-she">session history entry</span> is
      <var>previousEntryForActivation</var>.</p>
     </li>

     <li><p>Set <var>activation</var>'s <span data-x="nav-activation-new-entry">new entry</span> to
     <var>navigation</var>'s <span data-x="navigation-current-entry">current entry</span>.</p></li>

     <li><p>Set <var>activation</var>'s <span
     data-x="nav-activation-navigation-type">navigation type</span> to
     <var>navigationType</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>If <var>documentIsNew</var> is true, then:</p>

    <ol>
     <li><p><span>Assert</span>: <var>document</var>'s <span
     data-x="concept-document-navigation-id">during-loading navigation ID for WebDriver BiDi</span>
     is not null.</p></li>

     <li><p>Invoke <span>WebDriver BiDi navigation committed</span> with <var>navigable</var> and a
     new <span>WebDriver BiDi navigation status</span> whose <span
     data-x="navigation-status-id">id</span> is <var>document</var>'s <span
     data-x="concept-document-navigation-id">during-loading navigation ID for WebDriver BiDi</span>,
     <span data-x="navigation-status-status">status</span> is "<code
     data-x="navigation-status-committed">committed</code>", and <span
     data-x="navigation-status-url">url</span> is <var>document</var>'s <span
     data-x="concept-document-url">URL</span></p></li>

     <li><p><span>Try to scroll to the fragment</span> for <var>document</var>.</p></li>

     <li><p>At this point <dfn>scripts may run for the newly-created document</dfn>
     <var>document</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise, if <var>documentsEntryChanged</var> is false and <var>doNotReactivate</var> is
    false, then:</p>

    <ol>
     <li><p><span>Assert</span>: <var>entriesForNavigationAPI</var> is given.</p></li>

     <li><p><span data-x="reactivate a document">Reactivate</span> <var>document</var> given
     <var>entry</var> and <var>entriesForNavigationAPI</var>.</p>
    </ol>

    <p class="note"><var>documentsEntryChanged</var> can be false for one of two reasons: either we
    are restoring from <a href="#note-bfcache">bfcache</a>, or we are asynchronously finishing up a
    synchronous navigation which already synchronously set <var>document</var>'s <span>latest
    entry</span>. The <var>doNotReactivate</var> argument distinguishes between these two
    cases.</p>
   </li>
  </ol>

  <p>To <dfn>restore the history object state</dfn> given <code>Document</code>
  <var>document</var> and <span>session history entry</span> <var>entry</var>:</p>

  <ol>
   <li><p>Let <var>targetRealm</var> be <var>document</var>'s <span
   data-x="concept-relevant-realm">relevant realm</span>.</p></li>

   <li><p>Let <var>state</var> be <span>StructuredDeserialize</span>(<var>entry</var>'s <span
   data-x="she-classic-history-api-state">classic history API state</span>, <var>targetRealm</var>).
   If this throws an exception, catch it and let <var>state</var> be null.</p></li>

   <li><p>Set <var>document</var>'s <span data-x="doc-history">history object</span>'s <span
   data-x="concept-history-state">state</span> to <var>state</var>.</p></li>
  </ol>

  <p>To <dfn>make active</dfn> a <code>Document</code> <var>document</var>:</p>

  <ol>
   <li><p>Let <var>window</var> be <var>document</var>'s <span>relevant global
   object</span>.</p></li>

   <li><p>Set <var>document</var>'s <span data-x="concept-document-bc">browsing context</span>'s
   <code>WindowProxy</code>'s <span data-x="concept-windowproxy-window">[[Window]]</span> internal
   slot value to <var>window</var>.</p></li>

   <li><p>Set <var>document</var>'s <span>visibility state</span> to <var>document</var>'s
   <span>node navigable</span>'s <span data-x="nav-traversable">traversable navigable</span>'s
   <span>system visibility state</span>.</p></li>

   <li><p><span data-x="queue a performance entry">Queue</span> a new
   <code>VisibilityStateEntry</code> whose <span data-x="VisibilityStateEntry-state">visibility
   state</span> is <var>document</var>'s <span>visibility state</span> and whose <span
   data-x="VisibilityStateEntry-timestamp">timestamp</span> is zero.</p></li>

   <li><p>Set <var>window</var>'s <span>relevant settings object</span>'s <span
   data-x="concept-environment-execution-ready-flag">execution ready flag</span>.</p></li>
  </ol>

  <p>To <dfn data-x="reactivate a document" export for="Document">reactivate</dfn> a
  <code>Document</code> <var>document</var> given a <span>session history entry</span>
  <var>reactivatedEntry</var> and a <span>list</span> of <span data-x="session history
  entry">session history entries</span> <var>entriesForNavigationAPI</var>:</p>

  <p class="note">This algorithm updates <var>document</var> after it has come out of <a
  href="#note-bfcache">bfcache</a>, i.e., after it has been made <span>fully active</span>
  again. Other specifications that want to watch for this change to the <span>fully
  active</span> state are encouraged to add steps into this algorithm, so that the ordering
  of events that happen in effect of the change is clear.</p>

  <ol>
   <li id="history-autocomplete"><p><span data-x="list iterate">For each</span>
   <var>formControl</var> of form controls in <var>document</var> with an <span>autofill field
   name</span> of "<code data-x="attr-fe-autocomplete-off">off</code>", invoke the <span
   data-x="concept-form-reset-control">reset algorithm</span> for <var>formControl</var>.</p></li>

   <li>
    <p>If <var>document</var>'s <span>suspended timer handles</span> is not <span data-x="list
    empty">empty</span>:</p>

    <ol>
     <li><p><span>Assert</span>: <var>document</var>'s <span>suspension time</span> is not
     zero.</p></li>

     <li><p>Let <var>suspendDuration</var> be the <span>current high resolution time</span> minus
     <var>document</var>'s <span>suspension time</span>.</p></li>

     <li><p>Let <var>activeTimers</var> be <var>document</var>'s <span>relevant global
     object</span>'s <span>map of active timers</span>.</p></li>

     <li><p>For each <var>handle</var> in <var>document</var>'s <span>suspended timer
     handles</span>, if <var>activeTimers</var>[<var>handle</var>] <span data-x="map
     exists">exists</span>, then increase <var>activeTimers</var>[<var>handle</var>] by
     <var>suspendDuration</var>.</p></li>
    </ol>
   </li>

   <li><p><span>Update the navigation API entries for reactivation</span> given
   <var>document</var>'s <span>relevant global object</span>'s <span
   data-x="window-navigation-api">navigation API</span>, <var>entriesForNavigationAPI</var>, and
   <var>reactivatedEntry</var>.</p></li>

   <li>
    <p>If <var>document</var>'s <span>current document readiness</span> is "<code
    data-x="">complete</code>", and <var>document</var>'s <span>page showing</span> is false:</p>

    <ol>
     <li><p>Set <var>document</var>'s <span>page showing</span> to true.</p></li>

     <li><p>Set <var>document</var>'s <span>has been revealed</span> to false.</p>

     <li><p><span>Update the visibility state</span> of <var>document</var> to "<code
     data-x="">visible</code>".</p></li>

     <li><p><span>Fire a page transition event</span> named <code
     data-x="event-pageshow">pageshow</code> at <var>document</var>'s <span>relevant global
     object</span> with true.</p></li>
    </ol>

    <!-- an interesting thing to test would be to traverse back during onload, before the first
    pageshow has fired, and then to traverse forward again, and see if we get _two_ pageshows. If
    so, it indicates that browsers don't have a "page showing" flag like this and that the history
    traversal task source has a higher priority than the DOM manipulation task source. -->
   </li>
  </ol>

  <p>To <dfn>try to scroll to the fragment</dfn> for a <code>Document</code> <var>document</var>,
  perform the following steps <span>in parallel</span>:</p>

  <ol>
   <li><p>Wait for an <span>implementation-defined</span> amount of time. (This is intended to allow
   the user agent to optimize the user experience in the face of performance concerns.)</p></li>

   <li>
    <p><span>Queue a global task</span> on the <span>navigation and traversal task source</span>
    given <var>document</var>'s <span>relevant global object</span> to run these steps:</p>

    <ol>
     <li><p>If <var>document</var> has no parser, or its parser has <span data-x="stop
     parsing">stopped parsing</span>, or the user agent has reason to believe the user is no longer
     interested in scrolling to the <span data-x="concept-url-fragment">fragment</span>, then abort
     these steps.</p></li>

     <li><p><span>Scroll to the fragment</span> given <var>document</var>.</p></li>

     <li><p>If <var>document</var>'s <span>indicated part</span> is still null, then <span>try to
     scroll to the fragment</span> for <var>document</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>To <dfn export>make document unsalvageable</dfn>, given a <code>Document</code>
  <var>document</var> and a string <var>reason</var>:</p>

  <ol>
   <li><p>Let <var>details</var> be a new <span
   data-x="nrr-details-struct">not restored reason details</span> whose <span
   data-x="nrr-details-reason">reason</span> is <var>reason</var>.</p></li>

   <li><p><span data-x="set append">Append</span> <var>details</var> to <var>document</var>'s
   <span data-x="concept-document-bfcache-blocking-details">bfcache blocking details</span>.</p></li>

   <li><p>Set <var>document</var>'s <i data-x="concept-document-salvageable">salvageable</i>
   state to false.</p></li>
  </ol>

  <p>To <dfn>build not restored reasons for document state</dfn> given <code>Document</code>
  <var>document</var>:</p>

  <ol>
   <li><p>Let <var>notRestoredReasonsForDocument</var> be a new <span
   data-x="nrr-struct">not restored reasons</span>.</p></li>

   <li><p>Set <var>notRestoredReasonsForDocument</var>'s <span data-x="nrr-url">URL</span> to
   <var>document</var>'s <span data-x="concept-document-url">URL</span>.</p></li>

   <li><p>Let <var>container</var> be <var>document</var>'s <span>node navigable</span>'s <span
   data-x="nav-container">container</span>.</p></li>

   <li>
    <p>If <var>container</var> is an <code>iframe</code> element:</p>

    <ol>
     <li><p>Let <var>src</var> be the empty string.</p></li>

     <li>
      <p>If <var>container</var> has a <code data-x="dom-iframe-src">src</code> attribute:</p>

      <ol>
       <li><p>Let <var>src</var> be the result of <span>encoding-parsing-and-serializing a
       URL</span> given <var>container</var>'s <code data-x="attr-iframe-src">src</code> attribute's
       value, relative to <var>container</var>'s <span>node document</span>.</p></li>

       <li><p>If <var>src</var> is failure, then set <var>src</var> to <var>container</var>'s <code
       data-x="attr-iframe-src">src</code> attribute's value.</p></li>
      </ol>
     </li>

     <li><p>Set <var>notRestoredReasonsForDocument</var>'s <span data-x="nrr-src">src</span> to
     <var>src</var>.</p></li>

     <li><p>Set <var>notRestoredReasonsForDocument</var>'s <span data-x="nrr-id">id</span> to
     <var>container</var>'s <code data-x="attr-id">id</code> attribute's value, or the empty string
     if it has no such attribute.</p></li>

     <li><p>Set <var>notRestoredReasonsForDocument</var>'s <span data-x="nrr-name">name</span> to
     <var>container</var>'s <code data-x="attr-iframe-name">name</code> attribute's value, or the
     empty string if it has no such attribute.</p></li>
    </ol>
   </li>

   <li><p>Set <var>notRestoredReasonsForDocument</var>'s <span data-x="nrr-reasons">reasons</span>
   to a <span data-x="list clone">clone</span> of <var>document</var>'s <span
   data-x="concept-document-bfcache-blocking-details">bfcache blocking details</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>navigable</var> of <var>document</var>'s
    <span data-x="document-tree child navigables">document-tree child navigables</span>:</p>

    <ol>
     <li><p>Let <var>childDocument</var> be <var>navigable</var>'s <span
     data-x="nav-document">active document</span>.</p></li>

     <li><p><span>Build not restored reasons for document state</span> given
     <var>childDocument</var>.</p></li>

     <li><p><span data-x="list append">Append</span> <var>childDocument</var>'s <span
     data-x="concept-document-nrr">not restored reasons</span> to
     <var>notRestoredReasonsForDocument</var>'s <span data-x="nrr-children">children</span>.</p></li>
    </ol>
   </li>

   <li><p>Set <var>document</var>'s <span>node navigable</span>'s <span
   data-x="nav-active-history-entry">active session history entry</span>'s <span
   data-x="she-document-state">document state</span>'s <span
   data-x="document-state-not-restored-reasons">not restored reasons</span> to
   <var>notRestoredReasonsForDocument</var>.</p></li>
  </ol>

 <p>To <dfn>build not restored reasons for a top-level traversable and its descendants</dfn>
 given <span>top-level traversable</span> <var>topLevelTraversable</var>:
 <ol>
  <li><p><span>Build not restored reasons for document state</span> given
  <var>topLevelTraversable</var>'s <span data-x="nav-document">active document</span>.</p></li>

  <li><p>Let <var>crossOriginDescendants</var> be an empty <span>list</span>.</p></li>

  <li>
   <p><span data-x="list iterate">For each</span> <var>childNavigable</var> of
   <var>topLevelTraversable</var>'s <span data-x="nav-document">active document</span>'s
   <span>descendant navigables</span>:</p>
   <ol>
    <li><p>If <var>childNavigable</var>'s <span data-x="nav-document">active document</span>'s
    <span data-x="concept-document-origin">origin</span> is not <span>same origin</span> with
    <var>topLevelTraversable</var>'s <span data-x="nav-document">active document</span>'s <span
    data-x="concept-document-origin">origin</span>, then <span data-x="list append">append</span>
    <var>childNavigable</var> to <var>crossOriginDescendants</var>.</p></li>
   </ol>
  </li>

  <li><p>Let <var>crossOriginDescendantsPreventsBfcache</var> be false.</p></li>

  <li>
   <p><span data-x="list iterate">For each</span> <var>crossOriginNavigable</var> of
   <var>crossOriginDescendants</var>:</p>

   <ol>
    <li><p>Let <var>reasonsForCrossOriginChild</var> be <var>crossOriginNavigable</var>'s <span
    data-x="nav-document">active document</span>'s <span>document state</span>'s <span
    data-x="document-state-not-restored-reasons">not restored reasons</span>.</p></li>

    <li><p>If <var>reasonsForCrossOriginChild</var>'s <span data-x="nrr-reasons">reasons</span>
    is not empty, set <var>crossOriginDescendantsPreventsBfcache</var> to true.</p></li>

    <li><p>Set <var>reasonsForCrossOriginChild</var>'s <span data-x="nrr-url">URL</span> to
    null.</p></li>

    <li><p>Set <var>reasonsForCrossOriginChild</var>'s <span data-x="nrr-reasons">reasons</span>
    to null.</p></li>

    <li><p>Set <var>reasonsForCrossOriginChild</var>'s <span data-x="nrr-children">children</span>
    to null.</p></li>
   </ol>
  </li>

  <li><p>If <var>crossOriginDescendantsPreventsBfcache</var> is true, <span>make document
  unsalvageable</span> given <var>topLevelTraversable</var>'s <span
  data-x="nav-document">active document</span> and "<code
  data-x="blocking-masked">masked</code>".</p></li>
 </ol>

  <h5>Revealing the document</h5>

  <p>A <code>Document</code> has a boolean <dfn>has been revealed</dfn>, initially false. It is used
  to ensure that the <code data-x="event-pagereveal">pagereveal</code> event is fired once for each
  activation of the <code>Document</code> (once when it's rendered initially, and once for each
  <span data-x="reactivate a document">reactivation</span>).</p>

  <p>To <dfn>reveal</dfn> a <code>Document</code> <var>document</var>:</p>

  <ol>
   <li><p>If <var>document</var>'s <span>has been revealed</span> is true, then return.</p></li>

   <li><p>Set <var>document</var>'s <span>has been revealed</span> to true.</p></li>

   <li><p>Let <var>transition</var> be the result of
   <span>resolving inbound cross-document view-transition</span> for <var>document</var>.</p></li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named
   <code data-x="event-pagereveal">pagereveal</code> at <var>document</var>'s
   <span>relevant global object</span>, using <code>PageRevealEvent</code> with its <code
   data-x="dom-PageRevealEvent-viewTransition">viewTransition</code> set to
   <var>transition</var>.</p></li>

   <li>
    <p>If <var>transition</var> is not null, then:</p>

    <ol>
     <li><p><span>Prepare to run script</span> given <var>document</var>'s
     <span>relevant settings object</span>.</p></li>

     <li><p><span data-x="activate view transition">Activate</span> <var>transition</var>.</p></li>

     <li><p><span>Clean up after running script</span> given <var>document</var>'s
     <span>relevant settings object</span>.</p></li>
    </ol>

    <p class="note">Activating a view transition might resolve/reject promises, so by wrapping the
    activation with prepare/cleanup we ensure those promises are handled before the next rendering
    step.</p>
   </li>
  </ol>

  <p class="note">Though <code data-x="event-pagereveal">pagereveal</code> is guaranteed to be fired
  during the first <span>update the rendering</span> step that displays an up-to-date version of the
  page, user agents are free to display a cached frame of the page before firing it. This prevents
  the presence of a <code data-x="event-pagereveal">pagereveal</code> handler from delaying the
  presentation of such cached frame.</p>


  <h5>Scrolling to a fragment</h5>

  <p>To <dfn id="scroll-to-the-fragment-identifier">scroll to the fragment</dfn> given a
  <code>Document</code> <var>document</var>:</p>

  <ol>
   <li><p>If <var>document</var>'s <span>indicated part</span> is null, then set
   <var>document</var>'s <span>target element</span> to null.</p></li>

   <li>
    <p>Otherwise, if <var>document</var>'s <span>indicated part</span> is <span>top of the
    document</span>, then:</p>

    <ol>
     <li><p>Set <var>document</var>'s <span>target element</span> to null.</p></li>

     <li><p><span>Scroll to the beginning of the document</span> for <var>document</var>.
     <ref>CSSOMVIEW</ref></p></li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p><span>Assert</span>: <var>document</var>'s <span>indicated part</span> is an
     element.</p></li>

     <li><p>Let <var>target</var> be <var>document</var>'s <span>indicated part</span>.</p></li>

     <li><p>Set <var>document</var>'s <span>target element</span> to <var>target</var>.</p></li>

     <li><p>Run the <span>ancestor details revealing algorithm</span> on <var>target</var>.</p></li>

     <li><p>Run the <span>ancestor hidden-until-found revealing algorithm</span> on
     <var>target</var>.</p></li>

     <li><p><span data-x="scroll a target into view">Scroll <var>target</var> into view</span>,
     with <var>behavior</var> set to "auto", <var>block</var> set to "start", and <var>inline</var>
     set to "nearest". <ref>CSSOMVIEW</ref></p></li>

     <li><p>Run the <span>focusing steps</span> for <var>target</var>, with the
     <code>Document</code>'s <span>viewport</span> as the <var>fallback target</var>.</p></li>

     <li><p>Move the <span>sequential focus navigation starting point</span> to
     <var>target</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>A <code>Document</code>'s <dfn id="the-indicated-part-of-the-document">indicated part</dfn> is
  the one that its <span data-x="concept-document-url">URL</span>'s <span
  data-x="concept-url-fragment">fragment</span> identifies, or null if the fragment does not
  identify anything. The semantics of the <span data-x="concept-url-fragment">fragment</span> in
  terms of mapping it to a node is defined by the specification that defines the <span>MIME
  type</span> used by the <code>Document</code> (for example, the processing of <span
  data-x="concept-url-fragment">fragments</span> for <span data-x="XML MIME type">XML MIME
  types</span> is the responsibility of RFC7303). <ref>RFC7303</ref></p>

  <p>There is also a <dfn>target element</dfn> for each <code>Document</code>, which is used in
  defining the <code data-x="selector-target">:target</code> pseudo-class and is updated by the
  above algorithm. It is initially null.</p>

  <p>For an <span data-x="HTML documents">HTML document</span> <var>document</var>, its
  <span>indicated part</span> is the result
  of <span data-x="select the indicated part">selecting the indicated part</span> given
  <var>document</var> and <var>document</var>'s <span
  data-x="concept-document-url">URL</span>.</p>

  <p>To <dfn>select the indicated part</dfn> given a <code>Document</code> <var>document</var> and a
  <span>URL</span> <var>url</var>:</p>

  <ol>
   <li><p>If <var>document</var>'s <span
   data-x="concept-document-url">URL</span> does not <span data-x="concept-url-equals">equal</span>
   <var>url</var> with <i data-x="url equals exclude fragments">exclude fragments</i> set to
   true, then return null.</p></li>

   <li><p>Let <var>fragment</var> be <var>url</var>'s <span
   data-x="concept-url-fragment">fragment</span>.</p></li>

   <li><p>If <var>fragment</var> is the empty string, then return the special value <dfn>top of the
   document</dfn>.</p></li>

   <li><p>Let <var>potentialIndicatedElement</var> be the result of <span data-x="find a potential
   indicated element">finding a potential indicated element</span> given <var>document</var> and
   <var>fragment</var>.</p></li>

   <li><p>If <var>potentialIndicatedElement</var> is not null, then return
   <var>potentialIndicatedElement</var>.</p></li>

   <li><p>Let <var>fragmentBytes</var> be the result of <span
   data-x="percent-decode">percent-decoding</span> <var>fragment</var>.</p></li>

   <li><p>Let <var>decodedFragment</var> be the result of running <span>UTF-8 decode without
   BOM</span> on <var>fragmentBytes</var>.</p></li>

   <li><p>Set <var>potentialIndicatedElement</var> to the result of <span data-x="find a potential
   indicated element">finding a potential indicated element</span> given <var>document</var> and
   <var>decodedFragment</var>.</p></li>

   <li><p>If <var>potentialIndicatedElement</var> is not null, then return
   <var>potentialIndicatedElement</var>.</p></li>

   <li><p>If <var>decodedFragment</var> is an <span>ASCII case-insensitive</span> match for the
   string <code data-x="">top</code>, then return the <span>top of the document</span>.</p></li>

   <li><p>Return null.</p></li>
  </ol>

  <p>To <dfn>find a potential indicated element</dfn> given a <code>Document</code>
  <var>document</var> and a string <var>fragment</var>, run these steps:</p>

  <ol>
   <li><p>If there is an element <span data-x="in a document tree">in the document tree</span>
   whose <span>root</span> is <var>document</var> and that has an <span
   data-x="concept-id">ID</span> equal to <var>fragment</var>, then return the first such element
   in <span>tree order</span>.</p></li>

   <li><p>If there is an <code>a</code> element <span data-x="in a document tree">in the document
   tree</span> whose <span>root</span> is <var>document</var> that has a <code
   data-x="attr-a-name">name</code> attribute whose value is equal to <var>fragment</var>, then
   return the first such element in <span>tree order</span>.</p></li>

   <li><p>Return null.</p></li>
  </ol>


  <h5 id="persisted-user-state-restoration">Persisted history entry state</h5>

  <p>To <dfn>save persisted state</dfn> to a <span>session history entry</span>
  <var>entry</var>:</p>

  <ol>
   <li><p>Set the <span data-x="she-scroll-position">scroll position data</span> of <var>entry</var>
   to contain the scroll positions for all of <var>entry</var>'s <span
   data-x="she-document">document</span>'s <span>restorable scrollable regions</span>.</p></li>

   <li><p>Optionally, update <var>entry</var>'s <span data-x="she-other">persisted user state</span>
   to reflect any state that the user agent wishes to persist, such as the values of form
   fields.</p></li>
  </ol>

  <p id="restore-persisted-user-state">To <dfn>restore persisted state</dfn> from a <span>session
  history entry</span> <var>entry</var>:</p>

  <ol>
   <li>
    <p>If <var>entry</var>'s <span data-x="she-scroll-restoration-mode">scroll restoration
    mode</span> is "<code data-x="dom-ScrollRestoration-auto">auto</code>", and <var>entry</var>'s
    <span data-x="she-document">document</span>'s <span>relevant global object</span>'s <span
    data-x="window-navigation-api">navigation API</span>'s <span>suppress normal scroll restoration
    during ongoing navigation</span> is false, then <span>restore scroll position data</span> given
    <var>entry</var>.</p>

    <p class="note">The user agent not restoring scroll positions does not imply that scroll
    positions will be left at any particular value (e.g., (0,0)). The actual scroll position depends
    on the navigation type and the user agent's particular caching strategy. So web applications
    cannot assume any particular scroll position but rather are urged to set it to what they want it
    to be.</p>

    <p class="note">If <span>suppress normal scroll restoration during ongoing navigation</span> is
    true, then <span data-x="restore scroll position data">restoring scroll position data</span>
    might still happen at a later point, as part of <span
    data-x="NavigateEvent-finish">finishing</span> the relevant <code>NavigateEvent</code>, or via a
    <code data-x="dom-NavigateEvent-scroll">navigateEvent.scroll()</code> method call.</p>
   </li>

   <li>
    <p>Optionally, update other aspects of <var>entry</var>'s <span
    data-x="she-document">document</span> and its rendering, for instance values of form fields,
    that the user agent had previously recorded in <var>entry</var>'s <span
    data-x="she-other">persisted user state</span>.</p>

    <p class="note">This can even include updating the <code data-x="attr-dir">dir</code> attribute
    of <code>textarea</code> elements or <code>input</code> elements whose <code
    data-x="attr-input-type">type</code> attribute is in the <span
    data-x="attr-input-type-text">Text</span>, <span data-x="attr-input-type-search">Search</span>,
    <span data-x="attr-input-type-tel">Telephone</span>, <span
    data-x="attr-input-type-url">URL</span>, or <span data-x="attr-input-type-email">Email</span>
    state, if the persisted state includes the directionality of user input in such controls.</p>

    <p class="note">Restoring the value of form controls as part of this process does not fire any
    <code data-x="event-input">input</code> or <code data-x="event-change">change</code> events, but
    can trigger the <code data-x="">formStateRestoreCallback</code> of <span data-x="form-associated
    custom element">form-associated custom elements</span>.</p>
   </li>
  </ol>

  <hr>

  <p>Each <code>Document</code> has a boolean <dfn>has been scrolled by the user</dfn>, initially
  false. If the user scrolls the document, the user agent must set that document's <span>has been
  scrolled by the user</span> to true.</p>

  <p>The <dfn>restorable scrollable regions</dfn> of a <code>Document</code> <var>document</var>
  are <var>document</var>'s <span>viewport</span>, and all of <var>document</var>'s scrollable
  regions excepting any <span data-x="navigable container">navigable containers</span>.</p>

  <p class="note"><span>Child navigable</span> scroll restoration is handled as part of state
  restoration for the <span>session history entry</span> for those <span
  data-x="navigable">navigables</span>' <code>Document</code>s.</p>

  <p>To <dfn>restore scroll position data</dfn> given a <span>session history entry</span>
  <var>entry</var>:</p>

  <ol>
   <li><p>Let <var>document</var> be <var>entry</var>'s <span
   data-x="she-document">document</span>.</p></li>

   <li><p>If <var>document</var>'s <span>has been scrolled by the user</span> is true, then the
   user agent should return.</p></li>

   <li>
    <p>The user agent should attempt to use <var>entry</var>'s <span
    data-x="she-scroll-position">scroll position data</span> to restore the scroll positions of
    <var>entry</var>'s <span data-x="she-document">document</span>'s <span>restorable scrollable
    regions</span>. The user agent may continue to attempt to do so periodically, until
    <var>document</var>'s <span>has been scrolled by the user</span> becomes true.</p>

    <p class="note">This is formulated as an <em>attempt</em>, which is potentially repeated until
    success or until the user scrolls, due to the fact that relevant content indicated by the <span
    data-x="she-scroll-position">scroll position data</span> might take some time to load from the
    network.</p>

    <p class="note">Scroll restoration might be affected by scroll anchoring.
    <ref>CSSSCROLLANCHORING</ref></p>
   </li>
  </ol>


  <h3 split-filename="document-lifecycle">Document lifecycle</h3>

  <h4>Shared document creation infrastructure</h4>

  <p>When loading a document using one of the below algorithms, we use the
  following steps to <!--en-GB--><dfn id="initialise-the-document-object"
  data-x="create-the-document-object">create and initialize a <code>Document</code> object</dfn>,
  given a <span data-x="concept-document-type">type</span> <var>type</var>, <span
  data-x="concept-document-content-type">content type</span> <var>contentType</var>, and
  <span>navigation params</span> <var>navigationParams</var>:</p>

  <p class="note"><code>Document</code> objects are also created when <span>creating a new browsing
  context and document</span>; such <span data-x="is initial about:blank">initial
  <code>about:blank</code></span> <code>Document</code> are never created by this algorithm. Also,
  <span data-x="concept-document-bc">browsing context</span>-less <code>Document</code> objects can
  be created via various APIs, such as <code
  data-x="dom-DOMImplementation-createHTMLDocument">document.implementation.createHTMLDocument()</code>.</p>

  <ol>
   <li>
    <p>Let <var>browsingContext</var> be the result of <span
    data-x="obtain-browsing-context-navigation">obtaining a browsing context to use for a
    navigation response</span> given <var>navigationParams</var>.</p>

    <p class="note">This can result in a <a
    href="#browsing-context-group-switches-due-to-cross-origin-opener-policy">browsing context
    group switch</a>, in which case <var>browsingContext</var> will be a <span data-x="creating a
    new browsing context and document">newly-created</span> <span>browsing context</span> instead
    of being <var>navigationParams</var>'s <span
    data-x="navigation-params-navigable">navigable</span>'s <span data-x="nav-bc">active browsing
    context</span>. In such a case, the created <code>Window</code>, <code>Document</code>, and
    <span>agent</span> will not end up being used; because the created <code>Document</code>'s
    <span data-x="concept-document-origin">origin</span> is <span
    data-x="concept-origin-opaque">opaque</span>, we will end up creating a new <span>agent</span>
    and <code>Window</code> <a href="#create-new-agent-and-window">later in this algorithm</a> to
    go along with the new <code>Document</code>.</p>
   </li>

   <li>
    <p>Let <var>permissionsPolicy</var> be the result of <span>creating a permissions policy from a
    response</span> given <var>navigationParams</var>'s <span
    data-x="navigation-params-navigable">navigable</span>'s <span
    data-x="nav-container">container</span>, <var>navigationParams</var>'s <span
    data-x="navigation-params-origin">origin</span>, and <var>navigationParams</var>'s <span
    data-x="navigation-params-response">response</span>. <ref>PERMISSIONSPOLICY</ref></p>

    <div class="note">
     <p>The <span>creating a permissions policy from a response</span> algorithm makes use of the
     passed <span>origin</span>. If <code data-x="dom-document-domain">document.domain</code> has
     been used for <var>navigationParams</var>'s <span
     data-x="navigation-params-navigable">navigable</span>'s <span
     data-x="nav-container-document">container document</span>, then its <span
     data-x="concept-document-origin">origin</span> cannot be <span>same origin-domain</span> with
     the passed origin, because these steps run before the <var>document</var> is created, so it
     cannot itself yet have used <code data-x="dom-document-domain">document.domain</code>. Note
     that this means that Permissions Policy checks are less permissive compared to doing a
     <span>same origin</span> check instead.</p>

     <p>See below for some examples of this in action.</p>
    </div>
   </li>

   <li><p>Let <var>creationURL</var> be <var>navigationParams</var>'s <span
   data-x="navigation-params-response">response</span>'s <span
   data-x="concept-response-url">URL</span>.</p></li>

   <li><p>If <var>navigationParams</var>'s <span data-x="navigation-params-request">request</span>
   is non-null, then set <var>creationURL</var> to <var>navigationParams</var>'s <span
   data-x="navigation-params-request">request</span>'s <span
   data-x="concept-request-current-url">current URL</span>.</p></li>

   <li><p>Let <var>window</var> be null.</p></li>

   <li>
    <p>If <var>browsingContext</var>'s <span>active document</span>'s <span>is initial
    <code>about:blank</code></span> is true, and <var>browsingContext</var>'s <span>active
    document</span>'s <span data-x="concept-document-origin">origin</span> is <span>same
    origin-domain</span> with <var>navigationParams</var>'s <span
    data-x="navigation-params-origin">origin</span>, then set <var>window</var> to
    <var>browsingContext</var>'s <span>active window</span>.</p>

    <p class="note">This means that both the <span data-x="is initial about:blank">initial
    <code>about:blank</code></span> <code>Document</code>, and the new <code>Document</code> that
    is about to be created, will share the same <code>Window</code> object.</p>
  </li>

   <li id="create-new-agent-and-window">
    <p>Otherwise:</p>

    <ol>
     <li><p>Let <var>oacHeader</var> be the result of <span>getting a structured field value</span>
     given `<code>Origin-Agent-Cluster</code>` and "<code data-x="">item</code>" from
     <var>navigationParams</var>'s <span data-x="navigation-params-response">response</span>'s
     <span data-x="concept-response-header-list">header list</span>.</p></li>

     <li><p>Let <var>requestsOAC</var> be true if <var>oacHeader</var> is not
     null and <var>oacHeader</var>[0] is the <span
     data-x="http-structured-header-boolean">boolean</span> true; otherwise false.</p></li>

     <li><p>If <var>navigationParams</var>'s <span
     data-x="navigation-params-reserved-environment">reserved environment</span> is a
     <span>non-secure context</span>, then set <var>requestsOAC</var> to false.</p></li>

     <li><p>Let <var>agent</var> be the result of <span
     data-x="obtain-similar-origin-window-agent">obtaining a similar-origin window agent</span>
     given <var>navigationParams</var>'s <span data-x="navigation-params-origin">origin</span>,
     <var>browsingContext</var>'s <span data-x="tlbc group">group</span>, and
     <var>requestsOAC</var>.</p></li>

     <li>
      <p>Let <var>realmExecutionContext</var> be the result of <span>creating a new realm</span>
      given <var>agent</var> and the following customizations:</p>

      <ul>
       <li><p>For the global object, create a new <code>Window</code> object.</p></li>

       <li><p>For the global <b>this</b> binding, use <var>browsingContext</var>'s
       <code>WindowProxy</code> object.</p></li>
      </ul>
     </li>

     <li><p>Set <var>window</var> to the <span data-x="concept-realm-global">global
     object</span> of <var>realmExecutionContext</var>'s Realm component.</p></li>

     <li><p>Let <var>topLevelCreationURL</var> be <var>creationURL</var>.</p></li>

     <li><p>Let <var>topLevelOrigin</var> be <var>navigationParams</var>'s <span
     data-x="navigation-params-origin">origin</span>.</p></li>

     <li>
      <p>If <var>navigable</var>'s <span data-x="nav-container">container</span> is not null,
      then:</p>

      <ol>
       <li><p>Let <var>parentEnvironment</var> be <var>navigable</var>'s <span
       data-x="nav-container">container</span>'s <span>relevant settings object</span>.</p></li>

       <li><p>Set <var>topLevelCreationURL</var> to <var>parentEnvironment</var>'s <span>top-level
       creation URL</span>.</p></li>

       <li><p>Set <var>topLevelOrigin</var> to <var>parentEnvironment</var>'s <span>top-level
       origin</span>.</p></li>
      </ol>
     </li>

     <li><p><span>Set up a window environment settings object</span> with <var>creationURL</var>,
     <var>realmExecutionContext</var>, <var>navigationParams</var>'s <span
     data-x="navigation-params-reserved-environment">reserved environment</span>,
     <var>topLevelCreationURL</var>, and <var>topLevelOrigin</var>.</p></li>
    </ol>

    <p class="note">This is the usual case, where the new <code>Document</code> we're about to
    create gets a new <code>Window</code> to go along with it.</p>
   </li>

   <li><p>Let <var>loadTimingInfo</var> be a new <span>document load timing info</span> with its
   <span>navigation start time</span> set to <var>navigationParams</var>'s <span
   data-x="navigation-params-response">response</span>'s <span
   data-x="concept-response-timing-info">timing info</span>'s <span
   data-x="fetch-timing-info-start-time">start time</span>.</p></li>

   <li>
    <p>Let <var>document</var> be a new <code>Document</code>, with</p>

    <dl class="props">
     <dt><span data-x="concept-document-type">type</span></dt>
     <dd><var>type</var></dd>

     <dt><span data-x="concept-document-content-type">content type</span></dt>
     <dd><var>contentType</var></dd>

     <dt><span data-x="concept-document-origin">origin</span></dt>
     <dd><var>navigationParams</var>'s <span data-x="navigation-params-origin">origin</span></dd>

     <dt><span data-x="concept-document-bc">browsing context</span></dt>
     <dd><var>browsingContext</var>

     <dt><span data-x="concept-document-policy-container">policy container</span></dt>
     <dd><var>navigationParams</var>'s <span data-x="navigation-params-policy-container">policy
     container</span></dd>

     <dt><span data-x="concept-document-permissions-policy">permissions policy</span></dt>
     <dd><var>permissionsPolicy</var></dd>

     <dt><span>active sandboxing flag set</span></dt>
     <dd><var>navigationParams</var>'s <span data-x="navigation-params-sandboxing">final sandboxing
     flag set</span></dd>

     <dt><span data-x="concept-document-coop">opener policy</span></dt>
     <dd><var>navigationParams</var>'s <span data-x="navigation-params-coop">cross-origin opener
     policy</span></dd>

     <dt><span>load timing info</span></dt>
     <dd><var>loadTimingInfo</var></dd>

     <dt><span>was created via cross-origin redirects</span></dt>
     <dd><var>navigationParams</var>'s <span data-x="navigation-params-response">response</span>'s
     <span data-x="concept-response-has-cross-origin-redirects">has cross-origin
     redirects</span></dd>

     <dt><span data-x="concept-document-navigation-id">during-loading navigation ID for WebDriver BiDi</span></dt>
     <dd><var>navigationParams</var>'s <span data-x="navigation-params-id">id</span></dd>

     <dt><span data-x="concept-document-url">URL</span></dt>
     <dd><var>creationURL</var></dd>

     <dt><span>current document readiness</span></dt>
     <dd>"<code data-x="">loading</code>"</dd>

     <dt><span data-x="concept-document-about-base-url">about base URL</span></dt>
     <dd><var>navigationParams</var>'s <span data-x="navigation-params-about-base-URL">about base URL</span></dd>

     <dt><span data-x="concept-document-allow-declarative-shadow-roots">allow declarative shadow
     roots</span></dt>
     <dd>true</dd>
    </dl>
   </li>

   <li><p>Set <var>window</var>'s <span data-x="concept-document-window">associated
   <code>Document</code></span> to <var>document</var>.</p></li>

   <li><p><span>Run CSP initialization for a <code data-x="">Document</code></span> given
   <var>document</var>. <ref>CSP</ref></p>

   <li>
    <p>If <var>navigationParams</var>'s <span data-x="navigation-params-request">request</span> is
    non-null, then:</p>

    <ol>
     <li><p>Set <var>document</var>'s <span data-x="the document's referrer">referrer</span> to the
     empty string.</p></li>

     <li><p>Let <var>referrer</var> be <var>navigationParams</var>'s <span
     data-x="navigation-params-request">request</span>'s <span
     data-x="concept-request-referrer">referrer</span>.</p></li>

     <li>
      <p>If <var>referrer</var> is a <span>URL record</span>, then set <var>document</var>'s <span
      data-x="the document's referrer">referrer</span> to the <span
      data-x="concept-url-serializer">serialization</span> of <var>referrer</var>.</p>

      <p class="note">Per <cite>Fetch</cite>, <var>referrer</var> will be either a <span>URL
      record</span> or "<code data-x="">no-referrer</code>" at this point.</p>
     </li>
    </ol>
   </li>

   <li>
    <p>If <var>navigationParams</var>'s <span data-x="navigation-params-fetch-controller">fetch
    controller</span> is not null, then:</p>

    <ol>
     <li><p>Let <var>fullTimingInfo</var> be the result of <span data-x="extract full timing
     info">extracting the full timing info</span> from <var>navigationParams</var>'s <span
     data-x="navigation-params-fetch-controller">fetch controller</span>.</p></li>

     <li><p>Let <var>redirectCount</var> be 0 if <var>navigationParams</var>'s <span
     data-x="navigation-params-response">response</span>'s <span
     data-x="concept-response-has-cross-origin-redirects">has cross-origin redirects</span> is
     true; otherwise <var>navigationParams</var>'s <span
     data-x="navigation-params-request">request</span>'s <span
     data-x="concept-request-redirect-count">redirect count</span>.</p></li>

     <li><p><span>Create the navigation timing entry</span> for <var>document</var>, given
     <var>fullTimingInfo</var>, <var>redirectCount</var>, <var>navigationTimingType</var>,
     <var>navigationParams</var>'s <span data-x="navigation-params-response">response</span>'s
     <span data-x="concept-response-service-worker-timing-info">service worker timing info</span>,
     and <var>navigationParams</var>'s <span data-x="navigation-params-response">response</span>'s
     <span data-x="concept-response-body-info">body info</span>.</p></li>
    </ol>
   </li>

   <li><p><span>Create the navigation timing entry</span> for <var>document</var>, with
   <var>navigationParams</var>'s <span data-x="navigation-params-response">response</span>'s <span
   data-x="concept-response-timing-info">timing info</span>, <var>redirectCount</var>,
   <var>navigationParams</var>'s <span data-x="navigation-params-nav-timing-type">navigation timing
   type</span>, and <var>navigationParams</var>'s <span
   data-x="navigation-params-response">response</span>'s <span
   data-x="concept-response-service-worker-timing-info">service worker timing info</span>.</p></li>

   <li>
    <p>If <var>navigationParams</var>'s <span data-x="navigation-params-response">response</span>
    has a `<code>Refresh</code>` header, then:</p>

    <ol>
     <li><p>Let <var>value</var> be the <span data-x="isomorphic decode">isomorphic decoding</span>
     of the value of the header.</p></li>

     <li><p>Run the <span>shared declarative refresh steps</span> with <var>document</var> and
     <var>value</var>.</p></li>
    </ol>

    <p class="XXX">We do not currently have a spec for how to handle multiple `<code>Refresh</code>`
    headers. This is tracked as <a
    href="https://github.com/whatwg/html/issues/2900">issue #2900</a>.</p>
   </li>

   <li><p>If <var>navigationParams</var>'s <span
   data-x="navigation-params-commit-early-hints">commit early hints</span> is not null, then call
   <var>navigationParams</var>'s <span data-x="navigation-params-commit-early-hints">commit early
   hints</span> with <var>document</var>.</p></li>

   <li><p><span>Process link headers</span> given <var>document</var>,
   <var>navigationParams</var>'s <span data-x="navigation-params-response">response</span>, and
   "<code data-x="">pre-media</code>".</p></li>

   <li><p>Return <var>document</var>.</p></li>
  </ol>

  <div class="example">
   <p>In this example, the child document is not allowed to use <code>PaymentRequest</code>,
   despite being <span>same origin-domain</span> at the time the child document tries to use
   it. At the time the child document is initialized, only the parent document has set <code
   data-x="dom-document-domain">document.domain</code>, and the child document has not.</p>

   <pre><code class="html">&lt;!-- https://foo.example.com/a.html -->
&lt;!doctype html>
&lt;script>
document.domain = 'example.com';
&lt;/script>
&lt;iframe src=b.html>&lt;/iframe></code></pre>

  <pre><code class="html">&lt;!-- https://bar.example.com/b.html -->
&lt;!doctype html>
&lt;script>
document.domain = 'example.com'; // This happens after the document is initialized
new PaymentRequest(&hellip;); // Not allowed to use
&lt;/script></code></pre>
  </div>

  <div class="example">
   <p>In this example, the child document <em>is</em> allowed to use
   <code>PaymentRequest</code>, despite not being <span>same origin-domain</span> at the time
   the child document tries to use it. At the time the child document is initialized, none of
   the documents have set <code data-x="dom-document-domain">document.domain</code> yet so
   <span>same origin-domain</span> falls back to a normal <span>same origin</span> check.</p>

   <pre><code class="html">&lt;!-- https://example.com/a.html -->
&lt;!doctype html>
&lt;iframe src=b.html>&lt;/iframe>
&lt;!-- The child document is now initialized, before the script below is run. -->
&lt;script>
document.domain = 'example.com';
&lt;/script></code></pre>

   <pre><code class="html">&lt;!-- https://example.com/b.html -->
&lt;!doctype html>
&lt;script>
new PaymentRequest(&hellip;); // Allowed to use
&lt;/script></code></pre>
  </div>

  <p>To <dfn>populate with <code>html</code>/<code>head</code>/<code>body</code></dfn> given a
  <code>Document</code> <var>document</var>:</p>

  <ol>
   <li><p>Let <var>html</var> be the result of <span data-x="create an element">creating an
   element</span> given <var>document</var>, "<code data-x="">html</code>", and the <span>HTML
   namespace</span>.</p></li>

   <li><p>Let <var>head</var> be the result of <span data-x="create an element">creating an
   element</span> given <var>document</var>, "<code data-x="">head</code>", and the <span>HTML
   namespace</span>.</p></li>

   <li><p>Let <var>body</var> be the result of <span data-x="create an element">creating an
   element</span> given <var>document</var>, "<code data-x="">body</code>", and the <span>HTML
   namespace</span>.</p></li>

   <li><p><span data-x="concept-node-append">Append</span> <var>html</var> to
   <var>document</var>.</p></li>

   <li><p><span data-x="concept-node-append">Append</span> <var>head</var> to
   <var>html</var>.</p></li>

   <li><p><span data-x="concept-node-append">Append</span> <var>body</var> to
   <var>html</var>.</p></li>
  </ol>


  <h4 id="read-html">Loading HTML documents</h4>

  <p>To <dfn data-x="navigate-html">load an HTML document</dfn>, given <span>navigation
  params</span> <var>navigationParams</var>:</p>

  <ol>
   <li><p>Let <var>document</var> be the result of <span
   data-x="create-the-document-object">creating and initializing a <code>Document</code>
   object</span> given "<code data-x="">html</code>", "<code data-x="">text/html</code>", and
   <var>navigationParams</var>.</p></li>

   <li>
    <p>If <var>document</var>'s <span data-x="concept-document-url">URL</span> is
    <code>about:blank</code>, then <span>populate with
    <code>html</code>/<code>head</code>/<code>body</code></span> given <var>document</var>.</p>

    <p class="note">This special case, where even non-<span data-x="is initial about:blank">initial
    <code>about:blank</code></span> <code>Document</code>s are synchronously given their element
    nodes, is necessary for compatible with deployed content. In other words, it is not compatible
    to instead go down the "otherwise" branch and feed the empty <span>byte sequence</span> into
    an <span>HTML parser</span> to asynchronously populate <var>document</var>.</p>
   </li>

   <!-- See https://github.com/whatwg/html/issues/8095 for improving this step. -->
   <li>
    <p>Otherwise, create an <span>HTML parser</span> and associate it with the <var>document</var>.
    Each <span data-x="concept-task">task</span> that the <span>networking task source</span>
    places on the <span>task queue</span> while fetching runs must then fill the parser's
    <span>input byte stream</span> with the fetched bytes and cause the <span>HTML parser</span> to
    perform the appropriate processing of the input stream.</p>

    <p>The first <span data-x="concept-task">task</span> that the <span>networking task
    source</span> places on the <span>task queue</span> while fetching runs must <span>process link
    headers</span> given <var>document</var>, <var>navigationParams</var>'s <span
    data-x="navigation-params-response">response</span>, and "<code data-x="">media</code>", after
    the task has been processed by the <span>HTML parser</span>.</p>

    <p>Before any script execution occurs, the user agent must wait for <span>scripts may run for
    the newly-created document</span> to be true for <var>document</var>.</p>

    <p class="note">The <span>input byte stream</span> converts bytes into characters for use in the
    <span data-x="tokenization">tokenizer</span>. This process relies, in part, on character encoding
    information found in the real <span data-x="Content-Type">Content-Type metadata</span> of the
    resource; the computed type is not used for this purpose.</p>

    <p>When no more bytes are available, the user agent must <span>queue a global task</span> on
    the <span>networking task source</span> given <var>document</var>'s <span>relevant global
    object</span> to have the parser process the implied EOF character, which eventually causes
    a <code data-x="event-load">load</code> event to be fired.</p>
   </li>

   <li><p>Return <var>document</var>.</p></li>
  </ol>


  <h4 id="read-xml"><dfn data-x="navigate-xml">Loading XML documents</dfn></h4>

  <p>When faced with displaying an XML file inline, provided <span>navigation params</span>
  <var>navigationParams</var> and a string <var>type</var>, user agents must follow the
  requirements defined in <cite>XML</cite> and <cite>Namespaces in XML</cite>, <cite>XML Media
  Types</cite>, <cite>DOM</cite>, and other relevant specifications to <span
  data-x="create-the-document-object">create and initialize a <code>Document</code> object</span>
  <var>document</var>, given "<code data-x="">xml</code>", <var>type</var>, and
  <var>navigationParams</var>, and return that <code>Document</code>. They must also create a
  corresponding <span>XML parser</span>. <ref>XML</ref> <ref>XMLNS</ref> <ref>RFC7303</ref>
  <ref>DOM</ref></p>

  <p class="note">At the time of writing, the XML specification community had not actually yet
  specified how XML and the DOM interact.</p> <!--XMLPARSE-->

  <p>The first <span data-x="concept-task">task</span> that the <span>networking task source</span>
  places on the <span>task queue</span> while fetching runs must <span>process link headers</span>
  given <var>document</var>, <var>navigationParams</var>'s <span
  data-x="navigation-params-response">response</span>, and "<code data-x="">media</code>", after
  the task has been processed by the <span>XML parser</span>.</p>

  <p>The actual HTTP headers and other metadata, not the headers as mutated or implied by the
  algorithms given in this specification, are the ones that must be used when determining the
  character encoding according to the rules given in the above specifications. Once the character
  encoding is established, the <span>document's character encoding</span> must be set to that
  character encoding.</p>

  <p>Before any script execution occurs, the user agent must wait for <span>scripts may run for the
  newly-created document</span> to be true for the newly-created <code>Document</code>.</p>

  <p>Once parsing is complete, the user agent must set <var>document</var>'s <span
  data-x="concept-document-navigation-id">during-loading navigation ID for WebDriver BiDi</span> to
  null.</p>

  <p class="note">For HTML documents this is reset when parsing is complete, after firing the load
  event.</p>

  <p>Error messages from the parse process (e.g., XML namespace well-formedness errors) may be
  reported inline by mutating the <code>Document</code>.</p>


  <h4 id="read-text">Loading text documents</h4>

  <p>To <dfn data-x="navigate-text">load a text document</dfn>, given a <span>navigation
  params</span> <var>navigationParams</var> and a string <var>type</var>:</p>

  <ol>
   <li><p>Let <var>document</var> be the result of <span
   data-x="create-the-document-object">creating and initializing a <code>Document</code>
   object</span> given "<code data-x="">html</code>", <var>type</var>, and
   <var>navigationParams</var>.</p></li>

   <li><p>Set <var>document</var>'s <span>parser cannot change the mode flag</span> to
   true.</p></li>

   <li><p>Set <var>document</var>'s <span data-x="concept-document-mode">mode</span>
   to "<code data-x="">no-quirks</code>".</p></li>

   <li>
    <p>Create an <span>HTML parser</span> and associate it with the <var>document</var>. Act as
    if the tokenizer had emitted a start tag token with the tag name "pre" followed by a single
    U+000A LINE FEED (LF) character<!-- to get eaten, so that a leading LF in the text/plain stream
    doesn't get eaten itself-->, and switch the <span>HTML parser</span>'s tokenizer to the
    <span>PLAINTEXT state</span>. Each <span data-x="concept-task">task</span> that the
    <span>networking task source</span> places on the <span>task queue</span> while fetching runs
    must then fill the parser's <span>input byte stream</span> with the fetched bytes and cause the
    <span>HTML parser</span> to perform the appropriate processing of the input stream.</p>

    <p><var>document</var>'s <span data-x="document's character encoding">encoding</span> must be
    set to the character encoding used to decode the document during parsing.</p>

    <p>The first <span data-x="concept-task">task</span> that the <span>networking task
    source</span> places on the <span>task queue</span> while fetching runs must <span>process link
    headers</span> given <var>document</var>, <var>navigationParams</var>'s <span
    data-x="navigation-params-response">response</span>, and "<code data-x="">media</code>", after
    the task has been processed by the <span>HTML parser</span>.</p>

    <p>Before any script execution occurs, the user agent must wait for <span>scripts may run for
    the newly-created document</span> to be true for <var>document</var>.</p>

    <p>When no more bytes are available, the user agent must <span>queue a global task</span> on
    the <span>networking task source</span> given <var>document</var>'s <span>relevant global
    object</span> to have the parser process the implied EOF character, which eventually causes
    a <code data-x="event-load">load</code> event to be fired.</p>
   </li>

   <li>
    <p>User agents may add content to the <code>head</code> element of <var>document</var>, e.g.,
    linking to a style sheet, providing script, or giving the document a <code>title</code>.</p>

    <p class="note">In particular, if the user agent supports the <code
    data-x="">Format=Flowed</code> feature of RFC 3676 then the user agent would need to apply
    extra styling to cause the text to wrap correctly and to handle the quoting feature. This could
    be performed using, e.g., a CSS extension.</p>
   </li>

   <li><p>Return <var>document</var>.</p></li>
  </ol>

  <p>The rules for how to convert the bytes of the plain text document into actual characters, and
  the rules for actually rendering the text to the user, are defined by the specifications for the
  <span data-x="Content-Type sniffing">computed MIME type</span> of the resource (i.e.,
  <var>type</var>).</p>

  <h4 id="read-multipart-x-mixed-replace">Loading <code>multipart/x-mixed-replace</code>
  documents</h4>

  <p>To <dfn data-x="navigate-multipart-x-mixed-replace">load a
  <code>multipart/x-mixed-replace</code> document</dfn>, given <span>navigation params</span>
  <var>navigationParams</var>, <span>source snapshot params</span> <var>sourceSnapshotParams</var>,
  and <span>origin</span> <var>initiatorOrigin</var>:</p>

  <ol>
   <li><p>Parse <var>navigationParams</var>'s <span
   data-x="navigation-params-response">response</span>'s <span
   data-x="concept-response-body">body</span> using the rules for multipart types.
   <ref>RFC2046</ref></p></li>

   <li><p>Let <var>firstPartNavigationParams</var> be a copy of
   <var>navigationParams</var>.</p></li>

   <li><p>Set <var>firstPartNavigationParams</var>'s <span
   data-x="navigation-params-response">response</span> to a new <span
   data-x="concept-response">response</span> representing the first part of
   <var>navigationParams</var>'s <span data-x="navigation-params-response">response</span>'s <span
   data-x="concept-response-body">body</span>'s multipart stream.</p></li>

   <li>
    <p>Let <var>document</var> be the result of <span>loading a document</span> given
    <var>firstPartNavigationParams</var>, <var>sourceSnapshotParams</var>, and
    <var>initiatorOrigin</var>.</p>

    <p>For each additional body part obtained from <var>navigationParams</var>'s <span
    data-x="navigation-params-response">response</span>, the user agent must <span>navigate</span>
    <var>document</var>'s <span>node navigable</span> to <var>navigationParams</var>'s <span
    data-x="navigation-params-request">request</span>'s <span
    data-x="concept-request-url">URL</span>, using <var>document</var>, with <i
    data-x="navigation-response">response</i> set to <var>navigationParams</var>'s <span
    data-x="navigation-params-response">response</span> and <i
    data-x="navigation-hh">historyHandling</i> set to "<code
    data-x="NavigationHistoryBehavior-replace">replace</code>".</p>
   </li>

   <li><p>Return <var>document</var>.</p></li>
  </ol>

  <p>For the purposes of algorithms processing these body parts as if they were complete
  stand-alone resources, the user agent must act as if there were no more bytes for those resources
  whenever the boundary following the body part is reached.</p>

  <p class="note">Thus, <code data-x="event-load">load</code> events (and for that matter <code
  data-x="event-unload">unload</code> events) do fire for each body part loaded.</p>


  <h4 id="read-media">Loading media documents</h4>

  <p>To <dfn data-x="navigate-media">load a media document</dfn>, given <var>navigationParams</var>
  and a string <var>type</var>:</p>

  <ol>
   <li><p>Let <var>document</var> be the result of <span
   data-x="create-the-document-object">creating and initializing a <code>Document</code>
   object</span> given "<code data-x="">html</code>", <var>type</var>, and
   <var>navigationParams</var>.</p></li>

   <li><p>Set <var>document</var>'s <span data-x="concept-document-mode">mode</span>
   to "<code data-x="">no-quirks</code>".</p></li>

   <li><p><span>Populate with <code>html</code>/<code>head</code>/<code>body</code></span> given
   <var>document</var>.</p></li>

   <li><p>Append an element <var>host element</var> for the media, as described below, to the
   <code>body</code> element.</p></li>

   <li><p>Set the appropriate attribute of the element <var>host element</var>, as described below,
   to the address of the image, video, or audio resource.</p></li>

   <li><p>User agents may add content to the <code>head</code> element of <var>document</var>, or
   attributes to <var>host element</var>, e.g., to link to a style sheet, to provide a script, to
   give the document a <code>title</code>, or to make the media <span
   data-x="attr-media-autoplay">autoplay</span>.</p></li>

   <li><p><span>Process link headers</span> given <var>document</var>,
   <var>navigationParams</var>'s <span data-x="navigation-params-response">response</span>, and
   "<code data-x="">media</code>".</p></li>

   <li><p>Act as if the user agent had <span data-x="stop parsing">stopped parsing</span>
   <var>document</var>.</p></li>

   <li><p>Return <var>document</var>.</p></li>
  </ol>

  <p>The element <var>host element</var> to create for the media is the element given in
  the table below in the second cell of the row whose first cell describes the media. The
  appropriate attribute to set is the one given by the third cell in that same row.</p>

  <table>
   <thead>
    <tr> <th> Type of media
         <th> Element for the media
         <th> Appropriate attribute
   <tbody>
    <tr> <td> Image
         <td> <code>img</code>
         <td> <code data-x="attr-img-src">src</code>
    <tr> <td> Video
         <td> <code>video</code>
         <td> <code data-x="attr-media-src">src</code>
    <tr> <td> Audio
         <td> <code>audio</code>
         <td> <code data-x="attr-media-src">src</code>
  </table>

  <p>Before any script execution occurs, the user agent must wait for <span>scripts may run for the
  newly-created document</span> to be true for the <code>Document</code>.</p>


  <h4 id="read-ua-inline"><span id="read-ua-plugin"></span><span id="navigate-plugin"></span><dfn
  data-x="navigate-ua-inline">Loading a document for inline content that doesn't have a
  DOM</dfn></h4>

  <p>When the user agent is to create a document to display a user agent page or PDF viewer inline,
  provided a <span>navigable</span> <var>navigable</var>, a <span>navigation ID</span>
  <var>navigationId</var>, a <code>NavigationTimingType</code> <var>navTimingType</var>, and a
  <span>user navigation involvement</span> <var>userInvolvement</var>, the user agent should:</p>

  <ol>
   <li><p>Let <var>origin</var> be a new <span data-x="concept-origin-opaque">opaque
   origin</span>.</p></li>

   <li><p>Let <var>coop</var> be a new <span>opener policy</span>.</p></li>

   <li>
    <p>Let <var>coopEnforcementResult</var> be a new <span data-x="coop-enforcement-result">opener
    policy enforcement result</span> with</p>

    <dl class="props">
     <dt><span data-x="coop-enforcement-url">url</span></dt>
     <dd><var>response</var>'s <span data-x="concept-response-url">URL</span></dd>

     <dt><span data-x="coop-enforcement-origin">origin</span></dt>
     <dd><var>origin</var></dd>

     <dt><span data-x="coop-enforcement-coop">opener policy</span></dt>
     <dd><var>coop</var></dd>
    </dl>
   </li>

   <li>
    <p>Let <var>navigationParams</var> be a new <span>navigation params</span> with</p>

    <dl class="props">
     <dt><span data-x="navigation-params-id">id</span></dt>
     <dd><var>navigationId</var></dd>

     <dt><span data-x="navigation-params-navigable">navigable</span></dt>
     <dd><var>navigable</var></dd>

     <dt><span data-x="navigation-params-request">request</span></dt>
     <dd>null</dd>

     <dt><span data-x="navigation-params-response">response</span></dt>
     <dd>a new <span data-x="concept-response">response</span></dd>

     <dt><span data-x="navigation-params-origin">origin</span></dt>
     <dd><var>origin</var></dd>

     <dt><span data-x="navigation-params-fetch-controller">fetch controller</span></dt>
     <dd>null</dd>

     <dt><span data-x="navigation-params-commit-early-hints">commit early hints</span></dt>
     <dd>null</dd>

     <dt><span data-x="navigation-params-coop-enforcement-result">COOP enforcement result</span></dt>
     <dd><var>coopEnforcementResult</var></dd>

     <dt><span data-x="navigation-params-reserved-environment">reserved environment</span></dt>
     <dd>null</dd>

     <dt><span data-x="navigation-params-policy-container">policy container</span></dt>
     <dd>a new <span>policy container</span></dd>

     <dt><span data-x="navigation-params-sandboxing">final sandboxing flag set</span></dt>
     <dd>an empty set</dd>

     <dt><span data-x="navigation-params-coop">opener policy</span></dt>
     <dd><var>coop</var></dd>

     <dt><span data-x="navigation-params-nav-timing-type">navigation timing type</span></dt>
     <dd><var>navTimingType</var></dd>

     <dt><span data-x="navigation-params-about-base-url">about base URL</span></dt>
     <dd>null</dd>

     <dt><span data-x="navigation-params-user-involvement">user involvement</span></dt>
     <dd><var>userInvolvement</var></dd>
    </dl>
   </li>

   <li><p>Let <var>document</var> be the result of <span
   data-x="create-the-document-object">creating and initializing a <code>Document</code>
   object</span> given "<code data-x="">html</code>", "<code data-x="">text/html</code>", and
   <var>navigationParams</var>.</p></li>

   <li><p>Either associate <var>document</var> with a custom rendering that is not rendered using
   the normal <code>Document</code> rendering rules, or mutate <var>document</var> until it
   represents the content the user agent wants to render.</p></li>

   <li><p>Return <var>document</var>.</p></li>
  </ol>

  <p class="note">Because we ensure the resulting <code>Document</code>'s <span
  data-x="concept-document-origin">origin</span> is <span
  data-x="concept-origin-opaque">opaque</span>, and the resulting <code>Document</code> cannot run
  script with access to the DOM, the existence and properties of this <code>Document</code> are not
  observable to web developer code. This means that most of the above values, e.g., the
  <code>text/html</code> <span data-x="concept-document-type">type</span>, do not matter.
  Similarly, most of the items in <var>navigationParams</var> don't have any observable effect,
  besides preventing the <span data-x="create-the-document-object"><code>Document</code>-creation
  algorithm</span> from getting confused, and so are set to default values.</p>

  <p>Once the page has been set up, the user agent must act as if it had <span data-x="stop
  parsing">stopped parsing</span>.</p>


  <h4 id="loading-documents">Finishing the loading process</h4>

  <p>A <code>Document</code> has a <dfn>completely loaded time</dfn> (a time or null), which is
  initially null.</p>

  <p>A <code>Document</code> is considered <dfn>completely loaded</dfn> if its <span>completely
  loaded time</span> is non-null.

  <p>To <dfn>completely finish loading</dfn> a <code>Document</code> <var>document</var>:</p>

  <ol>
   <li><p><span>Assert</span>: <var>document</var>'s <span data-x="concept-document-bc">browsing
   context</span> is non-null.</p></li>

   <li><p>Set <var>document</var>'s <span>completely loaded time</span> to the current
   time.</p></li>

   <li>
    <p>Let <var>container</var> be <var>document</var>'s <span>node navigable</span>'s <span
    data-x="nav-container">container</span>.</p>

    <div class="note">
     <p>This will be null in the case where <var>document</var> is the <span data-x="is initial
     about:blank">initial <code>about:blank</code></span> <code>Document</code> in a
     <code>frame</code> or <code>iframe</code>, since at the point of <span data-x="creating a new
     browsing context and document">browsing context creation</span> which calls this algorithm,
     the container relationship has not yet been established. (That happens in a subsequent step of
     <span>create a new child navigable</span>.)</p>

     <p>The consequence of this is that the following steps do nothing, i.e., we do not fire an
     asynchronous <code data-x="event-load">load</code> event on the container element for such
     cases. Instead, a synchronous <code data-x="event-load">load</code> event is fired in a special
     initial-insertion case when <span data-x="process the iframe attributes">processing the
     <code>iframe</code> attributes</span>.</p>
    </div>
   </li>

   <li><p>If <var>container</var> is an <code>iframe</code> element, then <span>queue an element
   task</span> on the <span>DOM manipulation task source</span> given <var>container</var> to run
   the <span>iframe load event steps</span> given <var>container</var>.</p></li>

   <li><p>Otherwise, if <var>container</var> is non-null, then <span>queue an element task</span> on
   the <span>DOM manipulation task source</span> given <var>container</var> to <span
   data-x="concept-event-fire">fire an event</span> named <code data-x="event-load">load</code> at
   <var>container</var>.</p>
  </ol>

  <h4>Unloading documents</h4>

  <p>A <code>Document</code> has a <dfn
  data-x="concept-document-salvageable"><i>salvageable</i></dfn> state, which must initially be
  true, and a <dfn>page showing</dfn> boolean, which is initially false. The <span>page
  showing</span> boolean is used to ensure that scripts receive <code
  data-x="event-pageshow">pageshow</code> and <code data-x="event-pagehide">pagehide</code> events
  in a consistent manner (e.g., that they never receive two <code
  data-x="event-pagehide">pagehide</code> events in a row without an intervening <code
  data-x="event-pageshow">pageshow</code>, or vice versa).</p>

  <p>A <code>Document</code> has a <code>DOMHighResTimeStamp</code> <dfn>suspension time</dfn>,
  initially 0.</p>

  <p>A <code>Document</code> has a <span>list</span> of <dfn>suspended timer handles</dfn>,
  initially empty.</p>

  <p><span data-x="event loop">Event loops</span> have a <dfn>termination nesting level</dfn>
  counter, which must initially be 0.</p>

  <p id="ignore-opens-during-unload-counter"><code>Document</code> objects have an <dfn>unload
  counter</dfn>, which is used to ignore certain operations while the below algorithms run.
  Initially, the counter must be set to zero.</p>

  <p>To <dfn data-x="unload a document">unload</dfn> a <code>Document</code>
  <var>oldDocument</var>, given an optional <code>Document</code> <var>newDocument</var>:</p>

  <ol>
   <li><p><span>Assert</span>: this is running as part of a <span data-x="concept-task">task</span>
   queued on <var>oldDocument</var>'s <span>relevant agent</span>'s <span
   data-x="concept-agent-event-loop">event loop</span>.</p></li>

   <li><p>Let <var>unloadTimingInfo</var> be a new <span>document unload timing
   info</span>.</p></li>

   <li>
    <p>If <var>newDocument</var> is not given, then set <var>unloadTimingInfo</var> to null.</p>

    <p class="note">In this case there is no new document that needs to know about how long it took
    <var>oldDocument</var> to unload.</p>
   </li>

   <li>
    <p>Otherwise, if <var>newDocument</var>'s <span>event loop</span> is not
    <var>oldDocument</var>'s <span>event loop</span>, then the user agent may be <span
    data-x="unload a document">unloading</span> <var>oldDocument</var> <span>in parallel</span>. In
    that case, the user agent should set <var>unloadTimingInfo</var> to null.</p>

    <p class="note">In this case <var>newDocument</var>'s loading is not impacted by how long it
    takes to unload <var>oldDocument</var>, so it would be meaningless to communicate that timing
    info.</p>
   </li>

   <li>
    <p>Let <var>intendToKeepInBfcache</var> be true if the user agent intends to keep
    <var>oldDocument</var> alive in a <span>session history entry</span>, such that it can later be
    <a href="#note-bfcache">used for history traversal</a>.</p>

    <p>This must be false if <var>oldDocument</var> is not <i
    data-x="concept-document-salvageable">salvageable</i>, or if there are any descendants of
    <var>oldDocument</var> which the user agent does not intend to keep alive in the same way
    (including due to their lack of <i
    data-x="concept-document-salvageable">salvageability</i>).</p>
   </li>

   <li><p>Let <var>eventLoop</var> be <var>oldDocument</var>'s <span>relevant agent</span>'s <span
   data-x="concept-agent-event-loop">event loop</span>.</p></li>

   <li><p>Increase <var>eventLoop</var>'s <span>termination nesting level</span> by 1.</p></li>

   <li><p>Increase <var>oldDocument</var>'s <span>unload counter</span> by 1.</p></li>

   <li><p>If <var>intendToKeepInBfcache</var> is false, then set <var>oldDocument</var>'s <i
   data-x="concept-document-salvageable">salvageable</i> state to false.</p></li>

   <li>
    <p>If <var>oldDocument</var>'s <span>page showing</span> is true:</p>

    <ol>
     <li><p>Set <var>oldDocument</var>'s <span>page showing</span> to false.</p></li>

     <li><p><span>Fire a page transition event</span> named <code
     data-x="event-pagehide">pagehide</code> at <var>oldDocument</var>'s <span>relevant global
     object</span> with <var>oldDocument</var>'s <i
     data-x="concept-document-salvageable">salvageable</i> state.</p></li>

     <li><p><span>Update the visibility state</span> of <var>oldDocument</var> to
     "<code data-x="">hidden</code>".</p></li>
    </ol>
   </li>

   <li><p>If <var>unloadTimingInfo</var> is not null, then set <var>unloadTimingInfo</var>'s
   <span>unload event start time</span> to the <span>current high resolution time</span> given
   <var>newDocument</var>'s <span>relevant global object</span>, <span data-x="coarsen
   time">coarsened</span> given <var>oldDocument</var>'s <span>relevant settings object</span>'s
   <span data-x="concept-settings-object-cross-origin-isolated-capability">cross-origin isolated
   capability</span>.</p></li>

   <li><p>If <var>oldDocument</var>'s <i data-x="concept-document-salvageable">salvageable</i>
   state is false, then <span data-x="concept-event-fire">fire an event</span> named <code
   data-x="event-unload">unload</code> at <var>oldDocument</var>'s <span>relevant global
   object</span>, with <var>legacy target override flag</var> set.</p></li>

   <li><p>If <var>unloadTimingInfo</var> is not null, then set <var>unloadTimingInfo</var>'s
   <span>unload event end time</span> to the <span>current high resolution time</span> given
   <var>newDocument</var>'s <span>relevant global object</span>, <span data-x="coarsen
   time">coarsened</span> given <var>oldDocument</var>'s <span>relevant settings object</span>'s
   <span data-x="concept-settings-object-cross-origin-isolated-capability">cross-origin isolated
   capability</span>.</p></li>

   <li><p>Decrease <var>eventLoop</var>'s <span>termination nesting level</span> by 1.</p></li>

   <li><p>Set <var>oldDocument</var>'s <span>suspension time</span> to the <span>current high
   resolution time</span> given <var>document</var>'s <span>relevant global object</span>.</p></li>

   <li><p>Set <var>oldDocument</var>'s <span>suspended timer handles</span> to the result of
   <span data-x="map get the keys">getting the keys</span> for the
   <span>map of active timers</span>.</p></li>

   <li><p>Set <var>oldDocument</var>'s <span>has been scrolled by the user</span> to false.</p></li>

   <li><p>Run any <span>unloading document cleanup steps</span> for <var>oldDocument</var> that are
   defined by this specification and <span>other applicable specifications</span>.</p></li>

   <li><p>If <var>oldDocument</var>'s <span>node navigable</span> is a <span>top-level
   traversable</span>, <span>build not restored reasons for a top-level traversable and its
   descendants</span> given <var>oldDocument</var>'s <span>node navigable</span>.</p></li>

   <li><p>If <var>oldDocument</var>'s <i data-x="concept-document-salvageable">salvageable</i> state
   is false, then <span data-x="destroy a document">destroy</span> <var>oldDocument</var>.</p></li>

   <li><p>Decrease <var>oldDocument</var>'s <span>unload counter</span> by 1.</p></li>

   <li><p>If <var>newDocument</var> is given, <var>newDocument</var>'s <span>was created via
   cross-origin redirects</span> is false, and <var>newDocument</var>'s <span
   data-x="concept-document-origin">origin</span> is the <span data-x="same origin">same</span> as
   <var>oldDocument</var>'s <span data-x="concept-document-origin">origin</span>, then set
   <var>newDocument</var>'s <span>previous document unload timing</span> to
   <var>unloadTimingInfo</var>.</p></li>
  </ol>

  <p>To <dfn>unload a document and its descendants</dfn>, given a <code>Document</code>
  <var>document</var>, an optional <code>Document</code>-or-null <var>newDocument</var> (default
  null), an optional set of steps <var>afterAllUnloads</var>, and an optional set of steps
  <var>firePageSwapSteps</var>:</p>

  <ol>
   <li><p><span>Assert</span>: this is running within <var>document</var>'s <span>node
   navigable</span>'s <span data-x="nav-traversable">traversable navigable</span>'s <span
   data-x="tn-session-history-traversal-queue">session history traversal queue</span>.</p></li>

   <li><p>Let <var>childNavigables</var> be <var>document</var>'s <span data-x="child
   navigable">child navigables</span>.</p></li>

   <li><p>Let <var>numberUnloaded</var> be 0.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>childNavigable</var> of
    <var>childNavigables</var> <span class="XXX" data-x="">in what order?</span>, <span>queue a
    global task</span> on the <span>navigation and traversal task source</span> given
    <var>childNavigable</var>'s <span data-x="nav-window">active window</span> to perform the
    following steps:</p>

    <ol>
     <li><p>Let <var>incrementUnloaded</var> be an algorithm step which increments
     <var>numberUnloaded</var>.</p></li>

     <li><p><span>Unload a document and its descendants</span> given <var>childNavigable</var>'s
     <span data-x="nav-document">active document</span>, null, and
     <var>incrementUnloaded</var>.</p></li>
    </ol>
   </li>

   <li><p>Wait until <var>numberUnloaded</var> equals <var>childNavigable</var>'s <span
   data-x="list size">size</span>.</p></li>

   <li>
    <p><span>Queue a global task</span> on the <span>navigation and traversal task source</span>
    given <var>document</var>'s <span>relevant global object</span> to perform the following
    steps:</p>

    <ol>
     <li><p>If <var>firePageSwapSteps</var> is given, then run <var>firePageSwapSteps</var>.</p></li>

     <li><p><span data-x="unload a document">Unload</span> <var>document</var>, passing along
     <var>newDocument</var> if it is not null.</p></li>

     <li><p>If <var>afterAllUnloads</var> was given, then run it.</p></li>
    </ol>
   </li>
  </ol>

  <p>This specification defines the following <dfn export>unloading document cleanup steps</dfn>.
  Other specifications can define more. Given a <code>Document</code> <var>document</var>:</p>

  <ol>
   <li><p>Let <var>window</var> be <var>document</var>'s <span>relevant global
   object</span>.</p></li>

   <li>
    <p>For each <code>WebSocket</code> object <var>webSocket</var> whose <span>relevant global
    object</span> is <var>window</var>, <span>make disappear</span> <var>webSocket</var>.</p>

    <p>If this affected any <code>WebSocket</code> objects, then <span>make document
    unsalvageable</span> given <var>document</var> and "<code
    data-x="blocking-websocket">websocket</code>".</p>
   </li>

   <li><p>For each <code>WebTransport</code> object <var>transport</var> whose <span>relevant global
   object</span> is <var>window</var>, run the <span>context cleanup steps</span> given
   <var>transport</var>.</p></li>

   <li>
    <p>If <var>document</var>'s <i data-x="concept-document-salvageable">salvageable</i>
    state is false, then:</p>

    <ol>
     <li><p>For each <code>EventSource</code> object <var>eventSource</var> whose <span>relevant
     global object</span> is equal to <var>window</var>, <span
     data-x="concept-EventSource-forcibly-close">forcibly close</span>
     <var>eventSource</var>.</p></li>

     <li><p><span data-x="map clear">Clear</span> <var>window</var>'s
     <span>map of active timers</span>.</p></li>
    </ol>
   </li>
  </ol>

  <div class="XXX">
   <p>It would be better if specification authors sent a pull request to add calls from here into
   their specifications directly, instead of using the <span>unloading document cleanup
   steps</span> hook, to ensure well-defined cross-specification call order. As of the time of this
   writing the following specifications are known to have <span>unloading document cleanup
   steps</span>, which will be run in an unspecified order: <cite>Fullscreen API</cite>, <cite>Web
   NFC</cite>, <cite>WebDriver BiDi</cite>, <cite>Compute Pressure</cite>, <cite>File API</cite>,
   <cite>Media Capture and Streams</cite>, <cite>Picture-in-Picture</cite>, <cite>Screen
   Orientation</cite>, <cite>Service Workers</cite>, <cite>WebLocks API</cite>, <cite>WebAudio
   API</cite>, <cite>WebRTC</cite>. <ref>FULLSCREEN</ref> <ref>WEBNFC</ref> <ref>WEBDRIVERBIDI</ref>
   <ref>COMPUTEPRESSURE</ref> <ref>FILEAPI</ref> <ref>MEDIASTREAM</ref> <ref>PICTUREINPICTURE</ref>
   <ref>SCREENORIENTATION</ref> <ref>SW</ref> <ref>WEBLOCKS</ref> <ref>WEBAUDIO</ref>
   <ref>WEBRTC</ref>

   <p><a href="https://github.com/whatwg/html/issues/8906">Issue #8906</a> tracks the work to make
   the order of these steps clear.</p>
  </div>

  <h4>Destroying documents</h4>

  <p id="discard-a-document">To <dfn data-x="destroy a document">destroy</dfn> a
  <code>Document</code> <var>document</var>:</p>

  <ol>
   <li><p><span>Assert</span>: this is running as part of a <span data-x="concept-task">task</span>
   queued on <var>document</var>'s <span>relevant agent</span>'s <span
   data-x="concept-agent-event-loop">event loop</span>.</p></li>

   <li><p><span data-x="abort a document">Abort</span> <var>document</var>.</p></li>

   <li><p>Set <var>document</var>'s <i data-x="concept-document-salvageable">salvageable</i> state
   to false.</p></li>

   <li><p>Let <var>ports</var> be the list of <code>MessagePort</code>s whose <span>relevant global
   object</span>'s <span data-x="concept-document-window">associated <code>Document</code></span> is
   <var>document</var>.</p></li>

   <li><p>For each <var>port</var> in <var>ports</var>, <span>disentangle</span>
   <var>port</var>.</p></li>

   <li><p>Run any <span>unloading document cleanup steps</span> for <var>document</var> that
   are defined by this specification and <span>other applicable specifications</span>.</p></li>

   <li><p>Remove any <span data-x="concept-task">tasks</span> whose <span
   data-x="concept-task-document">document</span> is <var>document</var> from any <span>task
   queue</span> (without running those tasks).</p></li>

   <li><p>Set <var>document</var>'s <span data-x="concept-document-bc">browsing context</span> to
   null.</p></li>

   <li><p>Set <var>document</var>'s <span>node navigable</span>'s <span
   data-x="nav-active-history-entry">active session history entry</span>'s <span
   data-x="she-document-state">document state</span>'s <span
   data-x="document-state-document">document</span> to null.</p></li>

   <li><p><span data-x="list remove">Remove</span> <var>document</var> from the
   <span>owner set</span> of each <code>WorkerGlobalScope</code> object whose set <span
   data-x="list contains">contains</span> <var>document</var>.</p></li>

   <li><p><span data-x="list iterate">For each</span> <var>workletGlobalScope</var> in
   <var>document</var>'s <span data-x="concept-Document-worklet-global-scopes">worklet global
   scopes</span>, <span data-x="terminate a worklet global scope">terminate</span>
   <var>workletGlobalScope</var>.</p></li>
  </ol>

  <p class="note">Even after destruction, the <code>Document</code> object itself might still be
  accessible to script, in the case where we are <span data-x="destroy a child
  navigable">destroying a child navigable</span>.</p>

  <p>To <dfn>destroy a document and its descendants</dfn> given a <code>Document</code>
  <var>document</var> and an optional set of steps <var>afterAllDestruction</var>, perform the
  following steps <span>in parallel</span>:</p>

  <ol>
   <li>
    <p>If <var>document</var> is not <span>fully active</span>, then:</p>

    <ol>
     <li><p>Let <var>reason</var> be a string from <span
     data-x="ua-specific-blocking-reasons">user-agent specific blocking reasons</span>. If none
     apply, then let <var>reason</var> be "<code data-x="blocking-masked">masked</code>".</p></li>

     <li><p><span>Make document unsalvageable</span> given <var>document</var> and
     <var>reason</var>.</p></li>

     <li><p>If <var>document</var>'s <span>node navigable</span> is a
     <span>top-level traversable</span>,
     <span>build not restored reasons for a top-level traversable and its descendants</span>
     given <var>document</var>'s <span>node navigable</span>.</p></li>
    </ol>
   </li>

   <li><p>Let <var>childNavigables</var> be <var>document</var>'s <span data-x="child
   navigable">child navigables</span>.</p></li>

   <li><p>Let <var>numberDestroyed</var> be 0.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>childNavigable</var> of
    <var>childNavigables</var> <span class="XXX" data-x="">in what order?</span>, <span>queue a
    global task</span> on the <span>navigation and traversal task source</span> given
    <var>childNavigable</var>'s <span data-x="nav-window">active window</span> to perform the
    following steps:</p>

    <ol>
     <li><p>Let <var>incrementDestroyed</var> be an algorithm step which increments
     <var>numberDestroyed</var>.</p></li>

     <li><p><span>Destroy a document and its descendants</span> given <var>childNavigable</var>'s
     <span data-x="nav-document">active document</span> and <var>incrementDestroyed</var>.</p></li>
    </ol>
   </li>

   <li><p>Wait until <var>numberDestroyed</var> equals <var>childNavigable</var>'s <span
   data-x="list size">size</span>.</p></li>

   <li>
    <p><span>Queue a global task</span> on the <span>navigation and traversal task source</span>
    given <var>document</var>'s <span>relevant global object</span> to perform the following
    steps:</p>

    <ol>
     <li><p><span data-x="destroy a document">Destroy</span> <var>document</var>.</p></li>

     <li><p>If <var>afterAllDestruction</var> was given, then run it.</p></li>
    </ol>
   </li>
  </ol>

  <h4>Aborting a document load</h4>

  <p>To <dfn data-x="abort a document">abort</dfn> a <code>Document</code> <var>document</var>:</p>

  <ol>
   <li><p><span>Assert</span>: this is running as part of a <span data-x="concept-task">task</span>
   queued on <var>document</var>'s <span>relevant agent</span>'s <span
   data-x="concept-agent-event-loop">event loop</span>.</p></li>

   <li><p>Cancel any instances of the <span data-x="concept-fetch">fetch</span> algorithm in the
   context of <var>document</var>, discarding any <span data-x="concept-task">tasks</span> <span
   data-x="queue a task">queued</span> for them, and discarding any further data received from the
   network for them. If this resulted in any instances of the <span
   data-x="concept-fetch">fetch</span> algorithm being canceled or any <span data-x="queue a
   task">queued</span> <span data-x="concept-task">tasks</span> or any network data getting
   discarded, then <span>make document unsalvageable</span> given <var>document</var> and
   "<code data-x="blocking-fetch">fetch</code>".</p></li>

   <li>
    <p>If <var>document</var>'s <span data-x="concept-document-navigation-id">during-loading
    navigation ID for WebDriver BiDi</span> is non-null, then:</p>

    <ol>
     <li><p>Invoke <span>WebDriver BiDi navigation aborted</span> with <var>document</var>'s
     <span>node navigable</span> and a new <span>WebDriver BiDi navigation status</span> whose <span
     data-x="navigation-status-id">id</span> is <var>document</var>'s <span
     data-x="concept-document-navigation-id">during-loading navigation ID for WebDriver BiDi</span>,
     <span data-x="navigation-status-status">status</span> is "<code
     data-x="navigation-status-canceled" >canceled</code>", and <span
     data-x="navigation-status-url">url</span> is <var>document</var>'s <span
     data-x="concept-document-url">URL</span>.</p></li>

     <li><p>Set <var>document</var>'s <span data-x="concept-document-navigation-id">during-loading
     navigation ID for WebDriver BiDi</span> to null.</p></li>
    </ol>
   </li>

   <li>
    <p>If <var>document</var> has an <span>active parser</span>, then:</p>

    <ol>
     <li><p>Set <var>document</var>'s <span>active parser was aborted</span> to true.</p></li>

     <li><p><span data-x="abort a parser">Abort that parser</span>.</p></li>

     <li><p>Set <var>document</var>'s <i data-x="concept-document-salvageable">salvageable</i> to
     false.</p></li>
     <li><p><span>Make document unsalvageable</span> given <var>document</var> and
     "<code data-x="blocking-parser-aborted">parser-aborted</code>".</p></li>
    </ol>
   </li>
  </ol>

  <p>To <dfn>abort a document and its descendants</dfn> given a <code>Document</code>
  <var>document</var>:</p>

  <ol>
   <li><p><span>Assert</span>: this is running as part of a <span data-x="concept-task">task</span>
   queued on <var>document</var>'s <span>relevant agent</span>'s <span
   data-x="concept-agent-event-loop">event loop</span>.</p></li>

   <li><p>Let <var>descendantNavigables</var> be <var>document</var>'s <span>descendant
   navigables</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>descendantNavigable</var> of
    <var>descendantNavigables</var> <span class="XXX" data-x="">in what order?</span>, <span>queue
    a global task</span> on the <span>navigation and traversal task source</span> given
    <var>descendantNavigable</var>'s <span data-x="nav-window">active window</span> to perform the
    following steps:</p>

    <ol>
     <li><p><span data-x="abort a document">Abort</span> <var>descendantNavigable</var>'s <span
     data-x="nav-document">active document</span>.</p></li>

     <li><p>If <var>descendantNavigable</var>'s <span data-x="nav-document">active
     document</span>'s <i data-x="concept-document-salvageable">salvageable</i> is false, then set
     <var>document</var>'s <i data-x="concept-document-salvageable">salvageable</i> to
     false.</p></li>
    </ol>
   </li>

   <li><p><span data-x="abort a document">Abort</span> <var>document</var>.</p></li>
  </ol>

  <p id="stop-document-loading">To <dfn data-x="nav-stop">stop loading</dfn> a
  <span>navigable</span> <var>navigable</var>:</p>

  <ol>
   <li><p>Let <var>document</var> be <var>navigable</var>'s <span data-x="nav-document">active
   document</span>.</p></li>

   <li>
    <p>If <var>document</var>'s <span>unload counter</span> is 0, and <var>navigable</var>'s
    <span>ongoing navigation</span> is a <span>navigation ID</span>, then <span>set the ongoing
    navigation</span> for <var>navigable</var> to null.</p>

    <p class="note">This will have the effect of aborting any ongoing navigations of
    <var>navigable</var>, since at certain points during navigation, changes to the <span>ongoing
    navigation</span> will cause further work to be abandoned.</p>
   </li>

   <li><p><span>Abort a document and its descendants</span> given <var>document</var>.</p></li>
  </ol>

  <p class="XXX">Through their <a href="#nav-traversal-ui">user interface</a>, user agents also
  allow stopping traversals, i.e. cases where the <span>ongoing navigation</span> is "<code
  data-x="">traversal</code>". The above algorithm does not account for this. (On the other hand,
  user agents do not allow <code data-x="dom-window-stop">window.stop()</code> to stop traversals,
  so the above algorithm is correct for that caller.) See <a
  href="https://github.com/whatwg/html/issues/6905">issue #6905</a>.</p>


  <h3>The `<code>X-Frame-Options</code>` header</h3>

  <p>The `<dfn http-header><code>X-Frame-Options</code></dfn>` HTTP response header is a way
  of controlling whether and how a <code>Document</code> may be loaded inside of a <span>child
  navigable</span>. For sites using CSP, the <code data-x="frame-ancestors
  directive">frame-ancestors</code> directive provides more granular control over the
  same situations. It was originally defined in <cite>HTTP Header Field X-Frame-Options</cite>, but
  the definition<span w-nodev> and processing model</span> here supersedes that document.
  <ref>CSP</ref> <ref>RFC7034</ref>

  <p class="note">In particular, <cite>HTTP Header Field X-Frame-Options</cite> specified an `<code
  data-x="">ALLOW-FROM</code>` variant of the header, but that is not to be implemented.</p>

  <p class="note"><span w-nodev>Per the below processing model, if</span><span w-dev>If</span> both
  a CSP <code data-x="frame-ancestors directive">frame-ancestors</code> directive and an
  `<code>X-Frame-Options</code>` header are used in the same <span
  data-x="concept-response">response</span>, then `<code>X-Frame-Options</code>` is ignored.</p>

  <p>For web developers and conformance checkers, its value <span data-x="header-abnf">ABNF</span>
  is:</p>

  <pre><code data-x="" class="abnf">X-Frame-Options = "DENY" / "SAMEORIGIN"</code></pre>

  <div w-nodev>

  <p>To <dfn>check a navigation response's adherence to `<code>X-Frame-Options</code>`</dfn>, given
  a <span data-x="concept-response">response</span> <var>response</var>, a <span>navigable</span>
  <var>navigable</var>, a <span data-x="concept-csp-list">CSP list</span> <var>cspList</var>, and
  an <span>origin</span> <var>destinationOrigin</var>:</p>

  <ol>
   <li><p>If <var>navigable</var> is not a <span>child navigable</span>, then return true.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>policy</var> of <var>cspList</var>:</p>

    <ol>
     <li><p>If <var>policy</var>'s <span data-x="csp-disposition">disposition</span> is not "<code
     data-x="">enforce</code>", then <span>continue</span>.</p></li>

     <li><p>If <var>policy</var>'s <span data-x="csp-directive-set">directive set</span> <span
     data-x="list contains">contains</span> a <code data-x="frame-ancestors
     directive">frame-ancestors</code> directive, then return true.</p></li>
    </ol>
   </li>

   <li><p>Let <var>rawXFrameOptions</var> be the result of <span
   data-x="concept-header-list-get-decode-split">getting, decoding, and splitting</span>
   `<code>X-Frame-Options</code>` from <var>response</var>'s <span
   data-x="concept-response-header-list">header list</span>.</p></li>

   <li><p>Let <var>xFrameOptions</var> be a new <span>set</span>.</p></li>

   <li><p><span data-x="list iterate">For each</span> <var>value</var> of
   <var>rawXFrameOptions</var>, <span data-x="set append">append</span> <var>value</var>,
   <span>converted to ASCII lowercase</span>, to <var>xFrameOptions</var>.</p></li>

   <li>
    <p>If <var>xFrameOptions</var>'s <span data-x="list size">size</span> is greater than 1, and
    <var>xFrameOptions</var> <span data-x="list contains">contains</span> any of "<code
    data-x="">deny</code>", "<code data-x="">allowall</code>", or "<code
    data-x="">sameorigin</code>", then return false.</p>

    <p class="note">The intention here is to block any attempts at applying
    `<code>X-Frame-Options</code>` which were trying to do something valid, but appear confused.</p>

    <p class="note">This is the only impact of the legacy `<code data-x="">ALLOWALL</code>` value
    on the processing model.</p>
   </li>

   <li>
    <p>If <var>xFrameOptions</var>'s <span data-x="list size">size</span> is greater than 1, then
    return true.</p>

    <p class="note">This means it contains multiple invalid values, which we treat the same way as
    if the header was omitted entirely.</p>
   </li>

   <li><p>If <var>xFrameOptions</var>[0] is "<code data-x="">deny</code>", then return
   false.</p></li>

   <li>
    <p>If <var>xFrameOptions</var>[0] is "<code data-x="">sameorigin</code>", then:</p>

    <ol>
     <li><p>Let <var>containerDocument</var> be <var>navigable</var>'s <span
     data-x="nav-container-document">container document</span>.</p></li>

     <li>
      <p><span>While</span> <var>containerDocument</var> is not null:</p>

      <ol>
       <li><p>If <var>containerDocument</var>'s <span data-x="concept-document-origin">origin</span>
       is not <span>same origin</span> with <var>destinationOrigin</var>, then return
       false.</p></li>

       <li><p>Set <var>containerDocument</var> to <var>containerDocument</var>'s <span
       data-x="doc-container-document">container document</span>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li>
    <p>Return true.</p>

    <p class="note">If we've reached this point then we have a lone invalid value (which could
    potentially be one the legacy `<code data-x="">ALLOWALL</code>` or `<code
    data-x="">ALLOW-FROM</code>` forms). These are treated as if the header were omitted
    entirely.</p>
   </li>
  </ol>

  <hr>

  </div>

  <div class="example">
   <p>The following table illustrates the processing of various values for the header, including
   non-conformant ones:</p>

   <table class="data">
    <thead>
     <tr>
      <th>`<code>X-Frame-Options</code>`</th>
      <th>Valid</th>
      <th>Result</th>
     </tr>
    </thead>
    <tbody>
     <tr>
      <td>`<code data-x="">DENY</code>`</td>
      <td>✅</td>
      <td>embedding disallowed</td>
     </tr>
     <tr>
      <td>`<code data-x="">SAMEORIGIN</code>`</td>
      <td>✅</td>
      <td>same-origin embedding allowed</td>
     </tr>
     <tr>
      <td>`<code data-x="">INVALID</code>`</td>
      <td>❌</td>
      <td>embedding allowed</td>
     </tr>
     <tr>
      <td>`<code data-x="">ALLOWALL</code>`</td>
      <td>❌</td>
      <td>embedding allowed</td>
     </tr>
     <tr>
      <td>`<code data-x="">ALLOW-FROM=https://example.com/</code>`</td>
      <td>❌</td>
      <td>embedding allowed (from anywhere)</td>
     </tr>
    </tbody>
   </table>
  </div>

  <div class="example">
   <p>The following table illustrates how various non-conformant cases involving multiple values are
   processed:</p>

   <table class="data">
    <thead>
     <tr>
      <th>`<code>X-Frame-Options</code>`</th>
      <th>Result</th>
     </tr>
    </thead>
    <tbody>
     <tr>
      <td>`<code data-x="">SAMEORIGIN, SAMEORIGIN</code>`</td>
      <td>same-origin embedding allowed</td>
     </tr>
     <tr>
      <td>`<code data-x="">SAMEORIGIN, DENY</code>`</td>
      <td>embedding disallowed</td>
     </tr>
     <tr>
      <td>`<code data-x="">SAMEORIGIN,</code>`</td>
      <td>embedding disallowed</td>
     </tr>
     <tr>
      <td>`<code data-x="">SAMEORIGIN, ALLOWALL</code>`</td>
      <td>embedding disallowed</td>
     </tr>
     <tr>
      <td>`<code data-x="">SAMEORIGIN, INVALID</code>`</td>
      <td>embedding disallowed</td>
     </tr>
     <tr>
      <td>`<code data-x="">ALLOWALL, INVALID</code>`</td>
      <td>embedding disallowed</td>
     </tr>
     <tr>
      <td>`<code data-x="">ALLOWALL,</code>`</td>
      <td>embedding disallowed</td>
     </tr>
     <tr>
      <td>`<code data-x="">INVALID, INVALID</code>`</td>
      <td>embedding allowed</td>
     </tr>
    </tbody>
   </table>

   <p>The same results are obtained whether the values are delivered in a single header whose value is comma-delimited,
   or in multiple headers.</p>
  </div>


  <h3>The `<code>Refresh</code>` header</h3>

  <p>The `<dfn http-header><code>Refresh</code></dfn>` HTTP response header is the HTTP-equivalent
  to a <code>meta</code> element with an <code data-x="attr-meta-http-equiv">http-equiv</code>
  attribute in the <span data-x="attr-meta-http-equiv-refresh">Refresh state</span>. It takes <a
  href="#conformance-attr-meta-http-equiv-refresh">the same value</a> and works largely the
  same.<span w-nodev> Its processing model is detailed in <span
  data-x="create-the-document-object">create and initialize a <code>Document</code>
  object</span>.</span></p>
  <!-- This might be a retcon. -->


  <h3 id="nav-traversal-ui"><span id="history-notes"></span>Browser user interface
  considerations</h3>

  <p>Browser user agents should provide the ability to <!-- DONAV browser UI
  --><span>navigate</span>, <span>reload</span>, and <span data-x="nav-stop">stop loading</span>
  any <span>top-level traversable</span> in their <span>top-level traversable set</span>.</p>

  <p class="example">For example, via a location bar and reload/stop button UI.</p>

  <p>Browser user agents should provide the ability to <span data-x="traverse the history by a
  delta">traverse by a delta</span> any <span>top-level traversable</span> in their <span>top-level
  traversable set</span>.</p>

  <p class="example">For example, via back and forward buttons, possibly including long-press
  abilities to change the delta.</p>

  <p>It is suggested that such user agents allow traversal by deltas greater than one, to avoid
  letting a page "trap" the user by stuffing the session history with spurious entries. (For
  example, via repeated calls to <code data-x="dom-history-pushState">history.pushState()</code> or
  <span data-x="navigate-fragid">fragment navigations</span>.)</p>

  <p class="note">Some user agents have heuristics for translating a single "back" or "forward"
  button press into a larger delta, specifically to overcome such abuses. We are contemplating
  specifying these heuristics in <a href="https://github.com/whatwg/html/issues/7832">issue
  #7832</a>.</p>

  <p>Browser user agents should offer users the ability to <span>create a fresh top-level
  traversable</span>, given a user-provided or user agent-determined initial <span>URL</span>.</p>

  <p class="example">For example, via a "new tab" or "new window" button.</p>

  <p>Browser user agents should offer users the ability to arbitrarily <span data-x="close a
  top-level traversable">close</span> any <span>top-level traversable</span> in their
  <span>top-level traversable set</span>.</p>

  <p class="example">For example, by clicking a "close tab" button.</p>

  <hr>

  <p>Browser user agents may provide ways for the user to explicitly cause any
  <span>navigable</span> (not just a <span>top-level traversable</span>) to <!-- DONAV browser UI
  --><span>navigate</span>, <span>reload</span>, or <span data-x="nav-stop">stop
  loading</span>.</p>

  <p class="example">For example, via a context menu.</p>

  <p>Browser user agents may provide the ability for users to <span>destroy a top-level
  traversable</span>.</p>

  <p class="example">For example, by force-closing a window containing one or more such <span
  data-x="top-level traversable">top-level traversables</span>.</p>

  <hr>

  <p>When a user requests a <span>reload</span> of a <span>navigable</span> whose <span
  data-x="nav-active-history-entry">active session history entry</span>'s <span
  data-x="she-document-state">document state</span>'s <span
  data-x="document-state-resource">resource</span> is a <span>POST resource</span>, the user agent
  should prompt the user to confirm the operation first, since otherwise transactions (e.g.,
  purchases or database modifications) could be repeated.</p>

  <p>When a user requests a <span>reload</span> of a <span>navigable</span>, user agents may provide
  a mechanism for ignoring any caches when reloading.</p>

  <hr>

  <p>All calls to <span>navigate</span> initiated by the mechanisms mentioned above must have the <i
  data-x="navigation-user-involvement">userInvolvement</i> argument set to "<code
  data-x="uni-browser-ui">browser UI</code>".</p>

  <p>All calls to <span>reload</span> initiated by the mechanisms mentioned above must have the <i
  data-x="reload-user-involvement">userInvolvement</i> argument set to "<code
  data-x="uni-browser-ui">browser UI</code>".</p>

  <p>All calls to <span>traverse the history by a delta</span> initiated by the mechanisms mentioned
  above must not pass a value for the <i data-x="traverse-sourceDocument">sourceDocument</i>
  argument.</p>

  <hr>

  <p>The above recommendations, and the data structures in this specification, are not meant to
  place restrictions on how user agents represent the session history to the user.</p>

  <p>For example, although a <span>top-level traversable</span>'s <span
  data-x="tn-session-history-entries">session history entries</span> are stored and maintained as a
  list, and the user agent is recommended to give an interface for <span data-x="traverse the
  history by a delta">traversing that list by a delta</span>, a novel user agent could instead or
  in addition present a tree-like view, with each page having multiple "forward" pages that the user
  can choose between.</p>

  <p>Similarly, although session history for all descendant <span
  data-x="navigable">navigables</span> is stored in their <span>traversable navigable</span>, user
  agents could present the user with a more nuanced per-<span>navigable</span> view of the session
  history.</p>

  <hr>

  <p>Browser user agents may use a <span>top-level browsing context</span>'s <span>is
  popup</span> boolean for the following purposes:

  <ul>
   <li><p>Deciding whether or not to provide a minimal web browser user interface for the
   corresponding <span>top-level traversable</span>.</p></li>

   <li><p>Performing the optional steps in <span>set up browsing context features</span>.</p></li>
  </ul>

  <p>In both cases user agents might additionally incorporate user preferences, or present a choice
  as to whether to go down the popup route.</p>

  <p>User agents that provides a minimal user interface for such popups are encouraged to not hide
  the browser's location bar.</p>


  <h2 split-filename="webappapis" id="webappapis">Web application APIs</h2>

  <h3 id="scripting">Scripting</h3>

  <h4>Introduction</h4>

  <p>Various mechanisms can cause author-provided executable code to run in the context of a
  document. These mechanisms include, but are probably not limited to:</p>

  <ul>
   <li>Processing of <code>script</code> elements.</li>

   <li>Navigating to <span data-x="javascript protocol"><code>javascript:</code> URLs</span>.</li>

   <li>Event handlers, whether registered through the DOM using <code
   data-x="">addEventListener()</code>, by explicit <span>event handler content attributes</span>, by
   <span>event handler IDL attributes</span>, or otherwise.</li>

   <li>Processing of technologies like SVG that have their own scripting features.</li>
  </ul>

  <h4>Agents and agent clusters</h4>

  <h5>Integration with the JavaScript agent formalism</h5>

  <p>JavaScript defines the concept of an <span>agent</span>. This section gives the mapping of that
  language-level concept on to the web platform.</p>

  <div class="note">
   <p>Conceptually, the <span>agent</span> concept is an architecture-independent, idealized
   "thread" in which JavaScript code runs. Such code can involve multiple globals/<span
   data-x="concept-global-object-realm">realms</span> that can synchronously access each other, and
   thus needs to run in a single execution thread.</p>

   <p w-nodev>Two <code>Window</code> objects having the same <span>agent</span> does not indicate
   they can directly access all objects created in each other's realms. They would have to be
   <span>same origin-domain</span>; see <span>IsPlatformObjectSameOrigin</span>.</p>
  </div>

  <p>The following types of agents exist on the web platform:</p>

  <dl>
   <dt><dfn export>Similar-origin window agent</dfn></dt>
   <dd>
    <p>Contains various <code>Window</code> objects which can potentially reach each other, either
    directly or by using <code data-x="dom-document-domain">document.domain</code>.</p>

    <p>If the encompassing <span>agent cluster</span>'s <span>is origin-keyed</span> is true, then
    all the <code>Window</code> objects will be <span>same origin</span>, can reach each other
    directly, and <code data-x="dom-document-domain">document.domain</code> will no-op.</p>

    <p class="note">Two <code>Window</code> objects that are <span>same origin</span> can be in
    different <span data-x="similar-origin window agent">similar-origin window agents</span>, for
    instance if they are each in their own <span>browsing context group</span>.</p>
   </dd>

   <dt><dfn export>Dedicated worker agent</dfn></dt>
   <dd><p>Contains a single <code>DedicatedWorkerGlobalScope</code>.</p></dd>

   <dt><dfn export>Shared worker agent</dfn></dt>
   <dd><p>Contains a single <code>SharedWorkerGlobalScope</code>.</p></dd>

   <dt><dfn export>Service worker agent</dfn></dt>
   <dd><p>Contains a single <code>ServiceWorkerGlobalScope</code>.</p></dd>

   <dt><dfn export>Worklet agent</dfn></dt>
   <dd>
    <p>Contains a single <code>WorkletGlobalScope</code> object.</p>

    <p class="note">Although a given worklet can have multiple realms, each such realm needs its own
    agent, as each realm can be executing code independently and at the same time as the others.</p>
   </dd>
  </dl>

  <p>Only <span data-x="shared worker agent">shared</span> and <span data-x="dedicated worker
  agent">dedicated worker agents</span> allow the use of JavaScript <code>Atomics</code> APIs to
  potentially <span data-x="forward progress">block</span>.</p>

  <div w-nodev>

  <p>To <dfn>create an agent</dfn>, given a boolean <var>canBlock</var>:</p>

  <ol>
   <li><p>Let <var>signifier</var> be a new unique internal value.</p></li>

   <li><p>Let <var>candidateExecution</var> be a new <span>candidate execution</span>.</p></li>

   <li><p>Let <var>agent</var> be a new <span>agent</span> whose [[CanBlock]] is
   <var>canBlock</var>, [[Signifier]] is <var>signifier</var>, [[CandidateExecution]] is
   <var>candidateExecution</var>, and [[IsLockFree1]], [[IsLockFree2]], and [[LittleEndian]] are set
   at the implementation's discretion.</p></li>

   <li><p>Set <var>agent</var>'s <span data-x="concept-agent-event-loop">event loop</span> to a new
   <span>event loop</span>.</p></li>

   <li><p>Return <var>agent</var>.</p></li>
  </ol>

  <p>For a <span>realm</span> <var>realm</var>, the <span>agent</span> whose [[Signifier]] is
  <var>realm</var>.[[AgentSignifier]] is <dfn data-x="realm's agent" export data-lt="agent"
  for="realm">the realm's agent</dfn>.</p>

  <p>The <dfn export>relevant agent</dfn> for a <span>platform object</span>
  <var>platformObject</var> is <var>platformObject</var>'s <span
  data-x="concept-relevant-realm">relevant realm</span>'s <span data-x="realm's
  agent">agent</span>.</p>

  <p class="note">The agent equivalent of the <span>current realm</span> is the <span>surrounding
  agent</span>.</p>

  </div>

  <h5>Integration with the JavaScript agent cluster formalism</h5>

  <p>JavaScript also defines the concept of an <span>agent cluster</span>, which this standard maps
  to the web platform by placing agents appropriately when they are created<span w-nodev> using the
  <span data-x="obtain-similar-origin-window-agent">obtain a similar-origin window agent</span> or
  <span data-x="obtaining a worker/worklet agent">obtain a worker/worklet agent</span>
  algorithms</span>.</p>

  <p>The <span>agent cluster</span> concept is crucial for defining the JavaScript memory model, and
  in particular among which <span data-x="agent">agents</span> the backing data of
  <code>SharedArrayBuffer</code> objects can be shared.</p>

  <p class="note">Conceptually, the <span>agent cluster</span> concept is an
  architecture-independent, idealized "process boundary" that groups together multiple "threads"
  (<span data-x="agent">agents</span>). The <span data-x="agent cluster">agent clusters</span>
  defined by the specification are generally more restrictive than the actual process boundaries
  implemented in user agents. By enforcing these idealized divisions at the specification level, we
  ensure that web developers see interoperable behavior with regard to shared memory, even in the
  face of varying and changing user agent process models.</p>

  <div w-nodev>

  <p>An <span>agent cluster</span> has an associated <dfn
  data-x="agent-cluster-cross-origin-isolation">cross-origin isolation mode</dfn>, which is a
  <span>cross-origin isolation mode</span>. It is initially "<code
  data-x="cross-origin-isolation-none">none</code>".</p>

  <p>An <span>agent cluster</span> has an associated <dfn>is origin-keyed</dfn> (a boolean), which
  is initially false.</p>

  <hr>

  <p>The following defines the allocation of the <span data-x="agent cluster">agent clusters</span>
  of <span data-x="similar-origin window agent">similar-origin window agents</span>.</p>

  <p>An <dfn>agent cluster key</dfn> is a <span>site</span> or <span
  data-x="concept-origin-tuple">tuple origin</span>. Without web developer action to achieve <a
  href="#origin-keyed-agent-clusters">origin-keyed agent clusters</a>, it will be a
  <span>site</span>.</p>

  <p class="note">An equivalent formulation is that an <span>agent cluster key</span> can be a
  <span>scheme-and-host</span> or an <span>origin</span>.</p>

  <p>To <dfn data-x="obtain-similar-origin-window-agent">obtain a similar-origin window agent</dfn>,
  given an <span>origin</span> <var>origin</var>, a <span>browsing context group</span>
  <var>group</var>, and a boolean <var>requestsOAC</var>, run these steps:</p>

  <ol>
   <li><p>Let <var>site</var> be the result of <span data-x="obtain a site">obtaining a site</span>
   with <var>origin</var>.</p></li>

   <li><p>Let <var>key</var> be <var>site</var>.</p></li>

   <li><p>If <var>group</var>'s <span data-x="bcg-cross-origin-isolation">cross-origin isolation
   mode</span> is not "<code data-x="cross-origin-isolation-none">none</code>", then set
   <var>key</var> to <var>origin</var>.</p></li>

   <li><p>Otherwise, if <var>group</var>'s <span>historical agent cluster key
   map</span>[<var>origin</var>] <span data-x="map exists">exists</span>, then set <var>key</var> to
   <var>group</var>'s <span>historical agent cluster key map</span>[<var>origin</var>].</p></li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p>If <var>requestsOAC</var> is true, then set <var>key</var> to
     <var>origin</var>.</p></li>

     <li><p>Set <var>group</var>'s <span>historical agent cluster key map</span>[<var>origin</var>]
     to <var>key</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>If <var>group</var>'s <span>agent cluster map</span>[<var>key</var>] <span data-x="map
    exists">does not exist</span>, then:</p>

    <ol>
     <li><p>Let <var>agentCluster</var> be a new <span>agent cluster</span>.</p></li>

     <li><p>Set <var>agentCluster</var>'s <span
     data-x="agent-cluster-cross-origin-isolation">cross-origin isolation mode</span> to
     <var>group</var>'s <span data-x="bcg-cross-origin-isolation">cross-origin isolation
     mode</span>.</p></li>

     <li>
      <p>If <var>key</var> is an <span>origin</span>:</p>

      <ol>
       <li><p><span>Assert</span>: <var>key</var> is <var>origin</var>.</p></li>

       <li><p>Set <var>agentCluster</var>'s <span>is origin-keyed</span> to true.</p></li>
      </ol>
     </li>

     <li><p>Add the result of <span data-x="create an agent">creating an agent</span>, given false,
     to <var>agentCluster</var>.</p></li>

     <li><p>Set <var>group</var>'s <span>agent cluster map</span>[<var>key</var>] to
     <var>agentCluster</var>.</p></li>
    </ol>
   </li>

   <li><p>Return the single <span>similar-origin window agent</span> contained in <var>group</var>'s
   <span>agent cluster map</span>[<var>key</var>].</p></li>
  </ol>

  <p class="note">This means that there is only one <span>similar-origin window agent</span> per
  browsing context agent cluster. (However, <span data-x="dedicated worker agent">dedicated
  worker</span> and <span data-x="worklet agent">worklet agents</span> might be in the same
  cluster.)</p>

  <hr>

  <p>The following defines the allocation of the <span data-x="agent cluster">agent clusters</span>
  of all other types of agents.</p>

  <p>To <dfn data-x="obtaining a worker/worklet agent">obtain a worker/worklet agent</dfn>, given an
  <span>environment settings object</span> or null <var>outside settings</var>, a boolean
  <var>isTopLevel</var>, and a boolean <var>canBlock</var>, run these steps:</p>

  <ol>
   <li><p>Let <var>agentCluster</var> be null.

   <li>
    <p>If <var>isTopLevel</var> is true, then:</p>

    <ol>
     <li><p>Set <var>agentCluster</var> to a new <span>agent cluster</span>.</p></li>

     <li>
      <p>Set <var>agentCluster</var>'s <span>is origin-keyed</span> to true.</p>

      <p class="note">These workers can be considered to be origin-keyed. However, this is not
      exposed through any APIs (in the way that <code
      data-x="dom-originAgentCluster">originAgentCluster</code> exposes the origin-keyedness for
      windows).</p>
     </li>
    </ol>
   </li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p><span>Assert</span>: <var>outside settings</var> is not null.</p></li>

     <li><p>Let <var>ownerAgent</var> be <var>outside settings</var>'s <span data-x="environment
     settings object's realm">realm</span>'s <span data-x="realm's agent">agent</span>.</p></li>

     <li><p>Set <var>agentCluster</var> to the agent cluster which contains
     <var>ownerAgent</var>.</p></li>
    </ol>
   </li>

   <li><p>Let <var>agent</var> be the result of <span data-x="create an agent">creating an
   agent</span> given <var>canBlock</var>.</p></li>

   <li><p>Add <var>agent</var> to <var>agentCluster</var>.</p></li>

   <li><p>Return <var>agent</var>.</p></li>
  </ol>

  <p>To <dfn>obtain a dedicated/shared worker agent</dfn>, given an <span>environment settings
  object</span> <var>outside settings</var> and a boolean <var>isShared</var>, return the result of
  <span>obtaining a worker/worklet agent</span> given <var>outside settings</var>,
  <var>isShared</var>, and true.</p>

  <p>To <dfn>obtain a worklet agent</dfn>, given an <span>environment settings object</span>
  <var>outside settings</var>, return the result of <span>obtaining a worker/worklet agent</span>
  given <var>outside settings</var>, false, and false.</p>

  <p>To <dfn export>obtain a service worker agent</dfn>, return the result of <span>obtaining a
  worker/worklet agent</span> given null, true, and false.</p>

  <hr>

  </div>

  <div class="example" id="can-share-memory-with">
   <p>The following pairs of global objects are each within the same <span>agent cluster</span>, and
   thus can use <code>SharedArrayBuffer</code> instances to share memory with each other:</p>

   <ul>
    <li><p>A <code>Window</code> object and a dedicated worker that it created.</p></li>

    <li><p>A worker (of any type) and a dedicated worker it created.</p></li>

    <li><p>A <code>Window</code> object <var>A</var> and the <code>Window</code> object of an
    <code>iframe</code> element that <var>A</var> created that could be <span>same
    origin-domain</span> with <var>A</var>.</p></li>

    <li><p>A <code>Window</code> object and a <span>same origin-domain</span> <code>Window</code>
    object that opened it.</p></li>

    <li><p>A <code>Window</code> object and a worklet that it created.</p></li>
   </ul>

   <p>The following pairs of global objects are <em>not</em> within the same <span>agent
   cluster</span>, and thus cannot share memory:</p>

   <ul>
    <li><p>A <code>Window</code> object and a shared worker it created.</p></li>

    <li><p>A worker (of any type) and a shared worker it created.</p></li>

    <li><p>A <code>Window</code> object and a service worker it created.</p></li>

    <li><p>A <code>Window</code> object <var>A</var> and the <code>Window</code> object of an
    <code>iframe</code> element that <var>A</var> created that cannot be <span>same
    origin-domain</span> with <var>A</var>.</p></li>

    <li><p>Any two <code>Window</code> objects with no opener or ancestor relationship. This holds
    even if the two <code>Window</code> objects are <span>same origin</span>.</p></li>
   </ul>
  </div>

  <div w-nodev>

  <h4>Realms and their counterparts</h4>

  <p>The JavaScript specification introduces the <span>realm</span> concept, representing a global
  environment in which script is run. Each realm comes with an <span>implementation-defined</span>
  <span>global object</span>; much of this specification is devoted to defining that global object
  and its properties.</p>

  <p>For web specifications, it is often useful to associate values or algorithms with a
  realm/global object pair. When the values are specific to a particular type of realm, they are
  associated directly with the global object in question, e.g., in the definition of the
  <code>Window</code> or <code>WorkerGlobalScope</code> interfaces. When the values have utility
  across multiple realms, we use the <span>environment settings object</span> concept.</p>

  <p>Finally, in some cases it is necessary to track associated values before a
  realm/global object/environment settings object even comes into existence (for example, during
  <span data-x="navigate">navigation</span>). These values are tracked in the
  <span>environment</span> concept.</p>

  <h5>Environments</h5>

  <p>An <dfn export>environment</dfn> is an object that identifies the settings of a current or
  potential execution environment (i.e., a <span data-x="navigation params">navigation
  params</span>'s <span data-x="navigation-params-reserved-environment">reserved environment</span>
  or a <span data-x="concept-request">request</span>'s <span
  data-x="concept-request-reserved-client">reserved client</span>). An <span>environment</span> has
  the following fields:</p>

  <dl>
   <dt>An <dfn data-x="concept-environment-id" export
   for="environment">id</dfn></dt>
   <dd><p>An opaque string that uniquely identifies this <span>environment</span>.</p></dd>

   <dt>A <dfn data-x="concept-environment-creation-url" export
   for="environment">creation URL</dfn></dt>
   <dd>
    <p>A <span>URL</span> that represents the location of the resource with which this
    <span>environment</span> is associated.</p>

    <p class="note">In the case of a <code>Window</code> <span>environment settings object</span>,
    this URL might be distinct from its <span data-x="concept-settings-object-global">global
    object</span>'s <span data-x="concept-document-window">associated
    <code>Document</code></span>'s <span data-x="concept-document-url">URL</span>, due to
    mechanisms such as <code data-x="dom-history-pushstate">history.pushState()</code> which modify
    the latter.</p>
   </dd>

   <dt>A <dfn id="concept-environment-top-level-creation-url" export
   for="environment">top-level creation URL</dfn></dt>
   <dd><p>Null or a <span>URL</span> that represents the <span
   data-x="concept-environment-creation-url">creation URL</span> of the "top-level"
   <span>environment</span>. It is null for workers and worklets.</p></dd>

   <dt>A <dfn id="concept-environment-top-level-origin" export
   for="environment">top-level origin</dfn></dt>
   <dd>
    <p>A <span class="XXX" data-x="">for now</span> <span>implementation-defined</span> value, null,
    or an <span>origin</span>. For a "top-level" potential execution environment it is null (i.e.,
    when there is no response yet); otherwise it is the "top-level" <span>environment</span>'s <span
    data-x="concept-settings-object-origin">origin</span>. For a dedicated worker or worklet it is
    the <span>top-level origin</span> of its creator. For a shared or service worker it is an
    <span>implementation-defined</span> value.</p>

    <p class="note">This is distinct from the <span>top-level creation URL</span>'s <span
    data-x="concept-url-origin">origin</span> when sandboxing, workers, and worklets are
    involved.</p>
   </dd>

   <dt>A <dfn data-x="concept-environment-target-browsing-context" export
   for="environment">target browsing context</dfn></dt>
   <dd><p>Null or a target <span>browsing context</span> for a <span
   data-x="navigation-request">navigation request</span>.</p></dd>

   <dt>An <dfn data-x="concept-environment-active-service-worker" export
   for="environment">active service worker</dfn></dt>
   <dd><p>Null or a <span data-x="dfn-service-worker">service worker</span> that <span
   data-x="dfn-control">controls</span> the <span>environment</span>.</p></dd>

   <dt>An <dfn data-x="concept-environment-execution-ready-flag" export
   for="environment">execution ready flag</dfn></dt>
   <dd><p>A flag that indicates whether the environment setup is done. It is initially
   unset.</p></dd>
  </dl>

  <p>Specifications may define <dfn export>environment discarding steps</dfn> for environments. The
  steps take an <span>environment</span> as input.</p>

  <p class="note">The <span data-x="environment discarding steps">environment discarding
  steps</span> are run for only a select few environments: the ones that will
  never become execution ready because, for example, they failed to load.</p>

  <h5>Environment settings objects</h5>

  <p>An <dfn export>environment settings object</dfn> is an <span>environment</span> that
  additionally specifies algorithms for:</p>

  <dl>
   <dt>A <dfn export for="environment settings object">realm execution
   context</dfn></dt>

   <dd>
    <p>A <span>JavaScript execution context</span> shared by all
    <span data-x="script">scripts</span> that use this settings object, i.e., all scripts in a given
    <span>realm</span>. When we <span>run a classic script</span> or <span>run a module
    script</span>, this execution context becomes the top of the <span>JavaScript execution context
    stack</span>, on top of which another execution context specific to the script in question is
    pushed. (This setup ensures <span>Source Text Module Record</span>'s
    <span data-x="js-Evaluate">Evaluate</span> knows which realm to use.)</p>
   </dd>

   <dt>A <dfn data-x="concept-settings-object-module-map" export for="environment
   settings object">module map</dfn></dt>

   <dd>
    <p>A <span>module map</span> that is used when importing JavaScript modules.</p>
   </dd>

   <dt>An <dfn export for="environment settings object">API base URL</dfn></dt>

   <dd>
    <p>A <span>URL</span> used by APIs called by scripts that use this <span>environment settings
    object</span> to <span data-x="parse a URL">parse URLs</span>.</p>
   </dd>

   <dt>An <dfn data-x="concept-settings-object-origin" export for="environment
   settings object">origin</dfn></dt>

   <dd>
    <p>An <span>origin</span> used in security checks.</p>
   </dd>

   <dt>A <dfn data-x="concept-settings-object-policy-container" export
   for="environment settings object">policy container</dfn></dt>

   <dd>
    <p>A <span>policy container</span> containing policies used for security checks.</p>
   </dd>

   <dt>A <dfn data-x="concept-settings-object-cross-origin-isolated-capability" export
   for="environment settings object">cross-origin isolated capability</dfn></dt>

   <dd><p>A boolean representing whether scripts that use this <span>environment settings
   object</span> are allowed to use APIs that require cross-origin isolation.</p></dd>

   <dt>A <dfn data-x="concept-settings-object-time-origin" export
   for="environment settings object">time origin</dfn></dt>

   <dd>A number used as the baseline for performance-related timestamps. <ref>HRT</ref></dd>
  </dl>

  <p>An <span>environment settings object</span>'s <dfn export for="environment settings
  object">responsible event loop</dfn> is its <span data-x="concept-settings-object-global">global
  object</span>'s <span>relevant agent</span>'s <span data-x="concept-agent-event-loop">event
  loop</span>.</p>

  <h5 id="realms-settings-objects-global-objects">Realms, settings objects, and global objects</h5>

  <p>A <dfn export>global object</dfn> is a JavaScript object that is the [[GlobalObject]] field of
  a <span>realm</span>.</p>

  <p class="note">In this specification, all <span data-x="realm">realms</span> are <span
  data-x="creating a new realm">created</span> with <span data-x="global object">global
  objects</span> that are either <code>Window</code>, <code>WorkerGlobalScope</code>, or
  <code>WorkletGlobalScope</code> objects.</p>

  <p>A <span>global object</span> has an <dfn for="global object">in error reporting mode</dfn>
  boolean, which is initially false.</p>

  <p>A <span>global object</span> has an <dfn>outstanding rejected promises weak set</dfn>, a
  <span>set</span> of <code data-x="idl-Promise">Promise</code> objects, initially empty. This set
  must not create strong references to any of its members, and implementations are free to limit
  its size in an <span>implementation-defined</span> manner, e.g., by removing old entries from it
  when new ones are added.</p>

  <p>A <span>global object</span> has an <dfn>about-to-be-notified rejected promises list</dfn>, a
  <span>list</span> of <code data-x="idl-Promise">Promise</code> objects, initially empty.</p>

  <p>A <span>global object</span> has an <dfn data-x="concept-global-import-map">import map</dfn>,
  initially an <span>empty import map</span>.</p>

  <p class="note">For now, only <code>Window</code> <span data-x="global object">global
  objects</span> have their <span data-x="concept-global-import-map">import map</span> modified
  from the initial empty one. The <span data-x="concept-global-import-map">import map</span> is
  only accessed for the resolution of a root <span>module script</span>.</p>

  <p>A <span>global object</span> has a <dfn>resolved module set</dfn>, a <span>set</span>
  of <span data-x="specifier resolution record">specifier resolution records</span>, initially
  empty.</p>

  <p class="note">The <span>resolved module set</span> ensures that module specifier resolution
  returns the same result when called multiple times with the same (referrer, specifier) pair. It
  does that by ensuring that <span>import map</span> rules that impact the specifier in its
  referrer's scope cannot be defined after its initial resolution. For now, only <code>Window</code>
  <span data-x="global object">global objects</span> have their module set data structures modified
  from the initial empty one.</p>

  <hr>

  <p>There is always a 1-to-1-to-1 mapping between <span data-x="realm">realms</span>, <span
  data-x="global object">global objects</span>, and <span data-x="environment settings
  object">environment settings objects</span>:</p>

  <ul>
   <li><p>A <span>realm</span> has a [[HostDefined]] field, which contains <dfn
   data-x="concept-realm-settings-object" export data-lt="settings object" for="realm">the realm's
   settings object</dfn>.</p></li>

   <li><p>A <span>realm</span> has a [[GlobalObject]] field, which contains <dfn
   data-x="concept-realm-global" export data-lt="global object" for="realm">the realm's global
   object</dfn>.</p></li>

   <li><p>Each <span>global object</span> in this specification is created during the <span
   data-x="creating a new realm">creation</span> of a corresponding <span>realm</span>, known as
   <dfn data-x="concept-global-object-realm" export data-lt="realm" for="global object">the global
   object's realm</dfn>.</p></li>

   <li id="relevant-settings-object-for-a-global-object"><p>Each <span>global object</span> in this
   specification is created alongside a corresponding <span>environment settings object</span>,
   known as its <span>relevant settings object</span>.</p>

   <li><p>An <span>environment settings object</span>'s <span>realm execution context</span>'s
   Realm component is <dfn data-x="environment settings object's realm" export data-lt="realm"
   for="environment settings object">the environment settings object's realm</dfn>.</p></li>

   <li><p>An <span>environment settings object</span>'s <span data-x="environment settings object's
   realm">realm</span> then has a [[GlobalObject]] field, which contains <dfn export
   data-x="concept-settings-object-global" data-lt="global object" for="environment settings
   object">the environment settings object's global object</dfn>.</p></li>
  </ul>

  <p>To <dfn data-x="creating a new realm" export id="creating-a-new-javascript-realm">create a new
  realm</dfn> in an <span>agent</span> <var>agent</var>, optionally with instructions to create a
  global object or a global <b>this</b> binding (or both), the following steps are taken:</p>

  <ol>
   <li><p>Perform <span data-x="js-InitializeHostDefinedRealm">InitializeHostDefinedRealm</span>()
   with the provided customizations for creating the global object and the global <b>this</b>
   binding.</p></li>

   <li>
    <p>Let <var>realm execution context</var> be the <span>running JavaScript execution
    context</span>.</p>

    <p class="note">This is the <span>JavaScript execution context</span> created in the previous
    step.</p>
   </li>

   <li><p>Remove <var>realm execution context</var> from the <span>JavaScript execution context
   stack</span>.</p></li>

   <li><p>Let <var>realm</var> be <var>realm execution context</var>'s Realm component.</p></li>

   <li>
    <p>If <var>agent</var>'s <span>agent cluster</span>'s <span
    data-x="agent-cluster-cross-origin-isolation">cross-origin isolation mode</span> is "<code
    data-x="cross-origin-isolation-none">none</code>", then:</p>

    <ol>
     <li><p>Let <var>global</var> be <var>realm</var>'s <span data-x="concept-realm-global">global
     object</span>.</p></li>

     <li><p>Let <var>status</var> be ! <var>global</var>.[[Delete]]("<code
     data-x="">SharedArrayBuffer</code>").</p></li>

     <li><p><span>Assert</span>: <var>status</var> is true.</p></li>
    </ol>

    <p class="note">This is done for compatibility with web content and there is some hope that this
    can be removed in the future. Web developers can still get at the constructor through
    <code class="js" data-x="">new WebAssembly.Memory({ shared:true, initial:0, maximum:0
    }).buffer.constructor</code>.</p>
   </li>

   <li><p>Return <var>realm execution context</var>.</p></li>
  </ol>

  <hr>

  <p>When defining algorithm steps throughout this specification, it is often important to indicate
  what <span>realm</span> is to be used—or, equivalently, what <span>global object</span> or
  <span>environment settings object</span> is to be used. In general, there are at least four
  possibilities:</p>

  <dl>
    <dt><dfn data-x="concept-entry-everything">Entry</dfn></dt>
    <dd>This corresponds to the script that initiated the currently running script action: i.e.,
    the function or script that the user agent called into when it called into author code.</dd>

    <dt><dfn data-x="concept-incumbent-everything">Incumbent</dfn></dt>
    <dd>This corresponds to the most-recently-entered author function or script on the stack, or the
    author function or script that originally scheduled the currently-running callback.</dd>

    <dt><dfn data-x="concept-current-everything">Current</dfn></dt>
    <dd>This corresponds to the currently-running function object, including built-in user-agent
    functions which might not be implemented as JavaScript. (It is derived from the <span>current
    realm</span>.)</dd>

    <dt><dfn data-x="concept-relevant-everything">Relevant</dfn></dt>
    <dd>Every <span>platform object</span> has a <span data-x="concept-relevant-realm">relevant
    realm</span>, which is roughly the <span>realm</span> in which it was created. When writing
    algorithms, the most prominent <span>platform object</span> whose <span
    data-x="concept-relevant-realm">relevant realm</span> might be important is the <b>this</b>
    value of the currently-running function object. In some cases, there can be other important
    <span data-x="concept-relevant-realm">relevant realms</span>, such as those of any
    arguments.</dd>
  </dl>

  <p>Note how the <span data-x="concept-entry-everything">entry</span>, <span
  data-x="concept-incumbent-everything">incumbent</span>, and <span
  data-x="concept-current-everything">current</span> concepts are usable without qualification,
  whereas the <span data-x="concept-relevant-everything">relevant</span> concept must be applied to
  a particular <span>platform object</span>.</p>

  <p class="warning">The <span data-x="concept-incumbent-everything">incumbent</span> and <span
  data-x="concept-entry-everything">entry</span> concepts should not be used by new specifications,
  as they are excessively complicated and unintuitive to work with. We are working to remove almost
  all existing uses from the platform: see <a
  href="https://github.com/whatwg/html/issues/1430">issue #1430</a> for <span
  data-x="concept-incumbent-everything">incumbent</span>, and <a
  href="https://github.com/whatwg/html/issues/1431">issue #1431</a> for <span
  data-x="concept-entry-everything">entry</span>.</p>

  <p>In general, web platform specifications should use the <span
  data-x="concept-relevant-everything">relevant</span> concept, applied to the object being operated
  on (usually the <b>this</b> value of the current method). This mismatches the JavaScript
  specification, where <span data-x="concept-current-everything">current</span> is generally used as
  the default (e.g., in determining the <span>realm</span> whose <code data-x="">Array</code>
  constructor should be used to construct the result in <code data-x="">Array.prototype.map</code>).
  But this inconsistency is so embedded in the platform that we have to accept it going forward.</p>

  <div class="example">

   <p>Consider the following pages, with <code data-x="">a.html</code> being loaded in a browser
   window, <code data-x="">b.html</code> being loaded in an <code>iframe</code> as shown, and <code
   data-x="">c.html</code> and <code data-x="">d.html</code> omitted (they can simply be empty
   documents):</p>

   <pre><code class="html">&lt;!-- a.html -->
&lt;!DOCTYPE html>
&lt;html lang="en">
&lt;title>Entry page&lt;/title>

&lt;iframe src="b.html">&lt;/iframe>
&lt;button onclick="frames[0].hello()">Hello&lt;/button>

&lt;!--b.html -->
&lt;!DOCTYPE html>
&lt;html lang="en">
&lt;title>Incumbent page&lt;/title>

&lt;iframe src="c.html" id="c">&lt;/iframe>
&lt;iframe src="d.html" id="d">&lt;/iframe>

&lt;script>
  const c = document.querySelector("#c").contentWindow;
  const d = document.querySelector("#d").contentWindow;

  window.hello = () => {
    c.print.call(d);
  };
&lt;/script></code></pre>

   <p>Each page has its own <span>browsing context</span>, and thus its own <span>realm</span>,
   <span>global object</span>, and <span>environment settings object</span>.</p>

   <p>When the <code data-x="dom-print">print()</code> method is called in response to pressing the
   button in <code data-x="">a.html</code>, then:</p>

   <ul>
    <li><p>The <span data-x="concept-entry-realm">entry realm</span> is that of <code
    data-x="">a.html</code>.</p></li>
    <li><p>The <span data-x="concept-incumbent-realm">incumbent realm</span> is that of <code
    data-x="">b.html</code>.</p></li>
    <li><p>The <span>current realm</span> is that of <code data-x="">c.html</code> (since it is the
    <code data-x="dom-print">print()</code> method from <code data-x="">c.html</code> whose code is
    running).</p></li>
    <li><p>The <span data-x="concept-relevant-realm">relevant realm</span> of the object on which
    the <code data-x="dom-print">print()</code> method is being called is that of <code
    data-x="">d.html</code>.</p></li>
   </ul>
  </div>

  <div class="example">
   <p>One reason why the <span data-x="concept-relevant-everything">relevant</span> concept is
   generally a better default choice than the <span
   data-x="concept-current-everything">current</span> concept is that it is more suitable for
   creating an object that is to be persisted and returned multiple times. For example, the <code
   data-x="dom-navigator-getBattery">navigator.getBattery()</code> method creates promises in the
   <span data-x="concept-relevant-realm">relevant realm</span> for the <code>Navigator</code> object
   on which it is invoked. This has the following impact: <ref>BATTERY</ref>

   <pre><code class="html">&lt;!-- outer.html -->
&lt;!DOCTYPE html>
&lt;html lang="en">
&lt;title>Relevant realm demo: outer page&lt;/title>
&lt;script>
  function doTest() {
    const promise = navigator.getBattery.call(frames[0].navigator);

    console.log(promise instanceof Promise);           // logs false
    console.log(promise instanceof frames[0].Promise); // logs true

    frames[0].hello();
  }
&lt;/script>
&lt;iframe src="inner.html" onload="doTest()">&lt;/iframe>

&lt;!-- inner.html -->
&lt;!DOCTYPE html>
&lt;html lang="en">
&lt;title>Relevant realm demo: inner page&lt;/title>
&lt;script>
  function hello() {
    const promise = navigator.getBattery();

    console.log(promise instanceof Promise);        // logs true
    console.log(promise instanceof parent.Promise); // logs false
  }
&lt;/script></code></pre>

   <p>If the algorithm for the <code data-x="dom-navigator-getBattery">getBattery()</code> method
   had instead used the <span>current realm</span>, all the results would be reversed. That is,
   after the first call to <code data-x="dom-navigator-getBattery">getBattery()</code> in <code
   data-x="">outer.html</code>, the <code>Navigator</code> object in <code
   data-x="">inner.html</code> would be permanently storing a <code data-x="">Promise</code> object
   created in <code data-x="">outer.html</code>'s <span>realm</span>, and calls like that inside the
   <code data-x="">hello()</code> function would thus return a promise from the "wrong" realm. Since
   this is undesirable, the algorithm instead uses the <span
   data-x="concept-relevant-realm">relevant realm</span>, giving the sensible results indicated in
   the comments above.</p>
  </div>

  <hr>

  <p>The rest of this section deals with formally defining the <span
  data-x="concept-entry-everything">entry</span>, <span
  data-x="concept-incumbent-everything">incumbent</span>, <span
  data-x="concept-current-everything">current</span>, and <span
  data-x="concept-relevant-everything">relevant</span> concepts.</p>

  <h6>Entry</h6>

  <p>The process of <a href="#calling-scripts">calling scripts</a> will push or pop <span
  data-x="realm execution context">realm execution contexts</span> onto the <span>JavaScript
  execution context stack</span>, interspersed with other <span data-x="JavaScript execution
  context">execution contexts</span>.</p>

  <p>With this in hand, we define the <dfn>entry execution context</dfn> to be the most recently
  pushed item in the <span>JavaScript execution context stack</span> that is a <span data-x="realm
  execution context">realm execution context</span>. The <dfn data-x="concept-entry-realm">entry
  realm</dfn> is the <span>entry execution context</span>'s Realm component.</p>

  <p>Then, the <dfn>entry settings object</dfn> is the <span
  data-x="concept-realm-settings-object">environment settings object</span> of the <span
  data-x="concept-entry-realm">entry realm</span>.</p>

  <p>Similarly, the <dfn>entry global object</dfn> is the <span
  data-x="concept-realm-global">global object</span> of the <span data-x="concept-entry-realm">entry
  realm</span>.</p>

  <h6>Incumbent</h6>

  <p>All <span data-x="JavaScript execution context">JavaScript execution contexts</span> must
  contain, as part of their code evaluation state, a <dfn>skip-when-determining-incumbent
  counter</dfn> value, which is initially zero. In the process of <span data-x="prepare to run a
  callback">preparing to run a callback</span> and <span data-x="clean up after running a
  callback">cleaning up after running a callback</span>, this value will be incremented and
  decremented.</p>

  <p>Every <span>event loop</span> has an associated <dfn>backup incumbent settings object
  stack</dfn>, initially empty. Roughly speaking, it is used to determine the <span>incumbent
  settings object</span> when no author code is on the stack, but author code is responsible for the
  current algorithm having been run in some way. The process of <span data-x="prepare to run a
  callback">preparing to run a callback</span> and <span data-x="clean up after running a
  callback">cleaning up after running a callback</span> manipulate this stack. <ref>WEBIDL</ref></p>

  <p>When Web IDL is used to <span data-x="es-invoking-callback-functions">invoke</span> author
  code, or when <span>HostEnqueuePromiseJob</span> invokes a promise job, they use the following
  algorithms to track relevant data for determining the <span>incumbent settings object</span>:</p>

  <p>To <dfn export>prepare to run a callback</dfn> with an <span>environment settings object</span>
  <var>settings</var>:</p>

  <ol>
   <li><p>Push <var>settings</var> onto the <span>backup incumbent settings object
   stack</span>.</p></li>

   <li><p>Let <var>context</var> be the <span>topmost script-having execution
   context</span>.</p></li>

   <li><p>If <var>context</var> is not null, increment <var>context</var>'s
   <span>skip-when-determining-incumbent counter</span>.</p></li>
  </ol>

  <p>To <dfn export>clean up after running a callback</dfn> with an <span>environment settings
  object</span> <var>settings</var>:</p>

  <ol>
   <li>
    <p>Let <var>context</var> be the <span>topmost script-having execution context</span>.</p>

    <p class="note">This will be the same as the <span>topmost script-having execution
    context</span> inside the corresponding invocation of <span>prepare to run a
    callback</span>.</p>
   </li>

   <li><p>If <var>context</var> is not null, decrement <var>context</var>'s
   <span>skip-when-determining-incumbent counter</span>.</p></li>

   <li><p><span>Assert</span>: the topmost entry of the <span>backup incumbent settings object
   stack</span> is <var>settings</var>.</p></li>

   <li><p>Remove <var>settings</var> from the <span>backup incumbent settings object
   stack</span>.</p></li>
  </ol>

  <p>Here, the <dfn>topmost script-having execution context</dfn> is the topmost entry of the
  <span>JavaScript execution context stack</span> that has a non-null ScriptOrModule component, or
  null if there is no such entry in the <span>JavaScript execution context stack</span>.</p>

  <p>With all this in place, the <dfn export>incumbent settings object</dfn> is determined as
  follows:</p>

  <ol>
   <li><p>Let <var>context</var> be the <span>topmost script-having execution
   context</span>.</p></li>

   <li>
    <p>If <var>context</var> is null, or if <var>context</var>'s
    <span>skip-when-determining-incumbent counter</span> is greater than zero, then:</p>

    <ol>
     <li>
      <p><span>Assert</span>: the <span>backup incumbent settings object stack</span> is not
      empty.</p>

      <p class="note">This assert would fail if you try to obtain the <span>incumbent settings
      object</span> from inside an algorithm that was triggered neither by <a
      href="#calling-scripts">calling scripts</a> nor by Web IDL <span
      data-x="es-invoking-callback-functions">invoking</span> a callback. For example, it would
      trigger if you tried to obtain the <span>incumbent settings object</span> inside an algorithm
      that ran periodically as part of the <span>event loop</span>, with no involvement of author
      code. In such cases the <span data-x="concept-incumbent-everything">incumbent</span> concept
      cannot be used.</p>
     </li>

     <li><p>Return the topmost entry of the <span>backup incumbent settings object
     stack</span>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>context</var>'s Realm component's <span
   data-x="concept-realm-settings-object">settings object</span>.</p></li>
  </ol>

  <p>Then, the <dfn export data-x="concept-incumbent-realm">incumbent realm</dfn> is the <span
  data-x="environment settings object's realm">realm</span> of the <span>incumbent settings
  object</span>.</p>

  <p>Similarly, the <dfn export data-x="concept-incumbent-global">incumbent global object</dfn> is
  the <span data-x="concept-settings-object-global">global object</span> of the <span>incumbent
  settings object</span>.</p>

  <hr>

  <p>The following series of examples is intended to make it clear how all of the different
  mechanisms contribute to the definition of the <a href="#incumbent">incumbent</a> concept:</p>

  <div class="example" id="example-incumbent-1">
   <p>Consider the following starter example:</p>

   <pre><code class="html">&lt;!DOCTYPE html>
&lt;iframe>&lt;/iframe>
&lt;script>
  frames[0].postMessage("some data", "*");
&lt;/script></code></pre>

   <p>There are two interesting <span data-x="environment settings object">environment settings
   objects</span> here: that of <code data-x="">window</code>, and that of <code
   data-x="">frames[0]</code>. Our concern is: what is the <span>incumbent settings object</span> at
   the time that the algorithm for <code data-x="dom-window-postMessage">postMessage()</code>
   executes?</p>

   <p>It should be that of <code data-x="">window</code>, to capture the intuitive notion that the
   author script responsible for causing the algorithm to happen is executing in <code
   data-x="">window</code>, not <code data-x="">frames[0]</code>. This makes sense: the <span>window
   post message steps</span> use the <span>incumbent settings object</span> to determine the <code
   data-x="dom-MessageEvent-source">source</code> property of the resulting
   <code>MessageEvent</code>, and in this case <code data-x="">window</code> is definitely the
   source of the message.</p>

   <p>Let us now explain how the steps given above give us our intuitively-desired result of <code
   data-x="">window</code>'s <span>relevant settings object</span>.</p>

   <p>When the <span>window post message steps</span> look up the <span>incumbent settings
   object</span>, the <span>topmost script-having execution context</span> will be that
   corresponding to the <code>script</code> element: it was pushed onto the <span>JavaScript
   execution context stack</span> as part of <span
   data-x="js-ScriptEvaluation">ScriptEvaluation</span> during the <span>run a classic script</span>
   algorithm. Since there are no Web IDL callback invocations involved, the context's
   <span>skip-when-determining-incumbent counter</span> is zero, so it is used to determine the
   <span>incumbent settings object</span>; the result is the <span>environment settings
   object</span> of <code data-x="">window</code>.</p>

   <p>(Note how the <span>environment settings object</span> of <code data-x="">frames[0]</code> is
   the <span>relevant settings object</span> of <span>this</span> at the time the <code
   data-x="dom-window-postMessage">postMessage()</code> method is called, and thus is involved in
   determining the <em>target</em> of the message. Whereas the incumbent is used to determine the
   <em>source</em>.)
  </div>

  <div class="example" id="example-incumbent-2">
   <p>Consider the following more complicated example:</p>

   <pre><code class="html">&lt;!DOCTYPE html>
&lt;iframe>&lt;/iframe>
&lt;script>
  const bound = frames[0].postMessage.bind(frames[0], "some data", "*");
  window.setTimeout(bound);
&lt;/script></code></pre>

   <p>This example is very similar to the previous one, but with an extra indirection through <code
   data-x="">Function.prototype.bind</code> as well as <code
   data-x="dom-setTimeout">setTimeout()</code>. But, the answer should be the same: invoking
   algorithms asynchronously should not change the <span
   data-x="concept-incumbent-everything">incumbent</span> concept.</p>

   <p>This time, the result involves more complicated mechanisms:</p>

   <p>When <code data-x="">bound</code> is <span data-x="concept-idl-convert">converted</span> to a
   Web IDL callback type, the <span>incumbent settings object</span> is that corresponding to <code
   data-x="">window</code> (in the same manner as in our starter example above). Web IDL stores this
   as the resulting callback value's <span>callback context</span>.</p>

   <p>When the <span data-x="concept-task">task</span> posted by <code
   data-x="dom-setTimeout">setTimeout()</code> executes, the algorithm for that task uses Web IDL to
   <span data-x="es-invoking-callback-functions">invoke</span> the stored callback value. Web IDL in
   turn calls the above <span>prepare to run a callback</span> algorithm. This pushes the stored
   <span>callback context</span> onto the <span>backup incumbent settings object stack</span>. At
   this time (inside the timer task) there is no author code on the stack, so the <span>topmost
   script-having execution context</span> is null, and nothing gets its
   <span>skip-when-determining-incumbent counter</span> incremented.</p>

   <p>Invoking the callback then calls <code data-x="">bound</code>, which in turn calls
   the <code data-x="dom-window-postMessage">postMessage()</code> method of <code
   data-x="">frames[0]</code>. When the <code data-x="dom-window-postMessage">postMessage()</code>
   algorithm looks up the <span>incumbent settings object</span>, there is still no author code on
   the stack, since the bound function just directly calls the built-in method. So the
   <span>topmost script-having execution context</span> will be null: the <span>JavaScript execution
   context</span> stack only contains an execution context corresponding to <code
   data-x="dom-window-postMessage">postMessage()</code>, with no <span
   data-x="js-ScriptEvaluation">ScriptEvaluation</span> context or similar below it.</p>

   <p>This is where we fall back to the <span>backup incumbent settings object stack</span>. As
   noted above, it will contain as its topmost entry the <span>relevant settings object</span> of
   <code data-x="">window</code>. So that is what is used as the <span>incumbent settings
   object</span> while executing the <code data-x="dom-window-postMessage">postMessage()</code>
   algorithm.</p>
  </div>

  <div class="example" id="example-incumbent-3">
   <p>Consider this final, even more convoluted example:</p>

   <pre><code class="html">&lt;!-- a.html -->
&lt;!DOCTYPE html>
&lt;button>click me&lt;/button>
&lt;iframe>&lt;/iframe>
&lt;script>
const bound = frames[0].location.assign.bind(frames[0].location, "https://example.com/");
document.querySelector("button").addEventListener("click", bound);
&lt;/script></code></pre>

   <pre><code class="html">&lt;!-- b.html -->
&lt;!DOCTYPE html>
&lt;iframe src="a.html">&lt;/iframe>
&lt;script>
  const iframe = document.querySelector("iframe");
  iframe.onload = function onLoad() {
    iframe.contentWindow.document.querySelector("button").click();
  };
&lt;/script></code></pre>

   <p>Again there are two interesting <span data-x="environment settings object">environment
   settings objects</span> in play: that of <code data-x="">a.html</code>, and that of <code
   data-x="">b.html</code>. When the <code data-x="dom-location-assign">location.assign()</code>
   method triggers the <span><code>Location</code>-object navigate</span> algorithm, what will be
   the <span>incumbent settings object</span>? As before, it should intuitively be that of <code
   data-x="">a.html</code>: the <code data-x="event-click">click</code> listener was originally
   scheduled by <code data-x="">a.html</code>, so even if something involving <code
   data-x="">b.html</code> causes the listener to fire, the <span
   data-x="concept-incumbent-everything">incumbent</span> responsible is that of <code
   data-x="">a.html</code>.</p>

   <p>The callback setup is similar to the previous example: when <code data-x="">bound</code> is
   <span data-x="concept-idl-convert">converted</span> to a Web IDL callback type, the
   <span>incumbent settings object</span> is that corresponding to <code data-x="">a.html</code>,
   which is stored as the callback's <span>callback context</span>.</p>

   <p>When the <code data-x="dom-click">click()</code> method is called inside <code
   data-x="">b.html</code>, it <span data-x="concept-event-dispatch">dispatches</span> a <code
   data-x="event-click">click</code> event on the button that is inside <code
   data-x="">a.html</code>. This time, when the <span>prepare to run a callback</span> algorithm
   executes as part of event dispatch, there <em>is</em> author code on the stack; the <span>topmost
   script-having execution context</span> is that of the <code data-x="">onLoad</code> function,
   whose <span>skip-when-determining-incumbent counter</span> gets incremented. Additionally, <code
   data-x="">a.html</code>'s <span>environment settings object</span> (stored as the
   <code>EventHandler</code>'s <span>callback context</span>) is pushed onto the
   <span>backup incumbent settings object stack</span>.</p>

   <p>Now, when the <span><code>Location</code>-object navigate</span> algorithm looks up the
   <span>incumbent settings object</span>, the <span>topmost script-having execution
   context</span> is still that of the <code data-x="">onLoad</code> function (due to the fact we
   are using a bound function as the callback). Its <span>skip-when-determining-incumbent
   counter</span> value is one, however, so we fall back to the <span>backup incumbent settings
   object stack</span>. This gives us the <span>environment settings object</span> of <code
   data-x="">a.html</code>, as expected.</p>

   <p>Note that this means that even though it is the <code>iframe</code> inside <code
   data-x="">a.html</code> that navigates, it is <code data-x="">a.html</code> itself that is used
   as the source <code>Document</code>, which determines among other things the <span
   data-x="concept-request-client">request client</span>. This is <a
   href="https://www.w3.org/Bugs/Public/show_bug.cgi?id=26603#c36">perhaps the only justifiable use
   of the incumbent concept on the web platform</a>; in all other cases the consequences of using it
   are simply confusing and we hope to one day switch them to use <span
   data-x="concept-current-everything">current</span> or <span
   data-x="concept-relevant-everything">relevant</span> as appropriate.</p>
  </div>

  <h6>Current</h6>

  <p>The JavaScript specification defines the <span>current realm</span>, also known as the "current
  Realm Record". <ref>JAVASCRIPT</ref></p>

  <p>Then, the <dfn export>current settings object</dfn> is the <span
  data-x="concept-realm-settings-object">environment settings object</span> of the <span>current
  realm</span>.</p>

  <p>Similarly, the <dfn export>current global object</dfn> is the <span
  data-x="concept-realm-global">global object</span> of the <span>current realm</span>.</p>

  <h6>Relevant</h6>

  <p>The <dfn export data-x="concept-relevant-realm">relevant realm</dfn> for a <span>platform
  object</span> is the value of <span data-x="[[Realm]] field of a platform object">its [[Realm]]
  field</span>.</p>

  <p>Then, the <dfn export>relevant settings object</dfn> for a <span>platform object</span>
  <var>o</var> is the <span data-x="concept-realm-settings-object">environment settings
  object</span> of the <span data-x="concept-relevant-realm">relevant realm</span> for
  <var>o</var>.</p>

  <p>Similarly, the <dfn export id="concept-relevant-global">relevant global object</dfn> for a
  <span>platform object</span> <var>o</var> is the <span data-x="concept-realm-global">global
  object</span> of the <span data-x="concept-relevant-realm">relevant realm</span> for
  <var>o</var>.</p>

  <h5>Enabling and disabling scripting</h5>

  <p id="concept-bc-script"><dfn data-x="concept-environment-script">Scripting is enabled</dfn> for
  an <span>environment settings object</span> <var>settings</var> when all of the following
  conditions are true:</p>

  <ul>
   <li>The user agent supports scripting.</li>

   <li>
   <!--INSERT TRACKING-->
   The user has not disabled scripting for <var>settings</var> at this time. (User agents may
   provide users with the option to disable scripting globally, or in a finer-grained manner, e.g.,
   on a per-origin basis, down to the level of individual <span data-x="environment settings
   object">environment settings objects</span>.)</li>

   <li id="sandboxScriptBlocked">Either <var>settings</var>'s <span
   data-x="concept-settings-object-global">global object</span> is not a <code>Window</code> object,
   or <var>settings</var>'s <span data-x="concept-settings-object-global">global object</span>'s
   <span data-x="concept-document-window">associated <code>Document</code></span>'s <span>active
   sandboxing flag set</span> does not have its <span>sandboxed scripts browsing context flag</span>
   set.</li>
  </ul>

  <p id="concept-bc-noscript"><dfn data-x="concept-environment-noscript">Scripting is disabled</dfn>
  for an <span>environment settings object</span> when scripting is not <span
  data-x="concept-environment-script">enabled</span> for it, i.e., when any of the above conditions
  are false.</p>

  <hr>

  <p><dfn data-x="concept-n-script">Scripting is enabled</dfn> for a node <var>node</var> if
  <var>node</var>'s <span>node document</span>'s <span data-x="concept-document-bc">browsing
  context</span> is non-null, and <span data-x="concept-environment-script">scripting is
  enabled</span> for <var>node</var>'s <span>relevant settings object</span>.</p>

  <p><dfn data-x="concept-n-noscript">Scripting is disabled</dfn> for a node when scripting is not
  <span data-x="concept-n-script">enabled</span>, i.e., when its <span>node document</span>'s <span
  data-x="concept-document-bc">browsing context</span> is null or when <span
  data-x="concept-environment-noscript">scripting is disabled</span> for its <span>relevant settings
  object</span>.</p>

  <h5>Secure contexts</h5>

  <p>An <span>environment</span> <var>environment</var> is a <dfn export data-lt="secure context|Is
  an environment settings object contextually secure?">secure context</dfn> if the following
  algorithm returns true:</p>

  <ol>
   <li>
    <p>If <var>environment</var> is an <span>environment settings object</span>, then:</p>

    <ol>
     <li><p>Let <var>global</var> be <var>environment</var>'s <span
     data-x="concept-settings-object-global">global object</span>.</p></li>

     <li>
      <p>If <var>global</var> is a <code>WorkerGlobalScope</code>, then:

      <ol>
       <li>
        <p>If <var>global</var>'s <span>owner set</span>[0]'s <span>relevant settings object</span>
        is a <span>secure context</span>, then return true.</p>

        <p class="note">We only need to check the 0th item since they will necessarily all be
        consistent.</p>
       </li>

       <li><p>Return false.</p></li>
      </ol>
     </li>

     <li>
      <p>If <var>global</var> is a <code>WorkletGlobalScope</code>, then return true.</p>

      <p class="note">Worklets can only be created in secure contexts.</p>
     </li>
    </ol>
   </li>

   <li><p>If the result of <span>Is url potentially trustworthy?</span> given
   <var>environment</var>'s <span>top-level creation URL</span> is "<code data-x="">Potentially
   Trustworthy</code>", then return true.</p></li>

   <li><p>Return false.</p></li>
  </ol>

  <p>An <span>environment</span> is a <dfn export>non-secure context</dfn> if it is not a
  <span>secure context</span>.</p>

  </div>

  <h4 id="scripting-processing-model"><span id="processing-model-7"></span>Script processing
  model</h4>

  <div w-nodev>

  <h5 id="script-structs"><span id="definitions-2"></span>Scripts</h5>

  <p>A <dfn data-x="concept-script" export>script</dfn> is one of two possible <span
  data-x="struct">structs</span> (namely, a <span>classic script</span> or a
  <span>module script</span>). All scripts have:</p>

  <dl>
   <dt>A <dfn for="script" export data-x="concept-script-settings-object"
   id="settings-object">settings object</dfn></dt>

   <dd><p>An <span>environment settings object</span>, containing various settings that are shared
   with other <span data-x="concept-script">scripts</span> in the same context.</p></dd>

   <dt>A <dfn for="script" export data-x="concept-script-record">record</dfn></dt>

   <dd>
    <p>One of the following:</p>

    <ul>
     <li><p>a <span>script record</span>, for <span data-x="classic script">classic
     scripts</span>;</p></li>

     <li><p>a <span>Source Text Module Record</span>, for <span data-x="JavaScript module
     script">JavaScript module scripts</span>;</p></li>

     <li><p>a <span>Synthetic Module Record</span>, for <span data-x="CSS module script">CSS module
     scripts</span> and <span data-x="JSON module script">JSON module scripts</span>;</p></li>

     <li><p>a <span>WebAssembly Module Record</span>, for <span data-x="WebAssembly module
     script">WebAssembly module scripts</span>; or</p></li>

     <li><p>null, representing a parsing failure.</p></li>
    </ul>
   </dd>

   <dt>A <dfn for="script" data-x="concept-script-parse-error">parse error</dfn></dt>

   <dd>
    <p>A JavaScript value, which has meaning only if the <span
    data-x="concept-script-record">record</span> is null, indicating that the corresponding script
    source text could not be parsed.</p>

    <p class="note">This value is used for internal tracking of immediate parse errors when <a
    href="#creating-scripts">creating scripts</a>, and is not to be used directly. Instead, consult
    the <span data-x="concept-script-error-to-rethrow">error to rethrow</span> when determining
    "what went wrong" for this script.</p>
  </dd>

   <dt>An <dfn for="script" export data-x="concept-script-error-to-rethrow">error
   to rethrow</dfn></dt>

   <dd>
    <p>A JavaScript value representing an error that will prevent evaluation from succeeding. It
    will be re-thrown by any attempts to <a href="#calling-scripts">run</a> the script.</p>

    <p class="note">This could be the script's <span data-x="concept-script-parse-error">parse
    error</span>, but in the case of a <span>module script</span> it could instead be the
    <span data-x="concept-script-parse-error">parse error</span> from one of its dependencies, or
    an error from <span>resolve a module specifier</span>.</p>

    <p class="note">Since this exception value is provided by the JavaScript specification, we know
    that it is never null, so we use null to signal that no error has occurred.</p>
   </dd>

   <dt><dfn for="script" export
   data-x="concept-script-script-fetch-options">Fetch options</dfn></dt>

   <dd>Null or a <span>script fetch options</span>, containing various options related to fetching
   this script or <span data-x="module script">module scripts</span> that it imports.</dd>

   <dt>A <dfn for="script" export data-x="concept-script-base-url">base
   URL</dfn></dt>

   <dd><p>Null or a base <span>URL</span> used for <span data-x="resolve a module
   specifier">resolving module specifiers</span>. When non-null, this will either be the URL from
   which the script was obtained, for external scripts, or the <span>document base URL</span> of the
   containing document, for inline scripts.</p></dd>
  </dl>


  <p>A <dfn export>classic script</dfn> is a type of <span data-x="concept-script">script</span>
  that has the following additional <span data-x="struct item">item</span>:</p>

  <dl>
   <dt>A <dfn>muted errors</dfn> boolean</dt>

   <dd><p>A boolean which, if true, means that error information will not be provided for errors in
   this script. This is used to mute errors for cross-origin scripts, since that can leak private
   information.</p></dd>
  </dl>

  <p>A <dfn export>module script</dfn> is another type of <span
  data-x="concept-script">script</span>. It has no additional <span data-x="struct
  item">items</span>.</p>

  <p><span data-x="module script">Module scripts</span> can be classified into four types:</p>

  <ul>
   <li><p>A <span>module script</span> is a <dfn export>JavaScript module script</dfn> if
   its <span data-x="concept-script-record">record</span> is a <span>Source Text Module
   Record</span>.</p></li>

   <li>
    <p>A <span>module script</span> is a <dfn export>CSS module script</dfn> if its <span
    data-x="concept-script-record">record</span> is a <span>Synthetic Module Record</span>, and it
    was created via the <span data-x="creating a CSS module script">create a CSS module
    script</span> algorithm. CSS module scripts represent a parsed CSS stylesheet.</p>

    <!--
     This definition is not super-rigorous, but it doesn't need to be for now. If we ever start
     testing if something is a CSS module script in algorithms, instead of just referring to the
     concept, then we should consider adding a type item to the module script struct.
    -->
   </li>

   <li>
    <p>A <span>module script</span> is a <dfn export>JSON module script</dfn> if its <span
    data-x="concept-script-record">record</span> is a <span>Synthetic Module Record</span>, and it
    was created via the <span data-x="creating a JSON module script">create a JSON module
    script</span> algorithm. JSON module scripts represent a parsed JSON document.</p>

    <!--
     This definition is not super-rigorous, but it doesn't need to be for now. If we ever start
     testing if something is a JSON module script in algorithms, instead of just referring to the
     concept, then we should consider adding a type item to the module script struct.
    -->
   </li>

   <li>
    <p>A <span>module script</span> is a <dfn export>WebAssembly module script</dfn> if
    its <span data-x="concept-script-record">record</span> is a <span>WebAssembly Module
    Record</span>.</p>
   </li>
  </ul>

  <p class="note">As CSS stylesheets and JSON documents do not import dependent modules, and do not
  throw exceptions on evaluation, the <span data-x="concept-script-script-fetch-options">fetch
  options</span> and <span data-x="concept-script-base-url">base URL</span> of <span data-x="CSS
  module script">CSS module scripts</span> and <span data-x="JSON module script">JSON module
  scripts</span> and are always null.</p>

  <p>The <dfn>active script</dfn> is determined by the following algorithm:</p>

  <ol>
   <li><p>Let <var>record</var> be <span>GetActiveScriptOrModule</span>().</p></li>

   <li><p>If <var>record</var> is null, return null.</p></li>

   <li><p>Return <var>record</var>.[[HostDefined]].</p></li>
  </ol>

  <p class="note">The <span>active script</span> concept is so far only used by the
  <code>import()</code> feature, to determine the <span data-x="concept-script-base-url">base
  URL</span> to use for resolving relative module specifiers.</p>

  <h5 id="fetching-scripts">Fetching scripts</h5>

  <p>This section introduces a number of algorithms for fetching scripts, taking various necessary
  inputs and resulting in <span data-x="classic script">classic</span> or <span data-x="module
  script">module scripts</span>.</p>

  <hr>

  <p><dfn>Script fetch options</dfn> is a <span>struct</span> with the following <span
  data-x="struct item">items</span>:</p>

  <dl>
   <dt><dfn data-x="concept-script-fetch-options-nonce">cryptographic nonce</dfn></dt>

   <dd><p>The <span data-x="concept-request-nonce-metadata">cryptographic nonce metadata</span> used
   for the initial fetch and for fetching any imported modules</p></dd>

   <dt><dfn data-x="concept-script-fetch-options-integrity">integrity metadata</dfn></dt>

   <dd><p>The <span data-x="concept-request-integrity-metadata">integrity metadata</span> used for
   the initial fetch</dd>

   <dt><dfn data-x="concept-script-fetch-options-parser">parser metadata</dfn></dt>

   <dd><p>The <span data-x="concept-request-parser-metadata">parser metadata</span> used for the
   initial fetch and for fetching any imported modules</p></dd>

   <dt><dfn data-x="concept-script-fetch-options-credentials">credentials mode</dfn></dt>

   <dd><p>The <span data-x="concept-request-credentials-mode">credentials mode</span> used for the
   initial fetch (for <span data-x="module script">module scripts</span>) and for fetching any
   imported modules (for both <span data-x="module script">module scripts</span> and <span
   data-x="classic script">classic scripts</span>)</p></dd>

   <dt><dfn data-x="concept-script-fetch-options-referrer-policy">referrer policy</dfn></dt>

   <dd>
    <p>The <span data-x="concept-request-referrer-policy">referrer policy</span> used for the
    initial fetch and for fetching any imported modules</p>

    <p class="note">This policy can mutate after a <span>module script</span>'s <span
    data-x="concept-response">response</span> is received, to be the <span>referrer policy</span>
    <span data-x="parse-referrer-policy-header">parsed</span> from the <span
    data-x="concept-response">response</span>, and used when fetching any module dependencies.</p>
   </dd>

   <dt><dfn data-x="concept-script-fetch-options-render-blocking">render-blocking</dfn></dt>

   <dd><p>The boolean value of <span data-x="concept-request-render-blocking">render-blocking</span>
   used for the initial fetch and for fetching any imported modules. Unless otherwise stated, its
   value is false.</p></dd>

   <dt><dfn data-x="concept-script-fetch-options-fetch-priority">fetch priority</dfn></dt>

   <dd><p>The <span data-x="concept-request-priority">priority</span> used for the
   initial fetch</p></dd>

  </dl>

  <p class="note">Recall that via the <code>import()</code> feature, <span data-x="classic
  script">classic scripts</span> can import <span data-x="module script">module scripts</span>.</p>

  <p>The <dfn>default script fetch options</dfn> are a <span>script fetch options</span> whose <span
  data-x="concept-script-fetch-options-nonce">cryptographic nonce</span> is the empty string, <span
  data-x="concept-script-fetch-options-integrity">integrity metadata</span> is the empty string,
  <span data-x="concept-script-fetch-options-parser">parser metadata</span> is "<code
  data-x="">not-parser-inserted</code>", <span
  data-x="concept-script-fetch-options-credentials">credentials mode</span> is "<code
  data-x="">same-origin</code>", <span
  data-x="concept-script-fetch-options-referrer-policy">referrer policy</span> is the empty
  string, and <span data-x="concept-script-fetch-options-fetch-priority">fetch priority</span>
  is "<code data-x="">auto</code>".</p>

  <p>Given a <span data-x="concept-request">request</span> <var>request</var> and a <span>script
  fetch options</span> <var>options</var>, we define:</p>

  <dl>
   <dt><dfn>set up the classic script request</dfn></dt>
   <dd><p>Set <var>request</var>'s <span data-x="concept-request-nonce-metadata">cryptographic nonce
   metadata</span> to <var>options</var>'s <span
   data-x="concept-script-fetch-options-nonce">cryptographic nonce</span>, its <span
   data-x="concept-request-integrity-metadata">integrity metadata</span> to <var>options</var>'s
   <span data-x="concept-script-fetch-options-integrity">integrity metadata</span>, its <span
   data-x="concept-request-parser-metadata">parser metadata</span> to <var>options</var>'s <span
   data-x="concept-script-fetch-options-parser">parser metadata</span>, its <span
   data-x="concept-request-referrer-policy">referrer policy</span> to <var>options</var>'s <span
   data-x="concept-script-fetch-options-referrer-policy">referrer policy</span>, its
   <span data-x="concept-request-render-blocking">render-blocking</span> to <var>options</var>'s
   <span data-x="concept-script-fetch-options-render-blocking">render-blocking</span>, and its
   <span data-x="concept-request-priority">priority</span> to <var>options</var>'s
   <span data-x="concept-script-fetch-options-fetch-priority">fetch priority</span>.</p></dd>

   <dt><dfn>set up the module script request</dfn></dt>
   <dd><p>Set <var>request</var>'s <span data-x="concept-request-nonce-metadata">cryptographic nonce
   metadata</span> to <var>options</var>'s <span
   data-x="concept-script-fetch-options-nonce">cryptographic nonce</span>, its <span
   data-x="concept-request-integrity-metadata">integrity metadata</span> to <var>options</var>'s
   <span data-x="concept-script-fetch-options-integrity">integrity metadata</span>, its <span
   data-x="concept-request-parser-metadata">parser metadata</span> to <var>options</var>'s <span
   data-x="concept-script-fetch-options-parser">parser metadata</span>, its <span
   data-x="concept-request-credentials-mode">credentials mode</span> to <var>options</var>'s <span
   data-x="concept-script-fetch-options-credentials">credentials mode</span>, its <span
   data-x="concept-request-referrer-policy">referrer policy</span> to <var>options</var>'s <span
   data-x="concept-script-fetch-options-referrer-policy">referrer policy</span>, its
   <span data-x="concept-request-render-blocking">render-blocking</span> to <var>options</var>'s
   <span data-x="concept-script-fetch-options-render-blocking">render-blocking</span>, and its
   <span data-x="concept-request-priority">priority</span> to <var>options</var>'s
   <span data-x="concept-script-fetch-options-fetch-priority">fetch priority</span>.</p></dd>
  </dl>

  <p>To <dfn>get the descendant script fetch options</dfn> given a <span>script fetch options</span>
  <var>originalOptions</var>, a <span>URL</span> <var>url</var>, and an <span>environment settings
  object</span> <var>settingsObject</var>:</p>

  <ol>
   <li><p>Let <var>newOptions</var> be a copy of <var>originalOptions</var>.</p></li>

   <li><p>Let <var>integrity</var> be the result of <span>resolving a module integrity
   metadata</span> with <var>url</var> and <var>settingsObject</var>.</p></li>

   <li><p>Set <var>newOptions</var>'s <span
   data-x="concept-script-fetch-options-integrity">integrity metadata</span> to
   <var>integrity</var>.</p></li>

   <li><p>Set <var>newOptions</var>'s <span
   data-x="concept-script-fetch-options-fetch-priority">fetch priority</span> to "<code
   data-x="">auto</code>".</p></li>

   <li><p>Return <var>newOptions</var>.</p></li>
  </ol>

  <p>To <dfn data-x="resolving a module integrity metadata">resolve a module integrity
  metadata</dfn>, given a <span>URL</span> <var>url</var> and an <span>environment settings
  object</span> <var>settingsObject</var>:</p>

  <ol>
   <li><p>Let <var>map</var> be <var>settingsObject</var>'s <span
   data-x="concept-settings-object-global">global object</span>'s <span
   data-x="concept-global-import-map">import map</span>.</p></li>

   <li><p>If <var>map</var>'s <span
   data-x="concept-import-map-integrity">integrity</span>[<var>url</var>] does not <span
   data-x="map exists">exist</span>, then return the empty string.</p></li>

   <li><p>Return <var>map</var>'s <span
   data-x="concept-import-map-integrity">integrity</span>[<var>url</var>].</p></li>
  </ol>

  <hr>

  <p>Several of the below algorithms can be customized with a <dfn
  data-x="fetching-scripts-perform-fetch" export for="fetching scripts">perform the fetch
  hook</dfn> algorithm, which takes a <span data-x="concept-request">request</span>, a boolean <var
  data-x="fetching-scripts-is-top-level">isTopLevel</var>, and a <dfn export for="fetching
  scripts"><var
  data-x="fetching-scripts-processCustomFetchResponse">processCustomFetchResponse</var></dfn>
  algorithm. It runs <var
  data-x="fetching-scripts-processCustomFetchResponse">processCustomFetchResponse</var> with a <span
  data-x="concept-response">response</span> and either null (on failure) or a <span>byte
  sequence</span> containing the response body. <dfn export for="fetching scripts"><var
  data-x="fetching-scripts-is-top-level">isTopLevel</var></dfn> will be true for all <span>classic
  script</span> fetches, and for the initial fetch when <span data-x="fetch an external module
  script graph">fetching an external module script graph</span> or <span data-x="fetch a module
  worker script graph">fetching a module worker script graph</span>, but false for the fetches
  resulting from <code data-x="">import</code> statements encountered throughout the graph or from
  <code data-x="">import()</code> expressions.</p>

  <p>By default, not supplying a <span data-x="fetching-scripts-perform-fetch">perform the fetch
  hook</span> will cause the below algorithms to simply <span data-x="concept-fetch">fetch</span>
  the given <span data-x="concept-request">request</span>, with algorithm-specific customizations
  to the <span data-x="concept-request">request</span> and validations of the resulting <span
  data-x="concept-response">response</span>.</p>

  <p>To layer your own customizations on top of these algorithm-specific ones, supply a <span
  data-x="fetching-scripts-perform-fetch">perform the fetch hook</span> that modifies the given
  <span data-x="concept-request">request</span>, <span data-x="concept-fetch">fetches</span> it,
  and then performs specific validations of the resulting <span
  data-x="concept-response">response</span> (completing with a <span>network error</span> if the
  validations fail).</p>

  <p>The hook can also be used to perform more subtle customizations, such as keeping a cache of
  <span data-x="concept-response">responses</span> and avoiding performing a <span
  data-x="concept-fetch">fetch</span> at all.</p>

  <p class="note"><cite>Service Workers</cite> is an example of a specification that runs these
  algorithms with its own options for the hook. <ref>SW</ref></p>

  <hr>

  <p>Now for the algorithms themselves.</p>

  <p>To <dfn export>fetch a classic script</dfn> given a <span>URL</span> <var>url</var>, an
  <span>environment settings object</span> <var>settingsObject</var>, a <span>script fetch
  options</span> <var>options</var>, a <span data-x="CORS settings attribute">CORS settings
  attribute state</span> <var>corsSetting</var>, an <span>encoding</span> <var>encoding</var>, and
  an algorithm <var>onComplete</var>, run these steps. <var>onComplete</var> must be an algorithm
  accepting null (on failure) or a <span>classic script</span> (on success).</p>

  <ol>
   <li><p>Let <var>request</var> be the result of <span data-x="create a potential-CORS
   request">creating a potential-CORS request</span> given <var>url</var>, "<code
   data-x="">script</code>", and <var>corsSetting</var>.</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-client">client</span> to
   <var>settingsObject</var>.</p>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-initiator-type">initiator
   type</span> to "<code data-x="">script</code>".</p>

   <li><p><span>Set up the classic script request</span> given <var>request</var> and
   <var>options</var>.</p></li>

   <!--FETCH-->
   <li>
    <p><span data-x="concept-fetch">Fetch</span> <var>request</var> with
    the following <i data-x="processResponseConsumeBody">processResponseConsumeBody</i> steps given
    <span data-x="concept-response">response</span> <var>response</var> and null, failure, or a
    <span>byte sequence</span> <var>bodyBytes</var>:</p>

    <p class="note"><var>response</var> can be either <span>CORS-same-origin</span> or
    <span>CORS-cross-origin</span>. This only affects how error reporting happens.</p>

    <ol>
     <li><p>Set <var>response</var> to <var>response</var>'s <span>unsafe response</span>.</p></li>

     <li>
      <p>If any of the following are true:</p>

      <ul>
       <li><p><var>bodyBytes</var> is null or failure; or</p></li>

       <li><p><var>response</var>'s <span data-x="concept-response-status">status</span> is not an
       <span>ok status</span>,</p></li>
      </ul>

      <p>then run <var>onComplete</var> given null, and abort these steps.</p>

      <p class="note">For historical reasons, this algorithm does not include MIME type checking,
      unlike the other script-fetching algorithms in this section.</p>
     </li>

     <li><p>Let <var>potentialMIMETypeForEncoding</var> be the result of <span data-x="extract a
     MIME type">extracting a MIME type</span> given <var>response</var>'s <span
     data-x="concept-response-header-list">header list</span>.</p></li>

     <li>
      <p>Set <var>encoding</var> to the result of <span data-x="legacy extract an encoding">legacy
      extracting an encoding</span> given <var>potentialMIMETypeForEncoding</var> and
      <var>encoding</var>.</p>

      <p class="note">This intentionally ignores the <span>MIME type essence</span>.</p>
     </li>

     <li>
      <p>Let <var>sourceText</var> be the result of <span data-x="decode">decoding</span>
      <var>bodyBytes</var> to Unicode, using <var>encoding</var> as the fallback encoding.</p>

      <p class="note">The <span>decode</span> algorithm overrides <var>encoding</var> if the file
      contains a BOM.</p>
     </li>

     <li><p>Let <var>mutedErrors</var> be true if <var>response</var> was
     <span>CORS-cross-origin</span>, and false otherwise.</p></li>

     <li><p>Let <var>script</var> be the result of <span>creating a classic script</span> given
     <var>sourceText</var>, <var>settingsObject</var>, <var>response</var>'s <span
     data-x="concept-response-url">URL</span>, <var>options</var>,
     <var>mutedErrors</var>, and <var>url</var>.</p></li>

     <li>Run <var>onComplete</var> given <var>script</var>.</li>
    </ol>
   </li>
  </ol>

  <p>To <dfn export>fetch a classic worker script</dfn> given a <span>URL</span> <var>url</var>, an
  <span>environment settings object</span> <var>fetchClient</var>, a <span
  data-x="concept-request-destination">destination</span> <var>destination</var>, an
  <span>environment settings object</span> <var>settingsObject</var>, an algorithm
  <var>onComplete</var>, and an optional <span data-x="fetching-scripts-perform-fetch">perform the
  fetch hook</span> <var>performFetch</var>, run these steps. <var>onComplete</var> must be an
  algorithm accepting null (on failure) or a <span>classic script</span> (on success).</p>

  <ol>
   <li><p>Let <var>request</var> be a new <span data-x="concept-request">request</span> whose <span
   data-x="concept-request-url">URL</span> is <var>url</var>, <span
   data-x="concept-request-client">client</span> is <var>fetchClient</var>, <span
   data-x="concept-request-destination">destination</span> is <var>destination</var>, <span
   data-x="concept-request-initiator-type">initiator type</span> is "<code data-x="">other</code>",
   <span data-x="concept-request-mode">mode</span> is "<code data-x="">same-origin</code>", <span
   data-x="concept-request-credentials-mode">credentials mode</span> is "<code
   data-x="">same-origin</code>", <span data-x="concept-request-parser-metadata">parser
   metadata</span> is "<code data-x="">not parser-inserted</code>", and whose
   <span>use-URL-credentials flag</span> is set.</p></li>

   <!--FETCH-->
   <li>
    <p>If <var>performFetch</var> was given, run <var>performFetch</var> with <var>request</var>,
    true, and with <var>processResponseConsumeBody</var> as defined below.</p>

    <p>Otherwise, <span data-x="concept-fetch">fetch</span> <var>request</var> with <i
    data-x="processResponseConsumeBody">processResponseConsumeBody</i> set to
    <var>processResponseConsumeBody</var> as defined below.</p>

    <p>In both cases, let <var>processResponseConsumeBody</var> given <span
    data-x="concept-response">response</span> <var>response</var> and null, failure, or a <span>byte
    sequence</span> <var>bodyBytes</var> be the following algorithm:</p>

    <ol>
     <li><p>Set <var>response</var> to <var>response</var>'s <span>unsafe response</span>.</p></li>

     <li>
      <p>If any of the following are true:</p>

      <ul>
       <li><p><var>bodyBytes</var> is null or failure; or</p></li>

       <li><p><var>response</var>'s <span data-x="concept-response-status">status</span> is not an
       <span>ok status</span>,</p></li>
      </ul>

      <p>then run <var>onComplete</var> given null, and abort these steps.</p>
     </li>

     <li>
      <p>If all of the following are true:</p>

      <ul>
       <li><p><var>response</var>'s <span data-x="concept-response-url">URL</span>'s <span
       data-x="concept-url-scheme">scheme</span> is an <span>HTTP(S) scheme</span>; and</p></li>

       <li><p>the result of <span data-x="extract a MIME type">extracting a MIME type</span> from
       <var>response</var>'s <span data-x="concept-response-header-list">header list</span> is not
       a <span>JavaScript MIME type</span>,</p>
      </ul>

      <p>then run <var>onComplete</var> given null, and abort these steps.</p>

      <p class="note">Other <span data-x="fetch scheme">fetch schemes</span> are exempted from MIME
      type checking for historical web-compatibility reasons. We might be able to tighten this in
      the future; see <a href="https://github.com/whatwg/html/issues/3255">issue #3255</a>.</p>
     </li>

     <li><p>Let <var>sourceText</var> be the result of <span data-x="UTF-8 decode">UTF-8
     decoding</span> <var>bodyBytes</var>.</p></li>

     <li><p>Let <var>script</var> be the result of <span>creating a classic script</span> using
     <var>sourceText</var>, <var>settingsObject</var>, <var>response</var>'s <span
     data-x="concept-response-url">URL</span>, and the <span>default script fetch
     options</span>.</p></li>

     <li><p>Run <var>onComplete</var> given <var>script</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>To <dfn export>fetch a classic worker-imported script</dfn> given a <span>URL</span>
  <var>url</var>, an <span>environment settings object</span> <var>settingsObject</var>, and an
  optional <span data-x="fetching-scripts-perform-fetch">perform the fetch hook</span>
  <var>performFetch</var>, run these steps. The algorithm will return a <span>classic script</span>
  on success, or throw an exception on failure.</p>

  <ol>
   <li><p>Let <var>response</var> be null.</p></li>

   <li><p>Let <var>bodyBytes</var> be null.</p></li>

   <li><p>Let <var>request</var> be a new <span data-x="concept-request">request</span> whose <span
   data-x="concept-request-url">URL</span> is <var>url</var>, <span
   data-x="concept-request-client">client</span> is <var>settingsObject</var>, <span
   data-x="concept-request-destination">destination</span> is "<code data-x="">script</code>",
   <span data-x="concept-request-initiator-type">initiator type</span> is "<code
   data-x="">other</code>", <span data-x="concept-request-parser-metadata">parser metadata</span>
   is "<code data-x="">not parser-inserted</code>", and whose <span>use-URL-credentials flag</span>
   is set.</p></li>

   <li>
    <p>If <var>performFetch</var> was given, run <var>performFetch</var> with <var>request</var>,
    <var>isTopLevel</var>, and with <var>processResponseConsumeBody</var> as defined below.</p>

    <p>Otherwise, <span data-x="concept-fetch">fetch</span> <var>request</var> with <i
    data-x="processResponseConsumeBody">processResponseConsumeBody</i> set to
    <var>processResponseConsumeBody</var> as defined below.</p>

    <p>In both cases, let <var>processResponseConsumeBody</var> given <span
    data-x="concept-response">response</span> <var>res</var> and null, failure, or a
    <span>byte sequence</span> <var>bb</var> be the following algorithm:</p>

    <ol>
     <li><p>Set <var>bodyBytes</var> to <var>bb</var>.</p></li>

     <li><p>Set <var>response</var> to <var>res</var>.</p></li>
    </ol>
   </li>

   <li>
    <p><span>Pause</span> until <var>response</var> is not null.</p>

    <p class="note">Unlike other algorithms in this section, the fetching process is synchronous
    here.</p>
   </li>

   <li><p>Set <var>response</var> to <var>response</var>'s <span>unsafe response</span>.</p></li>

   <li>
    <p>If any of the following are true:</p>

    <ul>
     <li><p><var>bodyBytes</var> is null or failure;</p></li>

     <li><p><var>response</var>'s <span data-x="concept-response-status">status</span> is not an
     <span>ok status</span>; or</p></li>

     <li><p>the result of <span data-x="extract a MIME type">extracting a MIME type</span> from
     <var>response</var>'s <span data-x="concept-response-header-list">header list</span> is not a
     <span>JavaScript MIME type</span>,</p>
    </ul>

    <p>then throw a <span>"<code>NetworkError</code>"</span> <code>DOMException</code>.</p>
   </li>

   <li><p>Let <var>sourceText</var> be the result of <span data-x="UTF-8 decode">UTF-8
   decoding</span> <var>bodyBytes</var>.</p></li>

   <li><p>Let <var>mutedErrors</var> be true if <var>response</var> was
   <span>CORS-cross-origin</span>, and false otherwise.</p></li>

   <li><p>Let <var>script</var> be the result of <span>creating a classic script</span> given
   <var>sourceText</var>, <var>settingsObject</var>, <var>response</var>'s <span
   data-x="concept-response-url">URL</span>, the <span>default script fetch options</span>, and
   <var>mutedErrors</var>.</p></li>

   <li><p>Return <var>script</var>.</p></li>
  </ol>

  <p>To <dfn export id="fetch-a-module-script-tree">fetch an external module script graph</dfn>
  given a <span>URL</span> <var>url</var>, an <span>environment settings object</span>
  <var>settingsObject</var>, a <span>script fetch options</span> <var>options</var>, and an
  algorithm <var>onComplete</var>, run these steps. <var>onComplete</var> must be an algorithm
  accepting null (on failure) or a <span>module script</span> (on success).</p>

  <ol>
   <li>
    <p><span>Fetch a single module script</span> given <var>url</var>, <var>settingsObject</var>,
    "<code data-x="">script</code>", <var>options</var>, <var>settingsObject</var>, "<code
    data-x="">client</code>", true, and with the following steps given <var>result</var>:</p>

    <ol>
     <li><p>If <var>result</var> is null, run <var>onComplete</var> given null, and abort these
     steps.</p></li>

     <li><p><span data-x="fetch the descendants of and link a module script">Fetch the descendants of
     and link</span> <var>result</var> given <var>settingsObject</var>, "<code
     data-x="">script</code>", and <var>onComplete</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>To <dfn>fetch a modulepreload module script graph</dfn> given a <span>URL</span> <var>url</var>,
  a <span data-x="concept-request-destination">destination</span> <var>destination</var>, an
  <span>environment settings object</span> <var>settingsObject</var>, a <span>script fetch
  options</span> <var>options</var>, and an algorithm <var>onComplete</var>, run these steps.
  <var>onComplete</var> must be an algorithm accepting null (on failure) or a <span>module
  script</span> (on success).</p>

  <ol>
   <li>
    <p><span>Fetch a single module script</span> given <var>url</var>, <var>settingsObject</var>,
    <var>destination</var>, <var>options</var>, <var>settingsObject</var>, "<code
    data-x="">client</code>", true, and with the following steps given <var>result</var>:</p>

    <ol>
     <li><p>Run <var>onComplete</var> given <var>result</var>.</p></li>

     <li><p><span>Assert</span>: <var>settingsObject</var>'s <span
     data-x="concept-settings-object-global">global object</span> implements
     <code>Window</code>.</p></li>

     <li>
      <p>If <var>result</var> is not null, optionally <span data-x="fetch the descendants of and
      link a module script">fetch the descendants of and link</span> <var>result</var> given
      <var>settingsObject</var>, <var>destination</var>, and an empty algorithm.</p>

      <p class="note">Generally, performing this step will be beneficial for performance, as it
      allows pre-loading the modules that will invariably be requested later, via algorithms such as
      <span>fetch an external module script graph</span> that fetch the entire graph. However, user
      agents might wish to skip them in bandwidth-constrained situations, or situations where the
      relevant fetches are already in flight.</p>
     </li>
    </ol>
   </li>
  </ol>

  <p>To <dfn>fetch an inline module script graph</dfn> given a <span>string</span>
  <var>sourceText</var>, a <span>URL</span> <var>baseURL</var>, an <span>environment settings
  object</span> <var>settingsObject</var>, a <span>script fetch options</span> <var>options</var>,
  and an algorithm <var>onComplete</var>, run these steps. <var>onComplete</var> must be an
  algorithm accepting null (on failure) or a <span>module script</span> (on success).</p>

  <ol>
   <li><p>Let <var>script</var> be the result of <span>creating a JavaScript module script</span>
   using <var>sourceText</var>, <var>settingsObject</var>, <var>baseURL</var>, and
   <var>options</var>.</p></li>

   <li><p><span data-x="fetch the descendants of and link a module script">Fetch the descendants of
   and link</span> <var>script</var>, given <var>settingsObject</var>, "<code
   data-x="">script</code>", and <var>onComplete</var>.</p></li>
  </ol>

  <p>To <dfn export id="fetch-a-module-worker-script-tree">fetch a module worker script graph</dfn>
  given a <span>URL</span> <var>url</var>, an <span>environment settings object</span>
  <var>fetchClient</var>, a <span data-x="concept-request-destination">destination</span>
  <var>destination</var>, a <span data-x="concept-script-fetch-options-credentials">credentials
  mode</span> <var>credentialsMode</var>, an <span>environment settings object</span>
  <var>settingsObject</var>, and an algorithm <var>onComplete</var>, <span>fetch a worklet/module
  worker script graph</span> given <var>url</var>, <var>fetchClient</var>, <var>destination</var>,
  <var>credentialsMode</var>, <var>settingsObject</var>, and <var>onComplete</var>.</p>

  <p>To <dfn>fetch a worklet script graph</dfn> given a <span>URL</span> <var>url</var>, an
  <span>environment settings object</span> <var>fetchClient</var>, a <span
  data-x="concept-request-destination">destination</span> <var>destination</var>, a
  <span data-x="concept-script-fetch-options-credentials">credentials mode</span>
  <var>credentialsMode</var>, an <span>environment settings object</span>
  <var>settingsObject</var>, a <span data-x="concept-Worklet-module-responses-map">module responses
  map</span> <var>moduleResponsesMap</var>, and an algorithm <var>onComplete</var>, <span>fetch a
  worklet/module worker script graph</span> given <var>url</var>, <var>fetchClient</var>,
  <var>destination</var>, <var>credentialsMode</var>, <var>settingsObject</var>,
  <var>onComplete</var>, and the following <span data-x="fetching-scripts-perform-fetch">perform the
  fetch hook</span> given <var>request</var> and <i
  data-x="fetching-scripts-processCustomFetchResponse">processCustomFetchResponse</i>:</p>

  <ol>
   <li><p>Let <var>requestURL</var> be <var>request</var>'s <span
   data-x="concept-request-url">URL</span>.</p></li>

   <li><p>If <var>moduleResponsesMap</var>[<var>requestURL</var>] is "<code
   data-x="">fetching</code>", wait <span>in parallel</span> until that entry's value changes, then
   <span>queue a task</span> on the <span>networking task source</span> to proceed with running the
   following steps.</p></li>

   <li>
    <p>If <var>moduleResponsesMap</var>[<var>requestURL</var>] <span data-x="map
   exists">exists</span>, then:</p>

    <ol>
     <li><p>Let <var>cached</var> be <var>moduleResponsesMap</var>[<var>requestURL</var>].</p></li>

     <li><p>Run <var>processCustomFetchResponse</var> with <var>cached</var>[0] and
     <var>cached</var>[1].</p></li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li><p><span data-x="map set">Set</span> <var>moduleResponsesMap</var>[<var>requestURL</var>] to
   "<code data-x="">fetching</code>".</p></li>

   <li>
    <p><span data-x="concept-fetch">Fetch</span> <var>request</var>, with <i
    data-x="processResponseConsumeBody">processResponseConsumeBody</i> set to the following steps
    given <span data-x="concept-response">response</span> <var>response</var> and null, failure, or a
    <span>byte sequence</span> <var>bodyBytes</var>:</p>

    <ol>
     <li><p>Set <var>moduleResponsesMap</var>[<var>requestURL</var>] to (<var>response</var>,
     <var>bodyBytes</var>).</p></li>

     <li><p>Run <var>processCustomFetchResponse</var> with <var>response</var> and
     <var>bodyBytes</var>.</p></li>
    </ol>
   </li>
  </ol>

  <hr>

  <p>The following algorithms are meant for internal use by this specification only as part of <span
  data-x="fetch an external module script graph">fetching an external module script graph</span> or
  other similar concepts above, and should not be used directly by other specifications.</p>

  <p>This diagram illustrates how these algorithms relate to the ones above, as well as to each
  other:</p>

  <svg id="module-script-fetching-diagram" viewBox="0 0 941 166" role="img" aria-label="Fetch an external module script, fetch a modulepreload module script graph, fetch an inline module script graph, and fetch a module worker script graph all call fetch the descendants of and link a module script.">
   <defs>
    <marker id="module-script-fetching-diagram-arrow" orient="auto" markerWidth="6" markerHeight="6" refX="5" refY="3">
     <path d="M0,0 V6 L5,3 Z" style="fill: var(--text, CanvasText)"/>
    </marker>
   </defs>

   <g transform="translate(0.5,0.5)" class="caller">
    <rect width="180" height="50"/>
    <foreignObject width="180" height="50">
     <span xmlns="http://www.w3.org/1999/xhtml">fetch an external module script graph</span>
    </foreignObject>
   </g>
   <path d="M90.5,50.5 L470.5,140.5" marker-end="url(#module-script-fetching-diagram-arrow)"/>

   <g transform="translate(190.5,0.5)" class="caller">
    <rect width="180" height="50"/>
    <foreignObject width="180" height="50">
     <span xmlns="http://www.w3.org/1999/xhtml">fetch a modulepreload module script graph</span>
    </foreignObject>
   </g>
   <path d="M280.5,50.5 L470.5,140.5" marker-end="url(#module-script-fetching-diagram-arrow)"/>

   <g transform="translate(380.5,0.5)" class="caller">
    <rect width="180" height="50"/>
    <foreignObject width="180" height="50">
     <span xmlns="http://www.w3.org/1999/xhtml">fetch an inline module script graph</span>
    </foreignObject>
   </g>
   <path d="M470.5,50.5 L470.5,140.5" marker-end="url(#module-script-fetching-diagram-arrow)"/>

   <g transform="translate(570.5,0.5)" class="caller">
    <rect width="180" height="50"/>
    <foreignObject width="180" height="50">
     <span xmlns="http://www.w3.org/1999/xhtml">fetch a module worker script graph</span>
    </foreignObject>
   </g>
   <path d="M660.5,50.5 L755.5,70.5" marker-end="url(#module-script-fetching-diagram-arrow)"/>

   <g transform="translate(760.5,0.5)" class="caller">
    <rect width="180" height="50"/>
    <foreignObject width="180" height="50">
     <span xmlns="http://www.w3.org/1999/xhtml">fetch a worklet script graph</span>
    </foreignObject>
   </g>
   <path d="M850.5,50.5 L755.5,70.5" marker-end="url(#module-script-fetching-diagram-arrow)"/>

   <g transform="translate(665.5,70.5)" class="caller">
    <rect width="180" height="50"/>
    <foreignObject width="180" height="50">
     <span xmlns="http://www.w3.org/1999/xhtml">fetch a worklet/module worker script graph</span>
    </foreignObject>
   </g>
   <path d="M755.5,120.5 L470.5,140.5" marker-end="url(#module-script-fetching-diagram-arrow)"/>

   <g transform="translate(260.5,140.5)" class="subalgorithm">
    <rect width="420" height="25"/>
    <foreignObject width="420" height="25">
     <span xmlns="http://www.w3.org/1999/xhtml">fetch the descendants of and link a module script</span>
    </foreignObject>
   </g>
  </svg>

  <p>To <dfn>fetch a worklet/module worker script graph</dfn> given a <span>URL</span>
  <var>url</var>, an <span>environment settings object</span> <var>fetchClient</var>, a <span
  data-x="concept-request-destination">destination</span> <var>destination</var>, a <span
  data-x="concept-script-fetch-options-credentials">credentials mode</span>
  <var>credentialsMode</var>, an <span>environment settings object</span> <var>settingsObject</var>,
  an algorithm <var>onComplete</var>, and an optional <span
  data-x="fetching-scripts-perform-fetch">perform the fetch hook</span> <var>performFetch</var>, run
  these steps. <var>onComplete</var> must be an algorithm accepting null (on failure) or a
  <span>module script</span> (on success).</p>

  <ol>
   <li><p>Let <var>options</var> be a <span>script fetch options</span> whose <span
   data-x="concept-script-fetch-options-nonce">cryptographic nonce</span> is the empty string, <span
   data-x="concept-script-fetch-options-integrity">integrity metadata</span> is the empty string,
   <span data-x="concept-script-fetch-options-parser">parser metadata</span> is "<code
   data-x="">not-parser-inserted</code>", <span
   data-x="concept-script-fetch-options-credentials">credentials mode</span> is
   <var>credentialsMode</var>, <span data-x="concept-script-fetch-options-referrer-policy">referrer
   policy</span> is the empty string, and <span
   data-x="concept-script-fetch-options-fetch-priority">fetch priority</span> is "<code
   data-x="">auto</code>".</p>

   <li>
    <p><span>Fetch a single module script</span> given <var>url</var>,
    <var>fetchClient</var>, <var>destination</var>, <var>options</var>, <var>settingsObject</var>,
    "<code data-x="">client</code>", true, and <var>onSingleFetchComplete</var> as defined below. If
    <var>performFetch</var> was given, pass it along as well.</p>

    <p><var>onSingleFetchComplete</var> given <var>result</var> is the following algorithm:</p>

    <ol>
     <li><p>If <var>result</var> is null, run <var>onComplete</var> given null, and abort these
     steps.</p></li>

     <li><p><span data-x="fetch the descendants of and link a module script">Fetch the descendants
     of and link</span> <var>result</var> given <var>fetchClient</var>,
     <var>destination</var>, and <var>onComplete</var>. If <var>performFetch</var> was given, pass
     it along as well.</p></li>
    </ol>
   </li>
  </ol>

  <p>To <dfn data-x="fetch the descendants of and link a module script">fetch the descendants of and
  link</dfn> a <span>module script</span> <var>moduleScript</var>, given an <span>environment
  settings object</span> <var>fetchClient</var>, a <span
  data-x="concept-request-destination">destination</span> <var>destination</var>, an algorithm
  <var>onComplete</var>, and an optional <span data-x="fetching-scripts-perform-fetch">perform the
  fetch hook</span> <var>performFetch</var>, run these steps. <var>onComplete</var> must be an
  algorithm accepting null (on failure) or a <span>module script</span> (on success).</p>

  <ol>
   <li><p>Let <var>record</var> be <var>moduleScript</var>'s <span
   data-x="concept-script-record">record</span>.</p></li>

   <li>
    <p>If <var>record</var> is null, then:</p>

    <ol>
     <li><p>Set <var>moduleScript</var>'s <span data-x="concept-script-error-to-rethrow">error to
     rethrow</span> to <var>moduleScript</var>'s <span>parse error</span>.</p></li>

     <li><p>Run <var>onComplete</var> given <var>moduleScript</var>.</li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li><p>Let <var>state</var> be <span>Record</span> { [[ParseError]]: null, [[Destination]]:
   <var>destination</var>, [[PerformFetch]]: null, [[FetchClient]]: <var>fetchClient</var>
   }.</p></li>

   <li><p>If <var>performFetch</var> was given, set <var>state</var>.[[PerformFetch]] to
   <var>performFetch</var>.</p></li>

   <li>
    <p>Let <var>loadingPromise</var> be <var>record</var>.<span
    data-x="js-LoadRequestedModules">LoadRequestedModules</span>(<var>state</var>).</p>

    <p class="note">This step will recursively load all the module transitive dependencies.</p>
   </li>

   <li>
    <p><span>Upon fulfillment</span> of <var>loadingPromise</var>, run the following steps:</p>

    <ol>
     <li>
      <p>Perform <var>record</var>.<span data-x="js-Link">Link</span>().</p>

      <p class="note">This step will recursively call <span data-x="js-Link">Link</span> on all of
      the module's unlinked dependencies.</p>

      <p>If this throws an exception, catch it, and set <var>moduleScript</var>'s <span
      data-x="concept-script-error-to-rethrow">error to rethrow</span> to that exception.</p>
     </li>

     <li><p>Run <var>onComplete</var> given <var>moduleScript</var>.</p></li>
    </ol>
   </li>

   <li>
    <p><span>Upon rejection</span> of <var>loadingPromise</var>, run the
    following steps:</p>

    <ol>
     <li><p>If <var>state</var>.[[ParseError]] is not null, set <var>moduleScript</var>'s <span
     data-x="concept-script-error-to-rethrow">error to rethrow</span> to
     <var>state</var>.[[ParseError]] and run <var>onComplete</var> given
     <var>moduleScript</var>.</p></li>

     <li>
      <p>Otherwise, run <var>onComplete</var> given null.</p>

      <p class="note"><var>state</var>.[[ParseError]] is null when <var>loadingPromise</var> is
      rejected due to a loading error.</p>
     </li>
    </ol>
   </li>
  </ol>

  <p>To <dfn>fetch a single module script</dfn>, given a <span>URL</span> <var>url</var>, an
  <span>environment settings object</span> <var>fetchClient</var>, a <span
  data-x="concept-request-destination">destination</span> <var>destination</var>, a <span>script
  fetch options</span> <var>options</var>, an <span>environment settings object</span>
  <var>settingsObject</var>, a <span data-x="concept-request-referrer">referrer</span>
  <var>referrer</var>, an optional <span>ModuleRequest Record</span> <var>moduleRequest</var>, a
  boolean <var data-x="fetching-scripts-is-top-level">isTopLevel</var>, an algorithm
  <var>onComplete</var>, and an optional <span data-x="fetching-scripts-perform-fetch">perform the
  fetch hook</span> <var>performFetch</var>, run these steps. <var>onComplete</var> must be an
  algorithm accepting null (on failure) or a <span>module script</span> (on success).</p>

  <ol>
   <li><p>Let <var>moduleType</var> be "<code data-x="">javascript-or-wasm</code>".</p></li>

   <li><p>If <var>moduleRequest</var> was given, then set <var>moduleType</var> to the result of
   running the <span>module type from module request</span> steps given
   <var>moduleRequest</var>.</p></li>

   <li><p><span>Assert</span>: the result of running the <span>module type allowed</span> steps
   given <var>moduleType</var> and <var>settingsObject</var> is true. Otherwise, we would not have
   reached this point because a failure would have been raised when inspecting
   <var>moduleRequest</var>.[[Attributes]] in <a href="#validate-requested-module-specifiers">
   HostLoadImportedModule</a> or <span>fetch a single imported module script</span>.</p></li>

   <li><p>Let <var>moduleMap</var> be <var>settingsObject</var>'s <span
   data-x="concept-settings-object-module-map">module map</span>.</p></li>

   <li><p>If <var>moduleMap</var>[(<var>url</var>, <var>moduleType</var>)] is
   "<code data-x="">fetching</code>", wait <span>in parallel</span> until that entry's value
   changes, then <span>queue a task</span> on the <span>networking task source</span> to proceed
   with running the following steps.</p></li>

   <li><p>If <var>moduleMap</var>[(<var>url</var>, <var>moduleType</var>)] <span
   data-x="map exists">exists</span>, run <var>onComplete</var> given
   <var>moduleMap</var>[(<var>url</var>, <var>moduleType</var>)], and return.</p></li>

   <li><p><span data-x="map set">Set</span> <var>moduleMap</var>[(<var>url</var>,
   <var>moduleType</var>)] to "<code data-x="">fetching</code>".</p></li>

   <li><p>Let <var>request</var> be a new <span data-x="concept-request">request</span> whose
   <span data-x="concept-request-url">URL</span> is <var>url</var>, <span
   data-x="concept-request-mode">mode</span> is "<code data-x="">cors</code>", <span
   data-x="concept-request-referrer">referrer</span> is <var>referrer</var>, and <span
   data-x="concept-request-client">client</span> is <var>fetchClient</var>.</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-destination">destination</span> to
   the result of running the <span>fetch destination from module type</span> steps given
   <var>destination</var> and <var>moduleType</var>.</p></li>

   <!--
    "serviceworker" being included here is not redundant with the Service Worker spec's own
    same-origin check, because the request's mode also affects the Origin and Sec-Fetch-Mode
    headers.
   -->
   <li><p>If <var>destination</var> is "<code data-x="">worker</code>", "<code
   data-x="">sharedworker</code>", or "<code data-x="">serviceworker</code>", and
   <var>isTopLevel</var> is true, then set <var>request</var>'s <span
   data-x="concept-request-mode">mode</span> to "<code data-x="">same-origin</code>".</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-initiator-type">initiator
   type</span> to "<code data-x="">script</code>".</p>

   <li><p><span>Set up the module script request</span> given <var>request</var> and
   <var>options</var>.</p></li>

   <!--FETCH-->
   <li>
    <p>If <var>performFetch</var> was given, run <var>performFetch</var> with <var>request</var>,
    <var>isTopLevel</var>, and with <var>processResponseConsumeBody</var> as defined below.</p>

    <p>Otherwise, <span data-x="concept-fetch">fetch</span> <var>request</var> with <i
    data-x="processResponseConsumeBody">processResponseConsumeBody</i> set to
    <var>processResponseConsumeBody</var> as defined below.</p>

    <p>In both cases, let <var>processResponseConsumeBody</var> given <span
    data-x="concept-response">response</span> <var>response</var> and null, failure, or a
    <span>byte sequence</span> <var>bodyBytes</var> be the following algorithm:</p>

    <p class="note"><var>response</var> is always <span>CORS-same-origin</span>.</p>

    <ol>
     <li>
      <p>If any of the following are true:</p>

      <ul>
       <li><p><var>bodyBytes</var> is null or failure; or</p></li>

       <li><p><var>response</var>'s <span data-x="concept-response-status">status</span> is not an
       <span>ok status</span>,</p></li>
      </ul>

      <p>then <span data-x="map set">set</span> <var>moduleMap</var>[(<var>url</var>,
      <var>moduleType</var>)] to null, run <var>onComplete</var> given null, and abort these
      steps.</p>
     </li>

     <li><p>Let <var>mimeType</var> be the result of <span data-x="extract a MIME type">extracting
     a MIME type</span> from <var>response</var>'s <span
     data-x="concept-response-header-list">header list</span>.</p></li>

     <li><p>Let <var>moduleScript</var> be null.</p></li>

     <li><p>Let <var>referrerPolicy</var> be the result of <span
     data-x="parse-referrer-policy-header">parsing the `<code>Referrer-Policy</code>` header</span>
     given <var>response</var>. <ref>REFERRERPOLICY</ref></p></li>

     <li><p>If <var>referrerPolicy</var> is not the empty string, set <var>options</var>'s <span
     data-x="concept-script-fetch-options-referrer-policy">referrer policy</span> to
     <var>referrerPolicy</var>.</p></li>

     <li><p>If <var>mimeType</var>'s <span data-x="MIME type essence">essence</span> is
     "<code>application/wasm</code>" and <var>moduleType</var> is "<code
     data-x="">javascript-or-wasm</code>", then set <var>moduleScript</var> to the result of
     <span>creating a WebAssembly module script</span> given <var>bodyBytes</var>,
     <var>settingsObject</var>, <var>response</var>'s <span
     data-x="concept-response-url">URL</span>, and <var>options</var>.</p></li>

     <li>
      <p>Otherwise:</p>
      <ol>
       <li><p>Let <var>sourceText</var> be the result of <span data-x="UTF-8 decode">UTF-8
       decoding</span> <var>bodyBytes</var>.</p></li>

       <li><p>If <var>mimeType</var> is a <span>JavaScript MIME type</span> and <var>moduleType</var>
       is "<code data-x="">javascript-or-wasm</code>", then set <var>moduleScript</var> to the result of
       <span>creating a JavaScript module script</span> given <var>sourceText</var>,
       <var>settingsObject</var>, <var>response</var>'s <span
       data-x="concept-response-url">URL</span>, and <var>options</var>.</p></li>

       <li><p>If the <span>MIME type essence</span> of <var>mimeType</var> is "<code>text/css</code>"
       and <var>moduleType</var> is "<code data-x="">css</code>", then set <var>moduleScript</var> to
       the result of <span>creating a CSS module script</span> given <var>sourceText</var> and
       <var>settingsObject</var>.</p></li>

       <li><p>If <var>mimeType</var> is a <span>JSON MIME type</span> and
       <var>moduleType</var> is "<code data-x="">json</code>", then set <var>moduleScript</var> to
       the result of <span>creating a JSON module script</span> given <var>sourceText</var> and
       <var>settingsObject</var>.</p></li>
      </ol>
     </li>

     <li>
      <p><span data-x="map set">Set</span> <var>moduleMap</var>[(<var>url</var>,
      <var>moduleType</var>)] to <var>moduleScript</var>, and run <var>onComplete</var> given
      <var>moduleScript</var>.</p>

      <p class="note">It is intentional that the <span>module map</span> is keyed by the <span
      data-x="concept-request-url">request URL</span>, whereas the <span
      data-x="concept-script-base-url">base URL</span> for the <span>module script</span> is
      set to the <span data-x="concept-response-url">response URL</span>. The former is used to
      deduplicate fetches, while the latter is used for URL resolution.</p>
     </li>
    </ol>
   </li>
  </ol>

  <p>To <dfn>fetch a single imported module script</dfn>, given a <span>URL</span> <var>url</var>,
  an <span>environment settings object</span> <var>fetchClient</var>, a <span
  data-x="concept-request-destination">destination</span> <var>destination</var>, a <span>script
  fetch options</span> <var>options</var>, <span>environment settings object</span>
  <var>settingsObject</var>, a <span data-x="concept-request-referrer">referrer</span>
  <var>referrer</var>, a <span>ModuleRequest Record</span> <var>moduleRequest</var>, an
  algorithm <var>onComplete</var>, and an optional <span
  data-x="fetching-scripts-perform-fetch">perform the fetch hook</span> <var>performFetch</var>, run
  these steps. <var>onComplete</var> must be an algorithm accepting null (on failure) or a
  <span>module script</span> (on success).</p>

  <ol>
   <li><p><span>Assert</span>: <var>moduleRequest</var>.[[Attributes]] does not contain any
   <span>Record</span> <var>entry</var> such that <var>entry</var>.[[Key]] is not "<code
   data-x="">type</code>", because we only asked for "<code data-x="">type</code>" attributes in
   <span>HostGetSupportedImportAttributes</span>.</p></li>

   <li><p>Let <var>moduleType</var> be the result of running the <span>module type from module
   request</span> steps given <var>moduleRequest</var>.</p></li>

   <li><p>If the result of running the <span>module type allowed</span> steps given
   <var>moduleType</var> and <var>settingsObject</var> is false, then run <var>onComplete</var>
   given null, and return.</p></li>

   <li><p><span>Fetch a single module script</span> given <var>url</var>, <var>fetchClient</var>,
   <var>destination</var>, <var>options</var>, <var>settingsObject</var>, <var>referrer</var>,
   <var>moduleRequest</var>, false, and <var>onComplete</var>. If <var>performFetch</var> was given,
   pass it along as well.</p></li>
  </ol>

  <h5 id="creating-scripts">Creating scripts</h5>

  <p>To <dfn data-x="creating a classic script">create a classic script</dfn>, given a
  <span>string</span> <var>source</var>, an <span>environment settings object</span>
  <var>settings</var>, a <span>URL</span> <var>baseURL</var>, a <span>script fetch options</span>
  <var>options</var>, an optional boolean <var>mutedErrors</var> (default false), and an optional
  <span>URL</span>-or-null <var>sourceURLForWindowScripts</var> (default null):</p>

  <ol>
   <li>
    <p>If <var>mutedErrors</var> is true, then set <var>baseURL</var> to
    <code>about:blank</code>.</p>

    <p class="note">When <var>mutedErrors</var> is true, <var>baseURL</var> is the script's
    <span>CORS-cross-origin</span> <span data-x="concept-response">response</span>'s <span
    data-x="concept-response-url">url</span>, which shouldn't be exposed to JavaScript. Therefore,
    <var>baseURL</var> is sanitized here.</p>
   </li>

   <li><p>If <span data-x="concept-environment-noscript">scripting is disabled</span> for
   <var>settings</var>, then set <var>source</var> to the empty string.</p></li>

   <li><p>Let <var>script</var> be a new <span>classic script</span> that this algorithm will
   subsequently initialize.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-settings-object">settings
   object</span> to <var>settings</var>.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-base-url">base URL</span> to
   <var>baseURL</var>.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-script-fetch-options">fetch
   options</span> to <var>options</var>.</p></li>

   <li><p>Set <var>script</var>'s <span>muted errors</span> to <var>mutedErrors</var>.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-parse-error">parse error</span> and
   <span data-x="concept-script-error-to-rethrow">error to rethrow</span> to null.</p></li>

   <li><p><span>Record classic script creation time</span> given <var>script</var> and
   <var>sourceURLForWindowScripts</var>.</p></li>

   <li>
    <p>Let <var>result</var> be <span data-x="js-ParseScript">ParseScript</span>(<var>source</var>,
    <var>settings</var>'s <span data-x="environment settings object's realm">realm</span>,
    <var>script</var>).</p>

    <p class="note">Passing <var>script</var> as the last parameter here ensures
    <var>result</var>.[[HostDefined]] will be <var>script</var>.</p>
   </li>

   <li>
    <p>If <var>result</var> is a <span>list</span> of errors, then:

    <ol>
     <li><p>Set <var>script</var>'s <span data-x="concept-script-parse-error">parse error</span> and
     its <span data-x="concept-script-error-to-rethrow">error to rethrow</span> to
     <var>result</var>[0].</p></li>

     <li><p>Return <var>script</var>.</p></li>
    </ol>
   </li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-record">record</span> to
   <var>result</var>.</p></li>

   <li><p>Return <var>script</var>.</p></li>
  </ol>

  <p>To <dfn data-x="creating a JavaScript module script">create a JavaScript module script</dfn>,
  given a <span>string</span> <var>source</var>, an <span>environment settings
  object</span> <var>settings</var>, a <span>URL</span> <var>baseURL</var>, and a <span>script fetch
  options</span> <var>options</var>:</p>

  <ol>
   <li><p>If <span data-x="concept-environment-noscript">scripting is disabled</span> for
   <var>settings</var>, then set <var>source</var> to the empty string.</p></li>

   <li><p>Let <var>script</var> be a new <span>module script</span> that this algorithm will
   subsequently initialize.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-settings-object">settings
   object</span> to <var>settings</var>.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-base-url">base URL</span> to
   <var>baseURL</var>.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-script-fetch-options">fetch
   options</span> to <var>options</var>.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-parse-error">parse error</span> and
   <span data-x="concept-script-error-to-rethrow">error to rethrow</span> to null.</p></li>

   <li>
    <p>Let <var>result</var> be <span data-x="js-ParseModule">ParseModule</span>(<var>source</var>,
    <var>settings</var>'s <span data-x="environment settings object's realm">realm</span>,
    <var>script</var>).</p>

    <p class="note">Passing <var>script</var> as the last parameter here ensures
    <var>result</var>.[[HostDefined]] will be <var>script</var>.</p>
   </li>

   <li>
    <p>If <var>result</var> is a <span>list</span> of errors, then:</p>

    <ol>
     <li><p>Set <var>script</var>'s <span data-x="concept-script-parse-error">parse error</span> to
     <var>result</var>[0].</p></li>

     <li><p>Return <var>script</var>.</p></li>
    </ol>
   </li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-record">record</span> to
   <var>result</var>.</p></li>

   <li><p>Return <var>script</var>.</p></li>
  </ol>

  <p>To <dfn data-x="creating a WebAssembly module script">create a WebAssembly module script</dfn>,
  given a <span>byte sequence</span> <var>bodyBytes</var>, an <span>environment settings
  object</span> <var>settings</var>, a <span>URL</span> <var>baseURL</var>, and a <span>script fetch
  options</span> <var>options</var>:</p>

  <ol>
   <li>
    <p>If <span data-x="concept-environment-noscript">scripting is disabled</span> for
    <var>settings</var>, then set <var>bodyBytes</var> to the byte sequence 0x00 0x61 0x73 0x6d 0x01
    0x00 0x00 0x00.</p>

    <p class="note">This byte sequence corresponds to an empty WebAssembly module with only the magic
    bytes and version number provided.</p>
   </li>

   <li><p>Let <var>script</var> be a new <span>module script</span> that this algorithm will
   subsequently initialize.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-settings-object">settings
   object</span> to <var>settings</var>.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-base-url">base URL</span> to
   <var>baseURL</var>.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-script-fetch-options">fetch
   options</span> to <var>options</var>.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-parse-error">parse error</span> and
   <span data-x="concept-script-error-to-rethrow">error to rethrow</span> to null.</p></li>

   <li>
    <p>Let <var>result</var> be the result of <span data-x="parse a WebAssembly module">parsing a
    WebAssembly module</span> given <var>bodyBytes</var>,
    <var>settings</var>'s <span data-x="environment settings object's realm">realm</span>, and
    <var>script</var>.</p>

    <p class="note">Passing <var>script</var> as the last parameter here ensures
    <var>result</var>.[[HostDefined]] will be <var>script</var>.</p>
   </li>

   <li>
    <p>If the previous step threw an error <var>error</var>, then:</p>

    <ol>
     <li><p>Set <var>script</var>'s <span data-x="concept-script-parse-error">parse error</span> to
     <var>error</var>.</p></li>

     <li><p>Return <var>script</var>.</p></li>
    </ol>
   </li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-record">record</span> to
   <var>result</var>.</p></li>

   <li><p>Return <var>script</var>.</p></li>
  </ol>

  <p class="note"><cite>WebAssembly JavaScript Interface: ESM Integration</cite> specifies the
  hooks for the WebAssembly integration with ECMA-262 module loading. This includes support both
  for direct dependency imports, as well as for source phase imports, which support virtualization
  and multi-instantiation. <ref>WASMESM</ref></p>

  <p>To <dfn data-x="creating a CSS module script">create a CSS module script</dfn>, given a
  string <var>source</var> and an <span>environment settings object</span> <var>settings</var>:</p>

  <ol>
   <li><p>Let <var>script</var> be a new <span>module script</span> that this algorithm will
   subsequently initialize.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-settings-object">settings
   object</span> to <var>settings</var>.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-base-url">base URL</span> and
   <span data-x="concept-script-script-fetch-options">fetch options</span> to null.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-parse-error">parse error</span> and
   <span data-x="concept-script-error-to-rethrow">error to rethrow</span> to null.</p></li>

   <li><p>Let <var>sheet</var> be the result of running the steps to <span>create a constructed
   <code>CSSStyleSheet</code></span> with an empty dictionary as the argument.</p></li>

   <li>
    <p>Run the steps to <span>synchronously replace the rules of a <code>CSSStyleSheet</code></span>
    on <var>sheet</var> given <var>source</var>.</p>

    <p>If this throws an exception, catch it, and set <var>script</var>'s <span
    data-x="concept-script-parse-error">parse error</span> to that exception, and return
    <var>script</var>.</p>

    <p class="note">The steps to <span>synchronously replace the rules of a
    <code>CSSStyleSheet</code></span> will throw if <var>source</var> contains any <code
    data-x="">@import</code> rules.  This is by-design for now because there is not yet an
    agreement on how to handle these for CSS module scripts; therefore they are blocked altogether
    until a consensus is reached.</p>
   </li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-record">record</span> to the result
   of <span>CreateDefaultExportSyntheticModule</span>(<var>sheet</var>).</p></li>

   <li><p>Return <var>script</var>.</p></li>
  </ol>

  <p>To <dfn data-x="creating a JSON module script">create a JSON module script</dfn>, given a
  string <var>source</var> and an <span>environment settings object</span> <var>settings</var>:</p>

  <ol>
   <li><p>Let <var>script</var> be a new <span>module script</span> that this algorithm will
   subsequently initialize.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-settings-object">settings
   object</span> to <var>settings</var>.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-base-url">base URL</span> and
   <span data-x="concept-script-script-fetch-options">fetch options</span> to null.</p></li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-parse-error">parse error</span> and
   <span data-x="concept-script-error-to-rethrow">error to rethrow</span> to null.</p></li>

   <li>
    <p>Let <var>result</var> be <span>ParseJSONModule</span>(<var>source</var>).</p>

    <p>If this throws an exception, catch it, and set <var>script</var>'s <span
    data-x="concept-script-parse-error">parse error</span> to that exception, and return
    <var>script</var>.</p>
   </li>

   <li><p>Set <var>script</var>'s <span data-x="concept-script-record">record</span> to
   <var>result</var>.</p></li>

   <li><p>Return <var>script</var>.</p></li>
  </ol>

  <p>The <dfn>module type from module request</dfn> steps, given a <span>ModuleRequest
  Record</span> <var>moduleRequest</var>, are as follows:</p>

  <ol>
   <li><p>Let <var>moduleType</var> be "<code data-x="">javascript-or-wasm</code>".</p></li>

   <li>
    <p>If <var>moduleRequest</var>.[[Attributes]] has a <span>Record</span> <var>entry</var> such
    that <var>entry</var>.[[Key]] is "<code data-x="">type</code>", then:</p>

    <ol>
     <li>
      <p>If <var>entry</var>.[[Value]] is "<code data-x="">javascript-or-wasm</code>", then set
      <var>moduleType</var> to null.</p>

      <p class="note">This specification uses the "<code data-x="">javascript-or-wasm</code>" module
      type internally for <span data-x="JavaScript module script">JavaScript module scripts</span> or
      <span data-x="WebAssembly module script">WebAssembly module scripts</span>, so this step is
      needed to prevent modules from being imported using a
      "<code data-x="">javascript-or-wasm</code>" type attribute (a null <var>moduleType</var> will
      cause the <span>module type allowed</span> check to fail).</p>
     </li>

     <li><p>Otherwise, set <var>moduleType</var> to <var>entry</var>.[[Value]].</p></li>
    </ol>
   </li>

   <li><p>Return <var>moduleType</var>.</p></li>
  </ol>

  <p>The <dfn>module type allowed</dfn> steps, given a <span>string</span> <var>moduleType</var>
  and an <span>environment settings object</span> <var>settings</var>, are as follows:</p>

  <ol>
   <li><p>If <var>moduleType</var> is not "<code data-x="">javascript-or-wasm</code>", "<code
   data-x="">css</code>", or "<code data-x="">json</code>", then return false.</p></li>

   <li><p>If <var>moduleType</var> is "<code data-x="">css</code>" and the
   <code>CSSStyleSheet</code> interface is not <span data-x="idl-exposed">exposed</span> in
   <var>settings</var>'s <span data-x="environment settings object's realm">realm</span>, then
   return false.</p></li>

   <li><p>Return true.</p></li>
  </ol>

  <p>The <dfn>fetch destination from module type</dfn> steps, given a <span
  data-x="concept-request-destination">destination</span> <var>defaultDestination</var> and a
  <span>string</span> <var>moduleType</var>, are as follows:</p>

  <ol>
   <li>If <var>moduleType</var> is "<code data-x="">json</code>", then return "<code
   data-x="">json</code>".</li>

   <li>If <var>moduleType</var> is "<code data-x="">css</code>", then return "<code
   data-x="">style</code>".</li>

   <li>Return <var>defaultDestination</var>.</li>
  </ol>

  <h5 id="calling-scripts">Calling scripts</h5>

  <p>To <dfn export>run a classic script</dfn> given a <span>classic script</span> <var>script</var>
  and an optional boolean <var>rethrow errors</var> (default false):</p>

  <ol>
   <li><p>Let <var>settings</var> be the <span data-x="concept-script-settings-object">settings
   object</span> of <var>script</var>.</p></li>

   <li><p><span>Check if we can run script</span> with <var>settings</var>. If this returns "do
   not run" then return <span>NormalCompletion</span>(empty).</p></li>

   <li><p><span>Record classic script execution start time</span> given <var>script</var>.</p></li>

   <li><p><span>Prepare to run script</span> given <var>settings</var>.</p></li>

   <li><p>Let <var>evaluationStatus</var> be null.</p></li>

   <li><p>If <var>script</var>'s <span data-x="concept-script-error-to-rethrow">error to
   rethrow</span> is not null, then set <var>evaluationStatus</var> to Completion { [[Type]]: throw,
   [[Value]]: <var>script</var>'s <span data-x="concept-script-error-to-rethrow">error to
   rethrow</span>, [[Target]]: empty }.</p></li>

   <li>
    <p>Otherwise, set <var>evaluationStatus</var> to <span
    data-x="js-ScriptEvaluation">ScriptEvaluation</span>(<var>script</var>'s <span
    data-x="concept-script-record">record</span>).</p>

    <p>If <span data-x="js-ScriptEvaluation">ScriptEvaluation</span> does not complete because the
    user agent has <span data-x="abort a running script">aborted the running script</span>, leave
    <var>evaluationStatus</var> as null.</p>
   </li>

   <li>
    <p>If <var>evaluationStatus</var> is an <span data-x="Completion Record">abrupt
    completion</span>, then:</p>

    <ol>
     <li>
      <p>If <var>rethrow errors</var> is true and <var>script</var>'s <span>muted
      errors</span> is false, then:</p>

      <ol>
       <li><p><span>Clean up after running script</span> with <var>settings</var>.</p></li>

       <li><p>Rethrow <var>evaluationStatus</var>.[[Value]].</p></li>
      </ol>

     <li>
      <p>If <var>rethrow errors</var> is true and <var>script</var>'s <span>muted
      errors</span> is true, then:</p>

      <ol>
       <li><p><span>Clean up after running script</span> with <var>settings</var>.</p></li>

       <li><p>Throw a <span>"<code>NetworkError</code>"</span> <code>DOMException</code>.</p></li>
      </ol>
     </li>

     <li>
      <p>Otherwise, <var>rethrow errors</var> is false. Perform the following steps:</p>

      <ol>
       <li><p><span>Report an exception</span> given by <var>evaluationStatus</var>.[[Value]] for
       <var>script</var>'s <span data-x="concept-script-settings-object">settings object</span>'s
       <span data-x="concept-settings-object-global">global object</span>.</p></li>

       <li><p><span>Clean up after running script</span> with <var>settings</var>.</p></li>

       <li><p>Return <var>evaluationStatus</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p><span>Clean up after running script</span> with <var>settings</var>.</p></li>

   <li><p>If <var>evaluationStatus</var> is a normal completion, then return
   <var>evaluationStatus</var>.</p></li>

   <li><p>If we've reached this point, <var>evaluationStatus</var> was left as null because the
   script was <span data-x="abort a running script">aborted prematurely</span> during evaluation.
   Return Completion { [[Type]]: throw, [[Value]]: a new
   <span>"<code>QuotaExceededError</code>"</span> <code>DOMException</code>, [[Target]]: empty }.
   </p></li>
  </ol>

  <p>To <dfn export>run a module script</dfn> given a <span>module script</span> <var>script</var>
  and an optional boolean <var>preventErrorReporting</var> (default false):</p>

  <ol>
   <li><p>Let <var>settings</var> be the <span data-x="concept-script-settings-object">settings
   object</span> of <var>script</var>.</p></li>

   <li><p><span>Check if we can run script</span> with <var>settings</var>. If this returns "do not
   run", then return <span>a promise resolved with</span> undefined.</p></li>

   <li><p><span>Record module script execution start time</span> given <var>script</var>.</p></li>

   <li><p><span>Prepare to run script</span> given <var>settings</var>.</p></li>

   <li><p>Let <var>evaluationPromise</var> be null.</p></li>

   <li><p>If <var>script</var>'s <span data-x="concept-script-error-to-rethrow">error to
   rethrow</span> is not null, then set <var>evaluationPromise</var> to <span>a promise rejected
   with</span> <var>script</var>'s <span data-x="concept-script-error-to-rethrow">error to
   rethrow</span>.</p></li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p>Let <var>record</var> be <var>script</var>'s <span
     data-x="concept-script-record">record</span>.</p>

     <li>
      <p>Set <var>evaluationPromise</var> to <var>record</var>.<span
      data-x="js-Evaluate">Evaluate</span>().</p>

      <p class="note">This step will recursively evaluate all of the module's dependencies.</p>

      <p>If <span data-x="js-Evaluate">Evaluate</span> fails to complete as a result of the user agent
      <span data-x="abort a running script">aborting the running script</span>, then set
      <var>evaluationPromise</var> to <span>a promise rejected with</span> a new
      <span>"<code>QuotaExceededError</code>"</span> <code>DOMException</code>.</p>
      </li>
    </ol>
   </li>

   <li><p>If <var>preventErrorReporting</var> is false, then <span>upon rejection</span> of
   <var>evaluationPromise</var> with <var>reason</var>, <span>report an exception</span> given by
   <var>reason</var> for <var>script</var>'s <span data-x="concept-script-settings-object">settings
   object</span>'s <span data-x="concept-settings-object-global">global object</span>.</p></li>

   <li><p><span>Clean up after running script</span> with <var>settings</var>.</p></li>

   <li><p>Return <var>evaluationPromise</var>.</p></li>
  </ol>

  <p>The steps to <dfn>check if we can run script</dfn> with an <span>environment settings
  object</span> <var>settings</var> are as follows. They return either "run" or "do not run".</p>

  <ol>
   <li><p>If the <span data-x="concept-settings-object-global">global object</span> specified by
   <var>settings</var> is a <code>Window</code> object whose <code>Document</code> object is not
   <span>fully active</span>, then return "do not run".</p>

   <li><p>If <span data-x="concept-environment-noscript">scripting is disabled</span> for
   <var>settings</var>, then return "do not run".</p>

   <li><p>Return "run".</p></li>
  </ol>

  <p>The steps to <dfn export>prepare to run script</dfn> with an <span>environment settings
  object</span> <var>settings</var> are as follows:</p>

  <ol>
   <li><p>Push <var>settings</var>'s <span>realm execution context</span> onto the <span>JavaScript
   execution context stack</span>; it is now the <span>running JavaScript execution
   context</span>.</p></li>

   <li><p>Add <var>settings</var> to the <span>surrounding agent</span>'s <span
   data-x="concept-agent-event-loop">event loop</span>'s <span>currently running task</span>'s
   <span>script evaluation environment settings object set</span>.</p></li>
  </ol>

  <p>The steps to <dfn export>clean up after running script</dfn> with an <span>environment settings
  object</span> <var>settings</var> are as follows:</p>

  <ol>
   <li><p><span>Assert</span>: <var>settings</var>'s <span>realm execution context</span> is the
   <span>running JavaScript execution context</span>.</p></li>

   <li><p>Remove <var>settings</var>'s <span>realm execution context</span> from the
   <span>JavaScript execution context stack</span>.</p></li>

   <li><p>If the <span>JavaScript execution context stack</span> is now empty, <span>perform a
   microtask checkpoint</span>. (If this runs scripts, these algorithms will be invoked
   reentrantly.)</p></li>
  </ol>

  <p class="note">These algorithms are not invoked by one script directly calling another, but they
  can be invoked reentrantly in an indirect manner, e.g. if a script dispatches an event which has
  event listeners registered.</p>

  <p>The <dfn>running script</dfn> is the <span data-x="concept-script">script</span> in the
  [[HostDefined]] field in the ScriptOrModule component of the <span>running JavaScript execution
  context</span>.</p>


  <h5>Killing scripts</h5>

  <p>Although the JavaScript specification does not account for this possibility, it's sometimes
  necessary to <dfn data-lt="abort a running script|abort the running script" export>abort a
  running script</dfn>. This causes any <span data-x="js-ScriptEvaluation">ScriptEvaluation</span>
  or <span>Source Text Module Record</span> <span data-x="js-Evaluate">Evaluate</span> invocations
  to cease immediately, emptying the <span>JavaScript execution context stack</span> without
  triggering any of the normal mechanisms like <code data-x="">finally</code> blocks.
  <ref>JAVASCRIPT</ref></p>

  <p>User agents may impose resource limitations on scripts, for example CPU quotas, memory limits,
  total execution time limits, or bandwidth limitations. When a script exceeds a limit, the user
  agent may either throw a <span>"<code>QuotaExceededError</code>"</span> <code>DOMException</code>,
  <span data-x="abort a running script">abort the script</span> without an exception, prompt the
  user, or throttle script execution.</p>

  <div class="example">

   <p>For example, the following script never terminates. A user agent could, after waiting for a
   few seconds, prompt the user to either terminate the script or let it continue.</p>

   <pre><code class="html">&lt;script>
 while (true) { /* loop */ }
&lt;/script></code></pre>

  </div>

  <p>User agents are encouraged to allow users to disable scripting whenever the user is prompted
  either by a script (e.g. using the <code data-x="dom-alert">window.alert()</code> API) or because
  of a script's actions (e.g. because it has exceeded a time limit).</p>

  <p>If scripting is disabled while a script is executing, the script should be terminated
  immediately.</p>

  <p>User agents may allow users to specifically disable scripts just for the purposes of closing a
  <span>browsing context</span>.</p>

  <p class="example">For example, the prompt mentioned in the example above could also offer the
  user with a mechanism to just close the page entirely, without running any <code
  data-x="event-unload">unload</code> event handlers.</p>

  </div>

  <h5 id="runtime-script-errors">Runtime script errors</h5>

  <dl class="domintro">
   <dt><code data-x="">self.<span subdfn data-x="dom-reportError">reportError</span>(<var>e</var>)</code></dt>
   <dd><p>Dispatches an <code data-x="event-error">error</code> event at the global object for the
   given value <var>e</var>, in the same fashion as an unhandled exception.</p></dd>
  </dl>

  <p w-dev>In various scenarios, the user agent can report an exception by firing an <code
  data-x="event-error">error</code> event at the <code>Window</code>. If this event is not canceled,
  then the error is considered not handled, and can be reported to the developer console.</p>

  <div w-nodev>

  <p>To <dfn data-x="extract-error">extract error information</dfn> from a JavaScript value
  <var>exception</var>:</p>

  <ol>
   <li><p>Let <var>attributes</var> be an empty <span data-x="ordered map">map</span> keyed by IDL
   attributes.</p></li>

   <li><p>Set <var>attributes</var>[<code data-x="dom-ErrorEvent-error">error</code>] to
   <var>exception</var>.</p></li>

   <li>
    <p>
     <!-- INSERT TRACKING-->
     Set <var>attributes</var>[<code data-x="dom-ErrorEvent-message">message</code>],
     <var>attributes</var>[<code data-x="dom-ErrorEvent-filename">filename</code>],
     <var>attributes</var>[<code data-x="dom-ErrorEvent-lineno">lineno</code>], and
     <var>attributes</var>[<code data-x="dom-ErrorEvent-colno">colno</code>] to
     <span>implementation-defined</span> values derived from <var>exception</var>.
    </p>

    <p class="note">Browsers implement behavior not specified here or in the JavaScript
    specification to gather values which are helpful, including in unusual cases (e.g., <code
    data-x="">eval</code>). In the future, this might be specified in greater detail.</p>
   </li>

   <li><p>Return <var>attributes</var>.</p></li>
  </ol>

  <p id="runtime-script-errors-in-documents"><span id="report-the-error"></span>To <dfn
  export>report an exception</dfn> <var>exception</var> which is a JavaScript value, for a
  particular <span>global object</span> <var>global</var> and optional boolean <dfn
  data-x="report-exception-omitError"><var>omitError</var></dfn> (default false):</p>

  <ol>
   <li><p>Let <var>notHandled</var> be true.</p></li>

   <li><p>Let <var>errorInfo</var> be the result of <span data-x="extract-error">extracting error
   information</span> from <var>exception</var>.</p></li>

   <li>
    <p>Let <var>script</var> be a <span data-x="concept-script">script</span> found in an
    <span>implementation-defined</span> way, or null. This should usually be the <span>running
    script</span> (most notably during <span>run a classic script</span>).</p>

    <p class="note">Implementations have not yet settled on interoperable behavior for which script
    is used to determine whether errors are muted in less common cases.</p>
   </li>

   <li><p>If <var>script</var> is a <span>classic script</span> and <var>script</var>'s <span>muted
   errors</span> is true, then set <var>errorInfo</var>[<code
   data-x="dom-ErrorEvent-error">error</code>] to null, <var>errorInfo</var>[<code
   data-x="dom-ErrorEvent-message">message</code>] to "<code data-x="">Script error.</code>",
   <var>errorInfo</var>[<code data-x="dom-ErrorEvent-filename">filename</code>] to the empty
   string, <var>errorInfo</var>[<code data-x="dom-ErrorEvent-lineno">lineno</code>] to 0, and
   <var>errorInfo</var>[<code data-x="dom-ErrorEvent-colno">colno</code>] to 0.</p></li>

   <li><p>If <var>omitError</var> is true, then set <var>errorInfo</var>[<code
   data-x="dom-ErrorEvent-error">error</code>] to null.</p></li>

   <li>
    <p>If <var>global</var> is not <span>in error reporting mode</span>, then:</p>

    <ol>
     <li>Set <var>global</var>'s <span>in error reporting mode</span> to true.</li>

     <li>
      <p>If <var>global</var> implements <code>EventTarget</code>, then set <var>notHandled</var> to
      the result of <span data-x="concept-event-fire">firing an event</span> named <code
      data-x="event-error">error</code> at <var>global</var>, using <code>ErrorEvent</code>, with
      the <code data-x="dom-Event-cancelable">cancelable</code> attribute initialized to true,
      and additional attributes initialized according to <var>errorInfo</var>.</p>

      <p class="note">Returning true in an event handler cancels the event per <span>the event
      handler processing algorithm</span>.</p>
     </li>

     <li><p>Set <var>global</var>'s <span>in error reporting mode</span> to false.</p></li>
    </ol>
   </li>

   <li>
    <p>If <var>notHandled</var> is true, then:</p>

    <ol>
     <li><p>Set <var>errorInfo</var>[<code data-x="dom-ErrorEvent-error">error</code>] to
     null.</p></li>

     <li>
      <p>If <var>global</var> implements <code>DedicatedWorkerGlobalScope</code>, <span>queue a
      global task</span> on the <span>DOM manipulation task source</span> with the
      <var>global</var>'s associated <code>Worker</code>'s <span>relevant global object</span> to
      run these steps:</p>

      <ol>
       <li><p>Let <var>workerObject</var> be the <code>Worker</code> object associated with
       <var>global</var>.</p></li>

       <li><p>Set <var>notHandled</var> be the result of <span data-x="concept-event-fire">firing
       an event</span> named <code data-x="event-error">error</code> at <var>workerObject</var>,
       using <code>ErrorEvent</code>, with the <code
       data-x="dom-Event-cancelable">cancelable</code> attribute initialized to true,
       and additional attributes initialized according to <var>errorInfo</var>.</p></li>

       <li>
        <p>If <var>notHandled</var> is true, then <span data-x="report an exception">report</span>
        <var>exception</var> for <var>workerObject</var>'s <span>relevant global object</span> with
        <span data-x="report-exception-omitError"><var>omitError</var></span> set to true.</p>

        <p class="note">The actual <var>exception</var> value will not be available in the owner
        realm, but the user agent still carries through enough information to set the message,
        filename, and other attributes, as well as potentially report to a developer console.</p>
       </li>
      </ol>
     </li>

     <li><p>Otherwise, the user agent may report <var>exception</var> to a developer
     console.</p></li>
    </ol>
   </li>
  </ol>

  <p>If the implicit port connecting a worker to its <code>Worker</code> object has been
  disentangled (i.e. if the parent worker has been terminated), then the user agent must act as if
  the <code>Worker</code> object had no <code data-x="event-error">error</code> event handler and as
  if that worker's <code data-x="handler-WorkerGlobalScope-onerror">onerror</code> attribute was
  null, but must otherwise act as described above.</p>

  <p class="note">Thus, error reports propagate up to the chain of dedicated workers up to the
  original <code>Document</code>, even if some of the workers along this chain have been terminated
  and garbage collected.</p>

  <p>Previous revisions of this standard defined an algorithm to <dfn data-x="report the exception"
  data-lt="report the exception" export>report the exception</dfn>. As part of <a
  href="https://github.com/whatwg/html">issue #958</a>, this has been superseded by <span>report an
  exception</span> which behaves differently and takes different inputs. <a
  href="https://github.com/whatwg/html/issues/10516">Issue #10516</a> tracks updating the
  specification ecosystem.</p>

  <hr>

  <p>The <dfn method for="WindowOrWorkerGlobalScope"
  data-x="dom-reportError"><code>reportError(<var>e</var>)</code></dfn> method steps are to
  <span>report an exception</span> <var>e</var> for <span>this</span>.</p>

  <p class="XXX">It is unclear whether <span data-x="muted errors">muting</span> is
  applicable here. In Chrome and Safari it is muted, but in Firefox it is not. See also <a
  href="https://github.com/whatwg/html/issues/958">issue #958</a>.</p>

  <hr>

  <p id="the-errorevent-interface">The <code>ErrorEvent</code> interface is defined as follows:</p>

  <pre><code class="idl">[Exposed=*]
interface <dfn interface>ErrorEvent</dfn> : <span>Event</span> {
  <span data-x="dom-Event-constructor">constructor</span>(DOMString type, optional <span>ErrorEventInit</span> eventInitDict = {});

  readonly attribute DOMString <span data-x="dom-ErrorEvent-message">message</span>;
  readonly attribute USVString <span data-x="dom-ErrorEvent-filename">filename</span>;
  readonly attribute unsigned long <span data-x="dom-ErrorEvent-lineno">lineno</span>;
  readonly attribute unsigned long <span data-x="dom-ErrorEvent-colno">colno</span>;
  readonly attribute any <span data-x="dom-ErrorEvent-error">error</span>;
};

dictionary <dfn dictionary>ErrorEventInit</dfn> : <span>EventInit</span> {
  DOMString message = "";
  USVString filename = "";
  unsigned long lineno = 0;
  unsigned long colno = 0;
  any error;
};</code></pre>

  <p>The <dfn attribute for="ErrorEvent"><code data-x="dom-ErrorEvent-message">message</code></dfn>
  attribute must return the value it was initialized to. It represents the error message.</p>

  <p>The <dfn attribute for="ErrorEvent"><code
  data-x="dom-ErrorEvent-filename">filename</code></dfn> attribute must return the value it was
  initialized to. It represents the <span>URL</span> of the script in which the error originally
  occurred.</p>

  <p>The <dfn attribute for="ErrorEvent"><code data-x="dom-ErrorEvent-lineno">lineno</code></dfn>
  attribute must return the value it was initialized to. It represents the line number where the
  error occurred in the script.</p>

  <p>The <dfn attribute for="ErrorEvent"><code data-x="dom-ErrorEvent-colno">colno</code></dfn>
  attribute must return the value it was initialized to. It represents the column number where the
  error occurred in the script.</p>

  <p>The <dfn attribute for="ErrorEvent"><code data-x="dom-ErrorEvent-error">error</code></dfn>
  attribute must return the value it was initialized to. It must initially be initialized to
  undefined. Where appropriate, it is set to the object representing the error (e.g., the exception
  object in the case of an uncaught exception).</p>

  </div>

  <h5>Unhandled promise rejections</h5>

  <p>In addition to synchronous <a href="#runtime-script-errors">runtime script errors</a>, scripts
  may experience asynchronous promise rejections, tracked via the <code
  data-x="event-unhandledrejection">unhandledrejection</code> and <code
  data-x="event-rejectionhandled">rejectionhandled</code> events.<span w-nodev> Tracking these
  rejections is done via the <span>HostPromiseRejectionTracker</span> abstract operation, but
  reporting them is defined here.</span></p>

  <div w-nodev>

  <p>To <dfn>notify about rejected promises</dfn> given a <span>global object</span>
  <var>global</var>:</p>

  <ol>
   <li><p>Let <var>list</var> be a <span data-x="list clone">clone</span> of <var>global</var>'s
   <span>about-to-be-notified rejected promises list</span>.</p></li>

   <li><p>If <var>list</var> <span data-x="list is empty">is empty</span>, then return.</p></li>

   <li><p><span data-x="list empty">Empty</span> <var>global</var>'s <span>about-to-be-notified
   rejected promises list</span>.</p></li>

   <li>
    <p><span>Queue a global task</span> on the <span>DOM manipulation task source</span> given
    <var>global</var> to run the following step:</p>

    <ol>
     <li><p><span data-x="list iterate">For each</span> promise <var>p</var> of <var>list</var>:</p>

      <ol>
       <li><p>If <var>p</var>.[[PromiseIsHandled]] is true, then <span>continue</span>.</p></li>

       <li><p>Let <var>notCanceled</var> be the result of <span data-x="concept-event-fire">firing
       an event</span> named <code data-x="event-unhandledrejection">unhandledrejection</code> at
       <var>global</var>, using <code>PromiseRejectionEvent</code>, with the <code
       data-x="dom-Event-cancelable">cancelable</code> attribute initialized to true, the <code
       data-x="dom-PromiseRejectionEvent-promise">promise</code> attribute initialized to
       <var>p</var>, and the <code data-x="dom-PromiseRejectionEvent-reason">reason</code>
       attribute initialized to <var>p</var>.[[PromiseResult]].</p></li>

       <li id="concept-promise-rejection-handled"><p id="concept-promise-rejection-nothandled">If
       <var>notCanceled</var> is true, then the user agent may report
       <var>p</var>.[[PromiseResult]] to a developer console.</p></li>

       <li><p>If <var>p</var>.[[PromiseIsHandled]] is false, then <span data-x="set
       append">append</span> <var>p</var> to <var>global</var>'s <span>outstanding rejected
       promises weak set</span>.</p></li>
      </ol>
     </li>
    </ol>
   </li>
  </ol>

  <p id="the-promiserejectionevent-interface">The <code>PromiseRejectionEvent</code> interface is
  defined as follows:</p>

  <pre><code class="idl">[Exposed=*]
interface <dfn interface>PromiseRejectionEvent</dfn> : <span>Event</span> {
  <span data-x="dom-Event-constructor">constructor</span>(DOMString type, <span>PromiseRejectionEventInit</span> eventInitDict);

  readonly attribute <span data-x="idl-object">object</span> <span data-x="dom-PromiseRejectionEvent-promise">promise</span>;
  readonly attribute any <span data-x="dom-PromiseRejectionEvent-reason">reason</span>;
};

dictionary <dfn dictionary>PromiseRejectionEventInit</dfn> : <span>EventInit</span> {
  required <span data-x="idl-object">object</span> promise;
  any reason;
};</code></pre>

  <p>The <dfn attribute for="PromiseRejectionEvent"><code
  data-x="dom-PromiseRejectionEvent-promise">promise</code></dfn> attribute must return the value it
  was initialized to. It represents the promise which this notification is about.</p>

  <p class="note">Because of how Web IDL conversion rules for <code
  data-x=""><span data-x="idl-Promise">Promise</span>&lt;<var>T</var>></code> types always wrap the input into a new promise, the
  <code data-x="dom-PromiseRejectionEvent-promise">promise</code> attribute is of type <code
  data-x="idl-object">object</code> instead, which is more appropriate for representing an opaque
  handle to the original promise object.</p>

  <p>The <dfn attribute for="PromiseRejectionEvent"><code
  data-x="dom-PromiseRejectionEvent-reason">reason</code></dfn> attribute must return the value it
  was initialized to. It represents the rejection reason for the promise.</p>


  <h5>Import map parse results</h5>

  <p>An <dfn>import map parse result</dfn> is a <span>struct</span> that is similar to a <span
  data-x="concept-script">script</span>, and also can be stored in a <code>script</code> element's
  <span data-x="concept-script-result">result</span>, but is not counted as a <span
  data-x="concept-script">script</span> for other purposes. It has the following <span
  data-x="struct item">items</span>:</p>

  <dl>
   <dt>An <dfn data-x="impr-import-map">import map</dfn></dt>
   <dd>An <span>import map</span> or null.</dd>

   <dt>An <dfn data-x="impr-error-to-rethrow">error to rethrow</dfn></dt>
   <dd>A JavaScript value representing an error that will prevent using this import map, when
   non-null.</dd>
  </dl>

  <p>To <dfn>create an import map parse result</dfn> given a <span>string</span> <var>input</var>
  and a <span>URL</span> <var>baseURL</var>:</p>

  <ol>
   <li><p>Let <var>result</var> be an <span>import map parse result</span> whose <span
   data-x="impr-import-map">import map</span> is null and whose <span
   data-x="impr-error-to-rethrow">error to rethrow</span> is null.</p></li>

   <li><p><span>Parse an import map string</span> given <var>input</var> and <var>baseURL</var>,
   catching any exceptions. If this threw an exception, then set <var>result</var>'s <span
   data-x="impr-error-to-rethrow">error to rethrow</span> to that exception. Otherwise, set
   <var>result</var>'s <span data-x="impr-import-map">import map</span> to the return
   value.</p></li>

   <li><p>Return <var>result</var>.</p></li>
  </ol>

  <p>To <dfn>register an import map</dfn> given a <code>Window</code> <var>global</var> and an
  <span>import map parse result</span> <var>result</var>:</p>

  <ol>
   <li><p>If <var>result</var>'s <span data-x="impr-error-to-rethrow">error to rethrow</span> is
   not null, then <span>report an exception</span> given by <var>result</var>'s <span
   data-x="impr-error-to-rethrow">error to rethrow</span> for <var>global</var> and
   return.</p></li>

   <li><p><span data-x="merge existing and new import maps">Merge existing and new import
   maps</span>, given <var>global</var> and <var>result</var>'s <span
   data-x="impr-import-map">import map</span>.</p></li>
  </ol>

  </div>


  <h4>Module specifier resolution</h4>

  <h5>The resolution algorithm</h5>

  <p>The <span>resolve a module specifier</span> algorithm is the primary entry point for
  converting module specifier strings into <span data-x="URL">URLs</span>. When no <span
  data-x="import map">import maps</span> are involved, it is relatively straightforward, and
  reduces to <span>resolving a URL-like module specifier</span>.</p>

  <p>When there is a non-empty <span>import map</span> present, the behavior is more complex. It
  checks candidate entries from all applicable <span data-x="module specifier map">module specifier
  maps</span>, from most-specific to least-specific <span
  data-x="concept-import-map-scopes">scopes</span> (falling back to the top-level unscoped <span
  data-x="concept-import-map-imports">imports</span>), and from most-specific to least-specific
  prefixes. <span w-nodev>For each candidate, the <span data-x="resolving an imports match">resolve
  an imports match</span> algorithm will give on the following results:</span></p>

  <ul w-nodev> <li><p>Successful resolution of the specifier to a <span>URL</span>. Then the
  <span>resolve a module specifier</span> algorithm will return that URL.</p></li>

   <li><p>Throwing an exception. Then the <span>resolve a module specifier</span> algorithm will
   rethrow that exception, without any further fallbacks.</p></li>

   <li><p>Failing to resolve, without an error. In this case the outer <span>resolve a module
   specifier</span> algorithm will move on to the next candidate.</p></li>
  </ul>

  <p>In the end, if no successful resolution is found via any of the candidate <span data-x="module
  specifier map">module specifier maps</span>, <span>resolve a module specifier</span> will throw
  an exception. Thus the result is always either a <span>URL</span> or a thrown exception.</p>

  <div w-nodev>

  <p>To <dfn>resolve a module specifier</dfn> given a <span>script</span>-or-null
  <var>referringScript</var> and a <span>string</span> <var>specifier</var>:</p>

  <ol>
   <li><p>Let <var>settingsObject</var> and <var>baseURL</var> be null.</p></li>

   <li>
    <p>If <var>referringScript</var> is not null, then:</p>

    <ol>
     <li><p>Set <var>settingsObject</var> to <var>referringScript</var>'s
     <span data-x="concept-script-settings-object">settings object</span>.</p></li>

     <li><p>Set <var>baseURL</var> to <var>referringScript</var>'s <span
     data-x="concept-script-base-url">base URL</span>.</p></li>
    </ol>
   </li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p><span>Assert</span>: there is a <span>current settings object</span>.</p></li>

     <li><p>Set <var>settingsObject</var> to the <span>current settings object</span>.</p></li>

     <li><p>Set <var>baseURL</var> to <var>settingsObject</var>'s <span>API base
     URL</span>.</p></li>
    </ol>
   </li>

   <li><p>Let <var>importMap</var> be an <span>empty import map</span>.</p></li>

   <li><p>If <var>settingsObject</var>'s <span data-x="concept-settings-object-global">global
   object</span> implements <code>Window</code>, then set <var>importMap</var> to
   <var>settingsObject</var>'s <span data-x="concept-settings-object-global">global object</span>'s
   <span data-x="concept-global-import-map">import map</span>.</p></li>

   <li><p>Let <var>serializedBaseURL</var> be <var>baseURL</var>, <span
   data-x="concept-url-serializer">serialized</span>.</p></li>

   <li><p>Let <var>asURL</var> be the result of <span>resolving a URL-like module specifier</span>
   given <var>specifier</var> and <var>baseURL</var>.</p></li>

   <li><p>Let <var>normalizedSpecifier</var> be the <span
   data-x="concept-url-serializer">serialization</span> of <var>asURL</var>, if <var>asURL</var> is
   non-null; otherwise, <var>specifier</var>.</p></li>

   <li><p>Let <var>result</var> be a <span>URL</span>-or-null, initially null.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>scopePrefix</var> → <var>scopeImports</var>
    of <var>importMap</var>'s <span data-x="concept-import-map-scopes">scopes</span>:</p>

    <ol>
     <li>
      <p>If <var>scopePrefix</var> is <var>serializedBaseURL</var>, or if <var>scopePrefix</var> ends
      with U+002F (/) and <var>scopePrefix</var> is a <span>code unit prefix</span> of
      <var>serializedBaseURL</var>, then:</p>

      <ol>
       <li><p>Let <var>scopeImportsMatch</var> be the result of <span>resolving an imports
       match</span> given <var>normalizedSpecifier</var>, <var>asURL</var>, and
       <var>scopeImports</var>.</p></li>

       <li><p>If <var>scopeImportsMatch</var> is not null, then set <var>result</var> to
       <var>scopeImportsMatch</var>, and <span>break</span>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>If <var>result</var> is null, set <var>result</var> be the result of <span>resolving an
   imports match</span> given <var>normalizedSpecifier</var>, <var>asURL</var>, and
   <var>importMap</var>'s <span data-x="concept-import-map-imports">imports</span>.</p></li>

   <li>
    <p>If <var>result</var> is null, set it to <var>asURL</var>.</p>

    <p class="note">By this point, if <var>result</var> was null, <var>specifier</var> wasn't
    remapped to anything by <var>importMap</var>, but it might have been able to be turned into a
    URL.</p>
   </li>

   <li>
    <p>If <var>result</var> is not null, then:</p>

    <ol>
     <li><p><span>Add module to resolved module set</span> given <var>settingsObject</var>,
     <var>serializedBaseURL</var>, <var>normalizedSpecifier</var>, and <var>asURL</var>.</p></li>

     <li><p>Return <var>result</var>.</p></li>
    </ol>
   </li>

   <li><p>Throw a <code>TypeError</code> indicating that <var>specifier</var> was a bare specifier,
   but was not remapped to anything by <var>importMap</var>.</p></li>
  </ol>

  <p>To <dfn data-x="resolving an imports match">resolve an imports match</dfn>, given a
  <span>string</span> <var>normalizedSpecifier</var>, a <span>URL</span>-or-null <var>asURL</var>,
  and a <span>module specifier map</span> <var>specifierMap</var>:</p>

  <ol>
   <li>
    <p><span data-x="list iterate">For each</span> <var>specifierKey</var> →
    <var>resolutionResult</var> of <var>specifierMap</var>:</p>

    <ol>
     <li>
      <p>If <var>specifierKey</var> is <var>normalizedSpecifier</var>, then:</p>

      <ol>
       <li>
        <p>If <var>resolutionResult</var> is null, then throw a <code>TypeError</code> indicating
        that resolution of <var>specifierKey</var> was blocked by a null entry.</p>

        <p class="note">This will terminate the entire <span>resolve a module specifier</span>
        algorithm, without any further fallbacks.</p>
       </li>

       <li><p><span>Assert</span>: <var>resolutionResult</var> is a <span>URL</span>.</p></li>

       <li><p>Return <var>resolutionResult</var>.</p></li>
      </ol>
     </li>

     <li>
      <p>If all of the following are true:</p>

      <ul>
       <li><p><var>specifierKey</var> ends with U+002F (/);</p></li>

       <li><p><var>specifierKey</var> is a <span>code unit prefix</span> of
       <var>normalizedSpecifier</var>; and</p></li>

       <li><p>either <var>asURL</var> is null, or <var>asURL</var> <span>is special</span>,</p></li>
      </ul>

      <p>then:</p>

      <ol>
       <li>
        <p>If <var>resolutionResult</var> is null, then throw a <code>TypeError</code> indicating
        that the resolution of <var>specifierKey</var> was blocked by a null entry.</p>

        <p class="note">This will terminate the entire <span>resolve a module specifier</span>
        algorithm, without any further fallbacks.</p>
       </li>

       <li><p><span>Assert</span>: <var>resolutionResult</var> is a <span>URL</span>.</p></li>

       <li><p>Let <var>afterPrefix</var> be the portion of <var>normalizedSpecifier</var> after the
       initial <var>specifierKey</var> prefix.</p></li>

       <li><p><span>Assert</span>: <var>resolutionResult</var>, <span
       data-x="concept-url-serializer">serialized</span>, ends with U+002F (/), as enforced during
       <span data-x="parse an import map string">parsing</span>.</p></li>

       <li><p>Let <var>url</var> be the result of <span data-x="URL parser">URL parsing</span>
       <var>afterPrefix</var> with <var>resolutionResult</var>.</p></li>

       <li>
        <p>If <var>url</var> is failure, then throw a <code>TypeError</code> indicating that
        resolution of <var>normalizedSpecifier</var> was blocked since the <var>afterPrefix</var>
        portion could not be URL-parsed relative to the <var>resolutionResult</var> mapped to by
        the <var>specifierKey</var> prefix.</p>

        <p class="note">This will terminate the entire <span>resolve a module specifier</span>
        algorithm, without any further fallbacks.</p>
       </li>

       <li><p><span>Assert</span>: <var>url</var> is a <span>URL</span>.</p></li>

       <li>
        <p>If the <span data-x="concept-url-serializer">serialization</span> of
        <var>resolutionResult</var> is not a <span>code unit prefix</span> of the <span
        data-x="concept-url-serializer">serialization</span> of <var>url</var>, then throw a
        <code>TypeError</code> indicating that the resolution of <var>normalizedSpecifier</var> was
        blocked due to it backtracking above its prefix <var>specifierKey</var>.</p>

        <p class="note">This will terminate the entire <span>resolve a module specifier</span>
        algorithm, without any further fallbacks.</p>
       </li>

       <li><p>Return <var>url</var>.</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li>
    <p>Return null.</p>

    <p class="note">The <span>resolve a module specifier</span> algorithm will fall back to a
    less-specific scope, or to "<code data-x="">imports</code>", if possible.</p>
   </li>
  </ol>

  <p>To <dfn data-x="resolving a URL-like module specifier">resolve a URL-like module
  specifier</dfn>, given a <span>string</span> <var>specifier</var> and a <span>URL</span>
  <var>baseURL</var>:</p>

  <ol>
   <li>
    <p>If <var>specifier</var> <span>starts with</span> "<code data-x="">/</code>", "<code
    data-x="">./</code>", or "<code data-x="">../</code>", then:</p>

    <ol>
     <li><p>Let <var>url</var> be the result of <span data-x="URL parser">URL parsing</span>
     <var>specifier</var> with <var>baseURL</var>.</p></li>

     <li>
      <p>If <var>url</var> is failure, then return null.</p>

      <p class="example">One way this could happen is if <var>specifier</var> is "<code
      data-x="">../foo</code>" and <var>baseURL</var> is a <code data-x="data
      protocol">data:</code> URL.</p>
     </li>

     <li><p>Return <var>url</var>.</p></li>
    </ol>

    <p class="note">This includes cases where <var>specifier</var> <span>starts with</span> "<code
    data-x="">//</code>", i.e., scheme-relative URLs. Thus, <var>url</var> might end up with a
    different <span data-x="concept-url-host">host</span> than <var>baseURL</var>.</p>
   </li>

   <li><p>Let <var>url</var> be the result of <span data-x="URL parser">URL parsing</span>
   <var>specifier</var> (with no base URL).</p></li>

   <li><p>If <var>url</var> is failure, then return null.</p></li>

   <li><p>Return <var>url</var>.</p>
  </ol>

  </div>


  <h5>Import maps</h5>

  <p>An <span subdfn>import map</span> allows control over module specifier resolution. Import maps
  are delivered via inline <code>script</code> elements with their <code
  data-x="attr-script-type">type</code> attribute set to "<code data-x="">importmap</code>", and
  with their <span>child text content</span> containing a JSON representation of the import
  map.</p>

  <p>A <code>Document</code> can have multiple import maps processed, which can happen either
  before or after any modules have been imported, e.g., via <code>import()</code> expressions or
  <code>script</code> elements with their <code data-x="attr-script-type">type</code> attribute set
  to "<code data-x="">module</code>". The <span>merge existing and new import maps</span> algorithm
  ensures that new import maps cannot define the module resolution for modules that were already
  defined by past import maps, or for ones that were already resolved.</p>

  <div class="example" id="example-import-map-bare-specifier">
   <p>The simplest use of import maps is to globally remap a bare module specifier:</p>

   <pre><code class="json" data-x="">{
  "imports": {
    "moment": "/node_modules/moment/src/moment.js"
  }
}</code></pre>

   <p>This enables statements like <code class="js" data-x="">import moment from "moment";</code>
   to work, fetching and evaluating the JavaScript module at the <code
   data-x="">/node_modules/moment/src/moment.js</code> URL.</p>
  </div>

  <div class="example" id="example-import-map-trailing-slashes">
   <p>An import map can remap a class of module specifiers into a class of URLs by using trailing
   slashes, like so:</p>

   <pre><code class="json" data-x="">{
  "imports": {
    "moment/": "/node_modules/moment/src/"
  }
}</code></pre>

   <p>This enables statements like <code class="js" data-x="">import localeData from
   "moment/locale/zh-cn.js";</code> to work, fetching and evaluating the JavaScript module at the
   <code data-x="">/node_modules/moment/src/locale/zh-cn.js</code> URL. Such trailing-slash
   mappings are often combined with bare-specifier mappings, e.g.</p>

   <pre><code class="json" data-x="">{
  "imports": {
    "moment": "/node_modules/moment/src/moment.js",
    "moment/": "/node_modules/moment/src/"
  }
}</code></pre>

   <p>so that both the "main module" specified by "<code data-x="">moment</code>" and the
   "submodules" specified by paths such as "<code data-x="">moment/locale/zh-cn.js</code>" are
   available.</p>
  </div>

  <div class="example" id="example-import-map-url-like-specifier">
   <p>Bare specifiers are not the only type of module specifiers which import maps can remap.
   "URL-like" specifiers, i.e., those that are either parseable as absolute URLs or start with
   "<code data-x="">/</code>", "<code data-x="">./</code>", or "<code data-x="">../</code>", can be
   remapped as well:</p>

   <pre><code class="json" data-x="">{
  "imports": {
    "https://cdn.example.com/vue/dist/vue.runtime.esm.js": "/node_modules/vue/dist/vue.runtime.esm.js",
    "/js/app.mjs": "/js/app-8e0d62a03.mjs",
    "../helpers/": "https://cdn.example/helpers/"
  }
}</code></pre>

   <p>Note how the URL to be remapped, as well as the URL being mapped to, can be specified either
   as absolute URLs, or as relative URLs starting with "<code data-x="">/</code>", "<code
   data-x="">./</code>", or "<code data-x="">../</code>". (They cannot be specified as relative
   URLs without those starting sigils, as those help distinguish from bare module specifiers.) Also
   note how the <a href="#example-import-map-trailing-slashes">trailing slash mapping</a> works in
   this context as well.</p>

   <p>Such remappings operate on the post-canonicalization URL, and do not require a match between
   the literal strings supplied in the import map key and the imported module specifier. So for
   example, if this import map was included on <code data-x="">https://example.com/app.html</code>,
   then not only would <code class="js" data-x="">import "/js/app.mjs"</code> be remapped, but so
   would <code class="js" data-x="">import "./js/app.mjs"</code> and <code class="js"
   data-x="">import "./foo/../js/app.mjs"</code>.</p>
  </div>

  <div class="example" id="example-import-map-scopes">
   <p>All previous examples have globally remapped module specifiers, by using the top-level "<code
   data-x="">imports</code>" key in the import map. The top-level "<code data-x="">scopes</code>"
   key can be used to provide localized remappings, which only apply when the referring module
   matches a specific URL prefix. For example:</p>

   <pre><code class="json" data-x="">{
  "scopes": {
    "/a/" : {
      "moment": "/node_modules/moment/src/moment.js"
    },
    "/b/" : {
      "moment": "https://cdn.example.com/moment/src/moment.js"
    }
  }
}</code></pre>

   <p>With this import map, the statement <code data-x="">import "moment"</code> will have
   different meanings depending on which referrer script contains the statement:</p>

   <ul>
    <li><p>Inside scripts located under <code data-x="">/a/</code>, this will import <code
    data-x="">/node_modules/moment/src/moment.js</code>.</p></li>

    <li><p>Inside scripts located under <code data-x="">/b/</code>, this will import <code
    data-x="">https://cdn.example.com/moment/src/moment.js</code>.</p></li>

    <li><p>Inside scripts located under <code data-x="">/c/</code>, this will fail to resolve and
    thus throw an exception.</p></li>
   </ul>

   <p>A typical usage of scopes is to allow multiple versions of the "same" module to exist in a
   web application, with some parts of the module graph importing one version, and other parts
   importing another version.
  </div>

  <div class="example" id="example-import-map-scopes-overlapping">
   <p>Scopes can overlap each other, and overlap the global "<code data-x="">imports</code>"
   specifier map. At resolution time, scopes are consulted in order of most- to least-specific,
   where specificity is measured by sorting the scopes using the <span>code unit less than</span>
   operation. So, for example, "<code data-x="">/scope2/scope3/</code>" is treated as more specific
   than "<code data-x="">/scope2/</code>", which is treated as more specific than the top-level
   (unscoped) mappings.</p>

   <p>The following import map illustrates this:</p>

   <pre><code class="json" data-x="">{
  "imports": {
    "a": "/a-1.mjs",
    "b": "/b-1.mjs",
    "c": "/c-1.mjs"
  },
  "scopes": {
    "/scope2/": {
      "a": "/a-2.mjs"
    },
    "/scope2/scope3/": {
      "b": "/b-3.mjs"
    }
  }
}</code></pre>

   <p>This results in the following resolutions (using relative URLs for brevity):</p>

   <table id="table-import-map-scopes-example">
    <thead>
     <tr>
      <td colspan="2" rowspan="2">
      <th colspan="3">Specifier
     <tr>
      <th>"<code data-x="">a</code>"
      <th>"<code data-x="">b</code>"
      <th>"<code data-x="">c</code>"
    <tbody>
     <tr>
      <th rowspan="3">Referrer
      <th><code data-x="">/scope1/r.mjs</code>
      <td><code data-x="">/a-1.mjs</code>
      <td><code data-x="">/b-1.mjs</code>
      <td><code data-x="">/c-1.mjs</code>
     <tr>
      <th><code data-x="">/scope2/r.mjs</code>
      <td><code data-x="">/a-2.mjs</code>
      <td><code data-x="">/b-1.mjs</code>
      <td><code data-x="">/c-1.mjs</code>
     <tr>
      <th><code data-x="">/scope2/scope3/r.mjs</code>
      <td><code data-x="">/a-2.mjs</code>
      <td><code data-x="">/b-3.mjs</code>
      <td><code data-x="">/c-1.mjs</code>
   </table>
  </div>

  <div class="example" id="example-import-map-integrity">
   <p>Import maps can also be used to provide modules with integrity metadata to be used in
   <cite>Subresource Integrity</cite> checks. <ref>SRI</ref>
   </p>

   <p>The following import map illustrates this:</p>

   <pre><code class="json" data-x="">{
  "imports": {
    "a": "/a-1.mjs",
    "b": "/b-1.mjs",
    "c": "/c-1.mjs"
  },
  "integrity": {
    "/a-1.mjs": "sha384-Li9vy3DqF8tnTXuiaAJuML3ky+er10rcgNR/VqsVpcw+ThHmYcwiB1pbOxEbzJr7",
    "/d-1.mjs": "sha384-MBO5IDfYaE6c6Aao94oZrIOiC6CGiSN2n4QUbHNPhzk5Xhm0djZLQqTpL0HzTUxk"
  }
}</code></pre>
   <p>The above example provides integrity metadata to be enforced on the modules <code
   data-x="">/a-1.mjs</code> and <code data-x="">/d-1.mjs</code>, even if the latter is not defined
   as an import in the map.</p>
  </div>

  <hr>

  <p>The <span>child text content</span> of a <code>script</code> element representing an
  <span>import map</span> must match the following <dfn>import map authoring
  requirements</dfn>:</p>

  <ul>
   <li><p>It must be valid JSON. <ref>JSON</ref></p></li>

   <li><p>The JSON must represent a JSON object, with at most the three keys "<code
   data-x="">imports</code>", "<code data-x="">scopes</code>", and "<code
   data-x="">integrity</code>".</p></li>

   <li><p>The values corresponding to the "<code data-x="">imports</code>", "<code
   data-x="">scopes</code>", and "<code data-x="">integrity</code>" keys, if present,
   must themselves be JSON objects.</p></li>

   <li><p>The value corresponding to the "<code data-x="">imports</code>" key, if present, must be
   a <span>valid module specifier map</span>.</p></li>

   <li><p>The value corresponding to the "<code data-x="">scopes</code>" key, if present, must be a
   JSON object, whose keys are <span data-x="valid URL string">valid URL strings</span> and whose
   values are <span data-x="valid module specifier map">valid module specifier
   maps</span>.</p></li>

   <li><p>The value corresponding to the "<code data-x="">integrity</code>" key, if present, must
   be a JSON object, whose keys are <span data-x="valid URL string">valid URL strings</span> and
   whose values fit <span>the requirements of the integrity attribute</span>.</p></li>
  </ul>

  <p>A <dfn>valid module specifier map</dfn> is a JSON object that meets the following
  requirements:</p>

  <ul>
   <li><p>All of its keys must be nonempty.</p></li>

   <li><p>All of its values must be strings.</p></li>

   <li><p>Each value must be either a <span data-x="absolute URL">valid absolute URL</span> or a
   <span>valid URL string</span> that <span>starts with</span> "<code data-x="">/</code>", "<code
   data-x="">./</code>", or "<code data-x="">../</code>".</p></li>

   <li><p>If a given key <span>ends with</span> "<code data-x="">/</code>", then the corresponding
   value must also.</p></li>
  </ul>


  <div w-nodev>

  <h5>Import map processing model</h5>

  <p>Formally, an <dfn>import map</dfn> is a <span>struct</span> with three <span data-x="struct
  item">items</span>:</p>

  <ul>
   <li><p><dfn data-x="concept-import-map-imports">imports</dfn>, a <span>module specifier
   map</span>;</p></li>

   <li><p><dfn data-x="concept-import-map-scopes">scopes</dfn>, an <span>ordered map</span> of
   <span data-x="URL">URLs</span> to <span data-x="module specifier map">module specifier
   maps</span>; and</p></li>

   <li><p><dfn data-x="concept-import-map-integrity">integrity</dfn>, a <span>module integrity
   map</span>.</p></li>
  </ul>

  <p>A <dfn>module specifier map</dfn> is an <span>ordered map</span> whose <span data-x="map
  key">keys</span> are <span data-x="string">strings</span> and whose <span data-x="map
  value">values</span> are either <span data-x="URL">URLs</span> or nulls.</p>

  <p>A <dfn>module integrity map</dfn> is an <span>ordered map</span> whose <span data-x="map
  key">keys</span> are <span data-x="URL">URLs</span> and whose <span data-x="map
  value">values</span> are <span data-x="string">strings</span> that will be used as <span
  data-x="concept-request-integrity-metadata">integrity metadata</span>.</p>

  <p>An <dfn>empty import map</dfn> is an <span>import map</span> with its <span
  data-x="concept-import-map-imports">imports</span> and <span
  data-x="concept-import-map-scopes">scopes</span> both being empty maps.</p>

  <hr>

  <p>A <dfn>specifier resolution record</dfn> is a <span>struct</span>. It has the following <span
  data-x="struct item">items</span>:</p>
  <dl>
   <dt>A <dfn data-x="specifier-resolution-record-serialized-base-url">serialized base
   URL</dfn></dt>
   <dd>A <span>string</span>-or-null that represents the base URL of the specifier, when one
   exists.</dd>

   <dt>A <dfn data-x="specifier-resolution-record-specifier">specifier</dfn></dt>
   <dd>A <span>string</span> representing the specifier.</dd>

   <dt>A <dfn data-x="specifier-resolution-record-as-url">specifier as a URL</dfn></dt>
   <dd>A <span>URL</span>-or-null that represents the URL in case of a URL-like module
   specifier.</dd>
  </dl>

  <p class="note">Implementations can replace
  <span data-x="specifier-resolution-record-as-url">specifier as a URL</span> with a boolean that
  indicates that the specifier is either bare or URL-like that <span>is special</span>.</p>

  <p>To <dfn>add module to resolved module set</dfn> given an <span>environment settings
  object</span> <var>settingsObject</var>, a <span>string</span> <var>serializedBaseURL</var>, a
  <span>string</span> <var>normalizedSpecifier</var>, and a <span>URL</span>-or-null
  <var>asURL</var>:</p>

  <ol>
   <li><p>Let <var>global</var> be <var>settingsObject</var>'s <span
   data-x="concept-settings-object-global">global object</span>.</p></li>

   <li><p>If <var>global</var> does not implement <code>Window</code>, then return.</p></li>

   <li><p>Let <var>record</var> be a new <span>specifier resolution record</span>, with <span
   data-x="specifier-resolution-record-serialized-base-url">serialized base URL</span> set to
   <var>serializedBaseURL</var>, <span
   data-x="specifier-resolution-record-specifier">specifier</span> set to
   <var>normalizedSpecifier</var>, and <span
   data-x="specifier-resolution-record-as-url">specifier as a URL</span> set to
   <var>asURL</var>.</p></li>

   <li><p><span data-x="set append">Append</span> <var>record</var> to <var>global</var>'s
   <span>resolved module set</span>.</p></li>
  </ol>
  <hr>

  <p>To <dfn>parse an import map string</dfn>, given a <span>string</span> <var>input</var> and a
  <span>URL</span> <var>baseURL</var>:</p>

  <ol>
   <li><p>Let <var>parsed</var> be the result of <span data-x="parse a JSON string to an Infra
   value">parsing a JSON string to an Infra value</span> given <var>input</var>.</p></li>

   <li><p>If <var>parsed</var> is not an <span>ordered map</span>, then throw a
   <code>TypeError</code> indicating that the top-level value needs to be a JSON object.</p></li>

   <li><p>Let <var>sortedAndNormalizedImports</var> be an empty <span>ordered map</span>.</p></li>

   <li>
    <p>If <var>parsed</var>["<code data-x="">imports</code>"] <span data-x="map
    exists">exists</span>, then:</p>

    <ol>
     <li><p>If <var>parsed</var>["<code data-x="">imports</code>"] is not an <span>ordered
     map</span>, then throw a <code>TypeError</code> indicating that the value for the "<code
     data-x="">imports</code>" top-level key needs to be a JSON object.</p></li>

     <li><p>Set <var>sortedAndNormalizedImports</var> to the result of <span>sorting and
     normalizing a module specifier map</span> given <var>parsed</var>["<code
     data-x="">imports</code>"] and <var>baseURL</var>.</p></li>
    </ol>
   </li>

   <li><p>Let <var>sortedAndNormalizedScopes</var> be an empty <span>ordered map</span>.</p></li>

   <li>
    <p>If <var>parsed</var>["<code data-x="">scopes</code>"] <span data-x="map
    exists">exists</span>, then:</p>

    <ol>
     <li><p>If <var>parsed</var>["<code data-x="">scopes</code>"] is not an <span>ordered
     map</span>, then throw a <code>TypeError</code> indicating that the value for the "<code
     data-x="">scopes</code>" top-level key needs to be a JSON object.</p></li>

     <li><p>Set <var>sortedAndNormalizedScopes</var> to the result of <span>sorting and normalizing
     scopes</span> given <var>parsed</var>["<code data-x="">scopes</code>"] and
     <var>baseURL</var>.</p></li>
    </ol>
   </li>

   <li><p>Let <var>normalizedIntegrity</var> be an empty <span>ordered map</span>.</p></li>

   <li>
    <p>If <var>parsed</var>["<code data-x="">integrity</code>"] <span data-x="map
    exists">exists</span>, then:</p>

    <ol>
     <li><p>If <var>parsed</var>["<code data-x="">integrity</code>"] is not an <span>ordered
     map</span>, then throw a <code>TypeError</code> indicating that the value for the "<code
     data-x="">integrity</code>" top-level key needs to be a JSON object.</p></li>

     <li><p>Set <var>normalizedIntegrity</var> to the result of <span>normalizing a module
     integrity map</span> given <var>parsed</var>["<code data-x="">integrity</code>"] and
     <var>baseURL</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>If <var>parsed</var>'s <span data-x="map key">keys</span> <span data-x="list
    contains">contains</span> any items besides "<code data-x="">imports</code>", "<code
    data-x="">scopes</code>", or "<code data-x="">integrity</code>", then the user agent should
    <span>report a warning to the console</span> indicating that an invalid top-level key was
    present in the import map.</p>

    <p class="note">This can help detect typos. It is not an error, because that would prevent any
    future extensions from being added backward-compatibly.</p>
   </li>

   <li><p>Return an <span>import map</span> whose <span
   data-x="concept-import-map-imports">imports</span> are <var>sortedAndNormalizedImports</var>,
   whose <span data-x="concept-import-map-scopes">scopes</span> are
   <var>sortedAndNormalizedScopes</var>, and whose <span
   data-x="concept-import-map-integrity">integrity</span> are
   <var>normalizedIntegrity</var>.</p></li>
  </ol>

  <div class="example" id="example-import-map-normalization">
   <p>The <span>import map</span> that results from this parsing algorithm is highly normalized.
   For example, given a base URL of <code data-x="">https://example.com/base/page.html</code>, the
   input</p>

   <pre><code class="json" data-x="">{
  "imports": {
    "/app/helper": "node_modules/helper/index.mjs",
    "lodash": "/node_modules/lodash-es/lodash.js"
  }
}</code></pre>

   <p>will generate an <span>import map</span> with <span
   data-x="concept-import-map-imports">imports</span> of</p>

   <pre><code data-x="">«[
  "https://example.com/app/helper" → https://example.com/base/node_modules/helper/index.mjs
  "lodash" → https://example.com/node_modules/lodash-es/lodash.js
]»</code></pre>

   <p>and (despite nothing being present in the input string) an empty <span>ordered map</span> for
   its <span data-x="concept-import-map-scopes">scopes</span>.</p>
  </div>

  <hr>

  <p>To <dfn>merge module specifier maps</dfn>, given a <span>module specifier map</span>
  <var>newMap</var> and a <span>module specifier map</span> <var>oldMap</var>:</p>

  <ol>
   <li><p>Let <var>mergedMap</var> be a deep copy of <var>oldMap</var>.</p></li>
   <!-- XXX "deep copy" ought to be defined. -->

   <li>
    <p><span data-x="map iterate">For each</span> <var>specifier</var> →
    <var>url</var> of <var>newMap</var>:</p>

    <ol>
     <li>
      <p>If <var>specifier</var> <span data-x="map exists">exists</span> in
      <var>oldMap</var>, then:</p>

      <ol>
       <li><p>The user agent may <span>report a warning to the console</span> indicating the
       ignored rule. They may choose to avoid reporting if the rule is identical to an existing
       one.</p></li>

       <li><p><span>Continue</span>.</p></li>
      </ol>
     </li>

     <li><p>Set <var>mergedMap</var>[<var>specifier</var>] to <var>url</var>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>mergedMap</var>.</p></li>
  </ol>

  <p>To <dfn data-x="merge existing and new import maps">merge existing and new import maps</dfn>,
  given a <span data-x="concept-settings-object-global">global object</span> <var>global</var> and
  an <span>import map</span> <var>newImportMap</var>:</p>

  <ol>
   <li>
    <p>Let <var>newImportMapScopes</var> be a deep copy of <var>newImportMap</var>'s <span
    data-x="concept-import-map-scopes">scopes</span>.</p>

    <p class="note">We're mutating these copies and removing items from them when they are used to
    ignore scope-specific rules. This is true for <var>newImportMapScopes</var>, as well as to
    <var>newImportMapImports</var> below.</p>
   </li>

  <li><p>Let <var>oldImportMap</var> be <var>global</var>'s <span
  data-x="concept-global-import-map">import map</span>.</p></li>

   <li><p>Let <var>newImportMapImports</var> be a deep copy of <var>newImportMap</var>'s <span
   data-x="concept-import-map-imports">imports</span>.</p></li>
    <!-- XXX "deep copy" ought to be defined. -->

   <li>
    <p><span data-x="map iterate">For each</span> <var>scopePrefix</var> →
    <var>scopeImports</var> of <var>newImportMapScopes</var>:</p>

    <ol>
     <li>
      <p><span data-x="list iterate">For each</span> <var>record</var> of <var>global</var>'s
      <span>resolved module set</span>:</p>

      <ol>
       <li>
        <p>If <var>scopePrefix</var> is <var>record</var>'s <span
        data-x="specifier-resolution-record-serialized-base-url">serialized base URL</span>, or if
        <var>scopePrefix</var> ends with U+002F (/) and <var>scopePrefix</var> is a <span>code unit
        prefix</span> of <var>record</var>'s <span
        data-x="specifier-resolution-record-serialized-base-url">serialized base URL</span>,
        then:</p>

        <ol>
         <li>
          <p><span data-x="list iterate">For each</span> <var>specifierKey</var> →
          <var>resolutionResult</var> of <var>scopeImports</var>:</p>

          <ol>
           <li>
            <p>If <var>specifierKey</var> is <var>record</var>'s <span
            data-x="specifier-resolution-record-specifier">specifier</span>, or if all of the
            following conditions are true:</p>

            <ul>
             <li><p><var>specifierKey</var> ends with U+002F (/);</p></li>

             <li><p><var>specifierKey</var> is a <span>code unit prefix</span> of
             <var>record</var>'s <span
             data-x="specifier-resolution-record-specifier">specifier</span>;</p></li>

             <li><p>either <var>record</var>'s <span
             data-x="specifier-resolution-record-as-url">specifier as a URL</span> is null or
             <span>is special</span>,</p></li>
            </ul>

            <p>then:</p>

            <ol>
             <li><p>The user agent may <span>report a warning to the console</span> indicating the
             ignored rule. They may choose to avoid reporting if the rule is identical to an existing
             one.</p></li>

             <li><p>Remove <var>scopeImports</var>[<var>specifierKey</var>].</p></li>
            </ol>
           </li>
          </ol>
         </li>
        </ol>
       </li>
      </ol>

      <p class="note">Implementers are encouraged to implement a more efficient matching
      algorithm when working with the <span>resolved module set</span>. As guidance, the number of
      resolved/mapped modules in a large application can be on the order of thousands.</p>
     </li>

     <li><p>If <var>scopePrefix</var> <span data-x="map exists">exists</span> in
     <var>oldImportMap</var>'s <span data-x="concept-import-map-scopes">scopes</span>, then
     set <var>oldImportMap</var>'s <span
     data-x="concept-import-map-scopes">scopes</span>[<var>scopePrefix</var>] to the result of
     <span data-x="merge module specifier maps">merging module specifier maps</span>, given
     <var>scopeImports</var> and <var>oldImportMap</var>'s <span
     data-x="concept-import-map-scopes">scopes</span>[<var>scopePrefix</var>].</p></li>

     <li><p>Otherwise, set <var>oldImportMap</var>'s <span
     data-x="concept-import-map-scopes">scopes</span>[<var>scopePrefix</var>] to
     <var>scopeImports</var>.</p></li>
    </ol>
   </li>

   <li>
    <p><span data-x="map iterate">For each</span> <var>url</var> →
    <var>integrity</var> of <var>newImportMap</var>'s <span
    data-x="concept-import-map-integrity">integrity</span>:</p>

    <ol>
     <li>
      <p>If <var>url</var> <span data-x="map exists">exists</span> in <var>oldImportMap</var>'s
      <span data-x="concept-import-map-integrity">integrity</span>, then:</p>

      <ol>
       <li><p>The user agent may <span>report a warning to the console</span> indicating the
       ignored rule. They may choose to avoid reporting if the rule is identical to an existing
       one.</p></li>

       <li><p><span>Continue</span>.</p></li>
      </ol>
     </li>

     <li><p>Set <var>oldImportMap</var>'s <span
     data-x="concept-import-map-integrity">integrity</span>[<var>url</var>] to
     <var>integrity</var>.</p></li>
    </ol>
   </li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>record</var> of <var>global</var>'s
    <span>resolved module set</span>:</p>

    <ol>
     <li>
      <p><span data-x="list iterate">For each</span> <var>specifier</var> → <var>url</var> of
      <var>newImportMapImports</var>:</p>

      <ol>
       <li>
        <p>If <var>specifier</var> <span>starts with</span> <var>record</var>'s <span
        data-x="specifier-resolution-record-specifier">specifier</span>, then:</p>

        <ol>
         <li><p>The user agent may <span>report a warning to the console</span> indicating the
         ignored rule. They may choose to avoid reporting if the rule is identical to an existing
         one.</p></li>

         <li><p>Remove <var>newImportMapImports</var>[<var>specifier</var>].</p></li>
        </ol>
       </li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Set <var>oldImportMap</var>'s <span data-x="concept-import-map-imports">imports</span> to
   the result of <span data-x="merge module specifier maps">merge module specifier maps</span>,
   given <var>newImportMapImports</var> and <var>oldImportMap</var>'s <span
   data-x="concept-import-map-imports">imports</span>.</p></li>
  </ol>

  <p>The above algorithm merges a new import map into the given <span>environment settings
  object</span>'s <span>global object</span>'s <span>import map</span>. Let's examine a few
  examples:</p>

  <div class="example" id="example-import-map-merge-unrelated">
   <p>There are two cases when rules of the new import map don't get merged into the existing
   one.</p>

   <ol>
    <li><p>The new import map rule has the exact same scope and specifier as a rule in the existing
    import map. We'll call that "conflicting rule".</p></li>

    <li><p>The new import map rule may impact the resolution of an already resolved module. We'll
    call that "impacted already resolved module".</p></li>
   </ol>

   <p>When the new import map has no conflicting rules, and there are no impacted resolved modules,
   the resulting map would be a combination of the new and existing maps. Rules that would have
   individually impacted similar modules (e.g. "/app/" and "/app/helper") but are not an exact
   match are not conflicting, and all make it to the merged map.</p>

   <p>So, the following existing and new import maps:</p>
   <pre><code class="json" data-x="">{
   "imports": {
    "/app/": "./original-app/",
  }
}</code>
<code class="json" data-x="">{
  "imports": {
    "/app/helper": "./helper/index.mjs"
  },
  "scopes": {
    "/js": {
      "/app/": "./js-app/"
    }
  }
}</code></pre>
   <p>Would be equivalent to the following single import map:</p>
   <pre><code class="json" data-x="">{
  "imports": {
    "/app/": "./original-app/",
    "/app/helper": "./helper/index.mjs"
  },
  "scopes": {
    "/js": {
      "/app/": "./js-app/"
    }
  }
}</code></pre>
  </div>

  <div class="example" id="example-import-map-merge-conflict-imports">
   <p>When the new import map impacts an already resolved module, that rule gets dropped from the
   import map.</p>

   <p>So, if the <span>resolved module set</span> already contains the "<code
   data-x="">/app/helper</code>", the following new import map:</p>
   <pre><code class="json" data-x="">{
   "imports": {
    "/app/helper": "./helper/index.mjs",
    "lodash": "/node_modules/lodash-es/lodash.js"
  }
}</code></pre>
   <p>Would be equivalent to the following one:</p>
   <pre><code class="json" data-x="">{
  "imports": {
    "lodash": "/node_modules/lodash-es/lodash.js"
  }
}</code></pre>
  </div>

  <div class="example" id="example-import-map-merge-conflict-scopes">
   <p>The same is true for rules that impact already resolved modules defined in specific scopes.
   If we already resolved "<code data-x="">/app/helper</code>" from "<code
   data-x="">/app/main.mjs</code>" the following new import map:</p>
   <pre><code class="json" data-x="">{
  "scopes": {
    "/app/": {
      "/app/helper": "./helper/index.mjs"
    }
  },
   "imports": {
    "lodash": "/node_modules/lodash-es/lodash.js"
  }
}</code></pre>
   <p>Would similarly be equivalent to:</p>
   <pre><code class="json" data-x="">{
  "imports": {
    "lodash": "/node_modules/lodash-es/lodash.js"
  }
}</code></pre>
  </div>

  <div class="example" id="example-import-map-merge-conflict-imports-and-scopes">
   <p>We could also have cases where a single already-resolved module specifier has multiple rules
   for its resolution, depending on the referring script. In such cases, only the relevant rules
   would not be added to the map.</p>
   <p>For example, if we already resolved "<code data-x="">/app/helper</code>" from "<code
   data-x="">/app/vendor/main.mjs</code>", the following new import map:</p>
   <pre><code class="json" data-x="">{
  "scopes": {
    "/app/": {
      "/app/helper": "./helper/index.mjs"
    },
    "/app/vendor/": {
      "/app/": "./vendor_helper/"
    },
    "/vendor/": {
      "/app/helper": "./helper/vendor_index.mjs"
    }
  },
   "imports": {
    "lodash": "/node_modules/lodash-es/lodash.js"
    "/app/": "./general_app_path/"
    "/app/helper": "./other_path/helper/index.mjs"
  }
}</code></pre>
   <p>Would be equivalent to:</p>
   <pre><code class="json" data-x="">{
  "scopes": {
    "/vendor/": {
      "/app/helper": "./helper/vendor_index.mjs"
    }
  },
  "imports": {
    "lodash": "/node_modules/lodash-es/lodash.js"
  }
}</code></pre>

  <p>This is achieved by the fact that the merge algorithm tracks already resolved modules
  and removes rules affecting them from new import maps before they are merged into the existing
  one.</p>
  </div>

  <div class="example" id="example-import-map-merge-two-map-conflict">
   <p>When the new import map has conflicting rules to the existing import map, with no impacted
   already resolved modules, the existing import map rules persist.</p>

   <p>For example, the following existing and new import maps:</p>
   <pre><code class="json" data-x="">{
   "imports": {
    "/app/helper": "./helper/index.mjs",
    "lodash": "/node_modules/lodash-es/lodash.js"
  }
}</code>
<code class="json" data-x="">{
  "imports": {
    "/app/helper": "./main/helper/index.mjs"
  }
}</code></pre>
   <p>Would be equivalent to the following single import map:</p>
   <pre><code class="json" data-x="">{
  "imports": {
    "/app/helper": "./helper/index.mjs",
    "lodash": "/node_modules/lodash-es/lodash.js",
  }
}</code></pre>
  </div>

  <p>To <dfn data-x="sorting and normalizing a module specifier map">sort and normalize a module
  specifier map</dfn>, given an <span>ordered map</span> <var>originalMap</var> and a
  <span>URL</span> <var>baseURL</var>:</p>

  <ol>
   <li><p>Let <var>normalized</var> be an empty <span>ordered map</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>specifierKey</var> → <var>value</var> of
    <var>originalMap</var>:</p>

    <ol>
     <li><p>Let <var>normalizedSpecifierKey</var> be the result of <span>normalizing a specifier
     key</span> given <var>specifierKey</var> and <var>baseURL</var>.</p></li>

     <li><p>If <var>normalizedSpecifierKey</var> is null, then <span>continue</span>.</p></li>

     <li>
      <p>If <var>value</var> is not a <span>string</span>, then:</p>

      <ol>
       <li><p>The user agent may <span>report a warning to the console</span> indicating that
       addresses need to be strings.</p></li>

       <li><p>Set <var>normalized</var>[<var>normalizedSpecifierKey</var>] to null.</p></li>

       <li><p><span>Continue</span>.</p></li>
      </ol>
     </li>

     <li><p>Let <var>addressURL</var> be the result of <span>resolving a URL-like module
     specifier</span> given <var>value</var> and <var>baseURL</var>.</p></li>

     <li>
      <p>If <var>addressURL</var> is null, then:</p>

      <ol>
       <li><p>The user agent may <span>report a warning to the console</span> indicating that the
       address was invalid.</p></li>

       <li><p>Set <var>normalized</var>[<var>normalizedSpecifierKey</var>] to null.</p></li>

       <li><p><span>Continue</span>.</p></li>
      </ol>
     </li>

     <li>
      <p>If <var>specifierKey</var> ends with U+002F (/), and the <span
      data-x="concept-url-serializer">serialization</span> of <var>addressURL</var> does not end
      with U+002F (/), then:</p>

      <ol>
       <li><p>The user agent may <span>report a warning to the console</span> indicating that an
       invalid address was given for the specifier key <var>specifierKey</var>; since
       <var>specifierKey</var> ends with a slash, the address needs to as well.</p></li>

       <li><p>Set <var>normalized</var>[<var>normalizedSpecifierKey</var>] to null.</p></li>

       <li><p><span>Continue</span>.</p></li>
      </ol>
     </li>

     <li><p>Set <var>normalized</var>[<var>normalizedSpecifierKey</var>] to
     <var>addressURL</var>.</p></li>
    </ol>
   </li>

   <li><p>Return the result of <span data-x="map sort descending">sorting in descending order</span>
   <var>normalized</var>, with an entry <var>a</var> being less than an entry <var>b</var> if
   <var>a</var>'s <span data-x="map key">key</span> is <span>code unit less than</span>
   <var>b</var>'s <span data-x="map key">key</span>.</p></li>
  </ol>

  <p>To <dfn data-x="sorting and normalizing scopes">sort and normalize scopes</dfn>, given an
  <span>ordered map</span> <var>originalMap</var> and a <span>URL</span> <var>baseURL</var>:</p>

  <ol>
   <li><p>Let <var>normalized</var> be an empty <span>ordered map</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>scopePrefix</var> →
    <var>potentialSpecifierMap</var> of <var>originalMap</var>:</p>

    <ol>
     <li><p>If <var>potentialSpecifierMap</var> is not an <span>ordered map</span>, then throw a
     <code>TypeError</code> indicating that the value of the scope with prefix
     <var>scopePrefix</var> needs to be a JSON object.</p></li>

     <li><p>Let <var>scopePrefixURL</var> be the result of <span data-x="URL parser">URL
     parsing</span> <var>scopePrefix</var> with <var>baseURL</var>.</p></li>

     <li>
      <p>If <var>scopePrefixURL</var> is failure, then:</p>

      <ol>
       <li><p>The user agent may <span>report a warning to the console</span> that the scope prefix
       URL was not parseable.</p></li>

       <li><p><span>Continue</span>.</p></li>
      </ol>
     </li>

     <li><p>Let <var>normalizedScopePrefix</var> be the <span
     data-x="concept-url-serializer">serialization</span> of <var>scopePrefixURL</var>.</p></li>

     <li><p>Set <var>normalized</var>[<var>normalizedScopePrefix</var>] to the result of
     <span>sorting and normalizing a module specifier map</span> given
     <var>potentialSpecifierMap</var> and <var>baseURL</var>.</p></li>
    </ol>
   </li>

   <li><p>Return the result of <span data-x="map sort descending">sorting in descending
   order</span> <var>normalized</var>, with an entry <var>a</var> being less than an entry
   <var>b</var> if <var>a</var>'s <span data-x="map key">key</span> is <span>code unit less
   than</span> <var>b</var>'s <span data-x="map key">key</span>.</p></li>
  </ol>

  <p class="note">In the above two algorithms, sorting keys and scopes in descending order has the
  effect of putting "<code data-x="">foo/bar/</code>" before "<code data-x="">foo/</code>". This in
  turn gives "<code data-x="">foo/bar/</code>" a higher priority than "<code data-x="">foo/</code>"
  during <span data-x="resolve a module specifier">module specifier resolution</span>.</p>

  <p>To <dfn data-x="normalizing a module integrity map">normalize a module integrity map</dfn>,
  given an <span>ordered map</span> <var>originalMap</var>:</p>

  <ol>
   <li><p>Let <var>normalized</var> be an empty <span>ordered map</span>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>key</var> → <var>value</var> of
    <var>originalMap</var>:</p>

    <ol>
     <li>
      <p>Let <var>resolvedURL</var> be the result of <span>resolving a URL-like module
      specifier</span> given <var>key</var> and <var>baseURL</var>.</p>

      <p class="note">Unlike "<code data-x="">imports</code>", keys of the integrity map are treated
      as URLs, not module specifiers. However, we use the <span data-x="resolving a URL-like module
      specifier">resolve a URL-like module specifier</span> algorithm to prohibit "bare" relative
      URLs like <code data-x="">foo</code>, which could be mistaken for module specifiers.</p>
     </li>

     <li>
      <p>If <var>resolvedURL</var> is null, then:</p>

      <ol>
       <li><p>The user agent may <span>report a warning to the console</span> indicating that
       the key failed to resolve.</p></li>

       <li><p><span>Continue</span>.</p></li>
      </ol>
     </li>

     <li>
      <p>If <var>value</var> is not a <span>string</span>, then:</p>

      <ol>
       <li><p>The user agent may <span>report a warning to the console</span> indicating that
       <span data-x="concept-request-integrity-metadata">integrity metadata</span> values need to
       be <span data-x="string">strings</span>.</p></li>

       <li><p><span>Continue</span>.</p></li>
      </ol>
     </li>

     <li><p>Set <var>normalized</var>[<var>resolvedURL</var>] to <var>value</var>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>normalized</var>.</p></li>
  </ol>

  <p>To <dfn data-x="normalizing a specifier key">normalize a specifier key</dfn>, given a <span>string</span> <var>specifierKey</var> and a <span>URL</span> <var>baseURL</var>:</p>

  <ol>
   <li>
    <p>If <var>specifierKey</var> is the empty string, then:</p>

    <ol>
     <li><p>The user agent may <span>report a warning to the console</span> indicating that specifier keys may not be the empty string.</p></li>

     <li><p>Return null.</p></li>
    </ol>
   </li>

   <li><p>Let <var>url</var> be the result of <span>resolving a URL-like module specifier</span>, given <var>specifierKey</var> and <var>baseURL</var>.</p></li>

   <li><p>If <var>url</var> is not null, then return the <span data-x="concept-url-serializer">serialization</span> of <var>url</var>.</p></li>

   <li><p>Return <var>specifierKey</var>.</p></li>
  </ol>


  <h4>JavaScript specification host hooks</h4>

  <p>The JavaScript specification contains a number of <span>implementation-defined</span> abstract
  operations, that vary depending on the host environment. This section defines them for user
  agent hosts.</p>

  <h5 id="the-hostensurecanaddprivateelement-implementation"><dfn>HostEnsureCanAddPrivateElement</dfn>(<var>O</var>)</h5>

  <p>JavaScript contains an <span>implementation-defined</span> <span
  data-x="js-HostEnsureCanAddPrivateElement">HostEnsureCanAddPrivateElement</span>(<var>O</var>)
  abstract operation. User agents must use the following implementation: <ref>JAVASCRIPT</ref>

  <ol>
   <li><p>If <var>O</var> is a <code>WindowProxy</code> object, or <span>implements</span>
   <code>Location</code>, then return Completion { [[Type]]: throw, [[Value]]: a new
   <code>TypeError</code> }.</li>

   <li><p>Return <span>NormalCompletion</span>(unused).</p></li>
  </ol>

  <p class="note">JavaScript private fields can be applied to arbitrary objects. Since this can
  dramatically complicate implementation for particularly-exotic host objects, the JavaScript
  language specification provides this hook to allow hosts to reject private fields on objects
  meeting a host-defined criteria. In the case of HTML, <code>WindowProxy</code> and
  <code>Location</code> have complicated semantics &mdash; particularly around navigation and
  security &mdash; that make implementation of private field semantics challenging, so our
  implementation simply rejects those objects.</p>

  <h5><dfn
  data-x="the-hostensurecancompilestrings-implementation">HostEnsureCanCompileStrings</dfn>(<var>realm</var>,
  <var>parameterStrings</var>, <var>bodyString</var>, <var>codeString</var>, <var>compilationType</var>,
  <var>parameterArgs</var>, <var>bodyArg</var>)</h5>

  <p>JavaScript contains an <span>implementation-defined</span> <span
  data-x="js-HostEnsureCanCompileStrings">HostEnsureCanCompileStrings</span> abstract operation,
  redefined by the <cite>Dynamic Code Brand Checks</cite> proposal. User agents must use
  the following implementation: <ref>JAVASCRIPT</ref> <ref>JSDYNAMICCODEBRANDCHECKS</ref>

  <ol>
   <li><p>Perform ? <span
   data-x="csp-EnsureCSPDoesNotBlockStringCompilation">EnsureCSPDoesNotBlockStringCompilation</span>(<var>realm</var>,
   <var>parameterStrings</var>, <var>bodyString</var>, <var>codeString</var>,
   <var>compilationType</var>, <var>parameterArgs</var>, <var>bodyArg</var>).
   <ref>CSP</ref></p></li>
  </ol>

  <h5><dfn
  data-x="the-hostgetcodeforeval-implementation">HostGetCodeForEval</dfn>(<var>argument</var>)</h5>

  <p>The <cite>Dynamic Code Brand Checks</cite> proposal contains an
  <span>implementation-defined</span> <span
  data-x="js-HostGetCodeForEval">HostGetCodeForEval</span>(<var>argument</var>) abstract operation.
  User agents must use the following implementation: <ref>JSDYNAMICCODEBRANDCHECKS</ref>

  <ol>
   <li><p>If <var>argument</var> is a <code data-x="tt-trustedscript">TrustedScript</code> object,
   then return <var>argument</var>'s <span data-x="tt-trustedscript-data">data</span>.</p></li>

   <li><p>Otherwise, return no-code.</p></li>
  </ol>

  <h5 id="the-hostpromiserejectiontracker-implementation"><dfn>HostPromiseRejectionTracker</dfn>(<var>promise</var>, <var>operation</var>)</h5>

  <p>JavaScript contains an <span>implementation-defined</span> <span
  data-x="js-HostPromiseRejectionTracker">HostPromiseRejectionTracker</span>(<var>promise</var>,
  <var>operation</var>) abstract operation. User agents must use the following implementation:
  <ref>JAVASCRIPT</ref></p>

  <ol>
   <li><p>Let <var>script</var> be the <span>running script</span>.</p></li>

   <li><p>If <var>script</var> is a <span>classic script</span> and <var>script</var>'s <span>muted
   errors</span> is true, then return.</p></li>

   <li><p>Let <var>settingsObject</var> be the <span>current settings object</span>.</p></li>

   <li><p>If <var>script</var> is not null, then set <var>settingsObject</var> to
   <var>script</var>'s <span data-x="concept-script-settings-object">settings
   object</span>.</p></li>

   <li><p>Let <var>global</var> be <var>settingsObject</var>'s <span
   data-x="concept-settings-object-global">global object</span>.</p></li>

   <li>
    <p>If <var>operation</var> is "<code data-x="">reject</code>", then:</p>

    <ol>
     <li><p><span data-x="list append">Append</span> <var>promise</var> to <var>global</var>'s
     <span>about-to-be-notified rejected promises list</span>.</p></li>
    </ol>
   </li>

   <li>
    <p>If <var>operation</var> is "<code data-x="">handle</code>", then:</p>

    <ol>
     <li><p>If <var>global</var>'s <span>about-to-be-notified rejected promises list</span> <span
     data-x="list contains">contains</span> <var>promise</var>, then <span data-x="list
     remove">remove</span> <var>promise</var> from that list and return.</p></li>

     <li><p>If <var>global</var>'s <span>outstanding rejected promises weak set</span> does not
     <span data-x="list contains">contain</span> <var>promise</var>, then return.</p></li>

     <li><p><span data-x="list remove">Remove</span> <var>promise</var> from <var>global</var>'s
     <span>outstanding rejected promises weak set</span>.</p></li>

     <li><p><span>Queue a global task</span> on the <span>DOM manipulation task source</span> given
     <var>global</var> to <span data-x="concept-event-fire">fire an event</span> named <code
     data-x="event-rejectionhandled">rejectionhandled</code> at <var>global</var>, using
     <code>PromiseRejectionEvent</code>, with the <code
     data-x="dom-PromiseRejectionEvent-promise">promise</code> attribute initialized to
     <var>promise</var>, and the <code data-x="dom-PromiseRejectionEvent-reason">reason</code>
     attribute initialized to <var>promise</var>.[[PromiseResult]].</p></li>
    </ol>
   </li>
  </ol>

  <h5 id="hostsystemutcepochnanoseconds"><dfn>HostSystemUTCEpochNanoseconds</dfn>(<var>global</var>)</h5>

  <p>The Temporal proposal contains an <span>implementation-defined</span> <span
  data-x="js-HostSystemUTCEpochNanoseconds">HostSystemUTCEpochNanoseconds</span> abstract operation.
  User agents must use the following implementation: <ref>JSTEMPORAL</ref></p>

  <ol>
   <li><p>Let <var>settingsObject</var> be <var>global</var>'s <span>relevant settings
   object</span>.</p></li>

   <li><p>Let <var>time</var> be <var>settingsObject</var>'s <span>current wall
   time</span>.</p></li>

   <li><p>Let <var>ns</var> be the number of nanoseconds from the <span>Unix epoch</span> to
   <var>time</var>, rounded to the nearest integer.</p></li>

   <li><p>Return the result of <span>clamping</span> <var>ns</var> between
   <span>nsMinInstant</span> and <span>nsMaxInstant</span>.</p></li>
  </ol>

  <h5 id="integration-with-javascript-jobs">Job-related host hooks</h5>

  <p>The JavaScript specification defines Jobs to be scheduled and run later by the host, as well as
  <span data-x="JobCallback Record">JobCallback Records</span> which encapsulate JavaScript
  functions that are called as part of jobs. The JavaScript specification contains a number of
  <span>implementation-defined</span> abstract operations that lets the host define how jobs are
  scheduled and how JobCallbacks are handled. HTML uses these abstract operations to <span
  id="incumbent-settings-object-tracking-in-promises">track the <span>incumbent settings
  object</span> in promises and <code>FinalizationRegistry</code> callbacks</span> by saving and
  restoring the <span>incumbent settings object</span> and a <span>JavaScript execution
  context</span> for the <span>active script</span> in JobCallbacks. This section defines them for
  user agent hosts.</p>

  <h6 id="hostcalljobcallback"><dfn>HostCallJobCallback</dfn>(<var>callback</var>, <var>V</var>,
  <var>argumentsList</var>)</h6>

  <p>JavaScript contains an <span>implementation-defined</span> <span
  data-x="js-HostCallJobCallback">HostCallJobCallback</span>(<var>callback</var>, <var>V</var>,
  <var>argumentsList</var>) abstract operation to let hosts restore state when invoking JavaScript
  callbacks from inside tasks. User agents must use the following implementation:
  <ref>JAVASCRIPT</ref></p>

  <ol>
   <li><p>Let <var>incumbent settings</var> be
   <var>callback</var>.[[HostDefined]].[[IncumbentSettings]].</p></li>

   <li><p>Let <var>script execution context</var> be
   <var>callback</var>.[[HostDefined]].[[ActiveScriptContext]].</li>

   <li>
    <p><span>Prepare to run a callback</span> with <var>incumbent settings</var>.</p>

    <p class="note">This affects the <span data-x="concept-incumbent-everything">incumbent</span>
    concept while the callback runs.</p>
   </li>

   <li>
    <p>If <var>script execution context</var> is not null, then <span data-x="stack
    push">push</span> <var>script execution context</var> onto the <span>JavaScript execution
    context stack</span>.</p>

    <p class="note">This affects the <span>active script</span> while the callback runs.</p>
   </li>

   <li><p>Let <var>result</var> be <span>Call</span>(<var>callback</var>.[[Callback]], <var>V</var>,
   <var>argumentsList</var>).</p></li>

   <li><p>If <var>script execution context</var> is not null, then <span data-x="stack
   pop">pop</span> <var>script execution context</var> from the <span>JavaScript execution context
   stack</span>.</p></li>

   <li><p><span>Clean up after running a callback</span> with <var>incumbent
   settings</var>.</p></li>

   <li><p>Return <var>result</var>.</p></li>
  </ol>

  <h6
  id="hostenqueuefinalizationregistrycleanupjob"><dfn>HostEnqueueFinalizationRegistryCleanupJob</dfn>(<var>finalizationRegistry</var>)</h6>

  <p>JavaScript has the ability to register objects with <code>FinalizationRegistry</code> objects,
  in order to schedule a cleanup action if they are found to be garbage collected. The JavaScript
  specification contains an <span>implementation-defined</span> <span
  data-x="js-HostEnqueueFinalizationRegistryCleanupJob">HostEnqueueFinalizationRegistryCleanupJob</span>(<var>finalizationRegistry</var>)
  abstract operation to schedule the cleanup action.</p>

  <p class="note">The timing and occurrence of cleanup work is <span>implementation-defined</span>
  in the JavaScript specification. User agents might differ in when and whether an object is garbage
  collected, affecting both whether the return value of the <code>WeakRef.prototype.deref()</code>
  method is undefined, and whether <code>FinalizationRegistry</code> cleanup callbacks occur. There
  are well-known cases in popular web browsers where objects are not accessible to JavaScript, but
  they remain retained by the garbage collector indefinitely. HTML clears kept-alive
  <code>WeakRef</code> objects in the <span>perform a microtask checkpoint</span> algorithm. Authors
  would be best off not depending on the timing details of garbage collection implementations.</p>

  <p>Cleanup actions do not take place interspersed with synchronous JavaScript execution, but
  rather happen in queued <span data-x="concept-task">tasks</span>. User agents must use the
  following implementation: <ref>JAVASCRIPT</ref></p>

  <ol>
   <li><p>Let <var>global</var> be <var>finalizationRegistry</var>.[[Realm]]'s <span>global
   object</span>.</p></li>

   <li>
    <p><span>Queue a global task</span> on the <dfn>JavaScript engine task source</dfn> given
    <var>global</var> to perform the following steps:</p>

    <ol>
     <li><p>Let <var>entry</var> be
     <var>finalizationRegistry</var>.[[CleanupCallback]].[[Callback]].[[Realm]]'s <span
     data-x="concept-realm-settings-object">environment settings object</span>.</p></li>

     <li><p><span>Check if we can run script</span> with <var>entry</var>. If this returns "do not
     run", then return.</p></li>

     <li>
      <p><span>Prepare to run script</span> with <var>entry</var>.</p>

      <p class="note">This affects the <span data-x="concept-entry-everything">entry</span> concept
      while the cleanup callback runs.</p>
     </li>

     <li><p>Let <var>result</var> be the result of performing
     <span>CleanupFinalizationRegistry</span>(<var>finalizationRegistry</var>).</p></li>

     <li><p><span>Clean up after running script</span> with <var>entry</var>.</p></li>

     <li><p>If <var>result</var> is an <span data-x="Completion Record">abrupt completion</span>,
     then <span>report an exception</span> given by <var>result</var>.[[Value]] for
     <var>global</var>.</p></li>
    </ol>
   </li>
  </ol>

  <h6 id="hostenqueuegenericjob"><dfn>HostEnqueueGenericJob</dfn>(<var>job</var>,
  <var>realm</var>)</h6>

  <p>JavaScript contains an <span>implementation-defined</span> <span
  data-x="js-HostEnqueueGenericJob">HostEnqueueGenericJob</span>(<var>job</var>, <var>realm</var>)
  abstract operation to perform generic jobs in a particular realm (e.g., resolve promises resulting
  from <code>Atomics.waitAsync</code>). User agents must use the following implementation:
  <ref>JAVASCRIPT</ref></p>

  <ol>
   <li><p>Let <var>global</var> be <var>realm</var>'s <span data-x="concept-realm-global">global
   object</span>.</p></li>

   <li><p><span>Queue a global task</span> on the <span>JavaScript engine task source</span> given
   <var>global</var> to perform <var>job</var>().</p></li>
  </ol>

  <span id="integration-with-the-javascript-job-queue"></span>
  <span id="enqueuejob(queuename,-job,-arguments)"></span>
  <h6 id="hostenqueuepromisejob"><dfn>HostEnqueuePromiseJob</dfn>(<var>job</var>,
  <var>realm</var>)</h6>

  <p>JavaScript contains an <span>implementation-defined</span> <span
  data-x="js-HostEnqueuePromiseJob">HostEnqueuePromiseJob</span>(<var>job</var>, <var>realm</var>)
  abstract operation to schedule Promise-related operations. HTML schedules these operations in the
  microtask queue. User agents must use the following implementation: <ref>JAVASCRIPT</ref></p>

  <ol>
   <li>
    <p>If <var>realm</var> is not null, then let <var>job settings</var> be the <span
    data-x="concept-realm-settings-object">settings object</span> for <var>realm</var>. Otherwise,
    let <var>job settings</var> be null.</p>

     <div class="note">
      <p>If <var>realm</var> is not null, it is the <span>realm</span> of the author code that will
      run. When <var>job</var> is returned by <span>NewPromiseReactionJob</span>, it is the realm of
      the promise's handler function. When <var>job</var> is returned by
      <span>NewPromiseResolveThenableJob</span>, it is the realm of the <code data-x="">then</code>
      function.

      <p>If <var>realm</var> is null, either no author code will run or author code is guaranteed to
      throw. For the former, the author may not have passed in code to run, such as in <code
      data-x="">promise.then(null, null)</code>. For the latter, it is because a revoked Proxy was
      passed. In both cases, all the steps below that would otherwise use <var>job settings</var>
      get skipped.</p>

      <p class="XXX"><a
      href="https://tc39.es/ecma262/#sec-newpromiseresolvethenablejob">NewPromiseResolveThenableJob</a>
      and <a href="https://tc39.es/ecma262/#sec-newpromisereactionjob">NewPromiseReactionJob</a>
      both seem to provide non-null realms (the current Realm Record) in the case of a revoked
      proxy. The previous text could be updated to reflect that.</p>
     </div>
   </li>

   <li>
    <p><span>Queue a microtask</span> to perform the following steps:</p>

    <ol>
     <li><p>If <var>job settings</var> is not null, then <span>check if we can run script</span>
     with <var>job settings</var>. If this returns "do not run" then return.</p></li>

     <li>
      <p>If <var>job settings</var> is not null, then <span>prepare to run script</span> with
      <var>job settings</var>.</p>

      <p class="note">This affects the <span data-x="concept-entry-everything">entry</span> concept
      while the job runs.</p>
     </li>

     <li>
      <p>Let <var>result</var> be <var>job</var>().</p>

      <p class="note"><var>job</var> is an <span>abstract closure</span> returned by
      <span>NewPromiseReactionJob</span> or <span>NewPromiseResolveThenableJob</span>. The promise's
      handler function when <var>job</var> is returned by <span>NewPromiseReactionJob</span>, and
      the <code data-x="">then</code> function when <var>job</var> is returned by
      <span>NewPromiseResolveThenableJob</span>, are wrapped in <span data-x="JobCallback
      Record">JobCallback Records</span>. HTML saves the <span>incumbent settings object</span> and
      a <span>JavaScript execution context</span> for to the <span>active script</span> in
      <span>HostMakeJobCallback</span> and restores them in <span>HostCallJobCallback</span>.</p>
     </li>

     <li><p>If <var>job settings</var> is not null, then <span>clean up after running script</span>
     with <var>job settings</var>.</p></li>

     <li>
      <p>If <var>result</var> is an <span data-x="Completion Record">abrupt completion</span>, then
      <span>report an exception</span> given by <var>result</var>.[[Value]] for <var>realm</var>'s
      <span data-x="concept-realm-global">global object</span>.</p>

      <p class="XXX">There is a very gnarly case where HostEnqueuePromiseJob is called with a null
      realm (e.g., because Promise.prototype.then was called with null handlers) but also the job
      returns abruptly (because the promise capability's resolve or reject handler threw, possibly
      because this is a subclass of Promise that takes the supplied functions and wraps them in
      throwing functions before passing them on to the function passed to the Promise superclass
      constructor. Which global is to be used then, considering that the current realm could be
      different at each of those steps, by using a Promise constructor or Promise.prototype.then
      from another realm? See <a href="https://github.com/whatwg/html/issues/10526">issue
      #10526</a>.</p>
     </li>
    </ol>
   </li>
  </ol>

  <h6 id="hostenqueuetimeoutjob"><dfn>HostEnqueueTimeoutJob</dfn>(<var>job</var>, <var>realm</var>,
  <var>milliseconds</var>)</h6>

  <p>JavaScript contains an <span>implementation-defined</span> <span
  data-x="js-HostEnqueueTimeoutJob">HostEnqueueTimeoutJob</span>(<var>job</var>,
  <var>milliseconds</var>) abstract operation to schedule an operation to be performed after a
  timeout. HTML schedules these operations using <span>run steps after a timeout</span>. User agents
  must use the following implementation: <ref>JAVASCRIPT</ref></p>

  <ol>
   <li><p>Let <var>global</var> be <var>realm</var>'s <span data-x="concept-realm-global">global
   object</span>.</p></li>

   <li><p>Let <var>timeoutStep</var> be an algorithm step which <span data-x="queue a global
   task">queues a global task</span> on the <span>JavaScript engine task source</span> given
   <var>global</var> to perform <var>job</var>().</p></li>

   <li><p><span>Run steps after a timeout</span> given <var>global</var>, "<code
   data-x="">JavaScript</code>", <var>milliseconds</var>, and <var>timeoutStep</var>.</p></li>
  </ol>

  <h6 id="hostmakejobcallback"><dfn>HostMakeJobCallback</dfn>(<var>callable</var>)</h6>

  <p>JavaScript contains an <span>implementation-defined</span> <span
  data-x="js-HostMakeJobCallback">HostMakeJobCallback</span>(<var>callable</var>) abstract operation
  to let hosts attach state to JavaScript callbacks that are called from inside <span
  data-x="concept-task">task</span>s. User agents must use the following implementation:
  <ref>JAVASCRIPT</ref></p>

  <ol>
   <li><p>Let <var>incumbent settings</var> be the <span>incumbent settings object</span>.</p></li>

   <li><p>Let <var>active script</var> be the <span>active script</span>.</p></li>

   <li><p>Let <var>script execution context</var> be null.</p></li>

   <li>
    <p>If <var>active script</var> is not null, set <var>script execution context</var> to a new
    <span>JavaScript execution context</span>, with its Function field set to null, its Realm field
    set to <var>active script</var>'s <span data-x="concept-script-settings-object">settings
    object</span>'s <span data-x="environment settings object's realm">realm</span>, and its
    ScriptOrModule set to <var>active script</var>'s
    <span data-x="concept-script-record">record</span>.</p>

    <p class="note">As seen below, this is used in order to propagate the current <span>active
    script</span> forward to the time when the job callback is invoked.</p>

    <div class="example">
     <p>A case where <var>active script</var> is non-null, and saving it in this way is useful, is
     the following:</p>

     <pre><code class="js" data-x="">Promise.resolve('import(`./example.mjs`)').then(eval);</code></pre>

     <p>Without this step (and the steps that use it in <span>HostCallJobCallback</span>), there
     would be no <span>active script</span> when the <code>import()</code> expression is evaluated,
     since <code>eval()</code> is a built-in function that does not originate from any particular
     <span data-x="concept-script">script</span>.</p>

     <p>With this step in place, the active script is propagated from the above code into the job,
     allowing <code>import()</code> to use the original script's <span
     data-x="concept-script-base-url">base URL</span> appropriately.</p>
    </div>

    <div class="example">
     <p><var>active script</var> can be null if the user clicks on the following button:</p>

     <pre><code class="html" data-x="">&lt;button onclick="Promise.resolve('import(`./example.mjs`)').then(eval)">Click me&lt;/button></code></pre>

     <p>In this case, the JavaScript function for the <span data-x="event handlers">event
     handler</span> will be created by the <span data-x="getting the current value of the event
     handler">get the current value of the event handler</span> algorithm, which creates a function
     with null [[ScriptOrModule]] value. Thus, when the promise machinery calls
     <span>HostMakeJobCallback</span>, there will be no <span>active script</span> to pass
     along.</p>

     <p>As a consequence, this means that when the <code>import()</code> expression is evaluated,
     there will still be no <span>active script</span>. Fortunately that is handled by our
     implementation of <span>HostLoadImportedModule</span> by falling back to using the
     <span>current settings object</span>'s <span>API base URL</span>.</p>
    </div>
   </li>

   <li><p>Return the <span>JobCallback Record</span> { [[Callback]]: <var>callable</var>,
   [[HostDefined]]: { [[IncumbentSettings]]: <var>incumbent settings</var>, [[ActiveScriptContext]]:
   <var>script execution context</var> } }.</p></li>
  </ol>

  </div>


  <h5 id="integration-with-the-javascript-module-system">Module-related host hooks</h5>

  <p>The JavaScript specification defines a syntax for modules, as well as some host-agnostic parts
  of their processing model. This specification defines the rest of their processing model: how the
  module system is bootstrapped, via the <code>script</code> element with <code
  data-x="attr-script-type">type</code> attribute set to "<code data-x="">module</code>", and how
  modules are fetched, resolved, and executed. <ref>JAVASCRIPT</ref></p>

  <p class="note">Although the JavaScript specification speaks in terms of "scripts" versus
  "modules", in general this specification speaks in terms of <span data-x="classic script">classic
  scripts</span> versus <span data-x="module script">module scripts</span>, since both of them use
  the <code>script</code> element.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>modulePromise</var> = <span data-x="import()">import(<var>specifier</var>)</span></code></dt>

   <dd>
    <p>Returns a promise for the module namespace object for the <span>module script</span>
    identified by <var>specifier</var>. This allows dynamic importing of module scripts at runtime,
    instead of statically using the <code data-x="">import</code> statement form. The specifier will
    be <span data-x="resolve a module specifier">resolved</span> relative to the <span>active
    script</span>.</p>

    <p>The returned promise will be rejected if an invalid specifier is given, or if a failure is
    encountered while <span data-x="HostLoadImportedModule">fetching</span> or evaluating the
    resulting module graph.</p>

    <p>This syntax can be used inside both <span data-x="classic script">classic</span> and <span
    data-x="module script">module scripts</span>. It thus provides a bridge into the module-script
    world, from the classic-script world.</p>
   </dd>

   <dt><code data-x=""><var>url</var> = <span data-x="import.meta">import.meta</span>.<span data-x="import-meta-url">url</span></code></dt>

   <dd>
    <p>Returns the <span data-x="active script">active module script</span>'s <span
    data-x="concept-script-base-url">base URL</span>.</p>

    <p>This syntax can only be used inside <span data-x="module script">module scripts</span>.</p>
   </dd>

   <dt><code data-x=""><var>url</var> = <span data-x="import.meta">import.meta</span>.<span data-x="import-meta-resolve">resolve</span>(<var>specifier</var>)</code></dt>

   <dd>
    <p>Returns <var>specifier</var>, <span data-x="resolve a module specifier">resolved</span>
    relative to the <span>active script</span>. That is, this returns the URL that would be imported
    by using <code data-x="import()">import(<var>specifier</var>)</code>.</p>

    <p>Throws a <code>TypeError</code> exception if an invalid specifier is given.</p>

    <p>This syntax can only be used inside <span data-x="module script">module scripts</span>.</p>
   </dd>
  </dl>

  <p><span w-nodev>A <dfn>module map</dfn> is a <span data-x="ordered map">map</span> keyed by <span
  data-x="tuple">tuples</span> consisting of a <span>URL record</span> and a <span>string</span>.
  The <span>URL record</span> is the <span data-x="concept-request-url">request URL</span> at which
  the module was fetched, and the <span>string</span> indicates the type of the module (e.g. "<code
  data-x="">javascript-or-wasm</code>"). The <span>module map</span>'s values are either a
  <span>module script</span>, null (used to represent failed fetches), or a placeholder value "<code
  data-x="">fetching</code>". </span><span data-x="module map">Module maps</span> are used to ensure
  that imported module scripts are only fetched, parsed, and evaluated once per
  <code>Document</code> or <a href="#workers">worker</a>.</p>

  <div class="example">
   <p>Since <span data-x="module map">module maps</span> are keyed by (URL, module type), the
   following code will create three separate entries in the <span>module map</span>, since it
   results in three different (URL, module type) <span data-x="tuple">tuples</span> (all with "<code
   data-x="">javascript-or-wasm</code>" type):</p>

   <pre><code class="js">import "https://example.com/module.mjs";
import "https://example.com/module.mjs#map-buster";
import "https://example.com/module.mjs?debug=true";</code></pre>

   <p>That is, URL <span data-x="concept-url-query">queries</span> and <span
   data-x="concept-url-fragment">fragments</span> can be varied to create distinct entries in the
   <span>module map</span>; they are not ignored. Thus, three separate fetches and three separate
   module evaluations will be performed.</p>

   <p>In contrast, the following code would only create a single entry in the <span>module
   map</span>, since after applying the <span>URL parser</span> to these inputs, the resulting <span
   data-x="URL record">URL records</span> are equal:</p>

   <pre><code class="js">import "https://example.com/module2.mjs";
import "https:example.com/module2.mjs";
import "https://///example.com\\module2.mjs";
import "https://example.com/foo/../module2.mjs";</code></pre>

   <p>So in this second example, only one fetch and one module evaluation will occur.</p>

   <p>Note that this behavior is the same as how <span
   data-x="SharedWorker">shared workers</span> are keyed by their parsed <span
   data-x="concept-SharedWorkerGlobalScope-constructor-url">constructor url</span>.</p>
  </div>

  <div class="example">
   <p>Since module type is also part of the <span>module map</span> key, the following code will
   create two separate entries in the <span>module map</span> (the type is "<code
   data-x="">javascript-or-wasm</code>" for the first, and "<code data-x="">css</code>" for the
   second):</p>

   <pre><code class="html" data-x="">&lt;script type=module&gt;
  import "https://example.com/module";
&lt;/script&gt;
&lt;script type=module&gt;
  import "https://example.com/module" with { type: "css" };
&lt;/script&gt;</code></pre>

   <p>This can result in two separate fetches and two separate module evaluations being performed.</p>

   <p class="XXX">In practice, due to the as-yet-unspecified memory cache (see issue <a
   href="https://github.com/whatwg/html/issues/6110">#6110</a>) the resource may only be fetched
   once in WebKit and Blink-based browsers. Additionally, as long as all module types are mutually
   exclusive, the module type check in <span>fetch a single module script</span> will fail for at
   least one of the imports, so at most one module evaluation will occur.</p>

   <p>The purpose of including the type in the <span>module map</span> key is so that an import
   with the wrong type attribute does not prevent a different import of the same specifier but with
   the correct type from succeeding.</p>
  </div>

  <div class="example">
   <p>JavaScript module scripts are the default import type when importing from another JavaScript
    module; that is, when an <code data-x="">import</code> statement lacks a <code
    data-x="">type</code> import attribute the imported module script's type will be JavaScript.
    Attempting to import a JavaScript resource using an <code data-x="">import</code> statement with
    a <code data-x="">type</code> import attribute will fail:</p>

    <pre><code class="html">&lt;script type="module">
    // All of the following will fail, assuming that the imported .mjs files are served with a
    // JavaScript MIME type. JavaScript module scripts are the default and cannot be imported with
    // any import type attribute.
    import foo from "./foo.mjs" with { type: "javascript" };
    import foo2 from "./foo2.mjs" with { type: "js" };
    import foo3 from "./foo3.mjs" with { type: "" };
    await import("./foo4.mjs", { with: { type: null } });
    await import("./foo5.mjs", { with: { type: undefined } });
&lt;/script></code></pre>
  </div>

  <div w-nodev>


  <h6
  id="hostgetimportmetaproperties"><dfn>HostGetImportMetaProperties</dfn>(<var>moduleRecord</var>)</h6>

  <p>JavaScript contains an <span>implementation-defined</span> <span
  data-x="js-HostGetImportMetaProperties">HostGetImportMetaProperties</span> abstract operation.
  User agents must use the following implementation: <ref>JAVASCRIPT</ref></p>

  <ol>
   <li><p>Let <var>moduleScript</var> be <var>moduleRecord</var>.[[HostDefined]].</p></li>

   <li><p><span>Assert</span>: <var>moduleScript</var>'s <span data-x="concept-script-base-url">base
   URL</span> is not null, as <var>moduleScript</var> is a <span>JavaScript module
   script</span>.</p></li>

   <li><p>Let <var>urlString</var> be <var>moduleScript</var>'s <span
   data-x="concept-script-base-url">base URL</span>, <span
   data-x="concept-url-serializer">serialized</span>.</p></li>

   <li>
    <p>Let <var>steps</var> be the following steps, given the argument <var>specifier</var>:</p>

    <ol>
     <li><p>Set <var>specifier</var> to ? <span>ToString</span>(<var>specifier</var>).</p></li>

     <li><p>Let <var>url</var> be the result of <span data-x="resolve a module specifier">resolving
     a module specifier</span> given <var>moduleScript</var> and <var>specifier</var>.</p></li>

     <li><p>Return the <span data-x="concept-url-serializer">serialization</span> of
     <var>url</var>.</p></li>
    </ol>
   </li>

   <li><p>Let <var>resolveFunction</var> be ! <span>CreateBuiltinFunction</span>(<var>steps</var>,
   1, "<code data-x="">resolve</code>", « »).</p></li>

   <li><p>Return « <span>Record</span> { [[Key]]: "<dfn><code
   data-x="import-meta-url">url</code></dfn>", [[Value]]: <var>urlString</var> },
   <span>Record</span> { [[Key]]: "<dfn><code data-x="import-meta-resolve">resolve</code></dfn>",
   [[Value]]: <var>resolveFunction</var> } ».</p></li>
  </ol>

  <h6 id="hostgetsupportedimportattributes"><span
  id="hostgetsupportedimportassertions"></span><dfn>HostGetSupportedImportAttributes</dfn>()</h6>

  <p>JavaScript contains an an <span>implementation-defined</span> <span
  data-x="js-HostGetSupportedImportAttributes">HostGetSupportedImportAttributes</span> abstract
  operation. User agents must use the following implementation: <ref>JAVASCRIPT</ref></p>

  <ol>
   <li><p>Return « "<code data-x="">type</code>" ».</p></li>
  </ol>

  <h6 id="hostloadimportedmodule"><dfn>HostLoadImportedModule</dfn>(<var>referrer</var>,
  <var>moduleRequest</var>, <var>loadState</var>, <var>payload</var>)</h6>

  <p>JavaScript contains an <span>implementation-defined</span> <span
  data-x="js-HostLoadImportedModule">HostLoadImportedModule</span> abstract operation. User agents
  must use the following implementation: <ref>JAVASCRIPT</ref></p>

  <ol>
   <li><p>Let <var>settingsObject</var> be the <span>current settings object</span>.</p></li>

   <li>
    <p>If <var>settingsObject</var>'s <span data-x="concept-settings-object-global">global
    object</span> implements <code>WorkletGlobalScope</code> or <code>ServiceWorkerGlobalScope</code>
    and <var>loadState</var> is undefined, then:</p>

    <p class="note"><var>loadState</var> is undefined when the current fetching process has been
    initiated by a dynamic <code>import()</code> call, either directly or when loading the
    transitive dependencies of the dynamically imported module.</p>

    <ol>
     <li><p>Let <var>completion</var> be <span>Completion Record</span> { [[Type]]: throw,
     [[Value]]: a new <code>TypeError</code>, [[Target]]: empty }.</p></li>

     <li><p>Perform <span>FinishLoadingImportedModule</span>(<var>referrer</var>,
     <var>moduleRequest</var>, <var>payload</var>, <var>completion</var>).</p></li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li><p>Let <var>referencingScript</var> be null.</p></li>

   <li><p>Let <var>originalFetchOptions</var> be the <span>default script fetch
   options</span>.</p></li>

   <li><p>Let <var>fetchReferrer</var> be "<code data-x="">client</code>".</p></li>

   <li>
    <p>If <var>referrer</var> is a <span>Script Record</span> or a <span>Cyclic Module
    Record</span>, then:</p>

    <ol>
     <li><p>Set <var>referencingScript</var> to <var>referrer</var>.[[HostDefined]].</p>

     <li><p>Set <var>settingsObject</var> to <var>referencingScript</var>'s <span
     data-x="concept-script-settings-object">settings object</span>.</p></li>

     <li><p>Set <var>fetchReferrer</var> to <var>referencingScript</var>'s <span
     data-x="concept-script-base-url">base URL</span>.</p></li>

     <li><p>Set <var>originalFetchOptions</var> to <var>referencingScript</var>'s <span
     data-x="concept-script-script-fetch-options">fetch options</span>.</p></li>
    </ol>

    <div class="example">
     <p><var>referrer</var> is usually a <span>Script Record</span> or a <span>Cyclic Module
     Record</span>, but it will not be so for event handlers per the <span data-x="getting the
     current value of the event handler">get the current value of the event handler</span>
     algorithm. For example, given:</p>

     <pre><code class="html">&lt;button onclick="import('./foo.mjs')">Click me&lt;/button></code></pre>

     <p>If a <code data-x="event-click">click</code> event occurs, then at the time the
     <code>import()</code> expression runs, <span>GetActiveScriptOrModule</span> will return null,
     and this operation will receive the <span>current realm</span> as a fallback
     <var>referrer</var>.</p>
    </div>
   </li>

   <li>
    <p>If <var>referrer</var> is a <span>Cyclic Module Record</span> and <var>moduleRequest</var>
    is equal to the first element of <var>referrer</var>.[[RequestedModules]], then:</p>

    <ol>
     <li id="validate-requested-module-specifiers">
      <p><span data-x="list iterate">For each</span> <span>ModuleRequest record</span>
      <var>requested</var> of <var>referrer</var>.[[RequestedModules]]:</p>

      <ol>
       <li>
        <p>If <var>moduleRequest</var>.[[Attributes]] contains a <span>Record</span> <var>entry</var>
        such that <var>entry</var>.[[Key]] is not "<code data-x="">type</code>", then:</p>

        <ol>
         <li><p>Let <var>completion</var> be <span>Completion Record</span> { [[Type]]: throw,
         [[Value]]: a new <code>SyntaxError</code> exception, [[Target]]: empty }.</p></li>

         <li><p>Perform <span>FinishLoadingImportedModule</span>(<var>referrer</var>,
         <var>moduleRequest</var>, <var>payload</var>, <var>completion</var>).</p></li>

         <li><p>Return.</p></li>
        </ol>

        <p class="note">The JavaScript specification re-performs this validation but it is duplicated
        here to avoid unnecessarily loading any of the dependencies on validation failure.</p>
       </li>

       <li><p><span>Resolve a module specifier</span> given <var>referencingScript</var> and
       <var>moduleRequest</var>.[[Specifier]], catching any exceptions. If they throw an exception, let
       <var>resolutionError</var> be the thrown exception.</p></li>

       <li>
        <p>If the previous step threw an exception, then:</p>

        <ol>
         <li><p>Let <var>completion</var> be <span>Completion Record</span> { [[Type]]: throw,
         [[Value]]: <var>resolutionError</var>, [[Target]]: empty }.</p></li>

         <li><p>Perform <span>FinishLoadingImportedModule</span>(<var>referrer</var>,
         <var>moduleRequest</var>, <var>payload</var>, <var>completion</var>).</p></li>

         <li><p>Return.</p></li>
        </ol>
       </li>

       <li><p>Let <var>moduleType</var> be the result of running the <span>module type from module
       request</span> steps given <var>moduleRequest</var>.</p></li>

       <li>
        <p>If the result of running the <span>module type allowed</span> steps given
        <var>moduleType</var> and <var>settingsObject</var> is false, then:</p>

        <ol>
         <li><p>Let <var>completion</var> be <span>Completion Record</span> { [[Type]]: throw,
         [[Value]]: a new <code>TypeError</code> exception, [[Target]]: empty }.</p></li>

         <li><p>Perform <span>FinishLoadingImportedModule</span>(<var>referrer</var>,
         <var>moduleRequest</var>, <var>payload</var>, <var>completion</var>).</p></li>

         <li><p>Return.</p></li>
        </ol>
       </li>
      </ol>

      <p class="note">This step is essentially validating all of the requested module specifiers and
      type attributes when the first call to <span>HostLoadImportedModule</span> for a static module
      dependency list is made, to avoid further loading operations in the case any one of the
      dependencies has a static error. We treat a module with unresolvable module specifiers or
      unsupported type attributes the same as one that cannot be parsed; in both cases, a syntactic
      issue makes it impossible to ever contemplate linking the module later.</p>
     </li>
    </ol>
   </li>

   <li><p>Let <var>url</var> be the result of <span data-x="resolve a module specifier">resolving a
   module specifier</span> given <var>referencingScript</var> and
   <var>moduleRequest</var>.[[Specifier]], catching any exceptions. If they throw an exception, let
   <var>resolutionError</var> be the thrown exception.</p></li>

   <li>
    <p>If the previous step threw an exception, then:</p>

    <ol>
     <li><p>Let <var>completion</var> be <span>Completion Record</span> { [[Type]]: throw,
     [[Value]]: <var>resolutionError</var>, [[Target]]: empty }.</p></li>

     <li><p>Perform <span>FinishLoadingImportedModule</span>(<var>referrer</var>,
     <var>moduleRequest</var>, <var>payload</var>, <var>completion</var>).</p></li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li><p>Let <var>fetchOptions</var> be the result of <span data-x="get the descendant script
   fetch options">getting the descendant script fetch options</span> given
   <var>originalFetchOptions</var>, <var>url</var>, and <var>settingsObject</var>.</p></li>

   <li><p>Let <var>destination</var> be <code data-x="">"script"</code>.</p></li>

   <li><p>Let <var>fetchClient</var> be <var>settingsObject</var>.</p></li>

   <li>
    <p>If <var>loadState</var> is not undefined, then:</p>

    <ol>
     <li><p>Set <var>destination</var> to <var>loadState</var>.[[Destination]].</p></li>

     <li><p>Set <var>fetchClient</var> to <var>loadState</var>.[[FetchClient]].</p></li>
    </ol>
   </li>

   <li>
    <p><span>Fetch a single imported module script</span> given <var>url</var>,
    <var>fetchClient</var>, <var>destination</var>, <var>fetchOptions</var>,
    <var>settingsObject</var>, <var>fetchReferrer</var>, <var>moduleRequest</var>, and
    <var>onSingleFetchComplete</var> as defined below. If <var>loadState</var> is not undefined and
    <var>loadState</var>.[[PerformFetch]] is not null, pass <var>loadState</var>.[[PerformFetch]]
    along as well.</p>

    <p><var>onSingleFetchComplete</var> given <var>moduleScript</var> is the following
    algorithm:</p>

    <ol>
     <li><p>Let <var>completion</var> be null.</p></li>

     <li><p>If <var>moduleScript</var> is null, then set <var>completion</var> to <span>Completion
     Record</span> { [[Type]]: throw, [[Value]]: a new <code>TypeError</code>, [[Target]]: empty
     }.</p></li>

     <li>
      <p>Otherwise, if <var>moduleScript</var>'s <span data-x="concept-script-parse-error">parse
      error</span> is not null, then:</p>

      <ol>
       <li><p>Let <var>parseError</var> be <var>moduleScript</var>'s <span
       data-x="concept-script-parse-error">parse error</span>.</p></li>

       <li><p>Set <var>completion</var> to <span>Completion Record</span> { [[Type]]: throw,
       [[Value]]: <var>parseError</var>, [[Target]]: empty }.</p></li>

       <li><p>If <var>loadState</var> is not undefined and <var>loadState</var>.[[ParseError]] is
       null, set <var>loadState</var>.[[ParseError]] to <var>parseError</var>.</p></li>
      </ol>
     </li>

     <li><p>Otherwise, set <var>completion</var> to <span>Completion Record</span> { [[Type]]:
     normal, [[Value]]: <var>moduleScript</var>'s
     <span data-x="concept-script-record">record</span>, [[Target]]: empty }.</p></li>

     <li><p>Perform <span>FinishLoadingImportedModule</span>(<var>referrer</var>,
     <var>moduleRequest</var>, <var>payload</var>, <var>completion</var>).</p></li>
    </ol>
   </li>
  </ol>

  </div>

  <h4>Event loops</h4>

  <h5 w-nodev>Definitions</h5>

  <p>To coordinate events, user interaction, scripts, rendering, networking, and so forth, user
  agents must use <dfn data-x="event loop" data-lt="event loop" export>event loops</dfn> as
  described in this section. Each <span>agent</span> has an associated <dfn export for="agent"
  data-x="concept-agent-event-loop">event loop</dfn>, which is unique to that agent.</p>

  <p>The <span data-x="concept-agent-event-loop">event loop</span> of a <span>similar-origin window
  agent</span> is known as a <dfn>window event loop</dfn>. The <span
  data-x="concept-agent-event-loop">event loop</span> of a <span>dedicated worker agent</span>,
  <span>shared worker agent</span>, or <span>service worker agent</span> is known as a <dfn>worker
  event loop</dfn>. And the <span data-x="concept-agent-event-loop">event loop</span> of a
  <span>worklet agent</span> is known as a <dfn>worklet event loop</dfn>.</p>

  <div class="note">
   <p><span data-x="event loop">Event loops</span> do not necessarily correspond to implementation
   threads. For example, multiple <span data-x="window event loop">window event loops</span> could
   be cooperatively scheduled in a single thread.</p>

   <p>However, for the various worker <span data-x="agent">agents</span> that are allocated with
   [[CanBlock]] set to true, the JavaScript specification does place requirements on them regarding
   <span>forward progress</span>, which effectively amount to requiring dedicated per-agent threads
   in those cases.</p>
  </div>

  <div w-nodev>

  <hr>

  <p>An <span>event loop</span> has one or more <dfn data-x="task queue">task queues</dfn>. A
  <span>task queue</span> is a <span>set</span> of <span data-x="concept-task">tasks</span>.</p>

  <p class="note"><span data-x="task queue">Task queues</span> are <span data-x="set">sets</span>,
  not <span data-x="queue">queues</span>, because the <a href="#event-loop-processing-model">event
  loop processing model</a> grabs the first <span
  data-x="concept-task-runnable"><em>runnable</em></span> <span data-x="concept-task">task</span>
  from the chosen queue, instead of <span data-x="dequeue">dequeuing</span> the first task.</p>

  <p class="note">The <span>microtask queue</span> is not a <span>task queue</span>.</p>

  <p>Tasks encapsulate algorithms that are responsible for such work as:</p>

  <dl>
   <dt>Events</dt>
   <dd>
    <p>Dispatching an <code>Event</code> object at a particular
    <code>EventTarget</code> object is often done by a dedicated task.</p>

    <p class="note">Not all events are dispatched using the <span>task queue</span>; many are
    dispatched during other tasks.</p>
   </dd>

   <dt>Parsing</dt>
   <dd><p>The <span>HTML parser</span> tokenizing one or more bytes, and then processing any
   resulting tokens, is typically a task.</p></dd>

   <dt>Callbacks</dt>
   <dd><p>Calling a callback is often done by a dedicated task.</p></dd>


   <dt>Using a resource</dt>
   <dd><p>When an algorithm <span data-x="concept-fetch">fetches</span> a resource, if the fetching
   occurs in a non-blocking fashion then the processing of the resource once some or all of the
   resource is available is performed by a task.</p></dd>

   <dt>Reacting to DOM manipulation</dt>
   <dd><p>Some elements have tasks that trigger in response to DOM manipulation, e.g. when that
   element is <span data-x="node is inserted into a document">inserted into the document</span>.</p>
  </dl>

  <p>Formally, a <dfn data-x="concept-task" data-lt="task" export>task</dfn> is a
  <span>struct</span> which has:</p>

  <dl>
   <dt><dfn data-x="concept-task-steps">Steps</dfn></dt>
   <dd>A series of steps specifying the work to be done by the task.</dd>

   <dt>A <dfn data-x="concept-task-source">source</dfn></dt>
   <dd>One of the <span data-x="task source">task sources</span>, used to group and serialize
   related tasks.</dd>

   <dt>A <dfn data-x="concept-task-document">document</dfn></dt>
   <dd>A <code>Document</code> associated with the task, or null for tasks that are not in a
   <span>window event loop</span>.</dd>

   <dt>A <dfn export>script evaluation environment settings object set</dfn></dt>
   <dd>A <span>set</span> of <span data-x="environment settings object">environment settings
   objects</span> used for tracking script evaluation during the task.</dd>
  </dl>

  <p>A <span data-x="concept-task">task</span> is <dfn data-x="concept-task-runnable">runnable</dfn>
  if its <span data-x="concept-task-document">document</span> is either null or <span>fully
  active</span>.</p>

  <p>Per its <span data-x="concept-task-source">source</span> field, each <span
  data-x="concept-task">task</span> is defined as coming from a specific <dfn export>task
  source</dfn>. For each <span>event loop</span>, every <span>task source</span> must be associated
  with a specific <span>task queue</span>.</p>

  <p class="note">Essentially, <span data-x="task source">task sources</span> are used within
  standards to separate logically-different types of tasks, which a user agent might wish to
  distinguish between. <span data-x="task queue">Task queues</span> are used by user agents to
  coalesce task sources within a given <span>event loop</span>.</p>

  <p class="example">For example, a user agent could have one <span>task queue</span> for mouse and
  key events (to which the <span>user interaction task source</span> is associated), and another to
  which all other <span data-x="task source">task sources</span> are associated. Then, using the
  freedom granted in the initial step of the <a href="#event-loop-processing-model">event loop
  processing model</a>, it could give keyboard and mouse events preference over other tasks
  three-quarters of the time, keeping the interface responsive but not starving other task queues.
  Note that in this setup, the processing model still enforces that the user agent would never
  process events from any one <span>task source</span> out of order.</p>

  <hr>

  <p>Each <span>event loop</span> has a <dfn>currently running task</dfn>, which is either a <span
  data-x="concept-task">task</span> or null. Initially, this is null. It is used to handle
  reentrancy.</p>

  <p>Each <span>event loop</span> has a <dfn>microtask queue</dfn>, which is a <span>queue</span> of
  <span data-x="microtask">microtasks</span>, initially empty. A <dfn export>microtask</dfn> is a
  colloquial way of referring to a <span data-x="concept-task">task</span> that was created via the
  <span>queue a microtask</span> algorithm.</p>

  <p>Each <span>event loop</span> has a <dfn>performing a microtask checkpoint</dfn> boolean,
  which is initially false. It is used to prevent reentrant invocation of the <span>perform a
  microtask checkpoint</span> algorithm.</p>

  <p>Each <span>window event loop</span> has a <code>DOMHighResTimeStamp</code>
  <dfn>last render opportunity time</dfn>, initially set to zero.</p>

  <p>Each <span>window event loop</span> has a <code>DOMHighResTimeStamp</code>
  <dfn>last idle period start time</dfn>, initially set to zero.</p>

  <p>To get the <dfn>same-loop windows</dfn> for a <span>window event loop</span> <var>loop</var>,
  return all <code>Window</code> objects whose <span>relevant agent</span>'s
  <span data-x="concept-agent-event-loop">event loop</span> is <var>loop</var>.</p>


  <h5>Queuing tasks</h5>

  <p>To <dfn export>queue a task</dfn> on a <span>task source</span> <var>source</var>, which
  performs a series of steps <var>steps</var>, optionally given an event loop <var>event loop</var>
  and a document <var>document</var>:</p>

  <ol>
   <li><p>If <var>event loop</var> was not given, set <var>event loop</var> to the <span>implied
   event loop</span>.</p></li>

   <li><p>If <var>document</var> was not given, set <var>document</var> to the <span>implied
   document</span>.</p></li>

   <li><p>Let <var>task</var> be a new <span data-x="concept-task">task</span>.</p></li>

   <li><p>Set <var>task</var>'s <span data-x="concept-task-steps">steps</span> to
   <var>steps</var>.</p></li>

   <li><p>Set <var>task</var>'s <span data-x="concept-task-source">source</span> to
   <var>source</var>.</p></li>

   <li><p>Set <var>task</var>'s <span data-x="concept-task-document">document</span> to the
   <var>document</var>.</p></li>

   <li><p>Set <var>task</var>'s <span>script evaluation environment settings object set</span> to an
   empty <span>set</span>.</p></li>

   <li><p>Let <var>queue</var> be the <span>task queue</span> to which <var>source</var> is
   associated on <var>event loop</var>.</p></li>

   <li><p><span data-x="list append">Append</span> <var>task</var> to <var>queue</var>.</p></li>
  </ol>

  <p class="warning">Failing to pass an event loop and document to <span>queue a task</span> means
  relying on the ambiguous and poorly-specified <span>implied event loop</span> and <span>implied
  document</span> concepts. Specification authors should either always pass these values, or use the
  wrapper algorithms <span>queue a global task</span> or <span>queue an element task</span>
  instead. Using the wrapper algorithms is recommended.</p>

  <p>To <dfn export>queue a global task</dfn> on a <span>task source</span> <var>source</var>, with
  a <span>global object</span> <var>global</var> and a series of steps <var>steps</var>:</p>

  <ol>
   <li><p>Let <var>event loop</var> be <var>global</var>'s <span>relevant agent</span>'s <span
   data-x="concept-agent-event-loop">event loop</span>.</p></li>

   <li><p>Let <var>document</var> be <var>global</var>'s <span
   data-x="concept-document-window">associated <code>Document</code></span>, if <var>global</var> is
   a <code>Window</code> object; otherwise null.</p>

   <li><p><span>Queue a task</span> given <var>source</var>, <var>event loop</var>,
   <var>document</var>, and <var>steps</var>.</p></li>
  </ol>

  <p>To <dfn export>queue an element task</dfn> on a <span>task source</span> <var>source</var>,
  with an element <var>element</var> and a series of steps <var>steps</var>:</p>

  <ol>
   <li><p>Let <var>global</var> be <var>element</var>'s <span>relevant global
   object</span>.</p></li>

   <li><p><span>Queue a global task</span> given <var>source</var>, <var>global</var>, and
   <var>steps</var>.</p></li>
  </ol>

  <p>To <dfn export>queue a microtask</dfn> which performs a series of steps <var>steps</var>,
  optionally given a document <var>document</var>:</p>

  <ol>
   <li><p><span>Assert</span>: there is a <span>surrounding agent</span>. I.e., this algorithm is
   not called while <span>in parallel</span>.</p></li>

   <li><p>Let <var>eventLoop</var> be the <span>surrounding agent</span>'s <span
   data-x="concept-agent-event-loop">event loop</span>.</p></li>

   <li><p>If <var>document</var> was not given, set <var>document</var> to the <span>implied
   document</span>.</p></li>

   <li><p>Let <var>microtask</var> be a new <span data-x="concept-task">task</span>.</p></li>

   <li><p>Set <var>microtask</var>'s <span data-x="concept-task-steps">steps</span> to
   <var>steps</var>.</p></li>

   <li><p>Set <var>microtask</var>'s <span data-x="concept-task-source">source</span> to the
   <dfn>microtask task source</dfn>.</p></li>

   <li><p>Set <var>microtask</var>'s <span data-x="concept-task-document">document</span> to
   <var>document</var>.</p></li>

   <li><p>Set <var>microtask</var>'s <span>script evaluation environment settings object set</span>
   to an empty <span>set</span>.</p></li>

   <li><p><span>Enqueue</span> <var>microtask</var> on <var>eventLoop</var>'s <span>microtask
   queue</span>.</p></li>
  </ol>

  <p class="note">It is possible for a <span>microtask</span> to be moved to a regular <span>task
  queue</span>, if, during its initial execution, it <span data-x="spin the event loop">spins the
  event loop</span>. This is the only case in which the <span
  data-x="concept-task-source">source</span>, <span data-x="concept-task-document">document</span>,
  and <span>script evaluation environment settings object set</span> of the microtask are consulted;
  they are ignored by the <span>perform a microtask checkpoint</span> algorithm.</p>

  <p>The <dfn>implied event loop</dfn> when queuing a task is the one that can deduced from the
  context of the calling algorithm. This is generally unambiguous, as most specification algorithms
  only ever involve a single <span>agent</span> (and thus a single <span>event loop</span>). The
  exception is algorithms involving or specifying cross-agent communication (e.g., between a window
  and a worker); for those cases, the <span>implied event loop</span> concept must not be relied
  upon and specifications must explicitly provide an <span>event loop</span> when <span
  data-x="queue a task">queuing a task</span>.</p>

  <p>The <dfn>implied document</dfn> when queuing a task on an <span>event loop</span> <var>event
  loop</var> is determined as follows:</p>

  <ol>
   <li><p>If <var>event loop</var> is not a <span>window event loop</span>, then return
   null.</p></li>

   <li><p>If the task is being queued in the context of an element, then return the element's
   <span>node document</span>.</p></li>

   <li><p>If the task is being queued in the context of a <span>browsing context</span>, then return
   the browsing context's <span>active document</span>.</p></li>

   <li><p>If the task is being queued by or for a <span data-x="concept-script">script</span>, then
   return the script's <span data-x="concept-script-settings-object">settings object</span>'s <span
   data-x="concept-settings-object-global">global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p><span>Assert</span>: this step is never reached, because one of the previous conditions is
   true. <span class="&#x0058;&#x0058;&#x0058;">Really?</span></p></li>
  </ol>

  <p class="&#x0058;&#x0058;&#x0058;">Both <span>implied event loop</span> and <span>implied
  document</span> are vaguely-defined and have a lot of action-at-a-distance. The hope is to remove
  these, especially <span>implied document</span>. See <a
  href="https://github.com/whatwg/html/issues/4980">issue #4980</a>.</p>


  <h5 id="event-loop-processing-model" data-lt="event loop processing model" dfn><span
  id="processing-model-8"></span>Processing model</h5>
  <!-- EVENT LOOP -->

  <p>An <span>event loop</span> must continually run through the following steps for as long as it
  exists:</p>

  <!-- Don't refer to "step 1" or link to #step1 directly; we need to move steps around too often
       for such references to survive. Just link to "event loop processing model". (The id="step1"
       is kept for old documents which do not follow this advice.) -->
  <ol id="step1">
   <li><p>Let <var>oldestTask</var> and <var>taskStartTime</var> be null.</p></li>

   <li>
    <p>If the <span>event loop</span> has a <span>task queue</span> with at least one <span
    data-x="concept-task-runnable">runnable</span> <span data-x="concept-task">task</span>,
    then:</p>

    <ol>
     <li>
      <p>Let <var>taskQueue</var> be one such <span>task queue</span>, chosen in an
      <span>implementation-defined</span> manner.</p>

      <p class="note">Remember that the <span>microtask queue</span> is not a <span>task
      queue</span>, so it will not be chosen in this step. However, a <span>task queue</span> to
      which the <span>microtask task source</span> is associated might be chosen in this step. In
      that case, the <span data-x="concept-task">task</span> chosen in the next step was originally
      a <span>microtask</span>, but it got moved as part of <span data-x="spin the event
      loop">spinning the event loop</span>.</p>
     </li>

     <li><p>Set <var>taskStartTime</var> to the <span>unsafe shared current time</span>.</p></li>

     <li><p>Set <var>oldestTask</var> to the first <span
     data-x="concept-task-runnable">runnable</span> <span data-x="concept-task">task</span> in
     <var>taskQueue</var>, and <span data-x="list remove">remove</span> it from
     <var>taskQueue</var>.</p></li>

     <li><p>If <var>oldestTask</var>'s <span data-x="concept-task-document">document</span> is not
     null, then <span>record task start time</span> given <var>taskStartTime</var> and
     <var>oldestTask</var>'s <span data-x="concept-task-document">document</span>.</p></li>

     <li><p>Set the <span>event loop</span>'s <span>currently running task</span> to
     <var>oldestTask</var>.</p></li>

     <li><p>Perform <var>oldestTask</var>'s <span data-x="concept-task-steps">steps</span>.</p></li>

     <li><p>Set the <span>event loop</span>'s <span>currently running task</span> back to
     null.</p></li>

     <li><p><span>Perform a microtask checkpoint</span>.</p></li>
    </ol>
   </li>

   <li><p>Let <var>taskEndTime</var> be the <span>unsafe shared current time</span>.
   <ref>HRT</ref></p></li>

   <li>
    <p>If <var>oldestTask</var> is not null, then:</p>

    <ol>
     <li><p>Let <var>top-level browsing contexts</var> be an empty <span>set</span>.</p></li>

     <li>
      <p>For each <span>environment settings object</span> <var>settings</var> of
      <var>oldestTask</var>'s <span>script evaluation environment settings object set</span>:</p>

      <ol>
       <li><p>Let <var>global</var> be <var>settings</var>'s <span
       data-x="concept-settings-object-global">global object</span>.</p></li>

       <li><p>If <var>global</var> is not a <code>Window</code> object, then
       <span>continue</span>.</p></li>

       <li><p>If <var>global</var>'s <span data-x="window bc">browsing context</span> is null, then
       <span>continue</span>.</p></li>

       <li><p>Let <var>tlbc</var> be <var>global</var>'s <span data-x="window bc">browsing
       context</span>'s <span data-x="bc-tlbc">top-level browsing context</span>.</p></li>

       <li><p>If <var>tlbc</var> is not null, then <span data-x="set append">append</span> it to
       <var>top-level browsing contexts</var>.</p></li>
      </ol>
     </li>

     <li><p><span>Report long tasks</span>, passing in <var>taskStartTime</var>,
     <var>taskEndTime</var>, <var>top-level browsing contexts</var>, and
     <var>oldestTask</var>.</p></li>

     <li><p>If <var>oldestTask</var>'s <span data-x="concept-task-document">document</span> is not
     null, then <span>record task end time</span> given <var>taskEndTime</var> and
     <var>oldestTask</var>'s <span data-x="concept-task-document">document</span>.</p></li>
    </ol>
   </li>

   <li id="idle-deadline-computation">
    <p>If this is a <span>window event loop</span> that has no <span
    data-x="concept-task-runnable"><em>runnable</em></span> <span data-x="concept-task">task</span>
    in this <span>event loop</span>'s <span data-x="task queue">task queues</span>, then:</p>

    <ol>
     <li><p>Set this <span>event loop</span>'s <span>last idle period start time</span> to the
     <span>unsafe shared current time</span>.</p></li>

     <li>
      <p>Let <var>computeDeadline</var> be the following steps:</p>

      <ol>
       <li>
        <p>Let <var>deadline</var> be this <span>event loop</span>'s <span>last idle period start
        time</span> plus 50.</p>

        <p class="note">The cap of 50ms in the future is to ensure responsiveness to new user input
        within the threshold of human perception.</p>
       </li>

       <li><p>Let <var>hasPendingRenders</var> be false.</p></li>

       <li>
        <p>For each <var>windowInSameLoop</var> of the <span>same-loop windows</span> for this
        <span>event loop</span>:</p>

        <ol>
         <li><p>If <var>windowInSameLoop</var>'s <span>map of animation frame callbacks</span> is
         not <span data-x="map empty">empty</span>, or if the user agent believes that the
         <var>windowInSameLoop</var> might have pending rendering updates, set
         <var>hasPendingRenders</var> to true.</p></li>

         <li><p>Let <var>timerCallbackEstimates</var> be the result of <span data-x="map get the
         values">getting the values</span> of <var>windowInSameLoop</var>'s <span>map of active
         timers</span>.</p></li>

         <li><p>For each <var>timeoutDeadline</var> of <var>timerCallbackEstimates</var>, if
         <var>timeoutDeadline</var> is less than <var>deadline</var>, set <var>deadline</var> to
         <var>timeoutDeadline</var>.</p></li>
        </ol>
       </li>

       <li>
        <p>If <var>hasPendingRenders</var> is true, then:</p>

        <ol>
         <li>
          <p>Let <var>nextRenderDeadline</var> be this <span>event loop</span>'s <span>last render
          opportunity time</span> plus (1000 divided by the current refresh rate).</p>

          <p>The refresh rate can be hardware- or implementation-specific. For a refresh rate of
          60Hz, the <var>nextRenderDeadline</var> would be about 16.67ms after the <span>last
          render opportunity time</span>.</p>
         </li>

         <li><p>If <var>nextRenderDeadline</var> is less than <var>deadline</var>, then return
         <var>nextRenderDeadline</var>.</p></li>
        </ol>
       </li>

       <li><p>Return <var>deadline</var>.</p></li>
      </ol>
     </li>

     <li><p>For each <var>win</var> of the <span>same-loop windows</span> for this <span>event
     loop</span>, perform the <span>start an idle period algorithm</span> for <var>win</var> with
     the following step: return the result of calling <var>computeDeadline</var>, <span
     data-x="coarsen time">coarsened</span> given <var>win</var>'s <span>relevant settings
     object</span>'s <span
     data-x="concept-settings-object-cross-origin-isolated-capability">cross-origin isolated
     capability</span>. <ref>REQUESTIDLECALLBACK</ref></p></li>
    </ol>
   </li>

   <li>
    <p>If this is a <span>worker event loop</span>, then:</p>

    <ol>
     <li>
      <p>If this <span>event loop</span>'s <span>agent</span>'s single <span>realm</span>'s <span
      data-x="concept-realm-global">global object</span> is a <span
      data-x="concept-AnimationFrameProvider-supported">supported</span>
      <code>DedicatedWorkerGlobalScope</code> and the user agent believes that it would benefit
      from having its rendering updated at this time, then:</p>
      <!-- TODO: grab the realm from the task. https://github.com/whatwg/html/issues/4342 -->

      <ol>
       <li><p>Let <var>now</var> be the <span>current high resolution time</span> given the
       <code>DedicatedWorkerGlobalScope</code>. <ref>HRT</ref></p></li>

       <li><p><span>Run the animation frame callbacks</span> for that
       <code>DedicatedWorkerGlobalScope</code>, passing in <var>now</var> as the
       timestamp.</p></li>

       <li><p>Update the rendering of that dedicated worker to reflect the current state.</p></li>
      </ol>

      <p class="note">Similar to the notes for <span data-x="update the rendering">updating the
      rendering</span> in a <span>window event loop</span>, a user agent can determine the rate of
      rendering in the dedicated worker.</p>
     </li>

     <li><p>If there are no <span data-x="concept-task">tasks</span> in the <span>event
     loop</span>'s <span data-x="task queue">task queues</span> and the
     <code>WorkerGlobalScope</code> object's <span
     data-x="dom-WorkerGlobalScope-closing">closing</span> flag is true, then destroy the
     <span>event loop</span>, aborting these steps, resuming the <span>run a worker</span> steps
     described in the <a href="#workers">Web workers</a> section below.</p></li>
    </ol>
   </li>
  </ol>

  <p>A <span>window event loop</span> <var>eventLoop</var> must also run the following <span>in
  parallel</span>, as long as it exists:</p>

  <ol>
   <li><p>Wait until at least one <span>navigable</span> whose <span data-x="nav-document">active
   document</span>'s <span>relevant agent</span>'s <span data-x="concept-agent-event-loop">event
   loop</span> is <var>eventLoop</var> might have a <span>rendering opportunity</span>.</p></li>

   <li><p>Set <var>eventLoop</var>'s <span>last render opportunity time</span> to the <span>unsafe
   shared current time</span>.</p></li>

   <li>
    <p>For each <var>navigable</var> that has a <span>rendering opportunity</span>, <span>queue a
    global task</span> on the <span>rendering task source</span> given <var>navigable</var>'s <span
    data-x="nav-window">active window</span> to <dfn export>update the rendering</dfn>:</p>

    <p class="note">This might cause redundant calls to <span>update the rendering</span>. However,
    these calls would have no observable effect because there will be no rendering necessary, as
    per the <i>Unnecessary rendering</i> step. Implementations can introduce further optimizations
    such as only queuing this task when it is not already queued. However, note that the document
    associated with the task might become inactive before the task is processed.</p>

    <ol>
     <li><p>Let <var>frameTimestamp</var> be <var>eventLoop</var>'s <span>last render opportunity
     time</span>.</p></li>

     <li>
      <p>Let <var>docs</var> be all <span>fully active</span> <code>Document</code> objects whose
      <span>relevant agent</span>'s <span data-x="concept-agent-event-loop">event loop</span> is
      <var>eventLoop</var>, sorted arbitrarily except that the following conditions must be
      met:</p>

      <ul>
       <li><p>Any <code>Document</code> <var>B</var> whose <span
       data-x="doc-container-document">container document</span> is <var>A</var> must be listed
       after <var>A</var> in the list.</p></li>

       <li><p>If there are two documents <var>A</var> and <var>B</var> that both have the same
       non-null <span data-x="doc-container-document">container document</span> <var>C</var>, then
       the order of <var>A</var> and <var>B</var> in the list must match the <span>shadow-including
       tree order</span> of their respective <span data-x="navigable container">navigable
       containers</span> in <var>C</var>'s <span>node tree</span>.</p></li>
      </ul>

      <p>In the steps below that iterate over <var>docs</var>, each <code>Document</code> must be
      processed in the order it is found in the list.</p>
     </li>

     <li>
      <p><i>Filter non-renderable documents</i>: Remove from <var>docs</var> any
      <code>Document</code> object <var>doc</var> for which any of the following are true:

      <ul>
       <li><p><var>doc</var> is <span>render-blocked</span>;</p></li>

       <li><p><var>doc</var>'s <span>visibility state</span> is "<code
       data-x="">hidden</code>";</p></li>

       <li><p><var>doc</var>'s rendering is <span data-x="rendering suppression for view
       transitions">suppressed for view transitions</span>; or</p></li>

       <li><p><var>doc</var>'s <span>node navigable</span> doesn't currently have a <span>rendering
       opportunity</span>.</p></li>
      </ul>

      <p class="note">We have to check for rendering opportunities here, in addition to checking
      that in the <span>in parallel</span> steps, as some documents that share the same <span>event
      loop</span> might not have a <span>rendering opportunity</span> at the same time.</p>
     </li>

     <li>
      <p><i>Unnecessary rendering</i>: Remove from <var>docs</var> any <code>Document</code> object
      <var>doc</var> for which all of the following are true:</p>

      <ul>
       <li><p>the user agent believes that updating the rendering of <var>doc</var>'s <span>node
       navigable</span> would have no visible effect; and</p></li>

       <li><p><var>doc</var>'s <span>map of animation frame callbacks</span> is empty.</p></li>
      </ul>
     </li>

     <li>
      <p>Remove from <var>docs</var> all <code>Document</code> objects for which the user agent
      believes that it's preferable to skip updating the rendering for other reasons.</p>

      <div class="note">
       <p>The step labeled <i>Filter non-renderable documents</i> prevents the user agent from
       updating the rendering when it is unable to present new content to the user.</p>

       <p>The step labeled <i>Unnecessary rendering</i> prevents the user agent from updating the
       rendering when there's no new content to draw.</p>

       <p>This step enables the user agent to prevent the steps below from running for other
       reasons, for example, to ensure certain <span data-x="concept-task">tasks</span> are
       executed immediately after each other, with only <span data-x="perform a microtask
       checkpoint">microtask checkpoints</span> interleaved (and without, e.g., <span data-x="run
       the animation frame callbacks">animation frame callbacks</span> interleaved). Concretely, a
       user agent might wish to coalesce timer callbacks together, with no intermediate rendering
       updates.</p>
      </div>
     </li>

     <li><p>For each <var>doc</var> of <var>docs</var>, <span>reveal</span> <var>doc</var>.</p></li>

     <li><p>For each <var>doc</var> of <var>docs</var>, <span>flush autofocus candidates</span> for
     <var>doc</var> if its <span>node navigable</span> is a <span>top-level
     traversable</span>.</p></li>

     <li><p>For each <var>doc</var> of <var>docs</var>, <span>run the resize steps</span> for
     <var>doc</var>. <ref>CSSOMVIEW</ref></p></li>

     <li><p>For each <var>doc</var> of <var>docs</var>, <span>run the scroll steps</span> for
     <var>doc</var>. <ref>CSSOMVIEW</ref></p></li>

     <li><p>For each <var>doc</var> of <var>docs</var>, <span>evaluate media queries and report
     changes</span> for <var>doc</var>. <ref>CSSOMVIEW</ref></p></li>

     <li><p>For each <var>doc</var> of <var>docs</var>, <span>update animations and send
     events</span> for <var>doc</var>, passing in <span>relative high resolution time</span> given
     <var>frameTimestamp</var> and <var>doc</var>'s <span>relevant global object</span> as the
     timestamp <ref>WEBANIMATIONS</ref></p></li>

     <li><p>For each <var>doc</var> of <var>docs</var>, <span>run the fullscreen steps</span> for
     <var>doc</var>. <ref>FULLSCREEN</ref></p></li>

     <li>
      <p>For each <var>doc</var> of <var>docs</var>, if the user agent detects that the backing
      storage associated with a <code>CanvasRenderingContext2D</code> or an
      <code>OffscreenCanvasRenderingContext2D</code>, <var>context</var>, has been lost, then it
      must run the <dfn>context lost steps</dfn> for each such <var>context</var>:</p>

      <ol>
       <li><p>Let <var>canvas</var> be the value of <var>context</var>'s <code
       data-x="dom-context-2d-canvas">canvas</code> attribute, if <var>context</var> is a
       <code>CanvasRenderingContext2D</code>, or the <span>associated <code>OffscreenCanvas</code>
       object</span> for <var>context</var> otherwise.</p></li>

       <li><p>Set <var>context</var>'s <span data-x="concept-canvas-context-lost">context
       lost</span> to true.</p></li>

       <li><p><span>Reset the rendering context to its default state</span> given
       <var>context</var>.</p></li>

       <li><p>Let <var>shouldRestore</var> be the result of <span
       data-x="concept-event-fire">firing an event</span> named <code
       data-x="event-contextlost">contextlost</code> at <var>canvas</var>, with the <code
       data-x="dom-Event-cancelable">cancelable</code> attribute initialized to true.</p></li>

       <li><p>If <var>shouldRestore</var> is false, then abort these steps.</p></li>

       <li><p>Attempt to restore <var>context</var> by creating a backing storage using
       <var>context</var>'s attributes and associating them with <var>context</var>. If this fails,
       then abort these steps.</p></li>

       <li><p>Set <var>context</var>'s <span data-x="concept-canvas-context-lost">context
       lost</span> to false.</p></li>

       <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
       data-x="event-contextrestored">contextrestored</code> at <var>canvas</var>.</p></li>
      </ol>
     </li>

     <li><p>For each <var>doc</var> of <var>docs</var>, <span>run the animation frame
     callbacks</span> for <var>doc</var>, passing in the <span>relative high resolution time</span>
     given <var>frameTimestamp</var> and <var>doc</var>'s <span>relevant global object</span> as
     the timestamp.</p></li>

     <li><p>Let <var>unsafeStyleAndLayoutStartTime</var> be the <span>unsafe shared current
     time</span>.</p>

     <li>
      <p>For each <var>doc</var> of <var>docs</var>:</p>

      <ol>
       <li><p>Let <var>resizeObserverDepth</var> be 0.</p></li>

       <li>
        <p>While true:</p>

        <ol>
         <li><p>Recalculate styles and update layout for <var>doc</var>.</p></li>

         <li><p>Let <var>hadInitialVisibleContentVisibilityDetermination</var> be false.</p></li>

         <li>
          <p>For each element <var>element</var> with <span
          data-x="content-visibility-auto">'auto'</span> used value of
          <span>'content-visibility'</span>:</p>

          <ol>
           <li><p>Let <var>checkForInitialDetermination</var> be true if <var>element</var>'s
           <span>proximity to the viewport</span> is not determined and it is not <span>relevant to
           the user</span>. Otherwise, let <var>checkForInitialDetermination</var> be
           false.</p></li>

           <li><p>Determine <span>proximity to the viewport</span> for <var>element</var>.</p></li>

           <li><p>If <var>checkForInitialDetermination</var> is true and <var>element</var> is now
           <span>relevant to the user</span>, then set
           <var>hadInitialVisibleContentVisibilityDetermination</var> to true.</p></li>
          </ol>
         </li>

         <li>
          <p>If <var>hadInitialVisibleContentVisibilityDetermination</var> is true, then
          <span>continue</span>.</p>

          <p class="note">The intent of this step is for the initial viewport proximity
          determination, which takes effect immediately, to be reflected in the style and layout
          calculation which is carried out in a previous step of this loop. Proximity
          determinations other than the initial one take effect at the next <span>rendering
          opportunity</span>. <ref>CSSCONTAIN</ref></p>
         </li>

         <li><p><span>Gather active resize observations at depth</span>
         <var>resizeObserverDepth</var> for <var>doc</var>.</p></li>

         <li><p>If <var>doc</var> <span>has active resize observations</span>:</p>

          <ol>
           <li><p>Set <var>resizeObserverDepth</var> to the result of <span data-x="broadcast
           active resize observations">broadcasting active resize observations</span> given
           <var>doc</var>.</p></li>

           <li><span>Continue</span>.</li>
          </ol>
         </li>

         <li>Otherwise, <span>break</span>.</li>
        </ol>
       </li>

       <li><p>If <var>doc</var> <span>has skipped resize observations</span>, then <span>deliver
       resize loop error</span> given <var>doc</var>.</p></li>
      </ol>
     </li>

     <li id="focus-fixup-rule">
      <p>For each <var>doc</var> of <var>docs</var>, if the <span data-x="focused area of the
      document">focused area</span> of <var>doc</var> is not a <span>focusable area</span>, then
      run the <span>focusing steps</span> for <var>doc</var>'s <span>viewport</span>, and set
      <var>doc</var>'s <span>relevant global object</span>'s <span
      data-x="window-navigation-api">navigation API</span>'s <span>focus changed during ongoing
      navigation</span> to false.</p>

      <p class="example">For example, this might happen because an element has the <code
      data-x="attr-hidden">hidden</code> attribute added, causing it to stop <span>being
      rendered</span>. It might also happen to an <code>input</code> element when the element gets
      <span data-x="concept-fe-disabled">disabled</span>.</p>

      <p class="note">This will <a href="#note-sometimes-no-blur-event">usually</a> fire <code
      data-x="event-blur">blur</code> events, and possibly <code
      data-x="event-change">change</code> events.</p>

      <p class="note">In addition to this asynchronous fixup, if the <span>focused area of the
      document</span> is removed, there is a <a href="#node-remove-focus-fixup">synchronous
      fixup</a>. That one will <em>not</em> fire <code data-x="event-blur">blur</code> or <code
      data-x="event-change">change</code> events.</p>
     </li>

     <li><p>For each <var>doc</var> of <var>docs</var>, <span>perform pending transition
     operations</span> for <var>doc</var>. <ref>CSSVIEWTRANSITIONS</ref></p></li>

     <li><p>For each <var>doc</var> of <var>docs</var>, <span>run the update intersection
     observations steps</span> for <var>doc</var>, passing in the <span>relative high resolution
     time</span> given <var>now</var> and <var>doc</var>'s <span>relevant global object</span> as
     the timestamp. <ref>INTERSECTIONOBSERVER</ref></p></li>

     <li><p>For each <var>doc</var> of <var>docs</var>, <span>record rendering time</span> for
     <var>doc</var> given <var>unsafeStyleAndLayoutStartTime</var>.</p></li>

     <li><p>For each <var>doc</var> of <var>docs</var>, <span>mark paint timing</span> for
     <var>doc</var>.</p></li>

     <li><p>For each <var>doc</var> of <var>docs</var>, update the rendering or user interface of
     <var>doc</var> and its <span>node navigable</span> to reflect the current state.</p></li>

     <li><p>For each <var>doc</var> of <var>docs</var>, <span>process top layer removals</span>
     given <var>doc</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>A <span>navigable</span> has a <dfn export>rendering opportunity</dfn> if the user agent is
  currently able to present the contents of the <span>navigable</span> to the user, accounting for
  hardware refresh rate constraints and user agent throttling for performance reasons, but
  considering content presentable even if it's outside the viewport.</p>

  <p>A <span>navigable</span>'s <span data-x="rendering opportunity">rendering opportunities</span>
  are determined based on hardware constraints such as display refresh rates and other factors such
  as page performance or whether its <span data-x="nav-document">active document</span>'s
  <span>visibility state</span> is "<code data-x="">visible</code>". Rendering opportunities
  typically occur at regular intervals.</p>

  <p class="note">This specification does not mandate any particular model for selecting rendering
  opportunities. But for example, if the browser is attempting to achieve a 60Hz refresh rate, then
  rendering opportunities occur at a maximum of every 60th of a second (about 16.7ms). If the
  browser finds that a <span>navigable</span> is not able to sustain this rate, it might drop to a
  more sustainable 30 rendering opportunities per second for that <span>navigable</span>, rather
  than occasionally dropping frames. Similarly, if a <span>navigable</span> is not visible, the
  user agent might decide to drop that page to a much slower 4 rendering opportunities per second,
  or even less.</p>


  <hr>

  <p>When a user agent is to <dfn export>perform a microtask checkpoint</dfn>:</p>

  <ol>
   <li><p>If the <span>event loop</span>'s <span>performing a microtask checkpoint</span> is true,
   then return.</p></li>

   <li><p>Set the <span>event loop</span>'s <span>performing a microtask checkpoint</span> to
   true.</p></li>

   <li>
    <p>While the <span>event loop</span>'s <span>microtask queue</span> is not <span data-x="list is
    empty">empty</span>:</p>

    <ol>
     <li><p>Let <var>oldestMicrotask</var> be the result of <span data-x="dequeue">dequeuing</span>
     from the <span>event loop</span>'s <span>microtask queue</span>.</p></li>

     <li><p>Set the <span>event loop</span>'s <span>currently running task</span> to
     <var>oldestMicrotask</var>.</p></li>

     <li>
      <p>Run <var>oldestMicrotask</var>.</p>

      <p class="note">This might involve invoking scripted callbacks, which eventually calls the
      <span>clean up after running script</span> steps, which call this <span>perform a microtask
      checkpoint</span> algorithm again, which is why we use the <span>performing a microtask
      checkpoint</span> flag to avoid reentrancy.</p>
     </li>

     <li><p>Set the <span>event loop</span>'s <span>currently running task</span> back to
     null.</p></li>
    </ol>
   </li>

   <li><p>For each <span>environment settings object</span> <var>settingsObject</var> whose
   <span>responsible event loop</span> is this <span>event loop</span>, <span>notify about rejected
   promises</span> given <var>settingsObject</var>'s <span
   data-x="concept-settings-object-global">global object</span>.</p></li>

   <li><p><span>Cleanup Indexed Database transactions</span>.</p></li>

   <li>
    <p>Perform <span>ClearKeptObjects</span>().</p>

    <p class="note">When <code>WeakRef.prototype.deref()</code> returns an object, that object is
    kept alive until the next invocation of <span>ClearKeptObjects</span>(), after which it is again
    subject to garbage collection.</p>
   </li>

   <li><p>Set the <span>event loop</span>'s <span>performing a microtask checkpoint</span> to
   false.</p></li>

   <li><p><span>Record timing info for microtask checkpoint</span>.</p></li>
  </ol>

  <hr>

  <p>When an algorithm running <span>in parallel</span> is to <dfn>await a stable state</dfn>, the
  user agent must <span>queue a microtask</span> that runs the following steps, and must then stop
  executing (execution of the algorithm resumes when the microtask is run, as described in the
  following steps):</p>

  <ol>
   <li><p>Run the algorithm's <dfn>synchronous section</dfn>.</p></li>

   <li><p>Resumes execution of the algorithm <span>in parallel</span>, if appropriate, as described
   in the algorithm's steps.</p></li>
  </ol>

  <p class="note">Steps in <span data-x="synchronous section">synchronous sections</span> are marked
  with &#x231B;.</p>

  <hr>

  <p>Algorithm steps that say to <dfn>spin the event loop</dfn> until a condition <var>goal</var> is
  met are equivalent to substituting in the following algorithm steps:</p>

  <ol>
   <li>
    <p>Let <var>task</var> be the <span>event loop</span>'s <span>currently running
    task</span>.</p>

    <p class="note"><var>task</var> could be a <span>microtask</span>.</p>
   </li>

   <li><p>Let <var>task source</var> be <var>task</var>'s <span
   data-x="concept-task-source">source</span>.</p></li>

   <li><p>Let <var>old stack</var> be a copy of the <span>JavaScript execution context
   stack</span>.</p></li>

   <li><p>Empty the <span>JavaScript execution context stack</span>.</p></li>

   <li>
    <p><span>Perform a microtask checkpoint</span>.</p>

    <p class="note">If <var>task</var> is a <span>microtask</span> this step will be a no-op due to
    <span>performing a microtask checkpoint</span> being true.</p>
   </li>

   <li>
    <p><span>In parallel</span>:</p>

    <ol>
     <li><p>Wait until the condition <var>goal</var> is met.</p></li>

     <li>
      <p><span>Queue a task</span> on <var>task source</var> to:</p>

      <ol>
       <li><p>Replace the <span>JavaScript execution context stack</span> with <var>old
       stack</var>.</p></li>

       <li>
        <p>Perform any steps that appear after this <span>spin the event loop</span> instance in the
        original algorithm.</p>

        <p class="note">This resumes <var>task</var>.</p>
       </li>
      </ol>
     </li>
    </ol>
   </li>

   <li>
    <p>Stop <var>task</var>, allowing whatever algorithm that invoked it to resume.</p>

    <p class="note">This causes the <span>event loop</span>'s main set of steps or the <span>perform
    a microtask checkpoint</span> algorithm to continue.</p>
   </li>
  </ol>

  <p class="note">Unlike other algorithms in this and other specifications, which behave similar to
  programming-language function calls, <span>spin the event loop</span> is more like a macro, which
  saves typing and indentation at the usage site by expanding into a series of steps and
  operations.</p>

  <div class="example">
   <p>An algorithm whose steps are:</p>

   <ol>
    <li><p>Do something.</p></li>

    <li><p><span>Spin the event loop</span> until awesomeness happens.</p></li>

    <li><p>Do something else.</p></li>
   </ol>

   <p>is a shorthand which, after "macro expansion", becomes</p>

   <ol>
    <li><p>Do something.</p></li>

    <li><p>Let <var>old stack</var> be a copy of the <span>JavaScript execution context
    stack</span>.</p></li>

    <li><p>Empty the <span>JavaScript execution context stack</span>.</p></li>

    <li><p><span>Perform a microtask checkpoint</span>.</p></li>

    <li>
     <p><span>In parallel</span>:</p>

     <ol>
      <li><p>Wait until awesomeness happens.</p></li>

      <li>
       <p><span>Queue a task</span> on the task source in which "do something" was done to:</p>

       <ol>
        <li><p>Replace the <span>JavaScript execution context stack</span> with <var>old
        stack</var>.</p></li>

        <li><p>Do something else.</p></li>
       </ol>
      </li>
     </ol>
    </li>
   </ol>
  </div>

  <div class="example">
   <p>Here is a more full example of the substitution, where the event loop is spun from inside a
   task that is queued from work in parallel. The version using <span>spin the event loop</span>:</p>

   <ol>
    <li>
     <p><span>In parallel</span>:</p>

     <ol>
      <li><p>Do parallel thing 1.</p></li>

      <li>
       <p><span>Queue a task</span> on the <span>DOM manipulation task source</span> to:</p>

       <ol>
        <li><p>Do task thing 1.</p></li>

        <li><p><span>Spin the event loop</span> until awesomeness happens.</p></li>

        <li><p>Do task thing 2.</p></li>
       </ol>
      </li>

      <li><p>Do parallel thing 2.</p></li>
     </ol>
    </li>
   </ol>

   <p>The fully expanded version:</p>

   <ol>
    <li>
     <p><span>In parallel</span>:</p>

     <ol>
      <li><p>Do parallel thing 1.</p></li>

      <li><p>Let <var>old stack</var> be null.</p></li>

      <li>
       <p><span>Queue a task</span> on the <span>DOM manipulation task source</span> to:</p>

       <ol>
        <li><p>Do task thing 1.</p></li>

        <li><p>Set <var>old stack</var> to a copy of the <span>JavaScript execution context
        stack</span>.</p></li>

        <li><p>Empty the <span>JavaScript execution context stack</span>.</p></li>

        <li><p><span>Perform a microtask checkpoint</span>.</p></li>
       </ol>
      </li>

      <li><p>Wait until awesomeness happens.</p></li>

      <li>
       <p><span>Queue a task</span> on the <span>DOM manipulation task source</span> to:</p>

       <ol>
        <li><p>Replace the <span>JavaScript execution context stack</span> with <var>old
        stack</var>.</p></li>

        <li><p>Do task thing 2.</p></li>
       </ol>
      </li>

      <li><p>Do parallel thing 2.</p></li>
     </ol>
    </li>
   </ol>
  </div>

  <hr>

  <p>Some of the algorithms in this specification, for historical reasons, require the user agent to
  <dfn export>pause</dfn> while running a <span data-x="concept-task">task</span> until a condition
  <var>goal</var> is met. This means running the following steps:</p>

  <ol>
   <li><p>Let <var>global</var> be the <span>current global object</span>.</p></li>

   <li><p>Let <var>timeBeforePause</var> be the <span>current high resolution time</span> given
   <var>global</var>.</p></li>

   <li><p>If necessary, update the rendering or user interface of any <code>Document</code> or
   <span>navigable</span> to reflect the current state.</p></li>

   <li><p>Wait until the condition <var>goal</var> is met. While a user agent has a paused
   <span data-x="concept-task">task</span>, the corresponding <span>event loop</span> must not run
   further <span data-x="concept-task">tasks</span>, and any script in the currently running <span
   data-x="concept-task">task</span> must block. User agents should remain responsive to user input
   while paused, however, albeit in a reduced capacity since the <span>event loop</span> will not be
   doing anything.</p></li>

   <li><p><span>Record pause duration</span> given the <span>duration from</span>
   <var>timeBeforePause</var> to the <span>current high resolution time</span> given
   <var>global</var>.</p></li>
  </ol>

  <div class="warning">
   <p><span data-x="pause">Pausing</span> is highly detrimental to the user experience, especially
   in scenarios where a single <span>event loop</span> is shared among multiple documents. User
   agents are encouraged to experiment with alternatives to <span data-x="pause">pausing</span>,
   such as <span data-x="spin the event loop">spinning the event loop</span> or even simply
   proceeding without any kind of suspended execution at all, insofar as it is possible to do so
   while preserving compatibility with existing content. This specification will happily change if
   a less-drastic alternative is discovered to be web-compatible.</p>

   <p>In the interim, implementers should be aware that the variety of alternatives that user agents
   might experiment with can change subtle aspects of <span>event loop</span> behavior, including
   <span data-x="concept-task">task</span> and <span>microtask</span> timing. Implementations should
   continue experimenting even if doing so causes them to violate the exact semantics implied by the
   <span>pause</span> operation.</p>
  </div>

  <h5>Generic task sources</h5>

  <p>The following <span data-x="task source">task sources</span> are used by a number of mostly
  unrelated features in this and other specifications.</p>

  <dl>
   <dt>The <dfn export>DOM manipulation task source</dfn></dt>

   <dd>
    <p>This <span>task source</span> is used for features that react to DOM manipulations, such as
    things that happen in a non-blocking fashion when an element is <span data-x="node is inserted
    into a document">inserted into the document</span>.</p>
   </dd>

   <dt>The <dfn export>user interaction task source</dfn></dt>

   <dd>
    <p>This <span>task source</span> is used for features that react to user interaction, for
    example keyboard or mouse input.</p>

    <p>Events sent in response to user input (e.g. <code
    data-x="event-click">click</code> events) must be fired using <span
    data-x="concept-task">tasks</span> <span data-x="queue a task">queued</span> with the <span>user
    interaction task source</span>. <ref>UIEVENTS</ref></p> <!-- user
    interaction events integration point -->
   </dd>

   <dt>The <dfn export>networking task source</dfn></dt>

   <dd>
    <p>This <span>task source</span> is used for features that trigger in response to network
    activity.</p>
   </dd>

   <dt id="history-traversal-task-source">The <dfn>navigation and traversal task source</dfn></dt>

   <dd>
    <p>This <span>task source</span> is used to queue tasks involved in <span
    data-x="navigate">navigation</span> and <span data-x="apply the traverse history step">history
    traversal</span>.</p>
   </dd>

   <dt>The <dfn>rendering task source</dfn></dt>

   <dd>
    <p>This <span>task source</span> is used solely to <span>update the rendering</span>.
   </dd>
  </dl>

  <h5 id="event-loop-for-spec-authors">Dealing with the event loop from other specifications</h5>

  <p>Writing specifications that correctly interact with the <span>event loop</span> can be tricky.
  This is compounded by how this specification uses concurrency-model-independent terminology, so we
  say things like "<span>event loop</span>" and "<span>in parallel</span>" instead of using more
  familiar model-specific terms like "main thread" or "on a background thread".</p>

  <p>By default, specification text generally runs on the <span>event loop</span>. This falls out
  from the formal <a href="#event-loop-processing-model">event loop processing model</a>, in that
  you can eventually trace most algorithms back to a <span data-x="concept-task">task</span> <span
  data-x="queue a task">queued</span> there.</p>

  <p class="example">The algorithm steps for any JavaScript method will be invoked by author code
  calling that method. And author code can only be run via queued tasks, usually originating
  somewhere in the <a href="#script-processing-model"><code>script</code> processing model</a>.</p>

  <p>From this starting point, the overriding guideline is that any work a specification needs to
  perform that would otherwise block the <span>event loop</span> must instead be performed <span>in
  parallel</span> with it. This includes (but is not limited to):</p>

  <ul>
   <li><p>performing heavy computation;</p></li>

   <li><p>displaying a user-facing prompt;</p></li>

   <li><p>performing operations which could require involving outside systems (i.e. "going out of
   process").</p></li>
  </ul>

  <p>The next complication is that, in algorithm sections that are <span>in parallel</span>, you
  must not create or manipulate objects associated to a specific <span>realm</span>, <span
  data-x="global object">global</span>, or <span>environment settings object</span>. (Stated in more
  familiar terms, you must not directly access main-thread artifacts from a background thread.)
  Doing so would create data races observable to JavaScript code, since after all, your algorithm
  steps are running <em><span>in parallel</span></em> to the JavaScript code.</p>

  <p>By extension, you cannot access Web IDL's <span>this</span> value from steps running <span>in
  parallel</span>, even if those steps were activated by an algorithm that <em>does</em> have access
  to the <span>this</span> value.</p>

  <p>You can, however, manipulate specification-level data structures and values from
  <cite>Infra</cite>, as those are realm-agnostic. They are never directly exposed to JavaScript without
  a specific conversion taking place (often <span data-x="concept-idl-convert">via Web IDL</span>).
  <ref>INFRA</ref> <ref>WEBIDL</ref></p>

  <p>To affect the world of observable JavaScript objects, then, you must <span>queue a global
  task</span> to perform any such manipulations. This ensures your steps are properly interleaved
  with respect to other things happening on the <span>event loop</span>. Furthermore, you must
  choose a <span>task source</span> when <span data-x="queue a global task">queuing a global
  task</span>; this governs the relative order of your steps versus others. If you are unsure which
  <span>task source</span> to use, pick one of the <a href="#generic-task-sources">generic task
  sources</a> that sounds most applicable. Finally, you must indicate which <span>global
  object</span> your queued task is associated with; this ensures that if that global object is
  inactive, the task does not run.</p>

  <p class="note">The base primitive, on which <span>queue a global task</span> builds, is the
  <span>queue a task</span> algorithm. In general, <span>queue a global task</span> is better
  because it automatically picks the right <span>event loop</span> and, where appropriate, <span
  data-x="concept-task-document">document</span>. Older specifications often use <span>queue a
  task</span> combined with the <span>implied event loop</span> and <span>implied document</span>
  concepts, but this is discouraged.</p>

  <p>Putting this all together, we can provide a template for a typical algorithm that needs to do
  work asynchronously:</p>

  <ol>
   <li><p>Do any synchronous setup work, while still on the <span>event loop</span>. This may
   include converting <span>realm</span>-specific JavaScript values into realm-agnostic
   specification-level values.</p></li>

   <li><p>Perform a set of potentially-expensive steps <span>in parallel</span>, operating entirely
   on realm-agnostic values, and producing a realm-agnostic result.</p></li>

   <li><p><span>Queue a global task</span>, on a specified <span>task source</span> and given an
   appropriate <span>global object</span>, to convert the realm-agnostic result back into observable
   effects on the observable world of JavaScript objects on the <span>event loop</span>.</p></li>
  </ol>

  <div class="example" id="example-event-loop-using-algorithm">

   <p>The following is an algorithm that "encrypts" a passed-in <span>list</span> of <span
   data-x="scalar value string">scalar value strings</span> <var>input</var>, after parsing them as
   URLs:</p>

   <ol>
    <li><p>Let <var>urls</var> be an empty <span>list</span>.</p></li>

    <li>
     <p><span data-x="list iterate">For each</span> <var>string</var> of <var>input</var>:</p>

     <ol>
      <li><p>Let <var>parsed</var> be the result of <span>encoding-parsing a URL</span> given
      <var>string</var>, relative to the <span>current settings object</span>.</p></li>

      <li><p>If <var>parsed</var> is failure, then return <span>a promise rejected with</span> a
      <span>"<code>SyntaxError</code>"</span> <code>DOMException</code>.</p></li>

      <li><p>Let <var>serialized</var> be the result of applying the <span
      data-x="concept-url-serializer">URL serializer</span> to <var>parsed</var>.</p></li>

      <li><p><span data-x="list append">Append</span> <var>serialized</var> to <var>urls</var>.</p></li>
     </ol>
    </li>

    <li><p>Let <var>realm</var> be the <span>current realm</span>.</p></li>

    <li><p>Let <var>p</var> be a new promise.</p></li>

    <li>
     <p>Run the following steps <span>in parallel</span>:</p>

     <ol>
      <li><p>Let <var>encryptedURLs</var> be an empty <span>list</span>.</p></li>

      <li>
       <p><span data-x="list iterate">For each</span> <var>url</var> of <var>urls</var>:</p>

       <ol>
        <li><p>Wait 100 milliseconds, so that people think we're doing heavy-duty encryption.</p></li>

        <li><p>Let <var>encrypted</var> be a new <span>string</span> derived from <var>url</var>,
        whose <var>n</var>th <span>code unit</span> is equal to <var>url</var>'s <var>n</var>th
        <span>code unit</span> plus 13.</p></li>

        <li><p><span data-x="list append">Append</span> <var>encrypted</var> to
        <var>encryptedURLs</var>.</p>
       </ol>
      </li>

      <li>
       <p><span>Queue a global task</span> on the <span>networking task source</span>, given
       <var>realm</var>'s <span data-x="concept-realm-global">global object</span>, to perform the
       following steps:</p>

       <ol>
        <li><p>Let <var>array</var> be the result of <span
        data-x="concept-idl-convert">converting</span> <var>encryptedURLs</var> to a JavaScript
        array, in <var>realm</var>.</p></li>

        <li><p>Resolve <var>p</var> with <var>array</var>.</p></li>
       </ol>
      </li>
     </ol>
    </li>

    <li><p>Return <var>p</var>.</p></li>
   </ol>

   <p>Here are several things to notice about this algorithm:</p>

   <ul>
    <li><p>It does its URL parsing up front, on the <span>event loop</span>, before going to the
    <span>in parallel</span> steps. This is necessary, since parsing depends on the <span>current
    settings object</span>, which would no longer be current after going <span>in
    parallel</span>.</p></li>

    <li><p>Alternately, it could have saved a reference to the <span>current settings
    object</span>'s <span>API base URL</span> and used it during the <span>in parallel</span> steps;
    that would have been equivalent. However, we recommend instead doing as much work as possible up
    front, as this example does. Attempting to save the correct values can be error prone; for
    example, if we'd saved just the <span>current settings object</span>, instead of its <span>API
    base URL</span>, there would have been a potential race.</p></li>

    <li><p>It implicitly passes a <span>list</span> of <span data-x="string">strings</span> from the
    initial steps to the <span>in parallel</span> steps. This is OK, as both <span
    data-x="list">lists</span> and <span data-x="string">strings</span> are
    <span>realm</span>-agnostic.</p></li>

    <li><p>It performs "expensive computation" (waiting for 100 milliseconds per input URL) during
    the <span>in parallel</span> steps, thus not blocking the main <span>event loop</span>.</p></li>

    <li><p>Promises, as observable JavaScript objects, are never created and manipulated during the
    <span>in parallel</span> steps. <var>p</var> is created before entering those steps, and then is
    manipulated during a <span data-x="concept-task">task</span> that is <span data-x="queue a
    global task">queued</span> specifically for that purpose.</p></li>

    <li><p>The creation of a JavaScript array object also happens during the queued task, and is
    careful to specify which realm it creates the array in since that is no longer obvious from
    context.</p></li>
   </ul>

   <p>(On these last two points, see also <a
   href="https://github.com/whatwg/webidl/issues/135">whatwg/webidl issue #135</a> and <a
   href="https://github.com/whatwg/webidl/issues/371">whatwg/webidl issue #371</a>, where we are
   still mulling over the subtleties of the above promise-resolution pattern.)</p>

   <p>Another thing to note is that, in the event this algorithm was called from a Web IDL-specified
   operation taking a <code data-x="">sequence</code>&lt;<code
   data-x="idl-USVString">USVString</code>>, there was an automatic conversion from
   <span>realm</span>-specific JavaScript objects provided by the author as input, into the
   realm-agnostic <code data-x="">sequence</code>&lt;<code data-x="idl-USVString">USVString</code>>
   Web IDL type, which we then treat as a <span>list</span> of <span data-x="scalar value
   string">scalar value strings</span>. So depending on how your specification is structured, there
   may be other implicit steps happening on the main <span>event loop</span> that play a part in
   this whole process of getting you ready to go <span>in parallel</span>.</p>

  </div>


  </div>

  <h4>Events</h4>

  <h5 id="event-handler-attributes">Event handlers</h5>

  <p>Many objects can have <dfn export data-lt="event handler">event handlers</dfn> specified. These
  act as non-capture <span data-x="event listener">event listeners</span> for the object on which
  they are specified. <ref>DOM</ref></p>

  <div w-nodev>

  <p>An <span data-x="event handlers">event handler</span> is a <span>struct</span> with two <span
  data-x="struct item">items</span>:</p>

  <ul>
   <li><p>a <dfn export for="event handler" data-x="event handler value">value</dfn>, which is
   either null<span w-nodev>,</span><span w-dev> or</span> a callback object<span w-nodev>, or an
   <span>internal raw uncompiled handler</span></span>. The <code>EventHandler</code> callback
   function type describes how this is exposed to scripts. <span w-nodev>Initially, an <span
   data-x="event handlers">event handler</span>'s <span data-x="event handler value">value</span>
   must be set to null.</span></p></li>

   <li><p>a <dfn export for="event handler" data-x="event handler listener">listener</dfn>, which
   is either null or an <span>event listener</span><span w-nodev> responsible for running <span>the
   event handler processing algorithm</span></span>. <span w-nodev>Initially, an <span
   data-x="event handlers">event handler</span>'s <span data-x="event handler
   listener">listener</span> must be set to null.</span></p></li>
  </ul>

  </div>

  <p>Event handlers are exposed in two ways.</p>

  <p>The first way, common to all event handlers, is as an <span data-x="event handler IDL
  attributes">event handler IDL attribute</span>.</p>

  <p>The second way is as an <span data-x="event handler content attributes">event handler content
  attribute</span>. Event handlers on <span>HTML elements</span> and some of the event handlers on
  <code>Window</code> objects are exposed in this way.</p>

  <p>For both of these two ways, the <span data-x="event handlers">event handler</span> is exposed
  through a <dfn data-x="event handler name">name</dfn>, which is a string that always starts with
  "<code data-x="">on</code>" and is followed by the name of the event for which the handler is
  intended.</p>

  <hr>

  <p>Most of the time, the object that exposes an <span data-x="event handlers">event handler</span>
  is the same as the object on which the corresponding <span>event listener</span> is added.
  However, the <code>body</code> and <code>frameset</code> elements expose several <span>event
  handlers</span> that act upon the element's <code>Window</code> object, if one exists. In either
  case, we call the object an <span data-x="event handlers">event handler</span> acts upon the <dfn
  data-x="event handler target">target</dfn> of that <span data-x="event handlers">event
  handler</span>.</p>

  <div w-nodev>

  <p>To <dfn data-x="determining the target of an event handler">determine the target of an event
  handler</dfn>, given an <code>EventTarget</code> object <var>eventTarget</var> on which the <span
  data-x="event handlers">event handler</span> is exposed, and an <span>event handler name</span>
  <var>name</var>, the following steps are taken:</p>

  <ol>
   <li><p>If <var>eventTarget</var> is not a <code>body</code> element or a <code>frameset</code>
   element, then return <var>eventTarget</var>.</p></li>

   <li><p>If <var>name</var> is not the name of an attribute member of the
   <code>WindowEventHandlers</code> interface mixin and the <span><code>Window</code>-reflecting
   body element event handler set</span> does not <span data-x="list contains">contain</span>
   <var>name</var>, then return <var>eventTarget</var>.</p></li>

   <li>
    <p>If <var>eventTarget</var>'s <span>node document</span> is not an <span>active
    document</span>, then return null.</p>

    <p class="note">This could happen if this object is a <code>body</code> element without
    a corresponding <code>Window</code> object, for example.</p>

    <p class="note">This check does not necessarily prevent <code>body</code> and
    <code>frameset</code> elements that are not <span>the body element</span> of their <span>node
    document</span> from reaching the next step. In particular, a <code>body</code> element created
    in an <span>active document</span> (perhaps with <code
    data-x="dom-Document-createElement">document.createElement()</code>) but not
    <span>connected</span> will also have its corresponding <code>Window</code> object as the <span
    data-x="event handler target">target</span> of several <span>event handlers</span> exposed
    through it.</p>
   </li>

   <li><p>Return <var>eventTarget</var>'s <span>node document</span>'s <span>relevant global
   object</span>.</p></li>
  </ol>

  <hr>

  <p>Each <code>EventTarget</code> object that has one or more <span>event handlers</span> specified
  has an associated <dfn export for="EventTarget">event handler map</dfn>, which is a <span
  data-x="ordered map">map</span> of strings representing <span data-x="event handler
  name">names</span> of <span>event handlers</span> to <span>event handlers</span>.</p>

  <p>When an <code>EventTarget</code> object that has one or more <span>event handlers</span>
  specified is created, its <span>event handler map</span> must be initialized such that it contains
  an <span data-x="map entry">entry</span> for each <span data-x="event handlers">event
  handler</span> that has that object as <span data-x="event handler target">target</span>, with
  <span data-x="struct item">items</span> in those <span>event handlers</span> set to their initial
  values.</p>

  <p class="note">The order of the <span data-x="map entry">entries</span> of <span>event handler
  map</span> could be arbitrary. It is not observable through any algorithms that operate on the
  map.</p>

  <p class="note"><span data-x="map entry">Entries</span> are not created in the <span>event handler
  map</span> of an object for <span>event handlers</span> that are merely exposed on that object,
  but have some other object as their <span data-x="event handler target">targets</span>.</p>

  </div>

  <hr>

  <p>An <dfn data-x="event handler IDL attributes" export>event handler IDL attribute</dfn> is an
  IDL attribute for a specific <span data-x="event handlers">event handler</span>. The name of the
  IDL attribute is the same as the <span data-x="event handler name">name</span> of the <span
  data-x="event handlers">event handler</span>.</p>

  <div w-nodev>

  <p>The getter of an <span data-x="event handler IDL attributes">event handler IDL attribute</span>
  with name <var>name</var>, when called, must run these steps:</p>

  <ol>
   <li><p>Let <var>eventTarget</var> be the result of <span>determining the target of an event
   handler</span> given this object and <var>name</var>.</p></li>

   <li><p>If <var>eventTarget</var> is null, then return null.</p></li>

   <li><p>Return the result of <span>getting the current value of the event handler</span> given
   <var>eventTarget</var> and <var>name</var>.</p>
  </ol>

  <p>The setter of an <span data-x="event handler IDL attributes">event handler IDL attribute</span>
  with name <var>name</var>, when called, must run these steps:</p>

  <ol>
   <li><p>Let <var>eventTarget</var> be the result of <span>determining the target of an event
   handler</span> given this object and <var>name</var>.</p></li>

   <li><p>If <var>eventTarget</var> is null, then return.</p></li>

   <li><p>If the given value is null, then <span>deactivate an event handler</span> given
   <var>eventTarget</var> and <var>name</var>.</p></li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p>Let <var>handlerMap</var> be <var>eventTarget</var>'s <span>event handler
     map</span>.</p></li>

     <li><p>Let <var>eventHandler</var> be <var>handlerMap</var>[<var>name</var>].</p></li>

     <li><p>Set <var>eventHandler</var>'s <span data-x="event handler value">value</span> to the
     given value.</p></li>

     <li><p><span>Activate an event handler</span> given <var>eventTarget</var> and
     <var>name</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p class="note">Certain <span>event handler IDL attributes</span> have additional requirements, in
  particular the <code data-x="handler-MessageEventTarget-onmessage">onmessage</code> attribute of
  <code>MessagePort</code> objects.</p>

  </div>

  <hr>

  <p>An <dfn data-x="event handler content attributes" export>event handler content attribute</dfn>
  is a content attribute for a specific <span data-x="event handlers">event handler</span>. The name
  of the content attribute is the same as the <span data-x="event handler name">name</span> of the
  <span data-x="event handlers">event handler</span>.</p>

  <p><span>Event handler content attributes</span>, when specified, must contain valid JavaScript
  code which, when parsed, would match the <i data-x="js-prod-FunctionBody">FunctionBody</i>
  production after <span>automatic semicolon insertion</span>.</p>

  <div w-nodev>

  <p>The following <span data-x="concept-element-attributes-change-ext">attribute change
  steps</span> are used to synchronize between <span>event handler content attributes</span> and
  <span>event handlers</span>: <ref>DOM</ref></p>

  <ol>
   <li><p>If <var>namespace</var> is not null, or <var>localName</var> is not the name of an <span
   data-x="event handler content attributes">event handler content attribute</span> on
   <var>element</var>, then return.</p></li>

   <li><p>Let <var>eventTarget</var> be the result of <span>determining the target of an event
   handler</span> given <var>element</var> and <var>localName</var>.</p></li>

   <li><p>If <var>eventTarget</var> is null, then return.</p></li>

   <li><p>If <var>value</var> is null, then <span>deactivate an event handler</span> given
   <var>eventTarget</var> and <var>localName</var>.</p></li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p>If the <span>Should element's inline behavior be blocked by Content Security
     Policy?</span> algorithm returns "<code data-x="">Blocked</code>" when executed upon
     <var>element</var>, "<code data-x="">script attribute</code>", and <var>value</var>, then
     return. <ref>CSP</ref></p></li>

     <li><p>Let <var>handlerMap</var> be <var>eventTarget</var>'s <span>event handler
     map</span>.</p></li>

     <li><p>Let <var>eventHandler</var> be <var>handlerMap</var>[<var>localName</var>].</p></li>

     <li><p>Let <var>location</var> be the script location that triggered the execution of these
     steps.</p></li>

     <li><p>Set <var>eventHandler</var>'s <span data-x="event handler value">value</span> to the
     <span>internal raw uncompiled handler</span> <var>value</var>/<var>location</var>.</p></li>

     <li><p><span>Activate an event handler</span> given <var>eventTarget</var> and
     <var>localName</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p class="note">Per the DOM Standard, these steps are run even if <var>oldValue</var> and
  <var>value</var> are identical (setting an attribute to its current value), but <em>not</em> if
  <var>oldValue</var> and <var>value</var> are both null (removing an attribute that doesn't
  currently exist). <ref>DOM</ref></p>

  <hr>

  <p>To <dfn>deactivate an event handler</dfn> given an <code>EventTarget</code> object
  <var>eventTarget</var> and a string <var>name</var> that is the <span data-x="event handler
  name">name</span> of an <span data-x="event handlers">event handler</span>, run these steps:</p>

  <ol>
   <li><p>Let <var>handlerMap</var> be <var>eventTarget</var>'s <span>event handler
   map</span>.</p></li>

   <li><p>Let <var>eventHandler</var> be <var>handlerMap</var>[<var>name</var>].</p></li>

   <li><p>Set <var>eventHandler</var>'s <span data-x="event handler value">value</span> to
   null.</p></li>

   <li><p>Let <var>listener</var> be <var>eventHandler</var>'s <span data-x="event handler
   listener">listener</span>.</p></li>

   <li><p>If <var>listener</var> is not null, then <span>remove an event listener</span> with
   <var>eventTarget</var> and <var>listener</var>.</p></li>

   <li><p>Set <var>eventHandler</var>'s <span data-x="event handler listener">listener</span> to
   null.</p></li>
  </ol>

  <p>To <dfn>erase all event listeners and handlers</dfn> given an <code>EventTarget</code> object
  <var>eventTarget</var>, run these steps:</p>

  <ol>
   <li><p>If <var>eventTarget</var> has an associated <span>event handler map</span>, then for each
   <var>name</var> → <var>eventHandler</var> of <var>eventTarget</var>'s associated <span>event
   handler map</span>, <span>deactivate an event handler</span> given <var>eventTarget</var> and
   <var>name</var>.</p></li>

   <li><p><span>Remove all event listeners</span> given <var>eventTarget</var>.</p></li>
  </ol>

  <p class="note">This algorithm is used to define <code
  data-x="dom-document-open">document.open()</code>.</p>

  <p>To <dfn>activate an event handler</dfn> given an <code>EventTarget</code> object
  <var>eventTarget</var> and a string <var>name</var> that is the <span data-x="event handler
  name">name</span> of an <span data-x="event handlers">event handler</span>, run these steps:</p>

  <ol>
   <li><p>Let <var>handlerMap</var> be <var>eventTarget</var>'s <span>event handler
   map</span>.</p></li>

   <li><p>Let <var>eventHandler</var> be <var>handlerMap</var>[<var>name</var>].</p></li>

   <li><p>If <var>eventHandler</var>'s <span data-x="event handler listener">listener</span> is not
   null, then return.</p></li>

   <li>
    <p>Let <var>callback</var> be the result of creating a Web IDL <code
    data-x="dom-EventListener">EventListener</code> instance representing a reference to a function
    of one argument that executes the steps of <span>the event handler processing algorithm</span>,
    given <var>eventTarget</var>, <var>name</var>, and its argument.</p>

    <p>The <code data-x="dom-EventListener">EventListener</code>'s <span>callback context</span> can
    be arbitrary; it does not impact the steps of <span>the event handler processing
    algorithm</span>. <ref>DOM</ref></p>

    <p class="note">The callback is emphatically <em>not</em> the <span data-x="event
    handlers">event handler</span> itself. Every event handler ends up registering the same
    callback, the algorithm defined below, which takes care of invoking the right code, and
    processing the code's return value.</p>
   </li>

   <li>
    <p>Let <var>listener</var> be a new <span>event listener</span> whose <span data-x="event
    listener type">type</span> is the <dfn export>event handler event type</dfn> corresponding to
    <var>eventHandler</var> and <span data-x="event listener callback">callback</span> is
    <var>callback</var>.</p>

    <p class="note">To be clear, an <span>event listener</span> is different from an <code
    data-x="dom-EventListener">EventListener</code>.</p>
   </li>

   <li><p><span>Add an event listener</span> with <var>eventTarget</var> and
   <var>listener</var>.</p></li>

   <li><p>Set <var>eventHandler</var>'s <span data-x="event handler listener">listener</span> to
   <var>listener</var>.</p></li>
  </ol>

  <div class="note">
   <p>The event listener registration happens only if the <span data-x="event handlers">event
   handler</span>'s <span data-x="event handler value">value</span> is being set to non-null, and
   the <span data-x="event handlers">event handler</span> is not already activated. Since listeners
   are called in the order they were registered, assuming no <span data-x="deactivate an event
   handler">deactivation</span> occurred, the order of event listeners for a particular event type
   will always be:</p>

   <ol>
    <li><p>the event listeners registered with <code
    data-x="dom-EventTarget-addEventListener">addEventListener()</code> before the first time the
    <span data-x="event handlers">event handler</span>'s <span data-x="event handler
    value">value</span> was set to non-null</p></li>

    <li><p>then the callback to which it is currently set, if any</p></li>

    <li><p>and finally the event listeners registered with <code
    data-x="dom-EventTarget-addEventListener">addEventListener()</code> <em>after</em> the first
    time the <span data-x="event handlers">event handler</span>'s <span data-x="event handler
    value">value</span> was set to non-null.</p></li>
   </ol>
  </div>

  </div>

  <div class="example">

   <p>This example demonstrates the order in which event listeners are invoked. If the button in
   this example is clicked by the user, the page will show four alerts, with the text "ONE", "TWO",
   "THREE", and "FOUR" respectively.</p>

   <pre><code class="html">&lt;button id="test">Start Demo&lt;/button>
&lt;script>
 var button = document.getElementById('test');
 button.addEventListener('click', function () { alert('ONE') }, false);
 button.setAttribute('onclick', "alert('NOT CALLED')"); // event handler listener is registered here
 button.addEventListener('click', function () { alert('THREE') }, false);
 button.onclick = function () { alert('TWO'); };
 button.addEventListener('click', function () { alert('FOUR') }, false);
&lt;/script></code></pre>

   <p>However, in the following example, the event handler is <span data-x="deactivate an event
   handler">deactivated</span> after its initial activation (and its event listener is removed),
   before being reactivated at a later time. The page will show five alerts with "ONE", "TWO",
   "THREE", "FOUR", and "FIVE" respectively, in order.</p>

   <pre><code class="html">&lt;button id="test">Start Demo&lt;/button>
&lt;script>
 var button = document.getElementById('test');
 button.addEventListener('click', function () { alert('ONE') }, false);
 button.setAttribute('onclick', "alert('NOT CALLED')"); // event handler is activated here
 button.addEventListener('click', function () { alert('TWO') }, false);
 button.onclick = null;                                 // but deactivated here
 button.addEventListener('click', function () { alert('THREE') }, false);
 button.onclick = function () { alert('FOUR'); };       // and re-activated here
 button.addEventListener('click', function () { alert('FIVE') }, false);
&lt;/script></code></pre>

  </div>

  <div w-nodev>

  <p class="note">The interfaces implemented by the event object do not influence whether an <span
  data-x="event handlers">event handler</span> is triggered or not.</p>

  <p><dfn>The event handler processing algorithm</dfn> for an <code>EventTarget</code> object
  <var>eventTarget</var>, a string <var>name</var> representing the <span data-x="event handler
  name">name</span> of an <span data-x="event handlers">event handler</span>, and an
  <code>Event</code> object <var>event</var> is as follows:</p>

  <ol>
   <li><p>Let <var>callback</var> be the result of <span>getting the current value of the event
   handler</span> given <var>eventTarget</var> and <var>name</var>.</p></li>

   <li><p>If <var>callback</var> is null, then return.</p></li>

   <li><p>Let <var>special error event handling</var> be true if <var>event</var> is an
   <code>ErrorEvent</code> object, <var>event</var>'s <code data-x="dom-Event-type">type</code> is
   "<code data-x="event-error">error</code>", and <var>event</var>'s <code
   data-x="dom-Event-currentTarget">currentTarget</code> implements the
   <code>WindowOrWorkerGlobalScope</code> mixin. Otherwise, let <var>special error event
   handling</var> be false.</p></li>

   <li>
    <p>Process the <code>Event</code> object <var>event</var> as follows:</p>

    <dl class="switch">
     <dt>If <var>special error event handling</var> is true</dt>
     <dd>
      <p>Let <var>return value</var> be the result of <span
      data-x="es-invoking-callback-functions">invoking</span> <var>callback</var> with «
      <var>event</var>'s <code data-x="dom-ErrorEvent-message">message</code>, <var>event</var>'s
      <code data-x="dom-ErrorEvent-filename">filename</code>, <var>event</var>'s <code
      data-x="dom-ErrorEvent-lineno">lineno</code>, <var>event</var>'s <code
      data-x="dom-ErrorEvent-colno">colno</code>, <var>event</var>'s <code
      data-x="dom-ErrorEvent-error">error</code> », "<code data-x="">rethrow</code>", and with <i
      data-x="dfn-callback-this-value">callback this value</i> set to <var>event</var>'s <code
      data-x="dom-Event-currentTarget">currentTarget</code>.</p>
     </dd>

     <dt>Otherwise</dt>
     <dd>
      <p>Let <var>return value</var> be the result of <span
      data-x="es-invoking-callback-functions">invoking</span> <var>callback</var> with «
      <var>event</var> », "<code data-x="">rethrow</code>", and with <i
      data-x="dfn-callback-this-value">callback this value</i> set to <var>event</var>'s <code
      data-x="dom-Event-currentTarget">currentTarget</code>.</p>
     </dd>
    </dl>

    <p class="note">If an exception gets thrown by the callback, it will be rethrown, ending these
    steps. The exception will propagate to the <span data-x="concept-event-dispatch">DOM event
    dispatch logic</span>, which will then <span data-x="report an exception">report</span> it.</p>
   </li>

   <li>
    <p>Process <var>return value</var> as follows:</p>

    <dl class="switch">

     <dt>If <var>event</var> is a <code>BeforeUnloadEvent</code> object and <var>event</var>'s <code
     data-x="dom-Event-type">type</code> is "<code
     data-x="event-beforeunload">beforeunload</code>"</dt>
     <!-- It is possible to get a BeforeUnloadEvent object whose type is not beforeunload via
     document.createEvent("beforeunloadevent") + initEvent. This may be removed; see
     https://github.com/whatwg/dom/issues/362. But it's still probably a good idea to check both,
     just in case e.g. we end up adding a constructor to BeforeUnloadEvent. -->
     <dd>
      <p class="note">In this case, the <span data-x="event handler IDL attributes">event handler
      IDL attribute</span>'s type will be <code>OnBeforeUnloadEventHandler</code>, so <var>return
      value</var> will have been coerced into either null or a <code
      data-x="idl-DOMString">DOMString</code>.</p>

      <p>If <var>return value</var> is not null, then:</p>

      <ol>
       <li><p>Set <var>event</var>'s <span>canceled flag</span>.</p></li>

       <li><p>If <var>event</var>'s <code
       data-x="dom-BeforeUnloadEvent-returnValue">returnValue</code> attribute's value is the empty
       string, then set <var>event</var>'s <code
       data-x="dom-BeforeUnloadEvent-returnValue">returnValue</code> attribute's value to
       <var>return value</var>.</p></li>
      </ol>
     </dd>

     <dt>If <var>special error event handling</var> is true</dt>
     <dd><p>If <var>return value</var> is true, then set <var>event</var>'s <span>canceled
     flag</span>.</p></dd>

     <dt>Otherwise</dt>
     <dd>
      <p>If <var>return value</var> is false, then set <var>event</var>'s <span>canceled
      flag</span>.</p>

      <p class="note">If we've gotten to this "Otherwise" clause because <var>event</var>'s <code
      data-x="dom-Event-type">type</code> is "<code
      data-x="event-beforeunload">beforeunload</code>" but <var>event</var> is <em>not</em> a
      <code>BeforeUnloadEvent</code> object, then <var>return value</var> will never be false, since
      in such cases <var>return value</var> will have been coerced into either null or a <code
      data-x="idl-DOMString">DOMString</code>.</p>
     </dd>
    </dl>
   </li>
  </ol>

  </div>

  <hr>

  <p>The <code>EventHandler</code> callback function type represents a callback used for event
  handlers.<span w-nodev> It is represented in Web IDL as follows:</span></p>

  <pre><code class="idl">[<span>LegacyTreatNonObjectAsNull</span>]
callback <dfn callback>EventHandlerNonNull</dfn> = any (<span>Event</span> event);
typedef <span>EventHandlerNonNull</span>? <dfn typedef>EventHandler</dfn>;</code></pre>

  <p class="note">In JavaScript, any <code data-x="idl-Function">Function</code> object implements
  this interface.</p>

  <div class="example">

   <p>For example, the following document fragment:</p>

   <pre><code class="html">&lt;body onload="alert(this)" onclick="alert(this)"></code></pre>

   <p>...leads to an alert saying "<code data-x="">[object&nbsp;Window]</code>" when the document is
   loaded, and an alert saying "<code data-x="">[object&nbsp;HTMLBodyElement]</code>" whenever the
   user clicks something in the page.</p>

  </div>

  <div class="note">
   <p>The return value of the function affects whether the event is canceled or not: <span
   w-nodev>as described above, </span>if the return value is false, the event is canceled.</p>

   <p>There are two exceptions in the platform, for historical reasons:</p>

   <ul>
    <li><p>The <code data-x="handler-onerror">onerror</code> handlers on global objects, where
    returning <em>true</em> cancels the event.</p></li>

    <li><p>The <code data-x="handler-window-onbeforeunload">onbeforeunload</code> handler, where
    returning any non-null and non-undefined value will cancel the event.</p></li>
   </ul>
  </div>

  <p>For historical reasons, the <code data-x="handler-onerror">onerror</code> handler has different
  arguments:</p>

  <pre><code class="idl">[<span>LegacyTreatNonObjectAsNull</span>]
callback <dfn callback>OnErrorEventHandlerNonNull</dfn> = any ((<span>Event</span> or DOMString) event, optional DOMString source, optional unsigned long lineno, optional unsigned long colno, optional any error);
typedef <span>OnErrorEventHandlerNonNull</span>? <dfn typedef>OnErrorEventHandler</dfn>;</code></pre>

  <div class="example">
   <pre><code class="js">window.onerror = (message, source, lineno, colno, error) => { &hellip; };</code></pre>
  </div>

  <p>Similarly, the <code data-x="handler-window-onbeforeunload">onbeforeunload</code> handler has a
  different return value:<span w-dev> it will be cast to a string.</span></p>

  <pre><code class="idl">[<span>LegacyTreatNonObjectAsNull</span>]
callback <dfn callback>OnBeforeUnloadEventHandlerNonNull</dfn> = DOMString? (<span>Event</span> event);
typedef <span>OnBeforeUnloadEventHandlerNonNull</span>? <dfn typedef>OnBeforeUnloadEventHandler</dfn>;</code></pre>

  <!-- onreadystatechange is also defined specially, using [LenientThis]; see IDL -->

  <div w-nodev>

  <hr>

  <p>An <dfn>internal raw uncompiled handler</dfn> is a tuple with the following information:</p>

  <ul>
   <li><p>An uncompiled script body</p></li>

   <li><p>A location where the script body originated, in case an error needs to be
   reported</p></li>
  </ul>

  <p>When the user agent is to <dfn data-x="getting the current value of the event handler">get the
  current value of the event handler</dfn> given an <code>EventTarget</code> object
  <var>eventTarget</var> and a string <var>name</var> that is the <span data-x="event handler
  name">name</span> of an <span data-x="event handlers">event handler</span>, it must run these
  steps:</p>

  <ol>
   <li><p>Let <var>handlerMap</var> be <var>eventTarget</var>'s <span>event handler
   map</span>.</p></li>

   <li><p>Let <var>eventHandler</var> be <var>handlerMap</var>[<var>name</var>].</p></li>

   <li>
    <p>If <var>eventHandler</var>'s <span data-x="event handler value">value</span> is an
    <span>internal raw uncompiled handler</span>, then:</p>

    <ol>
     <li><p>If <var>eventTarget</var> is an element, then let <var>element</var> be
     <var>eventTarget</var>, and <var>document</var> be <var>element</var>'s <span>node
     document</span>. Otherwise, <var>eventTarget</var> is a <code>Window</code> object, let
     <var>element</var> be null, and <var>document</var> be <var>eventTarget</var>'s <span
     data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

     <li><p>If <span data-x="concept-n-noscript">scripting is disabled</span> for
     <var>document</var>, then return null.</p></li>

     <li><p>Let <var>body</var> be the uncompiled script body in <var>eventHandler</var>'s <span
     data-x="event handler value">value</span>.</p></li>

     <li><p>Let <var>location</var> be the location where the script body originated, as
     given by <var>eventHandler</var>'s <span data-x="event handler value">value</span>.</p></li>

     <li><p>If <var>element</var> is not null and <var>element</var> has a
     <span>form owner</span>, let <var>form owner</var> be that <span>form owner</span>.
     Otherwise, let <var>form owner</var> be null.</p></li>

     <li><p>Let <var>settings object</var> be the <span>relevant settings object</span> of
     <var>document</var>.</p></li>

     <li>
      <p>If <var>body</var> is not parsable as <i data-x="js-prod-FunctionBody">FunctionBody</i>
      or if parsing detects an <span>early error</span>, then follow these substeps:</p>

      <ol>
       <li>
        <p>Set <var>eventHandler</var>'s <span data-x="event handler value">value</span> to
        null.</p>

        <p class="note">This does not <span data-x="deactivate an event handler">deactivate</span>
        the event handler, which additionally <span data-x="remove an event listener">removes</span>
        the event handler's <span data-x="event handler listener">listener</span> (if present).</p>
       </li>

       <li><p>Let <var>syntaxError</var> be a new <code>SyntaxError</code> exception associated
       with <var>settings object</var>'s <span data-x="environment settings object's
       realm">realm</span> which describes the error while parsing. It should be based on
       <var>location</var>, where the script body originated.</p></li>

       <li><p><span>Report an exception</span> with <var>syntaxError</var>
       for <var>settings object</var>'s <span data-x="concept-settings-object-global">global
       object</span>.</p></li>

       <li><p>Return null.</p></li>
      </ol>
     </li>

     <li>
      <p>Push <var>settings object</var>'s <span>realm execution context</span> onto the
      <span>JavaScript execution context stack</span>; it is now the <span>running JavaScript
      execution context</span>.</p>

      <p class="note">This is necessary so the subsequent invocation of <span
      data-x="js-OrdinaryFunctionCreate">OrdinaryFunctionCreate</span> takes place in the correct
      <span>realm</span>.</p>
     </li>

     <li>
      <p>Let <var>function</var> be the result of calling <span
      data-x="js-OrdinaryFunctionCreate">OrdinaryFunctionCreate</span>, with arguments:</p>

      <dl>
       <dt><var>functionPrototype</var></dt>

       <dd><span>%Function.prototype%</span></dd>

       <dt><var>sourceText</var></dt>

       <dd>
        <dl class="switch">
         <dt>If <var>name</var> is <code data-x="handler-onerror">onerror</code> and
         <var>eventTarget</var> is a <code>Window</code> object</dt>

         <dd>The string formed by concatenating "<code data-x="">function </code>", <var>name</var>,
         "<code data-x="">(event, source, lineno, colno, error) {</code>", U+000A LF,
         <var>body</var>, U+000A LF, and "<code data-x="">}</code>".</dd>

         <dt>Otherwise</dt>

         <dd>The string formed by concatenating "<code data-x="">function </code>", <var>name</var>,
         "<code data-x="">(event) {</code>", U+000A LF, <var>body</var>, U+000A LF, and "<code
        data-x="">}</code>".</dd>
        </dl>
       </dd>

       <dt><var>ParameterList</var></dt>

       <dd>
        <dl class="switch">
         <dt>If <var>name</var> is <code data-x="handler-onerror">onerror</code> and
         <var>eventTarget</var> is a <code>Window</code> object</dt>

         <dd>Let the function have five arguments, named <code data-x="">event</code>, <code
         data-x="">source</code>, <code data-x="">lineno</code>, <code data-x="">colno</code>, and
         <code data-x="">error</code>.</dd>

         <dt>Otherwise</dt>

         <dd>Let the function have a single argument called <code data-x="">event</code>.</dd>
        </dl>
       </dd>

       <dt><var>body</var></dt>

       <dd>The result of parsing <var>body</var> above.</dd>

       <dt><var>thisMode</var></dt>

       <dd>non-lexical-this</dd>

       <dt><var>scope</var></dt>

       <dd>
        <ol>
         <li><p>Let <var>realm</var> be <var>settings object</var>'s <span
         data-x="environment settings object's realm">realm</span>.</p></li>

         <li><p>Let <var>scope</var> be <var>realm</var>.[[GlobalEnv]].</p></li>

         <li>
          <p>If <var>eventHandler</var> is an element's <span data-x="event handlers">event
          handler</span>, then set <var>scope</var> to <span
          data-x="js-NewObjectEnvironment">NewObjectEnvironment</span>(<var>document</var>, true,
          <var>scope</var>).</p>

          <p>(Otherwise, <var>eventHandler</var> is a <code>Window</code> object's <span
          data-x="event handlers">event handler</span>.)</p>
         </li>

         <li><p>If <var>form owner</var> is not null, then set <var>scope</var> to <span
         data-x="js-NewObjectEnvironment">NewObjectEnvironment</span>(<var>form owner</var>, true,
         <var>scope</var>).</p></li>

         <li><p>If <var>element</var> is not null, then set <var>scope</var> to <span
         data-x="js-NewObjectEnvironment">NewObjectEnvironment</span>(<var>element</var>, true,
         <var>scope</var>).</p></li>

         <li><p>Return <var>scope</var>.</p></li>
        </ol>
       </dd>
      </dl>
     </li>

     <li><p>Remove <var>settings object</var>'s <span>realm execution context</span> from the
     <span>JavaScript execution context stack</span>.</p></li>

     <li>
      <p>Set <var>function</var>.[[ScriptOrModule]] to null.</p>

      <div class="note">
       <p>This is done because the default behavior, of associating the created function
       with the nearest <span data-x="concept-script">script</span> on the stack, can lead to
       path-dependent results. For example, an event handler which is first invoked by user
       interaction would end up with null [[ScriptOrModule]] (since then this algorithm would be
       first invoked when the <span>active script</span> is null), whereas one that is first invoked
       by dispatching an event from script would have its [[ScriptOrModule]] set to that script.</p>

       <p>Instead, we just always set [[ScriptOrModule]] to null. This is more intuitive anyway; the
       idea that the first script which dispatches an event is somehow responsible for the event
       handler code is dubious.</p>

       <p>In practice, this only affects the resolution of relative URLs via <code>import()</code>,
       which consult the <span data-x="concept-script-base-url">base URL</span> of the associated
       script. Nulling out [[ScriptOrModule]] means that <span>HostLoadImportedModule</span> will
       fall back to the <span>current settings object</span>'s <span>API base URL</span>.</p>
      </div>
     </li>

     <li><p>Set <var>eventHandler</var>'s <span data-x="event handler value">value</span> to the
     result of creating a Web IDL <code>EventHandler</code> callback function object whose object
     reference is <var>function</var> and whose <span>callback context</span> is <var>settings
     object</var>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>eventHandler</var>'s <span data-x="event handler value">value</span>.</p></li>
  </ol>

  </div>



  <h5>Event handlers on elements, <code>Document</code> objects, and <code>Window</code> objects</h5>

  <p>The following are the <span>event handlers</span> (and their corresponding <span data-x="event
  handler event type">event handler event types</span>) <span w-nodev>that must be</span>
  supported by all <span>HTML elements</span>, as both <span>event handler content attributes</span>
  and <span>event handler IDL attributes</span>; and <span w-nodev>that must be</span>
  supported by all <code>Document</code> and <code>Window</code> objects, as <span>event handler IDL
  attributes</span>:</p>

  <table>
   <thead>
    <tr><th><span data-x="event handlers">Event handler</span> <th><span>Event handler event type</span>
   <tbody>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onabort">onabort</code></dfn> <td> <code data-x="">abort</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onauxclick">onauxclick</code></dfn> <td> <code data-x="event-auxclick">auxclick</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onbeforeinput">onbeforeinput</code></dfn> <td> <code data-x="event-beforeinput">beforeinput</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onbeforematch">onbeforematch</code></dfn> <td> <code data-x="event-beforematch">beforematch</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onbeforetoggle">onbeforetoggle</code></dfn> <td> <code data-x="event-beforetoggle">beforetoggle</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-oncancel">oncancel</code></dfn> <td> <code data-x="event-cancel">cancel</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-oncanplay">oncanplay</code></dfn> <td> <code data-x="event-media-canplay">canplay</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-oncanplaythrough">oncanplaythrough</code></dfn> <td> <code data-x="event-media-canplaythrough">canplaythrough</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onchange">onchange</code></dfn> <td> <code data-x="event-change">change</code> <!-- widely used -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onclick">onclick</code></dfn> <td> <code data-x="event-click">click</code> <!-- widely used -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onclose">onclose</code></dfn> <td> <code data-x="event-close">close</code> <!-- new for <dialog> -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-oncommand">oncommand</code></dfn> <td> <code data-x="event-command">command</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-oncontextlost">oncontextlost</code></dfn> <td> <code data-x="event-contextlost">contextlost</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-oncontextmenu">oncontextmenu</code></dfn> <td> <code data-x="event-contextmenu">contextmenu</code> <!-- widely used -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-oncontextrestored">oncontextrestored</code></dfn> <td> <code data-x="event-contextrestored">contextrestored</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-oncopy">oncopy</code></dfn> <td> <code data-x="event-copy">copy</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-oncuechange">oncuechange</code></dfn> <td> <code data-x="event-media-cuechange">cuechange</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-oncut">oncut</code></dfn> <td> <code data-x="event-cut">cut</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-ondblclick">ondblclick</code></dfn> <td> <code data-x="event-dblclick">dblclick</code> <!-- widely used -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-ondrag">ondrag</code></dfn> <td> <code data-x="event-dnd-drag">drag</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-ondragend">ondragend</code></dfn> <td> <code data-x="event-dnd-dragend">dragend</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-ondragenter">ondragenter</code></dfn> <td> <code data-x="event-dnd-dragenter">dragenter</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-ondragleave">ondragleave</code></dfn> <td> <code data-x="event-dnd-dragleave">dragleave</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-ondragover">ondragover</code></dfn> <td> <code data-x="event-dnd-dragover">dragover</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-ondragstart">ondragstart</code></dfn> <td> <code data-x="event-dnd-dragstart">dragstart</code> <!-- widely used -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-ondrop">ondrop</code></dfn> <td> <code data-x="event-dnd-drop">drop</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-ondurationchange">ondurationchange</code></dfn> <td> <code data-x="event-media-durationchange">durationchange</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onemptied">onemptied</code></dfn> <td> <code data-x="event-media-emptied">emptied</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onended">onended</code></dfn> <td> <code data-x="event-media-ended">ended</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onformdata">onformdata</code></dfn> <td> <code data-x="event-formdata">formdata</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-oninput">oninput</code></dfn> <td> <code data-x="event-input">input</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-oninvalid">oninvalid</code></dfn> <td> <code data-x="event-invalid">invalid</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onkeydown">onkeydown</code></dfn> <td> <code data-x="event-keydown">keydown</code> <!-- widely used -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onkeypress">onkeypress</code></dfn> <td> <code data-x="event-keypress">keypress</code> <!-- widely used -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onkeyup">onkeyup</code></dfn> <td> <code data-x="event-keyup">keyup</code> <!-- widely used -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onloadeddata">onloadeddata</code></dfn> <td> <code data-x="event-media-loadeddata">loadeddata</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onloadedmetadata">onloadedmetadata</code></dfn> <td> <code data-x="event-media-loadedmetadata">loadedmetadata</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onloadstart">onloadstart</code></dfn> <td> <code data-x="event-media-loadstart">loadstart</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onmousedown">onmousedown</code></dfn> <td> <code data-x="event-mousedown">mousedown</code> <!-- widely used -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onmouseenter">onmouseenter</code></dfn> <td> <code data-x="event-mouseenter">mouseenter</code> <!-- compat -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onmouseleave">onmouseleave</code></dfn> <td> <code data-x="event-mouseleave">mouseleave</code> <!-- compat -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onmousemove">onmousemove</code></dfn> <td> <code data-x="event-mousemove">mousemove</code> <!-- widely used -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onmouseout">onmouseout</code></dfn> <td> <code data-x="event-mouseout">mouseout</code> <!-- widely used -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onmouseover">onmouseover</code></dfn> <td> <code data-x="event-mouseover">mouseover</code> <!-- widely used -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onmouseup">onmouseup</code></dfn> <td> <code data-x="event-mouseup">mouseup</code> <!-- widely used -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onpaste">onpaste</code></dfn> <td> <code data-x="event-paste">paste</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onpause">onpause</code></dfn> <td> <code data-x="event-media-pause">pause</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onplay">onplay</code></dfn> <td> <code data-x="event-media-play">play</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onplaying">onplaying</code></dfn> <td> <code data-x="event-media-playing">playing</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onprogress">onprogress</code></dfn> <td> <code data-x="event-media-progress">progress</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onratechange">onratechange</code></dfn> <td> <code data-x="event-media-ratechange">ratechange</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onreset">onreset</code></dfn> <td> <code data-x="event-reset">reset</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onscrollend">onscrollend</code></dfn> <td> <code data-x="event-scrollend">scrollend</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onsecuritypolicyviolation">onsecuritypolicyviolation</code></dfn> <td> <code data-x="event-securitypolicyviolation">securitypolicyviolation</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onseeked">onseeked</code></dfn> <td> <code data-x="event-media-seeked">seeked</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onseeking">onseeking</code></dfn> <td> <code data-x="event-media-seeking">seeking</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onselect">onselect</code></dfn> <td> <code data-x="event-select">select</code> <!-- widely used --> <!-- [CSSOM] -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onslotchange">onslotchange</code></dfn> <td> <code data-x="event-slotchange">slotchange</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onstalled">onstalled</code></dfn> <td> <code data-x="event-media-stalled">stalled</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onsubmit">onsubmit</code></dfn> <td> <code data-x="event-submit">submit</code> <!-- widely used -->
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onsuspend">onsuspend</code></dfn> <td> <code data-x="event-media-suspend">suspend</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-ontimeupdate">ontimeupdate</code></dfn> <td> <code data-x="event-media-timeupdate">timeupdate</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-ontoggle">ontoggle</code></dfn> <td> <code data-x="event-toggle">toggle</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onvolumechange">onvolumechange</code></dfn> <td> <code data-x="event-media-volumechange">volumechange</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onwaiting">onwaiting</code></dfn> <td> <code data-x="event-media-waiting">waiting</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onwebkitanimationend">onwebkitanimationend</code></dfn> <td> <code data-x="">webkitAnimationEnd</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onwebkitanimationiteration">onwebkitanimationiteration</code></dfn> <td> <code data-x="">webkitAnimationIteration</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onwebkitanimationstart">onwebkitanimationstart</code></dfn> <td> <code data-x="">webkitAnimationStart</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onwebkittransitionend">onwebkittransitionend</code></dfn> <td> <code data-x="">webkitTransitionEnd</code>
    <tr><td><dfn attribute for="HTMLElement,Document,Window,GlobalEventHandlers"><code data-x="handler-onwheel">onwheel</code></dfn> <td> <code data-x="event-wheel">wheel</code> <!-- widely used -->

<!-- not supported, use dnd: -->
<!--<tr><td><dfn><code data-x="handler-onbeforecopy">onbeforecopy</code></dfn> <td> <code data-x="event-cp-beforecopy">beforecopy</code>--> <!-- widely used -->
<!-- not supported yet (v2?): -->
<!--<tr><td><dfn><code data-x="handler-onselectstart">onselectstart</code></dfn> <td> <code data-x="event-selectstart">selectstart</code>--> <!-- widely used -->
  </table>

  <hr>

  <p>The following are the <span>event handlers</span> (and their corresponding <span data-x="event
  handler event type">event handler event types</span>) <span w-nodev>that must be</span>
  supported by all <span>HTML elements</span> other than <code>body</code> and <code>frameset</code>
  elements, as both <span>event handler content attributes</span> and <span>event handler IDL
  attributes</span>; <span w-nodev>that must be</span> supported by all <code>Document</code>
  objects, as <span>event handler IDL attributes</span>; and <span w-nodev>that must be</span>
  supported by all <code>Window</code> objects, as <span>event handler IDL attributes</span> on the
  <code>Window</code> objects themselves, and with corresponding <span>event handler content
  attributes</span> and <span>event handler IDL attributes</span> exposed on all <code>body</code>
  and <code>frameset</code> elements that are owned by that <code>Window</code> object's <span
  data-x="concept-document-window">associated <code>Document</code></span>:</p>

  <table>
   <thead>
    <tr><th><span data-x="event handlers">Event handler</span> <th><span>Event handler event type</span>
   <tbody>
    <tr><td><dfn attribute for="GlobalEventHandlers"><code data-x="handler-onblur">onblur</code></dfn> <td> <code data-x="event-blur">blur</code> <!-- widely used -->
    <tr><td><dfn attribute for="GlobalEventHandlers"><code data-x="handler-onerror">onerror</code></dfn> <td> <code data-x="event-error">error</code>
    <tr><td><dfn attribute for="GlobalEventHandlers"><code data-x="handler-onfocus">onfocus</code></dfn> <td> <code data-x="event-focus">focus</code> <!-- widely used -->
    <tr><td><dfn attribute for="GlobalEventHandlers"><code data-x="handler-onload">onload</code></dfn> <td> <code data-x="event-load">load</code>
    <tr><td><dfn attribute for="GlobalEventHandlers"><code data-x="handler-onresize">onresize</code></dfn> <td> <code data-x="event-resize">resize</code>
    <tr><td><dfn attribute for="GlobalEventHandlers"><code data-x="handler-onscroll">onscroll</code></dfn> <td> <code data-x="event-scroll">scroll</code>
  </table>

  <p>We call the <span>set</span> of the <span data-x="event handler name">names</span> of the
  <span>event handlers</span> listed in the first column of this table the
  <dfn><code>Window</code>-reflecting body element event handler set</dfn>.</p>

  <hr>

  <p>The following are the <span>event handlers</span> (and their corresponding <span data-x="event
  handler event type">event handler event types</span>) <span w-nodev>that must be</span>
  supported by <code>Window</code> objects, as <span>event handler IDL attributes</span> on the
  <code>Window</code> objects themselves, and with corresponding <span>event handler content
  attributes</span> and <span>event handler IDL attributes</span> exposed on all <code>body</code>
  and <code>frameset</code> elements that are owned by that <code>Window</code> object's <span
  data-x="concept-document-window">associated <code>Document</code></span>:</p>

  <table>
   <thead>
    <tr><th><span data-x="event handlers">Event handler</span> <th><span>Event handler event type</span>
   <tbody>
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-onafterprint">onafterprint</code></dfn> <td> <code data-x="event-afterprint">afterprint</code>
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-onbeforeprint">onbeforeprint</code></dfn> <td> <code data-x="event-beforeprint">beforeprint</code>
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-onbeforeunload">onbeforeunload</code></dfn> <td> <code data-x="event-beforeunload">beforeunload</code>
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-onhashchange">onhashchange</code></dfn> <td> <code data-x="event-hashchange">hashchange</code> <!-- new -->
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-onlanguagechange">onlanguagechange</code></dfn> <td> <code data-x="event-languagechange">languagechange</code> <!-- new -->
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-onmessage">onmessage</code></dfn> <td> <code data-x="event-message">message</code> <!-- new for postMessage -->
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-onmessageerror">onmessageerror</code></dfn> <td> <code data-x="event-messageerror">messageerror</code> <!-- new for SAB -->
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-onoffline">onoffline</code></dfn> <td> <code data-x="event-offline">offline</code> <!-- new -->
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-ononline">ononline</code></dfn> <td> <code data-x="event-online">online</code> <!-- new -->
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-onpageswap">onpageswap</code></dfn> <td> <code data-x="event-pageswap">pageswap</code> <!-- new -->
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-onpagehide">onpagehide</code></dfn> <td> <code data-x="event-pagehide">pagehide</code> <!-- new -->
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-onpagereveal">onpagereveal</code></dfn> <td> <code data-x="event-pagereveal">pagereveal</code> <!-- new -->
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-onpageshow">onpageshow</code></dfn> <td> <code data-x="event-pageshow">pageshow</code> <!-- new -->
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-onpopstate">onpopstate</code></dfn> <td> <code data-x="event-popstate">popstate</code> <!-- new -->
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-onrejectionhandled">onrejectionhandled</code></dfn> <td> <code data-x="event-rejectionhandled">rejectionhandled</code>
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-onstorage">onstorage</code></dfn> <td> <code data-x="event-storage">storage</code> <!-- new -->
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-onunhandledrejection">onunhandledrejection</code></dfn> <td> <code data-x="event-unhandledrejection">unhandledrejection</code>
    <tr><td><dfn attribute for="WindowEventHandlers"><code data-x="handler-window-onunload">onunload</code></dfn> <td> <code data-x="event-unload">unload</code> <!-- widely used -->
  </table>

  <p w-nodev>This list of <span>event handlers</span> is reified as <span>event handler IDL
  attributes</span> through the <code>WindowEventHandlers</code> interface mixin.</p>

  <hr>

  <p>The following are the <span>event handlers</span> (and their corresponding <span data-x="event
  handler event type">event handler event types</span>) <span w-nodev>that must be</span>
  supported on <code>Document</code> objects as <span>event handler IDL attributes</span>:</p>

  <!-- This is only on Document and not on HTMLElement because otherwise HTMLScriptElement would
  have it and that causes compatibility issues since IE fires readystatechange events on <script>,
  not load events, and we can't fire both, and some sites try to decide which to look for based on
  the presence of script.onreadystatechange on HTMLScriptElement.
     https://www.w3.org/Bugs/Public/show_bug.cgi?id=13965
     https://lists.w3.org/Archives/Public/public-whatwg-archive/2011Sep/0074.html
  -->

  <table>
   <thead>
    <tr><th><span data-x="event handlers">Event handler</span> <th><span>Event handler event type</span>
   <tbody>
    <tr><td><dfn attribute for="Document"><code data-x="handler-onreadystatechange">onreadystatechange</code></dfn> <td> <code data-x="event-readystatechange">readystatechange</code>
    <tr><td><dfn attribute for="Document"><code data-x="handler-onvisibilitychange">onvisibilitychange</code></dfn> <td> <code data-x="event-visibilitychange">visibilitychange</code>
  </table>


  <div w-nodev>

  <h6>IDL definitions</h6>

  <pre><code class="idl">interface mixin <dfn interface>GlobalEventHandlers</dfn> {
  attribute <span>EventHandler</span> <span data-x="handler-onabort">onabort</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onauxclick">onauxclick</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onbeforeinput">onbeforeinput</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onbeforematch">onbeforematch</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onbeforetoggle">onbeforetoggle</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onblur">onblur</span>;
  attribute <span>EventHandler</span> <span data-x="handler-oncancel">oncancel</span>;
  attribute <span>EventHandler</span> <span data-x="handler-oncanplay">oncanplay</span>;
  attribute <span>EventHandler</span> <span data-x="handler-oncanplaythrough">oncanplaythrough</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onchange">onchange</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onclick">onclick</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onclose">onclose</span>;
  attribute <span>EventHandler</span> <span data-x="handler-oncommand">oncommand</span>;
  attribute <span>EventHandler</span> <span data-x="handler-oncontextlost">oncontextlost</span>;
  attribute <span>EventHandler</span> <span data-x="handler-oncontextmenu">oncontextmenu</span>;
  attribute <span>EventHandler</span> <span data-x="handler-oncontextrestored">oncontextrestored</span>;
  attribute <span>EventHandler</span> <span data-x="handler-oncopy">oncopy</span>;
  attribute <span>EventHandler</span> <span data-x="handler-oncuechange">oncuechange</span>;
  attribute <span>EventHandler</span> <span data-x="handler-oncut">oncut</span>;
  attribute <span>EventHandler</span> <span data-x="handler-ondblclick">ondblclick</span>;
  attribute <span>EventHandler</span> <span data-x="handler-ondrag">ondrag</span>;
  attribute <span>EventHandler</span> <span data-x="handler-ondragend">ondragend</span>;
  attribute <span>EventHandler</span> <span data-x="handler-ondragenter">ondragenter</span>;
  attribute <span>EventHandler</span> <span data-x="handler-ondragleave">ondragleave</span>;
  attribute <span>EventHandler</span> <span data-x="handler-ondragover">ondragover</span>;
  attribute <span>EventHandler</span> <span data-x="handler-ondragstart">ondragstart</span>;
  attribute <span>EventHandler</span> <span data-x="handler-ondrop">ondrop</span>;
  attribute <span>EventHandler</span> <span data-x="handler-ondurationchange">ondurationchange</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onemptied">onemptied</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onended">onended</span>;
  attribute <span>OnErrorEventHandler</span> <span data-x="handler-onerror">onerror</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onfocus">onfocus</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onformdata">onformdata</span>;
  attribute <span>EventHandler</span> <span data-x="handler-oninput">oninput</span>;
  attribute <span>EventHandler</span> <span data-x="handler-oninvalid">oninvalid</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onkeydown">onkeydown</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onkeypress">onkeypress</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onkeyup">onkeyup</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onload">onload</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onloadeddata">onloadeddata</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onloadedmetadata">onloadedmetadata</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onloadstart">onloadstart</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onmousedown">onmousedown</span>;
  [<span>LegacyLenientThis</span>] attribute <span>EventHandler</span> <span data-x="handler-onmouseenter">onmouseenter</span>;
  [<span>LegacyLenientThis</span>] attribute <span>EventHandler</span> <span data-x="handler-onmouseleave">onmouseleave</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onmousemove">onmousemove</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onmouseout">onmouseout</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onmouseover">onmouseover</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onmouseup">onmouseup</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onpaste">onpaste</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onpause">onpause</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onplay">onplay</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onplaying">onplaying</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onprogress">onprogress</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onratechange">onratechange</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onreset">onreset</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onresize">onresize</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onscroll">onscroll</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onscrollend">onscrollend</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onsecuritypolicyviolation">onsecuritypolicyviolation</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onseeked">onseeked</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onseeking">onseeking</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onselect">onselect</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onslotchange">onslotchange</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onstalled">onstalled</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onsubmit">onsubmit</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onsuspend">onsuspend</span>;
  attribute <span>EventHandler</span> <span data-x="handler-ontimeupdate">ontimeupdate</span>;
  attribute <span>EventHandler</span> <span data-x="handler-ontoggle">ontoggle</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onvolumechange">onvolumechange</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onwaiting">onwaiting</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onwebkitanimationend">onwebkitanimationend</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onwebkitanimationiteration">onwebkitanimationiteration</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onwebkitanimationstart">onwebkitanimationstart</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onwebkittransitionend">onwebkittransitionend</span>;
  attribute <span>EventHandler</span> <span data-x="handler-onwheel">onwheel</span>;
};

interface mixin <dfn interface>WindowEventHandlers</dfn> {
  attribute <span>EventHandler</span> <span data-x="handler-window-onafterprint">onafterprint</span>;
  attribute <span>EventHandler</span> <span data-x="handler-window-onbeforeprint">onbeforeprint</span>;
  attribute <span>OnBeforeUnloadEventHandler</span> <span data-x="handler-window-onbeforeunload">onbeforeunload</span>;
  attribute <span>EventHandler</span> <span data-x="handler-window-onhashchange">onhashchange</span>;
  attribute <span>EventHandler</span> <span data-x="handler-window-onlanguagechange">onlanguagechange</span>;
  attribute <span>EventHandler</span> <span data-x="handler-window-onmessage">onmessage</span>;
  attribute <span>EventHandler</span> <span data-x="handler-window-onmessageerror">onmessageerror</span>;
  attribute <span>EventHandler</span> <span data-x="handler-window-onoffline">onoffline</span>;
  attribute <span>EventHandler</span> <span data-x="handler-window-ononline">ononline</span>;
  attribute <span>EventHandler</span> <span data-x="handler-window-onpagehide">onpagehide</span>;
  attribute <span>EventHandler</span> <span data-x="handler-window-onpagereveal">onpagereveal</span>;
  attribute <span>EventHandler</span> <span data-x="handler-window-onpageshow">onpageshow</span>;
  attribute <span>EventHandler</span> <span data-x="handler-window-onpageswap">onpageswap</span>;
  attribute <span>EventHandler</span> <span data-x="handler-window-onpopstate">onpopstate</span>;
  attribute <span>EventHandler</span> <span data-x="handler-window-onrejectionhandled">onrejectionhandled</span>;
  attribute <span>EventHandler</span> <span data-x="handler-window-onstorage">onstorage</span>;
  attribute <span>EventHandler</span> <span data-x="handler-window-onunhandledrejection">onunhandledrejection</span>;
  attribute <span>EventHandler</span> <span data-x="handler-window-onunload">onunload</span>;
};</code></pre>

  <h5>Event firing</h5>

  <p>Certain operations and methods are defined as firing events on elements. For example, the <code
  data-x="dom-click">click()</code> method on the <code>HTMLElement</code> interface is defined as
  firing a <code data-x="event-click">click</code> event on the element. <ref>UIEVENTS</ref></p>

  <p><dfn data-x="fire a synthetic pointer event">Firing a synthetic pointer event named
  <var>e</var></dfn> at <var>target</var>, with an optional <var>not trusted flag</var>, means
  running these steps:</p>

  <ol>
   <li><p>Let <var>event</var> be the result of <span>creating an event</span> using
   <code>PointerEvent</code>.</p></li>

   <li><p>Initialize <var>event</var>'s <code data-x="dom-Event-type">type</code> attribute to
   <var>e</var>.</p></li>

   <li><p>Initialize <var>event</var>'s <code data-x="dom-Event-bubbles">bubbles</code> and <code
   data-x="dom-Event-cancelable">cancelable</code> attributes to true.</p></li>

   <li><p>Set <var>event</var>'s <span>composed flag</span>.</p></li>

   <li><p>If the <var>not trusted flag</var> is set, initialize <var>event</var>'s <code
   data-x="dom-Event-isTrusted">isTrusted</code> attribute to false.</p></li>

   <li><p>Initialize <var>event</var>'s <code data-x="">ctrlKey</code>, <code
   data-x="">shiftKey</code>, <code data-x="">altKey</code>, and <code data-x="">metaKey</code>
   attributes according to the current state of the key input device, if any (false for any keys
   that are not available).</p></li>

   <li><p>Initialize <var>event</var>'s <code data-x="dom-UIEvent-view">view</code> attribute to
   <var>target</var>'s <span>node document</span>'s <code>Window</code> object, if any, and null
   otherwise.</p></li>

   <li><p><var>event</var>'s <code data-x="">getModifierState()</code> method is to return values
   appropriately describing the current state of the key input device.</p></li>

   <li><p>Return the result of <span data-x="concept-event-dispatch">dispatching</span>
   <var>event</var> at <var>target</var>.</p></li>
  </ol>

  <p><dfn data-x="fire a click event">Firing a <code data-x="event-click">click</code> event</dfn>
  at <var>target</var> means <span data-x="fire a synthetic pointer event">firing a synthetic
  pointer event named <code data-x="event-click">click</code></span> at <var>target</var>.</p>

  </div>


  <h3 id="windoworworkerglobalscope-mixin">The <code>WindowOrWorkerGlobalScope</code> mixin</h3>

  <p>The <code>WindowOrWorkerGlobalScope</code> mixin is for use of APIs that are to be exposed on
  <code>Window</code> and <code>WorkerGlobalScope</code> objects.</p>

  <p class="note">Other standards are encouraged to further extend it using <code data-x="">partial
  interface mixin <span>WindowOrWorkerGlobalScope</span> { &hellip; };</code> along with an
  appropriate reference.</p>

  <pre><code class="idl">typedef (DOMString or <span data-x="idl-Function">Function</span> or <span data-x="tt-trustedscript">TrustedScript</span>) <dfn typedef>TimerHandler</dfn>;

interface mixin <dfn interface>WindowOrWorkerGlobalScope</dfn> {
  [Replaceable] readonly attribute USVString <span data-x="dom-origin">origin</span>;
  readonly attribute boolean <span data-x="dom-isSecureContext">isSecureContext</span>;
  readonly attribute boolean <span data-x="dom-crossOriginIsolated">crossOriginIsolated</span>;

  undefined <span data-x="dom-reportError">reportError</span>(any e);

  // base64 utility methods
  DOMString <span data-x="dom-btoa">btoa</span>(DOMString data);
  ByteString <span data-x="dom-atob">atob</span>(DOMString data);

  // timers
  long <span data-x="dom-setTimeout">setTimeout</span>(<span>TimerHandler</span> handler, optional long timeout = 0, any... arguments);
  undefined <span data-x="dom-clearTimeout">clearTimeout</span>(optional long id = 0);
  long <span data-x="dom-setInterval">setInterval</span>(<span>TimerHandler</span> handler, optional long timeout = 0, any... arguments);
  undefined <span data-x="dom-clearInterval">clearInterval</span>(optional long id = 0);

  // microtask queuing
  undefined <span data-x="dom-queueMicrotask">queueMicrotask</span>(<span data-x="idl-VoidFunction">VoidFunction</span> callback);

  // ImageBitmap
  <span data-x="idl-Promise">Promise</span>&lt;<span>ImageBitmap</span>&gt; <span data-x="dom-createImageBitmap">createImageBitmap</span>(<span>ImageBitmapSource</span> image, optional <span>ImageBitmapOptions</span> options = {});
  <span data-x="idl-Promise">Promise</span>&lt;<span>ImageBitmap</span>&gt; <span data-x="dom-createImageBitmap">createImageBitmap</span>(<span>ImageBitmapSource</span> image, long sx, long sy, long sw, long sh, optional <span>ImageBitmapOptions</span> options = {});

  // structured cloning
  any <span data-x="dom-structuredClone">structuredClone</span>(any value, optional <span>StructuredSerializeOptions</span> options = {});
};
<span>Window</span> includes <span>WindowOrWorkerGlobalScope</span>;
<span>WorkerGlobalScope</span> includes <span>WindowOrWorkerGlobalScope</span>;</code></pre>

  <dl class="domintro">
   <dt><code data-x="">self.<span subdfn data-x="dom-isSecureContext">isSecureContext</span></code></dt>
   <dd><p>Returns whether or not this global object represents a <span>secure context</span>.
   <ref>SECURE-CONTEXTS</ref></p></dd>

   <dt><code data-x="">self.<span subdfn data-x="dom-origin">origin</span></code></dt>
   <dd><p>Returns the global object's <span>origin</span>, serialized as string.</p></dd>

   <dt><code data-x="">self.<span subdfn data-x="dom-crossOriginIsolated">crossOriginIsolated</span></code></dt>
   <dd><p>Returns whether scripts running in this global are allowed to use APIs that require
   cross-origin isolation. This depends on the `<code>Cross-Origin-Opener-Policy</code>` and
   `<code>Cross-Origin-Embedder-Policy</code>` HTTP response headers and the "<code
   data-x="cross-origin-isolated-feature">cross-origin-isolated</code>" feature.</p></dd>
  </dl>

  <div class="example">
   <p>Developers are strongly encouraged to use <code data-x="">self.origin</code> over <code
   data-x="">location.origin</code>. The former returns the <span>origin</span> of the environment,
   the latter of the URL of the environment. Imagine the following script executing in a document on
   <code data-x="">https://stargate.example/</code>:</p>

   <pre><code class="js">var frame = document.createElement("iframe")
frame.onload = function() {
  var frameWin = frame.contentWindow
  console.log(frameWin.location.origin) // "null"
  console.log(frameWin.origin) // "https://stargate.example"
}
document.body.appendChild(frame)</code></pre>

   <p><code data-x="">self.origin</code> is a more reliable security indicator.</p>
  </div>

  <div w-nodev>

  <p>The <dfn attribute for="WindowOrWorkerGlobalScope"
  data-x="dom-isSecureContext"><code>isSecureContext</code></dfn> getter steps are to return true if
  <span>this</span>'s <span>relevant settings object</span> is a <span>secure context</span>, or
  false otherwise.</p>

  <p>The <dfn attribute for="WindowOrWorkerGlobalScope"
  data-x="dom-origin"><code>origin</code></dfn> getter steps are to return <span>this</span>'s
  <span>relevant settings object</span>'s <span
  data-x="concept-settings-object-origin">origin</span>, <span data-x="serialization of an
  origin">serialized</span>.</p>

  <p>The <dfn attribute for="WindowOrWorkerGlobalScope"
  data-x="dom-crossOriginIsolated"><code>crossOriginIsolated</code></dfn> getter steps are to return
  <span>this</span>'s <span>relevant settings object</span>'s <span
  data-x="concept-settings-object-cross-origin-isolated-capability">cross-origin isolated
  capability</span>.</p>

  </div>


  <h3 id="atob">Base64 utility methods</h3>

  <p>The <code data-x="dom-atob">atob()</code> and <code data-x="dom-btoa">btoa()</code> methods
  allow developers to transform content to and from the base64 encoding.</p>

  <p class="note">In these APIs, for mnemonic purposes, the "b" can be considered to stand for
  "binary", and the "a" for "ASCII". In practice, though, for primarily historical reasons, both the
  input and output of these functions are Unicode strings.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>result</var> = self.<span subdfn data-x="dom-btoa">btoa</span>(<var>data</var>)</code></dt>

   <dd>
    <p>Takes the input data, in the form of a Unicode string containing only characters in the range
    U+0000 to U+00FF, each representing a binary byte with values 0x00 to 0xFF respectively, and
    converts it to its base64 representation, which it returns.</p>

    <p>Throws an <span>"<code>InvalidCharacterError</code>"</span> <code>DOMException</code>
    exception if the input string contains any out-of-range characters.</p>
   </dd>

   <dt><code data-x=""><var>result</var> = self.<span subdfn data-x="dom-atob">atob</span>(<var>data</var>)</code></dt>

   <dd>
    <p>Takes the input data, in the form of a Unicode string containing base64-encoded binary data,
    decodes it, and returns a string consisting of characters in the range U+0000 to U+00FF, each
    representing a binary byte with values 0x00 to 0xFF respectively, corresponding to that binary
    data.</p>

    <p>Throws an <span>"<code>InvalidCharacterError</code>"</span> <code>DOMException</code> if the
    input string is not valid base64 data.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="WindowOrWorkerGlobalScope" data-x="dom-btoa"><code
  id="dom-windowbase64-btoa">btoa(<var>data</var>)</code></dfn> method must throw an
  <span>"<code>InvalidCharacterError</code>"</span> <code>DOMException</code> if <var>data</var>
  contains any character whose code point is greater than U+00FF. Otherwise, the user agent must
  convert <var>data</var> to a byte sequence whose <var>n</var>th byte is the eight-bit
  representation of the <var>n</var>th code point of <var>data</var>, and then must apply
  <span>forgiving-base64 encode</span> to that byte sequence and return the result.</p>

  <p>The <dfn method for="WindowOrWorkerGlobalScope" data-x="dom-atob"><code
  id="dom-windowbase64-atob">atob(<var>data</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>decodedData</var> be the result of running <span>forgiving-base64 decode</span>
   on <var>data</var>.</p></li>

   <li><p>If <var>decodedData</var> is failure, then throw an
   <span>"<code>InvalidCharacterError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Return <var>decodedData</var>.</p></li>
  </ol>

  </div>


  <h3 split-filename="dynamic-markup-insertion"><dfn>Dynamic markup insertion</dfn></h3>

  <p class="note">APIs for dynamically inserting markup into the document interact with the parser,
  and thus their behavior varies depending on whether they are used with <span>HTML
  documents</span> (and the <span>HTML parser</span>) or <span>XML documents</span> (and the
  <span>XML parser</span>).</p>

  <div w-nodev>

  <p><code>Document</code> objects have a <dfn>throw-on-dynamic-markup-insertion counter</dfn>,
  which is used in conjunction with the <span>create an element for the token</span> algorithm to
  prevent <span data-x="custom element constructor">custom element constructors</span> from being
  able to use <code data-x="dom-document-open">document.open()</code>, <code
  data-x="dom-document-close">document.close()</code>, and <code
  data-x="dom-document-write">document.write()</code> when they are invoked by the parser.
  Initially, the counter must be set to zero.</p>

  </div>



  <h4>Opening the input stream</h4>

  <dl class="domintro">
   <dt><code data-x=""><var>document</var> = <var>document</var>.<span subdfn data-x="dom-document-open">open</span>()</code></dt>
   <dd>
    <p>Causes the <code>Document</code> to be replaced in-place, as if it was a new
    <code>Document</code> object, but reusing the previous object, which is then returned.</p>

    <p>The resulting <code>Document</code> has an HTML parser associated with it, which can be given
    data to parse using <code data-x="dom-document-write">document.write()</code>.</p>

    <p>The method has no effect if the <code>Document</code> is still being parsed.</p>

    <p>Throws an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> if the
    <code>Document</code> is an <span data-x="XML documents">XML document</span>.</p>

    <p>Throws an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> if the
    parser is currently executing a <span>custom element constructor</span>.</p>
   </dd>

   <dt><code data-x=""><var>window</var> = <var>document</var>.<span data-x="dom-document-open-window">open</span>(<var>url</var>, <var>name</var>, <var>features</var>)</code></dt>
   <dd><p>Works like the <code data-x="dom-open">window.open()</code> method.</p></dd>
  </dl>

  <div w-nodev>

  <p><code>Document</code> objects have an <dfn>active parser was aborted</dfn> boolean, which is
  used to prevent scripts from invoking the <code data-x="dom-document-open">document.open()</code>
  and <code data-x="dom-document-write">document.write()</code> methods (directly or indirectly)
  after the document's <span>active parser</span> has been aborted. It is initially false.</p>

  <p>The <dfn>document open steps</dfn>, given a <var>document</var>, are as follows:</p>

  <ol>
   <li><p>If <var>document</var> is an <span data-x="XML documents">XML document</span>, then throw
   an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> exception.</p></li>

   <li><p>If <var>document</var>'s <span>throw-on-dynamic-markup-insertion counter</span> is greater
   than 0, then throw an <span>"<code>InvalidStateError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Let <var>entryDocument</var> be the <span>entry global object</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>If <var>document</var>'s <span data-x="concept-document-origin">origin</span> is not
   <span>same origin</span> to <var>entryDocument</var>'s <span
   data-x="concept-document-origin">origin</span>, then throw a
   <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li>
    <p>If <var>document</var> has an <span>active parser</span> whose <span>script nesting
    level</span> is greater than 0, then return <var>document</var>.</p>

    <p class="note">This basically causes <code data-x="dom-document-open">document.open()</code> to
    be ignored when it's called in an inline script found during parsing, while still letting it
    have an effect when called from a non-parser task such as a timer callback or event handler.</p>
   </li>

   <li>
    <p>Similarly, if <var>document</var>'s <span>unload counter</span> is greater than 0, then
    return <var>document</var>.</p>

    <p class="note">This basically causes <code data-x="dom-document-open">document.open()</code> to
    be ignored when it's called from a <code data-x="event-beforeunload">beforeunload</code>, <code
    data-x="event-pagehide">pagehide</code>, or <code data-x="event-unload">unload</code> event
    handler while the <code>Document</code> is being unloaded.</p>
   </li>

   <li>
    <p>If <var>document</var>'s <span>active parser was aborted</span> is true, then return
    <var>document</var>.</p>

    <p class="note">This notably causes <code data-x="dom-document-open">document.open()</code> to
    be ignored if it is called after a <span data-x="navigate">navigation</span> has started, but
    only during the initial parse. See <a href="https://github.com/whatwg/html/issues/4723">issue
    #4723</a> for more background.</p>
   </li>

   <li><p>If <var>document</var>'s <span>node navigable</span> is non-null and
   <var>document</var>'s <span>node navigable</span>'s <span>ongoing navigation</span> is a
   <span>navigation ID</span>, then <span data-x="nav-stop">stop loading</span>
   <var>document</var>'s <span>node navigable</span>.</p></li>

   <li><p>For each <span>shadow-including inclusive descendant</span> <var>node</var> of
   <var>document</var>, <span>erase all event listeners and handlers</span> given
   <var>node</var>.</p></li>

   <li><p>If <var>document</var> is the <span data-x="concept-document-window">associated
   <code>Document</code></span> of <var>document</var>'s <span>relevant global object</span>, then
   <span>erase all event listeners and handlers</span> given <var>document</var>'s <span>relevant
   global object</span>.</p></li>

   <li><p><span data-x="concept-node-replace-all">Replace all</span> with null within
   <var>document</var>.</p></li>

   <li>
    <p>If <var>document</var> is <span>fully active</span>, then:</p>

    <ol>
     <li><p>Let <var>newURL</var> be a copy of <var>entryDocument</var>'s <span
     data-x="concept-document-url">URL</span>.</p></li>

     <li><p>If <var>entryDocument</var> is not <var>document</var>, then set <var>newURL</var>'s
     <span data-x="concept-url-fragment">fragment</span> to null.</p></li>

     <li><p>Run the <span>URL and history update steps</span> with <var>document</var> and
     <var>newURL</var>.</p></li>
    </ol>
   </li>

   <!-- <span>the document's referrer</span> stays the same -->

   <li><p>Set <var>document</var>'s <span>is initial <code>about:blank</code></span> to
   false.</p></li>

   <li><p>If <var>document</var>'s <span>iframe load in progress</span> flag is set, then set
   <var>document</var>'s <span>mute iframe load</span> flag.</p></li>

   <li><p>Set <var>document</var> to <span>no-quirks mode</span>.</p></li>

   <li><p>Create a new <span>HTML parser</span> and associate it with <var>document</var>. This is a
   <dfn>script-created parser</dfn> (meaning that it can be closed by the <code
   data-x="dom-document-open">document.open()</code> and <code
   data-x="dom-document-close">document.close()</code> methods, and that the tokenizer will wait for
   an explicit call to <code data-x="dom-document-close">document.close()</code> before emitting an
   end-of-file token). The encoding <span data-x="concept-encoding-confidence">confidence</span> is
   <i>irrelevant</i>.</p></li>

   <li><p>Set the <span>insertion point</span> to point at just before the end of the <span>input
   stream</span> (which at this point will be empty).</p></li>

   <li>
    <p><span>Update the current document readiness</span> of <var>document</var> to "<code
    data-x="">loading</code>".</p>

    <p class="note">This causes a <code data-x="event-readystatechange">readystatechange</code>
    event to fire, but the event is actually unobservable to author code, because of the previous
    step which <span data-x="erase all event listeners and handlers">erased all event listeners and
    handlers</span> that could observe it.</p>
   </li>

   <li><p>Return <var>document</var>.</p></li>
  </ol>

  <p class="note">The <span>document open steps</span> do not affect whether a <code>Document</code>
  is <span>ready for post-load tasks</span> or <span>completely loaded</span>.</p>

  <p>The <dfn method for="Document" data-x="dom-document-open"><code>open(<var>unused1</var>,
  <var>unused2</var>)</code></dfn> method must return the result of running the <span>document open
  steps</span> with <span>this</span>.</p>

  <p class="note" id="dom-document-open-unused-arguments">The <var>unused1</var> and
  <var>unused2</var> arguments are ignored, but kept in the IDL to allow code that calls the
  function with one or two arguments to continue working. They are necessary due to Web IDL
  <span>overload resolution algorithm</span> rules, which would throw a <code>TypeError</code>
  exception for such calls had the arguments not been there. <a
  href="https://github.com/whatwg/webidl/issues/581">whatwg/webidl issue #581</a> investigates
  changing the algorithm to allow for their removal. <ref>WEBIDL</ref></p>

  <p>The <dfn method for="Document" data-x="dom-document-open-window"><code>open(<var>url</var>,
  <var>name</var>, <var>features</var>)</code></dfn> method must run these steps:</p>

  <ol>
   <li><p>If <span>this</span> is not <span>fully active</span>, then throw an
   <span>"<code>InvalidAccessError</code>"</span> <code>DOMException</code> exception.</p></li>

   <li><p>Return the result of running the <span>window open steps</span> with <var>url</var>,
   <var>name</var>, and <var>features</var>.</p></li>
  </ol>

  </div>



  <h4>Closing the input stream</h4>

  <dl class="domintro">
   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-close">close</span>()</code></dt>

   <dd>
    <p>Closes the input stream that was opened by the <code
    data-x="dom-document-open">document.open()</code> method.</p>

    <p>Throws an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> if the
    <code>Document</code> is an <span data-x="XML documents">XML document</span>.</p>

    <p>Throws an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> if the
    parser is currently executing a <span>custom element constructor</span>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="Document"><code data-x="dom-document-close">close()</code></dfn> method must run the following
  steps:</p>

  <ol>
   <li><p>If <span>this</span> is an <span data-x="XML documents">XML document</span>, then throw
   an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If <span>this</span>'s <span>throw-on-dynamic-markup-insertion counter</span> is greater
   than zero, then throw an <span>"<code>InvalidStateError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>If there is no <span>script-created parser</span> associated with <span>this</span>, then
   return.</p></li>

   <li><p>Insert an <span>explicit "EOF" character</span> at the end of the parser's <span>input
   stream</span>.</p></li>

   <li><p>If <span>this</span>'s <span>pending parsing-blocking script</span> is not null, then
   return.</p></li>

   <li><p>Run the tokenizer, processing resulting tokens as they are emitted, and stopping when the
   tokenizer reaches the <span>explicit "EOF" character</span> or <span data-x="spin the event
   loop">spins the event loop</span>.</p></li>
  </ol>

  </div>



  <h4><code data-x="dom-document-write">document.write()</code></h4>

  <dl class="domintro">
   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-write">write</span>(...<var>text</var>)</code></dt>

   <dd>
    <p>In general, adds the given string(s) to the <code>Document</code>'s input stream.</p>

    <p class="warning">This method has very idiosyncratic behavior. In some cases, this method can
    affect the state of the <span>HTML parser</span> while the parser is running, resulting in a DOM
    that does not correspond to the source of the document (e.g. if the string written is the string
    "<code data-x="">&lt;plaintext&gt;</code>" or "<code data-x="">&lt;!--</code>"). In other cases,
    the call can clear the current page first, as if <code data-x="dom-document-open">document.open()</code> had been called. In yet more cases, the method
    is simply ignored, or throws an exception. Users agents are <a href="#document-written-scripts-intervention">explicitly allowed to avoid executing
    <code>script</code> elements inserted via this method</a>. And to make matters even worse, the
    exact behavior of this method can in some cases be dependent on network latency<!--

     Namely, in the following case:

        <script>
         document.write('<link rel=stylesheet href=foo.css><script></script>x');
         // at this point, whether the DOM contains an "x" or not depends on how quickly the foo.css file could be processed
        </script>

    -->, which can lead to failures that are very hard to debug. <strong>For all these reasons, use
    of this method is strongly discouraged.</strong></p>

    <p>Throws an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> when
    invoked on <span>XML documents</span>.</p>

    <p>Throws an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> if the
    parser is currently executing a <span>custom element constructor</span>.</p>
   </dd>
  </dl>

  <p class="warning">This method performs no sanitization to remove potentially-dangerous elements
  and attributes like <code>script</code> or <span>event handler content attributes</span>.</p>

  <div w-nodev>

  <p><code>Document</code> objects have an <dfn>ignore-destructive-writes counter</dfn>, which is
  used in conjunction with the processing of <code>script</code> elements to prevent external
  scripts from being able to use <code data-x="dom-document-write">document.write()</code> to blow
  away the document by implicitly calling <code data-x="dom-document-open">document.open()</code>.
  Initially, the counter must be set to zero.</p>

  <p>The <dfn>document write steps</dfn>, given a <code>Document</code> object <var>document</var>,
  a list <var>text</var>, a boolean <var>lineFeed</var>, and a string <var>sink</var>, are as
  follows:</p>

  <ol>
   <li><p>Let <var>string</var> be the empty string.</p></li>

   <li><p>Let <var>isTrusted</var> be false if <var>text</var> <span data-x="list
   contains">contains</span> a string; otherwise true.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>value</var> of <var>text</var>:</p>

    <ol>
     <li><p>If <var>value</var> is a <code data-x="tt-trustedhtml">TrustedHTML</code> object, then
     append <var>value</var>'s associated <span data-x="tt-trustedhtml-data">data</span> to
     <var>string</var>.</p></li>

     <li><p>Otherwise, append <var>value</var> to <var>string</var>.</p></li>
    </ol>
   </li>

   <li><p>If <var>isTrusted</var> is false, set <var>string</var> to the result of invoking the
   <span data-x="tt-getcompliantstring">Get Trusted Type compliant string</span> algorithm with
   <code data-x="tt-trustedhtml">TrustedHTML</code>, <span>this</span>'s <span>relevant global
   object</span>, <var>string</var>, <var>sink</var>, and "<code data-x="">script</code>".</p></li>

   <li><p>If <var>lineFeed</var> is true, append U+000A LINE FEED to <var>string</var>.</p></li>

   <li><p>If <var>document</var> is an <span data-x="XML documents">XML document</span>, then throw
   an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.</p></li>
   <!-- Where would document.write() insert?
        Consider: data:text/xml,<script xmlns="http://www.w3.org/1999/xhtml"><![CDATA[ document.write('<foo>Test</foo>'); ]]></script>
   -->

   <li><p>If <var>document</var>'s <span>throw-on-dynamic-markup-insertion counter</span> is greater
   than 0, then throw an <span>"<code>InvalidStateError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>If <var>document</var>'s <span>active parser was aborted</span> is true, then
   return.</p></li>

   <li>
    <p>If the <span>insertion point</span> is undefined, then:

    <ol>
     <li><p>If <var>document</var>'s <span>unload counter</span> is greater than 0 or
     <var>document</var>'s <span>ignore-destructive-writes counter</span> is greater than 0, then
     return.</p></li>

     <li><p>Run the <span>document open steps</span> with <var>document</var>.</p></li>
    </ol>
   </li>

   <li><p>Insert <var>string</var> into the <span>input stream</span> just before the <span>insertion
   point</span>.</p></li>

   <li>
    <p>If <var>document</var>'s <span>pending parsing-blocking script</span> is null, then have the
    <span>HTML parser</span> process <var>string</var>, one code point at a time, processing
    resulting tokens as they are emitted, and stopping when the tokenizer reaches the insertion
    point or when the processing of the tokenizer is aborted by the tree construction stage (this
    can happen if a <code>script</code> end tag token is emitted by the tokenizer).

    <p class="note">If the <code data-x="dom-document-write">document.write()</code> method was
    called from script executing inline (i.e. executing because the parser parsed a set of
    <code>script</code> tags), then this is a <a href="#nestedParsing">reentrant invocation of the
    parser</a>. If the <span>parser pause flag</span> is set, the tokenizer will abort immediately
    and no HTML will be parsed, per the tokenizer's <a href="#check-parser-pause-flag">parser pause
    flag check</a>.</p>
   </li>
  </ol>

  <p>The <dfn method for="Document"><code
  data-x="dom-document-write">document.write(...<var>text</var>)</code></dfn> method steps are
  to run the <span>document write steps</span> with <span>this</span>, <var>text</var>, false, and
  "<code data-x="">Document write</code>".</p>

  </div>


  <h4><code data-x="dom-document-writeln">document.writeln()</code></h4>

  <dl class="domintro">
   <dt><code data-x=""><var>document</var>.<span subdfn data-x="dom-document-writeln">writeln</span>(...<var>text</var>)</code></dt>

   <dd>
    <p>Adds the given string(s) to the <code>Document</code>'s input stream, followed by a newline
    character. If necessary, calls the <code data-x="dom-document-open">open()</code> method
    implicitly first.</p>

    <p>Throws an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> when
    invoked on <span>XML documents</span>.</p>

    <p>Throws an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> if the
    parser is currently executing a <span>custom element constructor</span>.</p>
   </dd>
  </dl>

  <p class="warning">This method performs no sanitization to remove potentially-dangerous elements
  and attributes like <code>script</code> or <span>event handler content attributes</span>.</p>

  <div w-nodev>

  <p>The <dfn method for="Document"><code
  data-x="dom-document-writeln">document.writeln(...<var>text</var>)</code></dfn> method steps are
  to run the <span>document write steps</span> with <span>this</span>, <var>text</var>, true, and
  "<code data-x="">Document writeln</code>".</p>

  </div>


  <h3 id="dom-parsing-and-serialization">DOM parsing and serialization APIs</h3>

  <pre><code class="idl">partial interface <span id="Element-partial">Element</span> {
  [<span>CEReactions</span>] undefined <span data-x="dom-Element-setHTMLUnsafe">setHTMLUnsafe</span>((<code data-x="tt-trustedhtml">TrustedHTML</code> or DOMString) html);
  DOMString <span data-x="dom-Element-getHTML">getHTML</span>(optional <span>GetHTMLOptions</span> options = {});

  [<span>CEReactions</span>] attribute (<code data-x="tt-trustedhtml">TrustedHTML</code> or [<span>LegacyNullToEmptyString</span>] DOMString) <span data-x="dom-Element-innerHTML">innerHTML</span>;
  [<span>CEReactions</span>] attribute (<code data-x="tt-trustedhtml">TrustedHTML</code> or [<span>LegacyNullToEmptyString</span>] DOMString) <span data-x="dom-Element-outerHTML">outerHTML</span>;
  [<span>CEReactions</span>] undefined <span data-x="dom-Element-insertAdjacentHTML">insertAdjacentHTML</span>(DOMString position, (<code data-x="tt-trustedhtml">TrustedHTML</code> or DOMString) string);
};

partial interface <span id="ShadowRoot-partial">ShadowRoot</span> {
  [<span>CEReactions</span>] undefined <span data-x="dom-ShadowRoot-setHTMLUnsafe">setHTMLUnsafe</span>((<code data-x="tt-trustedhtml">TrustedHTML</code> or DOMString) html);
  DOMString <span data-x="dom-ShadowRoot-getHTML">getHTML</span>(optional <span>GetHTMLOptions</span> options = {});

  [<span>CEReactions</span>] attribute (<code data-x="tt-trustedhtml">TrustedHTML</code> or [<span>LegacyNullToEmptyString</span>] DOMString) <span data-x="dom-ShadowRoot-innerHTML">innerHTML</span>;
};

dictionary <dfn dictionary>GetHTMLOptions</dfn> {
  boolean <dfn dict-member for="GetHTMLOptions" data-x="dom-GetHTMLOptions-serializableShadowRoots">serializableShadowRoots</dfn> = false;
  sequence&lt;ShadowRoot> <dfn dict-member for="GetHTMLOptions" data-x="dom-GetHTMLOptions-shadowRoots">shadowRoots</dfn> = [];
};</code></pre>

  <h4>The <code>DOMParser</code> interface</h4>

  <p>The <code>DOMParser</code> interface allows authors to create new <code>Document</code> objects
  by parsing strings, as either HTML or XML.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>parser</var> = new <span subdfn data-x="dom-DOMParser-constructor">DOMParser</span>()</code></dt>
   <dd><p>Constructs a new <code>DOMParser</code> object.</p></dd>

   <dt><code data-x=""><var>document</var> = <var>parser</var>.<span subdfn data-x="dom-DOMParser-parseFromString">parseFromString</span>(<var>string</var>, <var>type</var>)</code></dt>
   <dd>
    <p>Parses <var>string</var> using either the HTML or XML parser, according to <var>type</var>,
    and returns the resulting <code>Document</code>. <var>type</var> can be "<code>text/html</code>"
    (which will invoke the HTML parser), or any of "<code>text/xml</code>",
    "<code>application/xml</code>", "<code>application/xhtml+xml</code>", or
    "<code>image/svg+xml</code>" (which will invoke the XML parser).</p>

    <p>For the XML parser, if <var>string</var> cannot be parsed, then the returned
    <code>Document</code> will contain elements describing the resulting error.</p>

    <p>Note that <code>script</code> elements are not evaluated during parsing, and the resulting
    document's <span data-x="document's character encoding">encoding</span> will always be
    <span>UTF-8</span>. The document's <span data-x="concept-document-url">URL</span> will be
    inherited from <var>parser</var>'s <span>relevant global object</span>.</p>

    <p>Values other than the above for <var>type</var> will cause a <code>TypeError</code> exception
    to be thrown.</p>
   </dd>
  </dl>

  <p class="note">The design of <code>DOMParser</code>, as a class that needs to be constructed and
  then have its <code data-x="dom-DOMParser-parseFromString">parseFromString()</code> method
  called, is an unfortunate historical artifact. If we were designing this functionality today it
  would be a standalone function. For parsing HTML, the modern alternative is <code
  data-x="dom-parseHTMLUnsafe">Document.parseHTMLUnsafe()</code>.</p>

  <p class="warning">This method performs no sanitization to remove potentially-dangerous elements
  and attributes like <code>script</code> or <span>event handler content attributes</span>.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>DOMParser</dfn> {
  <span data-x="dom-DOMParser-constructor">constructor</span>();

  [NewObject] <code>Document</code> <span data-x="dom-DOMParser-parseFromString">parseFromString</span>((<code data-x="tt-trustedhtml">TrustedHTML</code> or DOMString) string, <span>DOMParserSupportedType</span> type);
};

enum <dfn enum>DOMParserSupportedType</dfn> {
  "<span data-x="dom-DOMParserSupportedType-texthtml">text/html</span>",
  "<span data-x="dom-DOMParserSupportedType-otherwise">text/xml</span>",
  "<span data-x="dom-DOMParserSupportedType-otherwise">application/xml</span>",
  "<span data-x="dom-DOMParserSupportedType-otherwise">application/xhtml+xml</span>",
  "<span data-x="dom-DOMParserSupportedType-otherwise">image/svg+xml</span>"
};</code></pre>

  <div w-nodev>

  <p>The <dfn data-x="dom-DOMParser-constructor"><code>new DOMParser()</code></dfn> constructor
  steps are to do nothing.</p>

  <p>The <dfn method for="DOMParser"
  data-x="dom-DOMParser-parseFromString"><code>parseFromString(<var>string</var>,
  <var>type</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>compliantString</var> be the result of invoking the <span
   data-x="tt-getcompliantstring">Get Trusted Type compliant string</span> algorithm with <code
   data-x="tt-trustedhtml">TrustedHTML</code>, <span>this</span>'s <span>relevant global
   object</span>, <var>string</var>, "<code data-x="">DOMParser parseFromString</code>", and "<code
   data-x="">script</code>".</p></li>

   <li>
    <p>Let <var>document</var> be a new <code>Document</code>, whose <span
    data-x="concept-document-content-type">content type</span> is <var>type</var> and <span
    data-x="concept-document-URL">URL</span> is <span>this</span>'s <span>relevant global object</span>'s <span
    data-x="concept-document-window">associated <code>Document</code></span>'s <span
    data-x="concept-document-URL">URL</span>.</p>
    <!-- When https://github.com/whatwg/html/issues/4792 gets fixed we need to investigate which of
         HTMLDocument vs. XMLDocument gets returned, and when, with tests. In particular we'll want
         to check the application/xhtml+xml case as even for navigation it seems like browsers
         diverge there: https://github.com/whatwg/html/issues/5157#issuecomment-573052638. -->

    <p class="note">The document's <span data-x="document's character encoding">encoding</span> will
    be left as its default, of <span>UTF-8</span>. In particular, any XML declarations or
    <code>meta</code> elements found while parsing <var>compliantString</var> will have no effect.</p>
   </li>

   <li>
    <p>Switch on <var>type</var>:</p>

    <dl class="switch">
     <dt>"<dfn enum-value for="DOMParserSupportedType"
     data-x="dom-DOMParserSupportedType-texthtml"><code>text/html</code></dfn>"</dt>
     <dd>
      <ol>
       <li><p><span>Parse HTML from a string</span> given <var>document</var> and
       <var>compliantString</var>.</p></li>
      </ol>

      <p class="note">Since <var>document</var> does not have a <span
      data-x="concept-document-bc">browsing context</span>, <span
      data-x="concept-n-script">scripting is disabled</span>.</p>
     </dd>

     <dt><dfn data-x="dom-DOMParserSupportedType-otherwise">Otherwise</dfn></dt>
     <dd>
      <ol>
       <li><p>Create an <span>XML parser</span> <var>parser</var>, associated with
       <var>document</var>, and with <span>XML scripting support disabled</span>.</p></li>

       <li><p>Parse <var>compliantString</var> using <var>parser</var>.</p>

       <li>
        <p>If the previous step resulted in an XML well-formedness or XML namespace well-formedness
        error, then:</p>

        <ol>
         <li><p><span>Assert</span>: <var>document</var> has no child nodes.</p></li>

         <li><p>Let <var>root</var> be the result of <span data-x="create an element">creating an
         element</span> given <var>document</var>, "<code data-x="">parsererror</code>", and "<code
         data-x="">http://www.mozilla.org/newlayout/xml/parsererror.xml</code>".</p></li>

         <li><p>Optionally, add attributes or children to <var>root</var> to describe the nature of
         the parsing error.</p></li>

         <li><p><span data-x="concept-node-append">Append</span> <var>root</var> to
         <var>document</var>.</p></li>
        </ol>
       </li>
      </ol>
     </dd>
    </dl>
   </li>

   <li><p>Return <var>document</var>.</p>
  </ol>

  <p>To <dfn>parse HTML from a string</dfn>, given a <code>Document</code> <var>document</var> and a
  <span>string</span> <var>html</var>:</p>

  <ol>
   <li><p>Set <var>document</var>'s <span data-x="concept-document-type">type</span> to "<code
   data-x="">html</code>".</p></li>

   <li><p>Create an <span>HTML parser</span> <var>parser</var>, associated with
   <var>document</var>.</p></li>

   <li><p>Place <var>html</var> into the <span>input stream</span> for <var>parser</var>. The
   encoding <span data-x="concept-encoding-confidence">confidence</span> is
   <i>irrelevant</i>.</p></li>

   <li>
    <p>Start <var>parser</var> and let it run until it has consumed all the characters just
    inserted into the input stream.</p>

    <p class="note">This might mutate the document's <span
    data-x="concept-document-mode">mode</span>.</p>
   </li>
  </ol>

  </div>

  <h4>Unsafe HTML parsing methods</h4>

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn
   data-x="dom-Element-setHTMLUnsafe">setHTMLUnsafe</span>(<var>html</var>)</code></dt>

   <dd>
    <p>Parses <var>html</var> using the HTML parser, and replaces the children of <var>element</var>
    with the result. <var>element</var> provides context for the HTML parser.</p>
   </dd>

   <dt><code data-x=""><var>shadowRoot</var>.<span subdfn
   data-x="dom-ShadowRoot-setHTMLUnsafe">setHTMLUnsafe</span>(<var>html</var>)</code></dt>

   <dd>
    <p>Parses <var>html</var> using the HTML parser, and replaces the children of
    <var>shadowRoot</var> with the result. <var>shadowRoot</var>'s <span
    data-x="concept-DocumentFragment-host">host</span> provides context for the HTML parser.</p>
   </dd>

   <dt><code data-x=""><var>doc</var> = Document.<span
   data-x="dom-parseHTMLUnsafe">parseHTMLUnsafe</span>(<var>html</var>)</code></dt>

   <dd>
    <p>Parses <var>html</var> using the HTML parser, and returns the resulting
    <code>Document</code>.</p>

    <p>Note that <code>script</code> elements are not evaluated during parsing, and the resulting
    document's <span data-x="document's character encoding">encoding</span> will always be
    <span>UTF-8</span>. The document's <span data-x="concept-document-url">URL</span> will be
    <code>about:blank</code>.</p>
   </dd>
  </dl>

  <p class="warning">These methods perform no sanitization to remove potentially-dangerous elements
  and attributes like <code>script</code> or <span>event handler content attributes</span>.</p>

  <div w-nodev>

  <p><code>Element</code>'s <dfn method for="Element"><code
  data-x="dom-Element-setHTMLUnsafe">setHTMLUnsafe(<var>html</var>)</code></dfn> method steps
  are:</p>

  <ol>
   <li><p>Let <var>compliantHTML</var> be the result of invoking the <span
   data-x="tt-getcompliantstring">Get Trusted Type compliant string</span> algorithm with <code
   data-x="tt-trustedhtml">TrustedHTML</code>, <span>this</span>'s <span>relevant global
   object</span>, <var>html</var>, "<code data-x="">Element setHTMLUnsafe</code>", and "<code
   data-x="">script</code>".</p></li>

   <li><p>Let <var>target</var> be <span>this</span>'s <span>template contents</span> if
   <span>this</span> is a <code>template</code> element; otherwise <span>this</span>.</p></li>

   <li><p><span>Unsafely set HTML</span> given <var>target</var>, <span>this</span>, and
   <var>compliantHTML</var>.</p></li>
  </ol>

  <p><code>ShadowRoot</code>'s <dfn method for="ShadowRoot"><code
  data-x="dom-ShadowRoot-setHTMLUnsafe">setHTMLUnsafe(<var>html</var>)</code></dfn> method steps
  are:</p>

  <ol>
   <li><p>Let <var>compliantHTML</var> be the result of invoking the <span
   data-x="tt-getcompliantstring">Get Trusted Type compliant string</span> algorithm with <code
   data-x="tt-trustedhtml">TrustedHTML</code>, <span>this</span>'s <span>relevant global
   object</span>, <var>html</var>, "<code data-x="">ShadowRoot setHTMLUnsafe</code>", and "<code
   data-x="">script</code>".</p></li>

   <li><p><span>Unsafely set HTML</span> given <span>this</span>, <span>this</span>'s <span
   data-x="concept-DocumentFragment-host">shadow host</span>, and <var>compliantHTML</var>.</p></li>
  </ol>

  <p>To <dfn>unsafely set HTML</dfn>, given an <code>Element</code> or <code>DocumentFragment</code>
  <var>target</var>, an <code>Element</code> <var>contextElement</var>, and a <span>string</span>
  <var>html</var>:</p>

  <ol>
   <li><p>Let <var>newChildren</var> be the result of the <span>HTML fragment parsing
   algorithm</span> given <var>contextElement</var>, <var>html</var>, and true.</p></li>

   <li><p>Let <var>fragment</var> be a new <code>DocumentFragment</code> whose <span>node
   document</span> is <var>contextElement</var>'s <span>node document</span>.</p></li>

   <li><p>For each <var>node</var> in <var>newChildren</var>, <span
   data-x="concept-node-append">append</span> <var>node</var> to <var>fragment</var>.</p></li>

   <li><p><span data-x="concept-node-replace-all">Replace all</span> with <var>fragment</var> within
   <var>target</var>.</p></li>
  </ol>

  <hr>

  <p>The static <dfn method for="Document"><code
  data-x="dom-parseHTMLUnsafe">parseHTMLUnsafe(<var>html</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>compliantHTML</var> be the result of invoking the <span
   data-x="tt-getcompliantstring">Get Trusted Type compliant string</span> algorithm with <code
   data-x="tt-trustedhtml">TrustedHTML</code>, <span>this</span>'s <span>relevant global
   object</span>, <var>html</var>, "<code data-x="">Document parseHTMLUnsafe</code>", and "<code
   data-x="">script</code>".</p></li>

   <li>
    <p>Let <var>document</var> be a new <code>Document</code>, whose <span
    data-x="concept-document-content-type">content type</span> is "<code
    data-x="">text/html</code>".</p>

    <p class="note">Since <var>document</var> does not have a <span
    data-x="concept-document-bc">browsing context</span>, <span data-x="concept-n-script">scripting
    is disabled</span>.</p>
   </li>

   <li><p>Set <var>document</var>'s <span
   data-x="concept-document-allow-declarative-shadow-roots">allow declarative shadow roots</span> to
   true.</p></li>

   <li><p><span>Parse HTML from a string</span> given <var>document</var> and
   <var>compliantHTML</var>.</p></li>

   <li><p>Return <var>document</var>.</p></li>
  </ol>

  </div>

  <h4>HTML serialization methods</h4>

  <dl class="domintro">
   <dt><code data-x=""><var>html</var> = <var>element</var>.<span subdfn data-x="dom-Element-getHTML">getHTML</span>({ <span data-x="dom-GetHTMLOptions-serializableShadowRoots">serializableShadowRoots</span>, <span data-x="dom-GetHTMLOptions-shadowRoots">shadowRoots</span> })</code></dt>
   <dd>
    <p>Returns the result of serializing <var>element</var> to HTML. <span data-x="shadow root">Shadow roots</span> within <var>element</var> are serialized according to the provided options:</p>

    <ul>
     <li><p>If <code data-x="dom-GetHTMLOptions-serializableShadowRoots">serializableShadowRoots</code> is true, then all shadow roots marked as <span data-x="shadow-serializable">serializable</span> are serialized.</p></li>

     <li><p>If the <code data-x="dom-GetHTMLOptions-shadowRoots">shadowRoots</code> array is provided, then all shadow roots specified in the array are serialized, regardless of whether or not they are marked as serializable.</p></li>
    </ul>

    <p>If neither option is provided, then no shadow roots are serialized.</p>
   </dd>

   <dt><code data-x=""><var>html</var> = <var>shadowRoot</var>.<span subdfn data-x="dom-ShadowRoot-getHTML">getHTML</span>({ <span data-x="dom-GetHTMLOptions-serializableShadowRoots">serializableShadowRoots</span>, <span data-x="dom-GetHTMLOptions-shadowRoots">shadowRoots</span> })</code></dt>
   <dd>
    <p>Returns the result of serializing <var>shadowRoot</var> to HTML, using its <span data-x="concept-DocumentFragment-host">shadow host</span> as the context element. <span data-x="shadow root">Shadow roots</span> within <var>shadowRoot</var> are serialized according to the provided options, as above.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p><code>Element</code>'s <dfn method for="Element"><code
  data-x="dom-Element-getHTML">getHTML(<var>options</var>)</code></dfn> method steps
  are to return the result of <span>HTML fragment serialization algorithm</span> with
  <span>this</span>, <var>options</var>["<code
  data-x="dom-GetHTMLOptions-serializableShadowRoots">serializableShadowRoots</code>"],
  and <var>options</var>["<code data-x="dom-GetHTMLOptions-shadowRoots">shadowRoots</code>"].</p>

  <p><code>ShadowRoot</code>'s <dfn method for="ShadowRoot"><code
  data-x="dom-ShadowRoot-getHTML">getHTML(<var>options</var>)</code></dfn> method steps
  are to return the result of <span>HTML fragment serialization algorithm</span> with
  <span>this</span>, <var>options</var>["<code
  data-x="dom-GetHTMLOptions-serializableShadowRoots">serializableShadowRoots</code>"],
  and <var>options</var>["<code data-x="dom-GetHTMLOptions-shadowRoots">shadowRoots</code>"].</p>

  </div>

  <h4>The <code data-x="dom-element-innerHTML">innerHTML</code> property</h4>

  <p class="XXX">The <code data-x="dom-element-innerHTML">innerHTML</code> property has a number of outstanding issues
  in the <cite>DOM Parsing and Serialization</cite> <a href="https://github.com/w3c/DOM-Parsing/issues">issue
  tracker</a>, documenting various problems with its specification.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-Element-innerHTML">innerHTML</span></code></dt>
   <dd>
    <p>Returns a fragment of HTML or XML that represents the element's contents.</p>

    <p>In the case of an XML document, throws a <span>"<code>InvalidStateError</code>"</span>
    <code>DOMException</code> if the element cannot be serialized to XML.</p>
   </dd>
   <dt><code data-x=""><var>element</var>.<span data-x="dom-Element-innerHTML">innerHTML</span> = <var>value</var></code></dt>
   <dd>
    <p>Replaces the contents of the element with nodes parsed from the given string.</p>

    <p>In the case of an XML document, throws a <span>"<code>SyntaxError</code>"</span>
    <code>DOMException</code> if the given string is not well-formed.</p>
   </dd>

   <dt><code data-x=""><var>shadowRoot</var>.<span subdfn data-x="dom-ShadowRoot-innerHTML">innerHTML</span></code></dt>
   <dd>
    <p>Returns a fragment of HTML that represents the shadow roots's contents.</p>
   </dd>

   <dt><code data-x=""><var>shadowRoot</var>.<span data-x="dom-ShadowRoot-innerHTML">innerHTML</span> = <var>value</var></code></dt>
   <dd>
    <p>Replaces the contents of the shadow root with nodes parsed from the given string.</p>
   </dd>
  </dl>

  <p class="warning">These properties' setters perform no sanitization to remove
  potentially-dangerous elements and attributes like <code>script</code> or <span>event handler
  content attributes</span>.</p>

  <div w-nodev>

  <p>The <dfn export>fragment serializing algorithm steps</dfn>, given an <code>Element</code>,
  <code>Document</code>, or <code>DocumentFragment</code> <var>node</var> and a boolean <var>require
  well-formed</var>, are:</p>

  <ol>
    <li><p>Let <var>context document</var> be <var>node</var>'s <span>node document</span>.</p></li>

    <li><p>If <var>context document</var> is an <span data-x="HTML documents">HTML document</span>,
    return the result of <span>HTML fragment serialization algorithm</span> with <var>node</var>,
    false, and « ».</p></li>

    <li><p>Return the <span data-x="xml-serialization">XML serialization</span> of <var>node</var>
    given <var>require well-formed</var>.</p></li>
  </ol>

  <p>The <dfn export>fragment parsing algorithm steps</dfn>, given an <code>Element</code>
  <var>context</var> and a string <var>markup</var>, are:</p>

  <ol>
   <li><p>Let <var>algorithm</var> be the <span>HTML fragment parsing algorithm</span>.</p></li>

   <li><p>If <var>context</var>'s <span>node document</span> is an <span data-x="XML documents">XML
   document</span>, then set <var>algorithm</var> to the <span>XML fragment parsing
   algorithm</span>.</p></li>

   <li><p>Let <var>newChildren</var> be the result of invoking <var>algorithm</var> given
   <var>context</var> and <var>markup</var>.</p></li>

   <li><p>Let <var>fragment</var> be a new <code>DocumentFragment</code> whose <span>node
   document</span> is <var>context</var>'s <span>node document</span>.</p></li>

   <li>
    <p>For each <var>node</var> of <var>newChildren</var>, in <span>tree order</span>: <span
    data-x="concept-node-append">append</span> <var>node</var> to <var>fragment</var>.</p>

    <p class=note>This ensures the <span>node document</span> for the new <span
    data-x="node">nodes</span> is correct.</p>
   </li>

   <li><p>Return <var>fragment</var>.</p></li>
  </ol>

  <p><code>Element</code>'s <dfn attribute for="Element"><code
  data-x="dom-Element-innerHTML">innerHTML</code></dfn> getter steps are to return the result of
  running <span>fragment serializing algorithm steps</span> with <span>this</span> and true.</p>

  <p><code>ShadowRoot</code>'s <dfn attribute for="ShadowRoot"><code
  data-x="dom-ShadowRoot-innerHTML">innerHTML</code></dfn> getter steps are to return the result of
  running <span>fragment serializing algorithm steps</span> with <span>this</span> and true.</p>

  <p><code>Element</code>'s <code data-x="dom-Element-innerHTML">innerHTML</code> setter steps
  are:</p>

  <ol>
    <li><p>Let <var>compliantString</var> be the result of invoking the <span
    data-x="tt-getcompliantstring">Get Trusted Type compliant string</span> algorithm with <code
    data-x="tt-trustedhtml">TrustedHTML</code>, <span>this</span>'s <span>relevant global
    object</span>, the given value, "<code data-x="">Element innerHTML</code>", and "<code
    data-x="">script</code>".</p></li>

    <li><p>Let <var>context</var> be <span>this</span>.</p></li>

    <li><p>Let <var>fragment</var> be the result of invoking the <span>fragment parsing algorithm
    steps</span> with <var>context</var> and <var>compliantString</var>.</p></li>

    <li>
     <p>If <var>context</var> is a <code>template</code> element, then set <var>context</var> to
     the <code>template</code> element's <span>template contents</span> (a
     <code>DocumentFragment</code>).</p>

     <p class=note>Setting <code data-x="dom-Element-innerHTML">innerHTML</code> on a
     <code>template</code> element will replace all the nodes in its <span>template contents</span>
     rather than its <span data-x="concept-tree-child">children</span>.</p>
    </li>

    <li><p><span data-x="concept-node-replace-all">Replace all</span> with <var>fragment</var>
    within <var>context</var>.</p></li>
  </ol>

  <p><code>ShadowRoot</code>'s <code data-x="dom-ShadowRoot-innerHTML">innerHTML</code> setter
  steps are:</p>

  <ol>
    <li><p>Let <var>compliantString</var> be the result of invoking the <span
    data-x="tt-getcompliantstring">Get Trusted Type compliant string</span> algorithm with <code
    data-x="tt-trustedhtml">TrustedHTML</code>, <span>this</span>'s <span>relevant global
    object</span>, the given value, "<code data-x="">ShadowRoot innerHTML</code>", and "<code
    data-x="">script</code>".</p></li>

    <li><p>Let <var>context</var> be <span>this</span>'s <span
    data-x="concept-DocumentFragment-host">host</span>.</p></li>

    <li><p>Let <var>fragment</var> be the result of invoking the <span>fragment parsing algorithm
    steps</span> with <var>context</var> and <var>compliantString</var>.</p></li>

    <li><p><span data-x="concept-node-replace-all">Replace all</span> with <var>fragment</var>
    within <span>this</span>.</p></li>
  </ol>

  </div>

  <h4>The <code data-x="dom-element-outerHTML">outerHTML</code> property</h4>

  <p class="XXX">The <code data-x="dom-element-outerHTML">outerHTML</code> property has a number of outstanding issues
  in the <cite>DOM Parsing and Serialization</cite> <a href="https://github.com/w3c/DOM-Parsing/issues">issue
  tracker</a>, documenting various problems with its specification.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-Element-outerHTML">outerHTML</span></code></dt>
   <dd>
    <p>Returns a fragment of HTML or XML that represents the element and its contents.</p>

    <p>In the case of an XML document, throws a <span>"<code>InvalidStateError</code>"</span>
    <code>DOMException</code> if the element cannot be serialized to XML.</p>
   </dd>

   <dt><code data-x=""><var>element</var>.<span data-x="dom-Element-outerHTML">outerHTML</span> = <var>value</var></code></dt>
   <dd>
    <p>Replaces the element with nodes parsed from the given string.</p>

    <p>In the case of an XML document, throws a <span>"<code>SyntaxError</code>"</span>
    <code>DOMException</code> if the given string is not well-formed.</p>

    <p>Throws a <span>"<code>NoModificationAllowedError</code>"</span> <code>DOMException</code> if
    the parent of the element is a <span><code>Document</code></span>.</p>
   </dd>
  </dl>

  <p class="warning">This property's setter performs no sanitization to remove potentially-dangerous
  elements and attributes like <code>script</code> or <span>event handler content
  attributes</span>.</p>

  <div w-nodev>

  <p><code>Element</code>'s <dfn attribute for="Element"><code
  data-x="dom-Element-outerHTML">outerHTML</code></dfn> getter steps are:</p>

  <ol>
   <li><p>Let <var>element</var> be a fictional node whose only child is <span>this</span>.</p></li>
   <li><p>Return the result of running <span>fragment serializing algorithm steps</span> with
   <var>element</var> and true.
  </ol>

  <p><code>Element</code>'s <code data-x="dom-Element-outerHTML">outerHTML</code> setter steps
  are:</p>

  <ol>
    <li><p>Let <var>compliantString</var> be the result of invoking the <span
    data-x="tt-getcompliantstring">Get Trusted Type compliant string</span> algorithm with <code
    data-x="tt-trustedhtml">TrustedHTML</code>, <span>this</span>'s <span>relevant global
    object</span>, the given value, "<code data-x="">Element outerHTML</code>", and "<code
    data-x="">script</code>".</p></li>

    <li><p>Let <var>parent</var> be <span>this</span>'s <span>parent</span>.</p></li>

    <li><p>If <var>parent</var> is null, return. There would be no way to obtain a
    reference to the nodes created even if the remaining steps were run.</p></li>

    <li><p>If <var>parent</var> is a <code>Document</code>, throw a
    <span>"<code>NoModificationAllowedError</code>"</span> <code>DOMException</code>.</p></li>

    <li><p>If <var>parent</var> is a <code>DocumentFragment</code>, set <var>parent</var> to the
    result of <span data-x="create an element">creating an element</span> given <span>this</span>'s
    <span>node document</span>, "<code data-x="">body</code>", and the <span>HTML
    namespace</span>.</p></li>

    <li><p>Let <var>fragment</var> be the result of invoking the <span>fragment parsing algorithm
    steps</span> given <var>parent</var> and <var>compliantString</var>.</p></li>

    <li><p><span data-x="concept-node-replace">Replace</span> <span>this</span> with
    <var>fragment</var> within <span>this</span>'s <span>parent</span>.</p></li>
  </ol>

  </div>

  <h4>The <code data-x="dom-Element-insertAdjacentHTML">insertAdjacentHTML()</code> method</h4>

  <p class="XXX">The <code data-x="dom-Element-insertAdjacentHTML">insertAdjacentHTML()</code>
  method has a number of outstanding issues in the <cite>DOM Parsing and Serialization</cite> <a
  href="https://github.com/w3c/DOM-Parsing/issues">issue tracker</a>, documenting various problems
  with its specification.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>element</var>.<span subdfn data-x="dom-Element-insertAdjacentHTML">insertAdjacentHTML</span>(<var>position</var>, <var>string</var>)</code></dt>
   <dd>
    <p>Parses <var>string</var> as HTML or XML and inserts the resulting nodes into the tree in
    the position given by the <var>position</var> argument, as follows:</p>

    <dl>
     <dt>"<code data-x="">beforebegin</code>"</dt>
     <dd>Before the element itself (i.e., after <var>element</var>'s previous sibling)</dd>

     <dt>"<code data-x="">afterbegin</code>"</dt>
     <dd>Just inside the element, before its first child.</dd>

     <dt>"<code data-x="">beforeend</code>"</dt>
     <dd>Just inside the element, after its last child.</dd>

     <dt>"<code data-x="">afterend</code>"</dt>
     <dd>After the element itself (i.e., before <var>element</var>'s next sibling)</dd>
    </dl>

    <p>Throws a <span>"<code>SyntaxError</code>"</span> <code>DOMException</code> if the arguments
    have invalid values (e.g., in the case of an <span data-x="XML documents">XML document</span>,
    if the given string is not well-formed).</p>

    <p>Throws a <span>"<code>NoModificationAllowedError</code>"</span> <code>DOMException</code>
    if the given position isn't possible (e.g. inserting elements after the root element of a
    <code>Document</code>).</p>
   </dd>
  </dl>

  <p class="warning">This method performs no sanitization to remove potentially-dangerous elements
  and attributes like <code>script</code> or <span>event handler content attributes</span>.</p>

  <div w-nodev>

  <p><code>Element</code>'s <dfn method for="Element"><code
  data-x="dom-Element-insertAdjacentHTML">insertAdjacentHTML(<var>position</var>,
  <var>string</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>compliantString</var> be the result of invoking the <span
   data-x="tt-getcompliantstring">Get Trusted Type compliant string</span> algorithm with <code
   data-x="tt-trustedhtml">TrustedHTML</code>, <span>this</span>'s <span>relevant global
   object</span>, <var>string</var>, "<code data-x="">Element insertAdjacentHTML</code>", and "<code
   data-x="">script</code>".</p></li>

   <li><p>Let <var>context</var> be null.</p></li>

   <li><p>Use the first matching item from this list:</p>
    <dl class=switch>
     <dt>If <var>position</var> is an <span>ASCII case-insensitive</span> match for the string "<code data-x="">beforebegin</code>"</dt>
     <dt>If <var>position</var> is an <span>ASCII case-insensitive</span> match for the string "<code data-x="">afterend</code>"</dt>
     <dd>
      <ol>
       <li><p>Set <var>context</var> to <span>this</span>'s <span
       data-x="dom-parent">parent</span>.</p></li>

       <li><p>If <var>context</var> is null or a <code>Document</code>, throw a
       <span>"<code>NoModificationAllowedError</code>"</span> <code>DOMException</code>.</p></li>
      </ol>
     </dd>

     <dt>If <var>position</var> is an <span>ASCII case-insensitive</span> match for the string "<code data-x="">afterbegin</code>"</dt>
     <dt>If <var>position</var> is an <span>ASCII case-insensitive</span> match for the string "<code data-x="">beforeend</code>"</dt>
     <dd>Set <var>context</var> to <span>this</span>.</dd>

     <dt>Otherwise</dt>
     <dd><p>Throw a <span>"<code>SyntaxError</code>"</span> <code>DOMException</code>.</p></dd>
    </dl>
   </li>

   <li>
    <p>If <var>context</var> is not an <code>Element</code> or all of the following are true:</p>

    <ul>
     <li><p><var>context</var>'s <span>node document</span> is an HTML document;</p></li>
     <li><p><var>context</var>'s <span data-x="concept-element-local-name">local name</span> is
     "<code>html</code>"; and</p></li>
     <li><p><var>context</var>'s <span data-x="concept-element-namespace">namespace</span> is the
     <span>HTML namespace</span>,</p></li>
    </ul>

    <p>set <var>context</var> to the result of <span data-x="create an element">creating an
    element</span> given <span>this</span>'s <span>node document</span>, "<code
    data-x="">body</code>", and the <span>HTML namespace</span>.</p>
   </li>

   <li>
    <p>Let <var>fragment</var> be the result of invoking the <span>fragment parsing algorithm
    steps</span> with <var>context</var> and <var>compliantString</var>.</p>
   </li>

   <li>Use the first matching item from this list:
    <dl class=switch>
     <dt>If <var>position</var> is an <span>ASCII case-insensitive</span> match for the string "<code data-x="">beforebegin</code>"</dt>
     <dd>
      <p><span data-x="concept-node-insert">Insert</span> <var>fragment</var> into
      <span>this</span>'s <span data-x="dom-parent">parent</span> before <span>this</span>.</p>
     </dd>

     <dt>If <var>position</var> is an <span>ASCII case-insensitive</span> match for the string "<code data-x="">afterbegin</code>"</dt>
     <dd>
      <p><span data-x="concept-node-insert">Insert</span> <var>fragment</var> into
      <span>this</span> before its <span>first child</span>.</p>
     </dd>

     <dt>If <var>position</var> is an <span>ASCII case-insensitive</span> match for the string "<code data-x="">beforeend</code>"</dt>
     <dd>
      <p><span data-x="concept-node-append">Append</span> <var>fragment</var> to
      <span>this</span>.</p>
     </dd>

     <dt>If <var>position</var> is an <span>ASCII case-insensitive</span> match for the string "<code data-x="">afterend</code>"</dt>
     <dd>
      <p><span data-x="concept-node-insert">Insert</span> <var>fragment</var> into
      <span>this</span>'s <span data-x="dom-parent">parent</span> before <span>this</span>'s
      <span>next sibling</span>.</p>
     </dd>
    </dl>
   </li>
  </ol>

  <p class=note>As with other direct <code>Node</code>-manipulation APIs (and unlike <code
  data-x="dom-Element-innerHTML">innerHTML</code>), <code
  data-x="dom-Element-insertAdjacentHTML">insertAdjacentHTML()</code> does not include any special
  handling for <code>template</code> elements. In most cases you will want to use <code
  data-x="">templateEl.<span data-x="dom-template-content">content</span>.<span
  data-x="dom-Element-insertAdjacentHTML">insertAdjacentHTML()</span></code> instead of directly
  manipulating the child nodes of a <code>template</code> element.</p>

  </div>

  <h4>The <code data-x="dom-Range-createContextualFragment">createContextualFragment()</code>
  method</h4>

  <p class="XXX">The <code
  data-x="dom-Range-createContextualFragment">createContextualFragment()</code> method has a number
  of outstanding issues in the <cite>DOM Parsing and Serialization</cite> <a
  href="https://github.com/w3c/DOM-Parsing/issues">issue tracker</a>, documenting various problems
  with its specification.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>docFragment</var> = <var>range</var>.<span subdfn data-x="dom-Range-createContextualFragment">createContextualFragment</span>(<var>string</var>)</code></dt>
   <dd>
    <p>Returns a <code>DocumentFragment</code> created from the markup string <var>string</var> using
    <var>range</var>'s <span data-x="concept-range-start-node">start node</span> as the context in
    which <var>fragment</var> is parsed.</p>
   </dd>
  </dl>

  <p class="warning">This method performs no sanitization to remove potentially-dangerous elements
  and attributes like <code>script</code> or <span>event handler content attributes</span>.</p>

  <pre><code class="idl">partial interface <span id="Range-partial">Range</span> {
  [<span>CEReactions</span>, NewObject] <code>DocumentFragment</code> <span data-x="dom-Range-createContextualFragment">createContextualFragment</span>((<code data-x="tt-trustedhtml">TrustedHTML</code> or DOMString) string);
};</code></pre>

  <div w-nodev>

  <p><code>Range</code>'s <dfn method for="Range"><code
  data-x="dom-Range-createContextualFragment">createContextualFragment(<var>string</var>)</code></dfn>
  method steps are:</p>

  <ol>
   <li><p>Let <var>compliantString</var> be the result of invoking the <span data-x="tt-getcompliantstring">Get
   Trusted Type compliant string</span> algorithm with <code
   data-x="tt-trustedhtml">TrustedHTML</code>, <span>this</span>'s <span>relevant global
   object</span>, <var>string</var>, "<code data-x="">Range createContextualFragment</code>", and
   "<code data-x="">script</code>".</p></li>

   <li><p>Let <var>node</var> be <span>this</span>'s <span data-x="concept-range-start-node">start
   node</span>.</p></li>

   <li><p>Let <var>element</var> be null.</p></li>

   <li><p>If <var>node</var> <span>implements</span> <code>Element</code>, set <var>element</var>
   to <var>node</var>.</p></li>

   <li><p>Otherwise, if <var>node</var> <span>implements</span> <code>Text</code> or
   <code>Comment</code>, set <var>element</var> to <var>node</var>'s <span>parent
   element</span>.</p></li>

   <li>
    <p>If <var>element</var> is null or all of the following are true:</p>

    <ul>
     <li><p><var>element</var>'s <span>node document</span> is an HTML document;</p></li>

     <li><p><var>element</var>'s <span data-x="concept-element-local-name">local name</span> is
     "<code>html</code>"; and</p></li>

     <li><p><var>element</var>'s <span data-x="concept-element-namespace">namespace</span> is the
     <span>HTML namespace</span>,</p></li>
    </ul>

    <p>then set <var>element</var> to the result of <span data-x="create an element">creating an
    element</span> given <span>this</span>'s <span>node document</span>, "<code
    data-x="">body</code>", and the <span>HTML namespace</span>.</p>
   </li>

   <li><p>Let <var>fragment node</var> be the result of invoking the <span>fragment parsing
   algorithm steps</span> with <var>element</var> and <var>compliantString</var>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>script</var> of <var>fragment node</var>'s
    <code data-x="script">script</code> element <span
    data-x="concept-tree-descendant">descendants</span>:</p>

    <ol>
     <li><p>Set <var>script</var>'s <span>already started</span> to false.</p></li>

     <li><p>Set <var>script</var>'s <span>parser document</span> to null.</p></li>
    </ol>
   </li>

   <li><p>Return <var>fragment node</var>.</p></li>
  </ol>

  </div>

  <h4>The <code>XMLSerializer</code> interface</h4>

  <p class="XXX">The <code>XMLSerializer</code> interface has a number of outstanding issues in the
  <cite>DOM Parsing and Serialization</cite> <a
  href="https://github.com/w3c/DOM-Parsing/issues">issue tracker</a>, documenting various problems
  with its specification. The remainder of <cite>DOM Parsing and Serialization</cite> will be
  gradually upstreamed to this specification.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>xmlSerializer</var> = new <span subdfn data-x="dom-XMLSerializer-constructor">XMLSerializer</span>()</code></dt>
   <dd><p>Constructs a new <code>XMLSerializer</code> object.</p></dd>

   <dt><code data-x=""><var>string</var> = <var>xmlSerializer</var>.<span subdfn data-x="dom-XMLSerializer-serializeToString">serializeToString</span>(<var>root</var>)</code></dt>
   <dd>
    <p>Returns the result of serializing <var>root</var> to XML.</p>

    <p>Throws an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code> if
    <var>root</var> cannot be serialized to XML.</p>
   </dd>
  </dl>

  <p class="note">The design of <code>XMLSerializer</code>, as a class that needs to be constructed
  and then have its <code data-x="dom-XMLSerializer-serializeToString">serializeToString()</code>
  method called, is an unfortunate historical artifact. If we were designing this functionality
  today it would be a standalone function.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>XMLSerializer</dfn> {
  <span data-x="dom-XMLSerializer-constructor">constructor</span>();

  DOMString <span data-x="dom-XMLSerializer-serializeToString">serializeToString</span>(<code>Node</code> root);
};</code></pre>

  <div w-nodev>

  <p>The <dfn data-x="dom-XMLSerializer-constructor"><code>new XMLSerializer()</code></dfn>
  constructor steps are to do nothing.</p>

  <p>The <dfn method for="XMLSerializer"
  data-x="dom-XMLSerializer-serializeToString"><code>serializeToString(<var>root</var>)</code></dfn>
  method steps are:</p>

  <ol>
   <li><p>Return the <span data-x="xml-serialization">XML serialization</span> of <var>root</var>
   given false.</p></li>
  </ol>

  </div>

  <h3 split-filename="timers-and-user-prompts" id="timers">Timers</h3>

  <p>The <code data-x="dom-setTimeout">setTimeout()</code> and <code
  data-x="dom-setInterval">setInterval()</code> methods allow authors to schedule timer-based
  callbacks.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>id</var> = self.<span subdfn data-x="dom-setTimeout">setTimeout</span>(<var>handler</var> [, <var>timeout</var> [, ...<var>arguments</var> ] ])</code></dt>

   <dd>
    <p>Schedules a timeout to run <var>handler</var> after <var>timeout</var> milliseconds. Any
    <var>arguments</var> are passed straight through to the <var>handler</var>.</p>
   </dd>

   <dt><code data-x=""><var>id</var> = self.<span data-x="dom-setTimeout">setTimeout</span>(<var>code</var> [, <var>timeout</var> ])</code></dt>

   <dd>
    <p>Schedules a timeout to compile and run <var>code</var> after <var>timeout</var>
    milliseconds.</p>
   </dd>

   <dt><code data-x="">self.<span subdfn data-x="dom-clearTimeout">clearTimeout</span>(<var>id</var>)</code></dt>
   <!-- don't mention that the id is optional, since if omitted, the method does nothing -->

   <dd><p>Cancels the timeout set with <code data-x="dom-setTimeout">setTimeout()</code> or <code
   data-x="dom-setInterval">setInterval()</code> identified by <var>id</var>.</p></dd>

   <dt><code data-x=""><var>id</var> = self.<span subdfn data-x="dom-setInterval">setInterval</span>(<var>handler</var> [, <var>timeout</var> [, ...<var>arguments</var> ] ])</code></dt>

   <dd>
    <p>Schedules a timeout to run <var>handler</var> every <var>timeout</var> milliseconds. Any
    <var>arguments</var> are passed straight through to the <var>handler</var>.</p>
   </dd>

   <dt><code data-x=""><var>id</var> = self.<span data-x="dom-setInterval">setInterval</span>(<var>code</var> [, <var>timeout</var> ])</code></dt>

   <dd>
    <p>Schedules a timeout to compile and run <var>code</var> every <var>timeout</var>
    milliseconds.</p>
   </dd>

   <dt><code data-x="">self.<span subdfn data-x="dom-clearInterval">clearInterval</span>(<var>id</var>)</code></dt>
   <!-- don't mention that the id is optional, since if omitted, the method does nothing -->

   <dd><p>Cancels the timeout set with <code data-x="dom-setInterval">setInterval()</code> or <code
   data-x="dom-setTimeout">setTimeout()</code> identified by <var>id</var>.</p></dd>
  </dl>

  <p class="note">Timers can be nested; after five such nested timers, however, the interval is
  forced to be at least four milliseconds.</p>

  <p class="note">This API does not guarantee that timers will run exactly on schedule. Delays due
  to CPU load, other tasks, etc, are to be expected.</p>

  <div w-nodev>

  <p>Objects that implement the <code>WindowOrWorkerGlobalScope</code> mixin have a
  <dfn export for="WindowOrWorkerGlobalScope">map of setTimeout and setInterval IDs</dfn>, which is
  an <span>ordered map</span>, initially empty. Each <span data-x="map key">key</span> in this map
  is a positive integer, corresponding to the return value of a <code data-x="dom-setTimeout"
  >setTimeout()</code> or <code data-x="dom-setInterval">setInterval()</code> call.
  Each <span data-x="map value">value</span> is a <span>unique internal value</span>, corresponding
  to a key in the object's <span>map of active timers</span>.</p>

  <hr>

  <p>The <dfn method for="WindowOrWorkerGlobalScope" data-x="dom-setTimeout"><code
  id="dom-windowtimers-setTimeout">setTimeout(<var>handler</var>, <var>timeout</var>,
  ...<var>arguments</var>)</code></dfn> method steps are to return the result of running the
  <span>timer initialization steps</span> given <span>this</span>, <var>handler</var>,
  <var>timeout</var>, <var>arguments</var>, and false.</p>

  <p>The <dfn method for="WindowOrWorkerGlobalScope" data-x="dom-setInterval"><code
  id="dom-windowtimers-setInterval">setInterval(<var>handler</var>, <var>timeout</var>,
  ...<var>arguments</var>)</code></dfn> method steps are to return the result of running the
  <span>timer initialization steps</span> given <span>this</span>, <var>handler</var>,
  <var>timeout</var>, <var>arguments</var>, and true.</p>

  <p>The <dfn method for="WindowOrWorkerGlobalScope" data-x="dom-clearTimeout"><code
  id="dom-windowtimers-clearTimeout">clearTimeout(<var>id</var>)</code></dfn> and <dfn method
  for="WindowOrWorkerGlobalScope" data-x="dom-clearInterval"><code
  id="dom-windowtimers-clearInterval">clearInterval(<var>id</var>)</code></dfn> method steps
  are to <span data-x="map remove">remove</span> <span>this</span>'s <span>map of setTimeout and
  setInterval IDs</span>[<var>id</var>].</p>

  <p class="note">Because <code data-x="dom-clearTimeout">clearTimeout()</code> and <code
  data-x="dom-clearInterval">clearInterval()</code> clear entries from the same map, either method
  can be used to clear timers created by <code data-x="dom-setTimeout">setTimeout()</code> or <code
  data-x="dom-setInterval">setInterval()</code>.</p>

  <hr>

  <p>To perform the <!--en-GB--><dfn id="timer-initialisation-steps">timer initialization
  steps</dfn>, given a <code>WindowOrWorkerGlobalScope</code> <var>global</var>, a string or <code
  data-x="idl-Function">Function</code> or <code data-x="tt-trustedscript">TrustedScript</code>
  <var>handler</var>, a number <var>timeout</var>, a list <var>arguments</var>, a boolean
  <var>repeat</var>, and optionally (and only if <var>repeat</var> is true) a number
  <var>previousId</var>, perform the following steps. They return a number.</p>

  <ol>
   <li><p>Let <var>thisArg</var> be <var>global</var> if that is a <code>WorkerGlobalScope</code>
   object; otherwise let <var>thisArg</var> be the <code>WindowProxy</code> that corresponds to
   <var>global</var>.</p></li>

   <li><p>If <var>previousId</var> was given, let <var>id</var> be <var>previousId</var>;
   otherwise, let <var>id</var> be an <span>implementation-defined</span> integer that is greater
   than zero and does not already <span data-x="map exists">exist</span> in <var>global</var>'s
   <span>map of setTimeout and setInterval IDs</span>.</p></li>

   <li>
    <p>If the <span>surrounding agent</span>'s <span data-x="concept-agent-event-loop">event
    loop</span>'s <span>currently running task</span> is a task that was created by this algorithm,
    then let <var>nesting level</var> be the <span data-x="concept-task">task</span>'s <span>timer
    nesting level</span>. Otherwise, let <var>nesting level</var> be zero.</p>

    <p class="note">The task's <span>timer nesting level</span> is used both for nested calls to
    <code data-x="dom-setTimeout">setTimeout()</code>, and for the repeating timers created by
    <code data-x="dom-setInterval">setInterval()</code>. (Or, indeed, for any combination of the
    two.) In other words, it represents nested invocations of this algorithm, not of a particular
    method.</p>
   </li>

   <li><p>If <var>timeout</var> is less than 0, then set <var>timeout</var> to 0.</p></li>

   <li><p>If <var>nesting level</var> is greater than 5, and <var>timeout</var> is less than 4,
   then set <var>timeout</var> to 4.</p></li>

   <li><p>Let <var>realm</var> be <var>global</var>'s <span data-x="concept-relevant-realm">relevant
   realm</span>.</p></li>

   <li><p>Let <var>initiating script</var> be the <span>active script</span>.</p></li>

   <li><p>Let <var>uniqueHandle</var> be null.</p></li>

   <li>
    <p>Let <var>task</var> be a <span data-x="concept-task">task</span> that runs the following
    substeps:</p>

    <ol>
     <li><p><span>Assert</span>: <var>uniqueHandle</var> is a <span>unique internal value</span>,
     not null.</p></li>

     <li><p>If <var>id</var> does not <span data-x="map exists">exist</span> in <var>global</var>'s
     <span>map of setTimeout and setInterval IDs</span>, then abort these steps.</p></li>

     <li>
      <p>If <var>global</var>'s <span>map of setTimeout and setInterval IDs</span>[<var>id</var>]
      does not equal <var>uniqueHandle</var>, then abort these steps.</p>

      <p class="note">This accommodates for the ID having been cleared by a <code
      data-x="dom-clearTimeout">clearTimeout()</code> or <code
      data-x="dom-clearInterval">clearInterval()</code> call, and being reused by a subsequent
      <code data-x="dom-setTimeout">setTimeout()</code> or <code
      data-x="dom-setInterval">setInterval()</code> call.</p>
     </li>

     <li><p><span>Record timing info for timer handler</span> given <var>handler</var>,
     <var>global</var>'s <span>relevant settings object</span>, and <var>repeat</var>.</p></li>

     <li><p>If <var>handler</var> is a <code data-x="idl-Function">Function</code>, then <span
     data-x="es-invoking-callback-functions">invoke</span> <var>handler</var> given
     <var>arguments</var> and "<code data-x="">report</code>", and with <i
     data-x="dfn-callback-this-value">callback this value</i> set to <var>thisArg</var>.</p>
     </li>

     <li>
      <p>Otherwise:</p>

      <ol>
       <li>
        <p>If <var>previousId</var> was not given:</p>

        <ol>
         <li><p>Let <var>globalName</var> be "<code data-x="">Window</code>" if <var>global</var> is
         a <code>Window</code> object; "<code data-x="">WorkerGlobalScope</code>"
         otherwise.</p></li>

         <li><p>Let <var>methodName</var> be "<code data-x="">setInterval</code>" if
         <var>repeat</var> is true; "<code data-x="">setTimeout</code>" otherwise.</p></li>

         <li><p>Let <var>sink</var> be a concatenation of <var>globalName</var>, U+0020 SPACE, and
         <var>methodName</var>.</p></li>

         <li><p>Set <var>handler</var> to the result of invoking the <span
         data-x="tt-getcompliantstring">Get Trusted Type compliant string</span> algorithm with
         <code data-x="tt-trustedscript">TrustedScript</code>, <var>global</var>,
         <var>handler</var>, <var>sink</var>, and "<code data-x="">script</code>".</p></li>
        </ol>
       </li>

       <li><p><span>Assert</span>: <var>handler</var> is a string.</p></li>

       <li><p>Perform <span
       data-x="csp-EnsureCSPDoesNotBlockStringCompilation">EnsureCSPDoesNotBlockStringCompilation</span>(<var>realm</var>,
       « », <var>handler</var>, <var>handler</var>, timer, « », <var>handler</var>). If this throws
       an exception, catch it, <span data-x="report an exception">report</span> it for
       <var>global</var>, and abort these steps.</p></li>
       <!-- timer is an enum value. -->

       <li><p>Let <var>settings object</var> be <var>global</var>'s <span>relevant settings
       object</span>.</p></li>

       <li><p>Let <var>fetch options</var> be the <span>default script fetch
       options</span>.</p></li>

       <li><p>Let <var>base URL</var> be <var>settings object</var>'s <span>API base
       URL</span>.</p></li>

       <li>
        <p>If <var>initiating script</var> is not null, then:</p>

        <ol>
         <li><p>Set <var>fetch options</var> to a <span>script fetch options</span> whose <span
         data-x="concept-script-fetch-options-nonce">cryptographic nonce</span> is <var>initiating
         script</var>'s <span data-x="concept-script-script-fetch-options">fetch options</span>'s
         <span data-x="concept-script-fetch-options-nonce">cryptographic nonce</span>, <span
         data-x="concept-script-fetch-options-integrity">integrity metadata</span> is the empty
         string, <span data-x="concept-script-fetch-options-parser">parser metadata</span> is
         "<code data-x="">not-parser-inserted</code>", <span
         data-x="concept-script-fetch-options-credentials">credentials mode</span> is
         <var>initiating script</var>'s <span data-x="concept-script-script-fetch-options">fetch
         options</span>'s <span data-x="concept-script-fetch-options-credentials">credentials
         mode</span>, <span data-x="concept-script-fetch-options-referrer-policy">referrer
         policy</span> is <var>initiating script</var>'s <span
         data-x="concept-script-script-fetch-options">fetch options</span>'s <span
         data-x="concept-script-fetch-options-referrer-policy">referrer policy</span>, and <span
         data-x="concept-script-fetch-options-fetch-priority">fetch priority</span> is "<code
         data-x="">auto</code>".</p></li>

         <li><p>Set <var>base URL</var> to <var>initiating script</var>'s <span
         data-x="concept-script-base-url">base URL</span>.</p></li>
        </ol>

        <p class="note">The effect of these steps ensures that the string compilation done by <code
        data-x="dom-setTimeout">setTimeout()</code> and <code
        data-x="dom-setInterval">setInterval()</code> behaves equivalently to that done by
        <code>eval()</code>. That is, <span>module script</span> fetches via <code>import()</code>
        will behave the same in both contexts.</p>
       </li>

       <li><p>Let <var>script</var> be the result of <span>creating a classic script</span> given
       <var>handler</var>, <var>settings object</var>, <var>base URL</var>, and <var>fetch
       options</var>.</p></li>

       <li><p><span data-x="run a classic script">Run the classic script</span>
       <var>script</var>.</p></li>
      </ol>
     </li>

     <li><p>If <var>id</var> does not <span data-x="map exists">exist</span> in <var>global</var>'s
     <span>map of setTimeout and setInterval IDs</span>, then abort these steps.</p></li>

     <li>
      <p>If <var>global</var>'s <span>map of setTimeout and setInterval IDs</span>[<var>id</var>]
      does not equal <var>uniqueHandle</var>, then abort these steps.</p>

      <p class="note">The ID might have been removed via the author code in <var>handler</var>
      calling <code data-x="dom-clearTimeout">clearTimeout()</code> or <code
      data-x="dom-clearInterval">clearInterval()</code>. Checking that uniqueHandle isn't different
      accounts for the possibility of the ID, after having been cleared, being reused by a
      subsequent <code data-x="dom-setTimeout">setTimeout()</code> or <code
      data-x="dom-setInterval">setInterval()</code> call.</p>
     </li>

     <li><p>If <var>repeat</var> is true, then perform the <span>timer initialization
     steps</span> again, given <var>global</var>, <var>handler</var>, <var>timeout</var>,
     <var>arguments</var>, true, and <var>id</var>.</p></li>

     <li><p>Otherwise, <span data-x="map remove">remove</span> <var>global</var>'s <span>map of
     setTimeout and setInterval IDs</span>[<var>id</var>].</p></li>
    </ol>
   </li>

   <li><p>Increment <var>nesting level</var> by one.</p></li>

   <li><p>Set <var>task</var>'s <dfn>timer nesting level</dfn> to <var>nesting level</var>.</p></li>

   <li><p>Let <var>completionStep</var> be an algorithm step which <span data-x="queue a global
   task">queues a global task</span> on the <dfn export>timer task source</dfn> given
   <var>global</var> to run <var>task</var>.</p></li>

   <li><p>Set <var>uniqueHandle</var> to the result of <span data-x="run steps after a
   timeout">running steps after a timeout</span> given <var>global</var>, "<code
   data-x="">setTimeout/setInterval</code>", <var>timeout</var>,
   <var>completionStep</var>.</p></li>

   <li><p><span data-x="map set">Set</span> <var>global</var>'s <span>map of setTimeout and
   setInterval IDs</span>[<var>id</var>] to <var>uniqueHandle</var>.</p></li>

   <li><p>Return <var>id</var>.</p></li>
  </ol>

  <p class="note">Argument conversion as defined by Web IDL (for example, invoking <code
  data-x="">toString()</code> methods on objects passed as the first argument) happens in the
  algorithms defined in Web IDL, before this algorithm is invoked.</p>

  <div class="example">
   <p>So for example, the following rather silly code will result in the log containing "<code
   data-x="">ONE&nbsp;TWO&nbsp;</code>":</p>

   <pre><code class="js">var log = '';
function logger(s) { log += s + ' '; }

setTimeout({ toString: function () {
  setTimeout("logger('ONE')", 100);
  return "logger('TWO')";
} }, 100);</code></pre>
  </div>

  </div>

  <div class="example">
   <p>To run tasks of several milliseconds back to back without any delay, while still yielding back
   to the browser to avoid starving the user interface (and to avoid the browser killing the script
   for hogging the CPU), simply queue the next timer before performing work:</p>

   <pre><code class="js">function doExpensiveWork() {
  var done = false;
  // ...
  // this part of the function takes up to five milliseconds
  // set done to true if we're done
  // ...
  return done;
}

function rescheduleWork() {
  var id = setTimeout(rescheduleWork, 0); // preschedule next iteration
  if (doExpensiveWork())
    clearTimeout(id); // clear the timeout if we don't need it
}

function scheduleWork() {
  setTimeout(rescheduleWork, 0);
}

scheduleWork(); // queues a task to do lots of work</code></pre>
  </div>

  <div w-nodev>

  <p>Objects that implement the <code>WindowOrWorkerGlobalScope</code> mixin have a <dfn export
  for="WindowOrWorkerGlobalScope">map of active timers</dfn>, which is an <span>ordered map</span>,
  initially empty. Each <span data-x="map key">key</span> in this map is a <span>unique internal
  value</span> that represents a timer, and each <span data-x="map value">value</span> is a
  <code>DOMHighResTimeStamp</code>, representing the expiry time for that timer.</p>

  <p>To <dfn export>run steps after a timeout</dfn>, given a <code>WindowOrWorkerGlobalScope</code>
  <var>global</var>, a string <var>orderingIdentifier</var>, a number <var>milliseconds</var>, and a
  set of steps <var>completionSteps</var>, perform the following steps. They return a <span>unique
  internal value</span>.</p>

  <ol>

   <li><p>Let <var>timerKey</var> be a new <span>unique internal value</span>.</p></li>

   <li><p>Let <var>startTime</var> be the <span>current high resolution time</span> given
   <var>global</var>.</p></li>

   <li><p><span data-x="map set">Set</span> <var>global</var>'s <span>map of active
   timers</span>[<var>timerKey</var>] to <var>startTime</var> plus
   <var>milliseconds</var>.</p></li>

   <li>
    <p>Run the following steps <span>in parallel</span>:</p>

    <ol>
     <li>
      <p>If <var>global</var> is a <code>Window</code> object, wait until <var>global</var>'s <span
      data-x="concept-document-window">associated <code>Document</code></span> has been <span>fully
      active</span> for a further <var>milliseconds</var> milliseconds (not necessarily
      consecutively).</p>

      <p>Otherwise, <var>global</var> is a <code>WorkerGlobalScope</code> object; wait until
      <var>milliseconds</var> milliseconds have passed with the worker not suspended (not
      necessarily consecutively).</p>
     </li>

     <li><p>Wait until any invocations of this algorithm that had the same <var>global</var> and
     <var>orderingIdentifier</var>, that started before this one, and whose <var>milliseconds</var>
     is less than or equal to this one's, have completed.</p></li>

     <li>
      <p>Optionally, wait a further <span>implementation-defined</span> length of time.</p>

      <p class="note">This is intended to allow user agents to pad timeouts as needed to optimize
      the power usage of the device. For example, some processors have a low-power mode where the
      granularity of timers is reduced; on such platforms, user agents can slow timers down to fit
      this schedule instead of requiring the processor to use the more accurate mode with its
      associated higher power usage.</p>
     </li>

     <li><p>Perform <var>completionSteps</var>.</p></li>

     <li><p><span data-x="map remove">Remove</span> <var>global</var>'s
     <span>map of active timers</span>[<var>timerKey</var>].</p></li>
    </ol>
   </li>

   <li><p>Return <var>timerKey</var>.</p></li>
  </ol>

  <p class="note"><span>Run steps after a timeout</span> is meant to be used by other
  specifications that want to execute developer-supplied code after a developer-supplied timeout,
  in a similar manner to <code data-x="dom-setTimeout">setTimeout()</code>. (Note, however, it does
  not have the nesting and clamping behavior of <code data-x="dom-setTimeout">setTimeout()</code>.)
  Such specifications can choose an <var>orderingIdentifier</var> to ensure ordering within their
  specification's timeouts, while not constraining ordering with respect to other specification's
  timeouts.</p>

  </div>

  <h3>Microtask queuing</h3>

  <dl class="domintro">
   <dt><code data-x=""><var>self</var>.<span subdfn data-x="dom-queueMicrotask">queueMicrotask</span>(<var>callback</var>)</code></dt>
   <dd><p><span data-x="queue a microtask">Queues</span> a <span>microtask</span> to run the given
   <var>callback</var>.</p></dd>
  </dl>

  <p w-nodev>The <dfn method for="WindowOrWorkerGlobalScope"><code
  data-x="dom-queueMicrotask">queueMicrotask(<var>callback</var>)</code></dfn> method must
  <span>queue a microtask</span> to <span data-x="es-invoking-callback-functions">invoke</span>
  <var>callback</var> with « » and "<code data-x="">report</code>".</p>

  <p>The <code data-x="dom-queueMicrotask">queueMicrotask()</code> method allows authors to schedule
  a callback on the <span>microtask queue</span>. This allows their code to run once the
  <span>JavaScript execution context stack</span> is next empty, which happens once all currently
  executing synchronous JavaScript has run to completion. This doesn't yield control back to the
  <span>event loop</span>, as would be the case when using, for example, <code
  data-x="dom-setTimeout">setTimeout(<var>f</var>,&nbsp;0)</code>.</p>

  <p>Authors ought to be aware that scheduling a lot of microtasks has the same performance
  downsides as running a lot of synchronous code. Both will prevent the browser from doing its own
  work, such as rendering. In many cases, <code
  data-x="dom-AnimationFrameProvider-requestAnimationFrame">requestAnimationFrame()</code> or
  <code>requestIdleCallback()</code> is a better choice. In particular, if the goal is to run code
  before the next rendering cycle, that is the purpose of <code
  data-x="dom-AnimationFrameProvider-requestAnimationFrame">requestAnimationFrame()</code>.</p>

  <p>As can be seen from the following examples, the best way of thinking about <code
  data-x="dom-queueMicrotask">queueMicrotask()</code> is as a mechanism for rearranging synchronous
  code, effectively placing the queued code immediately after the currently executing synchronous
  JavaScript has run to completion.</p>

  <div class="example">
   <p>The most common reason for using <code data-x="dom-queueMicrotask">queueMicrotask()</code> is
   to create consistent ordering, even in the cases where information is available synchronously,
   without introducing undue delay.</p>

   <p>For example, consider a custom element firing a <code data-x="">load</code> event, that also
   maintains an internal cache of previously-loaded data. A na&iuml;ve implementation might look
   like:</p>

   <pre><code class="js">MyElement.prototype.loadData = function (url) {
  if (this._cache[url]) {
    this._setData(this._cache[url]);
    this.dispatchEvent(new Event("load"));
  } else {
    fetch(url).then(res => res.arrayBuffer()).then(data => {
      this._cache[url] = data;
      this._setData(data);
      this.dispatchEvent(new Event("load"));
    });
  }
};</code></pre>

   <p>This na&iuml;ve implementation is problematic, however, in that it causes its users to
   experience inconsistent behavior. For example, code such as</p>

   <pre><code class="js">element.addEventListener("load", () => console.log("loaded"));
console.log("1");
element.loadData();
console.log("2");</code></pre>

   <p>will sometimes log "1, 2, loaded" (if the data needs to be fetched), and sometimes log "1,
   loaded, 2" (if the data is already cached). Similarly, after the call to <code
   data-x="">loadData()</code>, it will be inconsistent whether or not the data is set on the
   element.</p>

   <p>To get a consistent ordering, <code data-x="dom-queueMicrotask">queueMicrotask()</code> can be
   used:</p>

   <pre><code class="js">MyElement.prototype.loadData = function (url) {
  if (this._cache[url]) {<strong>
    queueMicrotask(() => {
      this._setData(this._cache[url]);
      this.dispatchEvent(new Event("load"));
    });</strong>
  } else {
    fetch(url).then(res => res.arrayBuffer()).then(data => {
      this._cache[url] = data;
      this._setData(data);
      this.dispatchEvent(new Event("load"));
    });
  }
};</code></pre>

   <p>By essentially rearranging the queued code to be after the <span>JavaScript execution context
   stack</span> empties, this ensures a consistent ordering and update of the element's state.</p>
  </div>

  <div class="example">
   <p>Another interesting use of <code data-x="dom-queueMicrotask">queueMicrotask()</code> is to
   allow uncoordinated "batching" of work by multiple callers. For example, consider a library
   function that wants to send data somewhere as soon as possible, but doesn't want to make multiple
   network requests if doing so is easily avoidable. One way to balance this would be like so:</p>

   <pre><code class="js">const queuedToSend = [];

function sendData(data) {
  queuedToSend.push(data);

  if (queuedToSend.length === 1) {
    queueMicrotask(() => {
      const stringToSend = JSON.stringify(queuedToSend);
      queuedToSend.length = 0;

      fetch("/endpoint", stringToSend);
    });
  }
}</code></pre>

   <p>With this architecture, multiple subsequent calls to <code data-x="">sendData()</code> within
   the currently executing synchronous JavaScript will be batched together into one
   <code>fetch()</code> call, but with no intervening event loop tasks preempting the fetch (as
   would have happened with similar code that instead used <code
   data-x="dom-setTimeout">setTimeout()</code>).</p>
  </div>


  <h3>User prompts</h3>

  <!--
   v2 ideas:
    * in-window modal dialogs
       - escape/enter handling
       - dragging the window out of the tab
    * in-window non-modal palettes
       - with a solution for the mobile space
  -->

  <h4>Simple dialogs</h4>

  <dl class="domintro">
   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-alert">alert</span>(<var>message</var>)</code></dt>
   <dd><p>Displays a modal alert with the given message, and waits for the user to dismiss
   it.</p></dd>

   <dt><code data-x=""><var>result</var> = <var>window</var>.<span subdfn data-x="dom-confirm">confirm</span>(<var>message</var>)</code></dt>
   <dd>
    <p>Displays a modal OK/Cancel prompt with the given message, waits for the user to dismiss it,
    and returns true if the user clicks OK and false if the user clicks Cancel.</p>
   </dd>

   <dt><code data-x=""><var>result</var> = <var>window</var>.<span subdfn data-x="dom-prompt">prompt</span>(<var>message</var> [, <var>default</var>])</code></dt>
   <dd>
    <p>Displays a modal text control prompt with the given message, waits for the user to dismiss
    it, and returns the value that the user entered. If the user cancels the prompt, then returns
    null instead. If the second argument is present, then the given value is used as a default.</p>
   </dd>
  </dl>

  <p class="note">Logic that depends on <span data-x="concept-task">tasks</span> or <span
  data-x="microtask">microtasks</span>, such as <span data-x="media element">media elements</span>
  loading their <span>media data</span>, are stalled when these methods are invoked.</p>

  <div w-nodev>

  <p>The <dfn method for="Window"><code data-x="dom-alert-noargs">alert()</code></dfn> and <dfn
  method for="Window"><code data-x="dom-alert">alert(<var>message</var>)</code></dfn> method steps
  are:</p>

  <ol>
   <li><p>If we <span>cannot show simple dialogs</span> for <span>this</span>, then return.</p></li>

   <li><p>If the method was invoked with no arguments, then let <var>message</var> be the
   empty string; otherwise, let <var>message</var> be the method's first
   argument.</p></li>

   <li><p>Set <var>message</var> to the result of <span data-x="normalize newlines">normalizing
   newlines</span> given <var>message</var>.</p></li>

   <li><p>Set <var>message</var> to the result of <span data-x="optionally truncate a simple dialog
   string">optionally truncating</span> <var>message</var>.</p></li>

   <li><p>Let <var>userPromptHandler</var> be <span>WebDriver BiDi user prompt opened</span> with
   <span>this</span>, "<code data-x="">alert</code>", and <var>message</var>.</p></li>

   <li>
    <p>If <var>userPromptHandler</var> is "<code data-x="">none</code>", then:</p>

    <ol>
     <li><p>Show <var>message</var> to the user, treating U+000A LF as a line break.</p></li>

     <li><p>Optionally, <span>pause</span> while waiting for the user to acknowledge the
     message.</p></li>
    </ol>
   </li>

   <li><p>Invoke <span>WebDriver BiDi user prompt closed</span> with <span>this</span>,
   "<code data-x="">alert</code>", and true.</p></li>
  </ol>

  <p class="note">This method is defined using two overloads, instead of using an
  <!--non-normative-->optional argument, for historical reasons. The practical impact of this is
  that <code data-x="" class="js">alert(undefined)</code> is treated as <code data-x=""
  class="js">alert("undefined")</code>, but <code data-x="" class="js">alert()</code> is treated as
  <code data-x="" class="js">alert("")</code>.</p>

  <p>The <dfn method for="Window"><code
  data-x="dom-confirm">confirm(<var>message</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>If we <span>cannot show simple dialogs</span> for <span>this</span>, then return
   false.</p></li>

   <li><p>Set <var>message</var> to the result of <span data-x="normalize newlines">normalizing
   newlines</span> given <var>message</var>.</p></li>

   <li><p>Set <var>message</var> to the result of <span data-x="optionally truncate a simple dialog
   string">optionally truncating</span> <var>message</var>.</p></li>

   <li><p>Show <var>message</var> to the user, treating U+000A LF as a line break, and ask the user
   to respond with a positive or negative response.</p></li>

   <li><p>Let <var>userPromptHandler</var> be <span>WebDriver BiDi user prompt opened</span> with
   <span>this</span>, "<code data-x="">confirm</code>", and <var>message</var>.</p></li>

   <li><p>Let <var>accepted</var> be false.</p></li>

   <li>
    <p>If <var>userPromptHandler</var> is "<code data-x="">none</code>", then:</p>

    <ol>
     <li><p><span>Pause</span> until the user responds either positively or negatively.</p></li>

     <li><p>If the user responded positively, then set <var>accepted</var> to true.</p></li>
    </ol>
   </li>

   <li><p>If <var>userPromptHandler</var> is "<code data-x="">accept</code>", then set
   <var>accepted</var> to true.</p></li>

   <li><p>Invoke <span>WebDriver BiDi user prompt closed</span> with <span>this</span>,
   "<code data-x="">confirm</code>", and <var>accepted</var>.</p></li>

   <li><p>Return <var>accepted</var>.</p></li>
  </ol>

  <p>The <dfn method for="Window"><code data-x="dom-prompt">prompt(<var>message</var>,
  <var>default</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>If we <span>cannot show simple dialogs</span> for <span>this</span>, then return
   null.</p></li>

   <li><p>Set <var>message</var> to the result of <span data-x="normalize newlines">normalizing
   newlines</span> given <var>message</var>.</p></li>

   <li><p>Set <var>message</var> to the result of <span data-x="optionally truncate a simple dialog
   string">optionally truncating</span> <var>message</var>.</p></li>

   <li><p>Set <var>default</var> to the result of <span data-x="optionally truncate a simple dialog
   string">optionally truncating</span> <var>default</var>.</p></li>

   <li><p>Show <var>message</var> to the user, treating U+000A LF as a line break, and ask the user
   to either respond with a string value or abort. The response must be defaulted to the value given
   by <var>default</var>.</p></li>

   <li><p>Let <var>userPromptHandler</var> be <span>WebDriver BiDi user prompt opened</span> with
   <span>this</span>, "<code data-x="">prompt</code>", and <var>message</var>.</p></li>

   <li><p>Let <var>result</var> be null.</p></li>

   <li>
    <p>If <var>userPromptHandler</var> is "<code data-x="">none</code>", then:</p>

    <ol>
     <li><p><span>Pause</span> while waiting for the user's response.</p></li>

     <li><p>If the user did not abort, then set <var>result</var> to the string that the user
     responded with.</p></li>
    </ol>
   </li>

   <li><p>Otherwise, if <var>userPromptHandler</var> is "<code data-x="">accept</code>", then set
   <var>result</var> to the empty string.</p></li>

   <li><p>Invoke <span>WebDriver BiDi user prompt closed</span> with <span>this</span>,
   "<code data-x="">prompt</code>", false if <var>result</var> is null or true otherwise,
   and <var>result</var>.</p></li>

   <li><p>Return <var>result</var>.</p></li>
  </ol>

  <p>To <dfn>optionally truncate a simple dialog string</dfn> <var>s</var>, return either
  <var>s</var> itself or some string derived from <var>s</var> that is shorter. User agents should
  not provide UI for displaying the elided portion of <var>s</var>, as this makes it too easy for
  abusers to create dialogs of the form "Important security alert! Click 'Show More' for full
  details!".</p>

  <p class="note">For example, a user agent might want to only display the first 100 characters of a
  message. Or, a user agent might replace the middle of the string with "&hellip;". These types of
  modifications can be useful in limiting the abuse potential of unnaturally large,
  trustworthy-looking system dialogs.</p>

  <p>We <dfn>cannot show simple dialogs</dfn> for a <code>Window</code> <var>window</var> when the
  following algorithm returns true:</p>

  <ol>
   <li><p>If the <span>active sandboxing flag set</span> of <var>window</var>'s <span
   data-x="concept-document-window">associated <code>Document</code></span> has the <span>sandboxed
   modals flag</span> set, then return true.</p></li>

   <li><p>If <var>window</var>'s <span>relevant settings object</span>'s <span
   data-x="concept-settings-object-origin">origin</span> and <var>window</var>'s <span>relevant
   settings object</span>'s <span>top-level origin</span> are not <span>same origin-domain</span>,
   then return true.</p></li>

   <li>If <var>window</var>'s <span>relevant agent</span>'s <span
   data-x="concept-agent-event-loop">event loop</span>'s <span>termination nesting level</span> is
   nonzero, then optionally return true.</li>

   <li><p>Optionally, return true. (For example, the user agent might give the
   user the option to ignore all modal dialogs, and would thus abort at this step whenever the method was
   invoked.)</p></li>

   <li><p>Return false.</p></li>
  </ol>

  </div>


  <h4>Printing</h4>

  <dl class="domintro">
   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-print">print</span>()</code></dt>

   <dd><p>Prompts the user to print the page.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn method for="Window"><code data-x="dom-print">print()</code></dfn> method steps
  are:</p>

  <ol>
   <li><p>Let <var>document</var> be <span>this</span>'s <span
   data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>If <var>document</var> is not <span>fully active</span>, then return.</p></li>

   <li><p>If <var>document</var>'s <span>unload counter</span> is greater than 0, then
   return.</p></li>

   <li><p>If <var>document</var> is <span>ready for post-load tasks</span>, then run the
   <span>printing steps</span> for <var>document</var>.</p></li>

   <li><p>Otherwise, set <var>document</var>'s <dfn>print when loaded</dfn> flag.</p></li>
  </ol>

  <p>User agents should also run the <span>printing steps</span> whenever the user asks for the
  opportunity to <span>obtain a physical form</span> (e.g. printed copy), or the representation of a
  physical form (e.g. PDF copy), of a document.</p>

  <p>The <dfn>printing steps</dfn> for a <code>Document</code> <var>document</var> are:</p>

  <ol>
   <li>
    <p>The user agent may display a message to the user or return (or both).</p>

    <p class="example">For instance, a kiosk browser could silently ignore any invocations of the
    <code data-x="dom-print">print()</code> method.</p>

    <p class="example">For instance, a browser on a mobile device could detect that there are no
    printers in the vicinity and display a message saying so before continuing to offer a "save to
    PDF" option.</p>
   </li>

   <li>
    <p>If the <span>active sandboxing flag set</span> of <var>document</var> has the <span>sandboxed
    modals flag</span> set, then return.</p>

    <p class="note">If the printing dialog is blocked by a <code>Document</code>'s sandbox,
    then neither the <code data-x="event-beforeprint">beforeprint</code> nor <code
    data-x="event-afterprint">afterprint</code> events will be fired.</p>
   </li>

   <li>
    <p>The user agent must <span data-x="concept-event-fire">fire an event</span> named <code
    data-x="event-beforeprint">beforeprint</code> at the <span>relevant global object</span> of
    <var>document</var>, as well as any <span data-x="child navigable">child navigable</span> in
    it.</p>

    <p class="XXX">Firing in children only doesn't seem right here, and some tasks likely need to
    be queued. See <a href="https://github.com/whatwg/html/issues/5096">issue #5096</a>.</p>

    <p class="example">The <code data-x="event-beforeprint">beforeprint</code> event can be used to
    annotate the printed copy, for instance adding the time at which the document was printed.</p>
   </li>

   <li><p>The user agent should offer the user the opportunity to <span>obtain a physical
   form</span> (or the representation of a physical form) of <var>document</var>. The user agent may
   wait for the user to either accept or decline before returning; if so, the user agent must
   <span>pause</span> while the method is waiting. Even if the user agent doesn't wait at this
   point, the user agent must use the state of the relevant documents as they are at this point in
   the algorithm if and when it eventually creates the alternate form.</p></li>

   <li>
    <p>The user agent must <span data-x="concept-event-fire">fire an event</span> named <code
    data-x="event-afterprint">afterprint</code> at the <span>relevant global object</span> of
    <var>document</var>, as well as any <span data-x="child navigable">child navigables</span> in
    it.</p>

    <p class="XXX">Firing in children only doesn't seem right here, and some tasks likely need to
    be queued. See <a href="https://github.com/whatwg/html/issues/5096">issue #5096</a>.</p>

    <p class="example">The <code data-x="event-afterprint">afterprint</code> event can be used to
    revert annotations added in the earlier event, as well as showing post-printing UI. For
    instance, if a page is walking the user through the steps of applying for a home loan, the
    script could automatically advance to the next step after having printed a form or other.</p>
   </li>
  </ol>

  </div>


  <h3 split-filename="system-state">System state and capabilities</h3>

  <h4>The <code>Navigator</code> object</h4>

  <p>Instances of <code>Navigator</code> represent the identity and state of the user agent (the
  client). It has also been used as a generic global under which various APIs are located, but this
  is not precedent to build upon. Instead use the <code>WindowOrWorkerGlobalScope</code> mixin or
  equivalent.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>Navigator</dfn> {
  // objects implementing this interface also implement the interfaces given below
};
<span>Navigator</span> includes <span>NavigatorID</span>;
<span>Navigator</span> includes <span>NavigatorLanguage</span>;
<span>Navigator</span> includes <span>NavigatorOnLine</span>;
<span>Navigator</span> includes <span>NavigatorContentUtils</span>;
<span>Navigator</span> includes <span>NavigatorCookies</span>;
<span>Navigator</span> includes <span>NavigatorPlugins</span>;
<span>Navigator</span> includes <span>NavigatorConcurrentHardware</span>;</code></pre>

  <div w-nodev>

  <p class="note">These interface mixins are defined separately so that <code>WorkerNavigator</code> can reuse parts of
  the <code>Navigator</code> interface.</p>

  <p>Each <code>Window</code> has an <dfn>associated <code>Navigator</code></dfn>, which is a <code>Navigator</code>
  object. Upon creation of the <code>Window</code> object, its <span>associated <code>Navigator</code></span> must be
  set to a <span>new</span> <code>Navigator</code> object created in the <code>Window</code> object's <span
  data-x="concept-relevant-realm">relevant realm</span>.</p>

  <p>The <dfn attribute for="Window"><code data-x="dom-navigator">navigator</code></dfn> and <dfn attribute
  for="Window"><code data-x="dom-clientInformation">clientInformation</code></dfn> getter steps are to return
  <span>this</span>'s <span>associated <code>Navigator</code></span>.

  </div>


  <h5>Client identification</h5>

  <pre><code class="idl">interface mixin <dfn interface>NavigatorID</dfn> {
  readonly attribute DOMString <span data-x="dom-navigator-appCodeName">appCodeName</span>; // constant "Mozilla"
  readonly attribute DOMString <span data-x="dom-navigator-appName">appName</span>; // constant "Netscape"
  readonly attribute DOMString <span data-x="dom-navigator-appVersion">appVersion</span>;
  readonly attribute DOMString <span data-x="dom-navigator-platform">platform</span>;
  readonly attribute DOMString <span data-x="dom-navigator-product">product</span>; // constant "Gecko"
  [Exposed=Window] readonly attribute DOMString <span data-x="dom-navigator-productSub">productSub</span>;
  readonly attribute DOMString <span data-x="dom-navigator-userAgent">userAgent</span>;
  [Exposed=Window] readonly attribute DOMString <span data-x="dom-navigator-vendor">vendor</span>;
  [Exposed=Window] readonly attribute DOMString <span data-x="dom-navigator-vendorSub">vendorSub</span>; // constant ""
};</code></pre>

  <p>In certain cases, despite the best efforts of the entire industry, web browsers have bugs and
  limitations that web authors are forced to work around.</p>

  <p>This section defines a collection of attributes that can be used to determine, from script, the
  kind of user agent in use, in order to work around these issues.</p>

  <div w-nodev>

  <p>The user agent has a <dfn data-x="concept-navigator-compatibility-mode">navigator compatibility
  mode</dfn>, which is either <i>Chrome</i>, <i>Gecko</i>, or <i>WebKit</i>.</p>

  <p class="note">The <span data-x="concept-navigator-compatibility-mode">navigator compatibility
  mode</span> constrains the <code>NavigatorID</code> mixin to the combinations of attribute
  values and presence of <code data-x="dom-navigator-taintEnabled">taintEnabled()</code> and <code
  data-x="dom-navigator-oscpu">oscpu</code> that are known to be compatible with existing web
  content.</p>

  </div>

  <p>Client detection should always be limited to detecting known current versions; future versions
  and unknown versions should always be assumed to be fully compliant.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>self</var>.<span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-navigator-appCodeName">appCodeName</span></code></dt>
   <dd><p>Returns the string "<code data-x="">Mozilla</code>".</p></dd>

   <dt><code data-x=""><var>self</var>.<span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-navigator-appName">appName</span></code></dt>
   <dd><p>Returns the string "<code data-x="">Netscape</code>".</p></dd>

   <dt><code data-x=""><var>self</var>.<span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-navigator-appVersion">appVersion</span></code></dt>
   <dd><p>Returns the version of the browser.</p></dd>

   <dt><code data-x=""><var>self</var>.<span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-navigator-platform">platform</span></code></dt>
   <dd><p>Returns the name of the platform.</p></dd>

   <dt><code data-x=""><var>self</var>.<span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-navigator-product">product</span></code></dt>
   <dd><p>Returns the string "<code data-x="">Gecko</code>".</p></dd>

   <dt><code data-x=""><var>window</var>.<span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-navigator-productSub">productSub</span></code></dt>
   <dd><p>Returns either the string "<code data-x="">20030107</code>", or the string "<code
   data-x="">20100101</code>".</p></dd>

   <dt><code data-x=""><var>self</var>.<span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-navigator-userAgent">userAgent</span></code></dt>
   <dd><p>Returns the complete `<code data-x="">User-Agent</code>` header.</p></dd>

   <dt><code data-x=""><var>window</var>.<span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-navigator-vendor">vendor</span></code></dt>
   <dd><p>Returns either the empty string, the string "<code data-x="">Apple Computer, Inc.</code>",
   or the string "<code data-x="">Google Inc.</code>".</p></dd>

   <dt><code data-x=""><var>window</var>.<span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-navigator-vendorSub">vendorSub</span></code></dt>
   <dd><p>Returns the empty string.</p></dd>
  </dl>

  <div w-nodev>

  <dl>
   <dt><dfn attribute for="NavigatorID"><code data-x="dom-navigator-appCodeName">appCodeName</code></dfn></dt>
   <dd><p>Must return the string "<code data-x="">Mozilla</code>".</p></dd>

   <!-- appMinorVersion: IE only. In IE8b1, returns " Beta" (with the space) -->

   <dt><dfn attribute for="NavigatorID"><code data-x="dom-navigator-appName">appName</code></dfn></dt>
   <dd><p>Must return the string "<code data-x="">Netscape</code>".</p></dd>

   <dt><dfn attribute for="NavigatorID"><code
   data-x="dom-navigator-appVersion">appVersion</code></dfn></dt>
   <dd>
    <p>Must return the appropriate string that starts with "<code data-x="">5.0 (</code>", as
    follows:</p>

    <p>Let <var>trail</var> be the substring of <span data-x="default-user-agent-value">default
    `<code>User-Agent</code>` value</span> that follows the "<code data-x="">Mozilla/</code>"
    prefix.</p>
    <!-- Fetch doesn't actually require the value to start with "Mozilla/5.0 (". -->

    <dl class="switch"> <dt>If the <span data-x="concept-navigator-compatibility-mode">navigator
    compatibility mode</span> is <i>Chrome</i> or <i>WebKit</i></dt>
    <dd><p>Return <var>trail</var>.</p></dd>

     <dt>If the <span data-x="concept-navigator-compatibility-mode">navigator compatibility
     mode</span> is <i>Gecko</i></dt>
     <dd>
      <p>If <var>trail</var> starts with "<code data-x="">5.0 (Windows</code>", then return "<code
      data-x="">5.0 (Windows)</code>".</p>

      <p>Otherwise, return the prefix of <var>trail</var> up to but not including the first U+003B
      (;), concatenated with the character U+0029 RIGHT PARENTHESIS. For example, "<code
      data-x="">5.0 (Macintosh)</code>", "<code data-x="">5.0 (Android 10)</code>", or "<code
      data-x="">5.0 (X11)</code>".</p> <!-- Unclear if Gecko mode returning a shorter string that
      doesn't contain the whole trail is needed for Web compat. -->
     </dd>
    </dl>
   </dd>

   <!-- buildID: Mozilla only -->

   <!-- oscpu: Mozilla only -->

   <dt><dfn attribute for="NavigatorID"><code data-x="dom-navigator-platform">platform</code></dfn></dt>
   <dd><p>Must return a string representing the platform on which the
   browser is executing (e.g. "<code data-x="">MacIntel</code>", "<code data-x="">Win32</code>",
   "<code data-x="">Linux x86_64</code>", "<code data-x="">Linux armv81</code>") or, for privacy and
   compatibility, a string that is commonly returned on another platform.</p></dd>

   <dt><dfn attribute for="NavigatorID"><code data-x="dom-navigator-product">product</code></dfn></dt>
   <dd><p>Must return the string "<code data-x="">Gecko</code>".</p></dd>

   <dt><dfn attribute for="NavigatorID"><code data-x="dom-navigator-productSub">productSub</code></dfn></dt>
   <dd>
    <p>Must return the appropriate string from the following list:</p>

    <dl class="switch">

     <dt>If the <span data-x="concept-navigator-compatibility-mode">navigator compatibility
     mode</span> is <i>Chrome</i> or <i>WebKit</i></dt>
     <dd><p>The string "<code data-x="">20030107</code>".</p></dd>

     <dt>If the <span data-x="concept-navigator-compatibility-mode">navigator compatibility
     mode</span> is <i>Gecko</i></dt>
     <dd><p>The string "<code data-x="">20100101</code>".</p></dd>
    </dl>
   </dd>

   <!-- securityPolicy: Mozilla only; always returns "" -->

   <dt><dfn attribute for="NavigatorID"><code data-x="dom-navigator-userAgent">userAgent</code></dfn></dt>
   <dd><p>Must return the <span data-x="default-user-agent-value">default `<code>User-Agent</code>`
   value</span>.</p></dd>

   <dt><dfn attribute for="NavigatorID"><code data-x="dom-navigator-vendor">vendor</code></dfn></dt>
   <dd>
    <p>Must return the appropriate string from the following list:</p>

    <dl class="switch">

     <dt>If the <span data-x="concept-navigator-compatibility-mode">navigator compatibility
     mode</span> is <i>Chrome</i></dt>
     <dd><p>The string "<code data-x="">Google Inc.</code>".</p></dd>

     <dt>If the <span data-x="concept-navigator-compatibility-mode">navigator compatibility
     mode</span> is <i>Gecko</i></dt>
     <dd><p>The empty string.</p></dd>

     <dt>If the <span data-x="concept-navigator-compatibility-mode">navigator compatibility
     mode</span> is <i>WebKit</i></dt>
     <dd><p>The string "<code data-x="">Apple Computer, Inc.</code>".</p></dd>
    </dl>
   </dd>

   <dt><dfn attribute for="NavigatorID"><code data-x="dom-navigator-vendorSub">vendorSub</code></dfn></dt>
   <dd><p>Must return the empty string.</p></dd>
  </dl>

  <p>If the <span data-x="concept-navigator-compatibility-mode">navigator compatibility mode</span>
  is <i>Gecko</i>, then the user agent must also support the following partial interface:</p>

  <pre><code class="idl">partial interface mixin <span id="NavigatorID-partial">NavigatorID</span> {
  [Exposed=Window] boolean <span data-x="dom-navigator-taintEnabled">taintEnabled</span>(); // constant false
  [Exposed=Window] readonly attribute DOMString <span data-x="dom-navigator-oscpu">oscpu</span>;
};</code></pre>

  <p>The <dfn method for="NavigatorID"><code
  data-x="dom-navigator-taintEnabled">taintEnabled()</code></dfn> method must return false.</p>

  <p>The <dfn attribute for="NavigatorID"><code data-x="dom-navigator-oscpu">oscpu</code></dfn> attribute's getter must return
  either the empty string or a string representing the platform on which the browser is executing,
  e.g. "<code data-x="">Windows NT 10.0; Win64; x64</code>", "<code data-x="">Linux
  x86_64</code>".</p>

  <!-- similar paragraph in next section -->
  <p class="warning">
  <!--INSERT TRACKING-->
  Any information in this API that varies from user to user can be used to profile the user. In
  fact, if enough such information is available, a user can actually be uniquely identified. For
  this reason, user agent implementers are strongly urged to include as little information in this
  API as possible.</p>

  </div>

  <!-- next section refers to previous section as "previous section" -->

  <h5>Language preferences</h5>

  <pre><code class="idl">interface mixin <dfn interface>NavigatorLanguage</dfn> {
  readonly attribute DOMString <span data-x="dom-navigator-language">language</span>;
  readonly attribute FrozenArray&lt;DOMString&gt; <span data-x="dom-navigator-languages">languages</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>self</var>.<span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-navigator-language">language</span></code></dt>
   <dd><p>Returns a language tag representing the user's preferred language.</p></dd>

   <dt><code data-x=""><var>self</var>.<span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-navigator-languages">languages</span></code></dt>
   <dd>
    <p>Returns an array of language tags representing the user's preferred languages, with the most
    preferred language first.</p>

    <p>The most preferred language is the one returned by <code
    data-x="dom-navigator-language">navigator.language</code>.</p>
   </dd>
  </dl>

  <p class="note">A <code data-x="event-languagechange">languagechange</code> event is fired at the
  <code>Window</code> or <code>WorkerGlobalScope</code> object when the user agent's understanding
  of what the user's preferred languages are changes.</p>

  <div w-nodev>

  <dl>
   <!--
   <dt><dfn><code data-x="dom-navigator-browserLanguage">browserLanguage</code></dfn></dt> <!- - Opera and IE only - ->
   <dd><p>Must return a language tag representing either <span>a plausible language</span> or the
   language the browser uses in its interface.</p></dd>


   <dt><dfn><code data-x="dom-navigator-userLanguage">userLanguage</code></dfn></dt> <!- - Opera and IE only - ->
   -->
   <!-- at time of testing, this was supported by Opera, Safari, and Mozilla only -->
   <dt><dfn attribute for="NavigatorLanguage"><code
   data-x="dom-navigator-language">language</code></dfn></dt> <dd><p>Must return a valid BCP 47
   language tag representing either <span>a plausible language</span> or the user's most preferred
   language. <ref>BCP47</ref></p></dd>

   <dt><dfn attribute for="NavigatorLanguage"><code data-x="dom-navigator-languages">languages</code></dfn></dt>
   <dd>
    <p>Must return a <span>frozen array</span> of valid BCP 47 language tags representing either one
    or more <span data-x="a plausible language">plausible languages</span>, or the user's preferred
    languages, ordered by preference with the most preferred language first. The same object must be
    returned until the user agent needs to return different values, or values in a different order.
    <ref>BCP47</ref></p>

    <p>Whenever the user agent needs to make the <code
    data-x="dom-navigator-languages">navigator.languages</code> attribute of a <code>Window</code>
    or <code>WorkerGlobalScope</code> object <var>global</var> return a new set of language tags,
    the user agent must <span>queue a global task</span> on the <span>DOM manipulation task
    source</span> given <var>global</var> to <span data-x="concept-event-fire">fire an event</span>
    named <code data-x="event-languagechange">languagechange</code> at <var>global</var>, and wait
    until that task begins to be executed before actually returning a new value.</p>
   </dd>
  </dl>

  <p>To determine <dfn>a plausible language</dfn>, the user agent should bear in mind the following:</p>

  <ul>
   <li>
   <!--INSERT TRACKING-->
   Any information in this API that varies from user to user can be used to profile or identify the
   user.</li>

   <li>If the user is not using a service that obfuscates the user's point of origin (e.g. the Tor
   anonymity network), then the value that is least likely to distinguish the user from other users
   with similar origins (e.g. from the same IP address block) is the language used by the majority
   of such users. <ref>TOR</ref></li>

   <li>If the user is using an anonymizing service, then the value "<code data-x="">en-US</code>" is
   suggested; if all users of the service use that same value, that reduces the possibility of
   distinguishing the users from each other.</li>
  </ul>

  <p>
  <!--INSERT TRACKING-->
  To avoid introducing any more fingerprinting vectors, user agents should use the same list for the
  APIs defined in this function as for the HTTP `<code
  data-x="http-accept-language">Accept-Language</code>` header.</p>

  </div>


  <h5 id="navigator.online"><span id="browser-state"></span>Browser state</h5>

  <pre><code class="idl">interface mixin <dfn interface>NavigatorOnLine</dfn> {
  readonly attribute boolean <span data-x="dom-navigator-onLine">onLine</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>self</var>.<span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-navigator-onLine">onLine</span></code></dt>

   <dd>
    <p>Returns false if the user agent is definitely offline (disconnected from the network).
    Returns true if the user agent might be online.</p>

    <p>The events <code data-x="event-online">online</code> and <code
    data-x="event-offline">offline</code> are fired when the value of this attribute changes.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="NavigatorOnLine"><code
  data-x="dom-navigator-onLine">onLine</code></dfn> attribute must return false if the user agent
  will not contact the network when the user follows links or when a script requests a remote page
  (or knows that such an attempt would fail), and must return true otherwise.</p>

  <p>When the value that would be returned by the <code
  data-x="dom-navigator-onLine">navigator.onLine</code> attribute of a <code>Window</code> or
  <code>WorkerGlobalScope</code> <var>global</var> changes from true to false, the user agent must
  <span>queue a global task</span> on the <span>networking task source</span> given
  <var>global</var> to <span data-x="concept-event-fire">fire an event</span> named <code
  data-x="event-offline">offline</code> at <var>global</var>.</p>

  <p>On the other hand, when the value that would be returned by the <code
  data-x="dom-navigator-onLine">navigator.onLine</code> attribute of a <code>Window</code> or
  <code>WorkerGlobalScope</code> <var>global</var> changes from false to true, the user agent must
  <span>queue a global task</span> on the <span>networking task source</span> given
  <var>global</var> to <span data-x="concept-event-fire">fire an event</span> named <code
  data-x="event-online">online</code> at the <code>Window</code> or <code>WorkerGlobalScope</code>
  object.</p>

  </div>

  <p class="note">This attribute is inherently unreliable. A computer can be connected to a network
  without having Internet access.</p>

  <div class="example">
   <p>In this example, an indicator is updated as the browser goes online and offline.</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html lang="en">
 &lt;head>
  &lt;title>Online status&lt;/title>
  &lt;script>
   function updateIndicator() {
     document.getElementById('indicator').textContent = navigator.onLine ? 'online' : 'offline';
   }
  &lt;/script>
 &lt;/head>
 &lt;body onload="updateIndicator()" ononline="updateIndicator()" onoffline="updateIndicator()">
  &lt;p>The network is: &lt;span id="indicator">(state unknown)&lt;/span>
 &lt;/body>
&lt;/html></code></pre>
  </div>


  <h5 id="custom-handlers">Custom scheme handlers: the <code data-x="dom-navigator-registerProtocolHandler">registerProtocolHandler()</code> method</h5>

  <pre><code class="idl">interface mixin <dfn interface>NavigatorContentUtils</dfn> {
  [SecureContext] undefined <span data-x="dom-navigator-registerProtocolHandler">registerProtocolHandler</span>(DOMString scheme, USVString url);
  [SecureContext] undefined <span data-x="dom-navigator-unregisterProtocolHandler">unregisterProtocolHandler</span>(DOMString scheme, USVString url);
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>window</var>.<span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-navigator-registerProtocolHandler">registerProtocolHandler</span>(<var>scheme</var>, <var>url</var>)</code></dt>
   <dd>
    <p>Registers a handler for <var>scheme</var> at <var>url</var>. For example, an online telephone
    messaging service could register itself as a handler of the <code data-x="sms
    protocol">sms:</code> scheme, so that if the user clicks on such a link, they are given the
    opportunity to use that web site. <ref>SMS</ref></p>

    <p>The string "<code data-x="">%s</code>" in <var>url</var> is used as a placeholder for where
    to put the URL of the content to be handled.</p>

    <p>Throws a <span>"<code>SecurityError</code>"</span> <code>DOMException</code> if the user
    agent blocks the registration (this might happen if trying to register as a handler for "<code
    data-x="">http</code>", for instance).</p>

    <p>Throws a <span>"<code>SyntaxError</code>"</span> <code>DOMException</code> if the "<code
    data-x="">%s</code>" string is missing in <var>url</var>.</p>
   </dd>

   <dt><code data-x=""><var>window</var>.<span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-navigator-unregisterProtocolHandler">unregisterProtocolHandler</span>(<var>scheme</var>, <var>url</var>)</code></dt>
   <dd>
    <p>Unregisters the handler given by the arguments.</p>

    <p>Throws a <span>"<code>SecurityError</code>"</span> <code>DOMException</code> if the user
    agent blocks the deregistration (this might happen if with invalid schemes, for instance).</p>

    <p>Throws a <span>"<code>SyntaxError</code>"</span> <code>DOMException</code> if the "<code
    data-x="">%s</code>" string is missing in <var>url</var>.</p>
   </dd>
  </dl>


  <div w-nodev>

  <p>The <dfn method for="NavigatorContentUtils"><code
  data-x="dom-navigator-registerProtocolHandler">registerProtocolHandler(<var>scheme</var>,
  <var>url</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let (<var>normalizedScheme</var>, <var>normalizedURLString</var>) be the result of running
   <span>normalize protocol handler parameters</span> with <var>scheme</var>, <var>url</var>, and
   <span>this</span>'s <span>relevant settings object</span>.</p></li>

   <li>
    <p><span>In parallel</span>: <dfn data-x="protocol-handler-registration" export>register a
    protocol handler</dfn> for <var>normalizedScheme</var> and <var>normalizedURLString</var>. User
    agents may, within the constraints described, do whatever they like. A user agent could, for
    instance, prompt the user and offer the user the opportunity to add the site to a shortlist of
    handlers, or make the handlers their default, or cancel the request. User agents could also
    silently collect the information, providing it only when relevant to the user.</p>

    <p>User agents should keep track of which sites have registered handlers (even if the user has
    declined such registrations) so that the user is not repeatedly prompted with the same
    request.</p>

    <p>If the <span><code
    data-x="dom-navigator-registerProtocolHandler">registerProtocolHandler()</code> automation
    mode</span> of <span>this</span>'s <span>relevant global object</span>'s <span
    data-x="concept-document-window">associated <code>Document</code></span> is not "<dfn><code
    data-x="rph-automation-mode-none">none</code></dfn>", the user agent should first verify that it
    is in an automation context (see <span>WebDriver's security considerations</span>). The user
    agent should then bypass the above communication of information and gathering of user consent,
    and instead do the following based on the value of the <span><code
    data-x="dom-navigator-registerProtocolHandler">registerProtocolHandler()</code> automation
    mode</span>:</p>

    <dl>
     <dt>"<dfn data-x="rph-automation-mode-auto-accept"><code>autoAccept</code></dfn>"</dt>
     <dd><p>Act as if the user has seen the registration details and accepted the request.</p></dd>

     <dt>"<dfn data-x="rph-automation-mode-auto-reject"><code>autoReject</code></dfn>"</dt>
     <dd><p>Act as if the user has seen the registration details and rejected the request.</p></dd>
    </dl>

    <p>When the <dfn export data-x="protocol-handler-invocation" data-lt="invoke a protocol
    handler">user agent uses this handler</dfn> for a <span>URL</span> <var>inputURL</var>:</p>

    <ol>
     <li><p><span>Assert</span>: <var>inputURL</var>'s <span
     data-x="concept-url-scheme">scheme</span> is <var>normalizedScheme</var>.</p></li>

     <li><p><span data-x="set the username">Set the username</span> given <var>inputURL</var> and
     the empty string.</p></li>

     <li><p><span data-x="set the password">Set the password</span> given <var>inputURL</var> and
     the empty string.</p></li>

     <li><p>Let <var>inputURLString</var> be the <span
     data-x="concept-url-serializer">serialization</span> of <var>inputURL</var>.</p></li>

     <li><p>Let <var>encodedURL</var> be the result of running <span>UTF-8 percent-encode</span> on
     <var>inputURLString</var> using the <span>component percent-encode set</span>.</p></li>

     <li><p>Let <var>handlerURLString</var> be <var>normalizedURLString</var>.</p></li>

     <li><p>Replace the first instance of "<code data-x="">%s</code>" in <var>handlerURLString</var>
     with <var>encodedURL</var>.</p></li>

     <li><p>Let <var>resultURL</var> be the result of <span data-x="URL parser">parsing</span>
     <var>handlerURLString</var>.</p></li>

     <li><p><span>Navigate</span><!--DONAV user--> an appropriate <span>navigable</span> to
     <var>resultURL</var>.</p></li>
    </ol>

    <div class="example">
     <p>If the user had visited a site at <code data-x="">https://example.com/</code> that made the
     following call:</p>

     <pre><code class="js">navigator.registerProtocolHandler('web+soup', 'soup?url=%s')</code></pre>

     <p>...and then, much later, while visiting <code data-x="">https://www.example.net/</code>,
     clicked on a link such as:</p>

     <pre><code class="html">&lt;a href="web+soup:chicken-k&#xEF;wi">Download our Chicken K&#xEF;wi soup!&lt;/a></code></pre>

     <p>...then the UA might navigate to the following URL:</p>

     <pre>https://example.com/soup?url=web+soup:chicken-k%C3%AFwi</pre>

     <p>This site could then do whatever it is that it does with soup (synthesize it and ship it to
     the user, or whatever).</p>
    </div>

    <p>This does not define when the handler is used. To some extent, the <span
    data-x="navigate">processing model for navigating across documents</span> defines some cases
    where it is relevant, but in general user agents may use this information wherever they would
    otherwise consider handing schemes to native plugins or helper applications.</p>
   </li>
  </ol>

  <p>The <dfn method for="NavigatorContentUtils"><code
  data-x="dom-navigator-unregisterProtocolHandler">unregisterProtocolHandler(<var>scheme</var>,
  <var>url</var>)</code></dfn> method steps are:

  <ol>
   <li><p>Let (<var>normalizedScheme</var>, <var>normalizedURLString</var>) be the result of running
   <span>normalize protocol handler parameters</span> with <var>scheme</var>, <var>url</var>, and
   <span>this</span>'s <span>relevant settings object</span>.</p></li>

   <li><p><span>In parallel</span>: unregister the handler described by <var>normalizedScheme</var>
   and <var>normalizedURLString</var>.</p></li>
  </ol>

  <hr>

  <p>To <dfn export>normalize protocol handler parameters</dfn>, given a string <var>scheme</var>, a
  string <var>url</var>, and an <span>environment settings object</span> <var>environment</var>, run
  these steps:</p>

  <ol>
   <li><p>Set <var>scheme</var> to <var>scheme</var>, <span>converted to ASCII
   lowercase</span>.</p></li>

   <li>
    <p>If <var>scheme</var> is neither a <span>safelisted scheme</span> nor a string starting with
    "<code data-x="">web+</code>" followed by one or more <span data-x="ASCII lower alpha">ASCII
    lower alphas</span>, then throw a <span>"<code>SecurityError</code>"</span>
    <code>DOMException</code>.</p>

    <p class="note">This means that including a colon in <var>scheme</var> (as in "<code
    data-x="">mailto:</code>") will throw.</p>

    <p>The following schemes are the <dfn data-x="safelisted scheme" export>safelisted
    schemes</dfn>:</p>

    <ul class="brief">
     <li><code data-x="">bitcoin</code></li> <!-- https://en.bitcoin.it/wiki/BIP_0021 -->
     <li><code data-x="">ftp</code></li>
     <li><code data-x="">ftps</code></li>
     <li><code data-x="">geo</code></li>
     <li><code data-x="">im</code></li>
     <li><code data-x="">irc</code></li>
     <li><code data-x="">ircs</code></li>
     <li><code data-x="">magnet</code></li>
     <li><code data-x="">mailto</code></li>
     <li><code data-x="">matrix</code></li>
     <li><code data-x="">mms</code></li>
     <li><code data-x="">news</code></li>
     <li><code data-x="">nntp</code></li>
     <li><code data-x="">openpgp4fpr</code></li>
     <li><code data-x="">sftp</code></li>
     <li><code data-x="">sip</code></li>
     <li><code data-x="">sms</code></li>
     <li><code data-x="">smsto</code></li>
     <li><code data-x="">ssh</code></li>
     <li><code data-x="">tel</code></li>
     <li><code data-x="">urn</code></li>
     <li><code data-x="">webcal</code></li>
     <li><code data-x="">wtai</code></li>
     <li><code data-x="">xmpp</code></li>
    </ul>

    <p class="note">This list can be changed. If there are schemes that ought to be added, please
    send feedback.</p>
   </li>

   <li><p>If <var>url</var> does not contain "<code data-x="">%s</code>", then throw a
   <span>"<code>SyntaxError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>urlRecord</var> be the result of <span>encoding-parsing a URL</span> given
   <var>url</var>, relative to <var>environment</var>.</p></li>

   <li>
    <p>If <var>urlRecord</var> is failure, then throw a <span>"<code>SyntaxError</code>"</span>
    <code>DOMException</code>.</p>

    <p class="note">This is forcibly the case if the <code data-x="">%s</code> placeholder is in the
    host or port of the URL.</p>
   </li>

   <li><p>If <var>urlRecord</var>'s <span data-x="concept-url-scheme">scheme</span> is not an
   <span>HTTP(S) scheme</span> or <var>urlRecord</var>'s <span
   data-x="concept-url-origin">origin</span> is not <span>same origin</span> with
   <var>environment</var>'s <span data-x="concept-settings-object-origin">origin</span>, then throw
   a <span>"<code>SecurityError</code>"</span> <code>DOMException</code>.</p></li>

   <li>
    <p><span>Assert</span>: the result of <span>Is url potentially trustworthy?</span> given
    <var>urlRecord</var> is "<code data-x="">Potentially Trustworthy</code>".</p>

    <p class="note">Because <span>normalize protocol handler parameters</span> is run within a
    <span>secure context</span>, this is implied by the <span>same origin</span> condition.</p>
   </li>

   <li>
    <p>Return (<var>scheme</var>, <var>urlRecord</var>).</p>

    <p class="note">The <span data-x="concept-url-serializer">serialization</span> of
    <var>urlRecord</var> will by definition not be a <span>valid URL string</span> as it includes
    the string "<code data-x="">%s</code>" which is not a valid component in a URL.</p>
   </li>
  </ol>



  <h6>Security and privacy</h6>

  <p>Custom scheme handlers can introduce a number of concerns, in particular privacy concerns.</p>

  <p><strong>Hijacking all web usage.</strong> User agents should not allow schemes that are key to
  its normal operation, such as an <span>HTTP(S) scheme</span>, to be rerouted through third-party
  sites. This would allow a user's activities to be trivially tracked, and would allow user
  information, even in secure connections, to be collected.</p>

  <p><strong>Hijacking defaults.</strong> User agents are strongly urged to not automatically change
  any defaults, as this could lead the user to send data to remote hosts that the user is not
  expecting. New handlers registering themselves should never automatically cause those sites to be
  used.</p>

  <p><strong>Registration spamming.</strong> User agents should consider the possibility that a site
  will attempt to register a large number of handlers, possibly from multiple domains (e.g., by
  redirecting through a series of pages each on a different domain, and each registering a handler
  for <code data-x="">web+spam:</code> &mdash; analogous practices abusing other web browser
  features have been used by pornography web sites for many years). User agents should gracefully
  handle such hostile attempts, protecting the user.</p>

  <p><strong>Hostile handler metadata.</strong> User agents should protect against typical attacks
  against strings embedded in their interface, for example ensuring that markup or escape characters
  in such strings are not executed, that null bytes are properly handled, that over-long strings do
  not cause crashes or buffer overruns, and so forth.</p>

  <p><strong>Leaking private data.</strong> Web page authors may reference a custom scheme
  handler using URL data considered private. They might do so with the expectation that the user's
  choice of handler points to a page inside the organization, ensuring that sensitive data will not
  be exposed to third parties. However, a user may have registered a handler pointing to an
  external site, resulting in a data leak to that third party. Implementers might wish to consider
  allowing administrators to disable custom handlers on certain subdomains, content types, or
  schemes.</p>

  <p><strong>Interface interference.</strong> User agents should be prepared to handle intentionally
  long arguments to the methods. For example, if the user interface exposed consists of an "accept"
  button and a "deny" button, with the "accept" binding containing the name of the handler, it's
  important that a long name not cause the "deny" button to be pushed off the screen.</p>

  <h6><dfn id="rph-user-agent-automation">User agent automation</dfn></h6>

  <p>Each <code>Document</code> has a <dfn><code
  data-x="dom-navigator-registerProtocolHandler">registerProtocolHandler()</code> automation
  mode</dfn>. It defaults to "<code data-x="rph-automation-mode-none">none</code>", but it also can
  be either "<code data-x="rph-automation-mode-auto-accept">autoAccept</code>" or "<code
  data-x="rph-automation-mode-auto-reject">autoReject</code>".</p>

  <p>For the purposes of user agent automation and website testing, this standard defines <dfn
  data-x="set-rph-registration-mode">Set RPH Registration Mode</dfn> WebDriver <span>extension
  command</span>. It instructs the user agent to place a <code>Document</code> into a mode where it
  will automatically simulate a user either accepting or rejecting and registration confirmation
  prompt dialog.</p>

  <table class="data">
   <thead>
    <tr>
     <th>HTTP Method
     <th>URI Template
   <tbody>
    <tr>
     <td>`<code data-x="">POST</code>`
     <td><code data-x="">/session/{session id}/custom-handlers/set-mode</code>
  </table>

  <p>The <span>remote end steps</span> are:</p>

  <ol>
   <li><p>If <var>parameters</var> is not a JSON Object, return a <span>WebDriver error</span> with
   <span>WebDriver error code</span> <span>invalid argument</span>.</p></li>

   <li><p>Let <var>mode</var> be the result of <span>getting a property</span> named "<code
   data-x="">mode</code>" from <var>parameters</var>.</p></li>

   <li><p>If <var>mode</var> is not "<code
   data-x="rph-automation-mode-auto-accept">autoAccept</code>", "<code
   data-x="rph-automation-mode-auto-reject">autoReject</code>", or "<code
   data-x="rph-automation-mode-none">none</code>", return a <span>WebDriver error</span> with
   <span>WebDriver error code</span> <span>invalid argument</span>.</p></li>

   <li><p>Let <var>document</var> be the <span data-x="webdriver-current-browsing-context">current
   browsing context</span>'s <span>active document</span>.</p></li>

   <li><p>Set <var>document</var>'s <span><code
   data-x="dom-navigator-registerProtocolHandler">registerProtocolHandler()</code> automation
   mode</span> to <var>mode</var>.</p></li>

   <li><p>Return <span data-x="success-value">success</span> with data null.</p></li>
  </ol>

  </div>

  <h5>Cookies</h5>

  <pre><code class="idl">interface mixin <dfn interface>NavigatorCookies</dfn> {
  readonly attribute boolean <span data-x="dom-navigator-cookieEnabled">cookieEnabled</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>window</var>.<span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-navigator-cookieEnabled">cookieEnabled</span></code></dt>

   <dd><p>Returns false if setting a cookie will be ignored, and true otherwise.</p></dd>
  </dl>

  <div w-nodev>

   <p>The <dfn attribute for="NavigatorCookies"><code
   data-x="dom-navigator-cookieEnabled">cookieEnabled</code></dfn> attribute must return true if the
   user agent attempts to handle cookies according to <cite>HTTP State Management Mechanism</cite>,
   and false if it ignores cookie change requests. <ref>COOKIES</ref></p>

  </div>

  <h5>PDF viewing support</h5>

  <dl class="domintro">
   <dt><code data-x=""><var>window</var>.<span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-navigator-pdfViewerEnabled">pdfViewerEnabled</span></code></dt>

   <dd><p>Returns true if the user agent supports inline viewing of PDF files when <span
   data-x="navigate">navigating</span> to them, or false otherwise. In the latter case, PDF files
   will be handled by <span data-x="hand-off to external software">external
   software</span>.</p></dd>
  </dl>

  <pre><code class="idl">interface mixin <dfn interface>NavigatorPlugins</dfn> {
  [SameObject] readonly attribute <span>PluginArray</span> <span data-x="dom-navigator-plugins">plugins</span>;
  [SameObject] readonly attribute <span>MimeTypeArray</span> <span data-x="dom-navigator-mimeTypes">mimeTypes</span>;
  boolean <span data-x="dom-navigator-javaEnabled">javaEnabled</span>();
  readonly attribute boolean <span data-x="dom-navigator-pdfViewerEnabled">pdfViewerEnabled</span>;
};

[Exposed=Window,
 <span>LegacyUnenumerableNamedProperties</span>]
interface <dfn interface>PluginArray</dfn> {
  undefined <span data-x="dom-PluginArray-refresh">refresh</span>();
  readonly attribute unsigned long <span data-x="dom-PluginArray-length">length</span>;
  getter <span>Plugin</span>? <span data-x="dom-PluginArray-item">item</span>(unsigned long index);
  getter <span>Plugin</span>? <span data-x="dom-PluginArray-namedItem">namedItem</span>(DOMString name);
};

[Exposed=Window,
 <span>LegacyUnenumerableNamedProperties</span>]
interface <dfn interface>MimeTypeArray</dfn> {
  readonly attribute unsigned long <span data-x="dom-MimeTypeArray-length">length</span>;
  getter <span>MimeType</span>? <span data-x="dom-MimeTypeArray-item">item</span>(unsigned long index);
  getter <span>MimeType</span>? <span data-x="dom-MimeTypeArray-namedItem">namedItem</span>(DOMString name);
};

[Exposed=Window,
 <span>LegacyUnenumerableNamedProperties</span>]
interface <dfn interface data-x="dom-Plugin">Plugin</dfn> {
  readonly attribute DOMString <span data-x="dom-Plugin-name">name</span>;
  readonly attribute DOMString <span data-x="dom-Plugin-description">description</span>;
  readonly attribute DOMString <span data-x="dom-Plugin-filename">filename</span>;
  readonly attribute unsigned long <span data-x="dom-Plugin-length">length</span>;
  getter <span>MimeType</span>? <span data-x="dom-Plugin-item">item</span>(unsigned long index);
  getter <span>MimeType</span>? <span data-x="dom-Plugin-namedItem">namedItem</span>(DOMString name);
};

[Exposed=Window]
interface <dfn interface>MimeType</dfn> {
  readonly attribute DOMString <span data-x="dom-MimeType-type">type</span>;
  readonly attribute DOMString <span data-x="dom-MimeType-description">description</span>;
  readonly attribute DOMString <span data-x="dom-MimeType-suffixes">suffixes</span>;
  readonly attribute <span>Plugin</span> <span data-x="dom-MimeType-enabledPlugin">enabledPlugin</span>;
};</code></pre>

  <div w-nodev>

  <p>Although these days detecting PDF viewer support can be done via <code
  data-x="dom-navigator-pdfViewerEnabled">navigator.pdfViewerEnabled</code>, for historical reasons,
  there are a number of complex and intertwined interfaces that provide the same capability, which
  legacy code relies on. This section specifies both the simple modern variant and the complicated
  historical one.</p>

  <!-- INSERT TRACKING -->
  <p>Each user agent has a <dfn>PDF viewer supported</dfn> boolean, whose value is
  <span>implementation-defined</span> (and might vary according to user preferences).</p>

  <p class="note">This value also impacts the <span data-x="navigate">navigation</span> processing
  model.</p>

  <hr>

  <p>Each <code>Window</code> object has a <dfn>PDF viewer plugin objects</dfn> list. If the user
  agent's <span>PDF viewer supported</span> is false, then it is the empty list. Otherwise, it is a
  list containing five <code data-x="dom-Plugin">Plugin</code> objects, whose <span
  data-x="concept-Plugin-name">names</span> are, respectively:</p>

  <ol class="brief" start="0">
   <li>"<code data-x="">PDF Viewer</code>"</li>
   <li>"<code data-x="">Chrome PDF Viewer</code>"</li>
   <li>"<code data-x="">Chromium PDF Viewer</code>"</li>
   <li>"<code data-x="">Microsoft Edge PDF Viewer</code>"</li>
   <li>"<code data-x="">WebKit built-in PDF</code>"</li>
  </ol>

  <p>The values of the above list form the <dfn>PDF viewer plugin names</dfn> list.</p>

  <p class="note">These names were chosen based on evidence of what websites historically search
  for, and thus what is necessary for user agents to expose in order to maintain compatibility with
  existing content. They are ordered alphabetically. The "<code data-x="">PDF Viewer</code>" name
  was then inserted in the 0th position so that the <code
  data-x="dom-MimeType-enabledPlugin">enabledPlugin</code> getter could point to a generic plugin
  name.</p>

  <p>Each <code>Window</code> object has a <dfn>PDF viewer mime type objects</dfn> list. If the user
  agent's <span>PDF viewer supported</span> is false, then it is the empty list. Otherwise, it is a
  list containing two <code>MimeType</code> objects, whose <span
  data-x="concept-MimeType-type">types</span> are, respectively:</p>

  <ol class="brief" start="0">
   <li>"<code data-x="">application/pdf</code>"</li>
   <li>"<code data-x="">text/pdf</code>"</li>
  </ol>

  <p>The values of the above list form the <dfn>PDF viewer mime types</dfn> list.</p>

  <hr>

  <p>Each <code>NavigatorPlugins</code> object has a <dfn>plugins array</dfn>, which is a new
  <code>PluginArray</code>, and a <dfn>mime types array</dfn>, which is a new
  <code>MimeTypeArray</code>.</p>

  <p>The <code>NavigatorPlugins</code> mixin's <dfn attribute for="NavigatorPlugins"><code
  data-x="dom-navigator-plugins">plugins</code></dfn> getter steps are to return <span>this</span>'s
  <span>plugins array</span>.</p>

  <p>The <code>NavigatorPlugins</code> mixin's <dfn attribute for="NavigatorPlugins"><code
  data-x="dom-navigator-mimeTypes">mimeTypes</code></dfn> getter steps are to return
  <span>this</span>'s <span>mime types array</span>.</p>

  <p>The <code>NavigatorPlugins</code> mixin's <dfn method for="NavigatorPlugins"><code
  data-x="dom-navigator-javaEnabled">javaEnabled()</code></dfn> method steps are to return false.</p>

  <p>The <code>NavigatorPlugins</code> mixin's <dfn attribute for="NavigatorPlugins"><code
  data-x="dom-navigator-pdfViewerEnabled">pdfViewerEnabled</code></dfn> getter steps are to return
  the user agent's <span>PDF viewer supported</span>.</p>

  <hr>

  <p>The <code>PluginArray</code> interface <span data-x="support named properties">supports named
  properties</span>. If the user agent's <span>PDF viewer supported</span> is true, then they are
  the <span>PDF viewer plugin names</span>. Otherwise, they are the empty list.</p>

  <p>The <code>PluginArray</code> interface's <dfn method for="PluginArray"><code
  data-x="dom-PluginArray-namedItem">namedItem(<var>name</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p><span data-x="list iterate">For each</span> <code data-x="dom-Plugin">Plugin</code>
   <var>plugin</var> of <span>this</span>'s <span>relevant global object</span>'s <span>PDF viewer
   plugin objects</span>: if <var>plugin</var>'s <span data-x="concept-Plugin-name">name</span> is
   <var>name</var>, then return <var>plugin</var>.</p></li>

   <li><p>Return null.</p></li>
  </ol>

  <p>The <code>PluginArray</code> interface <span>supports indexed properties</span>. The
  <span>supported property indices</span> are the <span>indices</span> of <span>this</span>'s
  <span>relevant global object</span>'s <span>PDF viewer plugin objects</span>.</p>

  <p>The <code>PluginArray</code> interface's <dfn method for="PluginArray"><code
  data-x="dom-PluginArray-item">item(<var>index</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>plugins</var> be <span>this</span>'s <span>relevant global object</span>'s
   <span>PDF viewer plugin objects</span>.</p></li>

   <li><p>If <var>index</var> &lt; <var>plugins</var>'s <span data-x="list size">size</span>, then
   return <var>plugins</var>[<var>index</var>].</p></li>

   <li><p>Return null.</p></li>
  </ol>

  <p>The <code>PluginArray</code> interface's <dfn attribute for="PluginArray"><code
  data-x="dom-PluginArray-length">length</code></dfn> getter steps are to return <span>this</span>'s
  <span>relevant global object</span>'s <span>PDF viewer plugin objects</span>'s <span
  data-x="list size">size</span>.</p>

  <p>The <code>PluginArray</code> interface's <dfn method for="PluginArray"><code
  data-x="dom-PluginArray-refresh">refresh()</code></dfn> method steps are to do nothing.</p>

  <hr>

  <p>The <code>MimeTypeArray</code> interface <span data-x="support named properties">supports named
  properties</span>. If the user agent's <span>PDF viewer supported</span> is true, then they are
  the <span>PDF viewer mime types</span>. Otherwise, they are the empty list.</p>

  <p>The <code>MimeTypeArray</code> interface's <dfn method for="MimeTypeArray"><code
  data-x="dom-MimeTypeArray-namedItem">namedItem(<var>name</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p><span data-x="list iterate">For each</span> <code>MimeType</code> <var>mimeType</var> of
   <span>this</span>'s <span>relevant global object</span>'s <span>PDF viewer mime type
   objects</span>: if <var>mimeType</var>'s <span data-x="concept-MimeType-type">type</span> is
   <var>name</var>, then return <var>mimeType</var>.</p></li>

   <li><p>Return null.</p></li>
  </ol>

  <p>The <code>MimeTypeArray</code> interface <span>supports indexed properties</span>. The
  <span>supported property indices</span> are the <span>indices</span> of <span>this</span>'s
  <span>relevant global object</span>'s <span>PDF viewer mime type objects</span>.</p>

  <p>The <code>MimeTypeArray</code> interface's <dfn method for="MimeTypeArray"><code
  data-x="dom-MimeTypeArray-item">item(<var>index</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>mimeTypes</var> be <span>this</span>'s <span>relevant global object</span>'s
   <span>PDF viewer mime type objects</span>.</p></li>

   <li><p>If <var>index</var> &lt; <var>mimeTypes</var>'s <span data-x="list size">size</span>, then
   return <var>mimeTypes</var>[<var>index</var>].</p></li>

   <li><p>Return null.</p></li>
  </ol>

  <p>The <code>MimeTypeArray</code> interface's <dfn attribute for="MimeTypeArray"><code
  data-x="dom-MimeTypeArray-length">length</code></dfn> getter steps are to return
  <span>this</span>'s <span>relevant global object</span>'s <span>PDF viewer mime type
  objects</span>'s <span data-x="list size">size</span>.</p>

  <hr>

  <p>Each <code data-x="dom-Plugin">Plugin</code> object has a <dfn
  data-x="concept-Plugin-name">name</dfn>, which is set when the object is created.</p>

  <p>The <code data-x="dom-Plugin">Plugin</code> interface's <dfn attribute for="Plugin"><code
  data-x="dom-Plugin-name">name</code></dfn> getter steps are to return <span>this</span>'s <span
  data-x="concept-Plugin-name">name</span>.</p>

  <p>The <code data-x="dom-Plugin">Plugin</code> interface's <dfn attribute for="Plugin"><code
  data-x="dom-Plugin-description">description</code></dfn> getter steps are to return "<code
  data-x="">Portable Document Format</code>".</p>

  <p>The <code data-x="dom-Plugin">Plugin</code> interface's <dfn attribute for="Plugin"><code
  data-x="dom-Plugin-filename">filename</code></dfn> getter steps are to return "<code
  data-x="">internal-pdf-viewer</code>".</p>

  <p>The <code data-x="dom-Plugin">Plugin</code> interface <span data-x="support named
  properties">supports named properties</span>. If the user agent's <span>PDF viewer
  supported</span> is true, then they are the <span>PDF viewer mime types</span>. Otherwise, they
  are the empty list.</p>

  <p>The <code data-x="dom-Plugin">Plugin</code> interface's <dfn method for="Plugin"><code
  data-x="dom-Plugin-namedItem">namedItem(<var>name</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p><span data-x="list iterate">For each</span> <code>MimeType</code> <var>mimeType</var> of
   <span>this</span>'s <span>relevant global object</span>'s <span>PDF viewer mime type
   objects</span>: if <var>mimeType</var>'s <span data-x="concept-MimeType-type">type</span> is
   <var>name</var>, then return <var>mimeType</var>.</p></li>

   <li><p>Return null.</p></li>
  </ol>

  <p>The <code data-x="dom-Plugin">Plugin</code> interface <span>supports indexed properties</span>.
  The <span>supported property indices</span> are the <span>indices</span> of <span>this</span>'s
  <span>relevant global object</span>'s <span>PDF viewer mime type objects</span>.</p>

  <p>The <code data-x="dom-Plugin">Plugin</code> interface's <dfn method for="Plugin"><code
  data-x="dom-Plugin-item">item(<var>index</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>mimeTypes</var> be <span>this</span>'s <span>relevant global object</span>'s
   <span>PDF viewer mime type objects</span>.</p></li>

   <li><p>If <var>index</var> &lt; <var>mimeTypes</var>'s <span data-x="list size">size</span>, then
   return <var>mimeTypes</var>[<var>index</var>].</p></li>

   <li><p>Return null.</p></li>
  </ol>

  <p>The <code data-x="dom-Plugin">Plugin</code> interface's <dfn attribute for="Plugin"><code
  data-x="dom-Plugin-length">length</code></dfn> getter steps are to return
  <span>this</span>'s <span>relevant global object</span>'s <span>PDF viewer mime type
  objects</span>'s <span data-x="list size">size</span>.</p>

  <hr>

  <p>Each <code>MimeType</code> object has a <dfn data-x="concept-MimeType-type">type</dfn>, which
  is set when the object is created.</p>

  <p>The <code>MimeType</code> interface's <dfn attribute for="MimeType"><code
  data-x="dom-MimeType-type">type</code></dfn> getter steps are to return <span>this</span>'s <span
  data-x="concept-MimeType-type">type</span>.</p>

  <p>The <code>MimeType</code> interface's <dfn attribute for="MimeType"><code
  data-x="dom-MimeType-description">description</code></dfn> getter steps are to return "<code
  data-x="">Portable Document Format</code>".</p>

  <p>The <code>MimeType</code> interface's <dfn attribute for="MimeType"><code
  data-x="dom-MimeType-suffixes">suffixes</code></dfn> getter steps are to return "<code
  data-x="">pdf</code>".</p>

  <p>The <code>MimeType</code> interface's <dfn attribute for="MimeType"><code
  data-x="dom-MimeType-enabledPlugin">enabledPlugin</code></dfn> getter steps are to return
  <span>this</span>'s <span>relevant global object</span>'s <span>PDF viewer plugin
  objects</span>[0] (i.e., the generic "<code data-x="">PDF Viewer</code>" one).</p>

  </div>


  <h3 split-filename="imagebitmap-and-animations">Images</h3>

  <pre><code class="idl">[Exposed=(Window,Worker), <span>Serializable</span>, <span>Transferable</span>]
interface <dfn interface>ImageBitmap</dfn> {
  readonly attribute unsigned long <span data-x="dom-ImageBitmap-width">width</span>;
  readonly attribute unsigned long <span data-x="dom-ImageBitmap-height">height</span>;
  undefined <span data-x="dom-ImageBitmap-close">close</span>();
};

typedef (<span>CanvasImageSource</span> or
         <span>Blob</span> or
         <span>ImageData</span>) <dfn typedef>ImageBitmapSource</dfn>;

enum <dfn enum>ImageOrientation</dfn> { "<span data-x="dom-ImageOrientation-from-image">from-image</span>", "<span data-x="dom-ImageOrientation-flipY">flipY</span>" };
enum <dfn enum>PremultiplyAlpha</dfn> { "<span data-x="dom-PremultiplyAlpha-none">none</span>", "<span data-x="dom-PremultiplyAlpha-premultiply">premultiply</span>", "<span data-x="dom-PremultiplyAlpha-default">default</span>" };
enum <dfn enum>ColorSpaceConversion</dfn> { "<span data-x="dom-ColorSpaceConversion-none">none</span>", "<span data-x="dom-ColorSpaceConversion-default">default</span>" };
enum <dfn enum>ResizeQuality</dfn> { "<span data-x="dom-ResizeQuality-pixelated">pixelated</span>", "<span data-x="dom-ResizeQuality-low">low</span>", "<span data-x="dom-ResizeQuality-medium">medium</span>", "<span data-x="dom-ResizeQuality-high">high</span>" };

dictionary <dfn dictionary>ImageBitmapOptions</dfn> {
  <span>ImageOrientation</span> <span data-x="dom-ImageBitmapOptions-imageOrientation">imageOrientation</span> = "<span data-x="dom-ImageOrientation-from-image">from-image</span>";
  <span>PremultiplyAlpha</span> <span data-x="dom-ImageBitmapOptions-premultiplyAlpha">premultiplyAlpha</span> = "<span data-x="dom-PremultiplyAlpha-default">default</span>";
  <span>ColorSpaceConversion</span> <span data-x="dom-ImageBitmapOptions-colorSpaceConversion">colorSpaceConversion</span> = "<span data-x="dom-ColorSpaceConversion-default">default</span>";
  [EnforceRange] unsigned long <span data-x="dom-ImageBitmapOptions-resizeWidth">resizeWidth</span>;
  [EnforceRange] unsigned long <span data-x="dom-ImageBitmapOptions-resizeHeight">resizeHeight</span>;
  <span>ResizeQuality</span> <span data-x="dom-ImageBitmapOptions-resizeQuality">resizeQuality</span> = "<span data-x="dom-ResizeQuality-low">low</span>";
};</code></pre>

  <p>An <code>ImageBitmap</code> object represents a bitmap image that can be painted to a canvas
  without undue latency.</p>

  <p class="note">The exact judgement of what is undue latency of this is left up to the
  implementer, but in general if making use of the bitmap requires network I/O, or even local disk
  I/O, then the latency is probably undue; whereas if it only requires a blocking read from a GPU or
  system RAM, the latency is probably acceptable.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>promise</var> = self.<span subdfn data-x="dom-createImageBitmap">createImageBitmap</span>(<var>image</var> [, <var>options</var> ])</code></dt>
   <dt><code data-x=""><var>promise</var> = self.<span data-x="dom-createImageBitmap">createImageBitmap</span>(<var>image</var>, <var>sx</var>, <var>sy</var>, <var>sw</var>, <var>sh</var> [, <var>options</var> ])</code></dt>

   <dd>
    <p>Takes <var>image</var>, which can be an <code>img</code> element, an <span>SVG
    <code>image</code></span> element, a <code>video</code> element, a <code>canvas</code> element,
    a <code>Blob</code> object, an <code>ImageData</code> object, or another
    <code>ImageBitmap</code> object, and returns a promise that is resolved when a new
    <code>ImageBitmap</code> is created.</p>

    <p>If no <code>ImageBitmap</code> object can be constructed, for example because the provided
    <var>image</var> data is not actually an image, then the promise is rejected instead.</p>

    <p>If <var>sx</var>, <var>sy</var>, <var>sw</var>, and <var>sh</var> arguments are provided, the
    source image is cropped to the given pixels, with any pixels missing in the original replaced by
    <span>transparent black</span>. These coordinates are in the source image's pixel coordinate
    space, <em>not</em> in <span data-x="'px'">CSS pixels</span>.</p>

    <p>If <var>options</var> is provided, the <code>ImageBitmap</code> object's bitmap data is
    modified according to <var>options</var>. For example, if the <code
    data-x="dom-ImageBitmapOptions-premultiplyAlpha">premultiplyAlpha</code> option is set to "<code
    data-x="dom-PremultiplyAlpha-premultiply">premultiply</code>", the <span
    data-x="concept-ImageBitmap-bitmap-data">bitmap data</span>'s non-alpha color components are
    <span data-x="concept-premultiplied-alpha">premultiplied by the alpha component</span>.</p>

    <p>Rejects the promise with an <span>"<code>InvalidStateError</code>"</span>
    <code>DOMException</code> if the source image is not in a valid state (e.g., an <code>img</code>
    element that hasn't loaded successfully, an <code>ImageBitmap</code> object whose
    <span>[[Detached]]</span> internal slot value is true, an <code>ImageData</code> object whose
    <code data-x="dom-imagedata-data">data</code> attribute value's [[ViewedArrayBuffer]] internal
    slot is detached, or a <code>Blob</code> whose data cannot be interpreted as a bitmap
    image).</p>

    <p>Rejects the promise with a <span>"<code>SecurityError</code>"</span>
    <code>DOMException</code> if the script is not allowed to access the image data of the source
    image (e.g. a <code>video</code> that is <span>CORS-cross-origin</span>, or a
    <code>canvas</code> being drawn on by a script in a worker from another
    <span>origin</span>).</p>
   </dd>

   <dt><code data-x=""><var>imageBitmap</var>.<span subdfn data-x="dom-ImageBitmap-close">close</span>()</code></dt>

   <dd><p>Releases <var>imageBitmap</var>'s underlying <span
   data-x="concept-ImageBitmap-bitmap-data">bitmap data</span>.</p></dd>

   <dt><code data-x=""><var>imageBitmap</var>.<span subdfn data-x="dom-ImageBitmap-width">width</span></code></dt>

   <dd>
    <p>Returns the <span>natural width</span> of the image, in <span data-x="'px'">CSS
    pixels</span>.</p>
   </dd>

   <dt><code data-x=""><var>imageBitmap</var>.<span subdfn data-x="dom-ImageBitmap-height">height</span></code></dt>

   <dd>
    <p>Returns the <span>natural height</span> of the image, in <span data-x="'px'">CSS
    pixels</span>.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>An <code>ImageBitmap</code> object whose <span>[[Detached]]</span> internal slot value
  is false always has associated <dfn data-x="concept-ImageBitmap-bitmap-data">bitmap data</dfn>,
  with a width and a height. However, it is possible for this data to be corrupted. If an
  <code>ImageBitmap</code> object's media data can be decoded without errors, it is said to be <dfn
  data-x="concept-ImageBitmap-good">fully decodable</dfn>.</p>

  <p>An <code>ImageBitmap</code> object's bitmap has an <span
  data-x="concept-canvas-origin-clean">origin-clean</span> flag, which indicates whether the
  bitmap is tainted by content from a different <span>origin</span>. The flag is initially set to
  true and may be changed to false by the steps of <code
  data-x="dom-createImageBitmap">createImageBitmap()</code>.</p>

  <hr>

  <p><code>ImageBitmap</code> objects are <span>serializable objects</span> and <span>transferable
  objects</span>.</p>

  <p>Their <span>serialization steps</span>, given <var>value</var> and <var>serialized</var>,
  are:</p>

  <ol>
   <li><p>If <var>value</var>'s <span data-x="concept-canvas-origin-clean">origin-clean</span>
   flag is not set, then throw a <span>"<code>DataCloneError</code>"</span>
   <code>DOMException</code>.

   <li><p>Set <var>serialized</var>.[[BitmapData]] to a copy of <var>value</var>'s <span
   data-x="concept-ImageBitmap-bitmap-data">bitmap data</span>.</p></li>
  </ol>

  <p>Their <span>deserialization steps</span>, given <var>serialized</var>, <var>value</var>,
  and <var>targetRealm</var>, are:</p>

  <ol>
   <li><p>Set <var>value</var>'s <span data-x="concept-ImageBitmap-bitmap-data">bitmap data</span>
   to <var>serialized</var>.[[BitmapData]].</p></li>
  </ol>

  <p>Their <span>transfer steps</span>, given <var>value</var> and <var>dataHolder</var>, are:</p>

  <ol>
   <li><p>If <var>value</var>'s <span data-x="concept-canvas-origin-clean">origin-clean</span>
   flag is not set, then throw a <span>"<code>DataCloneError</code>"</span>
   <code>DOMException</code>.

   <li><p>Set <var>dataHolder</var>.[[BitmapData]] to <var>value</var>'s <span
   data-x="concept-ImageBitmap-bitmap-data">bitmap data</span>.</p></li>

   <li><p>Unset <var>value</var>'s <span data-x="concept-ImageBitmap-bitmap-data">bitmap
   data</span>.</p></li>
  </ol>

  <p>Their <span>transfer-receiving steps</span>, given <var>dataHolder</var> and <var>value</var>,
  are:</p>

  <ol>
   <li><p>Set <var>value</var>'s <span data-x="concept-ImageBitmap-bitmap-data">bitmap data</span>
   to <var>dataHolder</var>.[[BitmapData]].</p></li>
  </ol>

  <hr>

  <p>The <dfn method for="WindowOrWorkerGlobalScope"><code
  data-x="dom-createImageBitmap">createImageBitmap(<var>image</var>,
  <var>options</var>)</code></dfn> and <code data-x="">createImageBitmap(<var>image</var>,
  <var>sx</var>, <var>sy</var>, <var>sw</var>, <var>sh</var>, <var>options</var>)</code> methods,
  when invoked, must run these steps:</p>

  <ol>
   <li><p>If either <var>sw</var> or <var>sh</var> is given and is 0, then return <span>a promise
   rejected with</span> a <code data-x="js-RangeError">RangeError</code>.</p></li>

   <li><p>If either <var>options</var>'s <dfn dict-member for="ImageBitmapOptions"><code
   data-x="dom-ImageBitmapOptions-resizeWidth">resizeWidth</code></dfn> or <var>options</var>'s
   <dfn dict-member for="ImageBitmapOptions"><code
   data-x="dom-ImageBitmapOptions-resizeHeight">resizeHeight</code></dfn> is present and is 0, then
   return <span>a promise rejected with</span> an <span>"<code>InvalidStateError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p><span>Check the usability of the <var>image</var> argument</span>. If this throws an
   exception or returns <i>bad</i>, then return <span>a promise rejected with</span> an
   <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>.

   <li><p>Let <var>p</var> be a new promise.</p></li>

   <li><p>Let <var>imageBitmap</var> be a new <code>ImageBitmap</code> object.</p></li>

   <li>
    <p>Switch on <var>image</var>:</p>

    <dl class="switch">
     <dt><code>img</code>
     <dt><span>SVG <code>image</code></span>
     <dd>
      <ol>
       <li><p>If <var>image</var>'s media data has no <span>natural dimensions</span> (e.g., it's
       a vector graphic with no specified content size) and either <var>options</var>'s <code
       data-x="dom-ImageBitmapOptions-resizeWidth">resizeWidth</code> or <var>options</var>'s <code
       data-x="dom-ImageBitmapOptions-resizeHeight">resizeHeight</code> is not present, then return
       <span>a promise rejected with</span> an <span>"<code>InvalidStateError</code>"</span>
       <code>DOMException</code>.</p></li>

       <li><p>If <var>image</var>'s media data has no <span>natural dimensions</span> (e.g., it's
       a vector graphic with no specified content size), it should be rendered to a bitmap of the
       size specified by the <code data-x="dom-ImageBitmapOptions-resizeWidth">resizeWidth</code>
       and the <code data-x="dom-ImageBitmapOptions-resizeHeight">resizeHeight</code>
       options.</p></li>

       <li><p>Set <var>imageBitmap</var>'s <span data-x="concept-ImageBitmap-bitmap-data">bitmap
       data</span> to a copy of <var>image</var>'s media data, <span>cropped to the source rectangle
       with formatting</span>. If this is an animated image, <var>imageBitmap</var>'s <span
       data-x="concept-ImageBitmap-bitmap-data">bitmap data</span> must only be taken from the
       default image of the animation (the one that the format defines is to be used when animation
       is not supported or is disabled), or, if there is no such image, the first frame of the
       animation.</p></li>

       <li><p>If <var>image</var> <span>is not origin-clean</span>, then set the <span
       data-x="concept-canvas-origin-clean">origin-clean</span> flag of <var>imageBitmap</var>'s
       bitmap to false.</p></li>

       <li>
        <p>Run this step <span>in parallel</span>:</p>

        <ol>
         <li><p>Resolve <var>p</var> with <var>imageBitmap</var>.</p></li>
        </ol>
       </li>
      </ol>
     </dd>

     <dt><code>video</code>
     <dd>
      <ol>
       <li><p>If <var>image</var>'s <code data-x="dom-media-networkState">networkState</code>
       attribute is <code data-x="dom-media-NETWORK_EMPTY">NETWORK_EMPTY</code>, then return
       <span>a promise rejected with</span> an <span>"<code>InvalidStateError</code>"</span>
       <code>DOMException</code>.</p></li>

       <li><p>Set <var>imageBitmap</var>'s <span data-x="concept-ImageBitmap-bitmap-data">bitmap
       data</span> to a copy of the frame at the <span>current playback position</span>, at the
       <span>media resource</span>'s <span data-x="concept-video-natural-width">natural
       width</span> and <span data-x="concept-video-natural-height">natural height</span> (i.e.,
       after any aspect-ratio correction has been applied), <span>cropped to the source rectangle
       with formatting</span>.</p>

       <li><p>If <var>image</var> <span>is not origin-clean</span>, then set the <span
       data-x="concept-canvas-origin-clean">origin-clean</span> flag of <var>imageBitmap</var>'s
       bitmap to false.</p></li>

       <li>
        <p>Run this step <span>in parallel</span>:</p>

        <ol>
         <li><p>Resolve <var>p</var> with <var>imageBitmap</var>.</p></li>
        </ol>
       </li>
      </ol>
     </dd>

     <dt><code>canvas</code>
     <dd>
      <ol>
       <li><p>Set <var>imageBitmap</var>'s <span data-x="concept-ImageBitmap-bitmap-data">bitmap
       data</span> to a copy of <var>image</var>'s <span
       data-x="concept-ImageBitmap-bitmap-data">bitmap data</span>, <span>cropped to the source
       rectangle with formatting</span>.</p></li>

       <li><p>Set the <span data-x="concept-canvas-origin-clean">origin-clean</span> flag of the
       <var>imageBitmap</var>'s bitmap to the same value as the <span
       data-x="concept-canvas-origin-clean">origin-clean</span> flag of <var>image</var>'s
       bitmap.</p></li>

       <li>
        <p>Run this step <span>in parallel</span>:</p>

        <ol>
         <li><p>Resolve <var>p</var> with <var>imageBitmap</var>.</p></li>
        </ol>
       </li>
      </ol>
     </dd>

     <dt><code>Blob</code>
     <dd>
      <p>Run these steps <span>in parallel</span>:</p>

      <ol>
       <li><p>Let <var>imageData</var> be the result of reading <var>image</var>'s data. If an <span
       data-x="file-error-read">error occurs during reading of the object</span>, then reject
       <var>p</var> with an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>
       and abort these steps.</p></li>

       <li><p>Apply the <span data-x="Content-Type sniffing: image">image sniffing rules</span> to
       determine the file format of <var>imageData</var>, with MIME type of <var>image</var> (as
       given by <var>image</var>'s <code data-x="dom-Blob-type">type</code> attribute) giving the
       official type.</p></li>

       <li><p>If <var>imageData</var> is not in a supported image file format (e.g., it's not an
       image at all), or if <var>imageData</var> is corrupted in some fatal way such that the image
       dimensions cannot be obtained (e.g., a vector graphic with no natural size), then reject
       <var>p</var> with an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>
       and abort these steps.</p></li>

       <li><p>Set <var>imageBitmap</var>'s <span data-x="concept-ImageBitmap-bitmap-data">bitmap
       data</span> to <var>imageData</var>, <span>cropped to the source rectangle with
       formatting</span>. If this is an animated image, <var>imageBitmap</var>'s <span
       data-x="concept-ImageBitmap-bitmap-data">bitmap data</span> must only be taken from the
       default image of the animation (the one that the format defines is to be used when animation
       is not supported or is disabled), or, if there is no such image, the first frame of the
       animation.</p></li>

       <li><p>Resolve <var>p</var> with <var>imageBitmap</var>.</p></li>
      </ol>
     </dd>

     <dt><code>ImageData</code>
     <dd>
      <ol>
       <li><p>Let <var>buffer</var> be <var>image</var>'s <code
       data-x="dom-imagedata-data">data</code> attribute value's [[ViewedArrayBuffer]] internal
       slot.</p></li>

       <li><p>If <span>IsDetachedBuffer</span>(<var>buffer</var>) is true, then return <span>a
       promise rejected with</span> an <span>"<code>InvalidStateError</code>"</span>
       <code>DOMException</code>.</p></li>

       <li><p>Set <var>imageBitmap</var>'s <span data-x="concept-ImageBitmap-bitmap-data">bitmap
       data</span> to <var>image</var>'s image data, <span>cropped to the source rectangle with
       formatting</span>.</p></li>

       <li>
        <p>Run this step <span>in parallel</span>:</p>

        <ol>
         <li><p>Resolve <var>p</var> with <var>imageBitmap</var>.</p></li>
        </ol>
       </li>
      </ol>
     </dd>

     <dt><code>ImageBitmap</code>
     <dd>
      <ol>
       <li><p>Set <var>imageBitmap</var>'s <span data-x="concept-ImageBitmap-bitmap-data">bitmap
       data</span> to a copy of <var>image</var>'s <span
       data-x="concept-ImageBitmap-bitmap-data">bitmap data</span>, <span>cropped to the source
       rectangle with formatting</span>.</p></li>

       <li><p>Set the <span data-x="concept-canvas-origin-clean">origin-clean</span> flag of
       <var>imageBitmap</var>'s bitmap to the same value as the <span
       data-x="concept-canvas-origin-clean">origin-clean</span> flag of <var>image</var>'s
       bitmap.</p></li>

       <li>
        <p>Run this step <span>in parallel</span>:</p>

        <ol>
         <li><p>Resolve <var>p</var> with <var>imageBitmap</var>.</p></li>
        </ol>
       </li>
      </ol>
     </dd>

     <dt><code>VideoFrame</code>
     <dd>
      <ol>
       <li><p>Set <var>imageBitmap</var>'s <span data-x="concept-ImageBitmap-bitmap-data">bitmap
       data</span> to a copy of <var>image</var>'s visible pixel data, <span>cropped to the source
       rectangle with formatting</span>.</p></li>

       <li>
        <p>Run this step <span>in parallel</span>:</p>

        <ol>
         <li><p>Resolve <var>p</var> with <var>imageBitmap</var>.</p></li>
        </ol>
       </li>
      </ol>
     </dd>
    </dl>
   </li>

   <li><p>Return <var>p</var>.</p></li>
  </ol>

  <p>When the steps above require that the user agent <dfn data-x="cropped to the source
  rectangle with formatting">crop bitmap data to the source rectangle with formatting</dfn>,
  the user agent must run the following steps:</p>

  <ol>
   <li><p>Let <var>input</var> be the <span data-x="concept-ImageBitmap-bitmap-data">bitmap
   data</span> being transformed.</p></li>

   <li>
    <p>If <var>sx</var>, <var>sy</var>, <var>sw</var> and <var>sh</var> are specified, let
    <var>sourceRectangle</var> be a rectangle whose corners are the four points (<var>sx</var>,
    <var>sy</var>), (<var>sx</var>+<var>sw</var>, <var>sy</var>), (<var>sx</var>+<var>sw</var>,
    <var>sy</var>+<var>sh</var>), (<var>sx</var>, <var>sy</var>+<var>sh</var>). Otherwise, let
    <var>sourceRectangle</var> be a rectangle whose corners are the four points (0, 0), (width of
    <var>input</var>, 0), (width of <var>input</var>, height of <var>input</var>), (0, height of
    <var>input</var>).</p>

    <p class="note">If either <var>sw</var> or <var>sh</var> are negative, then
    the top-left corner of this rectangle will be to the left or above the (<var>sx</var>,
    <var>sy</var>) point.</p>
   </li>

   <li>
    <p>Let <var>outputWidth</var> be determined as follows:</p>

    <dl class="switch">
     <dt>If the <code data-x="dom-ImageBitmapOptions-resizeWidth">resizeWidth</code> member of
     <var>options</var> is specified</dt>
     <dd>the value of the <code data-x="dom-ImageBitmapOptions-resizeWidth">resizeWidth</code>
     member of <var>options</var></dd>

     <dt>If the <code data-x="dom-ImageBitmapOptions-resizeWidth">resizeWidth</code> member of
     <var>options</var> is not specified, but the <code
     data-x="dom-ImageBitmapOptions-resizeHeight">resizeHeight</code> member is specified</dt>
     <dd>the width of <var>sourceRectangle</var>, times the value of the <code
     data-x="dom-ImageBitmapOptions-resizeHeight">resizeHeight</code> member of <var>options</var>,
     divided by the height of <var>sourceRectangle</var>, rounded up to the nearest integer</dd>

     <dt>If neither <code data-x="dom-ImageBitmapOptions-resizeWidth">resizeWidth</code> nor <code
     data-x="dom-ImageBitmapOptions-resizeHeight">resizeHeight</code> are specified</dt>
     <dd>the width of <var>sourceRectangle</var></dd>
    </dl>
   </li>

   <li>
    <p>Let <var>outputHeight</var> be determined as follows:</p>

    <dl class="switch">
     <dt>If the <code data-x="dom-ImageBitmapOptions-resizeHeight">resizeHeight</code> member of
     <var>options</var> is specified</dt>
     <dd>the value of the <code data-x="dom-ImageBitmapOptions-resizeHeight">resizeHeight</code>
     member of <var>options</var></dd>

     <dt>If the <code data-x="dom-ImageBitmapOptions-resizeHeight">resizeHeight</code> member of
     <var>options</var> is not specified, but the <code
     data-x="dom-ImageBitmapOptions-resizeWidth">resizeWidth</code> member is specified</dt>
     <dd>the height of <var>sourceRectangle</var>, times the value of the <code
     data-x="dom-ImageBitmapOptions-resizeWidth">resizeWidth</code> member of <var>options</var>,
     divided by the width of <var>sourceRectangle</var>, rounded up to the nearest integer</dd>

     <dt>If neither <code data-x="dom-ImageBitmapOptions-resizeWidth">resizeWidth</code> nor <code
     data-x="dom-ImageBitmapOptions-resizeHeight">resizeHeight</code> are specified</dt>
     <dd>the height of <var>sourceRectangle</var></dd>
    </dl>
   </li>

   <li><p>Place <var>input</var> on an infinite <span>transparent black</span> grid plane,
   positioned so that its top left corner is at the origin of the plane, with the
   <var>x</var>-coordinate increasing to the right, and the <var>y</var>-coordinate increasing down,
   and with each pixel in the <var>input</var> image data occupying a cell on the plane's
   grid.</p></li>

   <li><p>Let <var>output</var> be the rectangle on the plane denoted by
   <var>sourceRectangle</var>.</p></li>

   <li><p>Scale <var>output</var> to the size specified by <var>outputWidth</var> and
   <var>outputHeight</var>. The user agent should use the value of the <dfn dict-member
   for="ImageBitmapOptions"><code
   data-x="dom-ImageBitmapOptions-resizeQuality">resizeQuality</code></dfn> option to guide the
   choice of scaling algorithm.</p></li>

   <li>
    <p>If the value of the <dfn dict-member for="ImageBitmapOptions"><code
    data-x="dom-ImageBitmapOptions-imageOrientation">imageOrientation</code></dfn> member of
    <var>options</var> is "<dfn enum-value for="ImageOrientation"><code
    data-x="dom-ImageOrientation-flipY">flipY</code></dfn>", <var>output</var> must be flipped
    vertically, disregarding any image orientation metadata of the source (such as EXIF metadata),
    if any. <ref>EXIF</ref></p>

    <p class="note">If the value is "<dfn enum-value for="ImageOrientation"><code
    data-x="dom-ImageOrientation-from-image">from-image</code></dfn>", no extra step is needed.</p>

    <p class="note">There used to be a "<dfn><code
    data-x="dom-ImageBitmapOptions-imageOrientation-none">none</code></dfn>" enum value. It was
    renamed to "<code data-x="dom-ImageOrientation-from-image">from-image</code>". In the future,
    "<code data-x="dom-ImageBitmapOptions-imageOrientation-none">none</code>" will be added back
    with a different meaning.</p>
   </li>

   <li>
    <p>If <var>image</var> is an <code>img</code> element or a <code>Blob</code> object, let
    <var>val</var> be the value of the <dfn dict-member for="ImageBitmapOptions"><code
    data-x="dom-ImageBitmapOptions-colorSpaceConversion">colorSpaceConversion</code></dfn> member
    of <var>options</var>, and then run these substeps:</p>

    <ol>
     <li><p>If <var>val</var> is "<dfn enum-value for="ColorSpaceConversion"><code
     data-x="dom-ColorSpaceConversion-default">default</code></dfn>",
     the color space conversion behavior is implementation-specific, and should be chosen according
     to the default color space that the implementation uses for drawing images onto the canvas.</p>
     </li>

     <li><p>If <var>val</var> is "<dfn enum-value for="ColorSpaceConversion"><code
     data-x="dom-ColorSpaceConversion-none">none</code></dfn>", <var>output</var> must be decoded
     without performing any color space conversions. This means that the image decoding algorithm
     must ignore color profile metadata embedded in the source data as well as the display device
     color profile.</p></li>
    </ol>
   </li>

   <li>
    <p>Let <var>val</var> be the value of <dfn dict-member for="ImageBitmapOptions"><code
    data-x="dom-ImageBitmapOptions-premultiplyAlpha">premultiplyAlpha</code></dfn> member of
    <var>options</var>, and then run these substeps:</p>

    <ol>
     <li><p>If <var>val</var> is "<dfn enum-value for="PremultiplyAlpha"><code
     data-x="dom-PremultiplyAlpha-default">default</code></dfn>", the alpha premultiplication
     behavior is implementation-specific, and should be chosen according to implementation deems
     optimal for drawing images onto the canvas.</p></li>

     <li><p>If <var>val</var> is "<dfn enum-value for="PremultiplyAlpha"><code
     data-x="dom-PremultiplyAlpha-premultiply">premultiply</code></dfn>", the <var>output</var> that
     is not premultiplied by alpha must have its color components <span
     data-x="convert-to-premultiplied">multiplied by alpha</span> and that is premultiplied by alpha
     must be left untouched.</p></li>

     <li><p>If <var>val</var> is "<dfn enum-value for="PremultiplyAlpha"><code
     data-x="dom-PremultiplyAlpha-none">none</code></dfn>", the <var>output</var> that is not
     premultiplied by alpha must be left untouched and that is premultiplied by alpha must have its
     color components <span data-x="convert-from-premultiplied">divided by alpha</span>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>output</var>.</p></li>
  </ol>

  <p>The <dfn method for="ImageBitmap"><code data-x="dom-ImageBitmap-close">close()</code></dfn>
  method steps are:</p>

  <ol>
   <li><p>Set <span>this</span>'s <span>[[Detached]]</span> internal slot value to true.</p></li>

   <li><p>Unset <span>this</span>'s <span data-x="concept-ImageBitmap-bitmap-data">bitmap
   data</span>.</p></li>
  </ol>

  <p>The <dfn attribute for="ImageBitmap"><code data-x="dom-ImageBitmap-width">width</code></dfn>
  getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>[[Detached]]</span> internal slot's value is true, then
   return 0.</p></li>

   <li><p>Return <span>this</span>'s width, in <span data-x="'px'">CSS pixels</span>.</p></li>
  </ol>

  <p>The <dfn attribute for="ImageBitmap"><code data-x="dom-ImageBitmap-height">height</code></dfn>
  getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span>[[Detached]]</span> internal slot's value is true, then
   return 0.</p></li>

   <li><p>Return <span>this</span>'s height, in <span data-x="'px'">CSS pixels</span>.</p></li>
  </ol>

  <p>The <code>ResizeQuality</code> enumeration is used to express a preference for the
  interpolation quality to use when scaling images.</p>

  <p>The "<dfn enum-value for="ResizeQuality"><code
  data-x="dom-ResizeQuality-pixelated">pixelated</code></dfn>" value indicates a preference for
  scaling the image to preserve the pixelation of the original as much as possible, with minor
  smoothing as necessary to avoid distorting the image when the target size is not a clean multiple
  of the original.</p>

  <p>To implement "<code data-x="dom-ResizeQuality-pixelated">pixelated</code>", for each axis
  independently, first determine the integer multiple of its natural size that puts it closest to
  the target size and is greater than zero. Scale it to this integer-multiple-size using nearest
  neighbor, then scale it the rest of the way to the target size using bilinear interpolation.</p>

  <p>The "<dfn enum-value for="ResizeQuality"><code data-x="dom-ResizeQuality-low">low</code></dfn>"
  value indicates a preference for a low level of image interpolation quality. Low-quality image
  interpolation may be more computationally efficient than higher settings.</p>

  <p>The "<dfn enum-value for="ResizeQuality"><code
  data-x="dom-ResizeQuality-medium">medium</code></dfn>" value indicates a preference for a medium
  level of image interpolation quality.</p>

  <p>The "<dfn enum-value for="ResizeQuality"><code
  data-x="dom-ResizeQuality-high">high</code></dfn>" value indicates a preference for a high level
  of image interpolation quality. High-quality image interpolation may be more computationally
  expensive than lower settings.</p>

  <p class="note">Bilinear scaling is an example of a relatively fast, lower-quality image-smoothing
  algorithm. Bicubic or Lanczos scaling are examples of image-scaling algorithms that produce
  higher-quality output. This specification does not mandate that specific interpolation algorithms
  be used, except for "<code data-x="dom-ResizeQuality-pixelated">pixelated</code>" as described
  above.</p>

  </div>

  <div class="example">

   <p>Using this API, a sprite sheet can be precut and prepared:</p>

   <pre><code class="js">var sprites = {};
function loadMySprites() {
  var image = new Image();
  image.src = 'mysprites.png';
  var resolver;
  var promise = new Promise(function (arg) { resolver = arg });
  image.onload = function () {
    resolver(Promise.all([
      createImageBitmap(image,  0,  0, 40, 40).then(function (image) { sprites.person = image }),
      createImageBitmap(image, 40,  0, 40, 40).then(function (image) { sprites.grass  = image }),
      createImageBitmap(image, 80,  0, 40, 40).then(function (image) { sprites.tree   = image }),
      createImageBitmap(image,  0, 40, 40, 40).then(function (image) { sprites.hut    = image }),
      createImageBitmap(image, 40, 40, 40, 40).then(function (image) { sprites.apple  = image }),
      createImageBitmap(image, 80, 40, 40, 40).then(function (image) { sprites.snake  = image })
    ]));
  };
  return promise;
}

function runDemo() {
  var canvas = document.querySelector('canvas#demo');
  var context = canvas.getContext('2d');
  context.drawImage(sprites.tree, 30, 10);
  context.drawImage(sprites.snake, 70, 10);
}

loadMySprites().then(runDemo);</code></pre>

  </div>



  <h3>Animation frames</h3>

  <p>Some objects include the <code>AnimationFrameProvider</code> interface mixin.</p>

  <pre><code class="idl">callback <dfn callback>FrameRequestCallback</dfn> = undefined (<span>DOMHighResTimeStamp</span> time);

interface mixin <dfn interface>AnimationFrameProvider</dfn> {
  unsigned long <span data-x="dom-AnimationFrameProvider-requestAnimationFrame">requestAnimationFrame</span>(<span>FrameRequestCallback</span> callback);
  undefined <span data-x="AnimationFrameProvider-cancelAnimationFrame">cancelAnimationFrame</span>(unsigned long handle);
};
<span>Window</span> includes <span>AnimationFrameProvider</span>;
<span>DedicatedWorkerGlobalScope</span> includes <span>AnimationFrameProvider</span>;</code></pre>

  <p>Each <code>AnimationFrameProvider</code> object also has a <dfn
  data-x="concept-AnimationFrameProvider-target-object">target object</dfn> that stores the
  provider's internal state. It is defined as follows:</p>

  <dl>
   <dt>If the <code>AnimationFrameProvider</code> is a <code>Window</code></dt>
   <dd>The <code>Window</code>'s <span data-x="concept-document-window">associated
   <code>Document</code></span></dd>

   <dt>If the <code>AnimationFrameProvider</code> is a <code>DedicatedWorkerGlobalScope</code></dt>
   <dd>The <code>DedicatedWorkerGlobalScope</code></dd>
  </dl>

  <p>Each <span data-x="concept-AnimationFrameProvider-target-object">target object</span> has a
  <dfn id="list-of-animation-frame-callbacks">map of animation frame callbacks</dfn>, which is an
  <span>ordered map</span> that must be initially empty, and an <dfn>animation frame callback
  identifier</dfn>, which is a number that must initially be zero.</p>

  <p>An <code>AnimationFrameProvider</code> <var>provider</var> is considered <dfn
  data-x="concept-AnimationFrameProvider-supported">supported</dfn> if any of the following are
  true:</p>

  <ul>
   <li><p><var>provider</var> is a <code>Window</code>.</p></li>

   <li><p><var>provider</var>'s <span>owner set</span> <span data-x="list contains">contains</span>
   a <code>Document</code> object.</p></li>

   <li><p>Any of the <code>DedicatedWorkerGlobalScope</code> objects in <var>provider</var>'s
   <span>owner set</span> are <span
   data-x="concept-AnimationFrameProvider-supported">supported</span>.</p></li>
  </ul>

  <hr>

  <p>The <dfn method for="AnimationFrameProvider"><code
  data-x="dom-AnimationFrameProvider-requestAnimationFrame">requestAnimationFrame(<var>callback</var>)</code></dfn>
  method steps are:</p>

  <ol>
   <li><p>If <span>this</span> is not <span
   data-x="concept-AnimationFrameProvider-supported">supported</span>, then throw a
   <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>target</var> be <span>this</span>'s <span
   data-x="concept-AnimationFrameProvider-target-object">target object</span>.</p></li>

   <li><p>Increment <var>target</var>'s <span>animation frame callback identifier</span> by one,
   and let <var>handle</var> be the result.</p></li>

   <li><p>Let <var>callbacks</var> be <var>target</var>'s <span>map of animation frame
   callbacks</span>.</p></li>

   <li><p><span data-x="map set">Set</span> <var>callbacks</var>[<var>handle</var>] to
   <var>callback</var>.</p></li>

   <li><p>Return <var>handle</var>.</p></li>
  </ol>

  <p>The <dfn method for="AnimationFrameProvider"><code
  data-x="AnimationFrameProvider-cancelAnimationFrame">cancelAnimationFrame(<var>handle</var>)</code></dfn>
  method steps are:</p>

  <ol>
   <li><p>If <span>this</span> is not <span
   data-x="concept-AnimationFrameProvider-supported">supported</span>, then throw a
   <span>"<code>NotSupportedError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>callbacks</var> be <span>this</span>'s <span
   data-x="concept-AnimationFrameProvider-target-object">target object</span>'s <span>map of
   animation frame callbacks</span>.</p></li>

   <li><p><span data-x="map remove">Remove</span> <var>callbacks</var>[<var>handle</var>].</p></li>
  </ol>

  <p>To <dfn>run the animation frame callbacks</dfn> for a <span
  data-x="concept-AnimationFrameProvider-target-object">target object</span> <var>target</var> with
  a timestamp <var>now</var>:</p>

  <ol>
   <li><p>Let <var>callbacks</var> be <var>target</var>'s <span>map of animation frame
   callbacks</span>.</p></li>

   <li><p>Let <var>callbackHandles</var> be the result of <span data-x="map get the keys">getting
   the keys</span> of <var>callbacks</var>.</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>handle</var> in
    <var>callbackHandles</var>, if <var>handle</var> <span data-x="map exists">exists</span> in
    <var>callbacks</var>:</p>

    <ol>
     <li><p>Let <var>callback</var> be <var>callbacks</var>[<var>handle</var>].</p></li>

     <li><p><span data-x="map remove">Remove</span>
     <var>callbacks</var>[<var>handle</var>].</p></li>

     <li><p><span data-x="es-invoking-callback-functions">Invoke</span> <var>callback</var> with «
     <var>now</var> » and "<code data-x="">report</code>".</p></li>
    </ol>
   </li>
  </ol>

  <div class="example">
   <p>Inside workers, <code
   data-x="dom-AnimationFrameProvider-requestAnimationFrame">requestAnimationFrame()</code> can be
   used together with an <code>OffscreenCanvas</code> transferred from a <code>canvas</code>
   element. First, in the document, transfer control to the worker:</p>

   <pre><code class="js">const offscreenCanvas = document.getElementById("c").transferControlToOffscreen();
worker.postMessage(offscreenCanvas, [offscreenCanvas]);</code></pre>

   <p>Then, in the worker, the following code will draw a rectangle moving from left to right:</p>

   <pre><code class="js">let ctx, pos = 0;
function draw(dt) {
  ctx.clearRect(0, 0, 100, 100);
  ctx.fillRect(pos, 0, 10, 10);
  pos += 10 * dt;
  requestAnimationFrame(draw);
}

self.onmessage = function(ev) {
  const transferredCanvas = ev.data;
  ctx = transferredCanvas.getContext("2d");
  draw();
};</code></pre>
  </div>

  <h2 split-filename="comms" id="comms">Communication</h2>

  <p class="note" id="network"><span id="network-intro"></span><span
  id="the-websocket-interface"></span><span id="feedback-from-the-protocol"></span><span
  id="ping-and-pong-frames"></span><span id="the-closeevent-interface"></span><span
  id="garbage-collection-2"></span><span id="websocket"></span><span id="binarytype"></span><span
  id="closeevent"></span><span id="closeeventinit"></span>The <code>WebSocket</code> interface used
  to be defined here. It is now defined in <cite>WebSockets</cite>. <ref>WEBSOCKETS</ref></p>

  <h3>The <code>MessageEvent</code> interface</h3>

  <p>Messages in <span>server-sent events</span>, <span>cross-document messaging</span>,
  <span>channel messaging</span>, <span>broadcast channels</span>, and <cite>WebSockets</cite> use
  the <code>MessageEvent</code> interface for their <code data-x="event-message">message</code>
  events: <ref>WEBSOCKETS</ref></p>

  <pre><code class="idl">[Exposed=(Window,Worker,AudioWorklet)]
interface <dfn interface>MessageEvent</dfn> : <span>Event</span> {
  <span data-x="dom-Event-constructor">constructor</span>(DOMString type, optional <span>MessageEventInit</span> eventInitDict = {});

  readonly attribute any <span data-x="dom-MessageEvent-data">data</span>;
  readonly attribute USVString <span data-x="dom-MessageEvent-origin">origin</span>;
  readonly attribute DOMString <span data-x="dom-MessageEvent-lastEventId">lastEventId</span>;
  readonly attribute <span>MessageEventSource</span>? <span data-x="dom-MessageEvent-source">source</span>;
  readonly attribute FrozenArray&lt;<span>MessagePort</span>&gt; <span data-x="dom-MessageEvent-ports">ports</span>;

  undefined <span data-x="dom-MessageEvent-initMessageEvent">initMessageEvent</span>(DOMString type, optional boolean bubbles = false, optional boolean cancelable = false, optional any data = null, optional USVString origin = "", optional DOMString lastEventId = "", optional <span>MessageEventSource</span>? source = null, optional sequence&lt;<span>MessagePort</span>&gt; ports = []);
};

dictionary <dfn dictionary>MessageEventInit</dfn> : <span>EventInit</span> {
  any <dfn dict-member for="MessageEventInit" data-x="dom-MessageEventInit-data">data</dfn> = null;
  USVString <dfn dict-member for="MessageEventInit" data-x="dom-MessageEventInit-origin">origin</dfn> = "";
  DOMString <dfn dict-member for="MessageEventInit" data-x="dom-MessageEventInit-lastEventId">lastEventId</dfn> = "";
  <span>MessageEventSource</span>? <dfn dict-member for="MessageEventInit" data-x="dom-MessageEventInit-source">source</dfn> = null;
  sequence&lt;<span>MessagePort</span>&gt; <dfn dict-member for="MessageEventInit" data-x="dom-MessageEventInit-ports">ports</dfn> = [];
};

typedef (<span>WindowProxy</span> or <span>MessagePort</span> or <span>ServiceWorker</span>) <dfn typedef>MessageEventSource</dfn>;</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-MessageEvent-data">data</span></code></dt>

   <dd><p>Returns the data of the message.</p></dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-MessageEvent-origin">origin</span></code></dt>

   <dd>
    <p>Returns the origin of the message, for <span>server-sent events</span> and
    <span>cross-document messaging</span>.</p>
   </dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-MessageEvent-lastEventId">lastEventId</span></code></dt>

   <dd>
    <p>Returns the <span data-x="concept-event-stream-last-event-id">last event ID string</span>,
    for <span>server-sent events</span>.</p>
   </dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-MessageEvent-source">source</span></code></dt>

   <dd>
    <p>Returns the <code>WindowProxy</code> of the source window, for <span>cross-document
    messaging</span>, and the <code>MessagePort</code> being attached, in the <code
    data-x="event-WorkerGlobalScope-connect">connect</code> event fired at
    <code>SharedWorkerGlobalScope</code> objects.</p>
   </dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-MessageEvent-ports">ports</span></code></dt>

   <dd>
    <p>Returns the <code>MessagePort</code> array sent with the message, for <span>cross-document
    messaging</span> and <span>channel messaging</span>.</p>
   </dd>

   <!-- initMessageEvent is deprecated -->
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="MessageEvent"><code data-x="dom-MessageEvent-data">data</code></dfn>
  attribute must return the value it was initialized to. It represents the message being sent.</p>

  <p>The <dfn attribute for="MessageEvent"><code
  data-x="dom-MessageEvent-origin">origin</code></dfn> attribute must return the value it was
  initialized to. It represents, in <span>server-sent events</span> and <span>cross-document
  messaging</span>, the <span data-x="concept-document-origin">origin</span> of the document that
  sent the message (typically the scheme, hostname, and port of the document, but not its path or
  <span data-x="concept-url-fragment">fragment</span>).</p>

  <p>The <dfn attribute for="MessageEvent"><code
  data-x="dom-MessageEvent-lastEventId">lastEventId</code></dfn> attribute must return the value it
  was initialized to. It represents, in <span>server-sent events</span>, the <span
  data-x="concept-event-stream-last-event-id">last event ID string</span> of the event source.</p>

  <p>The <dfn attribute for="MessageEvent"><code
  data-x="dom-MessageEvent-source">source</code></dfn> attribute must return the value it was
  initialized to. It represents, in <span>cross-document messaging</span>, the
  <code>WindowProxy</code> of the <span>browsing context</span> of the <code>Window</code> object
  from which the message came; and in the <code
  data-x="event-WorkerGlobalScope-connect">connect</code> events used by <span
  data-x="SharedWorkerGlobalScope">shared workers</span>, the newly connecting
  <code>MessagePort</code>.</p>

  <p>The <dfn attribute for="MessageEvent"><code data-x="dom-MessageEvent-ports">ports</code></dfn>
  attribute must return the value it was initialized to. It represents, in <span>cross-document
  messaging</span> and <span>channel messaging</span>, the <code>MessagePort</code> array being
  sent.</p>

  <!-- handwavy, file bugs if there are interop issues -->
  <p>The <dfn method for="MessageEvent"><code
  data-x="dom-MessageEvent-initMessageEvent">initMessageEvent(<var>type</var>, <var>bubbles</var>,
  <var>cancelable</var>, <var>data</var>, <var>origin</var>, <var>lastEventId</var>,
  <var>source</var>, <var>ports</var>)</code></dfn> method must initialize the event in a manner
  analogous to the similarly-named <code data-x="dom-Event-initEvent">initEvent()</code> method.
  <ref>DOM</ref></p>

  </div>

  <p class="note">Various APIs (e.g., <code>WebSocket</code>, <code>EventSource</code>) use the
  <code>MessageEvent</code> interface for their <code data-x="event-message">message</code> event
  without using the <code>MessagePort</code> API.</p>


  <h3 split-filename="server-sent-events" id="server-sent-events"><dfn>Server-sent events</dfn></h3>

  <h4 id="server-sent-events-intro">Introduction</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>To enable servers to push data to web pages over HTTP or using dedicated server-push protocols,
  this specification introduces the <code>EventSource</code> interface.</p>

  <p>Using this API consists of creating an <code>EventSource</code> object and registering an event
  listener.</p>

  <pre><code class="js">var source = new EventSource('updates.cgi');
source.onmessage = function (event) {
  alert(event.data);
};</code></pre>

  <p>On the server-side, the script ("<code data-x="">updates.cgi</code>" in this case) sends
  messages in the following form, with the <code>text/event-stream</code> MIME type:</p>

  <pre>data: This is the first message.

data: This is the second message, it
data: has two lines.

data: This is the third message.</pre>

  <hr>

  <p>Authors can separate events by using different event types. Here is a stream that has two event
  types, "add" and "remove":</p>

  <pre>event: add
data: 73857293

event: remove
data: 2153

event: add
data: 113411</pre>

  <p>The script to handle such a stream would look like this (where <code data-x="">addHandler</code>
  and <code data-x="">removeHandler</code> are functions that take one argument, the event):</p>

  <pre><code class="js">var source = new EventSource('updates.cgi');
source.addEventListener('add', addHandler, false);
source.addEventListener('remove', removeHandler, false);</code></pre>

  <p>The default event type is "message".</p>

  <p>Event streams are always decoded as UTF-8. There is no way to specify another character
  encoding.</p>

  <hr>

  <p>Event stream requests can be redirected using HTTP 301 and 307 redirects as with normal HTTP
  requests. Clients will reconnect if the connection is closed; a client can be told to stop
  reconnecting using the HTTP 204 No Content response code.</p>

  <p>Using this API rather than emulating it using <code>XMLHttpRequest</code> or an
  <code>iframe</code> allows the user agent to make better use of network resources in cases where
  the user agent implementer and the network operator are able to coordinate in advance. Amongst
  other benefits, this can result in significant savings in battery life on portable devices. This
  is discussed further in the section below on <a href="#eventsource-push">connectionless
  push</a>.</p>


  <h4>The <code>EventSource</code> interface</h4>

  <pre><code class="idl">[Exposed=(Window,Worker)]
interface <dfn interface>EventSource</dfn> : <span>EventTarget</span> {
  <span data-x="dom-EventSource">constructor</span>(USVString url, optional <span>EventSourceInit</span> eventSourceInitDict = {});

  readonly attribute USVString <span data-x="dom-EventSource-url">url</span>;
  readonly attribute boolean <span data-x="dom-EventSource-withCredentials">withCredentials</span>;

  // ready state
  const unsigned short <span data-x="dom-EventSource-CONNECTING">CONNECTING</span> = 0;
  const unsigned short <span data-x="dom-EventSource-OPEN">OPEN</span> = 1;
  const unsigned short <span data-x="dom-EventSource-CLOSED">CLOSED</span> = 2;
  readonly attribute unsigned short <span data-x="dom-EventSource-readyState">readyState</span>;

  // networking
  attribute <span>EventHandler</span> <span data-x="handler-EventSource-onopen">onopen</span>;
  attribute <span>EventHandler</span> <span data-x="handler-EventSource-onmessage">onmessage</span>;
  attribute <span>EventHandler</span> <span data-x="handler-EventSource-onerror">onerror</span>;
  undefined <span data-x="dom-EventSource-close">close</span>();
};

dictionary <dfn dictionary>EventSourceInit</dfn> {
  boolean <dfn dict-member for="EventSourceInit" data-x="dom-EventSourceInit-withCredentials">withCredentials</dfn> = false;
};</code></pre>

  <div w-nodev>

  <p>Each <code>EventSource</code> object has the following associated with it:</p>

  <ul>
   <li><p>A <dfn data-x="concept-EventSource-url" for="EventSource">url</dfn> (a <span>URL
   record</span>). Set during construction.</p></li>

   <li><p>A <dfn data-x="concept-event-stream-request">request</dfn>. This must initially be
   null.</p></li>

   <li><p>A <dfn data-x="concept-event-stream-reconnection-time">reconnection time</dfn>, in
   milliseconds. This must initially be an <span>implementation-defined</span> value, probably in
   the region of a few seconds.</p></li>

   <li><p>A <dfn data-x="concept-event-stream-last-event-id">last event ID string</dfn>. This must
   initially be the empty string.</p></li>
  </ul>

  <p>Apart from <span data-x="concept-EventSource-url">url</span> these are not currently exposed on
  the <code>EventSource</code> object.</p>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>source</var> = new <span subdfn data-x="dom-EventSource">EventSource</span>(
   <var>url</var> [, { <span data-x="dom-EventSourceInit-withCredentials">withCredentials</span>:
   true } ])</code></dt>
   <dd>
    <p>Creates a new <code>EventSource</code> object.</p>

    <p><var>url</var> is a string giving the <span>URL</span> that will provide the event
    stream.</p>

    <p>Setting <code data-x="dom-EventSourceInit-withCredentials">withCredentials</code> to true
    will set the <span data-x="concept-request-credentials-mode">credentials mode</span> for
    connection requests to <var>url</var> to "<code data-x="">include</code>".</p>
   </dd>

   <dt><code data-x=""><var>source</var>.<span subdfn data-x="dom-EventSource-close">close</span>()</code></dt>

   <dd>
    <p>Aborts any instances of the <span data-x="concept-fetch">fetch</span> algorithm started for
    this <code>EventSource</code> object, and sets the <code
    data-x="dom-EventSource-readyState">readyState</code> attribute to <code
    data-x="dom-EventSource-CLOSED">CLOSED</code>.</p>
   </dd>

   <dt><code data-x=""><var>source</var>.<span subdfn data-x="dom-EventSource-url">url</span></code></dt>

   <dd><p>Returns the <span data-x="concept-EventSource-url">URL providing the event
   stream</span>.</p></dd>

   <dt><code data-x=""><var>source</var>.<span subdfn data-x="dom-EventSource-withCredentials">withCredentials</span></code></dt>

   <dd>
    <p>Returns true if the <span data-x="concept-request-credentials-mode">credentials mode</span>
    for connection requests to the <span data-x="concept-EventSource-url">URL providing the event
    stream</span> is set to "<code data-x="">include</code>", and false otherwise.</p>
   </dd>

   <dt><code data-x=""><var>source</var>.<span subdfn data-x="dom-EventSource-readyState">readyState</span></code></dt>

   <dd>
    <p>Returns the state of this <code>EventSource</code> object's connection. It can have the
    values described below.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>The <dfn constructor for="EventSource"><code
  data-x="dom-EventSource">EventSource(<var>url</var>, <var>eventSourceInitDict</var>)</code></dfn>
  constructor, when invoked, must run these steps:</p>

  <ol>
   <li><p>Let <var>ev</var> be a new <code>EventSource</code> object.</p></li>

   <li><p>Let <var>settings</var> be <var>ev</var>'s <span>relevant settings object</span>.

   <li><p>Let <var>urlRecord</var> be the result of <span>encoding-parsing a URL</span> given
   <var>url</var>, relative to <var>settings</var>.</p></li>

   <li><p>If <var>urlRecord</var> is failure, then throw a <span>"<code>SyntaxError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Set <var>ev</var>'s <span data-x="concept-EventSource-url">url</span> to
   <var>urlRecord</var>.</p>

   <li><p>Let <var>corsAttributeState</var> be <span
   data-x="attr-crossorigin-anonymous">Anonymous</span>.</p></li>

   <li><p>If the value of <var>eventSourceInitDict</var>'s <code
   data-x="dom-EventSourceInit-withCredentials">withCredentials</code> member is true, then set
   <var>corsAttributeState</var> to <span data-x="attr-crossorigin-use-credentials">Use
   Credentials</span> and set <var>ev</var>'s <code
   data-x="dom-EventSource-withCredentials">withCredentials</code> attribute to true.</p></li>

   <li><p>Let <var>request</var> be the result of <span
   data-x="create a potential-CORS request">creating a potential-CORS request</span> given
   <var>urlRecord</var>, the empty string, and <var>corsAttributeState</var>.</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-client">client</span> to
   <var>settings</var>.</p></li>

   <li><p>User agents may <span data-x="concept-header-list-set">set</span> (`<code
   data-x="http-accept">Accept</code>`, `<code>text/event-stream</code>`) in <var>request</var>'s
   <span data-x="concept-request-header-list">header list</span>.</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-cache-mode">cache mode</span> to
   "<code data-x="">no-store</code>".</p></li>

   <li><p>Set <var>request</var>'s <span data-x="concept-request-initiator-type">initiator
   type</span> to "<code data-x="">other</code>".</p></li>

   <li><p>Set <var>ev</var>'s <span data-x="concept-event-stream-request">request</span> to
   <var>request</var>.</p></li>

   <li><p>Let <var>processEventSourceEndOfBody</var> given <span
   data-x="concept-response">response</span> <var>res</var> be the following step: if
   <var>res</var> is not a <span>network error</span>, then <span>reestablish the
   connection</span>.</p></li>

   <li>
    <p><span data-x="concept-fetch">Fetch</span> <var>request</var>, with <i
    data-x="processResponseEndOfBody">processResponseEndOfBody</i> set to
    <var>processEventSourceEndOfBody</var> and <i data-x="processResponse">processResponse</i> set
    to the following steps given <span data-x="concept-response">response</span>
    <var>res</var>:</p>

    <ol>
     <li><p>If <var>res</var> is an <span>aborted network error</span>, then <span>fail the
     connection</span>.</p></li>

     <li><p>Otherwise, if <var>res</var> is a <span>network error</span>, then <span>reestablish
     the connection</span>, unless the user agent knows that to be futile, in which case the user
     agent may <span>fail the connection</span>.</p></li>

     <li><p>Otherwise, if <var>res</var>'s <span data-x="concept-response-status">status</span> is
     not 200, or if <var>res</var>'s `<code>Content-Type</code>` is not
     `<code>text/event-stream</code>`, then <span>fail the connection</span>.</p>

     <li><p>Otherwise, <span>announce the connection</span> and <a
     href="#event-stream-interpretation">interpret</a> <var>res</var>'s <span
     data-x="concept-response-body">body</span> line by line.</p></li>
    </ol>
   </li>

   <li><p>Return <var>ev</var>.</p></li>
  </ol>

  <hr>

  <p>The <dfn attribute for="EventSource"><code data-x="dom-EventSource-url">url</code></dfn>
  attribute's getter must return the <span data-x="concept-url-serializer">serialization</span> of
  this <code>EventSource</code> object's <span data-x="concept-EventSource-url">url</span>.</p>

  <p>The <dfn attribute for="EventSource"><code
  data-x="dom-EventSource-withCredentials">withCredentials</code></dfn> attribute must return the
  value to which it was last initialized. When the object is created, it must be initialized to
  false.</p>

  <p>The <dfn attribute for="EventSource"><code
  data-x="dom-EventSource-readyState">readyState</code></dfn> attribute represents the state of the
  connection. It can have the following values:</p>

  </div>

  <dl>
   <dt><dfn const for="EventSource"><code data-x="dom-EventSource-CONNECTING">CONNECTING</code></dfn> (numeric value 0)</dt>

   <dd>The connection has not yet been established, or it was closed and the user agent is
   reconnecting.</dd>

   <dt><dfn const for="EventSource"><code data-x="dom-EventSource-OPEN">OPEN</code></dfn> (numeric value 1)</dt>

   <dd>The user agent has an open connection and is dispatching events as it receives them.</dd>

   <dt><dfn const for="EventSource"><code data-x="dom-EventSource-CLOSED">CLOSED</code></dfn> (numeric value 2)</dt>

   <dd>The connection is not open, and the user agent is not trying to reconnect. Either there was a
   fatal error or the <code data-x="dom-EventSource-close">close()</code> method was invoked.</dd>
  </dl>

  <div w-nodev>

  <p>When the object is created its <code data-x="dom-EventSource-readyState">readyState</code> must
  be set to <code data-x="dom-EventSource-CONNECTING">CONNECTING</code> (0). The rules given below
  for handling the connection define when the value changes.</p>

  <p>The <dfn method for="EventSource"><code data-x="dom-EventSource-close">close()</code></dfn>
  method must abort any instances of the <span data-x="concept-fetch">fetch</span> algorithm started
  for this <code>EventSource</code> object, and must set the <code
  data-x="dom-EventSource-readyState">readyState</code> attribute to <code
  data-x="dom-EventSource-CLOSED">CLOSED</code>.</p> <!-- this also causes all the message events to
  stop firing, even if they were queued before close() was called -->

  </div>

  <p>The following are the <span>event handlers</span> (and their corresponding <span data-x="event
  handler event type">event handler event types</span>) <span w-nodev>that must be</span> supported,
  as <span>event handler IDL attributes</span>, by all objects implementing the
  <code>EventSource</code> interface:</p>

  <table>
   <thead>
    <tr><th><span data-x="event handlers">Event handler</span> <th><span>Event handler event type</span>
   <tbody>
    <tr><td><dfn attribute for="EventSource"><code data-x="handler-EventSource-onopen">onopen</code></dfn> <td> <code data-x="event-open">open</code>
    <tr><td><dfn attribute for="EventSource"><code data-x="handler-EventSource-onmessage">onmessage</code></dfn> <td> <code data-x="event-message">message</code>
    <tr><td><dfn attribute for="EventSource"><code data-x="handler-EventSource-onerror">onerror</code></dfn> <td> <code data-x="event-error">error</code>
  </table>

  <hr>

  <div w-nodev>

  <h4 id="sse-processing-model"><span id="processing-model-9"></span>Processing model</h4>

  <p>When a user agent is to <dfn>announce the connection</dfn>, the user agent must <span>queue a
  task</span> which, if the <code data-x="dom-EventSource-readyState">readyState</code> attribute is
  set to a value other than <code data-x="dom-EventSource-CLOSED">CLOSED</code>, sets the <code
  data-x="dom-EventSource-readyState">readyState</code> attribute to <code
  data-x="dom-EventSource-OPEN">OPEN</code> and <span data-x="concept-event-fire">fires an
  event</span> named <code data-x="event-open">open</code> at the <code>EventSource</code>
  object.</p>

  <p>When a user agent is to <dfn>reestablish the connection</dfn>, the user agent must run the
  following steps. These steps are run <span>in parallel</span>, not as part of a <span
  data-x="concept-task">task</span>. (The tasks that it queues, of course, are run like normal tasks
  and not themselves <span>in parallel</span>.)</p>

  <ol>
   <li>
    <p><span>Queue a task</span> to run the following steps:</p>

    <ol>
     <li><p>If the <code data-x="dom-EventSource-readyState">readyState</code> attribute is set to
     <code data-x="dom-EventSource-CLOSED">CLOSED</code>, abort the task.</p></li>

     <li><p>Set the <code data-x="dom-EventSource-readyState">readyState</code> attribute to <code
     data-x="dom-EventSource-CONNECTING">CONNECTING</code>.</p></li>

     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-error">error</code> at the <code>EventSource</code> object.</p></li>
    </ol>
   </li>

   <li><p>Wait a delay equal to the reconnection time of the event source.</li>

   <li><p>Optionally, wait some more. In particular, if the previous attempt failed, then user
   agents might introduce an exponential backoff delay to avoid overloading a potentially already
   overloaded server. Alternatively, if the operating system has reported that there is no network
   connectivity, user agents might wait for the operating system to announce that the network
   connection has returned before retrying.</li>

   <li><p>Wait until the aforementioned task has run, if it has not yet run.</p></li>

   <li>
    <p><span>Queue a task</span> to run the following steps:</p>

    <ol>
     <li><p>If the <code>EventSource</code> object's <code
     data-x="dom-EventSource-readyState">readyState</code> attribute is not set to <code
     data-x="dom-EventSource-CONNECTING">CONNECTING</code>, then return.</p></li>

     <li><p>Let <var>request</var> be the <code>EventSource</code> object's <span
     data-x="concept-event-stream-request">request</span>.

     <li>
      <p>If the <code>EventSource</code> object's <span
      data-x="concept-event-stream-last-event-id">last event ID string</span> is not the empty
      string, then:</p>

      <ol>
       <li><p>Let <var>lastEventIDValue</var> be the <code>EventSource</code> object's <span
       data-x="concept-event-stream-last-event-id">last event ID string</span>, <span data-x="UTF-8
       encode">encoded as UTF-8</span>.</p></li>

       <li><p><span data-x="concept-header-list-set">Set</span> (`<code>Last-Event-ID</code>`,
       <var>lastEventIDValue</var>) in <var>request</var>'s <span
       data-x="concept-request-header-list">header list</span>.</p></li>
      </ol>
     </li>

     <!--FETCH--><li><p><span data-x="concept-fetch">Fetch</span> <var>request</var> and process the
     response obtained in this fashion, if any, as described earlier in this section.</p></li>
    </ol>
   </li>
  </ol>

  <p>When a user agent is to <dfn>fail the connection</dfn>, the user agent must <span>queue a
  task</span> which, if the <code data-x="dom-EventSource-readyState">readyState</code> attribute is
  set to a value other than <code data-x="dom-EventSource-CLOSED">CLOSED</code>, sets the <code
  data-x="dom-EventSource-readyState">readyState</code> attribute to <code
  data-x="dom-EventSource-CLOSED">CLOSED</code> and <span data-x="concept-event-fire">fires an
  event</span> named <code data-x="event-error">error</code> at the <code>EventSource</code> object.
  <strong>Once the user agent has <span data-x="fail the connection">failed the connection</span>,
  it does <em>not</em> attempt to reconnect.</strong></p>

  <hr>

  <p>The <span>task source</span> for any <span data-x="concept-task">tasks</span> that are <span
  data-x="queue a task">queued</span> by <code>EventSource</code> objects is the <dfn>remote event
  task source</dfn>.</p>

  </div>


  <h4>The `<code>Last-Event-ID</code>` header</h4>

  <p>The <dfn http-header><code>Last-Event-ID</code></dfn>` HTTP request header reports an
  <code>EventSource</code> object's <span data-x="concept-event-stream-last-event-id">last event ID
  string</span> to the server when the user agent is to <span>reestablish the connection</span>.

  <p class="XXX">See <a href="https://github.com/whatwg/html/issues/7363">whatwg/html issue
  #7363</a> to define the value space better. It is essentially any UTF-8 encoded string, that does
  not contain U+0000 NULL, U+000A LF, or U+000D CR.</p>


  <h4 w-nodev>Parsing an event stream</h4>

  <h4 w-dev>The event stream format</h4>

  <p>This event stream format's <span>MIME type</span> is <code>text/event-stream</code>.</p>

  <p>The event stream format is as described by the <code data-x="">stream</code> production of the
  following ABNF, the character set for which is Unicode. <ref>ABNF</ref></p>

  <pre><code data-x="" class="abnf">stream        = [ bom ] *event
event         = *( comment / field ) end-of-line
comment       = colon *any-char end-of-line
field         = 1*name-char [ colon [ space ] *any-char ] end-of-line
end-of-line   = ( cr lf / cr / lf )

; characters
lf            = %x000A ; U+000A LINE FEED (LF)
cr            = %x000D ; U+000D CARRIAGE RETURN (CR)
space         = %x0020 ; U+0020 SPACE
colon         = %x003A ; U+003A COLON (:)
bom           = %xFEFF ; U+FEFF BYTE ORDER MARK
name-char     = %x0000-0009 / %x000B-000C / %x000E-0039 / %x003B-10FFFF
                ; a <span>scalar value</span> other than U+000A LINE FEED (LF), U+000D CARRIAGE RETURN (CR), or U+003A COLON (:)
any-char      = %x0000-0009 / %x000B-000C / %x000E-10FFFF
                ; a <span>scalar value</span> other than U+000A LINE FEED (LF) or U+000D CARRIAGE RETURN (CR)</code></pre>

  <p>Event streams in this format must always be encoded as UTF-8. <ref>ENCODING</ref></p>

  <p>Lines must be separated by either a U+000D CARRIAGE RETURN U+000A LINE FEED (CRLF) character
  pair, a single U+000A LINE FEED (LF) character, or a single U+000D CARRIAGE RETURN (CR)
  character.</p>

  <p w-nodev>Since connections established to remote servers for such resources are expected to be
  long-lived, UAs should ensure that appropriate buffering is used. In particular, while line
  buffering with lines are defined to end with a single U+000A LINE FEED (LF) character is safe,
  block buffering or line buffering with different expected line endings can cause delays in event
  dispatch.</p>


  <div w-nodev>

  <h4 id="event-stream-interpretation">Interpreting an event stream</h4>

  <p>Streams must be decoded using the <span>UTF-8 decode</span> algorithm.</p>

  <p class="note">The <span>UTF-8 decode</span> algorithm strips one leading UTF-8 Byte Order Mark
  (BOM), if any.</p>

  <p>The stream must then be parsed by reading everything line by line, with a U+000D CARRIAGE
  RETURN U+000A LINE FEED (CRLF) character pair, a single U+000A LINE FEED (LF) character not
  preceded by a U+000D CARRIAGE RETURN (CR) character, and a single U+000D CARRIAGE RETURN (CR)
  character not followed by a U+000A LINE FEED (LF) character being the ways in which a line can
  end.</p>

  <p>When a stream is parsed, a <var>data</var> buffer, an <var>event type</var>
  buffer, and a <var>last event ID</var> buffer must be associated with it. They must be
  initialized to the empty string.</p>

  <p>Lines must be processed, in the order they are received, as follows:</p>

  <dl class="switch">

   <dt>If the line is empty (a blank line)</dt>

   <dd><p><span>Dispatch the event</span>, as defined below.</p></dd>


   <dt>If the line starts with a U+003A COLON character (:)</dt>

   <dd><p>Ignore the line.</p></dd>


   <dt>If the line contains a U+003A COLON character (:)</dt>

   <dd>
    <p>Collect the characters on the line before the first U+003A COLON character (:), and let <var>field</var> be that string.</p>

    <p>Collect the characters on the line after the first U+003A COLON character (:), and let <var>value</var> be that string. If <var>value</var> starts with a U+0020 SPACE
    character, remove it from <var>value</var>.</p>

    <p><a href="#processField">Process the field</a> using the steps described below, using <var>field</var> as the field name and <var>value</var> as the field value.</p>
   </dd>


   <dt>Otherwise, the string is not empty but does not contain a U+003A COLON character (:)</dt>

   <dd>
    <p><a href="#processField">Process the field</a> using the steps described below, using the
    whole line as the field name, and the empty string as the field value.</p>
   </dd>
  </dl>

  <p>Once the end of the file is reached, any pending data must be discarded. (If the file ends in
  the middle of an event, before the final empty line, the incomplete event is not dispatched.)</p>

  <hr>

  <p id="processField">The steps to <dfn data-x="">process the field</dfn> given a field name and a
  field value depend on the field name, as given in the following list. Field names must be compared
  literally, with no case folding performed.</p>

  <dl class="switch">

   <dt>If the field name is "event"</dt>

   <dd><p>Set the <var>event type</var> buffer to field value.</p></dd>


   <dt>If the field name is "data"</dt>

   <dd><p>Append the field value to the <var>data</var> buffer, then append a single U+000A
   LINE FEED (LF) character to the <var>data</var> buffer.</p></dd>


   <dt>If the field name is "id"</dt>

   <dd><p>If the field value does not contain U+0000 NULL, then set the <var>last event ID</var>
   buffer to the field value. Otherwise, ignore the field.</p></dd>


   <dt>If the field name is "retry"</dt>

   <dd><p>If the field value consists of only <span>ASCII digits</span>, then interpret the field
   value as an integer in base ten, and set the event stream's <span
   data-x="concept-event-stream-reconnection-time">reconnection time</span> to that integer.
   Otherwise, ignore the field.</p></dd>

   <!-- v2 feature request from Per-Erik Brodin:
      > > > Finally, it could be useful to be able to reset the reconnection
      > > > time to the user agent default value by sending the retry field only
      > > > and leave out the value similar to how you reset the last event id.
      > >
      > > What's the use case?
      >
      > Take the stock ticker as an example. When the stock market closes the
      > server logic knows that there won't be any new events for a number of
      > hours and so it can send the corresponding reconnection time and close
      > the connection. If the client is still running by the time the market
      > opens, it will reconnect, and the server can now reset the reconnection
      > time to a time that is convenient for the user agent (which is the user
      > agent default value, unknown to the server).
    -->

<!-- v2 feature request from John Fallows - https://lists.w3.org/Archives/Public/public-html-comments/2008Oct/0009.html

   <dt>If the field name is "reconnect"</dt>

   <dd><p>If the field value is the empty string, then: <span>dispatch the
   event</span> as defined below, and then drop the connection and immediately reconnect as if the
   <span data-x="concept-event-stream-reconnection-time">reconnection time</span> was zero for this
   one time.</p></dd>

  -->

   <dt>Otherwise</dt>

   <dd><p>The field is ignored.</p></dd>
  </dl>


  <p>When the user agent is required to <dfn id="dispatchMessage">dispatch the
  event</dfn>, the user agent must process the <var>data</var> buffer, the <var>event type</var> buffer, and the <var>last event ID</var> buffer using steps
  appropriate for the user agent.</p>

  <p>For web browsers, the appropriate steps to <span>dispatch the event</span> are as follows:</p>

  <ol>
   <li><p>Set the <span data-x="concept-event-stream-last-event-id">last event ID string</span> of
   the event source to the value of the <var>last event ID</var> buffer. The buffer does
   not get reset, so the <span data-x="concept-event-stream-last-event-id">last event ID
   string</span> of the event source remains set to this value until the next time it is set by the
   server.</p></li>

   <li><p>If the <var>data</var> buffer is an empty string, set the <var>data</var> buffer and the <var>event type</var> buffer to the empty string and
   return.</p></li>

   <li><p>If the <var>data</var> buffer's last character is a U+000A LINE FEED (LF)
   character, then remove the last character from the <var>data</var> buffer.</p></li>

   <li><p>Let <var>event</var> be the result of <span>creating an event</span> using
   <code>MessageEvent</code>, in the <span data-x="concept-relevant-realm">relevant realm</span> of
   the <code>EventSource</code> object.</p></li>

   <li><p>Initialize <var>event</var>'s <code data-x="dom-Event-type">type</code> attribute to
   "<code data-x="event-message">message</code>", its <code
   data-x="dom-MessageEvent-data">data</code> attribute to <var>data</var>, its <code
   data-x="dom-MessageEvent-origin">origin</code> attribute to the <span data-x="serialization of
   an origin">serialization</span> of the <span data-x="concept-url-origin">origin</span> of the
   event stream's final URL (i.e., the URL after redirects), and its <code
   data-x="dom-MessageEvent-lastEventId">lastEventId</code> attribute to the <span
   data-x="concept-event-stream-last-event-id">last event ID string</span> of the event
   source.</p></li>

   <li><p>If the <var>event type</var> buffer has a value other than the empty string, change the
   <span data-x="concept-event-type">type</span> of the newly created event to equal the value of
   the <var>event type</var> buffer.</p></li>

   <li><p>Set the <var>data</var> buffer and the <var>event type</var> buffer to
   the empty string.</p></li>

   <li><p><span>Queue a task</span> which, if the <code
   data-x="dom-EventSource-readyState">readyState</code> attribute is set to a value other than <code
   data-x="dom-EventSource-CLOSED">CLOSED</code>, <span
   data-x="concept-event-dispatch">dispatches</span> the newly created event at the
   <code>EventSource</code> object.</p></li> <!-- so calling close() drops the messages immediately.
   The connection dying doesn't interfere with already-sent messages since it sets us to CLOSED via
   a task, not immediately -->
  </ol>

  <p class="note">If an event doesn't have an "id" field, but an earlier event did set the event
  source's <span data-x="concept-event-stream-last-event-id">last event ID string</span>, then the
  event's <code data-x="dom-MessageEvent-lastEventId">lastEventId</code> field will be set to the
  value of whatever the last seen "id" field was.</p>

  <p>For other user agents, the appropriate steps to <span>dispatch the event</span> are
  implementation dependent, but at a minimum they must set the <var>data</var> and <var>event type</var> buffers to the empty string before returning.</p>

  </div>

  <div class="example">
   <p>The following event stream, once followed by a blank line:</p>
   <pre>data: YHOO
data: +2
data: 10</pre>

   <p>...would cause an event <code data-x="event-message">message</code> with the interface
   <code>MessageEvent</code> to be dispatched on the <code>EventSource</code> object. The event's
   <code data-x="dom-MessageEvent-data">data</code> attribute would contain the string "<code
   data-x="">YHOO\n+2\n10</code>" (where "<code data-x="">\n</code>" represents a newline).</p>

   <p>This could be used as follows:
   <pre><code class="js">var stocks = new EventSource("https://stocks.example.com/ticker.php");
stocks.onmessage = function (event) {
  var data = event.data.split('\n');
  updateStocks(data[0], data[1], data[2]);
};</code></pre>

   <p>...where <code data-x="">updateStocks()</code> is a function defined as:</p>

   <pre><code class="js">function updateStocks(symbol, delta, value) { ... }</code></pre>

   <p>...or some such.</p>
  </div>

  <div class="example">
   <p>The following stream contains four blocks. The first block has just a comment, and will fire
   nothing. The second block has two fields with names "data" and "id" respectively; an event will
   be fired for this block, with the data "first event", and will then set the last event ID to "1"
   so that if the connection died between this block and the next, the server would be sent a
   `<code>Last-Event-ID</code>` header with the value `<code data-x="">1</code>`. The third block
   fires an event with data "second event", and also has an "id" field, this time with no value,
   which resets the last event ID to the empty string (meaning no `<code>Last-Event-ID</code>`
   header will now be sent in the event of a reconnection being attempted). Finally, the last block
   just fires an event with the data "&nbsp;third&nbsp;event" (with a single leading space
   character). Note that the last still has to end with a blank line, the end of the stream is not
   enough to trigger the dispatch of the last event.</p>

   <pre>: test stream

data: first event
id: 1

data:second event
id

data:&nbsp;&nbsp;third event
</pre>
  </div>

  <div class="example">
   <p>The following stream fires two events:</p>

   <pre>data

data
data

data:</pre>

   <p>The first block fires events with the data set to the empty string, as would the last block if
   it was followed by a blank line. The middle block fires an event with the data set to a single
   newline character. The last block is discarded because it is not followed by a blank line.</p>
  </div>

  <div class="example">
   <p>The following stream fires two identical events:</p>

   <pre>data:test

data:&nbsp;test
</pre>

   <p>This is because the space after the colon is ignored if present.</p>
  </div>


  <h4>Authoring notes</h4>

  <p>Legacy proxy servers are known to, in certain cases, drop HTTP connections after a short
  timeout. To protect against such proxy servers, authors can include a comment line (one starting
  with a ':' character) every 15 seconds or so.</p>

  <p>Authors wishing to relate event source connections to each other or to specific documents
  previously served might find that relying on IP addresses doesn't work, as individual clients can
  have multiple IP addresses (due to having multiple proxy servers) and individual IP addresses can
  have multiple clients (due to sharing a proxy server). It is better to include a unique identifier
  in the document when it is served and then pass that identifier as part of the URL when the
  connection is established.</p>

  <p>Authors are also cautioned that HTTP chunking can have unexpected negative effects on the
  reliability of this protocol, in particular if the chunking is done by a different layer unaware
  of the timing requirements. If this is a problem, chunking can be disabled for serving event
  streams.</p> <!-- v2 can we get a better solution? -->

  <p>Clients that support HTTP's per-server connection limitation might run into trouble when
  opening multiple pages from a site if each page has an <code>EventSource</code> to the same
  domain. Authors can avoid this using the relatively complex mechanism of using unique domain names
  per connection, or by allowing the user to enable or disable the <code>EventSource</code>
  functionality on a per-page basis, or by sharing a single <code>EventSource</code> object using a
  <span data-x="SharedWorkerGlobalScope">shared worker</span>.</p>

  <div w-nodev>

  <h4 id="eventsource-push">Connectionless push and other features</h4>

  <p>User agents running in controlled environments, e.g. browsers on mobile handsets tied to
  specific carriers, may offload the management of the connection to a proxy on the network. In such
  a situation, the user agent for the purposes of conformance is considered to include both the
  handset software and the network proxy.</p>

  <div class="example">

   <p>For example, a browser on a mobile device, after having established a connection, might detect
   that it is on a supporting network and request that a proxy server on the network take over the
   management of the connection. The timeline for such a situation might be as follows:</p>

   <ol>
    <li>Browser connects to a remote HTTP server and requests the resource specified by the author
    in the <code data-x="dom-EventSource">EventSource</code> constructor.</li>

    <li>The server sends occasional messages.</li>

    <li>In between two messages, the browser detects that it is idle except for the network activity
    involved in keeping the TCP connection alive, and decides to switch to sleep mode to save
    power.</li>

    <li>The browser disconnects from the server.</li>

    <li>The browser contacts a service on the network, and requests that the service, a "push
    proxy", maintain the connection instead.</li>

    <li>The "push proxy" service contacts the remote HTTP server and requests the resource specified
    by the author in the <code data-x="dom-EventSource">EventSource</code> constructor (possibly
    including a `<code>Last-Event-ID</code>` HTTP header, etc.).</li>

    <li>The browser allows the mobile device to go to sleep.</li>

    <li>The server sends another message.</li>

    <li>The "push proxy" service uses a technology such as OMA push to convey the event to the
    mobile device, which wakes only enough to process the event and then returns to sleep.</li>
   </ol>

  </div>

  <p>This can reduce the total data usage, and can therefore result in considerable power
  savings.</p>

  <p>As well as implementing the existing API and <code>text/event-stream</code> wire format as
  defined by this specification and in more distributed ways as described above, formats of event
  framing defined by <span>other applicable specifications</span> may be supported. This
  specification does not define how they are to be parsed or processed.</p>


  <h4>Garbage collection</h4>

  <p>While an <code>EventSource</code> object's <code
  data-x="dom-EventSource-readyState">readyState</code> is <code
  data-x="dom-EventSource-CONNECTING">CONNECTING</code>, and the object has one or more event
  listeners registered for <code data-x="event-open">open</code>, <code
  data-x="event-message">message</code>, or <code data-x="event-error">error</code> events, there must
  be a strong reference from the <code>Window</code> or <code>WorkerGlobalScope</code> object that
  the <code>EventSource</code> object's constructor was invoked from to the <code>EventSource</code>
  object itself.</p>

  <p>While an <code>EventSource</code> object's <code
  data-x="dom-EventSource-readyState">readyState</code> is <code
  data-x="dom-EventSource-OPEN">OPEN</code>, and the object has one or more event listeners
  registered for <code data-x="event-message">message</code> or <code
  data-x="event-error">error</code> events, there must be a strong reference from the
  <code>Window</code> or <code>WorkerGlobalScope</code> object that the <code>EventSource</code>
  object's constructor was invoked from to the <code>EventSource</code> object itself.</p>

  <p>While there is a task queued by an <code>EventSource</code> object on the <span>remote event
  task source</span>, there must be a strong reference from the <code>Window</code> or
  <code>WorkerGlobalScope</code> object that the <code>EventSource</code> object's constructor was
  invoked from to that <code>EventSource</code> object.</p>

  <p>If a user agent is to <dfn data-x="concept-EventSource-forcibly-close">forcibly close</dfn> an
  <code>EventSource</code> object (this happens when a <code>Document</code> object goes away
  permanently), the user agent must abort any instances of the <span
  data-x="concept-fetch">fetch</span> algorithm started for this <code>EventSource</code> object,
  and must set the <code data-x="dom-EventSource-readyState">readyState</code> attribute to <code
  data-x="dom-EventSource-CLOSED">CLOSED</code>.</p> <!-- same as calling close() -->

  <p>If an <code>EventSource</code> object is garbage collected while its connection is still open,
  the user agent must abort any instance of the <span data-x="concept-fetch">fetch</span> algorithm
  opened by this <code>EventSource</code>.</p> <!-- no need to throw tasks away or anything; for it
  to get garbage collected, there can't be anything that would be able to receive those events -->


  <h4>Implementation advice</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>User agents are strongly urged to provide detailed diagnostic information about
  <code>EventSource</code> objects and their related network connections in their development
  consoles, to aid authors in debugging code using this API.</p>

  <p>For example, a user agent could have a panel displaying all the <code>EventSource</code>
  objects a page has created, each listing the constructor's arguments, whether there was a network
  error, what the CORS status of the connection is and what headers were sent by the client and
  received from the server to lead to that status, the messages that were received and how they were
  parsed, and so forth.</p>

  <p>Implementations are especially encouraged to report detailed information to their development
  consoles whenever an <code data-x="event-error">error</code> event is fired, since little to no
  information can be made available in the events themselves.</p>

  </div>


  <h3 split-filename="web-messaging" id="web-messaging"><dfn id="crossDocumentMessages">Cross-document messaging</dfn></h3>

  <p>Web browsers, for security and privacy reasons, prevent documents in different domains from
  affecting each other; that is, cross-site scripting is disallowed.</p>

  <p>While this is an important security feature, it prevents pages from different domains from
  communicating even when those pages are not hostile. This section introduces a messaging system
  that allows documents to communicate with each other regardless of their source domain, in a way
  designed to not enable cross-site scripting attacks.</p>

  <p class="note" id="fingerprint-postMessage">
  <!-- INSERT TRACKING -->
  The <code data-x="dom-window-postMessage">postMessage()</code> API can be used as a <span>tracking
  vector</span>.</p>


  <h4>Introduction</h4>

  <!-- NON-NORMATIVE SECTION -->

  <div class="example">

   <p>For example, if document A contains an <code>iframe</code> element that contains document B,
   and script in document A calls <code data-x="dom-window-postMessage">postMessage()</code> on the
   <code>Window</code> object of document B, then a message event will be fired on that object,
   marked as originating from the <code>Window</code> of document A. The script in document A might
   look like:</p>

   <pre><code class="js">var o = document.getElementsByTagName('iframe')[0];
o.contentWindow.postMessage('Hello world', 'https://b.example.org/');</code></pre>

   <p>To register an event handler for incoming events, the script would use <code
   data-x="">addEventListener()</code> (or similar mechanisms). For example, the script in document B
   might look like:</p>

   <pre><code class="js">window.addEventListener('message', receiver, false);
function receiver(e) {
  if (e.origin == 'https://example.com') {
    if (e.data == 'Hello world') {
      e.source.postMessage('Hello', e.origin);
    } else {
      alert(e.data);
    }
  }
}</code></pre>

   <p>This script first checks the domain is the expected domain, and then looks at the message,
   which it either displays to the user, or responds to by sending a message back to the document
   which sent the message in the first place.</p>

  </div>


  <h4 id="security-postmsg">Security</h4>

  <div w-nodev>

  <h5>Authors</h5>

  </div>

  <p id="security-4" class="warning">Use of this API requires extra care to protect users from
  hostile entities abusing a site for their own purposes.</p>

  <p>Authors should check the <code data-x="dom-MessageEvent-origin">origin</code> attribute to
  ensure that messages are only accepted from domains that they expect to receive messages from.
  Otherwise, bugs in the author's message handling code could be exploited by hostile sites.</p>

  <p>Furthermore, even after checking the <code data-x="dom-MessageEvent-origin">origin</code>
  attribute, authors should also check that the data in question is of the expected format.
  Otherwise, if the source of the event has been attacked using a cross-site scripting flaw, further
  unchecked processing of information sent using the <code
  data-x="dom-window-postMessage">postMessage()</code> method could result in the attack being
  propagated into the receiver.</p>

  <p>Authors should not use the wildcard keyword (*) in the <var>targetOrigin</var>
  argument in messages that contain any confidential information, as otherwise there is no way to
  guarantee that the message is only delivered to the recipient to which it was intended.</p>

  <hr>

  <p>Authors who accept messages from any origin are encouraged to consider the risks of a
  denial-of-service attack. An attacker could send a high volume of messages; if the receiving page
  performs expensive computation or causes network traffic to be sent for each such message, the
  attacker's message could be multiplied into a denial-of-service attack. Authors are encouraged to
  employ rate limiting (only accepting a certain number of messages per minute) to make such attacks
  impractical.</p>


  <div w-nodev>

  <h5>User agents</h5>

  <p>The integrity of this API is based on the inability for scripts of one <span>origin</span> to
  post arbitrary events (using <code data-x="">dispatchEvent()</code> or otherwise) to objects in
  other origins (those that are not the <span data-x="same origin">same</span>).</p>

  <p class="note">Implementers are urged to take extra care in the implementation of this feature.
  It allows authors to transmit information from one domain to another domain, which is normally
  disallowed for security reasons. It also requires that UAs be careful to allow access to certain
  properties but not others.</p>

  <hr>

  <p>User agents are also encouraged to consider rate-limiting message traffic between different
  <span data-x="origin">origins</span>, to protect na&iuml;ve sites from denial-of-service
  attacks.</p>

  </div>



  <h4>Posting messages</h4>

  <dl class="domintro">
   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-window-postMessage-options">postMessage</span>(<var>message</var> [, <var>options</var> ])</code></dt>

   <dd>
    <p>Posts a message to the given window. Messages can be structured objects, e.g. nested objects
    and arrays, can contain JavaScript values (strings, numbers, <code>Date</code> objects, etc.),
    and can contain certain data objects such as <code>File</code> <code>Blob</code>,
    <code>FileList</code>, and <code data-x="idl-ArrayBuffer">ArrayBuffer</code> objects.</p>

    <p>Objects listed in the <code data-x="dom-StructuredSerializeOptions-transfer">transfer</code> member
    of <var>options</var> are transferred, not just cloned, meaning that they are no longer usable
    on the sending side.</p>

    <p>A target origin can be specified using the <code
    data-x="dom-WindowPostMessageOptions-targetOrigin">targetOrigin</code> member of
    <var>options</var>. If not provided, it defaults to "<code data-x="">/</code>". This default
    restricts the message to same-origin targets only.</p>

    <p>If the origin of the target window doesn't match the given target origin, the message is
    discarded, to avoid information leakage. To send the message to the target regardless of origin,
    set the target origin to "<code data-x="">*</code>".</p>

    <p>Throws a <span>"<code>DataCloneError</code>"</span> <code>DOMException</code> if
    <var>transfer</var> array contains duplicate objects or if <var>message</var> could not be
    cloned.</p>
   </dd>

   <dt><code data-x=""><var>window</var>.<span data-x="dom-window-postMessage">postMessage</span>(<var>message</var>, <var>targetOrigin</var> [, <var>transfer</var> ])</code></dt>

   <dd><p>This is an alternate version of <code
   data-x="dom-window-postMessage-options">postMessage()</code> where the target origin is specified
   as a parameter. Calling <code data-x="">window.postMessage(message, target, transfer)</code> is
   equivalent to <code data-x="">window.postMessage(message, {targetOrigin,
   transfer})</code>.</p></dd>
  </dl>

  <p class="note">When posting a message to a <code>Window</code> of a <span>browsing context</span>
  that has just been navigated to a new <code>Document</code> is likely to result in the message not
  receiving its intended recipient: the scripts in the target <span>browsing context</span> have to
  have had time to set up listeners for the messages. Thus, for instance, in situations where a
  message is to be sent to the <code>Window</code> of newly created child <code>iframe</code>,
  authors are advised to have the child <code>Document</code> post a message to their parent
  announcing their readiness to receive messages, and for the parent to wait for this message before
  beginning posting messages.</p>

  <div w-nodev>

  <p>The <dfn>window post message steps</dfn>, given a <var>targetWindow</var>, <var>message</var>,
  and <var>options</var>, are as follows:</p>

  <ol> <!-- a lot of this is similar or identical to port.postMessage -->

   <li><p>Let <var>targetRealm</var> be <var>targetWindow</var>'s <span
   data-x="concept-global-object-realm">realm</span>.</p></li>

   <li><p>Let <var>incumbentSettings</var> be the <span>incumbent settings object</span>.</p></li>
   <!-- This is one of the few cases where incumbent is probably the right choice. Current or
   relevant would mean that the MessageEvent's source property always points to the "calling"
   window, in same-origin cases. See discussion at
   https://github.com/whatwg/html/issues/1542#issuecomment-233502636 -->

   <li><p>Let <var>targetOrigin</var> be <var>options</var>["<code
   data-x="dom-WindowPostMessageOptions-targetOrigin">targetOrigin</code>"].</p></li>

   <li><p>If <var>targetOrigin</var> is a single U+002F SOLIDUS character (/), then set
   <var>targetOrigin</var> to <var>incumbentSettings</var>'s <span
   data-x="concept-settings-object-origin">origin</span>.</p>

   <li>
    <p>Otherwise, if <var>targetOrigin</var> is not a single U+002A ASTERISK character (*),
    then:</p>

    <ol>
     <li><p>Let <var>parsedURL</var> be the result of running the <span>URL parser</span> on
     <var>targetOrigin</var>.</p></li>

     <li><p>If <var>parsedURL</var> is failure, then throw a <span>"<code>SyntaxError</code>"</span>
     <code>DOMException</code>.</p></li>

     <li><p>Set <var>targetOrigin</var> to <var>parsedURL</var>'s <span
     data-x="concept-url-origin">origin</span>.</p></li>
    </ol>
   </li>

   <li><p>Let <var>transfer</var> be <var>options</var>["<code
   data-x="dom-StructuredSerializeOptions-transfer">transfer</code>"].</p></li>

   <li><p>Let <var>serializeWithTransferResult</var> be
   <span>StructuredSerializeWithTransfer</span>(<var>message</var>, <var>transfer</var>). Rethrow
   any exceptions.</p></li>

   <li>
    <p><span>Queue a global task</span> on the <dfn>posted message task source</dfn> given
    <var>targetWindow</var> to run the following steps:</p>

    <ol>
     <li><p>If the <var>targetOrigin</var> argument is not a single literal U+002A ASTERISK character
     (*) and <var>targetWindow</var>'s <span data-x="concept-document-window">associated
     <code>Document</code></span>'s <span data-x="concept-document-origin">origin</span> is not
     <span>same origin</span> with <var>targetOrigin</var>, then return.</p></li>

     <li><p>Let <var>origin</var> be the <span data-x="serialization of an
     origin">serialization</span> of <var>incumbentSettings</var>'s <span
     data-x="concept-settings-object-origin">origin</span>.</p></li>

     <li><p>Let <var>source</var> be the <code>WindowProxy</code> object corresponding to
     <var>incumbentSettings</var>'s <span data-x="concept-settings-object-global">global
     object</span> (a <code>Window</code> object).</p></li>

     <li>
      <p>Let <var>deserializeRecord</var> be
      <span>StructuredDeserializeWithTransfer</span>(<var>serializeWithTransferResult</var>,
      <var>targetRealm</var>).</p>

      <p>If this throws an exception, catch it, <span data-x="concept-event-fire">fire an
      event</span> named <code data-x="event-messageerror">messageerror</code> at
      <var>targetWindow</var>, using <code>MessageEvent</code>, with the <code
      data-x="dom-MessageEvent-origin">origin</code> attribute initialized to <var>origin</var> and
      the <code data-x="dom-MessageEvent-source">source</code> attribute initialized to
      <var>source</var>, and then return.</p>
     </li>

     <li><p>Let <var>messageClone</var> be <var>deserializeRecord</var>.[[Deserialized]].</p></li>

     <li><p>Let <var>newPorts</var> be a new <span>frozen array</span> consisting of all
     <code>MessagePort</code> objects in <var>deserializeRecord</var>.[[TransferredValues]], if any,
     maintaining their relative order.</p></li>

     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-message">message</code> at <var>targetWindow</var>, using
     <code>MessageEvent</code>, with the <code data-x="dom-MessageEvent-origin">origin</code>
     attribute initialized to <var>origin</var>, the <code
     data-x="dom-MessageEvent-source">source</code> attribute initialized to <var>source</var>, the
     <code data-x="dom-MessageEvent-data">data</code> attribute initialized to
     <var>messageClone</var>, and the <code data-x="dom-MessageEvent-ports">ports</code> attribute
     initialized to <var>newPorts</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>The <code>Window</code> interface's <dfn method for="Window"><code
  data-x="dom-window-postMessage-options">postMessage(<var>message</var>,
  <var>options</var>)</code></dfn> method steps are to run the <span>window post message
  steps</span> given <span>this</span>, <var>message</var>, and <var>options</var>.</p>

  <p>The <code>Window</code> interface's <dfn method for="Window"><code
  data-x="dom-window-postMessage">postMessage(<var>message</var>, <var>targetOrigin</var>,
  <var>transfer</var>)</code></dfn> method steps are to run the <span>window post message
  steps</span> given <span>this</span>, <var>message</var>, and «[ "<code
  data-x="dom-WindowPostMessageOptions-targetOrigin">targetOrigin</code>" →
  <var>targetOrigin</var>, "<code data-x="dom-StructuredSerializeOptions-transfer">transfer</code>"
  → <var>transfer</var> ]».</p>

  </div>




  <h3><dfn>Channel messaging</dfn></h3>

  <h4>Introduction</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>To enable independent pieces of code (e.g. running in different <span data-x="browsing
  context">browsing contexts</span>) to communicate directly, authors can use <span>channel
  messaging</span>.</p>

  <p>Communication channels in this mechanism are implemented as two-ways pipes, with a port at each
  end. Messages sent in one port are delivered at the other port, and vice-versa. Messages are
  delivered as DOM events, without interrupting or blocking running <span data-x="concept-task">tasks</span>.</p>

  <p>To create a connection (two "entangled" ports), the <code data-x="">MessageChannel()</code>
  constructor is called:</p>

  <pre><code class="js">var channel = new MessageChannel();</code></pre>

  <p>One of the ports is kept as the local port, and the other port is sent to the remote code, e.g.
  using <code data-x="dom-window-postMessage">postMessage()</code>:</p>

  <pre><code class="js">otherWindow.postMessage('hello', 'https://example.com', [channel.port2]);</code></pre>

  <p>To send messages, the <code data-x="dom-MessagePort-postMessage">postMessage()</code> method on
  the port is used:</p>

  <pre><code class="js">channel.port1.postMessage('hello');</code></pre>

  <p>To receive messages, one listens to <code data-x="event-message">message</code> events:</p>

  <pre><code class="js">channel.port1.onmessage = handleMessage;
function handleMessage(event) {
  // message is in event.data
  // ...
}</code></pre>

  <p>Data sent on a port can be structured data; for example here an array of strings is passed on a
  <code>MessagePort</code>:</p>

  <pre><code class="js">port1.postMessage(['hello', 'world']);</code></pre>


  <h5>Examples</h5>

  <!-- NON-NORMATIVE SECTION -->

  <div class="example">

   <p>In this example, two JavaScript libraries are connected to each other using
   <code>MessagePort</code>s. This allows the libraries to later be hosted in different frames, or
   in <code>Worker</code> objects, without any change to the APIs.</p>

   <pre><code class="html">&lt;script src="contacts.js">&lt;/script> &lt;!-- exposes a contacts object -->
&lt;script src="compose-mail.js">&lt;/script> &lt;!-- exposes a composer object -->
&lt;script>
 var channel = new MessageChannel();
 composer.addContactsProvider(channel.port1);
 contacts.registerConsumer(channel.port2);
&lt;/script></code></pre>

   <p>Here's what the "addContactsProvider()" function's implementation could look like:</p>

   <pre><code class="js">function addContactsProvider(port) {
  port.onmessage = function (event) {
    switch (event.data.messageType) {
      case 'search-result': handleSearchResult(event.data.results); break;
      case 'search-done': handleSearchDone(); break;
      case 'search-error': handleSearchError(event.data.message); break;
      // ...
    }
  };
};</code></pre>

   <p>Alternatively, it could be implemented as follows:</p>

   <pre><code class="js">function addContactsProvider(port) {
  port.addEventListener('message', function (event) {
    if (event.data.messageType == 'search-result')
      handleSearchResult(event.data.results);
  });
  port.addEventListener('message', function (event) {
    if (event.data.messageType == 'search-done')
      handleSearchDone();
  });
  port.addEventListener('message', function (event) {
    if (event.data.messageType == 'search-error')
      handleSearchError(event.data.message);
  });
  // ...
  port.start();
};</code></pre>

   <p>The key difference is that when using <code
   data-x="dom-EventTarget-addEventListener">addEventListener()</code>, the <code
   data-x="dom-MessagePort-start">start()</code> method must also be invoked. When using <code
   data-x="handler-MessageEventTarget-onmessage">onmessage</code>, the call to <code
   data-x="dom-MessagePort-start">start()</code> is implied.</p>

   <p>The <code data-x="dom-MessagePort-start">start()</code> method, whether called explicitly or
   implicitly (by setting <code data-x="handler-MessageEventTarget-onmessage">onmessage</code>),
   starts the flow of messages: messages posted on message ports are initially paused, so that they
   don't get dropped on the floor before the script has had a chance to set up its handlers.</p>

  </div>


  <h5>Ports as the basis of an object-capability model on the web</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>Ports can be viewed as a way to expose limited capabilities (in the object-capability model
  sense) to other actors in the system. This can either be a weak capability system, where the ports
  are merely used as a convenient model within a particular origin, or as a strong capability model,
  where they are provided by one origin <var>provider</var> as the only mechanism by which
  another origin <var>consumer</var> can effect change in or obtain information from <var>provider</var>.</p>

  <p>For example, consider a situation in which a social web site embeds in one <code>iframe</code>
  the user's email contacts provider (an address book site, from a second origin), and in a second
  <code>iframe</code> a game (from a third origin). The outer social site and the game in the second
  <code>iframe</code> cannot access anything inside the first <code>iframe</code>; together they can
  only:</p>

  <ul>
   <li><p><span>Navigate</span> the <code>iframe</code> to a new <span>URL</span>, such as the same
   <span>URL</span> but with a different <span data-x="concept-url-fragment">fragment</span>,
   causing the <code>Window</code> in the <code>iframe</code> to receive a <code
   data-x="event-hashchange">hashchange</code> event.</p></li>

   <li><p>Resize the <code>iframe</code>, causing the <code>Window</code> in the <code>iframe</code>
   to receive a <code data-x="event-resize">resize</code> event.</p></li>

   <!-- anything else? -->

   <li><p>Send a <code data-x="event-message">message</code> event to the <code>Window</code> in the
   <code>iframe</code> using the <code data-x="dom-window-postMessage">window.postMessage()</code>
   API.</p></li>
  </ul>

  <p>The contacts provider can use these methods, most particularly the third one, to provide an API
  that can be accessed by other origins to manipulate the user's address book. For example, it could
  respond to a message "<code data-x="">add-contact Guillaume Tell
  &lt;tell@pomme.example.net></code>" by adding the given person and email address to the user's
  address book.</p>

  <p>To avoid any site on the web being able to manipulate the user's contacts, the contacts
  provider might only allow certain trusted sites, such as the social site, to do this.</p>

  <p>Now suppose the game wanted to add a contact to the user's address book, and that the social
  site was willing to allow it to do so on its behalf, essentially "sharing" the trust that the
  contacts provider had with the social site. There are several ways it could do this; most simply,
  it could just proxy messages between the game site and the contacts site. However, this solution
  has a number of difficulties: it requires the social site to either completely trust the game site
  not to abuse the privilege, or it requires that the social site verify each request to make sure
  it's not a request that it doesn't want to allow (such as adding multiple contacts, reading the
  contacts, or deleting them); it also requires some additional complexity if there's ever the
  possibility of multiple games simultaneously trying to interact with the contacts provider.</p>

  <p>Using message channels and <code>MessagePort</code> objects, however, all of these problems can
  go away. When the game tells the social site that it wants to add a contact, the social site can
  ask the contacts provider not for it to add a contact, but for the <em>capability</em> to add a
  single contact. The contacts provider then creates a pair of <code>MessagePort</code> objects, and
  sends one of them back to the social site, who forwards it on to the game. The game and the
  contacts provider then have a direct connection, and the contacts provider knows to only honor a
  single "add contact" request, nothing else. In other words, the game has been granted the
  capability to add a single contact.</p>


  <h5>Ports as the basis of abstracting out service implementations</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>Continuing the example from the previous section, consider the contacts provider in particular.
  While an initial implementation might have simply used <code>XMLHttpRequest</code> objects in the
  service's <code>iframe</code>, an evolution of the service might instead want to use a <span
  data-x="SharedWorker">shared worker</span> with a single <code>WebSocket</code> connection.</p>

  <p>If the initial design used <code>MessagePort</code> objects to grant capabilities, or even just
  to allow multiple simultaneous independent sessions, the service implementation can switch from
  the <code>XMLHttpRequest</code>s-in-each-<code>iframe</code> model to the
  shared-<code>WebSocket</code> model without changing the API at all: the ports on the service
  provider side can all be forwarded to the shared worker without it affecting the users of the API
  in the slightest.</p>



  <h4>Message channels</h4>

  <pre><code class="idl">[Exposed=(Window,Worker)]
interface <dfn interface>MessageChannel</dfn> {
  <span data-x="dom-MessageChannel">constructor</span>();

  readonly attribute <span>MessagePort</span> <span data-x="dom-MessageChannel-port1">port1</span>;
  readonly attribute <span>MessagePort</span> <span data-x="dom-MessageChannel-port2">port2</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>channel</var> = new <span subdfn data-x="dom-MessageChannel">MessageChannel</span>()</code></dt>

   <dd>
    <p>Returns a new <code>MessageChannel</code> object with two new <code>MessagePort</code>
    objects.</p>
   </dd>

   <dt><code data-x=""><var>channel</var>.<span subdfn data-x="dom-MessageChannel-port1">port1</span></code></dt>

   <dd><p>Returns the first <code>MessagePort</code> object.</p></dd>

   <dt><code data-x=""><var>channel</var>.<span subdfn data-x="dom-MessageChannel-port2">port2</span></code></dt>

   <dd><p>Returns the second <code>MessagePort</code> object.</p></dd>
  </dl>

  <div w-nodev>

  <p>A <code>MessageChannel</code> object has an associated <dfn>port 1</dfn> and an associated
  <dfn>port 2</dfn>, both <code>MessagePort</code> objects.</p>

  <p>The <dfn><code data-x="dom-MessageChannel">new MessageChannel()</code></dfn> constructor steps
  are:</p>

  <ol>
   <li><p>Set <span>this</span>'s <span>port 1</span> to a <span>new</span>
   <code>MessagePort</code> in <span>this</span>'s <span data-x="concept-relevant-realm">relevant
   realm</span>.</p></li>

   <li><p>Set <span>this</span>'s <span>port 2</span> to a <span>new</span>
   <code>MessagePort</code> in <span>this</span>'s <span data-x="concept-relevant-realm">relevant
   realm</span>.</p></li>

   <li><p><span>Entangle</span> <span>this</span>'s <span>port 1</span> and <span>this</span>'s
   <span>port 2</span>.</p></li>
  </ol>

  <p>The <dfn attribute for="MessageChannel"><code
  data-x="dom-MessageChannel-port1">port1</code></dfn> getter steps are to return
  <span>this</span>'s <span>port 1</span>.</p>

  <p>The <dfn attribute for="MessageChannel"><code
  data-x="dom-MessageChannel-port2">port2</code></dfn> getter steps are to return
  <span>this</span>'s <span>port 2</span>.</p>

  </div>

  <h4 w-nodev><span id="the-messageeventtarget-abstract-interface"></span>The
  <code>MessageEventTarget</code> mixin</h4>
  <h4 w-dev>Properties present on <code>MessagePort</code>, <code>Worker</code>, and
  <code>DedicatedWorkerGlobalScope</code></h4>

  <pre><code class="idl">interface mixin <dfn interface>MessageEventTarget</dfn> {
  attribute <span>EventHandler</span> <span data-x="handler-MessageEventTarget-onmessage">onmessage</span>;
  attribute <span>EventHandler</span> <span data-x="handler-MessageEventTarget-onmessageerror">onmessageerror</span>;
};</code></pre>

   <p>The following are the <span>event handlers</span> (and their corresponding <span data-x="event
   handler event type">event handler event types</span>) <span w-nodev>that must be</span> supported,
   as <span>event handler IDL attributes</span>, by <span w-nodev>objects implementing the
   <code>MessageEventTarget</code> interface</span><span w-dev><code>MessagePort</code>,
   <code>Worker</code>, and <code>DedicatedWorkerGlobalScope</code> objects</span>:</p>

   <table>
    <thead>
     <tr><th><span data-x="event handlers">Event handler</span> <th><span>Event handler event type</span>
    <tbody>
     <tr><td><span id="handler-messageport-onmessage"></span><span id="handler-worker-onmessage"></span><span id="handler-dedicatedworkerglobalscope-onmessage"></span><dfn attribute for="MessageEventTarget"><code data-x="handler-MessageEventTarget-onmessage">onmessage</code></dfn> <td> <code data-x="event-message">message</code>
     <tr><td><span id="handler-messageport-onmessageerror"></span><span id="handler-worker-onmessageerror"></span><span id="handler-dedicatedworkerglobalscope-onmessageerror"></span><dfn attribute for="MessageEventTarget"><code data-x="handler-MessageEventTarget-onmessageerror">onmessageerror</code></dfn> <td> <code data-x="event-messageerror">messageerror</code>
   </table>

  <h4>Message ports</h4>

  <p>Each channel has two message ports. Data sent through one port is received by the other port,
  and vice versa.</p>

  <pre><code class="idl">[Exposed=(Window,Worker,AudioWorklet), <span>Transferable</span>]
interface <dfn interface>MessagePort</dfn> : <span>EventTarget</span> {
  undefined <span data-x="dom-MessagePort-postMessage">postMessage</span>(any message, sequence&lt;<span data-x="idl-object">object</span>&gt; transfer);
  undefined <span data-x="dom-MessagePort-postMessage-options">postMessage</span>(any message, optional <span>StructuredSerializeOptions</span> options = {});
  undefined <span data-x="dom-MessagePort-start">start</span>();
  undefined <span data-x="dom-MessagePort-close">close</span>();

  // event handlers
  attribute <span>EventHandler</span> <span data-x="handler-MessagePort-onclose">onclose</span>;
};

<span>MessagePort</span> includes <span>MessageEventTarget</span>;

dictionary <dfn dictionary>StructuredSerializeOptions</dfn> {
  sequence&lt;<span data-x="idl-object">object</span>&gt; <dfn dict-member for="StructuredSerializeOptions" data-x="dom-StructuredSerializeOptions-transfer">transfer</dfn> = [];
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>port</var>.<span subdfn data-x="dom-MessagePort-postMessage">postMessage</span>(<var>message</var> [, <var>transfer</var>])</code></dt>
   <dt><code data-x=""><var>port</var>.<span subdfn data-x="dom-MessagePort-postMessage-options">postMessage</span>(<var>message</var> [, { <span data-x="">transfer</span> }])</code></dt>
   <dd>
    <p>Posts a message through the channel. Objects listed in <var>transfer</var> are transferred,
    not just cloned, meaning that they are no longer usable on the sending side.</p>

    <p>Throws a <span>"<code>DataCloneError</code>"</span> <code>DOMException</code> if
    <var>transfer</var> contains duplicate objects or <var>port</var>, or if <var>message</var>
    could not be cloned.</p>
   </dd>

   <dt><code data-x=""><var>port</var>.<span subdfn data-x="dom-MessagePort-start">start</span>()</code></dt>
   <dd><p>Begins dispatching messages received on the port.</p></dd>

   <dt><code data-x=""><var>port</var>.<span subdfn data-x="dom-MessagePort-close">close</span>()</code></dt>
   <dd><p>Disconnects the port, so that it is no longer active.</p></dd>
  </dl>

  <div w-nodev>

  <p>Each <code>MessagePort</code> object has a <dfn>message event target</dfn> (a
  <code>MessageEventTarget</code>), to which the <code data-x="event-message">message</code> and <code
  data-x="event-messageerror">messageerror</code> events are dispatched. Unless otherwise
  specified, it defaults to the <code>MessagePort</code> object itself.</p>

  <p>Each <code>MessagePort</code> object can be entangled with another (a symmetric relationship).
  Each <code>MessagePort</code> object also has a <span>task source</span> called the <dfn>port
  message queue</dfn>, initially empty. A <span>port message queue</span> can be enabled or
  disabled, and is initially disabled. Once enabled, a port can never be disabled again (though
  messages in the queue can get moved to another queue or removed altogether, which has much the
  same effect). A <code>MessagePort</code> also has a <dfn>has been shipped</dfn> flag, which must
  initially be false.</p>

  <p>When a port's <span>port message queue</span> is enabled, the <span>event loop</span> must use
  it as one of its <span data-x="task source">task sources</span>. When a port's <span>relevant
  global object</span> is a <code>Window</code>, all <span data-x="concept-task">tasks</span> <span
  data-x="queue a task">queued</span> on its <span>port message queue</span> must be associated with
  the port's <span>relevant global object</span>'s <span data-x="concept-document-window">associated
  <code>Document</code></span>.</p>

  <p class="note">If the document is <span>fully active</span>, but the event listeners were all
  created in the context of documents that are <em>not</em> <span>fully active</span>, then the
  messages will not be received unless and until the documents become <span>fully active</span>
  again.</p>

  <p>Each <span>event loop</span> has a <span>task source</span> called the <dfn>unshipped port
  message queue</dfn>. This is a virtual <span>task source</span>: it must act as if it contained
  the <span data-x="concept-task">tasks</span> of each <span>port message queue</span> of each
  <code>MessagePort</code> whose <span>has been shipped</span> flag is false, whose <span>port
  message queue</span> is enabled, and whose <span>relevant agent</span>'s <span
  data-x="concept-agent-event-loop">event loop</span> is that <span>event loop</span>, in the order
  in which they were added to their respective <span>task source</span>. When a <span
  data-x="concept-task">task</span> would be removed from the <span>unshipped port message
  queue</span>, it must instead be removed from its <span>port message queue</span>.</p>

  <p>When a <code>MessagePort</code>'s <span>has been shipped</span> flag is false, its <span>port
  message queue</span> must be ignored for the purposes of the <span>event loop</span>. (The
  <span>unshipped port message queue</span> is used instead.)</p>

  <p class="note">The <span>has been shipped</span> flag is set to true when a port, its twin, or
  the object it was cloned from, is or has been transferred. When a <code>MessagePort</code>'s
  <span>has been shipped</span> flag is true, its <span>port message queue</span> acts as a
  first-class <span>task source</span>, unaffected to any <span>unshipped port message
  queue</span>.</p>

  <p>When the user agent is to <dfn>entangle</dfn> two <code>MessagePort</code> objects, it must run
  the following steps:</p>

  <ol>
   <li>
    <p>If one of the ports is already entangled, then disentangle it and the port that it was
    entangled with.</p>

    <p class="note">If those two previously entangled ports were the two ports of a
    <code>MessageChannel</code> object, then that <code>MessageChannel</code> object no longer
    represents an actual channel: the two ports in that object are no longer entangled.</p>
   </li>

   <li>
    <p>Associate the two ports to be entangled, so that they form the two parts of a new channel.
    (There is no <code>MessageChannel</code> object that represents this channel.)</p>

    <p>Two ports <var>A</var> and <var>B</var> that have gone through this step
    are now said to be entangled; one is entangled to the other, and vice versa.</p>

    <p class="note">While this specification describes this process as instantaneous,
    implementations are more likely to implement it via message passing. As with all algorithms, the
    key is "merely" that the end result be indistinguishable, in a black-box sense, from the
    specification.</p>
   </li>
  </ol>

  <p>The <dfn>disentangle</dfn> steps, given a <code>MessagePort</code> <var>initiatorPort</var>
  which initiates disentangling, are as follows:</p>

  <ol>
   <li><p>Let <var>otherPort</var> be the <code>MessagePort</code> which <var>initiatorPort</var>
   was entangled with.</p></li>

   <li><p><span>Assert</span>: <var>otherPort</var> exists.</p></li>

   <li><p>Disentangle <var>initiatorPort</var> and <var>otherPort</var>, so that they are no longer
   entangled or associated with each other.</p></li>

   <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
   data-x="event-close">close</code> at <var>otherPort</var>.</p></li>
  </ol>

  <div class="note">
   <p>The <code data-x="event-close">close</code> event will be fired even if the port is not
   explicitly closed. The cases where this event is dispatched are:</p>

   <ul>
    <li>the <code data-x="dom-MessagePort-close">close()</code> method was called;</li>

    <li>the <code>Document</code> was <span data-x="destroy a document">destroyed</span>; or</li>

    <li>the <code>MessagePort</code> was <a href="#ports-and-garbage-collection">garbage
    collected</a>.</li>
   </ul>

   <p>We only dispatch the event on <var>otherPort</var> because <var>initiatorPort</var> explicitly
   triggered the close, its <code>Document</code> no longer exists, or it was already garbage
   collected, as described above.</p>
  </div>

  <hr>

  <p id="transferMessagePort"><code>MessagePort</code> objects are <span>transferable
  objects</span>. Their <span>transfer steps</span>, given <var>value</var> and
  <var>dataHolder</var>, are:</p>

  <ol>
   <li><p>Set <var>value</var>'s <span>has been shipped</span> flag to true.</p></li>

   <li><p>Set <var>dataHolder</var>.[[PortMessageQueue]] to <var>value</var>'s <span>port message
   queue</span>.</p></li>

   <li>
    <p>If <var>value</var> is entangled with another port <var>remotePort</var>, then:</p>

    <ol>
     <li><p>Set <var>remotePort</var>'s <span>has been shipped</span> flag to true.</p></li>

     <li><p>Set <var>dataHolder</var>.[[RemotePort]] to <var>remotePort</var>.</p></li>
    </ol>
   </li>

   <li><p>Otherwise, set <var>dataHolder</var>.[[RemotePort]] to null.</p></li>
  </ol>

  <p>Their <span>transfer-receiving steps</span>, given <var>dataHolder</var> and <var>value</var>,
  are:</p>

  <ol>
   <li><p>Set <var>value</var>'s <span>has been shipped</span> flag to true.</p></li>

   <li><p>Move all the <span data-x="concept-task">tasks</span> that are to fire <code
   data-x="event-message">message</code> events in <var>dataHolder</var>.[[PortMessageQueue]] to the
   <span>port message queue</span> of <var>value</var>, if any, leaving <var>value</var>'s
   <span>port message queue</span> in its initial disabled state, and, if <var>value</var>'s
   <span>relevant global object</span> is a <code>Window</code>, associating the moved <span
   data-x="concept-task">tasks</span> with <var>value</var>'s <span>relevant global object</span>'s
   <span data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>If <var>dataHolder</var>.[[RemotePort]] is not null, then <span>entangle</span>
   <var>dataHolder</var>.[[RemotePort]] and <var>value</var>. (This will disentangle
   <var>dataHolder</var>.[[RemotePort]] from the original port that was transferred.)</p></li>
  </ol>

  <hr>

  <p>The <dfn>message port post message steps</dfn>, given <var>sourcePort</var>,
  <var>targetPort</var>, <var>message</var>, and <var>options</var> are as follows:</p>

  <ol> <!-- a lot of this is similar or identical to the window post message steps -->
   <li><p>Let <var>transfer</var> be <var>options</var>["<code
   data-x="dom-StructuredSerializeOptions-transfer">transfer</code>"].</p></li>

   <li><p>If <var>transfer</var> <span data-x="list contains">contains</span>
   <var>sourcePort</var>, then throw a <span>"<code>DataCloneError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Let <var>doomed</var> be false.</p></li>

   <li><p>If <var>targetPort</var> is not null and <var>transfer</var> <span data-x="list
   contains">contains</span> <var>targetPort</var>, then set <var>doomed</var> to true and
   optionally report to a developer console that the target port was posted to itself, causing the
   communication channel to be lost.</p></li>

   <li><p>Let <var>serializeWithTransferResult</var> be
   <span>StructuredSerializeWithTransfer</span>(<var>message</var>, <var>transfer</var>). Rethrow
   any exceptions.</p></li>

   <li><p>If <var>targetPort</var> is null, or if <var>doomed</var> is true, then return.</p></li>
   <!-- we don't throw an exception if there is no target port because this can happen at a moment's
   notice. we don't return false because if the port is _about_ to be closed, the message might not
   be listened for anyway. we don't do this before the steps above because otherwise you can tell
   the difference (in whether the ports have become useless, or in when the structured clone
   algorithm runs scripts). We don't throw an exception for 'doomed' being true, because this can't
   necessarily be detected right now every time -->

   <li>
    <p>Add a <span data-x="concept-task">task</span> that runs the following steps to the <span>port
    message queue</span> of <var>targetPort</var>:</p>

    <ol>
     <li>
      <p>Let <var>finalTargetPort</var> be the <code>MessagePort</code> in whose <span>port message
      queue</span> the task now finds itself.</p>

      <p class="note">This can be different from <var>targetPort</var>, if <var>targetPort</var>
      itself was transferred and thus all its tasks moved along with it.</p>
     </li>

     <li><p>Let <var>messageEventTarget</var> be <var>finalTargetPort</var>'s <span>message event
     target</span>.</p></li>

     <li><p>Let <var>targetRealm</var> be <var>finalTargetPort</var>'s <span
     data-x="concept-relevant-realm">relevant realm</span>.</p></li>

     <li>
      <p>Let <var>deserializeRecord</var> be
      <span>StructuredDeserializeWithTransfer</span>(<var>serializeWithTransferResult</var>,
      <var>targetRealm</var>).</p>

      <p>If this throws an exception, catch it, <span data-x="concept-event-fire">fire an
      event</span> named <code data-x="event-messageerror">messageerror</code> at
      <var>messageEventTarget</var>, using <code>MessageEvent</code>, and then return.
     </li>

     <li><p>Let <var>messageClone</var> be <var>deserializeRecord</var>.[[Deserialized]].</p></li>

     <li><p>Let <var>newPorts</var> be a new <span>frozen array</span> consisting of all
     <code>MessagePort</code> objects in <var>deserializeRecord</var>.[[TransferredValues]], if any,
     maintaining their relative order.</p></li>

     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-message">message</code> at <var>messageEventTarget</var>, using
     <code>MessageEvent</code>, with the <code data-x="dom-MessageEvent-data">data</code> attribute
     initialized to <var>messageClone</var> and the <code
     data-x="dom-MessageEvent-ports">ports</code> attribute initialized to
     <var>newPorts</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>The <dfn method for="MessagePort"><code
  data-x="dom-MessagePort-postMessage-options">postMessage(<var>message</var>,
  <var>options</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>targetPort</var> be the port with which <span>this</span> is entangled, if any;
   otherwise let it be null.</p></li>

   <li><p>Run the <span>message port post message steps</span> providing <span>this</span>,
   <var>targetPort</var>, <var>message</var> and <var>options</var>.</p></li>
  </ol>

  <p>The <dfn method for="MessagePort"><code
  data-x="dom-MessagePort-postMessage">postMessage(<var>message</var>,
  <var>transfer</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>targetPort</var> be the port with which <span>this</span> is entangled, if any;
   otherwise let it be null.</p></li>

   <li><p>Let <var>options</var> be «[ "<code
   data-x="dom-StructuredSerializeOptions-transfer">transfer</code>" → <var>transfer</var>
   ]».</p></li>

   <li><p>Run the <span>message port post message steps</span> providing <span>this</span>,
   <var>targetPort</var>, <var>message</var> and <var>options</var>.</p></li>
  </ol>

  <hr>

  <p>The <dfn method for="MessagePort"><code data-x="dom-MessagePort-start">start()</code></dfn>
  method steps are to enable <span>this</span>'s <span>port message queue</span>, if it is not
  already enabled.</p>

  <hr>

  <p>The <dfn method for="MessagePort"><code data-x="dom-MessagePort-close">close()</code></dfn>
  method steps are:</p>

  <ol>
   <li><p>Set <span>this</span>'s <span>[[Detached]]</span> internal slot value to true.</p></li>

   <li><p>If <span>this</span> is entangled, <span>disentangle</span> it.</p></li>
  </ol>

  <hr>

  <p>The following are the <span>event handlers</span> (and their corresponding <span data-x="event
  handler event type">event handler event types</span>) <span w-nodev>that must be</span> supported,
  as <span>event handler IDL attributes</span>, by all objects implementing the
  <code>MessagePort</code> interface:</p>

  <table>
   <thead>
    <tr><th><span data-x="event handlers">Event handler</span> <th><span>Event handler event type</span>
   <tbody>
    <tr><td><dfn attribute for="MessagePort"><code data-x="handler-MessagePort-onclose">onclose</code></dfn> <td> <code data-x="event-close">close</code>
  </table>

  <p>The first time a <code>MessagePort</code> object's <code
  data-x="handler-MessageEventTarget-onmessage">onmessage</code> IDL attribute is set, the port's
  <span>port message queue</span> must be enabled, as if the <code
  data-x="dom-MessagePort-start">start()</code> method had been called.</p>

  </div>


  <h4 id="ports-and-garbage-collection">Ports and garbage collection</h4>

  <div w-nodev>
  <p>When a <code>MessagePort</code> object <var>o</var> is garbage collected, if <var>o</var> is
  entangled, then the user agent must <span data-x="disentangle">disentangle</span>
  <var>o</var>.</p>

  <p>When a <code>MessagePort</code> object <var>o</var> is entangled and <code
  data-x="event-message">message</code> or <code data-x="event-messageerror">messageerror</code>
  event listener is registered, user agents must act as if <var>o</var>'s entangled
  <code>MessagePort</code> object has a strong reference to <var>o</var>.</p>

  <p>Furthermore, a <code>MessagePort</code> object must not be garbage collected while there
  exists an event referenced by a <span data-x="concept-task">task</span> in a <span>task
  queue</span> that is to be dispatched on that <code>MessagePort</code> object, or while the
  <code>MessagePort</code> object's <span>port message queue</span> is enabled and not empty.</p>
  <!-- we might not need to explicitly say the first part if UI Events is fixed to say that events
  on a task queue prevent GC -->

  <!-- ports in the ports attribute of a MessageEvent that isn't dispatched yet are safe because
  the MessageEvent is safe -->

  <div class="note">

   <p>Thus, a message port can be received, given an event listener, and then forgotten, and so long
   as that event listener could receive a message, the channel will be maintained.</p>

   <p>Of course, if this was to occur on both sides of the channel, then both ports could be garbage
   collected, since they would not be reachable from live code, despite having a strong reference to
   each other. However, if a message port has a pending message, it is not garbage collected.</p>

  </div>

  </div>

  <p class="note">Authors are strongly encouraged to explicitly close <code>MessagePort</code>
  objects to disentangle them, so that their resources can be recollected. Creating many
  <code>MessagePort</code> objects and discarding them without closing them can lead to high
  transient memory usage since garbage collection is not necessarily performed promptly, especially
  for <code>MessagePort</code>s where garbage collection can involve cross-process coordination.</p>



  <h3><dfn data-x="broadcast channels">Broadcasting to other browsing contexts</dfn></h3>

  <p>Pages on a single <span>origin</span> opened by the same user in the same user agent but in
  different unrelated <span data-x="browsing context">browsing contexts</span> sometimes need to
  send notifications to each other, for example "hey, the user logged in over here, check your
  credentials again".</p>

  <p>For elaborate cases, e.g. to manage locking of shared state, to manage synchronization of
  resources between a server and multiple local clients, to share a <code>WebSocket</code>
  connection with a remote host, and so forth, <span data-x="SharedWorker">shared workers</span> are
  the most appropriate solution.</p>

  <p>For simple cases, though, where a shared worker would be an unreasonable overhead, authors can
  use the simple channel-based broadcast mechanism described in this section.</p>

  <pre><code class="idl">[Exposed=(Window,Worker)]
interface <dfn interface>BroadcastChannel</dfn> : <span>EventTarget</span> {
  <span data-x="dom-BroadcastChannel">constructor</span>(DOMString name);

  readonly attribute DOMString <span data-x="dom-BroadcastChannel-name">name</span>;
  undefined <span data-x="dom-BroadcastChannel-postMessage">postMessage</span>(any message);
  undefined <span data-x="dom-BroadcastChannel-close">close</span>();
  attribute <span>EventHandler</span> <span data-x="handler-BroadcastChannel-onmessage">onmessage</span>;
  attribute <span>EventHandler</span> <span data-x="handler-BroadcastChannel-onmessageerror">onmessageerror</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>broadcastChannel</var> = new <span subdfn data-x="dom-BroadcastChannel">BroadcastChannel</span>(<var>name</var>)</code></dt>

   <dd>
    <p>Returns a new <code>BroadcastChannel</code> object via which messages for the given channel
    name can be sent and received.</p>
   </dd>

   <dt><code data-x=""><var>broadcastChannel</var>.<span subdfn data-x="dom-BroadcastChannel-name">name</span></code></dt>

   <dd><p>Returns the channel name (as passed to the constructor).</p></dd>

   <dt><code data-x=""><var>broadcastChannel</var>.<span subdfn data-x="dom-BroadcastChannel-postMessage">postMessage</span>(<var>message</var>)</code></dt>

   <dd>
    <p>Sends the given message to other <code>BroadcastChannel</code> objects set up for this
    channel. Messages can be structured objects, e.g. nested objects and arrays.</p>
   </dd>

   <dt><code data-x=""><var>broadcastChannel</var>.<span subdfn data-x="dom-BroadcastChannel-close">close</span>()</code></dt>

   <dd><p>Closes the <code>BroadcastChannel</code> object, opening it up to garbage
   collection.</p></dd>
  </dl>

  <div w-nodev>

  <p>A <code>BroadcastChannel</code> object has a <dfn>channel name</dfn> and a <dfn
  data-x="concept-BroadcastChannel-closed">closed flag</dfn>.</p>

  <p>The <dfn><code data-x="dom-BroadcastChannel">new BroadcastChannel(<var>name</var>)</code></dfn>
  constructor steps are:</p>

  <ol>
   <li><p>Set <span>this</span>'s <span>channel name</span> to <var>name</var>.</p></li>

   <li><p>Set <span>this</span>'s <span data-x="concept-BroadcastChannel-closed">closed flag</span>
   to false.</p></li>
  </ol>

  <p>The <dfn attribute for="BroadcastChannel"><code
  data-x="dom-BroadcastChannel-name">name</code></dfn> getter steps are to return
  <span>this</span>'s <span>channel name</span>.</p>

  <p>A <code>BroadcastChannel</code> object is said to be <dfn>eligible for messaging</dfn> when
  its <span>relevant global object</span> is either:</p>

  <ul>
   <li><p>a <code>Window</code> object whose <span data-x="concept-document-window">associated
   <code>Document</code></span> is <span>fully active</span>, or</p></li>

   <li><p>a <code>WorkerGlobalScope</code> object whose <span
   data-x="dom-WorkerGlobalScope-closing">closing</span> flag is false and whose
   <span>worker</span> is not a <span>suspendable worker</span>.</p></li>
  </ul>

  <p>The <dfn method for="BroadcastChannel"><code
  data-x="dom-BroadcastChannel-postMessage">postMessage(<var>message</var>)</code></dfn> method
  steps are:</p>

  <ol>
   <li><p>If <span>this</span> is not <span>eligible for messaging</span>, then return.</p></li>

   <li><p>If <span>this</span>'s <span data-x="concept-BroadcastChannel-closed">closed flag</span>
   is true, then throw an <span>"<code>InvalidStateError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Let <var>serialized</var> be <span>StructuredSerialize</span>(<var>message</var>). Rethrow
   any exceptions.</p></li>

   <li><p>Let <var>sourceOrigin</var> be <span>this</span>'s <span>relevant settings object</span>'s
   <span data-x="concept-settings-object-origin">origin</span>.</p></li>

   <li><p>Let <var>sourceStorageKey</var> be the result of running <span>obtain a storage key for
   non-storage purposes</span> with <span>this</span>'s <span>relevant settings
   object</span>.</p></li>

   <li>
    <p>Let <var>destinations</var> be a list of <code>BroadcastChannel</code> objects that
    match the following criteria:</p>

    <ul>
     <li><p>They are <span>eligible for messaging</span>.</p></li>

     <li><p>The result of running <span>obtain a storage key for non-storage purposes</span> with
     their <span>relevant settings object</span> <span data-x="storage key equal">equals</span>
     <var>sourceStorageKey</var>.</p></li>

     <li><p>Their <span>channel name</span> <span>is</span> <span>this</span>'s <span>channel
     name</span>.</p></li>
    </ul>
   </li>

   <li><p>Remove <var>source</var> from <var>destinations</var>.</p></li>

   <li><p>Sort <var>destinations</var> such that all <code>BroadcastChannel</code> objects whose
   <span data-x="relevant agent">relevant agents</span> are the same are sorted in creation order,
   oldest first. (This does not define a complete ordering. Within this constraint, user agents may
   sort the list in any <span>implementation-defined</span> manner.)</p></li>

   <li>
    <p>For each <var>destination</var> in <var>destinations</var>, <span>queue a global task</span>
    on the <span>DOM manipulation task source</span> given <var>destination</var>'s <span>relevant
    global object</span> to perform the following steps:</p>

    <ol>
     <li><p>If <var>destination</var>'s <span data-x="concept-BroadcastChannel-closed">closed
     flag</span> is true, then abort these steps.</p></li>

     <li><p>Let <var>targetRealm</var> be <var>destination</var>'s <span
     data-x="concept-relevant-realm">relevant realm</span>.</p></li>

     <li>
      <p>Let <var>data</var> be <span>StructuredDeserialize</span>(<var>serialized</var>,
      <var>targetRealm</var>).</p>

      <p>If this throws an exception, catch it, <span data-x="concept-event-fire">fire an
      event</span> named <code data-x="event-messageerror">messageerror</code> at
      <var>destination</var>, using <code>MessageEvent</code>, with the <code
      data-x="dom-MessageEvent-origin">origin</code> attribute initialized to the <span
      data-x="serialization of an origin">serialization</span> of <var>sourceOrigin</var>, and then
      abort these steps.</p>
     </li>

     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-message">message</code> at <var>destination</var>, using
     <code>MessageEvent</code>, with the <code data-x="dom-MessageEvent-data">data</code> attribute
     initialized to <var>data</var> and the <code data-x="dom-MessageEvent-origin">origin</code>
     attribute initialized to the <span data-x="serialization of an origin">serialization</span> of
     <var>sourceOrigin</var>.</p></li>
    </ol>
   </li>
  </ol>

  <p>While a <code>BroadcastChannel</code> object whose <span
  data-x="concept-BroadcastChannel-closed">closed flag</span> is false has an event listener
  registered for <code data-x="event-message">message</code> or <code
  data-x="event-messageerror">messageerror</code> events, there must be a strong reference from the
  <code>BroadcastChannel</code> object's <span>relevant global object</span> to the
  <code>BroadcastChannel</code> object itself.</p>

  <p>The <dfn method for="BroadcastChannel"><code
  data-x="dom-BroadcastChannel-close">close()</code></dfn> method steps are to set
  <span>this</span>'s <span data-x="concept-BroadcastChannel-closed">closed flag</span> to true.</p>

  </div>

  <p class="note">Authors are strongly encouraged to explicitly close <code>BroadcastChannel</code>
  objects when they are no longer needed, so that they can be garbage collected. Creating many
  <code>BroadcastChannel</code> objects and discarding them while leaving them with an event
  listener and without closing them can lead to an apparent memory leak, since the objects will
  continue to live for as long as they have an event listener (or until their page or worker is
  closed).</p>

  <div w-nodev>

  <hr>

  <p>The following are the <span>event handlers</span> (and their corresponding <span data-x="event
  handler event type">event handler event types</span>) <span w-nodev>that must be</span> supported,
  as <span>event handler IDL attributes</span>, by all objects implementing the
  <code>BroadcastChannel</code> interface:</p>

  <table>
   <thead>
    <tr><th><span data-x="event handlers">Event handler</span> <th><span>Event handler event type</span>
   <tbody>
    <tr><td><dfn attribute for="BroadcastChannel"><code data-x="handler-BroadcastChannel-onmessage">onmessage</code></dfn> <td> <code data-x="event-message">message</code>
    <tr><td><dfn attribute for="BroadcastChannel"><code data-x="handler-BroadcastChannel-onmessageerror">onmessageerror</code></dfn> <td> <code data-x="event-messageerror">messageerror</code>
  </table>

  </div>

  <div class="example">

   <p>Suppose a page wants to know when the user logs out, even when the user does so from another
   tab at the same site:</p>

   <pre><code class="js">var authChannel = new BroadcastChannel('auth');
authChannel.onmessage = function (event) {
  if (event.data == 'logout')
    showLogout();
}

function logoutRequested() {
  // called when the user asks us to log them out
  doLogout();
  showLogout();
  authChannel.postMessage('logout');
}

function doLogout() {
  // actually log the user out (e.g. clearing cookies)
  // ...
}

function showLogout() {
  // update the UI to indicate we're logged out
  // ...
}</code></pre>

  </div>



  <h2 split-filename="workers" id="workers" dfn data-lt="web worker" export>Web workers</h2>

  <h3>Introduction</h3>

  <h4>Scope</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>This specification defines an API for running scripts in the background independently of any
  user interface scripts.</p>

  <p>This allows for long-running scripts that are not interrupted by scripts that respond to clicks
  or other user interactions, and allows long tasks to be executed without yielding to keep the page
  responsive.</p>

  <p>Workers (as these background scripts are called herein) are relatively heavy-weight, and are
  not intended to be used in large numbers. For example, it would be inappropriate to launch one
  worker for each pixel of a four megapixel image. The examples below show some appropriate uses of
  workers.</p>

  <p>Generally, workers are expected to be long-lived, have a high start-up performance cost, and a
  high per-instance memory cost.</p>


  <h4>Examples</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>There are a variety of uses that workers can be put to. The following subsections show various
  examples of this use.</p>


  <h5>A background number-crunching worker</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>The simplest use of workers is for performing a computationally
  expensive task without interrupting the user interface.</p>

  <p>In this example, the main document spawns a worker to (na&iuml;vely) compute prime numbers, and
  progressively displays the most recently found prime number.</p>

  <p>The main page is as follows:</p>

  <pre><code class="html">EXAMPLE workers/primes/page.html</code></pre>

  <p>The <code data-x="dom-Worker">Worker()</code> constructor call creates a worker and returns a
  <code>Worker</code> object representing that worker, which is used to communicate with the worker.
  That object's <code data-x="handler-MessageEventTarget-onmessage">onmessage</code> event handler
  allows the code to receive messages from the worker.</p>

  <p>The worker itself is as follows:</p>

  <pre><code class="js">EXAMPLE workers/primes/worker.js</code></pre>

  <p>The bulk of this code is simply an unoptimized search for a prime number. The <code
  data-x="dom-DedicatedWorkerGlobalScope-postMessage">postMessage()</code> method is used to send a
  message back to the page when a prime is found.</p>

  <p><a href="/demos/workers/primes/page.html">View this example online</a>.</p>


  <h5 id="module-worker-example">Using a JavaScript module as a worker</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>All of our examples so far show workers that run <span data-x="classic script">classic
  scripts</span>. Workers can instead be instantiated using <span data-x="module script">module
  scripts</span>, which have the usual benefits: the ability to use the JavaScript
  <code data-x="">import</code> statement to import other modules; strict mode by default; and
  top-level declarations not polluting the worker's global scope.</p>

  <p>As the <code data-x="">import</code> statement is available, the <code
  data-x="dom-WorkerGlobalScope-importScripts">importScripts()</code> method will automatically fail
  inside module workers.</p>

  <p>In this example, the main document uses a worker to do off-main-thread image manipulation.
  It imports the filters used from another module.</p>

  <p>The main page is as follows:</p>

  <pre><code class="html">EXAMPLE workers/modules/page.html</code></pre>

  <p>The worker file is then:</p>

  <pre><code class="js">EXAMPLE workers/modules/worker.js</code></pre>

  <p>Which imports the file <code data-x="">filters.js</code>:</p>

  <pre><code class="js">EXAMPLE workers/modules/filters.js</code></pre>

  <p><a href="/demos/workers/modules/page.html">View this example online</a>.</p>


  <h5>Shared workers introduction</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>This section introduces shared workers using a Hello World example. Shared workers use slightly
  different APIs, since each worker can have multiple connections.</p>

  <p>This first example shows how you connect to a worker and how a worker can send a message back
  to the page when it connects to it. Received messages are displayed in a log.</p>

  <p>Here is the HTML page:</p>

  <pre><code class="html">EXAMPLE workers/shared/001/test.html</code></pre>

  <p>Here is the JavaScript worker:</p>

  <pre><code class="js">EXAMPLE workers/shared/001/test.js</code></pre>

  <p><a href="/demos/workers/shared/001/test.html">View this example online</a>.</p>

  <hr>

  <p>This second example extends the first one by changing two things: first, messages are received
  using <code data-x="">addEventListener()</code> instead of an <span data-x="event handler IDL
  attributes">event handler IDL attribute</span>, and second, a message is sent <em>to</em> the
  worker, causing the worker to send another message in return. Received messages are again
  displayed in a log.</p>

  <p>Here is the HTML page:</p>

  <pre><code class="html">EXAMPLE workers/shared/002/test.html</code></pre>

  <p>Here is the JavaScript worker:</p>

  <pre><code class="js">EXAMPLE workers/shared/002/test.js</code></pre>

  <p><a href="/demos/workers/shared/002/test.html">View this example online</a>.</p>

  <hr>

  <p>Finally, the example is extended to show how two pages can connect to the same worker; in this
  case, the second page is merely in an <code>iframe</code> on the first page, but the same
  principle would apply to an entirely separate page in a separate <span>top-level
  traversable</span>.</p>

  <p>Here is the outer HTML page:</p>

  <pre><code class="html">EXAMPLE workers/shared/003/test.html</code></pre>

  <p>Here is the inner HTML page:</p>

  <pre><code class="html">EXAMPLE workers/shared/003/inner.html</code></pre>

  <p>Here is the JavaScript worker:</p>

  <pre><code class="js">EXAMPLE workers/shared/003/test.js</code></pre>

  <p><a href="/demos/workers/shared/003/test.html">View this example online</a>.</p>


  <h5>Shared state using a shared worker</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>In this example, multiple windows (viewers) can be opened that are all viewing the same map.
  All the windows share the same map information, with a single worker coordinating all the viewers.
  Each viewer can move around independently, but if they set any data on the map, all the viewers
  are updated.</p>

  <p>The main page isn't interesting, it merely provides a way to open the viewers:</p>

  <pre><code class="html">EXAMPLE workers/multiviewer/page.html</code></pre>

  <p>The viewer is more involved:</p>

  <pre><code class="html">EXAMPLE workers/multiviewer/viewer.html</code></pre>

  <p>There are several key things worth noting about the way the viewer is written.</p>

  <p><strong>Multiple listeners</strong>. Instead of a single message processing function, the code
  here attaches multiple event listeners, each one performing a quick check to see if it is relevant
  for the message. In this example it doesn't make much difference, but if multiple authors wanted
  to collaborate using a single port to communicate with a worker, it would allow for independent
  code instead of changes having to all be made to a single event handling function.</p>

  <p>Registering event listeners in this way also allows you to unregister specific listeners when
  you are done with them, as is done with the <code data-x="">configure()</code> method in this
  example.</p>

  <p>Finally, the worker:</p>

  <pre><code class="js">EXAMPLE workers/multiviewer/worker.js</code></pre>

  <p><strong>Connecting to multiple pages</strong>. The script uses the <code
  data-x="handler-SharedWorkerGlobalScope-onconnect">onconnect</code> event listener to listen for
  multiple connections.</p>

  <p><strong>Direct channels</strong>. When the worker receives a "msg" message from one viewer
  naming another viewer, it sets up a direct connection between the two, so that the two viewers can
  communicate directly without the worker having to proxy all the messages.</p>

  <p><a href="/demos/workers/multiviewer/page.html">View this example online</a>.</p>


  <h5>Delegation</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>With multicore CPUs becoming prevalent, one way to obtain better performance is to split
  computationally expensive tasks amongst multiple workers. In this example, a computationally
  expensive task that is to be performed for every number from 1 to 10,000,000 is farmed out to ten
  subworkers.</p>

  <p>The main page is as follows, it just reports the result:</p>

  <pre><code class="html">EXAMPLE workers/multicore/page.html</code></pre>

  <p>The worker itself is as follows:</p>

  <pre><code class="js">EXAMPLE workers/multicore/worker.js</code></pre>

  <p>It consists of a loop to start the subworkers, and then a handler
  that waits for all the subworkers to respond.</p>

  <p>The subworkers are implemented as follows:</p>

  <pre><code class="js">EXAMPLE workers/multicore/core.js</code></pre>

  <p>They receive two numbers in two events, perform the computation for the range of numbers thus
  specified, and then report the result back to the parent.</p>

  <p><a href="/demos/workers/multicore/page.html">View this example online</a>.</p>


  <h5>Providing libraries</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>Suppose that a cryptography library is made available that provides three tasks:</p>

  <dl>
   <dt>Generate a public/private key pair</dt>

   <dd>Takes a port, on which it will send two messages, first the public key and then the private
   key.</dd>

   <dt>Given a plaintext and a public key, return the corresponding ciphertext</dt>

   <dd>Takes a port, to which any number of messages can be sent, the first giving the public key,
   and the remainder giving the plaintext, each of which is encrypted and then sent on that same
   channel as the ciphertext. The user can close the port when it is done encrypting content.</dd>

   <dt>Given a ciphertext and a private key, return the corresponding plaintext</dt>

   <dd>Takes a port, to which any number of messages can be sent, the first giving the private key,
   and the remainder giving the ciphertext, each of which is decrypted and then sent on that same
   channel as the plaintext. The user can close the port when it is done decrypting content.</dd>
  </dl>

  <p>The library itself is as follows:</p>

  <pre><code class="js">EXAMPLE workers/crypto/libcrypto-v1.js</code></pre>

  <p>Note that the crypto functions here are just stubs and don't do real cryptography.</p>

  <p>This library could be used as follows:</p>

  <pre><code class="html">EXAMPLE workers/crypto/page.html</code></pre>

  <p>A later version of the API, though, might want to offload all the crypto work onto subworkers.
  This could be done as follows:</p>

  <pre><code class="js">EXAMPLE workers/crypto/libcrypto-v2.js</code></pre>

  <p>The little subworkers would then be as follows.</p>

  <p>For generating key pairs:</p>

  <pre><code class="js">EXAMPLE workers/crypto/libcrypto-v2-generator.js</code></pre>

  <p>For encrypting:</p>

  <pre><code class="js">EXAMPLE workers/crypto/libcrypto-v2-encryptor.js</code></pre>

  <p>For decrypting:</p>

  <pre><code class="js">EXAMPLE workers/crypto/libcrypto-v2-decryptor.js</code></pre>

  <p>Notice how the users of the API don't have to even know that this is happening &mdash; the API
  hasn't changed; the library can delegate to subworkers without changing its API, even though it is
  accepting data using message channels.</p>

  <p><a href="/demos/workers/crypto/page.html">View this example online</a>.</p>


  <h4>Tutorials</h4>

  <h5>Creating a dedicated worker</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>Creating a worker requires a URL to a JavaScript file. The <code
  data-x="dom-Worker">Worker()</code> constructor is invoked with the URL to that file as its only
  argument; a worker is then created and returned:</p>

  <pre><code class="js">var worker = new Worker('helper.js');</code></pre>

  <p>If you want your worker script to be interpreted as a <span>module script</span> instead of
  the default <span>classic script</span>, you need to use a slightly different signature:</p>

  <pre><code class="js">var worker = new Worker('helper.mjs', { type: "module" });</code></pre>


  <h5>Communicating with a dedicated worker</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>Dedicated workers use <code>MessagePort</code> objects behind the scenes, and thus support all
  the same features, such as sending structured data, transferring binary data, and transferring
  other ports.</p>

  <p>To receive messages from a dedicated worker, use the <code
  data-x="handler-MessageEventTarget-onmessage">onmessage</code> <span data-x="event handler IDL
  attributes">event handler IDL attribute</span> on the <code>Worker</code> object:</p>

  <pre><code class="js">worker.onmessage = function (event) { ... };</code></pre>

  <p>You can also use the <code data-x="dom-EventTarget-addEventListener">addEventListener()</code>
  method.</p>

  <p class="note">The implicit <code>MessagePort</code> used by dedicated workers has its <span>port
  message queue</span> implicitly enabled when it is created, so there is no equivalent to the
  <code>MessagePort</code> interface's <code data-x="dom-MessagePort-start">start()</code> method on
  the <code>Worker</code> interface.</p>

  <p>To <em>send</em> data to a worker, use the <code
  data-x="dom-Worker-postMessage">postMessage()</code> method. Structured data can be sent over this
  communication channel. To send <code data-x="idl-ArrayBuffer">ArrayBuffer</code> objects
  efficiently (by transferring them rather than cloning them), list them in an array in the second
  argument.</p>

  <pre><code class="js">worker.postMessage({
  operation: 'find-edges',
  input: buffer, // an ArrayBuffer object
  threshold: 0.6,
}, [buffer]);</code></pre>

  <p>To receive a message inside the worker, the <code
  data-x="handler-MessageEventTarget-onmessage">onmessage</code> <span data-x="event handler
  IDL attributes">event handler IDL attribute</span> is used.</p>

  <pre><code class="js">onmessage = function (event) { ... };</code></pre>

  <p>You can again also use the <code
  data-x="dom-EventTarget-addEventListener">addEventListener()</code> method.</p>

  <p>In either case, the data is provided in the event object's <code
  data-x="dom-MessageEvent-data">data</code> attribute.</p>

  <p>To send messages back, you again use <code
  data-x="dom-DedicatedWorkerGlobalScope-postMessage">postMessage()</code>. It supports the
  structured data in the same manner.</p>

  <pre><code class="js">postMessage(event.data.input, [event.data.input]); // transfer the buffer back</code></pre>


  <h5 dfn data-lt="shared worker">Shared workers</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>Shared workers are identified by the URL of the script used to create it, optionally with an
  explicit name. The name allows multiple instances of a particular shared worker to be started.</p>

  <p>Shared workers are scoped by <span>origin</span>. Two different sites using the same names will
  not collide. However, if a page tries to use the same shared worker name as another page on the
  same site, but with a different script URL, it will fail.</p>

  <p>Creating shared workers is done using the <code data-x="dom-SharedWorker">SharedWorker()</code>
  constructor. This constructor takes the URL to the script to use for its first argument, and the
  name of the worker, if any, as the second argument.</p>

  <pre><code class="js">var worker = new SharedWorker('service.js');</code></pre>

  <p>Communicating with shared workers is done with explicit <code>MessagePort</code> objects. The
  object returned by the <code data-x="dom-SharedWorker">SharedWorker()</code> constructor holds a
  reference to the port on its <code data-x="dom-SharedWorker-port">port</code> attribute.</p>

  <pre><code class="js">worker.port.onmessage = function (event) { ... };
worker.port.postMessage('some message');
worker.port.postMessage({ foo: 'structured', bar: ['data', 'also', 'possible']});</code></pre>

  <p>Inside the shared worker, new clients of the worker are announced using the <code
  data-x="event-WorkerGlobalScope-connect">connect</code> event. The port for the new client is
  given by the event object's <code data-x="dom-messageevent-source">source</code> attribute.</p>

  <pre><code class="js">onconnect = function (event) {
  var newPort = event.source;
  // set up a listener
  newPort.onmessage = function (event) { ... };
  // send a message back to the port
  newPort.postMessage('ready!'); // can also send structured data, of course
};</code></pre>




  <h3>Infrastructure</h3>

  <p>This standard defines two kinds of workers: dedicated workers, and shared workers. Dedicated
  workers, once created, are linked to their creator, but message ports can be used to communicate
  from a dedicated worker to multiple other browsing contexts or workers. Shared workers, on the
  other hand, are named, and once created any script running in the same <span>origin</span> can
  obtain a reference to that worker and communicate with it. <cite>Service Workers</cite> defines a
  third kind. <ref>SW</ref></p>


  <h4>The global scope</h4>

  <p>The global scope is the "inside" of a worker.</p>

  <h5>The <code subdfn>WorkerGlobalScope</code> common interface</h5>

  <pre><code class="idl">[Exposed=Worker]
interface <dfn interface>WorkerGlobalScope</dfn> : <span>EventTarget</span> {
  readonly attribute <span>WorkerGlobalScope</span> <span data-x="dom-WorkerGlobalScope-self">self</span>;
  readonly attribute <span>WorkerLocation</span> <span data-x="dom-WorkerGlobalScope-location">location</span>;
  readonly attribute <span>WorkerNavigator</span> <span data-x="dom-worker-navigator">navigator</span>;
  undefined <span data-x="dom-WorkerGlobalScope-importScripts">importScripts</span>((<code data-x="tt-trustedscripturl">TrustedScriptURL</code> or USVString)... urls);

  attribute <span>OnErrorEventHandler</span> <span data-x="handler-WorkerGlobalScope-onerror">onerror</span>;
  attribute <span>EventHandler</span> <span data-x="handler-WorkerGlobalScope-onlanguagechange">onlanguagechange</span>;
  attribute <span>EventHandler</span> <span data-x="handler-WorkerGlobalScope-onoffline">onoffline</span>;
  attribute <span>EventHandler</span> <span data-x="handler-WorkerGlobalScope-ononline">ononline</span>;
  attribute <span>EventHandler</span> <span data-x="handler-WorkerGlobalScope-onrejectionhandled">onrejectionhandled</span>;
  attribute <span>EventHandler</span> <span data-x="handler-WorkerGlobalScope-onunhandledrejection">onunhandledrejection</span>;
};</code></pre>

  <p><code>WorkerGlobalScope</code> serves as the base class for specific types of worker global
  scope objects, including <code>DedicatedWorkerGlobalScope</code>,
  <code>SharedWorkerGlobalScope</code>, and <code>ServiceWorkerGlobalScope</code>.</p>

  <div w-nodev>

  <p id="the-worker's-documents">A <code>WorkerGlobalScope</code> object has an associated <dfn
  id="concept-WorkerGlobalScope-owner-set" export for="WorkerGlobalScope">owner set</dfn> (a
  <span>set</span> of <code>Document</code> and <code>WorkerGlobalScope</code> objects). It is
  initially empty and populated when the worker is created or obtained.</p>

  <p class="note">It is a <span>set</span>, instead of a single owner, to accommodate
  <code>SharedWorkerGlobalScope</code> objects.</p>

  <p>A <code>WorkerGlobalScope</code> object has an associated <dfn export for="WorkerGlobalScope"
  data-x="concept-WorkerGlobalScope-type">type</dfn> ("<code data-x="">classic</code>" or "<code
  data-x="">module</code>"). It is set during creation.</p>

  <p>A <code>WorkerGlobalScope</code> object has an associated <dfn export for="WorkerGlobalScope"
  data-x="concept-WorkerGlobalScope-url">url</dfn> (null or a <span>URL</span>). It is initially
  null.</p>

  <p>A <code>WorkerGlobalScope</code> object has an associated <dfn export for="WorkerGlobalScope"
  data-x="concept-WorkerGlobalScope-name">name</dfn> (a string). It is set during creation.</p>

  <p class="note">The <span data-x="concept-WorkerGlobalScope-name">name</span> can have different
  semantics for each subclass of <code>WorkerGlobalScope</code>. For
  <code>DedicatedWorkerGlobalScope</code> instances, it is simply a developer-supplied name, useful
  mostly for debugging purposes. For <code>SharedWorkerGlobalScope</code> instances, it allows
  obtaining a reference to a common shared worker via the <code
  data-x="dom-SharedWorker">SharedWorker()</code> constructor. For
  <code>ServiceWorkerGlobalScope</code> objects, it doesn't make sense (and as such isn't exposed
  through the JavaScript API at all).</p>

  <p>A <code>WorkerGlobalScope</code> object has an associated <dfn export for="WorkerGlobalScope"
  data-x="concept-WorkerGlobalScope-policy-container">policy container</dfn> (a <span>policy
  container</span>). It is initially a new <span>policy container</span>.</p>

  <p>A <code>WorkerGlobalScope</code> object has an associated <dfn export for="WorkerGlobalScope"
  data-x="concept-WorkerGlobalScope-embedder-policy">embedder policy</dfn> (an <span>embedder
  policy</span>).</p>

  <p>A <code>WorkerGlobalScope</code> object has an associated <dfn export for="WorkerGlobalScope"
  data-x="concept-WorkerGlobalScope-module-map">module map</dfn>.  It is a <span>module map</span>,
  initially empty.</p>

  <p>A <code>WorkerGlobalScope</code> object has an associated <dfn export for="WorkerGlobalScope"
  data-x="concept-WorkerGlobalScope-cross-origin-isolated-capability">cross-origin isolated
  capability</dfn> boolean. It is initially false.</p>

  </div>

  <dl class="domintro">
   <dt><code data-x=""><var>workerGlobal</var>.<span subdfn data-x="dom-WorkerGlobalScope-self">self</span></code></dt>
   <dd>Returns <var>workerGlobal</var>.</dd>

   <dt><code data-x=""><var>workerGlobal</var>.<span subdfn data-x="dom-WorkerGlobalScope-location">location</span></code></dt>
   <dd>Returns <var>workerGlobal</var>'s <code>WorkerLocation</code> object.</dd>

   <dt><code data-x=""><var>workerGlobal</var>.<span subdfn data-x="dom-worker-navigator">navigator</span></code></dt>
   <dd>Returns <var>workerGlobal</var>'s <code>WorkerNavigator</code> object.</dd>

   <dt><code data-x=""><var>workerGlobal</var>.<span subdfn data-x="dom-WorkerGlobalScope-importScripts">importScripts</span>(...<var>urls</var>)</code></dt>
   <dd>Fetches each <span>URL</span> in <var>urls</var>, executes them one-by-one in the order they
   are passed, and then returns (or throws if something went amiss).</dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="WorkerGlobalScope"><code
  data-x="dom-WorkerGlobalScope-self">self</code></dfn> attribute must return the
  <code>WorkerGlobalScope</code> object itself.</p>

  <p>The <dfn attribute for="WorkerGlobalScope"><code
  data-x="dom-WorkerGlobalScope-location">location</code></dfn> attribute must return the
  <code>WorkerLocation</code> object whose associated <span
  data-x="concept-WorkerLocation-WorkerGlobalScope"><code>WorkerGlobalScope</code> object</span> is
  the <code>WorkerGlobalScope</code> object.</p>

  <p class="note">While the <code>WorkerLocation</code> object is created after the
  <code>WorkerGlobalScope</code> object, this is not problematic as it cannot be observed from
  script.</p>

  <hr>

  </div>

  <p>The following are the <span>event handlers</span> (and their corresponding <span data-x="event
  handler event type">event handler event types</span>) <span w-nodev>that must be</span> supported,
  as <span>event handler IDL attributes</span>, by objects implementing the
  <code>WorkerGlobalScope</code> interface:</p>

  <table>
   <thead>
    <tr><th><span data-x="event handlers">Event handler</span> <th><span>Event handler event type</span>
   <tbody>
    <tr><td><dfn attribute for="WorkerGlobalScope"><code data-x="handler-WorkerGlobalScope-onerror">onerror</code></dfn> <td> <code data-x="event-error">error</code>
    <tr><td><dfn attribute for="WorkerGlobalScope"><code data-x="handler-WorkerGlobalScope-onlanguagechange">onlanguagechange</code></dfn> <td> <code data-x="event-languagechange">languagechange</code> <!-- new -->
    <tr><td><dfn attribute for="WorkerGlobalScope"><code data-x="handler-WorkerGlobalScope-onoffline">onoffline</code></dfn> <td> <code data-x="event-offline">offline</code> <!-- new -->
    <tr><td><dfn attribute for="WorkerGlobalScope"><code data-x="handler-WorkerGlobalScope-ononline">ononline</code></dfn> <td> <code data-x="event-online">online</code> <!-- new -->
    <tr><td><dfn attribute for="WorkerGlobalScope"><code data-x="handler-WorkerGlobalScope-onrejectionhandled">onrejectionhandled</code></dfn> <td> <code data-x="event-rejectionhandled">rejectionhandled</code>
    <tr><td><dfn attribute for="WorkerGlobalScope"><code data-x="handler-WorkerGlobalScope-onunhandledrejection">onunhandledrejection</code></dfn> <td> <code data-x="event-unhandledrejection">unhandledrejection</code>
  </table>


  <h5>Dedicated workers and the <code subdfn>DedicatedWorkerGlobalScope</code> interface</h5>

  <pre><code class="idl">[Global=(Worker,DedicatedWorker),Exposed=DedicatedWorker]
interface <dfn interface>DedicatedWorkerGlobalScope</dfn> : <span>WorkerGlobalScope</span> {
  [Replaceable] readonly attribute DOMString <span data-x="dom-DedicatedWorkerGlobalScope-name">name</span>;

  undefined <span data-x="dom-DedicatedWorkerGlobalScope-postMessage">postMessage</span>(any message, sequence&lt;<span data-x="idl-object">object</span>&gt; transfer);
  undefined <span data-x="dom-DedicatedWorkerGlobalScope-postMessage-options">postMessage</span>(any message, optional <span>StructuredSerializeOptions</span> options = {});

  undefined <span data-x="dom-DedicatedWorkerGlobalScope-close">close</span>();
};

<span>DedicatedWorkerGlobalScope</span> includes <span>MessageEventTarget</span>;</code></pre>

  <p><code>DedicatedWorkerGlobalScope</code> objects have an associated <dfn>inside port</dfn> (a
  <code>MessagePort</code>). This port is part of a channel that is set up when the worker is
  created, but it is not exposed.<span w-nodev> This object must never be garbage collected before
  the <code>DedicatedWorkerGlobalScope</code> object.</span></p>

  <dl class="domintro">
   <dt><code data-x=""><var>dedicatedWorkerGlobal</var>.<span subdfn data-x="dom-DedicatedWorkerGlobalScope-name">name</span></code></dt>
   <dd><p>Returns <var>dedicatedWorkerGlobal</var>'s <span
   data-x="concept-WorkerGlobalScope-name">name</span>, i.e. the value given to the
   <code>Worker</code> constructor. Primarily useful for debugging.</p></dd>

   <dt><code data-x=""><var>dedicatedWorkerGlobal</var>.<span subdfn data-x="dom-DedicatedWorkerGlobalScope-postMessage">postMessage</span>(<var>message</var> [, <var>transfer</var> ])</code></dt>
   <dt><code data-x=""><var>dedicatedWorkerGlobal</var>.<span subdfn data-x="dom-DedicatedWorkerGlobalScope-postMessage-options">postMessage</span>(<var>message</var> [, { <span data-x="dom-StructuredSerializeOptions-transfer">transfer</span> } ])</code></dt>
   <dd><p>Clones <var>message</var> and transmits it to the <code>Worker</code> object associated
   with <var>dedicatedWorkerGlobal</var>. <var>transfer</var> can be passed as a list of objects
   that are to be transferred rather than cloned.</p></dd>

   <dt><code data-x=""><var>dedicatedWorkerGlobal</var>.<span subdfn data-x="dom-DedicatedWorkerGlobalScope-close">close</span>()</code></dt>
   <dd><p>Aborts <var>dedicatedWorkerGlobal</var>.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="DedicatedWorkerGlobalScope"><code
  data-x="dom-DedicatedWorkerGlobalScope-name">name</code></dfn> getter steps are to return
  <span>this</span>'s <span data-x="concept-WorkerGlobalScope-name">name</span>. Its value
  represents the name given to the worker using the <code>Worker</code> constructor, used primarily
  for debugging purposes.</p>

  <p>The <dfn method for="DedicatedWorkerGlobalScope"><code
  data-x="dom-DedicatedWorkerGlobalScope-postMessage">postMessage(<var>message</var>,
  <var>transfer</var>)</code></dfn> and <dfn method for="DedicatedWorkerGlobalScope"><code
  data-x="dom-DedicatedWorkerGlobalScope-postMessage-options">postMessage(<var>message</var>,
  <var>options</var>)</code></dfn> methods on <code>DedicatedWorkerGlobalScope</code> objects act as
  if, when invoked, it immediately invoked the respective <code
  data-x="dom-MessagePort-postMessage">postMessage(<var>message</var>, <var>transfer</var>)</code>
  and <code data-x="dom-MessagePort-postMessage-options">postMessage(<var>message</var>,
  <var>options</var>)</code> on the port, with the same arguments, and returned the same return
  value.</p>

  <p>To <dfn export>close a worker</dfn>, given a <var>workerGlobal</var>, run these steps:</p>

  <ol>
   <li><p>Discard any <span data-x="concept-task">tasks</span> that have been added to
   <var>workerGlobal</var>'s <span>relevant agent</span>'s <span
   data-x="concept-agent-event-loop">event loop</span>'s <span data-x="task queue">task
   queues</span>.</p>

   <li><p>Set <var>workerGlobal</var>'s <span data-x="dom-WorkerGlobalScope-closing">closing</span>
   flag to true. (This prevents any further tasks from being queued.)</p></li>
  </ol>

  <p>The <dfn method for="DedicatedWorkerGlobalScope"><code
  data-x="dom-DedicatedWorkerGlobalScope-close">close()</code></dfn> method steps are to
  <span>close a worker</span> given <span>this</span>.</p>

  <hr>

  </div>

  <h5>Shared workers and the <code subdfn>SharedWorkerGlobalScope</code> interface</h5>

  <pre><code class="idl">[Global=(Worker,SharedWorker),Exposed=SharedWorker]
interface <dfn interface>SharedWorkerGlobalScope</dfn> : <span>WorkerGlobalScope</span> {
  [Replaceable] readonly attribute DOMString <span data-x="dom-SharedWorkerGlobalScope-name">name</span>;

  undefined <span data-x="dom-SharedWorkerGlobalScope-close">close</span>();

  attribute <span>EventHandler</span> <span data-x="handler-SharedWorkerGlobalScope-onconnect">onconnect</span>;
};</code></pre>

  <div w-nodev>

  <p>A <code>SharedWorkerGlobalScope</code> object has an associated <dfn
  for="SharedWorkerGlobalScope"
  data-x="concept-SharedWorkerGlobalScope-constructor-origin">constructor origin</dfn>, <dfn
  for="SharedWorkerGlobalScope" data-x="concept-SharedWorkerGlobalScope-constructor-url">constructor
  url</dfn>, and <dfn for="SharedWorkerGlobalScope"
  data-x="concept-SharedWorkerGlobalScope-credentials">credentials</dfn>. They are initialized when
  the <code>SharedWorkerGlobalScope</code> object is created, in the <span>run a worker</span>
  algorithm.</p>

  </div>

  <p>Shared workers receive message ports through <code
  data-x="event-WorkerGlobalScope-connect">connect</code> events on their <code>SharedWorkerGlobalScope</code> object for each
  connection.</p>

  <dl class="domintro">
   <dt><code data-x=""><var>sharedWorkerGlobal</var>.<span subdfn data-x="dom-SharedWorkerGlobalScope-name">name</span></code></dt>
   <dd><p>Returns <var>sharedWorkerGlobal</var>'s <span
   data-x="concept-WorkerGlobalScope-name">name</span>, i.e. the value given to the
   <code>SharedWorker</code> constructor. Multiple <code>SharedWorker</code> objects can correspond
   to the same shared worker (and <code>SharedWorkerGlobalScope</code>), by reusing the same
   name.</p></dd>

   <dt><code data-x=""><var>sharedWorkerGlobal</var>.<span subdfn data-x="dom-SharedWorkerGlobalScope-close">close</span>()</code></dt>
   <dd><p>Aborts <var>sharedWorkerGlobal</var>.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="SharedWorkerGlobalScope"><code
  data-x="dom-SharedWorkerGlobalScope-name">name</code></dfn> getter steps are to return
  <span>this</span>'s <span data-x="concept-WorkerGlobalScope-name">name</span>. Its value
  represents the name that can be used to obtain a reference to the worker using the
  <code>SharedWorker</code> constructor.</p>

  <p>The <dfn method for="SharedWorkerGlobalScope"><code
  data-x="dom-SharedWorkerGlobalScope-close">close()</code></dfn> method steps are to <span>close a
  worker</span> given <span>this</span>.</p>

  <hr>

  </div>

  <p>The following are the <span>event handlers</span> (and their corresponding <span data-x="event
  handler event type">event handler event types</span>) <span w-nodev>that must be</span> supported,
  as <span>event handler IDL attributes</span>, by objects implementing the
  <code>SharedWorkerGlobalScope</code> interface:</p>

  <table>
   <thead>
    <tr><th><span data-x="event handlers">Event handler</span> <th><span>Event handler event type</span>
   <tbody>
    <tr><td><dfn attribute for="SharedWorkerGlobalScope"><code data-x="handler-SharedWorkerGlobalScope-onconnect">onconnect</code></dfn> <td> <code data-x="event-WorkerGlobalScope-connect">connect</code>
  </table>



  <h4 id="worker-event-loop">The event loop</h4>

  <p>A <span>worker event loop</span>'s <span data-x="task queue">task queues</span> only have
  events, callbacks, and networking activity as <span data-x="concept-task">tasks</span>.<span
  w-nodev> These <span data-x="worker event loop">worker event loops</span> are created by the
  <span>run a worker</span> algorithm.</span></p>

  <p>Each <code>WorkerGlobalScope</code> object has a <dfn data-x="dom-WorkerGlobalScope-closing"
  export for="WorkerGlobalScope">closing</dfn> flag, <span w-nodev>which must be</span> initially
  false, but which can get set to true <span w-nodev>by the algorithms in the processing model
  section below</span><span w-dev>when the worker is requested to close</span>.</p>

  <p>Once the <code>WorkerGlobalScope</code>'s <span
  data-x="dom-WorkerGlobalScope-closing">closing</span> flag is set to true, the <span>event
  loop</span>'s <span data-x="task queue">task queues</span> <span w-nodev>must</span> discard any
  further <span data-x="concept-task">tasks</span> that would be added to them (tasks already on the
  queue are unaffected except where otherwise specified). Effectively, once the <span
  data-x="dom-WorkerGlobalScope-closing">closing</span> flag is true, timers stop firing,
  notifications for all pending background operations are dropped, etc.</p>



  <div w-nodev>

  <h4>The worker's lifetime</h4>

  <p>Workers communicate with other workers and with <code>Window</code>s through <span
  data-x="channel messaging">message channels</span> and their <code>MessagePort</code>
  objects.</p>

  <p>Each <code>WorkerGlobalScope</code> object <var>worker global scope</var> has a list of
  <dfn export>the worker's ports</dfn>, which consists of all the <code>MessagePort</code> objects
  that are entangled with another port and that have one (but only one) port owned by <var>worker
  global scope</var>. This list includes the implicit <code>MessagePort</code> in the case of <span
  data-x="DedicatedWorkerGlobalScope">dedicated workers</span>.</p>

  <p id="list-of-relevant-document-objects-to-add">Given an <span>environment settings
  object</span> <var>o</var> when creating or obtaining a worker, the <dfn>relevant owner to
  add</dfn> depends on the type of <span data-x="concept-settings-object-global">global
  object</span> specified by <var>o</var>. If <var>o</var>'s <span
  data-x="concept-settings-object-global">global object</span> is a <code>WorkerGlobalScope</code>
  object (i.e., if we are creating a nested dedicated worker), then the relevant owner is that
  global object. Otherwise, <var>o</var>'s <span data-x="concept-settings-object-global">global
  object</span> is a <code>Window</code> object, and the relevant owner is that
  <code>Window</code>'s <span data-x="concept-document-window">associated
  <code>Document</code></span>.</p>

  <hr>

  <p>A worker is said to be a <dfn>permissible worker</dfn> if its <code>WorkerGlobalScope</code>'s
  <span>owner set</span> is not <span data-x="list is empty">empty</span> or:</p>

  <ul class=brief>
   <li>its <span>owner set</span> has been <span data-x="list is empty">empty</span> for no more
   than a short <span>implementation-defined</span> timeout value,</li>
   <li>its <code>WorkerGlobalScope</code> object is a <code>SharedWorkerGlobalScope</code> object
   (i.e., the worker is a shared worker), and</li>
   <li>the user agent has a <span>navigable</span> whose <span data-x="nav-document">active
   document</span> is not <span>completely loaded</span>.</li>
  </ul>

  <p class="note">The second part of this definition allows a shared worker to survive for a short
  time while a page is loading, in case that page is going to contact the shared worker again. This
  can be used by user agents as a way to avoid the cost of restarting a shared worker used by a site
  when the user is navigating from page to page within that site.</p>

  <p>A worker is said to be an <dfn>active needed worker</dfn> if any of its <span data-x="owner
  set">owners</span> are either <code>Document</code> objects that are <span>fully active</span> or
  <span data-x="active needed worker">active needed workers</span>.</p>

  <p>A worker is said to be a <dfn>protected worker</dfn> if it is an <span>active needed
  worker</span> and either it has outstanding timers, database transactions, or network connections,
  or its list of <span>the worker's ports</span> is not empty, or its <code>WorkerGlobalScope</code>
  is actually a <code>SharedWorkerGlobalScope</code> object (i.e., the worker is a shared
  worker).</p>

  <p>A worker is said to be a <dfn>suspendable worker</dfn> if it is not an <span>active needed
  worker</span> but it is a <span>permissible worker</span>.</p>


  <h4 id="worker-processing-model"><span id="processing-model-10"></span>Processing model</h4>

  <p>When a user agent is to <dfn export>run a worker</dfn> for a script with <code>Worker</code> or
  <code>SharedWorker</code> object <var>worker</var>, <span>URL</span> <var>url</var>,
  <span>environment settings object</span> <var>outside settings</var>, <code>MessagePort</code>
  <var>outside port</var>, and a <code>WorkerOptions</code> dictionary <var>options</var>, it must
  run the following steps.</p>

  <ol>
   <li><p>Let <var>is shared</var> be true if <var>worker</var> is a <code>SharedWorker</code>
   object, and false otherwise.</p></li>

   <li><p>Let <var>owner</var> be the <span>relevant owner to add</span> given <var>outside
   settings</var>.</p></li>

   <li><p>Let <var>unsafeWorkerCreationTime</var> be the <span>unsafe shared current
   time</span>.</p></li>

   <li><p>Let <var>agent</var> be the result of <span data-x="obtain a dedicated/shared worker
   agent">obtaining a dedicated/shared worker agent</span> given <var>outside settings</var> and
   <var>is shared</var>. Run the rest of these steps in that agent.</p></li>

   <li>
    <p>Let <var>realm execution context</var> be the result of <span>creating a new realm</span>
    given <var>agent</var> and the following customizations:</p>

    <ul>
     <li><p>For the global object, if <var>is shared</var> is true, create a new
     <code>SharedWorkerGlobalScope</code> object. Otherwise, create a new
     <code>DedicatedWorkerGlobalScope</code> object.</p></li>
    </ul>
   </li>

   <li>
    <p>Let <var>worker global scope</var> be the <span data-x="concept-realm-global">global
    object</span> of <var>realm execution context</var>'s Realm component.

    <p class="note">This is the <code>DedicatedWorkerGlobalScope</code> or
    <code>SharedWorkerGlobalScope</code> object created in the previous step.</p>
   </li>

   <li><p><span>Set up a worker environment settings object</span> with <var>realm execution
   context</var>, <var>outside settings</var>, and <var>unsafeWorkerCreationTime</var>, and let
   <var>inside settings</var> be the result.</p></li>

   <li><p>Set <var>worker global scope</var>'s <span
   data-x="concept-WorkerGlobalScope-name">name</span> to the value of <var>options</var>'s
   <code data-x="">name</code> member.</p></li>

   <li><p><span data-x="set append">Append</span> <var>owner</var> to <var>worker global
   scope</var>'s <span>owner set</span>.</p></li>

   <li>
    <p>If <var>is shared</var> is true, then:</p>

    <ol>
     <li><p>Set <var>worker global scope</var>'s <span
     data-x="concept-SharedWorkerGlobalScope-constructor-origin">constructor origin</span> to
     <var>outside settings</var>'s <span
     data-x="concept-settings-object-origin">origin</span>.</p></li>

     <li><p>Set <var>worker global scope</var>'s <span
     data-x="concept-SharedWorkerGlobalScope-constructor-url">constructor url</span> to
     <var>url</var>.</p></li>

     <li><p>Set <var>worker global scope</var>'s <span
     data-x="concept-WorkerGlobalScope-type">type</span> to the value of <var>options</var>'s
     <code data-x="">type</code> member.</p></li>

     <li><p>Set <var>worker global scope</var>'s <span
     data-x="concept-SharedWorkerGlobalScope-credentials">credentials</span> to the value of
     <var>options</var>'s <code data-x="">credentials</code> member.
     </p></li>
    </ol>
   </li>

   <li><p>Let <var>destination</var> be "<code data-x="">sharedworker</code>" if <var>is
   shared</var> is true, and "<code data-x="">worker</code>" otherwise.</p></li>

   <li>
    <p>Obtain <var>script</var> by switching on the value of <var>options</var>'s <code
    data-x="">type</code> member:</p>

    <dl class="switch">
     <dt>"<code data-x="">classic</code>"</dt>
     <dd><span>Fetch a classic worker script</span> given <var>url</var>, <var>outside
     settings</var>, <var>destination</var>, <var>inside settings</var>, and with
     <var>onComplete</var> and <var>performFetch</var> as defined below.</dd>

     <dt>"<code data-x="">module</code>"</dt>
     <dd><span>Fetch a module worker script graph</span> given <var>url</var>, <var>outside
     settings</var>, <var>destination</var>, the value of the <code data-x="">credentials</code>
     member of <var>options</var>, <var>inside settings</var>, and with <var>onComplete</var> and
     <var>performFetch</var> as defined below.</dd>
    </dl>

    <p>In both cases, let <var>performFetch</var> be the following <span
    data-x="fetching-scripts-perform-fetch">perform the fetch hook</span> given <var>request</var>,
    <i data-x="fetching-scripts-is-top-level">isTopLevel</i> and <i
    data-x="fetching-scripts-processCustomFetchResponse">processCustomFetchResponse</i>:</p>

    <ol>
     <li><p>If <var>isTopLevel</var> is false, <span data-x="concept-fetch">fetch</span>
     <var>request</var> with <i data-x="processResponseConsumeBody">processResponseConsumeBody</i>
     set to <var>processCustomFetchResponse</var>, and abort these steps.</p></li>

     <li>Set <var>request</var>'s <span data-x="concept-request-reserved-client">reserved
     client</span> to <var>inside settings</var>.</li>

     <li>
      <p><span data-x="concept-fetch">Fetch</span> <var>request</var> with <i
      data-x="processResponseConsumeBody">processResponseConsumeBody</i> set to the following steps
      given <span data-x="concept-response">response</span> <var>response</var> and null, failure,
      or a <span>byte sequence</span> <var>bodyBytes</var>:</p>

      <ol>
       <li><p>Set <var>worker global scope</var>'s <span
       data-x="concept-WorkerGlobalScope-url">url</span> to <var>response</var>'s <span
       data-x="concept-response-url">url</span>.</p></li>

       <li><p><span data-x="initialize worker policy container">Initialize worker global scope's
       policy container</span> given <var>worker global scope</var>, <var>response</var>, and
       <var>inside settings</var>.</p></li>

       <li><p>If the <span>Run CSP initialization for a global object</span> algorithm returns
       "<code data-x="">Blocked</code>" when executed upon <var>worker global scope</var>, set
       <var>response</var> to a <span>network error</span>. <ref>CSP</ref></p></li>

       <li>
        <p>If <var>worker global scope</var>'s <span
        data-x="concept-WorkerGlobalScope-embedder-policy">embedder policy</span>'s <span
        data-x="embedder-policy-value">value</span> is <span>compatible with cross-origin
        isolation</span> and <var>is shared</var> is true, then set <var>agent</var>'s <span>agent
        cluster</span>'s <span data-x="agent-cluster-cross-origin-isolation">cross-origin isolation
        mode</span> to "<code data-x="cross-origin-isolation-logical">logical</code>" or "<code
        data-x="cross-origin-isolation-concrete">concrete</code>". The one chosen is
        <span>implementation-defined</span>.</p>

        <p class="XXX">This really ought to be set when the agent cluster is created, which requires
        a redesign of this section.</p>
       </li>

       <li><p>If the result of <span data-x="check a global object's embedder policy">checking a
       global object's embedder policy</span> with <var>worker global scope</var>, <var>outside
       settings</var>, and <var>response</var> is false, then set <var>response</var> to a
       <span>network error</span>.</p></li>

       <li><p>Set <var>worker global scope</var>'s <span
       data-x="concept-WorkerGlobalScope-cross-origin-isolated-capability">cross-origin isolated
       capability</span> to true if <var>agent</var>'s <span>agent cluster</span>'s <span
       data-x="agent-cluster-cross-origin-isolation">cross-origin isolation mode</span> is "<code
       data-x="cross-origin-isolation-concrete">concrete</code>".</p></li>

       <li><p>If <var>is shared</var> is false and <var>owner</var>'s <span
       data-x="concept-settings-object-cross-origin-isolated-capability">cross-origin isolated
       capability</span> is false, then set <var>worker global scope</var>'s <span
       data-x="concept-WorkerGlobalScope-cross-origin-isolated-capability">cross-origin isolated
       capability</span> to false.</p></li>

       <li>
        <p>If <var>is shared</var> is false and <var>response</var>'s
        <span data-x="concept-response-url">url</span>'s <span
        data-x="concept-url-scheme">scheme</span> is "<code data-x="">data</code>", then set
        <var>worker global scope</var>'s <span
        data-x="concept-WorkerGlobalScope-cross-origin-isolated-capability">cross-origin isolated
        capability</span> to false.</p>

        <p class="note">This is a conservative default for now, while we figure out how workers in
        general, and <code data-x="data protocol">data:</code> URL workers in particular (which are
        cross-origin from their owner), will be treated in the context of permissions policies. See
        <a
        href="https://github.com/w3c/webappsec-permissions-policy/issues/207">w3c/webappsec-permissions-policy
        issue #207</a> for more details.</p>
       </li>

       <li><p>Run <var>processCustomFetchResponse</var> with <var>response</var> and
       <var>bodyBytes</var>.</p></li>
      </ol>
     </li>
    </ol>

    <p>In both cases, let <var>onComplete</var> given <var>script</var> be the following steps:</p>

    <ol>
     <li><p>If <var>script</var> is null or if <var>script</var>'s <span
    data-x="concept-script-error-to-rethrow">error to rethrow</span> is non-null, then:</p>

     <ol>
      <li><p><span>Queue a global task</span> on the <span>DOM manipulation task source</span> given
      <var>worker</var>'s <span>relevant global object</span> to <span
      data-x="concept-event-fire">fire an event</span> named <code data-x="event-error">error</code>
      at <var>worker</var>.</p></li>

      <li><p>Run the <span data-x="environment discarding steps">environment discarding steps</span>
      for <var>inside settings</var>.</p></li>

      <li><p>Abort these steps.</p></li>
     </ol>

     <li><p>Associate <var>worker</var> with <var>worker global scope</var>.</p></li>

     <li><p>Let <var>inside port</var> be a <span>new</span> <code>MessagePort</code> object in
     <var>inside settings</var>'s <span data-x="environment settings object's
     realm">realm</span>.</p></li>

     <li>
      <p>If <var>is shared</var> is false, then:</p>

      <ol>
       <li><p>Set <var>inside port</var>'s <span>message event target</span> to <var>worker global
       scope</var>.</p></li>

       <li><p>Set <var>worker global scope</var>'s <span>inside port</span> to <var>inside
       port</var>.</p></li>
      </ol>
     </li>

     <li><p><span>Entangle</span> <var>outside port</var> and <var>inside port</var>.</p></li>

     <li><p>Create a new <code>WorkerLocation</code> object and associate it with <var>worker global
     scope</var>.</p>

     <li>
      <p><strong>Closing orphan workers</strong>: Start monitoring the worker such that no sooner than
      it stops being a <span>protected worker</span>, and no later than it stops being a
      <span>permissible worker</span>, <var>worker global scope</var>'s <span
      data-x="dom-WorkerGlobalScope-closing">closing</span> flag is set to true.</p>
     </li>

     <li>
      <p><strong>Suspending workers</strong>: Start monitoring the worker, such that whenever
      <var>worker global scope</var>'s <span data-x="dom-WorkerGlobalScope-closing">closing</span>
      flag is false and the worker is a <span>suspendable worker</span>, the user agent suspends
      execution of script in that worker until such time as either the <span
      data-x="dom-WorkerGlobalScope-closing">closing</span> flag switches to true or the worker stops
      being a <span>suspendable worker</span>.</p>
     </li>

     <li><p>Set <var>inside settings</var>'s <span
     data-x="concept-environment-execution-ready-flag">execution ready flag</span>.</p></li>

     <li>
      <p>If <var>script</var> is a <span>classic script</span>, then <span data-x="run a classic
      script">run the classic script</span> <var>script</var>. Otherwise, it is a <span>module
      script</span>; <span data-x="run a module script">run the module script</span>
      <var>script</var>.</p>

      <p class="note">In addition to the usual possibilities of returning a value or failing due to
      an exception, this could be <span data-x="abort a running script">prematurely aborted</span> by
      the <span>terminate a worker</span> algorithm defined below.</p>
     </li>

     <li><p>Enable <var>outside port</var>'s <span>port message queue</span>.</p></li>

     <li><p>If <var>is shared</var> is false, enable the <span>port message queue</span>
     of the worker's implicit port.</p></li>

     <li><p>If <var>is shared</var> is true, then <span>queue a global task</span> on <span>DOM
     manipulation task source</span> given <var>worker global scope</var> to <span
     data-x="concept-event-fire">fire an event</span> named <code
     data-x="event-WorkerGlobalScope-connect">connect</code> at <var>worker global scope</var>, using
     <code>MessageEvent</code>, with the <code data-x="dom-MessageEvent-data">data</code> attribute
     initialized to the empty string, the <code data-x="dom-MessageEvent-ports">ports</code> attribute
     initialized to a new <span>frozen array</span> containing <var>inside port</var>, and the <code
     data-x="dom-MessageEvent-source">source</code> attribute initialized to <var>inside
     port</var>.</p></li>

     <li><p>Enable the <span data-x="dfn-client-message-queue">client message queue</span> of the
     <code>ServiceWorkerContainer</code> object whose associated <span
     data-x="serviceworkercontainer-service-worker-client">service worker client</span> is
     <var>worker global scope</var>'s <span>relevant settings object</span>.</p></li>

     <li>
      <p><strong>Event loop<!-- labeled for ease of discussion --></strong>: Run the <span>responsible
      event loop</span> specified by <var>inside settings</var> until it is destroyed.</p>

      <p class="note">The handling of events or the execution of callbacks by <span
      data-x="concept-task">tasks</span> run by the <span>event loop</span> might get <span
      data-x="abort a running script">prematurely aborted</span> by the <span>terminate a
      worker</span> algorithm defined below.</p>

      <p class="note">The worker processing model remains on this step until the event loop is
      destroyed, which happens after the <span data-x="dom-WorkerGlobalScope-closing">closing</span>
      flag is set to true, as described in the <span>event loop</span> processing model.</p>
     </li>

     <li>
      <p><span data-x="map clear">Clear</span> the <var>worker global scope</var>'s
      <span>map of active timers</span>.</p>
     </li>

     <li>
      <p>Disentangle all the ports in the list of <span>the worker's ports</span>.</p>
     </li>

     <li>
      <!-- this has no normative impact but makes it clearer that the worker is irrelevant now, and
      doesn't have to survive until its Documents all die off too --> <p><span data-x="list
      empty">Empty</span> <var>worker global scope</var>'s <span>owner set</span>.</p>
     </li>
    </ol>
   </li>
  </ol>

  <hr>

  <p>When a user agent is to <dfn export>terminate a worker</dfn>, it must run the following steps
  <span>in parallel</span> with the worker's main loop (the "<span>run a worker</span>" processing
  model defined above):</p>

  <ol>
   <li><p>Set the worker's <code>WorkerGlobalScope</code> object's <span
   data-x="dom-WorkerGlobalScope-closing">closing</span> flag to true.</p></li>

   <li><p>If there are any <span data-x="concept-task">tasks</span> queued in the
   <code>WorkerGlobalScope</code> object's <span>relevant agent</span>'s <span
   data-x="concept-agent-event-loop">event loop</span>'s <span data-x="task queue">task
   queues</span>, discard them without processing them.</p></li>

   <li><p><span data-x="abort a running script">Abort the script</span> currently running in the
   worker.</p></li>

   <li><p>If the worker's <code>WorkerGlobalScope</code> object is actually a
   <code>DedicatedWorkerGlobalScope</code> object (i.e. the worker is a dedicated worker), then
   empty the <span>port message queue</span> of the port that the worker's implicit port is
   entangled with.</p></li>
  </ol>

  <p>User agents may invoke the <span>terminate a worker</span> algorithm when a worker stops being
  an <span>active needed worker</span> and the worker continues executing even after its <span
  data-x="dom-WorkerGlobalScope-closing">closing</span> flag was set to true.</p>

  </div>


  <h4>Runtime script errors</h4>

  <p>Whenever an uncaught runtime script error occurs in one of the worker's scripts, if the error
  did not occur while handling a previous script error, the user agent will <span data-x="report an
  exception">report</span> it for the worker's <code>WorkerGlobalScope</code> object.</p>


  <h4>Creating workers</h4>

  <h5 w-nodev><span id="the-abstractworker-abstract-interface"></span>The
  <code>AbstractWorker</code> mixin</h5>
  <h5 w-dev>Properties present on both <code>Worker</code> and <code>SharedWorker</code></h5>

  <pre><code class="idl">interface mixin <dfn interface>AbstractWorker</dfn> {
  attribute <span>EventHandler</span> <span data-x="handler-AbstractWorker-onerror">onerror</span>;
};</code></pre>

  <p>The following are the <span>event handlers</span> (and their corresponding <span data-x="event
  handler event type">event handler event types</span>) <span w-nodev>that must be</span> supported,
  as <span>event handler IDL attributes</span>, by <span w-nodev>objects implementing the
  <code>AbstractWorker</code> interface</span><span w-dev><code>Worker</code> and
  <code>SharedWorker</code> objects</span>:</p>

  <table>
   <thead>
    <tr><th><span data-x="event handlers">Event handler</span> <th><span>Event handler event type</span>
   <tbody>
    <tr><td><dfn attribute for="AbstractWorker"><code data-x="handler-AbstractWorker-onerror">onerror</code></dfn> <td> <code data-x="event-error">error</code>
  </table>


  <div w-nodev>
  <h5>Script settings for workers</h5>

  <p>To <dfn>set up a worker environment settings object</dfn>, given a <span>JavaScript execution
  context</span> <var>execution context</var>, an <span>environment settings object</span>
  <var>outside settings</var>, and a number <var>unsafeWorkerCreationTime</var>:</p>

  <ol>
   <li><p>Let <var>inherited origin</var> be <var>outside settings</var>'s <span
   data-x="concept-settings-object-origin">origin</span>.</p></li>

   <li><p>Let <var>realm</var> be the value of <var>execution context</var>'s Realm
   component.</p></li>

   <li><p>Let <var>worker global scope</var> be <var>realm</var>'s <span
   data-x="concept-realm-global">global object</span>.</p></li>

   <li>
    <p>Let <var>settings object</var> be a new <span>environment settings object</span> whose algorithms
    are defined as follows:</p>

    <dl>
     <dt>The <span>realm execution context</span></dt>
     <dd>
      <p>Return <var>execution context</var>.
     </dd>

     <dt>The <span data-x="concept-settings-object-module-map">module map</span></dt>
     <dd>
      <p>Return <var>worker global scope</var>'s <span
      data-x="concept-WorkerGlobalScope-module-map">module map</span>.</p>
     </dd>

     <dt>The <span>API base URL</span></dt>
     <dd>
      <p>Return <var>worker global scope</var>'s <span
      data-x="concept-WorkerGlobalScope-url">url</span>.</p>
     </dd>

     <dt>The <span data-x="concept-settings-object-origin">origin</span></dt>
     <dd>
      <p>Return a unique <span data-x="concept-origin-opaque">opaque origin</span> if <var>worker
      global scope</var>'s <span data-x="concept-WorkerGlobalScope-url">url</span>'s <span
      data-x="concept-url-scheme">scheme</span> is "<code data-x="">data</code>", and <var>inherited
      origin</var> otherwise.</p>
     </dd>

     <dt>The <span data-x="concept-settings-object-policy-container">policy container</span></dt>
     <dd>
      <p>Return <var>worker global scope</var>'s <span
      data-x="concept-WorkerGlobalScope-policy-container">policy container</span>.</p>
     </dd>

     <dt>The <span data-x="concept-settings-object-cross-origin-isolated-capability">cross-origin
     isolated capability</span></dt>
     <dd><p>Return <var>worker global scope</var>'s <span
     data-x="concept-WorkerGlobalScope-cross-origin-isolated-capability">cross-origin isolated
     capability</span>.</p></dd>

     <dt>The <span data-x="concept-settings-object-time-origin">time origin</span></dt>
     <dd><p>Return the result of <span data-x="coarsen time">coarsening</span>
     <var>unsafeWorkerCreationTime</var> with <var>worker global scope</var>'s <span
     data-x="concept-WorkerGlobalScope-cross-origin-isolated-capability">cross-origin isolated
     capability</span>.</p></dd>
    </dl>
   </li>

   <li><p>Set <var>settings object</var>'s <span data-x="concept-environment-id">id</span> to a new
   unique opaque string, <span data-x="concept-environment-creation-url">creation URL</span> to
   <var>worker global scope</var>'s <span>url</span>, <span>top-level creation URL</span> to null,
   <span data-x="concept-environment-target-browsing-context">target browsing context</span> to
   null, and <span data-x="concept-environment-active-service-worker">active service worker</span>
   to null.</p></li>

   <li><p>If <var>worker global scope</var> is a <code>DedicatedWorkerGlobalScope</code> object,
   then set <var>settings object</var>'s <span>top-level origin</span> to <var>outside
   settings</var>'s <span>top-level origin</span>.</p></li>

   <li>
    <p>Otherwise, set <var>settings object</var>'s <span>top-level origin</span> to an
    <span>implementation-defined</span> value.</p>

    <p class="XXX">See <a href="https://privacycg.github.io/storage-partitioning/">Client-Side
    Storage Partitioning</a> for the latest on properly defining this.</p>
   </li>

   <li><p>Set <var>realm</var>'s [[HostDefined]] field to <var>settings object</var>.</p></li>

   <li><p>Return <var>settings object</var>.</p></li>
  </ol>
  </div>


  <h5>Dedicated workers and the <code subdfn>Worker</code> interface</h5>

  <pre><code class="idl">[Exposed=(Window,DedicatedWorker,SharedWorker)]
interface <dfn interface>Worker</dfn> : <span>EventTarget</span> {
  <span data-x="dom-Worker">constructor</span>((<code data-x="tt-trustedscripturl">TrustedScriptURL</code> or USVString) scriptURL, optional <span>WorkerOptions</span> options = {});

  undefined <span data-x="dom-Worker-terminate">terminate</span>();

  undefined <span data-x="dom-Worker-postMessage">postMessage</span>(any message, sequence&lt;<span data-x="idl-object">object</span>&gt; transfer);
  undefined <span data-x="dom-Worker-postMessage-options">postMessage</span>(any message, optional <span>StructuredSerializeOptions</span> options = {});
};

dictionary <dfn dictionary>WorkerOptions</dfn> {
  <span>WorkerType</span> type = "classic";
  <span>RequestCredentials</span> credentials = "same-origin"; // credentials is only used if type is "module"
  DOMString name = "";
};

enum <dfn enum>WorkerType</dfn> { "classic", "module" };

<span>Worker</span> includes <span>AbstractWorker</span>;
<span>Worker</span> includes <span>MessageEventTarget</span>;</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>worker</var> = new <span subdfn data-x="dom-Worker">Worker</span>(<var>scriptURL</var> [, <var>options</var> ])</code></dt>
   <dd><p>Returns a new <code>Worker</code> object. <var>scriptURL</var> will be fetched and
   executed in the background, creating a new global environment for which <var>worker</var>
   represents the communication channel. <var>options</var> can be used to define the <span
   data-x="concept-WorkerGlobalScope-name">name</span> of that global environment via the <code
   data-x="">name</code> option, primarily for debugging purposes. It can also ensure this new
   global environment supports JavaScript modules (specify <code data-x="">type: "module"</code>),
   and if that is specified, can also be used to specify how <var>scriptURL</var> is fetched through
   the <code data-x="">credentials</code> option.</p></dd>

   <dt><code data-x=""><var>worker</var>.<span subdfn data-x="dom-Worker-terminate">terminate</span>()</code></dt>
   <dd>Aborts <var>worker</var>'s associated global environment.</dd>

   <dt><code data-x=""><var>worker</var>.<span subdfn data-x="dom-Worker-postMessage">postMessage</span>(<var>message</var> [, <var>transfer</var> ])</code></dt>
   <dt><code data-x=""><var>worker</var>.<span subdfn data-x="dom-Worker-postMessage-options">postMessage</span>(<var>message</var> [, { <span data-x="dom-StructuredSerializeOptions-transfer">transfer</span> } ])</code></dt>
   <dd><p>Clones <var>message</var> and transmits it to <var>worker</var>'s global environment.
   <var>transfer</var> can be passed as a list of objects that are to be transferred rather than
   cloned.</p></dd>
  </dl>

  <div w-nodev>

  <p>Each <code>Worker</code> object has an associated <dfn for="Worker">outside port</dfn> (a
  <code>MessagePort</code>). This port is part of a channel that is set up when the worker is
  created, but it is not exposed. This object must never be garbage collected before the
  <code>Worker</code> object.</p>

  <p>The <dfn method for="Worker"><code data-x="dom-Worker-terminate">terminate()</code></dfn>
  method, when invoked, must cause the <span>terminate a worker</span> algorithm to be run on the
  worker with which the object is associated.</p>

  <p>The <dfn method for="Worker"><code
  data-x="dom-Worker-postMessage">postMessage(<var>message</var>, <var>transfer</var>)</code></dfn>
  and <dfn method for="Worker"><code
  data-x="dom-Worker-postMessage-options">postMessage(<var>message</var>,
  <var>options</var>)</code></dfn> methods on <code>Worker</code> objects act as if, when invoked,
  they immediately invoked the respective <code
  data-x="dom-MessagePort-postMessage">postMessage(<var>message</var>, <var>transfer</var>)</code>
  and <code data-x="dom-MessagePort-postMessage-options">postMessage(<var>message</var>,
  <var>options</var>)</code> on <span>this</span>'s <span>outside port</span>, with the same
  arguments, and returned the same return value.</p>

  </div>

  <div class="example">

   <p>The <code data-x="dom-Worker-postMessage">postMessage()</code>
   method's first argument can be structured data:</p>

   <pre><code class="js">worker.postMessage({opcode: 'activate', device: 1938, parameters: [23, 102]});</code></pre>

  </div>

  <div w-nodev>

  <hr>

  <p>When the <dfn constructor for="Worker"><code data-x="dom-Worker">Worker(<var>scriptURL</var>,
  <var>options</var>)</code></dfn> constructor is invoked, the user agent must run the following
  steps:</p>

  <ol>
   <li><p>Let <var>compliantScriptURL</var> be the result of invoking the <span
   data-x="tt-getcompliantstring">Get Trusted Type compliant string</span> algorithm with <code
   data-x="tt-trustedscripturl">TrustedScriptURL</code>, <span>this</span>'s <span>relevant global
   object</span>, <var>scriptURL</var>, "<code data-x="">Worker constructor</code>", and "<code
   data-x="">script</code>".</p></li>

   <li><p>Let <var>outside settings</var> be the <span>current settings object</span>.</p>

   <li>
    <p>Let <var>worker URL</var> be the result of <span>encoding-parsing a URL</span> given
    <var>compliantScriptURL</var>, relative to <var>outside settings</var>.</p>

    <p class="note">Any <span data-x="same origin">same-origin</span> URL (including <code
    data-x="blob protocol">blob:</code> URLs) can be used. <code data-x="data protocol">data:</code>
    URLs can also be used, but they create a worker with an <span
    data-x="concept-origin-opaque">opaque origin</span>.</p>
   </li>

   <li><p>If <var>worker URL</var> is failure, then throw a <span>"<code>SyntaxError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Let <var>worker</var> be a new <code>Worker</code> object.</p></li>

   <li><p>Let <var>outside port</var> be a <span>new</span> <code>MessagePort</code> in <var>outside
   settings</var>'s <span data-x="environment settings object's realm">realm</span>.</p></li>

   <li><p>Set <var>outside port</var>'s <span>message event target</span> to
   <var>worker</var>.</p></li>

   <li><p>Set <var>worker</var>'s <span>outside port</span> to <var>outside port</var>.</p></li>

   <li>
    <p>Run this step <span>in parallel</span>:</p>

    <ol>
     <li><p><span>Run a worker</span> given <var>worker</var>, <var>worker URL</var>, <var>outside
     settings</var>, <var>outside port</var>, and <var>options</var>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>worker</var>.</p></li>
  </ol>

  </div>


  <h5>Shared workers and the <code subdfn>SharedWorker</code> interface</h5>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>SharedWorker</dfn> : <span>EventTarget</span> {
  <span data-x="dom-SharedWorker">constructor</span>((<code data-x="tt-trustedscripturl">TrustedScriptURL</code> or USVString) scriptURL, optional (DOMString or <span>WorkerOptions</span>) options = {});

  readonly attribute <span>MessagePort</span> <span data-x="dom-SharedWorker-port">port</span>;
};
<span>SharedWorker</span> includes <span>AbstractWorker</span>;</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>sharedWorker</var> = new <span subdfn data-x="dom-SharedWorker">SharedWorker</span>(<var>scriptURL</var> [, <var>name</var> ])</code></dt>
   <dd><p>Returns a new <code>SharedWorker</code> object. <var>scriptURL</var> will be fetched and
   executed in the background, creating a new global environment for which <var>sharedWorker</var>
   represents the communication channel. <var>name</var> can be used to define the <span
   data-x="concept-WorkerGlobalScope-name">name</span> of that global environment.</p></dd>

   <dt><code data-x=""><var>sharedWorker</var> = new <span data-x="dom-SharedWorker">SharedWorker</span>(<var>scriptURL</var> [, <var>options</var> ])</code></dt>
   <dd><p>Returns a new <code>SharedWorker</code> object. <var>scriptURL</var> will be fetched and
   executed in the background, creating a new global environment for which <var>sharedWorker</var>
   represents the communication channel. <var>options</var> can be used to define the <span
   data-x="concept-WorkerGlobalScope-name">name</span> of that global environment via the <code
   data-x="">name</code> option. It can also ensure this new global environment supports JavaScript
   modules (specify <code data-x="">type: "module"</code>), and if that is specified, can also be
   used to specify how <var>scriptURL</var> is fetched through the <code
   data-x="">credentials</code> option. Note that attempting to construct a shared worker with
   <var>options</var> whose <code data-x="">type</code> or <code data-x="">credentials</code> values
   mismatch an existing shared worker will cause the returned <var>sharedWorker</var> to fire an
   error event and not connect to the existing shared worker.</p></dd>

   <dt><code data-x=""><var>sharedWorker</var>.<span subdfn data-x="dom-SharedWorker-port">port</span></code></dt>
   <dd><p>Returns <var>sharedWorker</var>'s <code>MessagePort</code> object which can be used to
   communicate with the global environment.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="SharedWorker"><code data-x="dom-SharedWorker-port">port</code></dfn>
  attribute must return the value it was assigned by the object's constructor. It represents the
  <code>MessagePort</code> for communicating with the shared worker.</p>

  <p>A user agent has an associated <dfn>shared worker manager</dfn> which is the result of
  <span>starting a new parallel queue</span>.</p>

  <p class="note">Each user agent has a single <span>shared worker manager</span> for simplicity.
  Implementations could use one per <span>origin</span>; that would not be observably different and
  enables more concurrency.</p>

  <p>When the <dfn constructor for="SharedWorker"><code
  data-x="dom-SharedWorker">SharedWorker(<var>scriptURL</var>, <var>options</var>)</code></dfn>
  constructor is invoked:</p>

  <ol>
   <li><p>Let <var>compliantScriptURL</var> be the result of invoking the <span
   data-x="tt-getcompliantstring">Get Trusted Type compliant string</span> algorithm with <code
   data-x="tt-trustedscripturl">TrustedScriptURL</code>, <span>this</span>'s <span>relevant global
   object</span>, <var>scriptURL</var>, "<code data-x="">SharedWorker constructor</code>", and
   "<code data-x="">script</code>".</p></li>

   <li><p>If <var>options</var> is a <code data-x="idl-DOMString">DOMString</code>, set
   <var>options</var> to a new <code>WorkerOptions</code> dictionary whose <code
   data-x="">name</code> member is set to the value of <var>options</var> and whose other members
   are set to their default values.</p></li>

   <li><p>Let <var>outside settings</var> be the <span>current settings object</span>.</p></li>

   <li>
    <p>Let <var>urlRecord</var> be the result of <span>encoding-parsing a URL</span> given
    <var>compliantScriptURL</var>, relative to <var>outside settings</var>.</p>

    <p class="note">Any <span data-x="same origin">same-origin</span> URL (including <code
    data-x="blob protocol">blob:</code> URLs) can be used. <code data-x="data protocol">data:</code>
    URLs can also be used, but they create a worker with an <span
    data-x="concept-origin-opaque">opaque origin</span>.</p>
   </li>

   <li><p>If <var>urlRecord</var> is failure, then throw a <span>"<code>SyntaxError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Let <var>worker</var> be a new <code>SharedWorker</code> object.</p></li>

   <li><p>Let <var>outside port</var> be a <span>new</span> <code>MessagePort</code> in <var>outside
   settings</var>'s <span data-x="environment settings object's realm">realm</span>.</p></li>

   <li><p>Assign <var>outside port</var> to the <code data-x="dom-SharedWorker-port">port</code>
   attribute of <var>worker</var>.</p></li>

   <li><p>Let <var>callerIsSecureContext</var> be true if <var>outside settings</var> is a
   <span>secure context</span>; otherwise, false.</p></li>

   <li><p>Let <var>outside storage key</var> be the result of running <span>obtain a storage key
   for non-storage purposes</span> given <var>outside settings</var>.</p></li>

   <li>
    <p><span>Enqueue the following steps</span> to the <span>shared worker manager</span>:</p>

    <ol>
     <li><p>Let <var>worker global scope</var> be null.</p></li>

     <li>
      <p><span data-x="list iterate">For each</span> <var>scope</var> in the list of all
      <code>SharedWorkerGlobalScope</code> objects:</p>

      <ol>
        <li><p>Let <var>worker storage key</var> be the result of running <span>obtain a storage
        key for non-storage purposes</span> given <var>scope</var>'s <span>relevant settings
        object</span>.</p></li>

        <li>
         <p>If all of the following are true:</p>

         <ul>
          <li><var>worker storage key</var> <span data-x="storage key equal">equals</span>
          <var>outside storage key</var>;</li>

          <li><var>scope</var>'s <span data-x="dom-WorkerGlobalScope-closing">closing</span> flag
          is false;</li>

          <li><var>scope</var>'s <span data-x="concept-SharedWorkerGlobalScope-constructor-url">
          constructor url</span> <span data-x="concept-url-equals">equals</span>
          <var>urlRecord</var>; and</li>

          <li><var>scope</var>'s <span data-x="concept-WorkerGlobalScope-name">name</span> equals
          the value of <var>options</var>'s <code data-x="">name</code> member,</li>
         </ul>

         <p>then:</p>

         <ol>
          <li><p>Set <var>worker global scope</var> to <var>scope</var>.</p></li>

          <li><p><span>Break</span>.</p></li>
         </ol>
        </li>
      </ol>

      <p class="note"><code data-x="data protocol">data:</code> URLs create a worker with an <span
      data-x="concept-origin-opaque">opaque origin</span>. Both the <span
      data-x="concept-SharedWorkerGlobalScope-constructor-origin">constructor origin</span> and
      <span data-x="concept-SharedWorkerGlobalScope-constructor-url">constructor url</span> are
      compared so the same <code data-x="data protocol">data:</code> URL can be used within an
      <span>origin</span> to get to the same <code>SharedWorkerGlobalScope</code> object, but cannot
      be used to bypass the <span>same origin</span> restriction.</p>
     </li>

     <li>
      <p>If <var>worker global scope</var> is not null, but the user agent has been
      configured to disallow communication between the worker represented by the <var>worker global
      scope</var> and the <span data-x="concept-script">scripts</span> whose <span
      data-x="concept-script-settings-object">settings object</span> is <var>outside settings</var>,
      then set <var>worker global scope</var> to null.</p>

      <p class="note">For example, a user agent could have a development mode that isolates a
      particular <span>top-level traversable</span> from all other pages, and scripts in that
      development mode could be blocked from connecting to shared workers running in the normal
      browser mode.</p>
     </li>

     <li><p>If <var>worker global scope</var> is not null, then check if <var>worker global
     scope</var>'s <span data-x="concept-WorkerGlobalScope-type">type</span> and <span
     data-x="concept-SharedWorkerGlobalScope-credentials">credentials</span> match the
     <var>options</var> values. If not, <span>queue a task</span> to <span
     data-x="concept-event-fire">fire an event</span> named <code data-x="event-error">error</code>
     and abort these steps.</p></li>

     <li>
      <p>If <var>worker global scope</var> is not null, then run these subsubsteps:</p>

      <ol>
       <li><p>Let <var>settings object</var> be the <span>relevant settings object</span> for
       <var>worker global scope</var>.</p></li>

       <li><p>Let <var>workerIsSecureContext</var> be true if <var>settings object</var> is a
       <span>secure context</span>; otherwise, false.</p></li>

       <li><p>If <var>workerIsSecureContext</var> is not <var>callerIsSecureContext</var>, then
       <span>queue a task</span> to <span data-x="concept-event-fire">fire an event</span> named
       <code data-x="event-error">error</code> at <var>worker</var> and abort these steps.
       <ref>SECURE-CONTEXTS</ref></p></li>

       <li><p>Associate <var>worker</var> with <var>worker global scope</var>.</p></li>

       <li><p>Let <var>inside port</var> be a <span>new</span> <code>MessagePort</code> in
       <var>settings object</var>'s <span data-x="environment settings object's
       realm">realm</span>.</p></li>

       <li><p><span>Entangle</span> <var>outside port</var> and <var>inside port</var>.</p></li>

       <li><p><span>Queue a task</span>, using the <span>DOM manipulation task source</span>, to
       <span data-x="concept-event-fire">fire an event</span> named <code
       data-x="event-WorkerGlobalScope-connect">connect</code> at <var>worker global scope</var>,
       using <code>MessageEvent</code>, with the <code data-x="dom-MessageEvent-data">data</code>
       attribute initialized to the empty string, the <code
       data-x="dom-MessageEvent-ports">ports</code> attribute initialized to a new <span>frozen
       array</span> containing only <var>inside port</var>, and the <code
       data-x="dom-MessageEvent-source">source</code> attribute initialized to <var>inside
       port</var>.</p></li>

       <li><p><span data-x="set append">Append</span> the <span>relevant owner to add</span> given
       <var>outside settings</var> to <var>worker global scope</var>'s <span>owner
       set</span>.</p></li>
      </ol>
     </li>

     <li><p>Otherwise, <span>in parallel</span>, <span>run a worker</span> given <var>worker</var>,
     <var>urlRecord</var>, <var>outside settings</var>, <var>outside port</var>, and
     <var>options</var>.</p></li>
    </ol>
   </li>

   <li><p>Return <var>worker</var>.</p></li>
  </ol>

  </div>


  <h4 id="navigator.hardwareconcurrency">Concurrent hardware capabilities</h4>

  <pre><code class="idl">interface mixin <dfn interface>NavigatorConcurrentHardware</dfn> {
  readonly attribute unsigned long long <span data-x="dom-navigator-hardwareConcurrency">hardwareConcurrency</span>;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>self</var>.<span data-x="dom-navigator">navigator</span>.<span subdfn data-x="dom-navigator-hardwareConcurrency">hardwareConcurrency</span></code></dt>

   <dd><p>Returns the number of logical processors potentially available to the user agent.</p></dd>
  </dl>

  <div w-nodev>

  <p>
  <!--INSERT TRACKING-->
  The <dfn attribute for="NavigatorConcurrentHardware"><code
  data-x="dom-navigator-hardwareConcurrency">navigator.hardwareConcurrency</code></dfn> attribute's
  getter must return a number between 1 and the number of logical processors potentially available
  to the user agent. If this cannot be determined, the getter must return 1.</p>

  <p>User agents should err toward exposing the number of logical processors available, using lower
  values only in cases where there are user-agent specific limits in place (such as a limitation
  on the number of <span data-x="Worker">workers</span> that can be created) or when the user agent
  desires to limit fingerprinting possibilities.</p>

  </div>

  <h3>APIs available to workers</h3>

  <div w-nodev>

  <h4>Importing scripts and libraries</h4>

  <p>The <dfn method for="WorkerGlobalScope"><code
  data-x="dom-WorkerGlobalScope-importScripts">importScripts(...<var>urls</var>)</code></dfn>
  method steps are:</p>

  <ol>
   <li><p>Let <var>urlStrings</var> be « ».</p></li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>url</var> of <var>urls</var>:</p>

    <ol>
     <li><p><span data-x="list append">Append</span> the result of invoking the <span
     data-x="tt-getcompliantstring">Get Trusted Type compliant string</span> algorithm with <code
     data-x="tt-trustedscripturl">TrustedScriptURL</code>, <span>this</span>'s <span>relevant global
     object</span>, <var>url</var>, "<code data-x="">WorkerGlobalScope importScripts</code>", and
     "<code data-x="">script</code>" to <var>urlStrings</var>.</p></li>
    </ol>
   </li>

   <li><p><span>Import scripts into worker global scope</span> given <span>this</span> and
   <var>urlStrings</var>.</p></li>
  </ol>

  <p>To <dfn export>import scripts into worker global scope</dfn>, given a
  <code>WorkerGlobalScope</code> object <var>worker global scope</var>, a <span>list</span> of
  <span data-x="scalar value string">scalar value strings</span> <var>urls</var>, and an optional
  <span data-x="fetching-scripts-perform-fetch">perform the fetch hook</span>
  <var>performFetch</var>:</p>

  <ol>
   <li><p>If <var>worker global scope</var>'s <span
   data-x="concept-WorkerGlobalScope-type">type</span> is "<code data-x="">module</code>", throw a
   <code>TypeError</code> exception.</p></li>

   <li><p>Let <var>settings object</var> be the <span>current settings object</span>.</p></li>

   <li><p>If <var>urls</var> is empty, return.</p></li>

   <li><p>Let <var>urlRecords</var> be « ».</p></li>

   <li>
    <p>For each <var>url</var> of <var>urls</var>:</p>

    <ol>
     <li><p>Let <var>urlRecord</var> be the result of <span>encoding-parsing a URL</span> given
     <var>url</var>, relative to <var>settings object</var>.</p></li>

     <li><p>If <var>urlRecord</var> is failure, then throw a <span>"<code>SyntaxError</code>"</span>
     <code>DOMException</code>.</p></li>

     <li><p><span data-x="list append">Append</span> <var>urlRecord</var> to
     <var>urlRecords</var>.</p></li>
    </ol>
   </li>

   <li>
    <p>For each <var>urlRecord</var> of <var>urlRecords</var>:</p>

    <ol>
     <li><p><span>Fetch a classic worker-imported script</span> given <var>urlRecord</var> and
     <var>settings object</var>, passing along <var>performFetch</var> if provided. If this
     succeeds, let <var>script</var> be the result. Otherwise, rethrow the exception.</p></li>

     <li>
      <p><span data-x="run a classic script">Run the classic script</span> <var>script</var>, with
      the rethrow errors argument set to true.</p>

      <p class="note"><var>script</var> will run until it either returns, fails to parse, fails to
      catch an exception, or gets <span data-x="abort a running script">prematurely aborted</span>
      by the <span>terminate a worker</span> algorithm defined above.</p>

      <p>If an exception was thrown or if the script was <span data-x="abort a running
      script">prematurely aborted</span>, then abort all these steps, letting the exception or
      aborting continue to be processed by the calling <span
      data-x="concept-script">script</span>.</p>
     </li>
    </ol>
   </li>
  </ol>

  <p class="note"><cite>Service Workers</cite> is an example of a specification that runs this
  algorithm with its own <span data-x="fetching-scripts-perform-fetch">perform the fetch
  hook</span>. <ref>SW</ref></p>

  </div>


  <h4 id="the-workernavigator-object">The <code subdfn>WorkerNavigator</code> interface</h4>

  <div w-nodev>

  <p>The <dfn attribute for="WorkerGlobalScope"><code
  data-x="dom-worker-navigator">navigator</code></dfn> attribute of the
  <code>WorkerGlobalScope</code> interface must return an instance of the
  <code>WorkerNavigator</code> interface, which represents the identity and state of the user agent
  (the client):</p>

  </div>

  <pre><code class="idl">[Exposed=Worker]
interface <dfn interface>WorkerNavigator</dfn> {};
<span>WorkerNavigator</span> includes <span>NavigatorID</span>;
<span>WorkerNavigator</span> includes <span>NavigatorLanguage</span>;
<span>WorkerNavigator</span> includes <span>NavigatorOnLine</span>;
<span>WorkerNavigator</span> includes <span>NavigatorConcurrentHardware</span>;</code></pre>

  <div w-dev>

  <p>The <code>WorkerNavigator</code> interface implements a subset of the <code>Navigator</code>
  interface, consisting of the following APIs:</p>

  <ul class="brief">
   <li><a href="#client-identification">Client identification</a></li>
   <li><a href="#language-preferences">Language preferences</a></li>
   <li><a href="#navigator.online">Browser state</a></li>
   <li><a href="#navigator.hardwareconcurrency">Concurrent hardware capabilities</a></li>
  </ul>

  </div>

  <h4 id="worker-locations">The <code subdfn>WorkerLocation</code> interface</h4>

  <pre><code class="idl">[Exposed=Worker]
interface <dfn interface>WorkerLocation</dfn> {
  stringifier readonly attribute USVString <span data-x="dom-WorkerLocation-href">href</span>;
  readonly attribute USVString <span data-x="dom-WorkerLocation-origin">origin</span>;
  readonly attribute USVString <span data-x="dom-WorkerLocation-protocol">protocol</span>;
  readonly attribute USVString <span data-x="dom-WorkerLocation-host">host</span>;
  readonly attribute USVString <span data-x="dom-WorkerLocation-hostname">hostname</span>;
  readonly attribute USVString <span data-x="dom-WorkerLocation-port">port</span>;
  readonly attribute USVString <span data-x="dom-WorkerLocation-pathname">pathname</span>;
  readonly attribute USVString <span data-x="dom-WorkerLocation-search">search</span>;
  readonly attribute USVString <span data-x="dom-WorkerLocation-hash">hash</span>;
};</code></pre>

  <p w-dev>The <code>WorkerLocation</code> interface is like the <code>Location</code> interface,
  but lacks the <code data-x="dom-Location-assign">assign()</code>, <code
  data-x="dom-Location-replace">replace()</code>, <code data-x="dom-Location-reload">reload()</code>,
  and <code data-x="dom-Location-ancestorOrigins">ancestorOrigins</code> members.</p>

  <div w-nodev>

  <p>A <code>WorkerLocation</code> object has an associated <dfn
  data-x="concept-WorkerLocation-WorkerGlobalScope"><code>WorkerGlobalScope</code> object</dfn> (a
  <code>WorkerGlobalScope</code> object).

  <p>The <dfn attribute for="WorkerLocation"><code
  data-x="dom-WorkerLocation-href">href</code></dfn> getter steps are to return <span>this</span>'s
  <span data-x="concept-WorkerLocation-WorkerGlobalScope"><code>WorkerGlobalScope</code>
  object</span>'s <span data-x="concept-WorkerGlobalScope-url">url</span>, <span
  data-x="concept-url-serializer">serialized</span>.</p>

  <p>The <dfn attribute for="WorkerLocation"><code
  data-x="dom-WorkerLocation-origin">origin</code></dfn> getter steps are to return the <span
  data-x="serialization of an origin">serialization</span> of <span>this</span>'s <span
  data-x="concept-WorkerLocation-WorkerGlobalScope"><code>WorkerGlobalScope</code> object</span>'s
  <span data-x="concept-WorkerGlobalScope-url">url</span>'s <span
  data-x="concept-url-origin">origin</span>.</p>

  <p>The <dfn attribute for="WorkerLocation"><code
  data-x="dom-WorkerLocation-protocol">protocol</code></dfn> getter steps are to return
  <span>this</span>'s <span
  data-x="concept-WorkerLocation-WorkerGlobalScope"><code>WorkerGlobalScope</code> object</span>'s
  <span data-x="concept-WorkerGlobalScope-url">url</span>'s <span
  data-x="concept-url-scheme">scheme</span>, followed by "<code data-x="">:</code>".</p>

  <p>The <dfn attribute for="WorkerLocation"><code
  data-x="dom-WorkerLocation-host">host</code></dfn> getter steps are:</p>

  <ol>
   <li><p>Let <var>url</var> be <span>this</span>'s <span
   data-x="concept-WorkerLocation-WorkerGlobalScope"><code>WorkerGlobalScope</code> object</span>'s
   <span data-x="concept-WorkerGlobalScope-url">url</span>.</p></li>

   <li><p>If <var>url</var>'s <span data-x="concept-url-host">host</span> is null, return the empty
   string.</p></li>

   <li><p>If <var>url</var>'s <span data-x="concept-url-port">port</span> is null, return
   <var>url</var>'s <span data-x="concept-url-host">host</span>, <span data-x="host
   serializer">serialized</span>.</p></li>

   <li><p>Return <var>url</var>'s <span data-x="concept-url-host">host</span>, <span data-x="host
   serializer">serialized</span>, followed by "<code data-x="">:</code>" and <var>url</var>'s <span
   data-x="concept-url-port">port</span>, <span data-x="serialize an
   integer">serialized</span>.</p></li>
  </ol>

  <p>The <dfn attribute for="WorkerLocation"><code
  data-x="dom-WorkerLocation-hostname">hostname</code></dfn> getter steps are:</p>

  <ol>
   <li><p>Let <var>host</var> be <span>this</span>'s <span
   data-x="concept-WorkerLocation-WorkerGlobalScope"><code>WorkerGlobalScope</code> object</span>'s
   <span data-x="concept-WorkerGlobalScope-url">url</span>'s <span
   data-x="concept-url-host">host</span>.</p></li>

   <li><p>If <var>host</var> is null, return the empty string.</p></li>

   <li><p>Return <var>host</var>, <span data-x="host serializer">serialized</span>.</p></li>
  </ol>

  <p>The <dfn attribute for="WorkerLocation"><code
  data-x="dom-WorkerLocation-port">port</code></dfn> getter steps are:</p>

  <ol>
   <li><p>Let <var>port</var> be <span>this</span>'s <span
   data-x="concept-WorkerLocation-WorkerGlobalScope"><code>WorkerGlobalScope</code> object</span>'s
   <span data-x="concept-WorkerGlobalScope-url">url</span>'s <span
   data-x="concept-url-port">port</span>.</p></li>

   <li><p>If <var>port</var> is null, return the empty string.</p></li>

   <li><p>Return <var>port</var>, <span data-x="serialize an integer">serialized</span>.</p></li>
  </ol>

  <p>The <dfn attribute for="WorkerLocation"><code
  data-x="dom-WorkerLocation-pathname">pathname</code></dfn> getter steps are to return the result
  of <span data-x="URL path serializer">URL path serializing</span> <span>this</span>'s <span
  data-x="concept-WorkerLocation-WorkerGlobalScope"><code>WorkerGlobalScope</code> object</span>'s
  <span data-x="concept-WorkerGlobalScope-url">url</span>.</p>

  <p>The <dfn attribute for="WorkerLocation"><code
  data-x="dom-WorkerLocation-search">search</code></dfn> getter steps are:</p>

  <ol>
   <li><p>Let <var>query</var> be <span>this</span>'s <span
   data-x="concept-WorkerLocation-WorkerGlobalScope"><code>WorkerGlobalScope</code> object</span>'s
   <span data-x="concept-WorkerGlobalScope-url">url</span>'s <span
   data-x="concept-url-query">query</span>.</p></li>

   <li><p>If <var>query</var> is either null or the empty string, return the empty string.</p></li>

   <li><p>Return "<code data-x="">?</code>", followed by <var>query</var>.</p></li>
  </ol>

  <p>The <dfn attribute for="WorkerLocation"><code
  data-x="dom-WorkerLocation-hash">hash</code></dfn> getter steps are:</p>

  <ol>
   <li><p>Let <var>fragment</var> be <span>this</span>'s <span
   data-x="concept-WorkerLocation-WorkerGlobalScope"><code>WorkerGlobalScope</code> object</span>'s
   <span data-x="concept-WorkerGlobalScope-url">url</span>'s <span
   data-x="concept-url-fragment">fragment</span>.</p></li>

   <li><p>If <var>fragment</var> is either null or the empty string, return the empty string.</p></li>

   <li><p>Return "<code data-x="">#</code>", followed by <var>fragment</var>.</p></li>
  </ol>

  </div>


  <h2 split-filename="worklets" id="worklets">Worklets</h2>

  <h3 id="worklets-intro">Introduction</h3>

  <!-- NON-NORMATIVE SECTION -->

  <p>Worklets are a piece of specification infrastructure which can be used for running scripts
  independent of the main JavaScript execution environment, while not requiring any particular
  implementation model.</p>

  <p>The worklet infrastructure specified here cannot be used directly by web developers. Instead,
  other specifications build upon it to create directly-usable worklet types, specialized for
  running in particular parts of the browser implementation pipeline.</p>

  <h4 id="worklets-motivations">Motivations</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>Allowing extension points to rendering, or other sensitive parts of the implementation pipeline
  such as audio output, is difficult. If extension points were done with full access to the APIs
  available on <code>Window</code>, engines would need to abandon previously-held assumptions for
  what could happen in the middle of those phases. For example, during the layout phase, rendering
  engines assume that no DOM will be modified.</p>

  <p>Additionally, defining extension points in the <code>Window</code> environment would restrict
  user agents to performing work in the same thread as the <code>Window</code> object. (Unless
  implementations added complex, high-overhead infrastructure to allow thread-safe APIs, as well
  as thread-joining guarantees.)</p>

  <p>Worklets are designed to allow extension points, while keeping guarantees that user agents
  currently rely on. This is done through new global environments, based on subclasses of
  <code>WorkletGlobalScope</code>.</p>

  <p>Worklets are similar to web workers. However, they:</p>

  <ul>
   <li><p>Are thread-agnostic. That is, they are not designed to run on a dedicated separate thread,
   like each worker is. Implementations can run worklets wherever they choose (including on the main
   thread).</p></li>

   <li><p>Are able to have multiple duplicate instances of the global scope created, for the purpose
   of parallelism.</p></li>

   <li><p>Do not use an event-based API. Instead, classes are registered on the global scope, whose
   methods are invoked by the user agent.</p></li>

   <li><p>Have a reduced API surface on the global scope.</p></li>

   <li><p>Have a lifetime for their <span>global object</span> which is defined by other
   specifications, often in an <span>implementation-defined</span> manner.</p></li>
  </ul>

  <p>As worklets have relatively high overhead, they are best used sparingly. Due to this, a given
  <code>WorkletGlobalScope</code> is expected to be shared between multiple separate scripts. (This
  is similar to how a single <code>Window</code> is shared between multiple separate scripts.)</p>

  <p>Worklets are a general technology that serve different use cases. Some worklets, such as those
  defined in <cite>CSS Painting API</cite>, provide extension points intended for stateless,
  idempotent, and short-running computations, which have special considerations as described in the
  next couple of sections. Others, such as those defined in <cite>Web Audio API</cite>, are used for
  stateful, long-running operations. <ref>CSSPAINT</ref> <ref>WEBAUDIO</ref></p>

  <h4 id="worklets-idempotent">Code idempotence</h4>

  <p>Some specifications which use worklets are intended to allow user agents to parallelize work
  over multiple threads, or to move work between threads as required. In these specifications, user
  agents might invoke methods on a web-developer-provided class in an
  <span>implementation-defined</span> order.</p>

  <p>As a result of this, to prevent interoperability issues, authors who register classes on such
  <code>WorkletGlobalScope</code>s should make their code idempotent. That is, a method or set of
  methods on the class should produce the same output given a particular input.</p>

  <p>This specification uses the following techniques in order to encourage authors to write code in
  an idempotent way:</p>

  <ul>
   <li>
    <p>No reference to the global object is available (i.e., there is no counterpart to <code
    data-x="dom-WorkerGlobalScope-self">self</code> on <code>WorkletGlobalScope</code>).</p>

    <p class="XXX">Although this was the intention when worklets were first specified, the
    introduction of <code data-x="">globalThis</code> has made it no longer true. See <a
    href="https://github.com/whatwg/html/issues/6059">issue #6059</a> for more discussion.</p>
   </li>

   <li><p>Code is loaded as a <span>module script</span>, which results in the code being executed
   in strict mode and with no shared <code data-x="">this</code> referencing the global
   proxy.</p></li>
  </ul>

  <p>Together, these restrictions help prevent two different scripts from sharing state using
  properties of the <span>global object</span>.</p>

  <p>Additionally, specifications which use worklets and intend to allow
  <span>implementation-defined</span> behavior must obey the following:</p>

  <ul>
   <li><p>They must require user agents to always have at least two <code>WorkletGlobalScope</code>
   instances per <code>Worklet</code>, and randomly assign a method or set of methods on a class to
   a particular <code>WorkletGlobalScope</code> instance. These specifications may provide an
   opt-out under memory constraints.</p></li>

   <li><p>These specifications must allow user agents to create and destroy instances of their
   <code>WorkletGlobalScope</code> subclasses at any time.</p></li>
  </ul>

  <h4 id="worklets-speculative">Speculative evaluation</h4>

  <p>Some specifications which use worklets can invoke methods on a web-developer-provided class
  based on the state of the user agent. To increase concurrency between threads, a user agent may
  invoke a method speculatively, based on potential future states.</p>

  <p>In these specifications, user agents might invoke such methods at any time, and with any
  arguments, not just ones corresponding to the current state of the user agent. The results of such
  speculative evaluations are not displayed immediately, but can be cached for use if the user agent
  state matches the speculated state. This can increase the concurrency between the user agent and
  worklet threads.</p>

  <p>As a result of this, to prevent interoperability risks between user agents, authors who
  register classes on such <code>WorkletGlobalScope</code>s should make their code stateless. That
  is, the only effect of invoking a method should be its result, and not any side effects such as
  updating mutable state.</p>

  <p>The same techniques which encourage <a href="#worklets-idempotent">code idempotence</a> also
  encourage authors to write stateless code.</p>

  <div class="non-normative">

  <h3 id="worklets-examples">Examples</h3>

  <!-- NON-NORMATIVE SECTION -->

  <p>For these examples, we'll use a fake worklet. The <code>Window</code> object provides two
  <code>Worklet</code> instances, which each run code in their own collection of
  <code>FakeWorkletGlobalScope</code>s:

  <pre class="extract"><code class="idl">partial interface <span>Window</span> {
  [SameObject, SecureContext] readonly attribute <span>Worklet</span> <span>fakeWorklet1</span>;
  [SameObject, SecureContext] readonly attribute <span>Worklet</span> <span>fakeWorklet2</span>;
};</code></pre>

  <dl class="domintro" w-dev>
   <dt><code data-x="">window.fakeWorklet1</code></dt>
   <dd>Returns one of the fake worklets.</dd>

   <dt><code data-x="">window.fakeWorklet2</code></dt>
   <dd>Returns another of the fake worklets.</dd>
  </dl>

  <p><span w-nodev>Each <code>Window</code> has two <code>Worklet</code> instances, <dfn>fake
  worklet 1</dfn> and <dfn>fake worklet 2</dfn>. </span>Both of these have their <span>worklet
  global scope type</span> set to <code>FakeWorkletGlobalScope</code>, and their <span>worklet
  destination type</span> set to "<code data-x="">fakeworklet</code>". User agents should create at
  least two <code>FakeWorkletGlobalScope</code> instances per worklet.</p>

  <p class="note">"<code data-x="">fakeworklet</code>" is not actually a valid <span
  data-x="concept-request-destination">destination</span> per <cite>Fetch</cite>. But this
  illustrates how real worklets would generally have their own worklet-type-specific destination.
  <ref>FETCH</ref></p>

  <div w-nodev>

  <p>The <dfn noexport><code>fakeWorklet1</code></dfn> getter steps are to return
  <span>this</span>'s <span>fake worklet 1</span>.</p>

  <p>The <dfn noexport><code>fakeWorklet2</code></dfn> getter steps are to return
  <span>this</span>'s <span>fake worklet 2</span>.</p>

  </div>

  <hr>

  <pre class="extract"><code class="idl">[Global=(Worklet,FakeWorklet),
 Exposed=FakeWorklet,
 SecureContext]
interface <dfn interface noexport>FakeWorkletGlobalScope</dfn> : <span>WorkletGlobalScope</span> {
  undefined <span data-x="dom-FakeWorkletGlobalScope-registerFake">registerFake</span>(DOMString type, <span data-x="idl-Function">Function</span> classConstructor);
};</code></pre>

  <p w-dev>Inside a <code subdfn>FakeWorkletGlobalScope</code>, the following global method is
  available:</p>

  <dl class="domintro" w-dev>
   <dt><code data-x="">registerFake(<var>type</var>, <var>classConstructor</var>)</code></dt>
   <dd>Registers the JavaScript class given by <var>classConstructor</var> for use when the user
   agent later wants to do some operation specified by <var>type</var>.</dd>
  </dl>

  <div w-nodev>

  <p>Each <code>FakeWorkletGlobalScope</code> has a <dfn>registered class constructors map</dfn>,
  which is an <span>ordered map</span>, initially empty.</p>

  <p>The <dfn data-x="dom-FakeWorkletGlobalScope-registerFake"
  noexport><code>registerFake(<var>type</var>, <var>classConstructor</var>)</code></dfn> method
  steps are to set <span>this</span>'s <span>registered class constructors
  map</span>[<var>type</var>] to <var>classConstructor</var>.</p>

  </div>

  <h4 id="worklets-examples-loading">Loading scripts</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>To load scripts into <span>fake worklet 1</span>, a web developer would write:</p>

  <pre><code class="js">window.fakeWorklet1.addModule('script1.mjs');
window.fakeWorklet1.addModule('script2.mjs');</code></pre>

  <p>Note that which script finishes fetching and runs first is dependent on network timing: it
  could be either <code data-x="">script1.mjs</code> or <code data-x="">script2.mjs</code>. This
  generally won't matter for well-written scripts intended to be loaded in worklets, if they follow
  the suggestions about preparing for <a href="#worklets-speculative">speculative
  evaluation</a>.</p>

  <p>If a web developer wants to perform a task only after the scripts have successfully run and
  loaded into some worklets, they could write:</p>

  <pre><code class="js">Promise.all([
    window.fakeWorklet1.addModule('script1.mjs'),
    window.fakeWorklet2.addModule('script2.mjs')
]).then(() => {
    // Do something which relies on those scripts being loaded.
});</code></pre>

  <hr>

  <p>Another important point about script-loading is that loaded scripts can be run in multiple
  <code>WorkletGlobalScope</code>s per <code>Worklet</code>, as discussed in the section on <a
  href="#worklets-idempotent">code idempotence</a>. In particular, the specification above
  for <span>fake worklet 1</span> and <span>fake worklet 2</span> require this. So, consider a
  scenario such as the following:</p>

  <pre><code class="js">// script.mjs
console.log("Hello from a FakeWorkletGlobalScope!");</code></pre>

  <pre><code class="js">// app.mjs
window.fakeWorklet1.addModule("script.mjs");</code></pre>

  <p>This could result in output such as the following from a user agent's console:</p>

  <pre><code data-x="">[fakeWorklet1#1] Hello from a FakeWorkletGlobalScope!
[fakeWorklet1#4] Hello from a FakeWorkletGlobalScope!
[fakeWorklet1#2] Hello from a FakeWorkletGlobalScope!
[fakeWorklet1#3] Hello from a FakeWorkletGlobalScope!</code></pre>

  <p>If the user agent at some point decided to kill and restart the third instance of
  <code>FakeWorkletGlobalScope</code>, the console would again print <code
  data-x="">[fakeWorklet1#3] Hello from a FakeWorkletGlobalScope!</code> when this occurs.</p>

  <h4 id="worklets-example-registering">Registering a class and invoking its methods</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>Let's say that one of the intended usages of our fake worklet by web developers is to allow
  them to customize the highly-complex process of boolean negation. They might register their
  customization as follows:</p>

  <pre><code class="js">// script.mjs
registerFake('negation-processor', class {
  process(arg) {
    return !arg;
  }
});</code></pre>

  <pre><code class="js">// app.mjs
window.fakeWorklet1.addModule("script.mjs");</code></pre>

  <p>To make use of such registered classes, the specification for fake worklets could define a <dfn
  data-x="FakeWorkletGlobalScope-process">find the opposite of true</dfn> algorithm, given a
  <code>Worklet</code> <var>worklet</var><span w-dev>, which invokes the <code
  data-x="">process</code> method on any class registered to one of <var>worklet</var>'s global
  scopes as having type "<code data-x="">negation-processor</code>", with true as the argument, and
  then uses the result in some way.</span><span w-nodev>:</span></p>

  <div w-nodev>

  <ol>
   <li><p>Optionally, <span>create a worklet global scope</span> for <var>worklet</var>.</p></li>

   <li><p>Let <var>workletGlobalScope</var> be one of <var>worklet</var>'s <span
   data-x="concept-Worklet-global-scopes">global scopes</span>, chosen in an
   <span>implementation-defined</span> manner.</p></li>

   <li><p>Let <var>classConstructor</var> be <var>workletGlobalScope</var>'s <span>registered class
   constructors map</span>["<code data-x="">negation-processor</code>"].</p></li>

   <li><p>Let <var>classInstance</var> be the result of <span
   data-x="es-constructing-callback-functions">constructing</span> <var>classConstructor</var>, with
   no arguments.</p></li>

   <li><p>Let <var>function</var> be <span data-x="js-Get">Get</span>(<var>classInstance</var>,
   "<code data-x="">process</code>"). Rethrow any exceptions.</p>

   <li><p>Let <var>callback</var> be the result of <span
   data-x="concept-idl-convert">converting</span> <var>function</var> to a Web IDL <code
   data-x="idl-Function">Function</code> instance.</p></li>

   <li><p>Return the result of <span data-x="es-invoking-callback-functions">invoking</span>
   <var>callback</var> with « true » and "<code data-x="">rethrow</code>", and with <i
   data-x="dfn-callback-this-value">callback this value</i> set to
   <var>classInstance</var>.</p></li>
  </ol>

  <p class="note">Another, perhaps better, specification architecture would be to extract the "<code
  data-x="">process</code>" property and convert it into a <code
  data-x="idl-Function">Function</code> at registration time, as part of the <code
  data-x="dom-FakeWorkletGlobalScope-registerFake">registerFake()</code> method steps.</p>

  </div>

  </div>

  <h3 id="worklets-infrastructure">Infrastructure</h3>

  <h4 id="worklets-global">The global scope</h4>

  <p>Subclasses of <code subdfn>WorkletGlobalScope</code> are used to create <span data-x="global
  object">global objects</span> wherein code loaded into a particular <code>Worklet</code> can
  execute.</p>

  <pre><code class="idl">[Exposed=Worklet, SecureContext]
interface <dfn interface>WorkletGlobalScope</dfn> {};</code></pre>

  <p class="note">Other specifications are intended to subclass <code>WorkletGlobalScope</code>,
  adding APIs to register a class, as well as other APIs specific for their worklet type.</p>

  <div w-nodev>

  <p>Each <code>WorkletGlobalScope</code> has an associated <dfn
  data-x="concept-WorkletGlobalScope-module-map">module map</dfn>. It is a <span>module map</span>,
  initially empty.</p>

  </div>

  <h5 id="worklet-agents-and-event-loops">Agents and event loops</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>Each <code>WorkletGlobalScope</code> is contained in its own <span>worklet agent</span>, which
  has its corresponding <span data-x="concept-agent-event-loop">event loop</span>. However, in
  practice, implementation of these agents and event loops is expected to be different from most
  others.</p>

  <p>A <span>worklet agent</span> exists for each <code>WorkletGlobalScope</code> since, in theory,
  an implementation could use a separate thread for each <code>WorkletGlobalScope</code> instance,
  and allowing this level of parallelism is best done using agents. However, because their
  [[CanBlock]] value is false, there is no requirement that agents and threads are one-to-one. This
  allows implementations the freedom to execute scripts loaded into a worklet on any thread,
  including one running code from other agents with [[CanBlock]] of false, such as the thread of a
  <span>similar-origin window agent</span> ("the main thread"). Contrast this with <span
  data-x="dedicated worker agent">dedicated worker agents</span>, whose true value for [[CanBlock]]
  effectively requires them to get a dedicated operating system thread.</p>

  <p>Worklet <span data-x="event loop">event loops</span> are also somewhat special. They are only
  used for <span data-x="concept-task">tasks</span> associated with <code
  data-x="dom-Worklet-addModule">addModule()</code>, tasks wherein the user agent invokes
  author-defined methods, and <span data-x="microtask">microtasks</span>. Thus, even though the <a
  href="#event-loop-processing-model">event loop processing model</a> specifies that all event loops
  run continuously, implementations can achieve observably-equivalent results using a simpler
  strategy, which just <span data-x="es-invoking-callback-functions">invokes</span> author-provided
  methods and then relies on that process to <span>perform a microtask checkpoint</span>.</p>

  <div w-nodev>

  <h5 id="worklets-creation-termination">Creation and termination</h5>

  <p>To <dfn export>create a worklet global scope</dfn> for a <code>Worklet</code>
  <var>worklet</var>:</p>

  <ol>
   <li><p>Let <var>outsideSettings</var> be <var>worklet</var>'s <span>relevant settings
   object</span>.</p></li>

   <li><p>Let <var>agent</var> be the result of <span data-x="obtain a worklet agent">obtaining a
   worklet agent</span> given <var>outsideSettings</var>. Run the rest of these steps in that
   agent.</p></li>

   <li>
    <p>Let <var>realmExecutionContext</var> be the result of <span>creating a new realm</span> given
    <var>agent</var> and the following customizations:</p>

    <ul>
     <li><p>For the global object, create a new object of the type given by <var>worklet</var>'s
     <span>worklet global scope type</span>.</p></li>
    </ul>
   </li>

   <li><p>Let <var>workletGlobalScope</var> be the <span data-x="concept-realm-global">global
   object</span> of <var>realmExecutionContext</var>'s Realm component.</p></li>

   <li><p>Let <var>insideSettings</var> be the result of <span data-x="set up a worklet environment
   settings object">setting up a worklet environment settings object</span> given
   <var>realmExecutionContext</var> and <var>outsideSettings</var>.</p></li>

   <li><p>Let <var>pendingAddedModules</var> be a <span data-x="list clone">clone</span> of
   <var>worklet</var>'s <span data-x="concept-Worklet-added-modules-list">added modules
   list</span>.</p></li>

   <li>
    <p>Let <var>runNextAddedModule</var> be the following steps:</p>

    <ol>
     <li>
      <p>If <var>pendingAddedModules</var> <span data-x="list is empty">is not empty</span>, then:</p>

      <ol>
       <li><p>Let <var>moduleURL</var> be the result of <span data-x="dequeue">dequeueing</span>
       from <var>pendingAddedModules</var>.</p></li>

       <li>
        <p><span>Fetch a worklet script graph</span> given <var>moduleURL</var>,
        <var>insideSettings</var>, <var>worklet</var>'s <span>worklet destination type</span>, <span
        class="XXX">what credentials mode?</span>, <var>insideSettings</var>, <var>worklet</var>'s
        <span data-x="concept-Worklet-module-responses-map">module responses map</span>, and with
        the following steps given <var>script</var>:</p>

        <p class="note">This will not actually perform a network request, as it will just reuse
        <span data-x="concept-response">responses</span> from <var>worklet</var>'s <span
        data-x="concept-Worklet-module-responses-map">module responses map</span>. The main
        purpose of this step is to create a new <var>workletGlobalScope</var>-specific <span>module
        script</span> from the <span data-x="concept-response">response</span>.</p>

        <ol>
         <li><p><span>Assert</span>: <var>script</var> is not null, since the fetch succeeded and
         the source text was successfully parsed when <var>worklet</var>'s <span
         data-x="concept-Worklet-module-responses-map">module responses map</span> was initially
         populated with <var>moduleURL</var>.</p></li>

         <li><p><span>Run a module script</span> given <var>script</var>.</p></li>

         <li><p>Run <var>runNextAddedModule</var>.</p></li>
        </ol>
       </li>

       <li>Abort these steps.</li>
      </ol>
     </li>

     <li><p><span data-x="list append">Append</span> <var>workletGlobalScope</var> to
     <var>outsideSettings</var>'s <span data-x="concept-settings-object-global">global
     object</span>'s <span data-x="concept-document-window">associated
     <code>Document</code></span>'s <span data-x="concept-Document-worklet-global-scopes">worklet
     global scopes</span>.</p></li>

     <li><p><span data-x="list append">Append</span> <var>workletGlobalScope</var> to
     <var>worklet</var>'s <span data-x="concept-Worklet-global-scopes">global
     scopes</span>.</p></li>

     <li><p>Run the <span>responsible event loop</span> specified by
     <var>insideSettings</var>.</p></li>
    </ol>
   </li>

   <li><p>Run <var>runNextAddedModule</var>.</p></li>
  </ol>

  <p>To <dfn export>terminate a worklet global scope</dfn> given a <code>WorkletGlobalScope</code>
  <var>workletGlobalScope</var>:</p>

  <ol>
   <li><p>Let <var>eventLoop</var> be <var>workletGlobalScope</var>'s <span>relevant agent</span>'s
   <span data-x="concept-agent-event-loop">event loop</span>.</p></li>

   <li><p>If there are any <span data-x="concept-task">tasks</span> queued in <var>eventLoop</var>'s
   <span data-x="task queue">task queues</span>, discard them without processing them.</p></li>

   <li><p>Wait for <var>eventLoop</var> to complete the <span>currently running
   task</span>.</p></li>

   <li><p>If the previous step doesn't complete within an <span>implementation-defined</span> period
   of time, then <span data-x="abort a running script">abort the script</span> currently running in
   the worklet.</p></li>

   <li><p>Destroy <var>eventLoop</var>.</p></li>

   <li><p><span data-x="list remove">Remove</span> <var>workletGlobalScope</var> from the <span
   data-x="concept-Worklet-global-scopes">global scopes</span> of the <code>Worklet</code> whose
   <span data-x="concept-Worklet-global-scopes">global scopes</span> contains
   <var>workletGlobalScope</var>.</p></li>

   <li><p><span data-x="list remove">Remove</span> <var>workletGlobalScope</var> from the <span
   data-x="concept-Document-worklet-global-scopes">worklet global scopes</span> of the
   <code>Document</code> whose <span data-x="concept-Document-worklet-global-scopes">worklet global
   scopes</span> contains <var>workletGlobalScope</var>.</p></li>
  </ol>

  <h5>Script settings for worklets</h5>

  <p>To <dfn>set up a worklet environment settings object</dfn>, given a <span>JavaScript
  execution context</span> <var>executionContext</var> and an <span>environment settings
  object</span> <var>outsideSettings</var>:</p>

  <ol>
   <li><p>Let <var>origin</var> be a unique <span data-x="concept-origin-opaque">opaque
   origin</span>.</p></li>

   <li><p>Let <var>inheritedAPIBaseURL</var> be <var>outsideSettings</var>'s <span>API base
   URL</span>.</p></li>

   <li><p>Let <var>inheritedPolicyContainer</var> be a <span data-x="clone a policy
   container">clone</span> of <var>outsideSettings</var>'s <span
   data-x="concept-settings-object-policy-container">policy container</span>.</p></li>

   <li><p>Let <var>realm</var> be the value of <var>executionContext</var>'s Realm
   component.</p></li>

   <li><p>Let <var>workletGlobalScope</var> be <var>realm</var>'s <span
   data-x="concept-realm-global">global object</span>.</p></li>

   <li>
    <p>Let <var>settingsObject</var> be a new <span>environment settings object</span> whose
    algorithms are defined as follows:</p>

    <dl>
     <dt>The <span>realm execution context</span></dt>
     <dd>
      <p>Return <var>executionContext</var>.</p>
     </dd>

     <dt>The <span data-x="concept-settings-object-module-map">module map</span></dt>
     <dd>
      <p>Return <var>workletGlobalScope</var>'s <span
      data-x="concept-WorkletGlobalScope-module-map">module map</span>.</p>
     </dd>

     <dt>The <span>API base URL</span></dt>
     <dd>
      <p>Return <var>inheritedAPIBaseURL</var>.</p>

      <p class="note">Unlike workers or other globals derived from a single resource, worklets have
      no primary resource; instead, multiple scripts, each with their own URL, are loaded into the
      global scope via <code
      data-x="dom-Worklet-addModule">worklet.addModule()</code>. So this <span>API base URL</span>
      is rather unlike that of other globals. However, so far this doesn't matter, as no APIs
      available to worklet code make use of the <span>API base URL</span>.</p>
     </dd>

     <dt>The <span data-x="concept-settings-object-origin">origin</span></dt>
     <dd>
      <p>Return <var>origin</var>.</p>
     </dd>

     <dt>The <span data-x="concept-settings-object-policy-container">policy container</span></dt>
     <dd>
      <p>Return <var>inheritedPolicyContainer</var>.</p>
     </dd>

     <dt>The <span data-x="concept-settings-object-cross-origin-isolated-capability">cross-origin
     isolated capability</span></dt>
     <dd><p>Return <span class="XXX">TODO</span>.</p></dd>

     <dt>The <span data-x="concept-settings-object-time-origin">time origin</span></dt>
     <dd><p><span>Assert</span>: this algorithm is never called, because the <span
     data-x="concept-settings-object-time-origin">time origin</span> is not available in a worklet
     context.</p></dd>
    </dl>
   </li>

   <li><p>Set <var>settingsObject</var>'s <span data-x="concept-environment-id">id</span> to a new
   unique opaque string, <span data-x="concept-environment-creation-url">creation URL</span> to
   <var>inheritedAPIBaseURL</var>, <span>top-level creation URL</span> to null, <span>top-level
   origin</span> to <var>outsideSettings</var>'s <span>top-level origin</span>, <span
   data-x="concept-environment-target-browsing-context">target browsing context</span> to null, and
   <span data-x="concept-environment-active-service-worker">active service worker</span> to
   null.</p></li>

   <li><p>Set <var>realm</var>'s [[HostDefined]] field to <var>settingsObject</var>.</p></li>

   <li><p>Return <var>settingsObject</var>.</p></li>
  </ol>

  </div>

  <h4 id="worklets-worklet">The <code>Worklet</code> class</h4>

  <p>The <code subdfn>Worklet</code> class provides the capability to add module scripts into its
  associated <code>WorkletGlobalScope</code>s. The user agent can then create classes registered on
  the <code>WorkletGlobalScope</code>s and invoke their methods.</p>

  <pre><code class="idl">[Exposed=Window, SecureContext]
interface <dfn interface>Worklet</dfn> {
  [NewObject] <span data-x="idl-Promise">Promise</span>&lt;undefined> <span data-x="dom-Worklet-addModule">addModule</span>(USVString moduleURL, optional <span>WorkletOptions</span> options = {});
};

dictionary <dfn dictionary>WorkletOptions</dfn> {
  <span>RequestCredentials</span> <dfn dict-member for="WorkletOptions" data-x="dom-WorkletOptions-credentials">credentials</dfn> = "same-origin";
};</code></pre>

  <p>Specifications that create <code>Worklet</code> instances must specify the following for a
  given instance:</p>

  <ul>
   <li><p>its <dfn export>worklet global scope type</dfn>, which must be a Web IDL type that <span
   data-x="inherit">inherits</span> from <code>WorkletGlobalScope</code>; and</p></li>

   <li><p>its <dfn export>worklet destination type</dfn>, which must be a <span
   data-x="concept-request-destination">destination</span>, and is used when fetching
   scripts.</p></li>
  </ul>

  <dl class="domintro">
   <dt><code data-x="">await <var>worklet</var>.<span subdfn data-x="dom-Worklet-addModule">addModule</span>(<var>moduleURL</var>[, { <span data-x="dom-WorkletOptions-credentials">credentials</span> }])</code></dt>
   <dd>
    <p>Loads and executes the <span>module script</span> given by <var>moduleURL</var> into all of
    <var>worklet</var>'s <span data-x="concept-Worklet-global-scopes">global scopes</span>. It can
    also create additional global scopes as part of this process, depending on the worklet type. The
    returned promise will fulfill once the script has been successfully loaded and run in all global
    scopes.</p>

    <p>The <code data-x="dom-WorkletOptions-credentials">credentials</code> option can be set to a
    <span data-x="concept-request-credentials-mode">credentials mode</span> to modify the
    script-fetching process. It defaults to "<code data-x="">same-origin</code>".</p>

    <p>Any failures in <span data-x="fetch a worklet script graph">fetching</span> the script or its
    dependencies will cause the returned promise to be rejected with an
    <span>"<code>AbortError</code>"</span> <code>DOMException</code>. Any errors in parsing the
    script or its dependencies will cause the returned promise to be rejected with the exception
    generated during parsing.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>A <code>Worklet</code> has a <span>list</span> of <dfn
  data-x="concept-Worklet-global-scopes" export for="Worklet">global scopes</dfn>, which contains
  instances of the <code>Worklet</code>'s <span>worklet global scope type</span>. It is initially
  empty.</p>

  <p>A <code>Worklet</code> has an <dfn data-x="concept-Worklet-added-modules-list">added modules
  list</dfn>, which is a <span>list</span> of <span data-x="URL">URLs</span>, initially empty.
  Access to this list should be thread-safe.</p>

  <p>A <code>Worklet</code> has a <dfn data-x="concept-Worklet-module-responses-map">module
  responses map</dfn>, which is an <span>ordered map</span> from <span data-x="URL">URLs</span> to
  either "<code data-x="">fetching</code>" or <span data-x="tuple">tuples</span> consisting of a
  <span data-x="concept-response">response</span> and either null, failure, or a <span>byte
  sequence</span> representing the response body. This map is initially empty, and access to it
  should be thread-safe.</p>

  <div class="note">
   <p>The <span data-x="concept-Worklet-added-modules-list">added modules list</span> and <span
   data-x="concept-Worklet-module-responses-map">module responses map</span> exist to ensure that
   <code>WorkletGlobalScope</code>s created at different times get equivalent <span data-x="module
   script">module scripts</span> run in them, based on the same source text. This allows the
   creation of additional <code>WorkletGlobalScope</code>s to be transparent to the author.</p>

   <p>In practice, user agents are not expected to implement these data structures, and the
   algorithms that consult them, using thread-safe programming techniques. Instead, when <code
   data-x="dom-Worklet-addModule">addModule()</code> is called, user agents can fetch the module
   graph on the main thread, and send the fetched source text (i.e., the important data contained in
   the <span data-x="concept-Worklet-module-responses-map">module responses map</span>) to each
   thread which has a <code>WorkletGlobalScope</code>.</p>

   <p>Then, when a user agent <span data-x="create a worklet global scope">creates</span> a new
   <code>WorkletGlobalScope</code> for a given <code>Worklet</code>, it can simply send the map of
   fetched source text and the list of entry points from the main thread to the thread containing
   the new <code>WorkletGlobalScope</code>.</p>
  </div>

  <p>The <dfn method for="Worklet"
  data-x="dom-Worklet-addModule"><code>addModule(<var>moduleURL</var>,
  <var>options</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>outsideSettings</var> be the <span>relevant settings object</span> of
   <span>this</span>.</p></li>

   <li><p>Let <var>moduleURLRecord</var> be the result of <span>encoding-parsing a URL</span> given
   <var>moduleURL</var>, relative to <var>outsideSettings</var>.</p></li>

   <li><p>If <var>moduleURLRecord</var> is failure, then return <span>a promise rejected with</span>
   a <span>"<code>SyntaxError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>Let <var>promise</var> be a new promise.</p></li>

   <li><p>Let <var>workletInstance</var> be <span>this</span>.</p></li>

   <li>
    <p>Run the following steps <span>in parallel</span>:</p>

    <ol>
     <li>
      <p>If <var>workletInstance</var>'s <span data-x="concept-Worklet-global-scopes">global
      scopes</span> <span data-x="list is empty">is empty</span>, then:</p>

      <ol>
       <li><p><span>Create a worklet global scope</span> given <var>workletInstance</var>.</p></li>

       <li><p>Optionally, <span data-x="create a worklet global scope">create</span> additional
       global scope instances given <var>workletInstance</var>, depending on the specific worklet in
       question and its specification.</p></li>

       <li><p>Wait for all steps of the <span data-x="create a worklet global scope">creation</span>
       process(es) — including those taking place within the <span data-x="worklet agent">worklet
       agents</span> — to complete, before moving on.</p></li>
      </ol>
     </li>

     <li><p>Let <var>pendingTasks</var> be <var>workletInstance</var>'s <span
     data-x="concept-Worklet-global-scopes">global scopes</span>'s <span data-x="list
     size">size</span>.</p></li>

     <li><p>Let <var>addedSuccessfully</var> be false.</p></li>

     <li>
      <p><span data-x="list iterate">For each</span> <var>workletGlobalScope</var> of
      <var>workletInstance</var>'s <span data-x="concept-Worklet-global-scopes">global
      scopes</span>, <span>queue a global task</span> on the <span>networking task source</span>
      given <var>workletGlobalScope</var> to <span>fetch a worklet script graph</span> given
      <var>moduleURLRecord</var>, <var>outsideSettings</var>, <var>workletInstance</var>'s
      <span>worklet destination type</span>, <var>options</var>["<code
      data-x="dom-WorkletOptions-credentials">credentials</code>"], <var>workletGlobalScope</var>'s
      <span>relevant settings object</span>, <var>workletInstance</var>'s <span
      data-x="concept-Worklet-module-responses-map">module responses map</span>, and the following
      steps given <var>script</var>:</p>

      <p class="note">Only the first of these fetches will actually perform a network request; the
      ones for other <code>WorkletGlobalScope</code>s will reuse <span
      data-x="concept-response">responses</span> from <var>workletInstance</var>'s <span
      data-x="concept-Worklet-module-responses-map">module responses map</span>.</p>

      <ol>
       <li>
        <p>If <var>script</var> is null, then:</p>

        <ol>
         <li>
          <p><span>Queue a global task</span> on the <span>networking task source</span> given
          <var>workletInstance</var>'s <span>relevant global object</span> to perform the following
          steps:</p>

          <ol>
           <li>
            <p>If <var>pendingTasks</var> is not &#x2212;1, then:</p>

            <ol>
             <li><p>Set <var>pendingTasks</var> to &#x2212;1.</p></li>

             <li><p>Reject <var>promise</var> with an <span>"<code>AbortError</code>"</span>
             <code>DOMException</code>.</p></li>
            </ol>
           </li>
          </ol>
         </li>

         <li><p>Abort these steps.</p></li>
        </ol>
       </li>

       <li>
        <p>If <var>script</var>'s <span data-x="concept-script-error-to-rethrow">error to
        rethrow</span> is not null, then:</p>

        <ol>
         <li>
          <p><span>Queue a global task</span> on the <span>networking task source</span> given
          <var>workletInstance</var>'s <span>relevant global object</span> to perform the following
          steps:</p>

          <ol>
           <li>
            <p>If <var>pendingTasks</var> is not &#x2212;1, then:</p>

            <ol>
             <li><p>Set <var>pendingTasks</var> to &#x2212;1.</p></li>

             <li><p>Reject <var>promise</var> with <var>script</var>'s <span
             data-x="concept-script-error-to-rethrow">error to rethrow</span>.</p></li>
            </ol>
           </li>
          </ol>
         </li>

         <li><p>Abort these steps.</p></li>
        </ol>
       </li>

       <li>
        <p>If <var>addedSuccessfully</var> is false, then:</p>

        <ol>
         <li><p><span data-x="list append">Append</span> <var>moduleURLRecord</var> to
         <var>workletInstance</var>'s <span data-x="concept-Worklet-added-modules-list">added
         modules list</span>.</p></li>

         <li><p>Set <var>addedSuccessfully</var> to true.</p></li>
        </ol>
       </li>

       <li><p><span>Run a module script</span> given <var>script</var>.</p></li>

       <li>
        <p><span>Queue a global task</span> on the <span>networking task source</span> given
        <var>workletInstance</var>'s <span>relevant global object</span> to perform the following
        steps:</p>

        <ol>
         <li>
          <p>If <var>pendingTasks</var> is not &#x2212;1, then:</p>

          <ol>
           <li><p>Set <var>pendingTasks</var> to <var>pendingTasks</var> &#x2212; 1.</p></li>

           <li><p>If <var>pendingTasks</var> is 0, then resolve <var>promise</var>.</p></li>
          </ol>
         </li>
        </ol>
       </li>
      </ol>
     </li>
    </ol>
   </li>

   <li><p>Return <var>promise</var>.</p></li>
  </ol>

  </div>

  <h4 id="worklets-lifetime">The worklet's lifetime</h4>

  <p>The lifetime of a <code>Worklet</code> has no special considerations; it is tied to the object
  it belongs to, such as the <code>Window</code>.</p>

  <div w-nodev>

  <p>Each <code>Document</code> has a <dfn data-x="concept-Document-worklet-global-scopes">worklet
  global scopes</dfn>, which is a <span>set</span> of <code>WorkletGlobalScope</code>s, initially
  empty.</p>

  </div>

  <p>The lifetime of a <code>WorkletGlobalScope</code> is, at a minimum, tied to the
  <code>Document</code> whose <span data-x="concept-Document-worklet-global-scopes">worklet global
  scopes</span> contain it. In particular, <span data-x="destroy a document">destroying</span> the
  <code>Document</code> will <span data-x="terminate a worklet global scope">terminate</span> the
  corresponding <code>WorkletGlobalScope</code> and allow it to be garbage-collected.</p>

  <p>Additionally, user agents may, at any time, <span data-x="terminate a worklet global
  scope">terminate</span> a given <code>WorkletGlobalScope</code>, unless the specification defining
  the corresponding worklet type says otherwise. For example, they might terminate them if the
  <span>worklet agent</span>'s <span data-x="concept-agent-event-loop">event loop</span> has no
  <span data-x="concept-task">tasks</span> queued, or if the user agent has no pending operations
  planning to make use of the worklet, or if the user agent detects abnormal operations such as
  infinite loops or callbacks exceeding imposed time limits.</p>

  <p>Finally, specifications for specific worklet types can give more specific details on when to
  <span data-x="create a worklet global scope">create</span> <code>WorkletGlobalScope</code>s for a
  given worklet type. For example, they might create them during specific processes that call upon
  worklet code, as in the <a href="#worklets-example-registering">example</a>.</p>

  <h2 split-filename="webstorage" id="webstorage">Web storage</h2>

  <h3>Introduction</h3>

  <!-- NON-NORMATIVE SECTION -->

  <p>This specification introduces two related mechanisms, similar to HTTP session cookies, for
  storing name-value pairs on the client side. <ref>COOKIES</ref></p>

  <p>The first is designed for scenarios where the user is carrying out a single transaction, but
  could be carrying out multiple transactions in different windows at the same time.</p>

  <p>Cookies don't really handle this case well. For example, a user could be buying plane tickets
  in two different windows, using the same site. If the site used cookies to keep track of which
  ticket the user was buying, then as the user clicked from page to page in both windows, the ticket
  currently being purchased would "leak" from one window to the other, potentially causing the user
  to buy two tickets for the same flight without noticing.</p>

  <p>To address this, this specification introduces the <code
  data-x="dom-sessionStorage">sessionStorage</code> getter. Sites can add data to the session
  storage, and it will be accessible to any page from the same site opened in that window.</p>

  <div class="example">
   <p>For example, a page could have a checkbox that the user ticks to
   indicate that they want insurance:</p>

   <pre><code class="html">&lt;label>
 &lt;input type="checkbox" onchange="sessionStorage.insurance = checked ? 'true' : ''">
  I want insurance on this trip.
&lt;/label></code></pre>

   <p>A later page could then check, from script, whether the user had checked the checkbox or
   not:</p>

   <pre><code class="js">if (sessionStorage.insurance) { ... }</code></pre>

   <p>If the user had multiple windows opened on the site, each one would have its own individual
   copy of the session storage object.</p>
  </div>

  <!--

   sessionStorage.flightDeparture = 'OSL';
   sessionStorage.flightArrival = 'NYC';

   for (var i in forms[0].elements)
      sessionStorage["data_" + i.name] = i.value;

   // if we add storage of non-strings
   if (!sessionStorage[documents])
     sessionStorage[documents] = {};
   sessionStorage[documents][filename] = <document/>;

  -->

  <p>The second storage mechanism is designed for storage that spans multiple windows, and lasts
  beyond the current session. In particular, web applications might wish to store megabytes of user
  data, such as entire user-authored documents or a user's mailbox, on the client side for
  performance reasons.</p>

  <p>Again, cookies do not handle this case well, because they are transmitted with every
  request.</p>

  <p>The <code data-x="dom-localStorage">localStorage</code> getter is used to access a page's local
  storage area.</p>

  <div class="example">
   <p>The site at example.com can display a count of how many times the user has loaded its page by
   putting the following at the bottom of its page:</p>

   <pre><code class="html">&lt;p>
  You have viewed this page
  &lt;span id="count">an untold number of&lt;/span>
  time(s).
&lt;/p>
&lt;script>
  if (!localStorage.pageLoadCount)
    localStorage.pageLoadCount = 0;
  localStorage.pageLoadCount = parseInt(localStorage.pageLoadCount) + 1;
  document.getElementById('count').textContent = localStorage.pageLoadCount;
&lt;/script></code></pre>
  </div>

  <p>Each site has its own separate storage area.</p>

  <p class="warning">The <code data-x="dom-localStorage">localStorage</code> getter provides access
  to shared state. This specification does not define the interaction with other agent clusters
  in a multiprocess user agent, and authors are encouraged to assume that there is no locking
  mechanism. A site could, for instance, try to read the value of a key, increment its value, then
  write it back out, using the new value as a unique identifier for the session; if the site does
  this twice in two different browser windows at the same time, it might end up using the same
  "unique" identifier for both sessions, with potentially disastrous effects.</p>




  <h3 id="storage">The API</h3>

  <h4>The <code>Storage</code> interface</h4>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>Storage</dfn> {
  readonly attribute unsigned long <span data-x="dom-Storage-length">length</span>;
  DOMString? <span data-x="dom-Storage-key">key</span>(unsigned long index);
  getter DOMString? <span data-x="dom-Storage-getItem">getItem</span>(DOMString key);
  setter undefined <span data-x="dom-Storage-setItem">setItem</span>(DOMString key, DOMString value);
  deleter undefined <span data-x="dom-Storage-removeItem">removeItem</span>(DOMString key);
  undefined <span data-x="dom-Storage-clear">clear</span>();
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>storage</var>.<span subdfn data-x="dom-Storage-length">length</span></code></dt>
   <dd><p>Returns the number of key/value pairs.</p></dd>

   <dt><code data-x=""><var>storage</var>.<span subdfn data-x="dom-Storage-key">key</span> (<var>n</var>)</code></dt>
   <dd><p>Returns the name of the <var>n</var>th key, or null if <var>n</var> is greater than or
   equal to the number of key/value pairs.</p></dd>

   <dt><code data-x=""><var>value</var> = <var>storage</var>.<span subdfn data-x="dom-Storage-getItem">getItem</span> (<var>key</var>)</code></dt>
   <dt><code data-x=""><var>value</var> = <var>storage</var>[<var>key</var>]</code></dt>
   <dd><p>Returns the current value associated with the given <var>key</var>, or null if the given
   <var>key</var> does not exist.</p></dd>

   <dt><code data-x=""><var>storage</var>.<span subdfn data-x="dom-Storage-setItem">setItem</span> (<var>key</var>, <var>value</var>)</code></dt>
   <dt><code data-x=""><var>storage</var>[<var>key</var>] = <var>value</var></code></dt>
   <dd>
    <p>Sets the value of the pair identified by <var>key</var> to <var>value</var>, creating a new
    key/value pair if none existed for <var>key</var> previously.</p>

    <p>Throws a <span>"<code>QuotaExceededError</code>"</span> <code>DOMException</code> exception
    if the new value couldn't be set. (Setting could fail if, e.g., the user has disabled storage
    for the site, or if the quota has been exceeded.)</p>

    <p>Dispatches a <code data-x="event-storage">storage</code> event on <code>Window</code> objects
    holding an equivalent <code>Storage</code> object.</p>
   </dd>

   <dt><code data-x=""><var>storage</var>.<span subdfn data-x="dom-Storage-removeItem">removeItem</span> (<var>key</var>)</code></dt>
   <dt><code>delete</code> <var>storage</var>[<var>key</var>]</dt>
   <dd>
    <p>Removes the key/value pair with the given <var>key</var>, if a key/value pair with the given
    <var>key</var> exists.</p>

    <p>Dispatches a <code data-x="event-storage">storage</code> event on <code>Window</code> objects
    holding an equivalent <code>Storage</code> object.</p>
   </dd>

   <dt><code data-x=""><var>storage</var>.<span subdfn data-x="dom-Storage-clear">clear</span>()</code></dt>
   <dd>
    <p>Removes all key/value pairs, if there are any.</p>

    <p>Dispatches a <code data-x="event-storage">storage</code> event on <code>Window</code> objects
    holding an equivalent <code>Storage</code> object.</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>A <code>Storage</code> object has an associated:</p>

  <dl>
   <dt><dfn data-x="concept-Storage-map">map</dfn>
   <dd>A <span>storage proxy map</span>.</dd>

   <dt><dfn data-x="concept-Storage-type">type</dfn>
   <dd>"<code data-x="">local</code>" or "<code data-x="">session</code>".
  </dl>

  <p>To <dfn data-x="concept-Storage-reorder">reorder</dfn> a <code>Storage</code> object
  <var>storage</var>, reorder <var>storage</var>'s <span data-x="concept-Storage-map">map</span>'s
  <span data-x="map entry">entries</span> in an <span>implementation-defined</span> manner.</p>

  <p class="note">Unfortunate as it is, iteration order is not defined and can change upon most
  mutations.</p>

  <p>To <dfn data-x="concept-Storage-broadcast">broadcast</dfn> a <code>Storage</code> object
  <var>storage</var>, given a <var>key</var>, <var>oldValue</var>, and <var>newValue</var>, run
  these steps:</p>

  <ol>
   <li><p>Let <var>thisDocument</var> be <var>storage</var>'s <span>relevant global object</span>'s <span data-x="concept-document-window">associated <code>Document</code></span>.</p></li>

   <li><p>Let <var>url</var> be the <span data-x="concept-url-serializer">serialization</span> of
   <var>thisDocument</var>'s <span data-x="concept-document-url">URL</span>.</p></li>

   <li>
    <p>Let <var>remoteStorages</var> be all <code>Storage</code> objects excluding
    <var>storage</var> whose:</p>

    <ul>
     <li><span data-x="concept-Storage-type">type</span> is <var>storage</var>'s <span
     data-x="concept-Storage-type">type</span></li>

     <li><span>relevant settings object</span>'s <span>origin</span> is <span>same origin</span>
     with <var>storage</var>'s <span>relevant settings object</span>'s <span>origin</span>.</li>
    </ul>

    <p>and, if <span data-x="concept-Storage-type">type</span> is "<code data-x="">session</code>",
    whose <span>relevant settings object</span>'s <span data-x="concept-document-window">associated
    <code>Document</code></span>'s <span>node navigable</span>'s <span
    data-x="nav-traversable">traversable navigable</span> is <var>thisDocument</var>'s <span>node
    navigable</span>'s <span data-x="nav-traversable">traversable navigable</span>.</p>
   </li>

   <li>
    <p><span data-x="list iterate">For each</span> <var>remoteStorage</var> of
    <var>remoteStorages</var>: <span>queue a global task</span> on the <span>DOM manipulation task
    source</span> given <var>remoteStorage</var>'s <span>relevant global object</span> to <span
    data-x="concept-event-fire">fire an event</span> named <code
    data-x="event-storage">storage</code> at <var>remoteStorage</var>'s <span>relevant global
    object</span>, using <code>StorageEvent</code>, with <code
    data-x="dom-StorageEvent-key">key</code> initialized to <var>key</var>, <code
    data-x="dom-StorageEvent-oldValue">oldValue</code> initialized to <var>oldValue</var>, <code
    data-x="dom-StorageEvent-newValue">newValue</code> initialized to <var>newValue</var>, <code
    data-x="dom-StorageEvent-url">url</code> initialized to <var>url</var>, and <code
    data-x="dom-StorageEvent-storageArea">storageArea</code> initialized to
    <var>remoteStorage</var>.</p>
    <!-- This ignores the fact that Storage objects that are part of the same agent should not get
         their own tasks, much like the text that predates it. -->

    <p class="note">The <code>Document</code> object associated with the resulting <span
    data-x="concept-task">task</span> is not necessarily <span>fully active</span>, but events fired
    on such objects are ignored by the <span>event loop</span> until the <code>Document</code>
    becomes <span>fully active</span> again.</p>
   </li>
  </ol>

  <hr>

  <p>The <dfn attribute for="Storage"><code data-x="dom-Storage-length">length</code></dfn> getter
  steps are to return <span>this</span>'s <span data-x="concept-Storage-map">map</span>'s <span
  data-x="map size">size</span>.</p>

  <p>The <dfn method for="Storage"><code data-x="dom-Storage-key">key(<var>index</var>)</code></dfn>
  method steps are:</p>

  <ol>
   <li><p>If <var>index</var> is greater than or equal to <span>this</span>'s <span
   data-x="concept-Storage-map">map</span>'s <span data-x="map size">size</span>, then return
   null.</p></li>

   <li><p>Let <var>keys</var> be the result of running <span data-x="map get the keys">get the
   keys</span> on <span>this</span>'s <span data-x="concept-Storage-map">map</span>.

   <li><p>Return <var>keys</var>[<var>index</var>].</p></li>
  </ol>

  <p>The <span>supported property names</span> on a <code>Storage</code> object <var>storage</var>
  are the result of running <span data-x="map get the keys">get the keys</span> on
  <var>storage</var>'s <span data-x="concept-Storage-map">map</span>.</p>

  <p>The <dfn method for="Storage"><code
  data-x="dom-Storage-getItem">getItem(<var>key</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="concept-Storage-map">map</span>[<var>key</var>] does
   not <span data-x="map exists">exist</span>, then return null.</p></li>

   <li><p>Return <span>this</span>'s <span
   data-x="concept-Storage-map">map</span>[<var>key</var>].</p></li>
  </ol>

  <p>The <dfn method for="Storage"><code data-x="dom-Storage-setItem">setItem(<var>key</var>,
  <var>value</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>Let <var>oldValue</var> be null.</p></li>

   <li><p>Let <var>reorder</var> be true.</p></li>

   <li>
    <p>If <span>this</span>'s <span data-x="concept-Storage-map">map</span>[<var>key</var>]
    <span data-x="map exists">exists</span>:</p>

    <ol>
     <li><p>Set <var>oldValue</var> to <span>this</span>'s <span
     data-x="concept-Storage-map">map</span>[<var>key</var>].</p></li>

     <li><p>If <var>oldValue</var> <span>is</span> <var>value</var>, then return.</p></li>

     <li><p>Set <var>reorder</var> to false.</p></li>
    </ol>
   </li>

   <li><p>If <var>value</var> cannot be stored, then throw a
   <span>"<code>QuotaExceededError</code>"</span> <code>DOMException</code> exception.</p></li>

   <li><p><span data-x="map set">Set</span> <span>this</span>'s <span
   data-x="concept-Storage-map">map</span>[<var>key</var>] to <var>value</var>.</p></li>

   <li><p>If <var>reorder</var> is true, then <span data-x="concept-Storage-reorder">reorder</span>
   <span>this</span>.</p></li>

   <li><p><span data-x="concept-Storage-broadcast">Broadcast</span> <span>this</span> with
   <var>key</var>, <var>oldValue</var>, and <var>value</var>.</p></li>
  </ol>

  <p>The <dfn method for="Storage"><code
  data-x="dom-Storage-removeItem">removeItem(<var>key</var>)</code></dfn> method steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="concept-Storage-map">map</span>[<var>key</var>] does
   not <span data-x="map exists">exist</span>, then return.</p></li>

   <li><p>Set <var>oldValue</var> to <span>this</span>'s <span
   data-x="concept-Storage-map">map</span>[<var>key</var>].</p></li>

   <li><p><span data-x="map remove">Remove</span> <span>this</span>'s <span
   data-x="concept-Storage-map">map</span>[<var>key</var>].</p></li>

   <li><p><span data-x="concept-Storage-reorder">Reorder</span> <span>this</span>.</p></li>

   <li><p><span data-x="concept-Storage-broadcast">Broadcast</span> <span>this</span> with
   <var>key</var>, <var>oldValue</var>, and null.</p></li>
  </ol>

  <p>The <dfn method for="Storage"><code data-x="dom-Storage-clear">clear()</code></dfn> method
  steps are:</p>

  <ol>
   <li><p><span data-x="map clear">Clear</span> <span>this</span>'s <span
   data-x="concept-Storage-map">map</span>.</p></li>

   <li><p><span data-x="concept-Storage-broadcast">Broadcast</span> <span>this</span> with null,
   null, and null.</p></li>
  </ol>

  </div>


  <h4 id="the-sessionstorage-attribute">The <code data-x="dom-sessionStorage">sessionStorage</code> getter</h4>

  <pre><code class="idl">interface mixin <dfn interface>WindowSessionStorage</dfn> {
  readonly attribute <span>Storage</span> <span data-x="dom-sessionStorage">sessionStorage</span>;
};
<span>Window</span> includes <span>WindowSessionStorage</span>;</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-sessionStorage">sessionStorage</span></code></dt>
   <dd>
    <p>Returns the <code>Storage</code> object associated with that <var>window</var>'s origin's
    session storage area.</p>

    <p>Throws a <span>"<code>SecurityError</code>"</span> <code>DOMException</code> if the
    <code>Document</code>'s <span data-x="concept-document-origin">origin</span> is an <span
    data-x="concept-origin-opaque">opaque origin</span> or if the request violates a policy decision
    (e.g., if the user agent is configured to not allow the page to persist data).</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>A <code>Document</code> object has an associated <dfn>session storage holder</dfn>, which is
  null or a <code>Storage</code> object. It is initially null.</p>

  <p>
  <!--INSERT TRACKING-->
  The <dfn attribute for="WindowSessionStorage"><code
  data-x="dom-sessionStorage">sessionStorage</code></dfn> getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="concept-document-window">associated
   <code>Document</code></span>'s <span>session storage holder</span> is non-null, then return
   <span>this</span>'s <span data-x="concept-document-window">associated
   <code>Document</code></span>'s <span>session storage holder</span>.</p></li>

   <li><p>Let <var>map</var> be the result of running <span>obtain a session storage bottle
   map</span> with <span>this</span>'s <span>relevant settings object</span> and "<code
   data-x="">sessionStorage</code>".</p></li>

   <li><p>If <var>map</var> is failure, then throw a <span>"<code>SecurityError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Let <var>storage</var> be a new <code>Storage</code> object whose <span
   data-x="concept-Storage-map">map</span> is <var>map</var>.

   <li><p>Set <span>this</span>'s <span data-x="concept-document-window">associated
   <code>Document</code></span>'s <span>session storage holder</span> to
   <var>storage</var>.</p></li>

   <li><p>Return <var>storage</var>.</p></li>
  </ol>

  <p class="note">After <span>creating a new auxiliary browsing context and document</span>, the
  session storage <a href="#copy-session-storage">is copied</a> over.</p>

  </div>


  <h4 id="the-localstorage-attribute">The <code data-x="dom-localStorage">localStorage</code> getter</h4>

  <pre><code class="idl">interface mixin <dfn interface>WindowLocalStorage</dfn> {
  readonly attribute <span>Storage</span> <span data-x="dom-localStorage">localStorage</span>;
};
<span>Window</span> includes <span>WindowLocalStorage</span>;</code></pre>


  <dl class="domintro">
   <dt><code data-x=""><var>window</var>.<span subdfn data-x="dom-localStorage">localStorage</span></code></dt>
   <dd>
    <p>Returns the <code>Storage</code> object associated with <var>window</var>'s origin's local
    storage area.</p>

    <p>Throws a <span>"<code>SecurityError</code>"</span> <code>DOMException</code> if the
    <code>Document</code>'s <span data-x="concept-document-origin">origin</span> is an <span
    data-x="concept-origin-opaque">opaque origin</span> or if the request violates a policy decision
    (e.g., if the user agent is configured to not allow the page to persist data).</p>
   </dd>
  </dl>

  <div w-nodev>

  <p>A <code>Document</code> object has an associated <dfn>local storage holder</dfn>, which is null
  or a <code>Storage</code> object. It is initially null.</p>

  <p>
  <!--INSERT TRACKING-->
  The <dfn attribute for="WindowLocalStorage"><code
  data-x="dom-localStorage">localStorage</code></dfn> getter steps are:</p>

  <ol>
   <li><p>If <span>this</span>'s <span data-x="concept-document-window">associated
   <code>Document</code></span>'s <span>local storage holder</span> is non-null, then return
   <span>this</span>'s <span data-x="concept-document-window">associated
   <code>Document</code></span>'s <span>local storage holder</span>.</p></li>

   <li><p>Let <var>map</var> be the result of running <span>obtain a local storage bottle map</span>
   with <span>this</span>'s <span>relevant settings object</span> and "<code
   data-x="">localStorage</code>".</p></li>

   <li><p>If <var>map</var> is failure, then throw a <span>"<code>SecurityError</code>"</span>
   <code>DOMException</code>.</p></li>

   <li><p>Let <var>storage</var> be a new <code>Storage</code> object whose <span
   data-x="concept-Storage-map">map</span> is <var>map</var>.

   <li><p>Set <span>this</span>'s <span data-x="concept-document-window">associated
   <code>Document</code></span>'s <span>local storage holder</span> to <var>storage</var>.</p></li>

   <li><p>Return <var>storage</var>.</p></li>
  </ol>

  </div>


  <h4>The <code>StorageEvent</code> interface</h4>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>StorageEvent</dfn> : <span>Event</span> {
  constructor(DOMString type, optional <span>StorageEventInit</span> eventInitDict = {});

  readonly attribute DOMString? <span data-x="dom-StorageEvent-key">key</span>;
  readonly attribute DOMString? <span data-x="dom-StorageEvent-oldValue">oldValue</span>;
  readonly attribute DOMString? <span data-x="dom-StorageEvent-newValue">newValue</span>;
  readonly attribute USVString <span data-x="dom-StorageEvent-url">url</span>;
  readonly attribute <span>Storage</span>? <span data-x="dom-StorageEvent-storageArea">storageArea</span>;

  undefined <span data-x="dom-StorageEvent-initStorageEvent">initStorageEvent</span>(DOMString type, optional boolean bubbles = false, optional boolean cancelable = false, optional DOMString? key = null, optional DOMString? oldValue = null, optional DOMString? newValue = null, optional USVString url = "", optional Storage? storageArea = null);
};

dictionary <dfn dictionary>StorageEventInit</dfn> : <span>EventInit</span> {
  DOMString? key = null;
  DOMString? oldValue = null;
  DOMString? newValue = null;
  USVString url = "";
  <span>Storage</span>? storageArea = null;
};</code></pre>

  <dl class="domintro">
   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-StorageEvent-key">key</span></code></dt>
   <dd><p>Returns the key of the storage item being changed.</p></dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-StorageEvent-oldValue">oldValue</span></code></dt>
   <dd><p>Returns the old value of the key of the storage item whose value is being
   changed.</p></dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-StorageEvent-newValue">newValue</span></code></dt>
   <dd><p>Returns the new value of the key of the storage item whose value is being
   changed.</p></dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-StorageEvent-url">url</span></code></dt>
   <dd><p>Returns the <span data-x="concept-document-url">URL</span> of the document whose storage
   item changed.</p></dd>

   <dt><code data-x=""><var>event</var>.<span subdfn data-x="dom-StorageEvent-storageArea">storageArea</span></code></dt>
   <dd><p>Returns the <code>Storage</code> object that was affected.</p></dd>
  </dl>

  <div w-nodev>

  <p>The <dfn attribute for="StorageEvent"><code data-x="dom-StorageEvent-key">key</code></dfn>,
  <dfn attribute for="StorageEvent"><code data-x="dom-StorageEvent-oldValue">oldValue</code></dfn>,
  <dfn attribute for="StorageEvent"><code data-x="dom-StorageEvent-newValue">newValue</code></dfn>,
  <dfn attribute for="StorageEvent"><code data-x="dom-StorageEvent-url">url</code></dfn>, and <dfn
  attribute for="StorageEvent"><code data-x="dom-StorageEvent-storageArea">storageArea</code></dfn>
  attributes must return the values they were initialized to.</p>

  <!-- handwavy, file bugs if there are interop issues -->
  <p>The <dfn method for="StorageEvent"><code
  data-x="dom-StorageEvent-initStorageEvent">initStorageEvent(<var>type</var>, <var>bubbles</var>,
  <var>cancelable</var>, <var>key</var>, <var>oldValue</var>, <var>newValue</var>, <var>url</var>,
  <var>storageArea</var>)</code></dfn> method must initialize the event in a manner analogous to the
  similarly-named <code data-x="dom-Event-initEvent">initEvent()</code> method. <ref>DOM</ref></p>

  </div>


  <h3>Privacy</h3>

  <h4>User tracking</h4>

  <p>A third-party advertiser (or any entity capable of getting content distributed to multiple
  sites) could use a unique identifier stored in its local storage area to track a user across
  multiple sessions, building a profile of the user's interests to allow for highly targeted
  advertising. In conjunction with a site that is aware of the user's real identity (for example an
  e-commerce site that requires authenticated credentials), this could allow oppressive groups to
  target individuals with greater accuracy than in a world with purely anonymous web usage.</p>

  <p>There are a number of techniques that can be used to mitigate the risk of user tracking:</p>

  <dl>
   <dt>Blocking third-party storage</dt>
   <dd>
    <p>User agents may restrict access to the <code data-x="dom-localStorage">localStorage</code>
    objects to scripts originating at the domain of the <span data-x="nav-document">active
    document</span> of the <span>top-level traversable</span>, for instance denying access to the
    API for pages from other domains running in <code>iframe</code>s.</p>
   </dd>

   <dt>Expiring stored data</dt>
   <dd>
    <p>User agents may, possibly in a manner configured by the user, automatically delete stored
    data after a period of time.</p>

    <p>For example, a user agent could be configured to treat third-party local storage areas as
    session-only storage, deleting the data once the user had closed all the <span
    data-x="navigable">navigables</span> that could access it.</p>

    <p>This can restrict the ability of a site to track a user, as the site would then only be able
    to track the user across multiple sessions when they authenticate with the site itself (e.g. by
    making a purchase or logging in to a service).</p>

    <p>However, this also reduces the usefulness of the API as a long-term storage mechanism. It can
    also put the user's data at risk, if the user does not fully understand the implications of data
    expiration.</p>
   </dd>

   <dt>Treating persistent storage as cookies</dt>
   <dd>
    <p>If users attempt to protect their privacy by clearing cookies without also clearing data
    stored in the local storage area, sites can defeat those attempts by using the two features as
    redundant backup for each other. User agents should present the interfaces for clearing these in
    a way that helps users to understand this possibility and enables them to delete data in all
    persistent storage features simultaneously. <ref>COOKIES</ref></p>
   </dd>

   <dt>Site-specific safelisting of access to local storage areas</dt> <dd>
    <p>User agents may allow sites to access session storage areas in
    an unrestricted manner, but require the user to authorize access
    to local storage areas.</p>
   </dd>

   <dt>Origin-tracking of stored data</dt>
   <dd>
    <p>User agents may record the <span data-x="origin">origins</span> of sites that contained
    content from third-party origins that caused data to be stored.</p>

    <p>If this information is then used to present the view of data currently in persistent storage,
    it would allow the user to make informed decisions about which parts of the persistent storage
    to prune. Combined with a blocklist ("delete this data and prevent this domain from ever storing
    data again"), the user can restrict the use of persistent storage to sites that they trust.</p>
   </dd>

   <dt>Shared blocklists</dt>
   <dd>
    <p>User agents may allow users to share their persistent storage domain blocklists.</p>

    <p>This would allow communities to act together to protect their privacy.</p>
   </dd>
  </dl>

  <p>While these suggestions prevent trivial use of this API for user tracking, they do not block it
  altogether. Within a single domain, a site can continue to track the user during a session, and
  can then pass all this information to the third party along with any identifying information
  (names, credit card numbers, addresses) obtained by the site. If a third party cooperates with
  multiple sites to obtain such information, a profile can still be created.</p>

  <p>However, user tracking is to some extent possible even with no cooperation from the user agent
  whatsoever, for instance by using session identifiers in URLs, a technique already commonly used
  for innocuous purposes but easily repurposed for user tracking (even retroactively). This
  information can then be shared with other sites, using visitors' IP addresses and other
  user-specific data (e.g. user-agent headers and configuration settings) to combine separate
  sessions into coherent user profiles.</p>

  <div w-nodev>

  <h4>Sensitivity of data</h4>

  <p>User agents should treat persistently stored data as potentially sensitive; it's quite possible
  for emails, calendar appointments, health records, or other confidential documents to be stored
  in this mechanism.</p>

  <p>To this end, user agents should ensure that when deleting data, it is promptly deleted from the
  underlying storage.</p>

  </div>


  <h3 id="security-storage">Security</h3>

  <h4>DNS spoofing attacks</h4>

  <p>Because of the potential for DNS spoofing attacks, one cannot guarantee that a host claiming to
  be in a certain domain really is from that domain. To mitigate this, pages can use TLS. Pages
  using TLS can be sure that only the user, software working on behalf of the user, and other pages
  using TLS that have certificates identifying them as being from the same domain, can access their
  storage areas.</p>


  <h4>Cross-directory attacks</h4>

  <p>Different authors sharing one host name, for example users hosting content on the now defunct
  <code data-x="">geocities.com</code>, all share one local storage object. There is no feature to
  restrict the access by pathname. Authors on shared hosts are therefore urged to avoid using these
  features, as it would be trivial for other authors to read the data and overwrite it.</p>

  <p class="note">Even if a path-restriction feature was made available, the usual DOM scripting
  security model would make it trivial to bypass this protection and access the data from any
  path.</p>


  <div w-nodev>

  <h4>Implementation risks</h4>

  <p>The two primary risks when implementing these persistent storage features are letting hostile
  sites read information from other domains, and letting hostile sites write information that is
  then read from other domains.</p>

  <p>Letting third-party sites read data that is not supposed to be read from their domain causes
  <em>information leakage</em>. For example, a user's shopping wishlist on one domain could be used
  by another domain for targeted advertising; or a user's work-in-progress confidential documents
  stored by a word-processing site could be examined by the site of a competing company.</p>

  <p>Letting third-party sites write data to the persistent storage of other domains can result in
  <em>information spoofing</em>, which is equally dangerous. For example, a hostile site could add
  items to a user's wishlist; or a hostile site could set a user's session identifier to a known ID
  that the hostile site can then use to track the user's actions on the victim site.</p>

  <p>Thus, strictly following the <span>origin</span> model described in this specification is
  important for user security.</p>

  </div>



  <h2 split-filename="syntax" id="syntax"><dfn>The HTML syntax</dfn></h2>

  <p class="note">This section only describes the rules for resources labeled with an <span>HTML
  MIME type</span>. Rules for XML resources are discussed in the section below entitled "<span>The
  XML syntax</span>".</p>


  <h3 id="writing">Writing HTML documents</h3>

  <div w-nodev>

  <p><i>This section only applies to documents, authoring tools, and markup generators. In
  particular, it does not apply to conformance checkers; conformance checkers must use the
  requirements given in the next section ("parsing HTML documents").</i></p>

  </div>

  <p>Documents must consist of the following parts, in the given
  order:</p>

  <ol>
   <li>Optionally, a single U+FEFF BYTE ORDER MARK (BOM) character.</li>

   <li>Any number of <span data-x="syntax-comments">comments</span> and <span>ASCII
   whitespace</span>.</li>

   <li>A <span data-x="syntax-doctype">DOCTYPE</span>.

   <li>Any number of <span data-x="syntax-comments">comments</span> and <span>ASCII
   whitespace</span>.</li>

   <li>The <span>document element</span>, in the form of an <code>html</code> <span
   data-x="syntax-elements">element</span>.</li>

   <li>Any number of <span data-x="syntax-comments">comments</span> and <span>ASCII
   whitespace</span>.</li>
  </ol>

  <p>The various types of content mentioned above are described in the next few sections.</p>

  <p>In addition, there are some restrictions on how <span data-x="character encoding
  declaration">character encoding declarations</span> are to be serialized, as discussed in the
  section on that topic.</p>

  <div class="note">

   <p><span>ASCII whitespace</span> before the <code>html</code> element, at the start of the
   <code>html</code> element and before the <code>head</code> element, will be dropped when the
   document is parsed; <span>ASCII whitespace</span> <em>after</em> the <code>html</code> element
   will be parsed as if it were at the end of the <code>body</code> element. Thus, <span>ASCII
   whitespace</span> around the <span>document element</span> does not round-trip.</p>

   <p>It is suggested that newlines be inserted after the DOCTYPE, after any comments that are
   before the document element, after the <code>html</code> element's start tag (if it is not <span
   data-x="syntax-tag-omission">omitted</span>), and after any comments that are inside the
   <code>html</code> element but before the <code>head</code> element.</p>

  </div>

  <p>Many strings in the HTML syntax (e.g. the names of elements and their attributes) are
  case-insensitive, but only for <span data-x="ASCII upper alpha">ASCII upper alphas</span> and
  <span data-x="ASCII lower alpha">ASCII lower alphas</span>. For convenience, in this section this
  is just referred to as "case-insensitive".</p>


  <h4>The DOCTYPE</h4>

  <p>A <dfn data-x="syntax-doctype">DOCTYPE</dfn> is a <!-- mostly useless but nonetheless -->
  required preamble.</p>

  <p class="note">DOCTYPEs are required for legacy reasons. When omitted, browsers tend to use a
  different rendering mode that is incompatible with some specifications. Including the DOCTYPE in a
  document ensures that the browser makes a best-effort attempt at following the relevant
  specifications.</p>

  <p>A DOCTYPE must consist of the following components, in this order:</p>

  <ol class="brief">
   <li>A string that is an <span>ASCII case-insensitive</span> match for the string "<code data-x="">&lt;!DOCTYPE</code>".</li>
   <li>One or more <span>ASCII whitespace</span>.</li>
   <li>A string that is an <span>ASCII case-insensitive</span> match for the string "<code data-x="">html</code>".</li>
   <li>Optionally, a <span>DOCTYPE legacy string</span>.</li>
   <li>Zero or more <span>ASCII whitespace</span>.</li>
   <li>A U+003E GREATER-THAN SIGN character (&gt;).</li>
  </ol>

  <p class="note">In other words, <code data-x="">&lt;!DOCTYPE html></code>, case-insensitively.</p>

  <hr>

  <p>For the purposes of HTML generators that cannot output HTML markup with the short DOCTYPE
  "<code data-x="">&lt;!DOCTYPE html></code>", a <dfn>DOCTYPE legacy string</dfn> may be inserted
  into the DOCTYPE (in the position defined above). This string must consist of:</p>

  <ol class="brief">
   <li>One or more <span>ASCII whitespace</span>.</li>
   <li>A string that is an <span>ASCII case-insensitive</span> match for the string "<code data-x="">SYSTEM</code>".</li>
   <li>One or more <span>ASCII whitespace</span>.</li>
   <li>A U+0022 QUOTATION MARK or U+0027 APOSTROPHE character (the <i>quote mark</i>).</li>
   <li>The literal string "<code>about:legacy-compat</code>".</li>
   <li>A matching U+0022 QUOTATION MARK or U+0027 APOSTROPHE character (i.e. the same character as in the earlier step labeled <i>quote mark</i>).</li>
  </ol>

  <p class="note">In other words, <code data-x="">&lt;!DOCTYPE html SYSTEM "about:legacy-compat"></code> or
  <code data-x="">&lt;!DOCTYPE html SYSTEM 'about:legacy-compat'></code>, case-insensitively except for the
  part in single or double quotes.</p>

  <p>The <span>DOCTYPE legacy string</span> should not be used unless the document is generated from
  a system that cannot output the shorter string.</p>



  <h4>Elements</h4>

  <p>There are six different kinds of <dfn data-x="syntax-elements">elements</dfn>: <span>void
  elements</span>, <span>the <code data-x="">template</code> element</span>, <span>raw text
  elements</span>, <span>escapable raw text elements</span>, <span>foreign elements</span>, and
  <span>normal elements</span>.</p>

  <dl>
   <dt><dfn>Void elements</dfn></dt>

   <dd><code>area</code>, <code>base</code>, <code>br</code>, <code>col</code>, <code>embed</code>,
   <code>hr</code>, <code>img</code>, <code>input</code>, <code>link</code>, <code>meta</code>,
   <code>source</code>, <code>track</code>, <code>wbr</code></dd>
   <!-- non-conforming elements treated as void by the parser are not listed here: basefont,
   bgsound, frame, param, keygen -->

   <dt><dfn>The <code data-x="">template</code> element</dfn></dt>
   <dd><code>template</code></dd>

   <dt><dfn>Raw text elements</dfn></dt>

   <dd><code>script</code>, <code>style</code></dd> <!-- iframe and noscript don't count as raw text
   for syntax purposes -->

   <dt><dfn>Escapable raw text elements</dfn></dt>

   <dd><code>textarea</code>, <code>title</code></dd>

   <dt><dfn>Foreign elements</dfn></dt>

   <dd>Elements from the <span>MathML namespace</span> and the <span>SVG namespace</span>.</dd>

   <dt><dfn>Normal elements</dfn></dt>

   <dd>All other allowed <span>HTML elements</span> are normal elements.</dd>
  </dl>

  <p><dfn data-x="syntax-tags">Tags</dfn> are used to delimit the start and end of elements in the
  markup. <span data-x="raw text elements">Raw text</span>, <span data-x="escapable raw text
  elements">escapable raw text</span>, and <span data-x="normal elements">normal</span> elements have
  a <span data-x="syntax-start-tag">start tag</span> to indicate where they begin, and an <span
  data-x="syntax-end-tag">end tag</span> to indicate where they end. The start and end tags of
  certain <span>normal elements</span> can be <span data-x="syntax-tag-omission">omitted</span>, as
  described below in the section on <span data-x="syntax-tag-omission">optional tags</span>. Those
  that cannot be omitted must not be omitted. <span>Void elements</span> only have a start tag; end
  tags must not be specified for <span>void elements</span>. <span>Foreign elements</span> must
  either have a start tag and an end tag, or a start tag that is marked as self-closing, in which
  case they must not have an end tag.</p>

  <p>The <span data-x="concept-html-contents">contents</span> of the element must be placed between
  just after the start tag (which <span data-x="syntax-tag-omission">might be implied, in certain
  cases</span>) and just before the end tag (which again, <span data-x="syntax-tag-omission">might be
  implied in certain cases</span>). The exact allowed contents of each individual element depend on
  the <span data-x="content models">content model</span> of that element, as described earlier in
  this specification. Elements must not contain content that their content model disallows. In
  addition to the restrictions placed on the contents by those content models, however, the five
  types of elements have additional <em>syntactic</em> requirements.</p>

  <p><span>Void elements</span> can't have any contents (since there's no end tag, no content can be
  put between the start tag and the end tag).</p>

  <p id="template-syntax"><span>The <code data-x="">template</code> element</span> can have
  <span>template contents</span>, but such <span>template contents</span> are not children of the
  <code>template</code> element itself. Instead, they are stored in a <code>DocumentFragment</code>
  associated with a different <code>Document</code> — without a <span>browsing context</span> — so
  as to avoid the <code>template</code> contents interfering with the main <code>Document</code>.
  The markup for the <span>template contents</span> of a <code>template</code> element is placed
  just after the <code>template</code> element's start tag and just before <code>template</code>
  element's end tag (as with other elements), and may consist of any <span
  data-x="syntax-text">text</span>, <span data-x="syntax-charref">character references</span>, <span
  data-x="syntax-elements">elements</span>, and <span data-x="syntax-comments">comments</span>, but
  the text must not contain the character U+003C LESS-THAN SIGN (&lt;) or an <span
  data-x="syntax-ambiguous-ampersand">ambiguous ampersand</span>.</p>

  <p><span>Raw text elements</span> can have <span data-x="syntax-text">text</span>, though it has <a
  href="#cdata-rcdata-restrictions">restrictions</a> described below.</p>

  <p><span>Escapable raw text elements</span> can have <span data-x="syntax-text">text</span> and
  <span data-x="syntax-charref">character references</span>, but the text must not contain an <span
  data-x="syntax-ambiguous-ampersand">ambiguous ampersand</span>. There are also <a
  href="#cdata-rcdata-restrictions">further restrictions</a> described below.</p>

  <p><span>Foreign elements</span> whose start tag is marked as self-closing can't have any contents
  (since, again, as there's no end tag, no content can be put between the start tag and the end
  tag). <span>Foreign elements</span> whose start tag is <em>not</em> marked as self-closing can
  have <span data-x="syntax-text">text</span>, <span data-x="syntax-charref">character
  references</span>, <span data-x="syntax-cdata">CDATA sections</span>, other <span
  data-x="syntax-elements">elements</span>, and <span data-x="syntax-comments">comments</span>, but
  the text must not contain the character U+003C LESS-THAN SIGN (&lt;) or an <span
  data-x="syntax-ambiguous-ampersand">ambiguous ampersand</span>.</p>

  <div class="note">

   <p>The HTML syntax does not support namespace declarations, even in <span>foreign
   elements</span>.</p>

   <p>For instance, consider the following HTML fragment:</p>

   <pre><code class="html">&lt;p>
 &lt;svg>
  &lt;metadata>
   &lt;!-- this is invalid -->
   &lt;cdr:license xmlns:cdr="https://www.example.com/cdr/metadata" name="MIT"/>
  &lt;/metadata>
 &lt;/svg>
&lt;/p></code></pre>

   <p>The innermost element, <code data-x="">cdr:license</code>, is actually in the SVG namespace, as
   the "<code data-x="">xmlns:cdr</code>" attribute has no effect (unlike in XML). In fact, as the
   comment in the fragment above says, the fragment is actually non-conforming. This is because
   <cite>SVG 2</cite> does not define any elements called "<code data-x="">cdr:license</code>" in
   the SVG namespace.</p>

  </div>

  <p><span>Normal elements</span> can have <span data-x="syntax-text">text</span>, <span
  data-x="syntax-charref">character references</span>, other <span
  data-x="syntax-elements">elements</span>, and <span data-x="syntax-comments">comments</span>, but
  the text must not contain the character U+003C LESS-THAN SIGN (&lt;) or an <span
  data-x="syntax-ambiguous-ampersand">ambiguous ampersand</span>. Some <span>normal elements</span>
  also have <a href="#element-restrictions">yet more restrictions</a> on what content they are
  allowed to hold, beyond the restrictions imposed by the content model and those described in this
  paragraph. Those restrictions are described below.</p>

  <p>Tags contain a <dfn data-x="syntax-tag-name">tag name</dfn>, giving the element's name. HTML
  elements all have names that only use <span data-x="ASCII alphanumeric">ASCII
  alphanumerics</span>. In the HTML syntax, tag names, even those for <span>foreign elements</span>,
  may be written with any mix of lower- and uppercase letters that, when converted to all-lowercase,
  matches the element's tag name; tag names are case-insensitive.</p>


  <h5>Start tags</h5>

  <p><dfn data-x="syntax-start-tag">Start tags</dfn> must have the following format:</p>

  <ol>
   <li>The first character of a start tag must be a U+003C LESS-THAN SIGN character (&lt;).</li>

   <li>The next few characters of a start tag must be the element's <span
   data-x="syntax-tag-name">tag name</span>.</li>

   <li>If there are to be any attributes in the next step, there must first be one or more
   <span>ASCII whitespace</span>.</li>

   <li>Then, the start tag may have a number of attributes, the <span
   data-x="syntax-attributes">syntax for which</span> is described below. Attributes must be
   separated from each other by one or more <span>ASCII whitespace</span>.</li>

   <li>After the attributes, or after the <span data-x="syntax-tag-name">tag name</span> if there
   are no attributes, there may be one or more <span>ASCII whitespace</span>. (Some attributes are
   required to be followed by a space. See the <span data-x="syntax-attributes">attributes
   section</span> below.)</li>

   <li>Then, if the element is one of the <span>void elements</span>, or if the element is a <span
   data-x="foreign elements">foreign element</span>, then there may be a single U+002F SOLIDUS
   character (/), which on <span>foreign elements</span> marks the start tag as self-closing. On
   <span>void elements</span>, it does not mark the start tag as self-closing but instead is
   unnecessary and has no effect of any kind. For such void elements, it should be used only with
   caution — especially since, if directly preceded by an <a href="#unquoted">unquoted attribute
   value</a>, it becomes part of the attribute value rather than being discarded by the
   parser.</li>

   <li>Finally, start tags must be closed by a U+003E GREATER-THAN SIGN character (&gt;).</li>
  </ol>


  <h5>End tags</h5>

  <p><dfn data-x="syntax-end-tag">End tags</dfn> must have the following format:</p>

  <ol>
   <li>The first character of an end tag must be a U+003C LESS-THAN SIGN character (&lt;).</li>

   <li>The second character of an end tag must be a U+002F SOLIDUS character (/).</li>

   <li>The next few characters of an end tag must be the element's <span data-x="syntax-tag-name">tag
   name</span>.</li>

   <li>After the tag name, there may be one or more <span>ASCII whitespace</span>.</li>

   <li>Finally, end tags must be closed by a U+003E GREATER-THAN SIGN character (&gt;).</li>
  </ol>


  <h5>Attributes</h5>

  <p><dfn data-x="syntax-attributes">Attributes</dfn> for an element are expressed inside the
  element's start tag.</p>

  <p>Attributes have a name and a value. <dfn data-x="syntax-attribute-name">Attribute names</dfn>
  must consist of one or more characters other than <span data-x="control">controls</span>,
  U+0020 SPACE, U+0022 ("), U+0027 ('), U+003E (&gt;), U+002F (/), U+003D (=), and <span
  data-x="noncharacter">noncharacters</span>. In the HTML syntax, attribute names, even those for
  <span>foreign elements</span>, may be written with any mix of <span data-x="ASCII lower
  alpha">ASCII lower</span> and <span data-x="ASCII upper alpha">ASCII upper alphas</span>.</p>

  <p><dfn data-x="syntax-attribute-value">Attribute values</dfn> are a mixture of <span
  data-x="syntax-text">text</span> and <span data-x="syntax-charref">character references</span>,
  except with the additional restriction that the text cannot contain an <span
  data-x="syntax-ambiguous-ampersand">ambiguous ampersand</span>.</p>

  <p>Attributes can be specified in four different ways:</p>

  <dl>
   <dt>Empty attribute syntax</dt>

   <dd>
    <p>Just the <span data-x="syntax-attribute-name">attribute name</span>. The value is implicitly
    the empty string.</p>

    <div class="example">

     <p>In the following example, the <code data-x="attr-fe-disabled">disabled</code> attribute is
     given with the empty attribute syntax:</p>

     <pre><code class="html">&lt;input <em>disabled</em>&gt;</code></pre>

    </div>

    <p>If an attribute using the empty attribute syntax is to be followed by another attribute, then
    there must be <span>ASCII whitespace</span> separating the two.</p>
   </dd>

   <dt id="unquoted">Unquoted attribute value syntax</dt>

   <dd>
    <p>The <span data-x="syntax-attribute-name">attribute name</span>, followed by zero or more
    <span>ASCII whitespace</span>, followed by a single U+003D EQUALS SIGN character, followed by
    zero or more <span>ASCII whitespace</span>, followed by the <span
    data-x="syntax-attribute-value">attribute value</span>, which, in addition to the requirements
    given above for attribute values, must not contain any literal <span>ASCII whitespace</span>,
    any U+0022 QUOTATION MARK characters (&#x22;), U+0027 APOSTROPHE characters (&#x27;), U+003D
    EQUALS SIGN characters (=), U+003C LESS-THAN SIGN characters (&lt;), U+003E GREATER-THAN SIGN
    characters (&gt;), or U+0060 GRAVE ACCENT characters (`), and must not be the empty string.</p>

    <!-- The ` character is in this list on a temporary basis, waiting for IE to fix its parsing bug
         whereby it treats ` as an attribute value delimiter. Otherwise, escaping software that
         tries to be clever and not use quotes when it doesn't need to could be tricked by an
         attacker.

         Posit a site that allows the user to input text that is used verbatim in two attributes,
         such that the user can set the first attribute's value to:

            `

         ...and the second to:

            ` onload='...payload...' end=x

         ...with the assumption that the site is going to not quote the first one, and quote the
         second one with double quotes:

            <body data-x=` class="` onload='...payload...' end=x">

         In IE, this is treated as:

            <body data-x=' class="'
                  onload='...payload...'
                  end='x"'>

    -->


    <div class="example">

     <p>In the following example, the <code data-x="attr-input-value">value</code> attribute is given
     with the unquoted attribute value syntax:</p>

     <pre><code class="html">&lt;input <em>value=yes</em>&gt;</code></pre>

    </div>

    <p>If an attribute using the unquoted attribute syntax is to be followed by another attribute or
    by the optional U+002F SOLIDUS character (/) allowed in step 6 of the <span
    data-x="syntax-start-tag">start tag</span> syntax above, then there must be <span>ASCII
    whitespace</span> separating the two.</p>
   </dd>

   <dt>Single-quoted attribute value syntax</dt>

   <dd>
    <p>The <span data-x="syntax-attribute-name">attribute name</span>, followed by zero or more
    <span>ASCII whitespace</span>, followed by a single U+003D EQUALS SIGN character, followed by
    zero or more <span>ASCII whitespace</span>, followed by a single U+0027 APOSTROPHE character
    ('), followed by the <span data-x="syntax-attribute-value">attribute value</span>, which, in
    addition to the requirements given above for attribute values, must not contain any literal
    U+0027 APOSTROPHE characters ('), and finally followed by a second single U+0027 APOSTROPHE
    character (').</p>

    <div class="example">

     <p>In the following example, the <code data-x="attr-input-type">type</code> attribute is given
     with the single-quoted attribute value syntax:</p>

     <pre><code class="html">&lt;input <em>type='checkbox'</em>&gt;</code></pre>

    </div>

    <p>If an attribute using the single-quoted attribute syntax is to be followed by another
    attribute, then there must be <span>ASCII whitespace</span> separating the two.</p>
   </dd>

   <dt>Double-quoted attribute value syntax</dt>

   <dd>
    <p>The <span data-x="syntax-attribute-name">attribute name</span>, followed by zero or more
    <span>ASCII whitespace</span>, followed by a single U+003D EQUALS SIGN character, followed by
    zero or more <span>ASCII whitespace</span>, followed by a single U+0022 QUOTATION MARK character
    ("), followed by the <span data-x="syntax-attribute-value">attribute value</span>, which, in
    addition to the requirements given above for attribute values, must not contain any literal
    U+0022 QUOTATION MARK characters ("), and finally followed by a second single U+0022 QUOTATION
    MARK character (").</p>

    <div class="example">

     <p>In the following example, the <code data-x="attr-fe-name">name</code> attribute is given with
     the double-quoted attribute value syntax:</p>

     <pre><code class="html">&lt;input <em>name="be evil"</em>&gt;</code></pre>

    </div>

    <p>If an attribute using the double-quoted attribute syntax is to be followed by another
    attribute, then there must be <span>ASCII whitespace</span> separating the two.</p>
   </dd>
  </dl>

  <p>There must never be two or more attributes on the same start tag whose names are an <span>ASCII
  case-insensitive</span> match for each other.</p>

  <hr>

  <p>When a <span data-x="foreign elements">foreign element</span> has one of the namespaced
  attributes given by the local name and namespace of the first and second cells of a row from the
  following table, it must be written using the name given by the third cell from the same row.</p>

  <table>
   <thead>
    <tr> <th> Local name <th> Namespace <th> Attribute name
   <tbody>
    <tr> <td> <code data-x="">actuate</code> <td> <span>XLink namespace</span> <td> <code data-x="">xlink:actuate</code>
    <tr> <td> <code data-x="">arcrole</code> <td> <span>XLink namespace</span> <td> <code data-x="">xlink:arcrole</code>
    <tr> <td> <code data-x="">href</code> <td> <span>XLink namespace</span> <td> <code data-x="">xlink:href</code>
    <tr> <td> <code data-x="">role</code> <td> <span>XLink namespace</span> <td> <code data-x="">xlink:role</code>
    <tr> <td> <code data-x="">show</code> <td> <span>XLink namespace</span> <td> <code data-x="">xlink:show</code>
    <tr> <td> <code data-x="">title</code> <td> <span>XLink namespace</span> <td> <code data-x="">xlink:title</code>
    <tr> <td> <code data-x="">type</code> <td> <span>XLink namespace</span> <td> <code data-x="">xlink:type</code>
    <tr> <td> <code data-x="">lang</code> <td> <span>XML namespace</span> <td> <code data-x="">xml:lang</code>
    <tr> <td> <code data-x="">space</code> <td> <span>XML namespace</span> <td> <code data-x="">xml:space</code>
    <tr> <td> <code data-x="">xmlns</code> <td> <span>XMLNS namespace</span> <td> <code data-x="">xmlns</code>
    <tr> <td> <code data-x="">xlink</code> <td> <span>XMLNS namespace</span> <td> <code data-x="">xmlns:xlink</code>
  </table>

  <p>No other namespaced attribute can be expressed in <span>the HTML syntax</span>.</p>

  <p class="note">Whether the attributes in the table above are conforming or not is defined by
  other specifications (e.g. <cite>SVG 2</cite> and <cite>MathML</cite>); this section only
  describes the syntax rules if the attributes are serialized using the HTML syntax.</p>


  <h5>Optional tags</h5>

  <p>Certain tags can be <dfn data-x="syntax-tag-omission">omitted</dfn>.</p>

  <p class="note">Omitting an element's <span data-x="syntax-start-tag">start tag</span> in the
  situations described below does not mean the element is not present; it is implied, but it is
  still there. For example, an HTML document always has a root <code>html</code> element, even if
  the string <code data-x="">&lt;html></code> doesn't appear anywhere in the markup.</p>

  <!-- <html> -->
  <p>An <code>html</code> element's <span data-x="syntax-start-tag">start tag</span> may be omitted
  if the first thing inside the <code>html</code> element is not a <span
  data-x="syntax-comments">comment</span>.</p>

  <div class="example">

   <p>For example, in the following case it's ok to remove the "<code data-x="">&lt;html></code>"
   tag:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
<strong>&lt;html></strong>
  &lt;head>
    &lt;title>Hello&lt;/title>
  &lt;/head>
  &lt;body>
    &lt;p>Welcome to this example.&lt;/p>
  &lt;/body>
&lt;/html></code></pre>

   <p>Doing so would make the document look like this:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>

  &lt;head>
    &lt;title>Hello&lt;/title>
  &lt;/head>
  &lt;body>
    &lt;p>Welcome to this example.&lt;/p>
  &lt;/body>
&lt;/html></code></pre>

   <p>This has the exact same DOM. In particular, note that whitespace around the <span>document
   element</span> is ignored by the parser. The following example would also have the exact same
   DOM:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>&lt;head>
    &lt;title>Hello&lt;/title>
  &lt;/head>
  &lt;body>
    &lt;p>Welcome to this example.&lt;/p>
  &lt;/body>
&lt;/html></code></pre>

   <p>However, in the following example, removing the start tag moves the comment to before the
   <code>html</code> element:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html>
  <strong>&lt;!-- where is this comment in the DOM? --></strong>
  &lt;head>
    &lt;title>Hello&lt;/title>
  &lt;/head>
  &lt;body>
    &lt;p>Welcome to this example.&lt;/p>
  &lt;/body>
&lt;/html></code></pre>

   <p>With the tag removed, the document actually turns into the same as this:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;!-- where is this comment in the DOM? -->
<small>&lt;html></small>
  &lt;head>
    &lt;title>Hello&lt;/title>
  &lt;/head>
  &lt;body>
    &lt;p>Welcome to this example.&lt;/p>
  &lt;/body>
&lt;/html></code></pre>

   <p>This is why the tag can only be removed if it is not followed by a comment: removing the tag
   when there is a comment there changes the document's resulting parse tree. Of course, if the
   position of the comment does not matter, then the tag can be omitted, as if the comment had been
   moved to before the start tag in the first place.</p>

  </div>

  <!-- </html> -->
  <p>An <code>html</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted if
  the <code>html</code> element is not immediately followed by a <span
  data-x="syntax-comments">comment</span>.</p>

  <!-- <head> -->
  <p>A <code>head</code> element's <span data-x="syntax-start-tag">start tag</span> may be omitted if
  the element is empty, or if the first thing inside the <code>head</code> element is an
  element.</p>

  <!-- </head> -->
  <p>A <code>head</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted if
  the <code>head</code> element is not immediately followed by <span>ASCII whitespace</span> or a
  <span data-x="syntax-comments">comment</span>.</p>

  <!-- <body> -->
  <p>A <code>body</code> element's <span data-x="syntax-start-tag">start tag</span> may be omitted
  if the element is empty, or if the first thing inside the <code>body</code> element is not
  <span>ASCII whitespace</span> or a <span data-x="syntax-comments">comment</span>, except if the
  first thing inside the <code>body</code> element is a <code>meta</code>, <code>noscript</code>,
  <code>link</code>, <code>script</code>, <code>style</code>, or <code>template</code> element.
  <!-- Note that even if the </head> end tag is present, the parser makes <style> and <script>
  elements between </head> and <body> end up in the <head> instead of implying the <body> --></p>

  <!-- </body> -->
  <p>A <code>body</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted if the
  <code>body</code> element is not immediately followed by a <span
  data-x="syntax-comments">comment</span>.</p>

  <div class="example">

   <p>Note that in the example above, the <code>head</code> element start and end tags, and the
   <code>body</code> element start tag, can't be omitted, because they are surrounded by
   whitespace:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;html><strong>
  </strong>&lt;head><strong>
    </strong>&lt;title>Hello&lt;/title><strong>
  </strong>&lt;/head><strong>
  </strong>&lt;body><strong>
    </strong>&lt;p>Welcome to this example.&lt;/p>
  &lt;/body>
&lt;/html></code></pre>

   <p>(The <code>body</code> and <code>html</code> element end tags could be omitted without
   trouble; any spaces after those get parsed into the <code>body</code> element anyway.)</p>

   <p>Usually, however, whitespace isn't an issue. If we first remove the whitespace we don't care
   about:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>&lt;html>&lt;head>&lt;title>Hello&lt;/title>&lt;/head>&lt;body>&lt;p>Welcome to this example.&lt;/p>&lt;/body>&lt;/html></code></pre>

   <p>Then we can omit a number of tags without affecting the DOM:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>&lt;title>Hello&lt;/title>&lt;p>Welcome to this example.&lt;/p></code></pre>

   <p>At that point, we can also add some whitespace back:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
&lt;title>Hello&lt;/title>
&lt;p>Welcome to this example.&lt;/p></code></pre>

   <p>This would be equivalent to this document, with the omitted tags shown in their
   parser-implied positions; the only whitespace text node that results from this is the newline at
   the end of the <code>head</code> element:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>
<small>&lt;html>&lt;head></small>&lt;title>Hello&lt;/title>
<small>&lt;/head>&lt;body></small>&lt;p>Welcome to this example.&lt;/p><small>&lt;/body>&lt;/html></small></code></pre>

  </div>

  <!-- </li> -->
  <p>An <code>li</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted if the
  <code>li</code> element is immediately followed by another <code>li</code> element or if there is
  no more content in the parent element.</p>

  <!-- </dt> -->
  <p>A <code>dt</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted if the
  <code>dt</code> element is immediately followed by another <code>dt</code> element or a
  <code>dd</code> element.</p>

  <!-- </dd> -->
  <p>A <code>dd</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted if the
  <code>dd</code> element is immediately followed by another <code>dd</code> element or a
  <code>dt</code> element, or if there is no more content in the parent element.</p>

  <!-- </p> -->
  <p>A <code>p</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted if the
  <code>p</code> element is immediately followed by an <code>address</code>, <code>article</code>,
  <code>aside</code>, <code>blockquote</code>, <code>details</code>, <code>dialog</code>,
  <code>div</code>, <code>dl</code>, <code>fieldset</code>, <code>figcaption</code>,
  <code>figure</code>, <code>footer</code>, <code>form</code>, <code>h1</code>, <code>h2</code>,
  <code>h3</code>, <code>h4</code>, <code>h5</code>, <code>h6</code>, <code>header</code>,
  <code>hgroup</code>, <code>hr</code>, <code>main</code>, <code>menu</code>, <code>nav</code>,
  <code>ol</code>, <code>p</code>, <code>pre</code>, <code>search</code>, <code>section</code>,
  <code>table</code>, or <code>ul</code> element, or if there is no more content in the parent
  element and the parent element is an <span data-x="HTML elements">HTML element</span> that is not
  an <code>a</code>, <code>audio</code>, <code>del</code>, <code>ins</code>, <code>map</code>,
  <code>noscript</code>, or <code>video</code> element, or an <span>autonomous custom
  element</span>.</p>

  <div class="example">

   <p>We can thus simplify the earlier example further:

   <pre><code class="html">&lt;!DOCTYPE HTML>&lt;title>Hello&lt;/title>&lt;p>Welcome to this example.</code></pre>

  </div>

  <!-- </rt> -->
  <p>An <code>rt</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted if the
  <code>rt</code> element is immediately followed by an <code>rt</code> or <code>rp</code> element,
  or if there is no more content in the parent element.</p>

  <!-- </rp> -->
  <p>An <code>rp</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted if the
  <code>rp</code> element is immediately followed by an <code>rt</code> or <code>rp</code> element,
  or if there is no more content in the parent element.</p>

  <!-- </optgroup> (the text assumes <optgroup> can only be inside a <select>; commented out text
  below can handle the non-<select> case if we ever allow it) -->
  <p>An <code>optgroup</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted
  if the <code>optgroup</code> element <!--has an ancestor <code>select</code> element and--> is
  immediately followed by another <code>optgroup</code> element, if it is immediately followed by an
  <code>hr</code> element, or if <!--all of the elements that are ancestors of the
  <code>optgroup</code> element, up to and including the first ancestor element that is not an
  <code>optgroup</code> element, have no more content--> there is no more content in the parent
  element.</p>
  <!-- so e.g. the max number of </optgroup>s are omitted here:
   <select><optgroup></select>
   <p id=x><optgroup></optgroup>x</p>
   <p id=x><optgroup><optgroup></optgroup></optgroup>x</p>
   <p><optgroup id=x><optgroup></optgroup>x</p>
   <p><optgroup><optgroup id=x>x</p>
  -->

  <!-- </option> -->
  <p>An <code>option</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted if
  the <code>option</code> element is immediately followed by another <code>option</code> element, if
  it is immediately followed by an <code>optgroup</code> element, if it is immediately followed by
  an <code>hr</code> element, or if there is no more content in the parent element.</p>

  <!-- <colgroup> -->
  <p>A <code>colgroup</code> element's <span data-x="syntax-start-tag">start tag</span> may be
  omitted if the first thing inside the <code>colgroup</code> element is a <code>col</code> element,
  and if the element is not immediately preceded by another <code>colgroup</code> element whose
  <span data-x="syntax-end-tag">end tag</span> has been omitted. (It can't be omitted if the element
  is empty.)</p>

  <!-- </colgroup> -->
  <p>A <code>colgroup</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted
  if the <code>colgroup</code> element is not immediately followed by <span>ASCII whitespace</span>
  or a <span data-x="syntax-comments">comment</span>.</p>

  <!-- </caption> -->
  <p>A <code>caption</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted if
  the <code>caption</code> element is not immediately followed by <span>ASCII whitespace</span> or a
  <span data-x="syntax-comments">comment</span>.</p>

  <!-- </thead> -->
  <p>A <code>thead</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted if
  the <code>thead</code> element is immediately followed by a <code>tbody</code> or
  <code>tfoot</code> element.</p>

  <!-- <tbody> -->
  <p>A <code>tbody</code> element's <span data-x="syntax-start-tag">start tag</span> may be omitted
  if the first thing inside the <code>tbody</code> element is a <code>tr</code> element, and if the
  element is not immediately preceded by a <code>tbody</code>, <code>thead</code>, or
  <code>tfoot</code> element whose <span data-x="syntax-end-tag">end tag</span> has been omitted. (It
  can't be omitted if the element is empty.)</p>

  <!-- </tbody> -->
  <p>A <code>tbody</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted if
  the <code>tbody</code> element is immediately followed by a <code>tbody</code> or
  <code>tfoot</code> element, or if there is no more content in the parent element.</p>

  <!-- </tfoot> -->
  <p>A <code>tfoot</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted if
  there is no more content in the parent element.</p>

  <!-- </tr> -->
  <p>A <code>tr</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted if the
  <code>tr</code> element is immediately followed by another <code>tr</code> element, or if there is
  no more content in the parent element.</p>

  <!-- </td> -->
  <p>A <code>td</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted if the
  <code>td</code> element is immediately followed by a <code>td</code> or <code>th</code> element,
  or if there is no more content in the parent element.</p>

  <!-- </th> -->
  <p>A <code>th</code> element's <span data-x="syntax-end-tag">end tag</span> may be omitted if the
  <code>th</code> element is immediately followed by a <code>td</code> or <code>th</code> element,
  or if there is no more content in the parent element.</p>

  <div class="example">

   <p>The ability to omit all these table-related tags makes table markup much terser.</p>

   <p>Take this example:</p>

   <pre><code class="html">&lt;table>
 &lt;caption>37547 TEE Electric Powered Rail Car Train Functions (Abbreviated)&lt;/caption>
 &lt;colgroup>&lt;col>&lt;col>&lt;col>&lt;/colgroup>
 &lt;thead>
  &lt;tr>
   &lt;th>Function&lt;/th>
   &lt;th>Control Unit&lt;/th>
   &lt;th>Central Station&lt;/th>
  &lt;/tr>
 &lt;/thead>
 &lt;tbody>
  &lt;tr>
   &lt;td>Headlights&lt;/td>
   &lt;td>&#x2714;&lt;/td>
   &lt;td>&#x2714;&lt;/td>
  &lt;/tr>
  &lt;tr>
   &lt;td>Interior Lights&lt;/td>
   &lt;td>&#x2714;&lt;/td>
   &lt;td>&#x2714;&lt;/td>
  &lt;/tr>
  &lt;tr>
   &lt;td>Electric locomotive operating sounds&lt;/td>
   &lt;td>&#x2714;&lt;/td>
   &lt;td>&#x2714;&lt;/td>
  &lt;/tr>
  &lt;tr>
   &lt;td>Engineer's cab lighting&lt;/td>
   &lt;td>&lt;/td>
   &lt;td>&#x2714;&lt;/td>
  &lt;/tr>
  &lt;tr>
   &lt;td>Station Announcements - Swiss&lt;/td>
   &lt;td>&lt;/td>
   &lt;td>&#x2714;&lt;/td>
  &lt;/tr>
 &lt;/tbody>
&lt;/table></code></pre>

   <p>The exact same table, modulo some whitespace differences, could be marked up as follows:</p>

   <pre><code class="html">&lt;table>
 &lt;caption>37547 TEE Electric Powered Rail Car Train Functions (Abbreviated)
 &lt;colgroup>&lt;col>&lt;col>&lt;col>
 &lt;thead>
  &lt;tr>
   &lt;th>Function
   &lt;th>Control Unit
   &lt;th>Central Station
 &lt;tbody>
  &lt;tr>
   &lt;td>Headlights
   &lt;td>&#x2714;
   &lt;td>&#x2714;
  &lt;tr>
   &lt;td>Interior Lights
   &lt;td>&#x2714;
   &lt;td>&#x2714;
  &lt;tr>
   &lt;td>Electric locomotive operating sounds
   &lt;td>&#x2714;
   &lt;td>&#x2714;
  &lt;tr>
   &lt;td>Engineer's cab lighting
   &lt;td>
   &lt;td>&#x2714;
  &lt;tr>
   &lt;td>Station Announcements - Swiss
   &lt;td>
   &lt;td>&#x2714;
&lt;/table></code></pre>

   <p>Since the cells take up much less room this way, this can be made even terser by having each
   row on one line:</p>

   <pre><code class="html">&lt;table>
 &lt;caption>37547 TEE Electric Powered Rail Car Train Functions (Abbreviated)
 &lt;colgroup>&lt;col>&lt;col>&lt;col>
 &lt;thead>
  &lt;tr> &lt;th>Function                              &lt;th>Control Unit     &lt;th>Central Station
 &lt;tbody>
  &lt;tr> &lt;td>Headlights                            &lt;td>&#x2714;                &lt;td>&#x2714;
  &lt;tr> &lt;td>Interior Lights                       &lt;td>&#x2714;                &lt;td>&#x2714;
  &lt;tr> &lt;td>Electric locomotive operating sounds  &lt;td>&#x2714;                &lt;td>&#x2714;
  &lt;tr> &lt;td>Engineer's cab lighting               &lt;td>                 &lt;td>&#x2714;
  &lt;tr> &lt;td>Station Announcements - Swiss         &lt;td>                 &lt;td>&#x2714;
&lt;/table></code></pre>

   <p>The only differences between these tables, at the DOM level, is with the precise position of
   the (in any case semantically-neutral) whitespace.</p>

  </div>

  <p><strong>However</strong>, a <span data-x="syntax-start-tag">start tag</span> must never be
  omitted if it has any attributes.</p>

  <div class="example">

   <p>Returning to the earlier example with all the whitespace removed and then all the optional
   tags removed:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>&lt;title>Hello&lt;/title>&lt;p>Welcome to this example.</code></pre>

   <p>If the <code>body</code> element in this example had to have a <code
   data-x="attr-class">class</code> attribute and the <code>html</code> element had to have a <code
   data-x="attr-lang">lang</code> attribute, the markup would have to become:</p>

   <pre><code class="html">&lt;!DOCTYPE HTML>&lt;html lang="en">&lt;title>Hello&lt;/title>&lt;body class="demo">&lt;p>Welcome to this example.</code></pre>

  </div>

  <p class="note">This section assumes that the document is conforming, in particular, that there
  are no <span data-x="content models">content model</span> violations. Omitting tags in the fashion
  described in this section in a document that does not conform to the <span>content models</span>
  described in this specification is likely to result in unexpected DOM differences (this is, in
  part, what the content models are designed to avoid).</p>


  <h5 id="element-restrictions">Restrictions on content models</h5>

  <p>For historical reasons, certain elements have extra restrictions beyond even the restrictions
  given by their content model.</p>

  <p>A <code>table</code> element must not contain <code>tr</code> elements, even though these
  elements are technically allowed inside <code>table</code> elements according to the content
  models described in this specification. (If a <code>tr</code> element is put inside a
  <code>table</code> in the markup, it will in fact imply a <code>tbody</code> start tag before
  it.)</p>

  <p>A single <span data-x="syntax-newlines">newline</span> may be placed immediately after the <span
  data-x="syntax-start-tag">start tag</span> of <code>pre</code> and <code>textarea</code> elements.
  This does not affect the processing of the element. The otherwise optional <span
  data-x="syntax-newlines">newline</span> <em>must</em> be included if the element's contents
  themselves start with a <span data-x="syntax-newlines">newline</span> (because otherwise the
  leading newline in the contents would be treated like the optional newline, and ignored).</p>

  <div class="example">
   <p>The following two <code>pre</code> blocks are equivalent:</p>
   <pre><code class="html">&lt;pre>Hello&lt;/pre></code></pre>
   <pre><code class="html">&lt;pre><br>Hello&lt;/pre></code></pre>
  </div>


  <h5 id="cdata-rcdata-restrictions">Restrictions on the contents of raw text and escapable raw text elements</h5>

  <p>The text in <span data-x="raw text elements">raw text</span> and <span>escapable raw text
  elements</span> must not contain any occurrences of the string "<code data-x="">&lt;/</code>"
  (U+003C LESS-THAN SIGN, U+002F SOLIDUS) followed by characters that case-insensitively match the
  tag name of the element followed by one of U+0009 CHARACTER TABULATION (tab), U+000A LINE FEED
  (LF), U+000C FORM FEED (FF), U+000D CARRIAGE RETURN (CR), U+0020 SPACE, U+003E GREATER-THAN SIGN
  (>), or U+002F SOLIDUS (/).</p>


  <h4>Text</h4>

  <p><dfn data-x="syntax-text">Text</dfn> is allowed inside elements, attribute values, and comments.
  Extra constraints are placed on what is and what is not allowed in text based on where the text is
  to be put, as described in the other sections.</p>


  <h5>Newlines</h5>

  <p><dfn data-x="syntax-newlines">Newlines</dfn> in HTML may be represented either as U+000D
  CARRIAGE RETURN (CR) characters, U+000A LINE FEED (LF) characters, or pairs of U+000D CARRIAGE
  RETURN (CR), U+000A LINE FEED (LF) characters in that order.</p>

  <p>Where <span data-x="syntax-charref">character references</span> are allowed, a character
  reference of a U+000A LINE FEED (LF) character (but not a U+000D CARRIAGE RETURN (CR) character)
  also represents a <span data-x="syntax-newlines">newline</span>.</p>


  <h4>Character references</h4>

  <p>In certain cases described in other sections, <span data-x="syntax-text">text</span> may be
  mixed with <dfn data-x="syntax-charref">character references</dfn>. These can be used to escape
  characters that couldn't otherwise legally be included in <span
  data-x="syntax-text">text</span>.</p>

  <p>Character references must start with a U+0026 AMPERSAND character (&amp;). Following this,
  there are three possible kinds of character references:</p>

  <dl>
   <dt>Named character references</dt>

   <dd>The ampersand must be followed by one of the names given in the <span>named character
   references</span> section, using the same case. <span w-nodev>The name must be one that is
   terminated by a U+003B SEMICOLON character (;).</span></dd>


   <dt>Decimal numeric character reference</dt>

   <dd>The ampersand must be followed by a U+0023 NUMBER SIGN character (#), followed by one or more
   <span>ASCII digits</span>, representing a base-ten integer that corresponds to a code point that
   is allowed according to the definition below. The digits must then be followed by a U+003B
   SEMICOLON character (;).</dd>


   <dt>Hexadecimal numeric character reference</dt>

   <dd>The ampersand must be followed by a U+0023 NUMBER SIGN character (#), which must be followed
   by either a U+0078 LATIN SMALL LETTER X character (x) or a U+0058 LATIN CAPITAL LETTER X
   character (X), which must then be followed by one or more <span>ASCII hex digits</span>,
   representing a hexadecimal integer that corresponds to a code point that is allowed according to
   the definition below. The digits must then be followed by a U+003B SEMICOLON character (;).</dd>
  </dl>

  <p>The numeric character reference forms described above are allowed to reference any code point
  excluding U+000D CR, <span data-x="noncharacter">noncharacters</span>, and <span
  data-x="control">controls</span> other than <span>ASCII whitespace</span>.</p>

  <p>An <dfn data-x="syntax-ambiguous-ampersand">ambiguous ampersand</dfn> is a U+0026 AMPERSAND
  character (&amp;) that is followed by one or more <span data-x="ASCII alphanumeric">ASCII
  alphanumerics</span>, followed by a U+003B SEMICOLON character (;), where these characters do not
  match any of the names given in the <span>named character references</span> section.</p>


  <h4>CDATA sections</h4>

  <p><dfn data-x="syntax-cdata">CDATA sections</dfn> must consist of the following components, in
  this order:</p>

  <ol>
   <li>The string "<code data-x="">&lt;![CDATA[</code>".</li>

   <li>Optionally, <span data-x="syntax-text">text</span>, with the additional restriction that the
   text must not contain the string "<code data-x="">]]></code>".</li>

   <li>The string "<code data-x="">]]></code>".</li>
  </ol>

  <div class="example">

   <p>CDATA sections can only be used in foreign content (MathML or SVG). In this example, a CDATA
   section is used to escape the contents of a <span>MathML <code>ms</code></span> element:</p>

   <pre><code class="html">&lt;p>You can add a string to a number, but this stringifies the number:&lt;/p>
&lt;math>
 &lt;ms>&lt;![CDATA[x&lt;y]]>&lt;/ms>
 &lt;mo>+&lt;/mo>
 &lt;mn>3&lt;/mn>
 &lt;mo>=&lt;/mo>
 &lt;ms>&lt;![CDATA[x&lt;y3]]>&lt;/ms>
&lt;/math></code></pre>

  </div>


  <h4>Comments</h4>

  <p><dfn data-x="syntax-comments">Comments</dfn> must have the following format:</p>

  <ol>
   <li>The string "<code data-x="">&lt;!--</code>".</li>

   <li>Optionally, <span data-x="syntax-text">text</span>, with the additional restriction that the
   text must not start with the string "<code data-x="">&gt;</code>", nor start with the string
   "<code data-x="">-></code>", nor contain the strings "<code data-x="">&lt;!--</code>", "<code
   data-x="">--&gt;</code>", or "<code data-x="">--!&gt;</code>", nor end with the string "<code
   data-x="">&lt;!-</code>".</li>

   <li>The string "<code data-x="">--&gt;</code>".</li>
  </ol>

  <p class="note">The <span data-x="syntax-text">text</span> is allowed to end with the string
  "<code data-x="">&lt;!</code>", as in <code data-x="">&lt;!--My favorite operators are > and
  &lt;!--></code>.</p>


<!--HTMLPARSER-->

  <div w-nodev>

  <h3 split-filename="parsing" id="parsing">Parsing HTML documents</h3>

  <p><i>This section only applies to user agents, data mining tools, and conformance
  checkers.</i></p>

  <p class="note">The rules for parsing XML documents into DOM trees are covered by the next
  section, entitled "<span>The XML syntax</span>".</p>

  <p>User agents must use the parsing rules described in this section to generate the DOM trees from
  <code>text/html</code> resources. Together, these rules define what is referred to as the
  <dfn export>HTML parser</dfn>.</p>

  <div class="note">

   <p>While the HTML syntax described in this specification bears a close resemblance to SGML and
   XML, it is a separate language with its own parsing rules.</p>

   <p>Some earlier versions of HTML (in particular from HTML2 to HTML4) were based on SGML and used
   SGML parsing rules. However, few (if any) web browsers ever implemented true SGML parsing for
   HTML documents; the only user agents to strictly handle HTML as an SGML application have
   historically been validators. The resulting confusion &mdash; with validators claiming documents
   to have one representation while widely deployed web browsers interoperably implemented a
   different representation &mdash; has wasted decades of productivity. This version of HTML thus
   returns to a non-SGML basis.</p>

   <p>Authors interested in using SGML tools in their authoring pipeline are encouraged to use XML
   tools and the XML serialization of HTML.</p>

  </div>

  <p>For the purposes of conformance checkers, if a resource is determined to be in <span>the HTML
  syntax</span>, then it is an <span data-x="HTML documents">HTML document</span>.</p>

  <p class="note">As stated <span data-x="HTML elements">in the terminology section</span>,
  references to <span data-x="element type">element types</span> that do not explicitly specify a
  namespace always refer to elements in the <span>HTML namespace</span>. For example, if the spec
  talks about "a <code>menu</code> element", then that is an element with the local name "<code
  data-x="">menu</code>", the namespace "<code data-x="">http://www.w3.org/1999/xhtml</code>", and
  the interface <code>HTMLMenuElement</code>. Where possible, references to such elements are
  hyperlinked to their definition.</p>

  </div>


  <div w-nodev>

  <h4>Overview of the parsing model</h4>

  <p class="overview"><img src="/images/parsing-model-overview.svg" width="345" height="535" alt="" class="darkmode-aware"></p>

  <p>The input to the HTML parsing process consists of a stream of <span data-x="code point">code
  points</span>, which is passed through a <span>tokenization</span> stage followed by a <span>tree
  construction</span> stage. The output is a <code>Document</code> object.</p>

  <p class="note">Implementations that <a href="#non-scripted">do not support scripting</a> do not
  have to actually create a DOM <code>Document</code> object, but the DOM tree in such cases is
  still used as the model for the rest of the specification.</p>

  <p>In the common case, the data handled by the tokenization stage comes from the network, but
  <span data-x="dynamic markup insertion">it can also come from script</span> running in the user
  agent, e.g. using the <code data-x="dom-document-write">document.write()</code> API.</p>

  <p id="nestedParsing">There is only one set of states for the tokenizer stage and the tree
  construction stage, but the tree construction stage is reentrant, meaning that while the tree
  construction stage is handling one token, the tokenizer might be resumed, causing further tokens
  to be emitted and processed before the first token's processing is complete.</p>

  <div class="example">

   <p>In the following example, the tree construction stage will be called upon to handle a "p"
   start tag token while handling the "script" end tag token:</p>

   <pre><code class="html">...
&lt;script>
 document.write('&lt;p>');
&lt;/script>
...</code></pre>

  </div>

  <p>To handle these cases, parsers have a <dfn>script nesting level</dfn>, which must be initially
  set to zero, and a <dfn>parser pause flag</dfn>, which must be initially set to false.</p>

  </div>



  <div w-nodev>

  <h4><dfn data-x="parse error">Parse errors</dfn></h4>

  <p>This specification defines the parsing rules for HTML documents, whether they are syntactically
  correct or not. Certain points in the parsing algorithm are said to be <span data-x="parse
  error">parse errors</span>. The error handling for parse errors is well-defined (that's the
  processing rules described throughout this specification), but user agents, while parsing an HTML
  document, may <span data-x="abort a parser">abort the parser</span> at the first <span>parse
  error</span> that they encounter for which they do not wish to apply the rules described in this
  specification.</p>

  <p>Conformance checkers must report at least one parse error condition to the user if one or more
  parse error conditions exist in the document and must not report parse error conditions if none
  exist in the document. Conformance checkers may report more than one parse error condition if more
  than one parse error condition exists in the document.</p>

  <p class="note">Parse errors are only errors with the <em>syntax</em> of HTML. In addition to
  checking for parse errors, conformance checkers will also verify that the document obeys all the
  other conformance requirements described in this specification.</p>

  <p>Some parse errors have dedicated codes outlined in the table below that should be used by
  conformance checkers in reports.</p>

  <p><i>Error descriptions in the table below are non-normative.</i></p>

  <table class="parse-error-table">
   <thead>
    <tr>
     <th>Code
     <th>Description
    <tbody>
    <tr>
     <td><dfn data-x="parse-error-abrupt-closing-of-empty-comment">abrupt-closing-of-empty-comment</dfn>
     <td><p>This error occurs if the parser encounters an empty <span
     data-x="syntax-comments">comment</span> that is abruptly closed by a U+003E (>) <span>code
     point</span> (i.e., <code data-x="">&lt;!--></code> or <code data-x="">&lt;!---></code>). The
     parser behaves as if the comment is closed correctly.</p>

    <tr>
     <td><dfn data-x="parse-error-abrupt-doctype-public-identifier">abrupt-doctype-public-identifier</dfn>
     <td><p>This error occurs if the parser encounters a U+003E (>) <span>code point</span> in the
     <span data-x="syntax-doctype">DOCTYPE</span> public identifier (e.g., <code
     data-x="">&lt;!DOCTYPE html PUBLIC "foo></code>). In such a case, if the DOCTYPE is correctly
     placed as a document preamble, the parser sets the <code>Document</code> to <span>quirks
     mode</span>.</p>

    <tr>
     <td><dfn data-x="parse-error-abrupt-doctype-system-identifier">abrupt-doctype-system-identifier</dfn>
     <td><p>This error occurs if the parser encounters a U+003E (>) <span>code point</span> in the
     <span data-x="syntax-doctype">DOCTYPE</span> system identifier (e.g., <code
     data-x="">&lt;!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01//EN" "foo></code>). In such a case,
     if the DOCTYPE is correctly placed as a document preamble, the parser sets the
     <code>Document</code> to <span>quirks mode</span>.</p>

    <tr>
     <td><dfn data-x="parse-error-absence-of-digits-in-numeric-character-reference">absence-of-digits-in-numeric-character-reference</dfn>
     <td><p>This error occurs if the parser encounters a numeric <span
     data-x="syntax-charref">character reference</span> that doesn't contain any digits (e.g., <code
     data-x="">&amp;#qux;</code>). In this case the parser doesn't resolve the character
     reference.</p>

    <tr>
     <td><dfn data-x="parse-error-cdata-in-html-content">cdata-in-html-content</dfn>
     <td><p>This error occurs if the parser encounters a <span data-x="syntax-cdata">CDATA
     section</span> outside of foreign content (SVG or MathML). The parser treats such CDATA
     sections (including leading "<code data-x="">[CDATA[</code>" and trailing "<code
     data-x="">]]</code>" strings) as comments.</p>

    <tr>
     <td><dfn data-x="parse-error-character-reference-outside-unicode-range">character-reference-outside-unicode-range</dfn>
     <td><p>This error occurs if the parser encounters a numeric <span
     data-x="syntax-charref">character reference</span> that references a <span>code point</span>
     that is greater than the valid Unicode range. The parser resolves such a character reference to
     a U+FFFD REPLACEMENT CHARACTER.</p>

    <tr>
     <td><dfn data-x="parse-error-control-character-in-input-stream">control-character-in-input-stream</dfn>
     <td><p>This error occurs if the <span>input stream</span> contains a <span
     data-x="control">control</span> <span>code point</span> that is not <span>ASCII
     whitespace</span> or U+0000 NULL. Such code points are parsed as-is and usually, where parsing
     rules don't apply any additional restrictions, make their way into the DOM.</p>

    <tr>
     <td><dfn data-x="parse-error-control-character-reference">control-character-reference</dfn>
     <td><p>This error occurs if the parser encounters a numeric <span
     data-x="syntax-charref">character reference</span> that references a <span
     data-x="control">control</span> <span>code point</span> that is not <span>ASCII
     whitespace</span> or is a U+000D CARRIAGE RETURN. The parser resolves such character references
     as-is except C1 control references that are replaced according to the <span>numeric character
     reference end state</span>.</p>

    <tr>
     <td><dfn data-x="parse-error-duplicate-attribute">duplicate-attribute</dfn>
     <td><p>This error occurs if the parser encounters an <span
     data-x="syntax-attributes">attribute</span> in a tag that already has an attribute with the
     same name. The parser ignores all such duplicate occurrences of the attribute.

    <tr>
     <td><dfn data-x="parse-error-end-tag-with-attributes">end-tag-with-attributes</dfn>
     <td><p>This error occurs if the parser encounters an <span data-x="syntax-end-tag">end
     tag</span> with <span data-x="syntax-attributes">attributes</span>. Attributes in end tags are
     ignored and do not make their way into the DOM.</p>

    <tr>
     <td><dfn data-x="parse-error-end-tag-with-trailing-solidus">end-tag-with-trailing-solidus</dfn>
     <td><p>This error occurs if the parser encounters an <span data-x="syntax-end-tag">end
     tag</span> that has a U+002F (/) <span>code point</span> right before the closing U+003E (>)
     code point (e.g., <code data-x="">&lt;/div/></code>). Such a tag is treated as a regular end
     tag.</p>

    <tr>
     <td><dfn data-x="parse-error-eof-before-tag-name">eof-before-tag-name</dfn>
     <td><p>This error occurs if the parser encounters the end of the <span>input stream</span>
     where a tag name is expected. In this case the parser treats the beginning of a <span
     data-x="syntax-start-tag">start tag</span> (i.e., <code data-x="">&lt;</code>) or an <span
     data-x="syntax-end-tag">end tag</span> (i.e., <code data-x="">&lt;/</code>) as text
     content.</p>

    <tr>
     <td><dfn data-x="parse-error-eof-in-cdata">eof-in-cdata</dfn>
     <td><p>This error occurs if the parser encounters the end of the <span>input stream</span> in a
     <span data-x="syntax-cdata">CDATA section</span>. The parser treats such CDATA sections as if
     they are closed immediately before the end of the input stream.</p>

    <tr>
     <td><dfn data-x="parse-error-eof-in-comment">eof-in-comment</dfn>
     <td><p>This error occurs if the parser encounters the end of the <span>input stream</span> in a
     <span data-x="syntax-comments">comment</span>. The parser treats such comments as if they are
     closed immediately before the end of the input stream.</p>

    <tr>
     <td><dfn data-x="parse-error-eof-in-doctype">eof-in-doctype</dfn>
     <td><p>This error occurs if the parser encounters the end of the input stream in a <span
     data-x="syntax-doctype">DOCTYPE</span>. In such a case, if the DOCTYPE is correctly placed as a
     document preamble, the parser sets the <code>Document</code> to <span>quirks mode</span>.</p>

    <tr>
     <td><dfn data-x="parse-error-eof-in-script-html-comment-like-text">eof-in-script-html-comment-like-text</dfn>
     <td>
      <p>This error occurs if the parser encounters the end of the <span>input stream</span> in text
      that resembles an <span data-x="syntax-comments">HTML comment</span> inside
      <code>script</code> element content (e.g., <code data-x="">&lt;script>&lt;!-- foo</code>).</p>

      <p class="note">Syntactic structures that resemble HTML comments in <code>script</code>
      elements are parsed as text content. They can be a part of a scripting language-specific
      syntactic structure or be treated as an HTML-like comment, if the scripting language supports
      them (e.g., parsing rules for HTML-like comments can be found in Annex B of the JavaScript
      specification). The common reason for this error is a violation of the <span data-x="script
      content restrictions">restrictions for contents of <code>script</code> elements</span>. <ref>JAVASCRIPT</ref></p>

    <tr>
     <td><dfn data-x="parse-error-eof-in-tag">eof-in-tag</dfn>
     <td><p>This error occurs if the parser encounters the end of the <span>input stream</span> in a
     <span data-x="syntax-start-tag">start tag</span> or an <span data-x="syntax-end-tag">end
     tag</span> (e.g., <code data-x="">&lt;div id=</code>). Such a tag is ignored.</p>

    <tr>
     <td><dfn data-x="parse-error-incorrectly-closed-comment">incorrectly-closed-comment</dfn>
     <td><p>This error occurs if the parser encounters a <span
     data-x="syntax-comments">comment</span> that is closed by the "<code data-x="">--!></code>"
     <span>code point</span> sequence. The parser treats such comments as if they are correctly
     closed by the "<code data-x="">--></code>" code point sequence.</p>

    <tr>
     <td><dfn data-x="parse-error-incorrectly-opened-comment">incorrectly-opened-comment</dfn>
     <td>
      <p>This error occurs if the parser encounters the "<code data-x="">&lt;!</code>" <span>code
      point</span> sequence that is not immediately followed by two U+002D (-) code points and that
      is not the start of a <span data-x="syntax-doctype">DOCTYPE</span> or a <span
      data-x="syntax-cdata">CDATA section</span>. All content that follows the "<code
      data-x="">&lt;!</code>" code point sequence up to a U+003E (>) code point (if present) or to
      the end of the <span>input stream</span> is treated as a comment.</p>

      <p class="note">One possible cause of this error is using an XML markup declaration (e.g.,
      <code data-x="">&lt;!ELEMENT br EMPTY></code>) in HTML.</p>

    <tr>
     <td><dfn data-x="parse-error-invalid-character-sequence-after-doctype-name">invalid-character-sequence-after-doctype-name</dfn>
     <td><p>This error occurs if the parser encounters any <span>code point</span> sequence other
     than "<code data-x="">PUBLIC</code>" and "<code data-x="">SYSTEM</code>" keywords after a <span
     data-x="syntax-doctype">DOCTYPE</span> name. In such a case, the parser ignores any following
     public or system identifiers, and if the DOCTYPE is correctly placed as a document preamble,
     and if the <span>parser cannot change the mode flag</span> is false, sets the <code>Document</code>
     to <span>quirks mode</span>.</p>

    <tr>
     <td><dfn data-x="parse-error-invalid-first-character-of-tag-name">invalid-first-character-of-tag-name</dfn>
     <td>
      <p>This error occurs if the parser encounters a <span>code point</span> that is not an
      <span>ASCII alpha</span> where first code point of a <span data-x="syntax-start-tag">start
      tag</span> name or an <span data-x="syntax-end-tag">end tag</span> name is expected. If a
      start tag was expected such code point and a preceding U+003C (&lt;) is treated as text
      content, and all content that follows is treated as markup. Whereas, if an end tag was
      expected, such code point and all content that follows up to a U+003E (>) code point (if
      present) or to the end of the <span>input stream</span> is treated as a comment.</p>

      <div class="example">
       <p>For example, consider the following markup:</p>

       <pre><code class="html">&lt;42>&lt;/42></code></pre>

       <p>This will be parsed into:</p>

       <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t3"><code>#text</code>: <span data-x="">&lt;42></span></li><li class="t8"><code>#comment</code>: <span data-x="">42</span></li></ul></li></ul></li></ul>
      </div>

      <p class="note">While the first code point of a tag name is limited to an <span>ASCII
      alpha</span>, a wide range of code points (including <span>ASCII digits</span>) is allowed in
      subsequent positions.</p>

    <tr>
     <td><dfn data-x="parse-error-missing-attribute-value">missing-attribute-value</dfn>
     <td><p>This error occurs if the parser encounters a U+003E (>) <span>code point</span> where an
     <span data-x="syntax-attributes">attribute</span> value is expected (e.g., <code
     data-x="">&lt;div id=></code>). The parser treats the attribute as having an empty value.</p>

    <tr>
     <td><dfn data-x="parse-error-missing-doctype-name">missing-doctype-name</dfn>
     <td><p>This error occurs if the parser encounters a <span
     data-x="syntax-doctype">DOCTYPE</span> that is missing a name (e.g., <code
     data-x="">&lt;!DOCTYPE></code>). In such a case, if the DOCTYPE is correctly placed as a
     document preamble, the parser sets the <code>Document</code> to <span>quirks mode</span>.</p>

    <tr>
     <td><dfn data-x="parse-error-missing-doctype-public-identifier">missing-doctype-public-identifier</dfn>
     <td><p>This error occurs if the parser encounters a U+003E (>) <span>code point</span> where
     start of the <span data-x="syntax-doctype">DOCTYPE</span> public identifier is expected (e.g.,
     <code data-x="">&lt;!DOCTYPE html PUBLIC ></code>). In such a case, if the DOCTYPE is correctly
     placed as a document preamble, the parser sets the <code>Document</code> to <span>quirks
     mode</span>.</p>

    <tr>
     <td><dfn data-x="parse-error-missing-doctype-system-identifier">missing-doctype-system-identifier</dfn>
     <td><p>This error occurs if the parser encounters a U+003E (>) <span>code point</span> where
     start of the <span data-x="syntax-doctype">DOCTYPE</span> system identifier is expected (e.g.,
     <code data-x="">&lt;!DOCTYPE html SYSTEM ></code>). In such a case, if the DOCTYPE is correctly
     placed as a document preamble, the parser sets the <code>Document</code> to <span>quirks
     mode</span>.</p>

    <tr>
     <td><dfn data-x="parse-error-missing-end-tag-name">missing-end-tag-name</dfn>
     <td><p>This error occurs if the parser encounters a U+003E (>) <span>code point</span> where an
     <span data-x="syntax-end-tag">end tag</span> name is expected, i.e., <code
     data-x="">&lt;/></code>. The parser ignores the whole "<code data-x="">&lt;/></code>" code
     point sequence.</p>

    <tr>
     <td><dfn data-x="parse-error-missing-quote-before-doctype-public-identifier">missing-quote-before-doctype-public-identifier</dfn>
     <td><p>This error occurs if the parser encounters the <span
     data-x="syntax-doctype">DOCTYPE</span> public identifier that is not preceded by a quote (e.g.,
     <code data-x="">&lt;!DOCTYPE html PUBLIC -//W3C//DTD HTML 4.01//EN"></code>). In such a case,
     the parser ignores the public identifier, and if the DOCTYPE is correctly placed as a document
     preamble, sets the <code>Document</code> to <span>quirks mode</span>.</p>

    <tr>
     <td><dfn data-x="parse-error-missing-quote-before-doctype-system-identifier">missing-quote-before-doctype-system-identifier</dfn>
     <td><p>This error occurs if the parser encounters the <span
     data-x="syntax-doctype">DOCTYPE</span> system identifier that is not preceded by a quote (e.g.,
     <code data-x="">&lt;!DOCTYPE html SYSTEM
     http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"></code>). In such a case, the parser
     ignores the system identifier, and if the DOCTYPE is correctly placed as a document preamble,
     sets the <code>Document</code> to <span>quirks mode</span>.</p>

    <tr>
     <td><dfn data-x="parse-error-missing-semicolon-after-character-reference">missing-semicolon-after-character-reference</dfn>
     <td>
      <p>This error occurs if the parser encounters a <span data-x="syntax-charref">character
      reference</span> that is not terminated by a U+003B (;) <span>code point</span>. Usually the
      parser behaves as if character reference is terminated by the U+003B (;) code point; however,
      there are some ambiguous cases in which the parser includes subsequent code points in the
      character reference.</p>

      <p class="example">For example, <code data-x="">&amp;not;in</code> will be parsed as "<code
      data-x="">&not;in</code>" whereas <code data-x="">&amp;notin</code> will be parsed as "<code
      data-x="">&notin;</code>".</p>

    <tr>
     <td><dfn data-x="parse-error-missing-whitespace-after-doctype-public-keyword">missing-whitespace-after-doctype-public-keyword</dfn>
     <td><p>This error occurs if the parser encounters a <span
     data-x="syntax-doctype">DOCTYPE</span> whose "<code data-x="">PUBLIC</code>" keyword and public
     identifier are not separated by <span>ASCII whitespace</span>. In this case the parser behaves
     as if ASCII whitespace is present.</p>

    <tr>
     <td><dfn data-x="parse-error-missing-whitespace-after-doctype-system-keyword">missing-whitespace-after-doctype-system-keyword</dfn>
     <td><p>This error occurs if the parser encounters a <span
     data-x="syntax-doctype">DOCTYPE</span> whose "<code data-x="">SYSTEM</code>" keyword and system
     identifier are not separated by <span>ASCII whitespace</span>. In this case the parser behaves
     as if ASCII whitespace is present.</p>

    <tr>
     <td><dfn data-x="parse-error-missing-whitespace-before-doctype-name">missing-whitespace-before-doctype-name</dfn>
     <td><p>This error occurs if the parser encounters a <span
     data-x="syntax-doctype">DOCTYPE</span> whose "<code data-x="">DOCTYPE</code>" keyword and name
     are not separated by <span>ASCII whitespace</span>. In this case the parser behaves as if ASCII
     whitespace is present.</p>

    <tr>
     <td><dfn data-x="parse-error-missing-whitespace-between-attributes">missing-whitespace-between-attributes</dfn>
     <td><p>This error occurs if the parser encounters <span
     data-x="syntax-attributes">attributes</span> that are not separated by <span>ASCII
     whitespace</span> (e.g., <code data-x="">&lt;div id="foo"class="bar"></code>). In this case the
     parser behaves as if ASCII whitespace is present.</p>

    <tr>
     <td><dfn data-x="parse-error-missing-whitespace-between-doctype-public-and-system-identifiers">missing-whitespace-between-doctype-public-and-system-identifiers</dfn>
     <td><p>This error occurs if the parser encounters a <span
     data-x="syntax-doctype">DOCTYPE</span> whose public and system identifiers are not separated by
     <span>ASCII whitespace</span>. In this case the parser behaves as if ASCII whitespace is
     present.</p>

    <tr>
     <td><dfn data-x="parse-error-nested-comment">nested-comment</dfn>
     <td><p>This error occurs if the parser encounters a nested <span
     data-x="syntax-comments">comment</span> (e.g., <code data-x="">&lt;!-- &lt;!-- nested -->
     --></code>). Such a comment will be closed by the first occurring "<code data-x="">--></code>"
     <span>code point</span> sequence and everything that follows will be treated as markup.</p>

    <tr>
     <td><dfn data-x="parse-error-noncharacter-character-reference">noncharacter-character-reference</dfn>
     <td><p>This error occurs if the parser encounters a numeric <span
     data-x="syntax-charref">character reference</span> that references a <span>noncharacter</span>.
     The parser resolves such character references as-is.</p>

    <tr>
     <td><dfn data-x="parse-error-noncharacter-in-input-stream">noncharacter-in-input-stream</dfn>
     <td><p>This error occurs if the <span>input stream</span> contains a <span>noncharacter</span>.
     Such <span data-x="code point">code points</span> are parsed as-is and usually, where parsing
     rules don't apply any additional restrictions, make their way into the DOM.</p>

    <tr>
     <td><dfn data-x="parse-error-non-void-html-element-start-tag-with-trailing-solidus">non-void-html-element-start-tag-with-trailing-solidus</dfn>
     <td>
      <p>This error occurs if the parser encounters a <span data-x="syntax-start-tag">start
      tag</span> for an element that is not in the list of <span>void elements</span> or is not a
      part of foreign content (i.e., not an SVG or MathML element) that has a U+002F (/) <span>code
      point</span> right before the closing U+003E (>) code point. The parser behaves as if the
      U+002F (/) is not present.</p>

      <div class="example">
       <p>For example, consider the following markup:</p>

       <pre><code class="html">&lt;div/>&lt;span>&lt;/span>&lt;span>&lt;/span></code></pre>

       <p>This will be parsed into:</p>

       <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>div</code><ul><li class="t1"><code>span</code></li><li class="t1"><code>span</code></li></ul></li></ul></li></ul></li></ul>
      </div>

      <p class="note">The trailing U+002F (/) in a start tag name can be used only in foreign
      content to specify self-closing tags. (Self-closing tags don't exist in HTML.) It is also
      allowed for void elements, but doesn't have any effect in this case.</p>

    <tr>
     <td><dfn data-x="parse-error-null-character-reference">null-character-reference</dfn>
     <td><p>This error occurs if the parser encounters a numeric <span
     data-x="syntax-charref">character reference</span> that references a U+0000 NULL <span>code
     point</span>. The parser resolves such character references to a U+FFFD REPLACEMENT
     CHARACTER.</p>

    <tr>
     <td><dfn data-x="parse-error-surrogate-character-reference">surrogate-character-reference</dfn>
     <td><p>This error occurs if the parser encounters a numeric <span
     data-x="syntax-charref">character reference</span> that references a <span>surrogate</span>.
     The parser resolves such character references to a U+FFFD REPLACEMENT CHARACTER.</p>

    <tr>
     <td><dfn data-x="parse-error-surrogate-in-input-stream">surrogate-in-input-stream</dfn>
     <td>
      <p>This error occurs if the <span>input stream</span> contains a <span
      data-x="surrogate">surrogate</span>. Such <span data-x="code point">code points</span> are
      parsed as-is and usually, where parsing rules don't apply any additional restrictions, make
      their way into the DOM.</p>

      <p class="note">Surrogates can only find their way into the input stream via script APIs such
      as <code data-x="dom-document-write">document.write()</code>.</p>

    <tr>
     <td><dfn data-x="parse-error-unexpected-character-after-doctype-system-identifier">unexpected-character-after-doctype-system-identifier</dfn>
     <td><p>This error occurs if the parser encounters any <span data-x="code point">code
     points</span> other than <span>ASCII whitespace</span> or closing U+003E (>) after the <span
     data-x="syntax-doctype">DOCTYPE</span> system identifier. The parser ignores these code
     points.</p>

    <tr>
     <td><dfn data-x="parse-error-unexpected-character-in-attribute-name">unexpected-character-in-attribute-name</dfn>
     <td>
      <p>This error occurs if the parser encounters a U+0022 ("), U+0027 ('), or U+003C (&lt;)
      <span>code point</span> in an <span data-x="syntax-attribute-name">attribute name</span>. The
      parser includes such code points in the attribute name.</p>

      <p class="note">Code points that trigger this error are usually a part of another syntactic
      construct and can be a sign of a typo around the attribute name.</p>

      <div class="example">
       <p>For example, consider the following markup:</p>

       <pre><code class="html">&lt;div foo&lt;div></code></pre>

       <p>Due to a forgotten U+003E (>) code point after <code data-x="">foo</code> the parser
       treats this markup as a single <code>div</code> element with a "<code
       data-x="">foo&lt;div</code>" attribute.</p>

       <p>As another example of this error, consider the following markup:</p>

       <pre><code class="html">&lt;div id'bar'></code></pre>

       <p>Due to a forgotten U+003D (=) code point between an attribute name and value the parser
       treats this markup as a <code>div</code> element with the attribute "<code
       data-x="">id'bar'</code>" that has an empty value.</p>
     </div>

    <tr>
     <td><dfn data-x="parse-error-unexpected-character-in-unquoted-attribute-value">unexpected-character-in-unquoted-attribute-value</dfn>
     <td>
      <p>This error occurs if the parser encounters a U+0022 ("), U+0027 ('), U+003C (&lt;), U+003D
      (=), or U+0060 (`) <span>code point</span> in an unquoted <span
      data-x="syntax-attribute-value">attribute value</span>. The parser includes such code points
      in the attribute value.</p>

      <p class="note">Code points that trigger this error are usually a part of another syntactic
      construct and can be a sign of a typo around the attribute value.</p>

      <p class="note">U+0060 (`) is in the list of code points that trigger this error because
      certain legacy user agents treat it as a quote.</p>

      <div class="example">
       <p>For example, consider the following markup:</p>

       <pre><code class="html">&lt;div foo=b'ar'></code></pre>

       <p>Due to a misplaced U+0027 (') code point the parser sets the value of the "<code
       data-x="">foo</code>" attribute to "<code data-x="">b'ar'</code>".</p>
     </div>

    <tr>
     <td><dfn data-x="parse-error-unexpected-equals-sign-before-attribute-name">unexpected-equals-sign-before-attribute-name</dfn>
     <td>
      <p>This error occurs if the parser encounters a U+003D (=) <span>code point</span> before an
      attribute name. In this case the parser treats U+003D (=) as the first code point of the
      attribute name.</p>

      <p class="note">The common reason for this error is a forgotten attribute name.</p>

      <div class="example">
       <p>For example, consider the following markup:</p>

       <pre><code class="html">&lt;div foo="bar" ="baz"></code></pre>

       <p>Due to a forgotten attribute name the parser treats this markup as a <code>div</code>
       element with two attributes: a "<code data-x="">foo</code>" attribute with a "<code
       data-x="">bar</code>" value and a "<code data-x="">="baz"</code>" attribute with an empty
       value.</p>
      </div>

    <tr>
     <td><dfn data-x="parse-error-unexpected-null-character">unexpected-null-character</dfn>
     <td><p>This error occurs if the parser encounters a U+0000 NULL <span>code point</span> in the
     <span>input stream</span> in certain positions. In general, such code points are either
     ignored or, for security reasons, replaced with a U+FFFD REPLACEMENT CHARACTER.</p>

    <tr>
     <td><dfn data-x="parse-error-unexpected-question-mark-instead-of-tag-name">unexpected-question-mark-instead-of-tag-name</dfn>
     <td>
      <p>This error occurs if the parser encounters a U+003F (?) <span>code point</span> where first
      code point of a <span data-x="syntax-start-tag">start tag</span> name is expected. The U+003F
      (?) and all content that follows up to a U+003E (>) code point (if present) or to the end of
      the <span>input stream</span> is treated as a comment.</p>

      <div class="example">
       <p>For example, consider the following markup:</p>

       <pre><code class="html">&lt;?xml-stylesheet type="text/css" href="style.css"?></code></pre>

       <p>This will be parsed into:</p>

       <ul class="domTree"><li class="t8"><code>#comment</code>: <span data-x="">?xml-stylesheet type="text/css" href="style.css"?</span></li><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code></li></ul></li></ul>
      </div>

      <p class="note">The common reason for this error is an XML processing instruction (e.g., <code
      data-x="">&lt;?xml-stylesheet type="text/css" href="style.css"?></code>) or an XML declaration
      (e.g., <code data-x="">&lt;?xml version="1.0" encoding="UTF-8"?></code>) being used in
      HTML.</p>

    <tr>
     <td><dfn data-x="parse-error-unexpected-solidus-in-tag">unexpected-solidus-in-tag</dfn>
     <td><p>This error occurs if the parser encounters a U+002F (/) <span>code point</span> that is
     not a part of a quoted <span data-x="syntax-attributes">attribute</span> value and not
     immediately followed by a U+003E (>) code point in a tag (e.g., <code data-x="">&lt;div /
     id="foo"></code>). In this case the parser behaves as if it encountered <span>ASCII
     whitespace</span>.</p>

    <tr>
     <td><dfn data-x="parse-error-unknown-named-character-reference">unknown-named-character-reference</dfn>
     <td><p>This error occurs if the parser encounters an <span
     data-x="syntax-ambiguous-ampersand">ambiguous ampersand</span>. In this case the parser doesn't
     resolve the <span data-x="syntax-charref">character reference</span>.</p>

  </table>

  </div>


  <div w-nodev>

  <h4>The <dfn>input byte stream</dfn></h4>

  <p>The stream of code points that comprises the input to the tokenization stage will be initially
  seen by the user agent as a stream of bytes (typically coming over the network or from the local
  file system). The bytes encode the actual characters according to a particular <i>character
  encoding</i>, which the user agent uses to decode the bytes into characters.</p>

  <p class="note">For XML documents, the algorithm user agents are required to use to determine the
  character encoding is given by <cite>XML</cite>. This section does not apply to XML documents.
  <ref>XML</ref></p>

  <p>Usually, the <span>encoding sniffing algorithm</span> defined below is used to determine the
  character encoding.</p>

  <p>Given a character encoding, the bytes in the <span>input byte stream</span> must be converted
  to characters for the tokenizer's <span>input stream</span>, by passing the <span>input byte
  stream</span> and character encoding to <span>decode</span>.</p>

  <p class="note">A leading Byte Order Mark (BOM) causes the character encoding argument to be
  ignored and will itself be skipped.</p>

  <p class="note">Bytes or sequences of bytes in the original byte stream that did not conform to
  the Encoding standard (e.g. invalid UTF-8 byte sequences in a UTF-8 input byte stream) are errors
  that conformance checkers are expected to report. <ref>ENCODING</ref></p>

  <p class="warning">The decoder algorithms describe how to handle invalid input; for security
  reasons, it is imperative that those rules be followed precisely. Differences in how invalid byte
  sequences are handled can result in, amongst other problems, script injection vulnerabilities
  ("XSS").</p>

  <p>When the HTML parser is decoding an input byte stream, it uses a character encoding and a <dfn
  data-x="concept-encoding-confidence">confidence</dfn>. The confidence is either <i>tentative</i>,
  <i>certain</i>, or <i>irrelevant</i>. The encoding used, and whether the confidence in that
  encoding is <i>tentative</i> or <i>certain</i>, is <a href="#meta-charset-during-parse">used
  during the parsing</a> to determine whether to <span>change the encoding</span>. If no encoding is
  necessary, e.g. because the parser is operating on a Unicode stream and doesn't have to use a
  character encoding at all, then the <span data-x="concept-encoding-confidence">confidence</span> is
  <i>irrelevant</i>.</p>

  <p class="note">Some algorithms feed the parser by directly adding characters to the <span>input
  stream</span> rather than adding bytes to the <span>input byte stream</span>.</p>



  <h5>Parsing with a known character encoding</h5>

  <p>When the HTML parser is to operate on an input byte stream that has <dfn export>a known
  definite encoding</dfn>, then the character encoding is that encoding and the <span
  data-x="concept-encoding-confidence">confidence</span> is <i>certain</i>.</p>


  <h5>Determining the character encoding</h5>

  <p>In some cases, it might be impractical to unambiguously determine the encoding before parsing
  the document. Because of this, this specification provides for a two-pass mechanism with an
  optional pre-scan. Implementations are allowed, as described below, to apply a simplified parsing
  algorithm to whatever bytes they have available before beginning to parse the document. Then, the
  real parser is started, using a tentative encoding derived from this pre-parse and other
  out-of-band metadata. If, while the document is being loaded, the user agent discovers a character
  encoding declaration that conflicts with this information, then the parser can get reinvoked to
  perform a parse of the document with the real encoding.</p>

  <p id="documentEncoding">User agents must use the following algorithm, called the <dfn>encoding
  sniffing algorithm</dfn>, to determine the character encoding to use when decoding a document in
  the first pass. This algorithm takes as input any out-of-band metadata available to the user agent
  (e.g. the <span data-x="Content-Type">Content-Type metadata</span> of the document) and all the
  bytes available so far, and returns a character encoding and a <span
  data-x="concept-encoding-confidence">confidence</span> that is either <i>tentative</i> or
  <i>certain</i>.</p>

  <ol>
   <li>
    <p>If the result of <span data-x="BOM sniff">BOM sniffing</span> is an encoding, return that
    encoding with <span data-x="concept-encoding-confidence">confidence</span> <i>certain</i>.</p>

    <p class="note">Although the <span>decode</span> algorithm will itself change the encoding to
    use based on the presence of a byte order mark, this algorithm sniffs the BOM as well in order
    to set the correct <span>document's character encoding</span> and <span
    data-x="concept-encoding-confidence">confidence</span>.</p>
   </li>

   <li>
    <p>If the user has explicitly instructed the user agent to override the document's character
    encoding with a specific encoding, optionally return that encoding with the <span
    data-x="concept-encoding-confidence">confidence</span> <i>certain</i>.</p>

    <p class="note">Typically, user agents remember such user requests across sessions, and in some
    cases apply them to documents in <code>iframe</code>s as well.</p>
   </li>

   <li>
    <p>The user agent may wait for more bytes of the resource to be available, either in this step
    or at any later step in this algorithm. For instance, a user agent might wait 500ms or 1024
    bytes, whichever came first. In general preparsing the source to find the encoding improves
    performance, as it reduces the need to throw away the data structures used when parsing upon
    finding the encoding information. However, if the user agent delays too long to obtain data to
    determine the encoding, then the cost of the delay could outweigh any performance improvements
    from the preparse.</p>

    <p class="note">The authoring conformance requirements for character encoding declarations limit
    them to only appearing <a href="#charset1024">in the first 1024 bytes</a>. User agents are
    therefore encouraged to use the prescan algorithm below (as invoked by these steps) on the first
    1024 bytes, but not to stall beyond that.</p>
   </li>

   <li><p>If the transport layer specifies a character encoding, and it is supported, return that
   encoding with the <span data-x="concept-encoding-confidence">confidence</span> <i>certain</i>.</p></li>

   <li>
    <p>Optionally <span data-x="prescan a byte stream to determine its encoding">prescan the byte
    stream to determine its encoding</span>, with the <i><span data-x="prescan-end-condition">end
    condition</span></i> being when the user agent decides that scanning further bytes would not be
    efficient. User agents are encouraged to only prescan the first 1024 bytes. User agents may
    decide that scanning <em>any</em> bytes is not efficient, in which case these substeps are
    entirely skipped.</p>

    <p>The aforementioned algorithm returns either a character encoding or failure. If it returns a
    character encoding, then return the same encoding, with <span
    data-x="concept-encoding-confidence">confidence</span> <i>tentative</i>.</p>
   </li>

   <li>
    <p>If the <span>HTML parser</span> for which this algorithm is being run is associated with a
    <code>Document</code> <var>d</var> whose <span data-x="doc-container-document">container
    document</span> is non-null, then:</p>

    <ol>
     <li><p>Let <var>parentDocument</var> be <var>d</var>'s <span
     data-x="doc-container-document">container document</span>.</p></li>

     <li><p>If <var>parentDocument</var>'s <span data-x="concept-document-origin">origin</span> is
     <span>same origin</span> with <var>d</var>'s <span
     data-x="concept-document-origin">origin</span> and <var>parentDocument</var>'s <span
     data-x="document's character encoding">character encoding</span> is not
     <span>UTF-16BE/LE</span>, then return <var>parentDocument</var>'s <span data-x="document's
     character encoding">character encoding</span>, with the <span
     data-x="concept-encoding-confidence">confidence</span> <i>tentative</i>.</p></li>
    </ol>
   </li>

   <li><p>Otherwise, if the user agent has information on the likely encoding for this page, e.g.
   based on the encoding of the page when it was last visited, then return that encoding, with the
   <span data-x="concept-encoding-confidence">confidence</span> <i>tentative</i>.</p></li>

   <li>
    <p>The user agent may attempt to autodetect the character encoding from applying frequency
    analysis or other algorithms to the data stream. Such algorithms may use information about the
    resource other than the resource's contents, including the address of the resource. If
    autodetection succeeds in determining a character encoding, and that encoding is a supported
    encoding, then return that encoding, with the <span
    data-x="concept-encoding-confidence">confidence</span> <i>tentative</i>.
    <ref>UNIVCHARDET</ref></p>

    <p class="note">User agents are generally discouraged from attempting to autodetect encodings
    for resources obtained over the network, since doing so involves inherently non-interoperable
    heuristics. Attempting to detect encodings based on an HTML document's preamble is especially
    tricky since HTML markup typically uses only ASCII characters, and HTML documents tend to begin
    with a lot of markup rather than with text content.</p>

    <p class="note">The UTF-8 encoding has a highly detectable bit pattern. Files from the local
    file system that contain bytes with values greater than 0x7F which match the UTF-8 pattern are
    very likely to be UTF-8, while documents with byte sequences that do not match it are very
    likely not. When a user agent can examine the whole file, rather than just the preamble,
    detecting for UTF-8 specifically can be especially effective. <ref>PPUTF8</ref>
    <ref>UTF8DET</ref></p>
   </li>

   <li>
    <p>Otherwise, return an <span>implementation-defined</span> or user-specified default character
    encoding, with the <span data-x="concept-encoding-confidence">confidence</span>
    <i>tentative</i>.</p>

    <p>In controlled environments or in environments where the encoding of documents can be
    prescribed (for example, for user agents intended for dedicated use in new networks), the
    comprehensive <code data-x="">UTF-8</code> encoding is suggested.</p>

    <p>In other environments, the default encoding is typically dependent on the user's locale (an
    approximation of the languages, and thus often encodings, of the pages that the user is likely
    to frequent). The following table gives suggested defaults based on the user's locale, for
    compatibility with legacy content. Locales are identified by BCP 47 language tags.
    <ref>BCP47</ref> <ref>ENCODING</ref></p>

    <!-- based on three sources:
          1. Mozilla 1.9.1 localizations: http://mxr.mozilla.org/l10n-mozilla1.9.1/find?string=global%2Fintl.properties&tree=l10n-mozilla1.9.1&hint= (though this data was apparently misinterpreted in some cases)
          2. Windows Vista encodings: http://msdn.microsoft.com/en-us/goglobal/bb896001
          3. Chrome encodings: https://code.google.com/p/chromium/codesearch#search/&q=IDS_DEFAULT_ENCODING
         several assumptions were made in this process; amongst them:
          - ISO-8859-1 and windows-1252 are the same (supported by encoding.spec.whatwg.org)
          - ISO-8859-9 and windows-1254 are the same (supported by encoding.spec.whatwg.org)
          - windows-31J and Shift_JIS are the same (supported by encoding.spec.whatwg.org)
          - windows-932 is close enough to Shift_JIS to be treated as equivalent (supported by wikipedia)
          - windows-936 is basically the same as GBK which is a subset of gb18030 (supported by wikipedia), but use GBK for its conservative encoder
          - windows-950 is basically the same as Big5 (supported by wikipedia)
          - Firefox's UTF-8 defaults are all bogus
    -->

    <table>
     <thead>
      <tr>
       <th colspan=2>Locale language
       <th>Suggested default encoding
     <tbody>

      <!-- af, Afrikaans, uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- am, Amharic, uses windows-1252: Firefox and Chrome agreed -->

      <tr>
       <td>ar
       <td>Arabic
       <td><span>windows-1256</span> <!-- Windows Vista and Chrome agreed -->

      <!-- arn-CL, Mapudungun (Chile), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- az, Azeri, is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1254 -->
      <!-- Subsequent observation from Firefox telemetry changed this: -->
      <tr>
       <td>az
       <td>Azeri
       <td><span>windows-1254</span>

      <!-- az-Cyrl-AZ, Azeri (Cyrillic, Azerbaijan), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1251 -->

      <!-- ba-RU, Bashkir (Russia), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1251 -->
      <!-- ba wasn't listed at all because none of the sources knew about it. However, further feedback has changed this: -->
      <tr>
       <td>ba
       <td>Bashkir
       <td><span>windows-1251</span> <!-- per https://www.w3.org/Bugs/Public/show_bug.cgi?id=23089 -->

      <!-- be, Belarusian, was not initially listed here because Windows Vista wanted windows-1251, Chrome wanted <none>, and Firefox wanted ISO-8859-5 -->
      <!-- further feedback has changed this: -->
      <tr>
       <td>be
       <td>Belarusian
       <td><span>windows-1251</span> <!-- per https://www.w3.org/Bugs/Public/show_bug.cgi?id=23089 -->

      <!-- be-BY, Belarusian (Belarus), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1251 -->

      <tr>
       <td>bg
       <td>Bulgarian
       <td><span>windows-1251</span> <!-- Windows Vista, Chrome, and Firefox agreed -->

      <!-- bn, Bengali, uses windows-1252: Firefox and Chrome agreed -->

      <!-- br-FR, Breton (France), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- bs-Cyrl-BA, Bosnian (Cyrillic, Bosnia and Herzegovina), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1251 -->

      <!-- bs-Latn-BA, Bosnian (Latin, Bosnia and Herzegovina), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1250 -->

      <!-- ca, Catalan, uses windows-1252: Windows Vista, Chrome, and Firefox agreed -->

      <!-- co-FR, Corsican (France), uses windows-1252: Windows Vista and Firefox agreed -->

      <tr>
       <td>cs
       <td>Czech
       <td><span>windows-1250</span> <!-- Windows Vista and Chrome agreed (but disagreed with Firefox, which reportedly agreed on Windows but thought the encoding should be ISO-8859-2 on Mac and Linux) -->

      <!-- cy-GB, Welsh (United Kingdom), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- da, Danish, uses windows-1252: Windows Vista, Chrome, and Firefox agreed -->

      <!-- de, German, uses windows-1252: Windows Vista, Chrome, and Firefox agreed -->

      <!-- el, Greek, was not initially listed here because Windows Vista wanted windows-1253, Chrome wanted ISO-8859-7, and Firefox wanted ISO-8859-7 but looked liked it wanted windows-1252 -->
      <!-- el-GR, Greek (Greece), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1253 -->
      <!-- further feedback has changed this: -->
      <tr>
       <td>el
       <td>Greek
       <td><span>ISO-8859-7</span> <!-- per https://www.w3.org/Bugs/Public/show_bug.cgi?id=23090 -->

      <!-- en, English, uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- es, Spanish, uses windows-1252: Windows Vista, Chrome, and Firefox agreed -->

      <tr>
       <td>et
       <td>Estonian
       <td><span>windows-1257</span> <!-- Windows Vista and Chrome agreed -->

      <!-- eu, Basque, uses windows-1252: Windows Vista and Firefox agreed -->

      <tr>
       <td>fa
       <td>Persian
       <td><span>windows-1256</span> <!-- Windows Vista and Chrome agreed -->

      <!-- fi, Finnish, uses windows-1252: Windows Vista, Chrome, and Firefox agreed -->

      <!-- fil, Filipino, uses windows-1252: Firefox and Chrome agreed -->

      <!-- fo, Faroese, uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- fr, French, uses windows-1252: Windows Vista, Chrome, and Firefox agreed -->

      <!-- fy-NL, Frisian (Netherlands), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- ga-IE, Irish (Ireland), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- gl, Galician, uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- gsw-FR, Alsatian (France), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- gu, Gujarati, uses windows-1252: Firefox and Chrome agreed -->

      <!-- ha-Latn-NG, Hausa (Latin, Nigeria), uses windows-1252: Windows Vista and Firefox agreed -->

      <tr>
       <td>he
       <td>Hebrew
       <td><span>windows-1255</span> <!-- Windows Vista, Chrome, and Firefox agreed -->

      <!-- hi, Hindi, uses windows-1252: Firefox and Chrome agreed -->

      <tr>
       <td>hr
       <td>Croatian
       <td><span>windows-1250</span> <!-- Windows Vista and Chrome agreed -->

      <tr>
       <td>hu
       <td>Hungarian
       <td><span>ISO-8859-2</span> <!-- Chrome and Firefox agreed (but disagreed with Windows Vista, which thought the encoding should be windows-1250) -->

      <!-- hu-HU, Hungarian (Hungary), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1250 -->

      <!-- id, Indonesian, uses windows-1252: Windows Vista, Chrome, and Firefox agreed -->

      <!-- ig-NG, Igbo (Nigeria), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- is, Icelandic, uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- it, Italian, uses windows-1252: Windows Vista, Chrome, and Firefox agreed -->

      <!-- iu-Latn-CA, Inuktitut (Latin, Canada), uses windows-1252: Windows Vista and Firefox agreed -->

      <tr>
       <td>ja
       <td>Japanese
       <td><span>Shift_JIS</span> <!-- Windows Vista, Chrome, and Firefox agreed -->

      <!-- kk, Kazakh, was not initially listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1251 -->
      <!-- further feedback has changed this: -->
      <tr>
       <td>kk
       <td>Kazakh
       <td><span>windows-1251</span> <!-- per https://www.w3.org/Bugs/Public/show_bug.cgi?id=23089 -->

      <!-- kl-GL, Greenlandic (Greenland), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- kn, Kannada, uses windows-1252: Firefox and Chrome agreed -->

      <tr>
       <td>ko
       <td>Korean
       <td><span>EUC-KR</span> <!-- Windows Vista, Chrome, and Firefox agreed -->

      <tr>
       <td>ku
       <td>Kurdish
       <td><span>windows-1254</span> <!-- Best guess -->

      <!-- ky, Kyrgyz, was not initially listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1251 -->
      <!-- further feedback has changed this: -->
      <tr>
       <td>ky
       <td>Kyrgyz
       <td><span>windows-1251</span> <!-- per https://www.w3.org/Bugs/Public/show_bug.cgi?id=23089 -->

      <!-- lb-LU, Luxembourgish (Luxembourg), uses windows-1252: Windows Vista and Firefox agreed -->

      <tr>
       <td>lt
       <td>Lithuanian
       <td><span>windows-1257</span> <!-- Windows Vista, Chrome, and Windows Firefox agreed; Linux and Mac Firefox reportedly disagreed -->

      <tr>
       <td>lv
       <td>Latvian
       <td><span>windows-1257</span> <!-- Windows Vista and Chrome agreed (but disagreed with Firefox, which thought the encoding should be ISO-8859-13) -->

      <!-- mk, Macedonian, was not initially listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1251 -->
      <!-- further feedback has changed this: -->
      <tr>
       <td>mk
       <td>Macedonian
       <td><span>windows-1251</span> <!-- per https://www.w3.org/Bugs/Public/show_bug.cgi?id=23089 -->

      <!-- ml, Malayalam, uses windows-1252: Firefox and Chrome agreed -->

      <!-- mn, Mongolian, is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1251 -->

      <!-- moh-CA, Mohawk (Mohawk), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- mr, Marathi, uses windows-1252: Firefox and Chrome agreed -->

      <!-- ms, Malay, uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- nb, Norwegian Bokm&aring;l, uses windows-1252: Firefox and Chrome agreed -->

      <!-- nl, Dutch, uses windows-1252: Windows Vista, Chrome, and Firefox agreed -->

      <!-- nn-NO, Norwegian, Nynorsk (Norway), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- no, Norwegian, uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- nso-ZA, Sesotho sa Leboa (South Africa), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- oc-FR, Occitan (France), uses windows-1252: Windows Vista and Firefox agreed -->

      <tr>
       <td>pl
       <td>Polish
       <td><span>ISO-8859-2</span> <!-- Chrome and Firefox agreed (but disagreed with Windows Vista, which thought the encoding should be windows-1250) -->

      <!-- pl-PL, Polish (Poland), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1250 -->

      <!-- prs-AF, Dari (Afghanistan), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1256 -->

      <!-- pt, Portuguese, uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- qut-GT, K'iche (Guatemala), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- quz-BO, Quechua (Bolivia), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- quz-EC, Quechua (Ecuador), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- quz-PE, Quechua (Peru), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- rm-CH, Romansh (Switzerland), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- ro, Romanian, is not listed here because Windows Vista wanted windows-1250, Chrome wanted ISO-8859-2, and Firefox wanted <none> -->

      <!-- ro-RO, Romanian (Romania), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1250 -->

      <tr>
       <td>ru
       <td>Russian
       <td><span>windows-1251</span> <!-- Windows Vista, Chrome, and Firefox agreed -->

      <!-- rw-RW, Kinyarwanda (Rwanda), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- sah-RU, Yakut (Russia), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1251 -->
      <!-- sah wasn't listed at all because none of the sources knew about it. However, further feedback has changed this: -->
      <tr>
       <td>sah
       <td>Yakut
       <td><span>windows-1251</span> <!-- per https://www.w3.org/Bugs/Public/show_bug.cgi?id=23089 -->

      <!-- se-FI, Sami, Northern (Finland), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- se-NO, Sami, Northern (Norway), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- se-SE, Sami, Northern (Sweden), uses windows-1252: Windows Vista and Firefox agreed -->

      <tr>
       <td>sk
       <td>Slovak
       <td><span>windows-1250</span> <!-- Windows Vista, Chrome, and Firefox agreed -->

      <tr>
       <td>sl
       <td>Slovenian
       <td><span>ISO-8859-2</span> <!-- Chrome and Firefox agreed (but disagreed with Windows Vista, which thought the encoding should be windows-1250) -->

      <!-- sl-SI, Slovenian (Slovenia), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1250 -->

      <!-- sma-NO, Sami, Southern (Norway), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- sma-SE, Sami, Southern (Sweden), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- smj-NO, Sami, Lule (Norway), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- smj-SE, Sami, Lule (Sweden), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- smn-FI, Sami, Inari (Finland), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- sms-FI, Sami, Skolt (Finland), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- sq, Albanian, is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1250 -->

      <tr>
       <td>sr
       <td>Serbian
       <td><span>windows-1251</span> <!-- Windows Vista and Chrome agreed -->

      <!-- sr-Latn-BA, Serbian (Latin, Bosnia and Herzegovina), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1250 -->

      <!-- sr-Latn-SP, Serbian (Latin, Serbia), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1250 -->

      <!-- sv, Swedish, uses windows-1252: Windows Vista, Chrome, and Firefox agreed -->

      <!-- sw, Kiswahili, uses windows-1252: Windows Vista, Chrome, and Firefox agreed -->

      <!-- ta, Tamil, uses windows-1252: Firefox and Chrome agreed -->

      <!-- te, Telugu, uses windows-1252: Firefox and Chrome agreed -->

      <!-- tg-Cyrl-TJ, Tajik (Cyrillic, Tajikistan), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1251 -->
      <!-- tg wasn't listed at all because none of the sources knew about it. However, further feedback has changed this: -->
      <tr>
       <td>tg
       <td>Tajik
       <td><span>windows-1251</span> <!-- per https://www.w3.org/Bugs/Public/show_bug.cgi?id=23089 -->

      <tr>
       <td>th
       <td>Thai
       <td><span>windows-874</span> <!-- Windows Vista, Chrome, and Firefox agree (though Firefox didn't always) -->

      <!-- tk-TM, Turkmen (Turkmenistan), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1250 -->

      <!-- tn-ZA, Setswana (South Africa), uses windows-1252: Windows Vista and Firefox agreed -->

      <tr>
       <td>tr
       <td>Turkish
       <td><span>windows-1254</span> <!-- Windows Vista, Chrome, and Firefox agreed -->

      <!-- tt, Tatar, was not initially listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1251 -->
      <!-- further feedback has changed this: -->
      <tr>
       <td>tt
       <td>Tatar
       <td><span>windows-1251</span> <!-- per https://www.w3.org/Bugs/Public/show_bug.cgi?id=23089 -->

      <!-- tzm-Latn-DZ, Tamazight (Latin, Algeria), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- ug-CN, Uighur (PRC), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1256 -->

      <tr>
       <td>uk
       <td>Ukrainian
       <td><span>windows-1251</span> <!-- Windows Vista, Chrome, and Firefox agreed -->

      <!-- ur, Urdu, is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1256 -->

      <!-- uz, Uzbek, is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1254 -->

      <!-- uz-Cyrl-UZ, Uzbek (Cyrillic, Uzbekistan), is not listed here because neither Chrome nor Firefox knew about it. For what it's worth, Windows Vista wanted windows-1251 -->

      <tr>
       <td>vi
       <td>Vietnamese
       <td><span>windows-1258</span> <!-- Windows Vista and Chrome agreed -->

      <!-- wee-DE, Lower Sorbian (Germany), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- wen-DE, Upper Sorbian (Germany), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- wo-SN, Wolof (Senegal), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- xh-ZA, isiXhosa (South Africa), uses windows-1252: Windows Vista and Firefox agreed -->

      <!-- yo-NG, Yoruba (Nigeria), uses windows-1252: Windows Vista and Firefox agreed -->

      <tr>
       <td>zh-Hans, zh-CN, zh-SG
       <td>Chinese, Simplified
       <td><span>GBK</span> <!-- Windows Vista, Chrome, and Firefox agreed -->

      <tr>
       <td>zh-Hant, zh-HK, zh-MO, zh-TW
       <td>Chinese, Traditional
       <td><span>Big5</span> <!-- Windows Vista, Chrome, and Firefox agreed (though Firefox didn't always) -->

      <!-- zu-ZA, isiZulu (South Africa), uses windows-1252: Windows Vista and Firefox agreed -->

      <tr>
       <td colspan=2>All other locales
       <td><span>windows-1252</span>

    </table>

    <p class="tablenote"><small>The contents of this table are derived from the intersection of
    Windows, Chrome, and Firefox defaults.</small></p>
   </li>
  </ol>

  <p>The <span>document's character encoding</span> must immediately be set to the value returned
  from this algorithm, at the same time as the user agent uses the returned value to select the
  decoder to use for the input byte stream.</p>

  <hr>

  <p>When an algorithm requires a user agent to <dfn export>prescan a byte stream to determine its
  encoding</dfn>, given some defined <dfn data-x="prescan-end-condition" export for="prescan a byte
  stream to determine its encoding"><var>end condition</var></dfn>, then it must run the following
  steps. If at any point during these steps (including during instances of the <span
  data-x="concept-get-attributes-when-sniffing">get an attribute</span> algorithm invoked by this
  one) the user agent either runs out of bytes (meaning the <var>position</var> pointer created in
  the first step below goes beyond the end of the byte stream obtained so far) or reaches its
  <var>end condition</var>, then abort the <span>prescan a byte stream to determine its
  encoding</span> algorithm and return the result <span
  data-x="concept-get-xml-encoding-when-sniffing">get an XML encoding</span> applied to the same
  bytes that the <span>prescan a byte stream to determine its encoding</span> algorithm was applied
  to. Otherwise, these steps will return a character encoding.</p>

  <ol>
   <li><p>Let <var>position</var> be a pointer to a byte in the input byte stream, initially
   pointing at the first byte.</p></li>

   <li>
    <p>Prescan for UTF-16 XML declarations: If <var>position</var> points to:</p>

    <dl class="switch">
     <dt>A sequence of bytes starting with: 0x3C, 0x0, 0x3F, 0x0, 0x78, 0x0 (case-sensitive UTF-16
     little-endian '&lt;?x')</dt>
     <dd><p>Return <span>UTF-16LE</span>.</p></dd>

     <dt>A sequence of bytes starting with: 0x0, 0x3C, 0x0, 0x3F, 0x0, 0x78 (case-sensitive UTF-16
     big-endian '&lt;?x')</dt>
     <dd><p>Return <span>UTF-16BE</span>.</p></dd>
    </dl>

    <p class="note">For historical reasons, the prefix is two bytes longer than in <a
    href="https://www.w3.org/TR/REC-xml/#sec-guessing">Appendix F</a> of <cite>XML</cite> and the
    encoding name is not checked.</p>
   </li>

   <li>
    <p><i>Loop</i>: If <var>position</var> points to:</p>

    <dl class="switch">

     <dt>A sequence of bytes starting with: 0x3C 0x21 0x2D 0x2D (`<code
     data-x="">&lt;!--</code>`)</dt>
     <dd>
      <p>Advance the <var>position</var> pointer so that it points at the first 0x3E byte
      which is preceded by two 0x2D bytes (i.e. at the end of an ASCII '-->' sequence) and comes
      after the 0x3C byte that was found. (The two 0x2D bytes can be the same as those in the
      '&lt;!--' sequence.)</p>
     </dd>

     <dt>A sequence of bytes starting with: 0x3C, 0x4D or 0x6D, 0x45 or 0x65, 0x54 or 0x74, 0x41 or 0x61, and one of 0x09, 0x0A, 0x0C, 0x0D, 0x20, 0x2F (case-insensitive ASCII '&lt;meta' followed by a space or slash)</dt>
     <dd>
      <ol>
       <li><p>Advance the <var>position</var> pointer so that it points at the next 0x09,
       0x0A, 0x0C, 0x0D, 0x20, or 0x2F byte (the one in sequence of characters matched
       above).</p></li>

       <li><p>Let <var>attribute list</var> be an empty list of strings.</p></li> <!-- so
       long as we only care about http-equiv, content, and charset, this can be a 3-bit bitfield -->

       <li><p>Let <var>got pragma</var> be false.</p></li>

       <li><p>Let <var>need pragma</var> be null.</p></li>

       <li><p>Let <var>charset</var> be the null value (which, for the purposes of this
       algorithm, is distinct from an unrecognized encoding or the empty string).</p></li>

       <li><p><i>Attributes</i>: <span data-x="concept-get-attributes-when-sniffing">Get an
       attribute</span> and its value. If no attribute was sniffed, then jump to the
       <i>processing</i> step below.</p></li>

       <li><p>If the attribute's name is already in <var>attribute list</var>, then return
       to the step labeled <i>attributes</i>.</p>

       <li><p>Add the attribute's name to <var>attribute list</var>.</p>

       <li>
        <p>Run the appropriate step from the following list, if one applies:</p>

        <dl class="switch">

         <dt>If the attribute's name is "<code data-x="">http-equiv</code>"</dt>

         <dd><p>If the attribute's value is "<code data-x="">content-type</code>", then set <var>got pragma</var> to true.</p></dd>

         <dt>If the attribute's name is "<code data-x="">content</code>"</dt>

         <dd><p>Apply the <span>algorithm for extracting a character encoding from a
         <code>meta</code> element</span>, giving the attribute's value as the string to parse. If a
         character encoding is returned, and if <var>charset</var> is still set to null,
         let <var>charset</var> be the encoding returned, and set <var>need
         pragma</var> to true.</p></dd>

         <dt>If the attribute's name is "<code data-x="">charset</code>"</dt>

         <dd><p>Let <var>charset</var> be the result of <span>getting an encoding</span>
         from the attribute's value, and set <var>need pragma</var> to false.</p></dd>
        </dl>
       </li>

       <li><p>Return to the step labeled <i>attributes</i>.</p></li>

       <li><p><i>Processing</i>: If <var>need pragma</var> is null, then jump to the step
       below labeled <i>next byte</i>.</p></li>

       <li><p>If <var>need pragma</var> is true but <var>got pragma</var> is
       false, then jump to the step below labeled <i>next byte</i>.</p></li>

       <li><p>If <var>charset</var> is failure, then jump to the step below labeled <i>next
       byte</i>.</p></li>

       <!-- the next two steps are redundant with steps in the 'change the encoding' algorithm -->

       <li><p>If <var>charset</var> is <span>UTF-16BE/LE</span>, then set <var>charset</var> to
       <span>UTF-8</span>.

       <li><p>If <var>charset</var> is <span>x-user-defined</span>, then set <var>charset</var> to
       <span>windows-1252</span>.</p></li>

       <li><p>Return <var>charset</var>.</p></li>
      </ol>
     </dd>

     <dt>A sequence of bytes starting with a 0x3C byte (&lt;), optionally a 0x2F byte (/), and
     finally a byte in the range 0x41-0x5A or 0x61-0x7A (A-Z or a-z)</dt>
     <dd>
      <ol>
       <li><p>Advance the <var>position</var> pointer so that it points at the next 0x09 (HT),
       0x0A (LF), 0x0C (FF), 0x0D (CR), 0x20 (SP), or 0x3E (>) byte.</p></li>

       <li><p>Repeatedly <span data-x="concept-get-attributes-when-sniffing">get an attribute</span>
       until no further attributes can be found, then jump to the step below labeled <i>next
       byte</i>.</p></li>
      </ol>
     </dd>

     <dt>A sequence of bytes starting with: 0x3C 0x21 (`<code data-x="">&lt;!</code>`)</dt>
     <dt>A sequence of bytes starting with: 0x3C 0x2F (`<code data-x="">&lt;/</code>`)</dt>
     <dt>A sequence of bytes starting with: 0x3C 0x3F (`<code data-x="">&lt;?</code>`)</dt>
     <dd>
      <p>Advance the <var>position</var> pointer so that it points at the first 0x3E byte (>) that
      comes after the 0x3C byte that was found.</p>
     </dd>

     <dt>Any other byte</dt>
     <dd>
      <p>Do nothing with that byte.</p>
     </dd>
    </dl>
   </li>

   <li><i>Next byte</i>: Move <var>position</var> so it points at the next byte in the
   input byte stream, and return to the step above labeled <i>loop</i>.</li>
  </ol>

  <p>When the <span>prescan a byte stream to determine its encoding</span> algorithm says to <dfn
  data-x="concept-get-attributes-when-sniffing">get an attribute</dfn>, it means doing this:</p>

  <ol>
   <li><p>If the byte at <var>position</var> is one of 0x09 (HT), 0x0A (LF), 0x0C (FF), 0x0D (CR),
   0x20 (SP), or 0x2F (/) then advance <var>position</var> to the next byte and redo this
   step.</p></li>

   <li><p>If the byte at <var>position</var> is 0x3E (>), then abort the <span
   data-x="concept-get-attributes-when-sniffing">get an attribute</span> algorithm. There isn't
   one.</p></li>

   <li><p>Otherwise, the byte at <var>position</var> is the start of the attribute name.
   Let <var>attribute name</var> and <var>attribute value</var> be the empty
   string.</p></li>

   <li><p>Process the byte at <var>position</var> as follows:</p>

    <dl class="switch">

     <dt>If it is 0x3D (=), and the <var>attribute name</var> is longer than the empty string</dt>

     <dd>Advance <var>position</var> to the next byte and jump to the step below labeled
     <i>value</i>.</dd>

     <dt>If it is 0x09 (HT), 0x0A (LF), 0x0C (FF), 0x0D (CR), or 0x20 (SP)</dt>

     <dd>Jump to the step below labeled <i>spaces</i>.</dd>

     <dt>If it is 0x2F (/) or 0x3E (>)</dt>

     <dd>Abort the <span data-x="concept-get-attributes-when-sniffing">get an attribute</span>
     algorithm. The attribute's name is the value of <var>attribute name</var>, its value
     is the empty string.</dd>

     <dt>If it is in the range 0x41 (A) to 0x5A (Z)</dt>

     <dd>Append the code point <var>b</var>+0x20 to <var>attribute name</var>
     (where <var>b</var> is the value of the byte at <var>position</var>). (This converts the input
     to lowercase.)</dd>

     <dt>Anything else</dt>

     <dd>Append the code point with the same value as the byte at <var>position</var> to
     <var>attribute name</var>. (It doesn't actually matter how bytes outside the ASCII range are
     handled here, since only ASCII bytes can contribute to the detection of a character
     encoding.)</dd>
    </dl>
   </li>

   <li><p>Advance <var>position</var> to the next byte and return to the previous
   step.</p></li>

   <li><p><i>Spaces</i>: If the byte at <var>position</var> is one of 0x09 (HT), 0x0A (LF), 0x0C
   (FF), 0x0D (CR), or 0x20 (SP) then advance <var>position</var> to the next byte, then, repeat
   this step.</p></li>

   <li><p>If the byte at <var>position</var> is <em>not</em> 0x3D (=), abort the <span
   data-x="concept-get-attributes-when-sniffing">get an attribute</span> algorithm. The attribute's
   name is the value of <var>attribute name</var>, its value is the empty string.</p></li>

   <li><p>Advance <var>position</var> past the 0x3D (=) byte.</p></li>

   <li><p><i>Value</i>: If the byte at <var>position</var> is one of 0x09 (HT), 0x0A (LF), 0x0C
   (FF), 0x0D (CR), or 0x20 (SP) then advance <var>position</var> to the next byte, then, repeat
   this step.</p></li>

   <li><p>Process the byte at <var>position</var> as follows:</p>

    <dl class="switch">

     <dt>If it is 0x22 (") or 0x27 (')</dt>

     <dd>
      <ol>
       <li>Let <var>b</var> be the value of the byte at <var>position</var>.</li>

       <li><i>Quote loop</i>: Advance <var>position</var> to the next byte.</li>

       <li>If the value of the byte at <var>position</var> is the value of <var>b</var>, then advance <var>position</var> to the next byte and abort the
       "get an attribute" algorithm. The attribute's name is the value of <var>attribute
       name</var>, and its value is the value of <var>attribute value</var>.</li>

       <li>Otherwise, if the value of the byte at <var>position</var> is in the range 0x41 (A) to
       0x5A (Z), then append a code point to <var>attribute value</var> whose value is 0x20 more
       than the value of the byte at <var>position</var>.</li>

       <li>Otherwise, append a code point to <var>attribute value</var> whose value is the same as
       the value of the byte at <var>position</var>.</li>

       <li>Return to the step above labeled <i>quote loop</i>.</li>
      </ol>
     </dd>

     <dt>If it is 0x3E (>)</dt>

     <dd>Abort the <span data-x="concept-get-attributes-when-sniffing">get an attribute</span>
     algorithm. The attribute's name is the value of <var>attribute name</var>, its value
     is the empty string.</dd>


     <dt>If it is in the range 0x41 (A) to 0x5A (Z)</dt>

     <dd>Append a code point <var>b</var>+0x20 to <var>attribute value</var>
     (where <var>b</var> is the value of the byte at <var>position</var>). Advance
     <var>position</var> to the next byte.</dd>

     <dt>Anything else</dt>

     <dd>Append a code point with the same value as the byte at <var>position</var> to
     <var>attribute value</var>. Advance <var>position</var> to the next byte.</dd>
    </dl>
   </li>

   <li><p>Process the byte at <var>position</var> as
   follows:</p>

    <dl class="switch">

     <dt>If it is 0x09 (HT), 0x0A (LF), 0x0C (FF), 0x0D (CR), 0x20 (SP), or 0x3E (>)</dt>

     <dd>Abort the <span data-x="concept-get-attributes-when-sniffing">get an attribute</span>
     algorithm. The attribute's name is the value of <var>attribute name</var> and its
     value is the value of <var>attribute value</var>.</dd>

     <dt>If it is in the range 0x41 (A) to 0x5A (Z)</dt>

     <dd>Append a code point <var>b</var>+0x20 to <var>attribute value</var>
     (where <var>b</var> is the value of the byte at <var>position</var>).</dd>

     <dt>Anything else</dt>

     <dd>Append a code point with the same value as the byte at <var>position</var> to
     <var>attribute value</var>.</dd>
    </dl>
   </li>

   <li><p>Advance <var>position</var> to the next byte and return to the previous
   step.</p></li>
  </ol>

  <p>When the <span>prescan a byte stream to determine its encoding</span> algorithm is aborted
  without returning an encoding, <dfn data-x="concept-get-xml-encoding-when-sniffing">get an XML
  encoding</dfn> means doing this.</p>

  <p class="note">Looking for syntax resembling an XML declaration, even in <code>text/html</code>,
  is necessary for compatibility with existing content.</p>

  <ol>
   <li><p>Let <var>encodingPosition</var> be a pointer to the start of the stream.</p></li>

   <li><p>If <var>encodingPosition</var> does not point to the start of a byte sequence 0x3C, 0x3F,
   0x78, 0x6D, 0x6C (`<code data-x="">&lt;?xml</code>`), then return failure.</p></li>

   <li><p>Let <var>xmlDeclarationEnd</var> be a pointer to the next byte in the input byte stream
   which is 0x3E (>). If there is no such byte, then return failure.</p></li>

   <li><p>Set <var>encodingPosition</var> to the position of the first occurrence of the subsequence
   of bytes 0x65, 0x6E, 0x63, 0x6F, 0x64, 0x69, 0x6E, 0x67 (`<code data-x="">encoding</code>`) at or
   after the current <var>encodingPosition</var>. If there is no such sequence, then return
   failure.</p></li>

   <li><p>Advance <var>encodingPosition</var> past the 0x67 (g) byte.</p></li>

   <li><p>While the byte at <var>encodingPosition</var> is less than or equal to 0x20 (i.e., it is
   either an ASCII space or control character), advance <var>encodingPosition</var> to the next
   byte.</p></li>

   <li><p>If the byte at <var>encodingPosition</var> is not 0x3D (=), then return failure.</p></li>

   <li><p>Advance <var>encodingPosition</var> to the next byte.</p></li>

   <li><p>While the byte at <var>encodingPosition</var> is less than or equal to 0x20 (i.e., it is
   either an ASCII space or control character), advance <var>encodingPosition</var> to the next
   byte.</p></li>

   <li><p>Let <var>quoteMark</var> be the byte at <var>encodingPosition</var>.</p></li>

   <li><p>If <var>quoteMark</var> is not either 0x22 (") or 0x27 ('), then return failure.</p></li>

   <li><p>Advance <var>encodingPosition</var> to the next byte.</p></li>

   <li><p>Let <var>encodingEndPosition</var> be the position of the next occurrence of
   <var>quoteMark</var> at or after <var>encodingPosition</var>. If <var>quoteMark</var> does not
   occur again, then return failure.</p></li>

   <li><p>Let <var>potentialEncoding</var> be the sequence of the bytes between
   <var>encodingPosition</var> (inclusive) and <var>encodingEndPosition</var> (exclusive).</p></li>

   <li><p>If <var>potentialEncoding</var> contains one or more bytes whose byte value is 0x20 or
   below, then return failure.</p></li>

   <li><p>Let <var>encoding</var> be the result of <span>getting an encoding</span> given
   <var>potentialEncoding</var> <span data-x="isomorphic decode">isomorphic decoded</span>.</p></li>

   <li><p>If the <var>encoding</var> is <span>UTF-16BE/LE</span>, then change it to
   <span>UTF-8</span>.</p></li>

   <li><p>Return <var>encoding</var>.</p></li>
  </ol>

  <p>For the sake of interoperability, user agents should not use a pre-scan algorithm that returns
  different results than the one described above. (But, if you do, please at least let us know, so
  that we can improve this algorithm and benefit everyone...)</p>



  <h5>Character encodings</h5>

  <p>User agents must support the encodings defined in <cite>Encoding</cite>, including, but
  not limited to,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#utf-8">UTF-8</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#iso-8859-2">ISO-8859-2</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#iso-8859-7">ISO-8859-7</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#iso-8859-8">ISO-8859-8</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#windows-874">windows-874</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#windows-1250">windows-1250</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#windows-1251">windows-1251</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#windows-1252">windows-1252</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#windows-1254">windows-1254</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#windows-1255">windows-1255</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#windows-1256">windows-1256</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#windows-1257">windows-1257</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#windows-1258">windows-1258</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#gbk">GBK</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#big5">Big5</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#iso-2022-jp">ISO-2022-JP</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#shift_jis">Shift_JIS</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#euc-kr">EUC-KR</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#utf-16be">UTF-16BE</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#utf-16le">UTF-16LE</dfn>,
  <dfn data-x-href="https://encoding.spec.whatwg.org/#utf-16be-le"
  id="utf-16-encoding">UTF-16BE/LE</dfn>, and
  <dfn data-x-href="https://encoding.spec.whatwg.org/#x-user-defined">x-user-defined</dfn>.
  User agents must not support other encodings.</p>

  <p class="note">The above prohibits supporting, for example, CESU-8, UTF-7, BOCU-1, SCSU, EBCDIC,
  and UTF-32. This specification does not make any attempt to support prohibited encodings in its
  algorithms; support and use of prohibited encodings would thus lead to unexpected behavior.
  <ref>CESU8</ref> <ref>UTF7</ref> <ref>BOCU1</ref> <ref>SCSU</ref></p>


  <h5>Changing the encoding while parsing</h5>

  <p>When the parser requires the user agent to <dfn>change the encoding</dfn>, it must run the
  following steps. This might happen if the <span>encoding sniffing algorithm</span> described above
  failed to find a character encoding, or if it found a character encoding that was not the actual
  encoding of the file.</p>

  <ol>
   <li><p>If the encoding that is already being used to interpret the input stream is
   <span>UTF-16BE/LE</span>, then set the <span
   data-x="concept-encoding-confidence">confidence</span> to <i>certain</i> and return. The new
   encoding is ignored; if it was anything but the same encoding, then it would be clearly
   incorrect.</p></li>

   <!-- the next two steps are redundant with similar logic in the sniffer -->
   <!-- if you add anything else here, then factor it out into a common algorithm -->

   <li><p>If the new encoding is <span>UTF-16BE/LE</span>, then change it to
   <span>UTF-8</span>.</p></li>

   <li><p>If the new encoding is <span>x-user-defined</span>, then change it to
   <span>windows-1252</span>.</p></li> <!-- apparently this was a Chrome invention, later picked up
   by Mozilla -->

   <li><p>If the new encoding is identical or equivalent to the encoding that is already being used
   to interpret the input stream, then set the <span
   data-x="concept-encoding-confidence">confidence</span> to <i>certain</i> and return.
   This happens when the encoding information found in the file matches what the <span>encoding
   sniffing algorithm</span> determined to be the encoding, and in the second pass through the
   parser if the first pass found that the encoding sniffing algorithm described in the earlier
   section failed to find the right encoding.</p></li>

   <li><p>If all the bytes up to the last byte converted by the current decoder have the same
   Unicode interpretations in both the current encoding and the new encoding, and if the user agent
   supports changing the converter on the fly, then the user agent may change to the new converter
   for the encoding on the fly. Set the <span>document's character encoding</span> and the encoding
   used to convert the input stream to the new encoding, set the <span
   data-x="concept-encoding-confidence">confidence</span> to <i>certain</i>, and return.</p></li>

   <li><p>Otherwise, restart the <span>navigate</span><!--DONAV reparse--> algorithm, with <i
   data-x="navigation-hh">historyHandling</i> set to "<code
   data-x="NavigationHistoryBehavior-replace">replace</code>" and other inputs kept the same, but
   this time skip the <span>encoding sniffing algorithm</span> and instead just set the encoding to
   the new encoding and the <span data-x="concept-encoding-confidence">confidence</span> to
   <i>certain</i>. Whenever possible, this should be done without actually contacting the network
   layer (the bytes should be re-parsed from memory), even if, e.g., the document is marked as not
   being cacheable. If this is not possible and contacting the network layer would involve repeating
   a request that uses a method other than `<code data-x="">GET</code>`, then instead set the <span
   data-x="concept-encoding-confidence">confidence</span> to <i>certain</i> and ignore the new
   encoding. The resource will be misinterpreted. User agents may notify the user of the situation,
   to aid in application development.</p></li>
  </ol>

  <p class="note">This algorithm is only invoked when a new encoding is found declared on a
  <code>meta</code> element.</p> <!-- this is important for the x-user-defined stuff in particular
  -->


  <h5>Preprocessing the input stream</h5>

  <p>The <dfn>input stream</dfn> consists of the characters pushed into it as the <span>input byte
  stream</span> is decoded or from the various APIs that directly manipulate the input stream.</p>

  <p>Any occurrences of <span data-x="surrogate">surrogates</span> are <span
  data-x="parse-error-surrogate-in-input-stream">surrogate-in-input-stream</span>
  <span data-x="parse error">parse errors</span>. Any occurrences of <span
  data-x="noncharacter">noncharacters</span> are <span
  data-x="parse-error-noncharacter-in-input-stream">noncharacter-in-input-stream</span>
  <span data-x="parse error">parse errors</span> and any occurrences of <span
  data-x="control">controls</span> other than <span>ASCII whitespace</span> and U+0000 NULL
  characters are <span
  data-x="parse-error-control-character-in-input-stream">control-character-in-input-stream</span>
  <span data-x="parse error">parse errors</span>.</p>

  <p class="note">The handling of U+0000 NULL characters varies based on where the characters are
  found and happens at the later stages of the parsing. They are either ignored or, for security
  reasons, replaced with a U+FFFD REPLACEMENT CHARACTER. This handling is, by necessity, spread
  across both the tokenization stage and the tree construction stage.</p>

  <p>Before the <span>tokenization</span> stage, the input stream must be preprocessed by <span
  data-x="normalize newlines">normalizing newlines</span>. Thus, newlines in HTML DOMs are
  represented by U+000A LF characters, and there are never any U+000D CR characters in the input to
  the <span>tokenization</span> stage.</p>

  <p>The <dfn>next input character</dfn> is the first character in the <span>input stream</span>
  that has not yet been <dfn data-x="">consumed</dfn> or explicitly ignored by the requirements in
  this section. Initially, the <i data-x="next input character">next input character</i> is the
  first character in the input. The <dfn>current input character</dfn> is the last character to have
  been <i>consumed</i>.</p>

  <p>The <dfn>insertion point</dfn> is the position (just before a character or just before the end
  of the input stream) where content inserted using <code
  data-x="dom-document-write">document.write()</code> is actually inserted. The insertion point is
  relative to the position of the character immediately after it, it is not an absolute offset into
  the input stream. Initially, the insertion point is undefined.</p>

  <p>The "EOF" character in the tables below is a conceptual character representing the end of the
  <span>input stream</span>. If the parser is a <span>script-created parser</span>, then the end of
  the <span>input stream</span> is reached when an <dfn>explicit "EOF" character</dfn> (inserted by
  the <code data-x="dom-document-close">document.close()</code> method) is consumed. Otherwise, the
  "EOF" character is not a real character in the stream, but rather the lack of any further
  characters.</p>

  </div>


  <div w-nodev>

  <h4>Parse state</h4>

  <h5>The insertion mode</h5>

  <p>The <dfn>insertion mode</dfn> is a state variable that controls the primary operation of the
  tree construction stage.</p>

  <p>Initially, the <span>insertion mode</span> is "<span data-x="insertion mode:
  initial">initial</span>". It can change to "<span data-x="insertion mode: before html">before
  html</span>", "<span data-x="insertion mode: before head">before head</span>", "<span
  data-x="insertion mode: in head">in head</span>", "<span data-x="insertion mode: in head
  noscript">in head noscript</span>", "<span data-x="insertion mode: after head">after head</span>",
  "<span data-x="insertion mode: in body">in body</span>", "<span data-x="insertion mode:
  text">text</span>", "<span data-x="insertion mode: in table">in table</span>", "<span
  data-x="insertion mode: in table text">in table text</span>", "<span data-x="insertion mode: in
  caption">in caption</span>", "<span data-x="insertion mode: in column group">in column
  group</span>", "<span data-x="insertion mode: in table body">in table body</span>", "<span
  data-x="insertion mode: in row">in row</span>", "<span data-x="insertion mode: in cell">in
  cell</span>", "<span data-x="insertion mode: in select">in select</span>", "<span data-x="insertion
  mode: in select in table">in select in table</span>", "<span data-x="insertion mode: in
  template">in template</span>", "<span data-x="insertion mode: after body">after body</span>",
  "<span data-x="insertion mode: in frameset">in frameset</span>", "<span data-x="insertion mode:
  after frameset">after frameset</span>", "<span data-x="insertion mode: after after body">after
  after body</span>", and "<span data-x="insertion mode: after after frameset">after after
  frameset</span>" during the course of the parsing, as described in the <span>tree
  construction</span> stage. The insertion mode affects how tokens are processed and whether CDATA
  sections are supported.</p>

  <p>Several of these modes, namely "<span data-x="insertion mode: in head">in head</span>", "<span
  data-x="insertion mode: in body">in body</span>", "<span data-x="insertion mode: in table">in
  table</span>", and "<span data-x="insertion mode: in select">in select</span>", are special, in
  that the other modes defer to them at various times. When the algorithm below says that the user
  agent is to do something "<dfn>using the rules for</dfn> the <var>m</var> insertion
  mode", where <var>m</var> is one of these modes, the user agent must use the rules
  described under the <var>m</var> <span>insertion mode</span>'s section, but must leave
  the <span>insertion mode</span> unchanged unless the rules in <var>m</var> themselves
  switch the <span>insertion mode</span> to a new value.</p>

  <p>When the insertion mode is switched to "<span data-x="insertion mode: text">text</span>" or
  "<span data-x="insertion mode: in table text">in table text</span>", the <dfn>original insertion
  mode</dfn> is also set. This is the insertion mode to which the tree construction stage will
  return.</p>

  <p>Similarly, to parse nested <code>template</code> elements, a <dfn>stack of template insertion
  modes</dfn> is used. It is initially empty. The <dfn>current template insertion mode</dfn> is the
  insertion mode that was most recently added to the <span>stack of template insertion modes</span>.
  The algorithms in the sections below will <i>push</i> insertion modes onto this stack, meaning
  that the specified insertion mode is to be added to the stack, and <i>pop</i> insertion modes from
  the stack, which means that the most recently added insertion mode must be removed from the
  stack.</p>

  <hr>

  <p>When the steps below require the UA to <dfn>reset the insertion mode appropriately</dfn>, it
  means the UA must follow these steps:</p>

  <ol>
   <li><p>Let <var>last</var> be false.</p></li>

   <li><p>Let <var>node</var> be the last node in the <span>stack of open
   elements</span>.</p></li>

   <li><p><i>Loop</i>: If <var>node</var> is the first node in the stack of open elements, then set
   <var>last</var> to true, and, if the parser was created as part of the <span>HTML fragment
   parsing algorithm</span> (<span>fragment case</span>), set <var>node</var> to the <var
   data-x="concept-frag-parse-context">context</var> element passed to that algorithm.</p></li>

   <li>
    <p>If <var>node</var> is a <code>select</code> element, run these substeps:</p>

    <ol>
     <li><p>If <var>last</var> is true, jump to the step below labeled
     <i>done</i>.</p></li>

     <li><p>Let <var>ancestor</var> be <var>node</var>.</p></li>

     <li><p><i>Loop</i>: If <var>ancestor</var> is the first node in the <span>stack of
     open elements</span>, jump to the step below labeled <i>done</i>.</p></li>

     <li><p>Let <var>ancestor</var> be the node before <var>ancestor</var> in the
     <span>stack of open elements</span>.</p></li>

     <li><p>If <var>ancestor</var> is a <code>template</code> node, jump to the step below
     labeled <i>done</i>.</p></li>

     <li><p>If <var>ancestor</var> is a <code>table</code> node, switch the <span>insertion
     mode</span> to "<span data-x="insertion mode: in select in table">in select in table</span>"
     and return.</p></li> <!-- consider
     <table><tr><td><select><template></template><caption></table>
     https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2374 -->

     <li><p>Jump back to the step labeled <i>loop</i>.</p></li>

     <li><p><i>Done</i>: Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in
     select">in select</span>" and return.</p></li>
    </ol>
   </li>

   <li><p>If <var>node</var> is a <code>td</code> or <code>th</code> element and <var>last</var> is
   false, then switch the <span>insertion mode</span> to "<span data-x="insertion mode: in cell">in
   cell</span>" and return.</p></li>

   <li><p>If <var>node</var> is a <code>tr</code> element, then switch the <span>insertion
   mode</span> to "<span data-x="insertion mode: in row">in row</span>" and return.</p></li>

   <li><p>If <var>node</var> is a <code>tbody</code>, <code>thead</code>, or
   <code>tfoot</code> element, then switch the <span>insertion mode</span> to "<span
   data-x="insertion mode: in table body">in table body</span>" and return.</p></li>

   <li><p>If <var>node</var> is a <code>caption</code> element, then switch the
   <span>insertion mode</span> to "<span data-x="insertion mode: in caption">in caption</span>" and
   return.</p></li>

   <li><p>If <var>node</var> is a <code>colgroup</code> element, then switch the
   <span>insertion mode</span> to "<span data-x="insertion mode: in column group">in column
   group</span>" and return.</p></li>

   <li><p>If <var>node</var> is a <code>table</code> element, then switch the
   <span>insertion mode</span> to "<span data-x="insertion mode: in table">in table</span>" and
   return.</p></li>

   <li><p>If <var>node</var> is a <code>template</code> element, then switch the
   <span>insertion mode</span> to the <span>current template insertion mode</span> and
   return.</p></li>

   <!-- <li><p>If <var>node</var> is a <code>head</code> element and <var>last</var> is true, then switch the <span>insertion mode</span> to "<span
   data-x="insertion mode: in body">in body</span>" ("<span data-x="insertion mode: in body">in
   body</span>"! <em> not "<span data-x="insertion mode: in head">in head</span>"</em>!) and return.
   (<span>fragment case</span>)</p></li> --><!-- The above is only here in case people
   think that the spec accidentally omitted it and try to "fix" it. Note that noscript-in-head is
   also handled this way. This is all intentional. The only thing it doesn't handle is the
   scripting-disabled fragment parsing case for a <head> element containing a <noscript> which
   itself contains something other than a <link> or a <style> element; you'd expect that to break
   out of the <noscript> but it doesn't. This is an edge case that doesn't affect the spec, since
   the algorithm for fragment parsing is only used for innerHTML/outerHTML/insertAdjacentHTML(),
   where we know scripting is enabled. -->

   <li><p>If <var>node</var> is a <code>head</code> element and <var>last</var> is
   false, then switch the <span>insertion mode</span> to "<span data-x="insertion mode: in head">in
   head</span>" and return.</p></li> <!-- for the case of <head><template></template>...
   -->

   <li><p>If <var>node</var> is a <code>body</code> element, then switch the
   <span>insertion mode</span> to "<span data-x="insertion mode: in body">in body</span>" and
   return.</p></li>

   <li><p>If <var>node</var> is a <code>frameset</code> element, then switch the
   <span>insertion mode</span> to "<span data-x="insertion mode: in frameset">in frameset</span>" and
   return. (<span>fragment case</span>)</p></li>

   <li>
    <p>If <var>node</var> is an <code>html</code> element, run these substeps:</p>

    <ol>
     <li><p>If the <span><code>head</code> element pointer</span> is null, switch the
     <span>insertion mode</span> to "<span data-x="insertion mode: before head">before head</span>"
     and return. (<span>fragment case</span>)</p></li>

     <li><p>Otherwise, the <span><code>head</code> element pointer</span> is not null, switch the
     <span>insertion mode</span> to "<span data-x="insertion mode: after head">after head</span>" and
     return.</p></li> <!-- consider <html><head></head><template></template> -->
    </ol>
   </li>

   <li><p>If <var>last</var> is true, then switch the <span>insertion mode</span> to "<span
   data-x="insertion mode: in body">in body</span>" and return. (<span>fragment
   case</span>)</p></li>

   <li><p>Let <var>node</var> now be the node before <var>node</var> in the
   <span>stack of open elements</span>.</p></li>

   <li><p>Return to the step labeled <i>loop</i>.</p></li>
  </ol>


  <h5>The stack of open elements</h5>

  <p>Initially, the <dfn>stack of open elements</dfn> is empty. The stack grows downwards; the
  topmost node on the stack is the first one added to the stack, and the bottommost node of the
  stack is the most recently added node in the stack (notwithstanding when the stack is manipulated
  in a random access fashion as part of <a href="#adoptionAgency">the handling for misnested
  tags</a>).</p>

  <p class="note">The "<span data-x="insertion mode: before html">before html</span>"
  <span>insertion mode</span> creates the <code>html</code> <span>document element</span>, which is
  then added to the stack.</p>

  <p class="note">In the <span>fragment case</span>, the <span>stack of open elements</span> is
  initialized to contain an <code>html</code> element that is created as part of <span data-x="html
  fragment parsing algorithm">that algorithm</span>. (The <span>fragment case</span> skips the
  "<span data-x="insertion mode: before html">before html</span>" <span>insertion mode</span>.)</p>

  <p>The <code>html</code> node, however it is created, is the topmost node of the stack. It only
  gets popped off the stack when the parser <span data-x="stop parsing">finishes</span>.</p>

  <p>The <dfn>current node</dfn> is the bottommost node in this <span>stack of open
  elements</span>.</p>

  <p>The <dfn>adjusted current node</dfn> is the <i data-x="concept-frag-parse-context">context</i>
  element if the parser was created as part of the <span>HTML fragment parsing algorithm</span> and
  the <span>stack of open elements</span> has only one element in it (<span>fragment case</span>);
  otherwise, the <span>adjusted current node</span> is the <span>current node</span>.</p>

  <p>When the <span>current node</span> is removed from the <span>stack of open elements</span>,
  <span>process internal resource links</span> given the <span>current node</span>'s
  <span>node document</span>.</p>

  <p>Elements in the <span>stack of open elements</span> fall into the following categories:</p>

  <dl>
   <dt><dfn>Special</dfn></dt>
   <dd>
    <p>The following elements have varying levels of special parsing rules: HTML's
    <code>address</code>, <code>applet</code>, <code>area</code>, <code>article</code>,
    <code>aside</code>, <code>base</code>, <code>basefont</code>, <code>bgsound</code>,
    <code>blockquote</code>, <code>body</code>, <code>br</code>, <code>button</code>,
    <code>caption</code>, <code>center</code>, <code>col</code>, <code>colgroup</code>,
    <code>dd</code>, <code>details</code>, <code>dir</code>, <code>div</code>, <code>dl</code>,
    <code>dt</code>, <code>embed</code>, <code>fieldset</code>, <code>figcaption</code>,
    <code>figure</code>, <code>footer</code>, <code>form</code>, <code>frame</code>,
    <code>frameset</code>, <code>h1</code>, <code>h2</code>, <code>h3</code>, <code>h4</code>,
    <code>h5</code>, <code>h6</code>, <code>head</code>, <code>header</code>, <code>hgroup</code>,
    <code>hr</code>, <code>html</code>, <code>iframe</code>, <!-- <code>image</code>, (commented
    out because this isn't an element that can end up on the stack, so it doesn't matter) -->
    <code>img</code>, <code>input</code>, <code>keygen</code>, <code>li</code>, <code>link</code>,
    <code>listing</code>, <code>main</code>, <code>marquee</code>, <code>menu</code>,
    <code>meta</code>, <code>nav</code>, <code>noembed</code>, <code>noframes</code>,
    <code>noscript</code>, <code>object</code>, <code>ol</code>, <code>p</code>,
    <code>param</code>, <code>plaintext</code>, <code>pre</code>, <code>script</code>,
    <code>search</code>, <code>section</code>, <code>select</code>, <code>source</code>,
    <code>style</code>, <code>summary</code>, <code>table</code>, <code>tbody</code>,
    <code>td</code>, <code>template</code>, <code>textarea</code>, <code>tfoot</code>,
    <code>th</code>, <code>thead</code>, <code>title</code>, <code>tr</code>, <code>track</code>,
    <code>ul</code>, <code>wbr</code>, <code>xmp</code>; <span>MathML <code>mi</code></span>,
    <span>MathML <code>mo</code></span>, <span>MathML <code>mn</code></span>, <span>MathML
    <code>ms</code></span>, <span>MathML <code>mtext</code></span>, and <span>MathML
    <code>annotation-xml</code></span>; and <span>SVG <code>foreignObject</code></span>, <span>SVG
    <code>desc</code></span>, and <span>SVG <code>title</code></span>.</p> <!-- we could actually
    put all non-HTML elements in this list, I think -->

    <p class="note">An <code data-x="">image</code> start tag token is handled by the tree builder,
    but it is not in this list because it is not an element; it gets turned into an <code>img</code>
    element.</p>
   </dd>

   <dt><dfn>Formatting</dfn></dt>
   <dd><p>The following HTML elements are those that end up in the <span>list of active formatting
   elements</span>: <code>a</code>, <code>b</code>, <code>big</code>, <code>code</code>,
   <code>em</code>, <code>font</code>, <code>i</code>, <code>nobr</code>, <code>s</code>,
   <code>small</code>, <code>strike</code>, <code>strong</code>, <code>tt</code>, and
   <code>u</code>.</p></dd>

   <dt><dfn>Ordinary</dfn></dt>
   <dd><p>All other elements found while parsing an HTML document.</p></dd>
  </dl>

  <p class="note">Typically, the <span>special</span> elements have the start and end tag tokens
  handled specifically, while <span>ordinary</span> elements' tokens fall into "any other start tag"
  and "any other end tag" clauses, and some parts of the tree builder check if a particular element
  in the <span>stack of open elements</span> is in the <span>special</span> category. However, some
  elements (e.g., the <code>option</code> element) have their start or end tag tokens handled
  specifically, but are still not in the <span>special</span> category, so that they get the
  <span>ordinary</span> handling elsewhere.</p>

  <p>The <span>stack of open elements</span> is said to <dfn data-x="has an element in the specific
  scope">have an element <var>target node</var> in a specific scope</dfn> consisting of a
  list of element types <var>list</var> when the following algorithm terminates in a match
  state:</p>

  <ol>
   <li><p>Initialize <var>node</var> to be the <span>current node</span> (the bottommost
   node of the stack).</p></li>

   <li><p>If <var>node</var> is <var>target node</var>, terminate in a match state.</p></li>

   <li><p>Otherwise, if <var>node</var> is one of the element types in <var>list</var>, terminate in a failure state.</p></li>

   <li><p>Otherwise, set <var>node</var> to the previous entry in the <span>stack of open
   elements</span> and return to step 2. (This will never fail, since the loop will always terminate
   in the previous step if the top of the stack &mdash; an <code>html</code> element &mdash; is
   reached.)</p></li>
  </ol>

  <p>The <span>stack of open elements</span> is said to <dfn data-x="has an element in scope">have a
  particular element in scope</dfn> when it <span data-x="has an element in the specific scope">has
  that element in the specific scope</span> consisting of the following element types:</p>

  <ul class="brief">
   <li><code>applet</code></li>
   <li><code>caption</code></li>
   <li><code>html</code></li> <!-- (This can only happen if the <var>node</var> is the topmost node of the <span>stack of open elements</span>, and prevents the next step from being invoked if there are no more elements in the stack.) -->
   <li><code>table</code></li>
   <li><code>td</code></li>
   <li><code>th</code></li>
   <li><code>marquee</code></li>
   <li><code>object</code></li>
   <li><code>template</code></li>
   <li><span>MathML <code>mi</code></span></li>
   <li><span>MathML <code>mo</code></span></li>
   <li><span>MathML <code>mn</code></span></li>
   <li><span>MathML <code>ms</code></span></li>
   <li><span>MathML <code>mtext</code></span></li>
   <li><span>MathML <code>annotation-xml</code></span></li>
   <li><span>SVG <code>foreignObject</code></span></li>
   <li><span>SVG <code>desc</code></span></li>
   <li><span>SVG <code>title</code></span></li>
  </ul>

  <p>The <span>stack of open elements</span> is said to <dfn data-x="has an element in list item
  scope">have a particular element in list item scope</dfn> when it <span data-x="has an element in
  the specific scope">has that element in the specific scope</span> consisting of the following
  element types:</p>

  <ul class="brief">
   <li>All the element types listed above for the <i data-x="has an element in scope">has an element in scope</i> algorithm.</li>
   <li><code>ol</code> in the <span>HTML namespace</span></li>
   <li><code>ul</code> in the <span>HTML namespace</span></li>
  </ul>

  <p>The <span>stack of open elements</span> is said to <dfn data-x="has an element in button
  scope">have a particular element in button scope</dfn> when it <span data-x="has an element in the
  specific scope">has that element in the specific scope</span> consisting of the following element
  types:</p>

  <ul class="brief">
   <li>All the element types listed above for the <i data-x="has an element in scope">has an element in scope</i> algorithm.</li>
   <li><code>button</code> in the <span>HTML namespace</span></li>
  </ul>

  <p>The <span>stack of open elements</span> is said to <dfn data-x="has an element in table
  scope">have a particular element in table scope</dfn> when it <span data-x="has an element in the
  specific scope">has that element in the specific scope</span> consisting of the following element
  types:</p>

  <ul class="brief">
   <li><code>html</code> in the <span>HTML namespace</span></li> <!-- (This can only happen if the
   <var>node</var> is the topmost node of the <span>stack of open elements</span>, and
   prevents the next step from being invoked if there are no more elements in the stack.) -->
   <li><code>table</code> in the <span>HTML namespace</span></li>
   <li><code>template</code> in the <span>HTML namespace</span></li>
  </ul>

  <p>The <span>stack of open elements</span> is said to <dfn data-x="has an element in select
  scope">have a particular element in select scope</dfn> when it <span data-x="has an element in the
  specific scope">has that element in the specific scope</span> consisting of all element types
  <em>except</em> the following:</p>

  <ul class="brief">
   <!--<li><code>select</code> in the <span>HTML namespace</span></li>-->
   <li><code>optgroup</code> in the <span>HTML namespace</span></li>
   <li><code>option</code> in the <span>HTML namespace</span></li>
  </ul>

  <p>Nothing happens if at any time any of the elements in the <span>stack of open elements</span>
  are moved to a new location in, or removed from, the <code>Document</code> tree. In particular,
  the stack is not changed in this situation. This can cause, amongst other strange effects, content
  to be appended to nodes that are no longer in the DOM.</p>

  <p class="note">In some cases (namely, when <a href="#adoptionAgency">closing misnested formatting
  elements</a>), the stack is manipulated in a random-access fashion.</p>


  <h5>The list of active formatting elements</h5>

  <p>Initially, the <dfn>list of active formatting elements</dfn> is empty. It is used to handle
  mis-nested <span data-x="formatting">formatting element tags</span>.</p>

  <p>The list contains elements in the <span>formatting</span> category, and <span
  data-x="concept-parser-marker">markers</span>. The <dfn
  data-x="concept-parser-marker">markers</dfn> are inserted when entering <code>applet</code>,
  <code>object</code>, <code>marquee</code>, <code>template</code>, <code>td</code>,
  <code>th</code>, and <code>caption</code> elements, and are used to prevent formatting from
  "leaking" <em>into</em> <code>applet</code>, <code>object</code>, <code>marquee</code>,
  <code>template</code>, <code>td</code>, <code>th</code>, and <code>caption</code> elements.</p>

  <p>In addition, each element in the <span>list of active formatting elements</span> is associated
  with the token for which it was created, so that further elements can be created for that token if
  necessary.</p>

  <p>When the steps below require the UA to <dfn>push onto the list of active formatting
  elements</dfn> an element <var>element</var>, the UA must perform the following
  steps:</p>

  <ol id="noah">

   <li><p>If there are already three elements in the <span>list of active formatting elements</span>
   after the last <span data-x="concept-parser-marker">marker</span>, if any, or anywhere in the
   list if there are no <span data-x="concept-parser-marker">markers</span>, that have the same tag
   name, namespace, and attributes as <var>element</var>, then remove the earliest such
   element from the <span>list of active formatting elements</span>. For these purposes, the
   attributes must be compared as they were when the elements were created by the parser; two
   elements have the same attributes if all their parsed attributes can be paired such that the two
   attributes in each pair have identical names, namespaces, and values (the order of the attributes
   does not matter).</p>

   <p class="note">This is the Noah's Ark clause. But with three per family instead of two.</p></li>
   <!-- A sort of polyamorous Noah's Ark, if you will. -->

   <li><p>Add <var>element</var> to the <span>list of active formatting
   elements</span>.</p></li>
  </ol>

  <p>When the steps below require the UA to <dfn>reconstruct the active formatting elements</dfn>,
  the UA must perform the following steps:</p>

  <ol>
   <li><p>If there are no entries in the <span>list of active formatting elements</span>, then there
   is nothing to reconstruct; stop this algorithm.</p></li>

   <li><p>If the last (most recently added) entry in the <span>list of active formatting
   elements</span> is a <span data-x="concept-parser-marker">marker</span>, or if it is an element
   that is in the <span>stack of open elements</span>, then there is nothing to reconstruct; stop
   this algorithm.</p></li>

   <li><p>Let <var>entry</var> be the last (most recently added) element in the <span>list
   of active formatting elements</span>.</p></li>

   <li><p><i>Rewind</i>: If there are no entries before <var>entry</var> in the <span>list
   of active formatting elements</span>, then jump to the step labeled <i>create</i>.</p></li>

   <li><p>Let <var>entry</var> be the entry one earlier than <var>entry</var> in
   the <span>list of active formatting elements</span>.</p></li>

   <li><p>If <var>entry</var> is neither a <span
   data-x="concept-parser-marker">marker</span> nor an element that is also in the <span>stack of
   open elements</span>, go to the step labeled <i>rewind</i>.</p></li>

   <li><p><i>Advance</i>: Let <var>entry</var> be the element one later than <var>entry</var> in the <span>list of active formatting elements</span>.</p></li>

   <li><p><i>Create</i>: <span>Insert an HTML element</span> for the token for which the element
   <var>entry</var> was created, to obtain <var>new element</var>.</p></li>

   <li><p>Replace the entry for <var>entry</var> in the list with an entry for <var>new element</var>.</p></li>

   <li><p>If the entry for <var>new element</var> in the <span>list of active formatting
   elements</span> is not the last entry in the list, return to the step labeled
   <i>advance</i>.</p></li>
  </ol>

  <p>This has the effect of reopening all the formatting elements that were opened in the current
  body, cell, or caption (whichever is youngest) that haven't been explicitly closed.</p>

  <p class="note">The way this specification is written, the <span>list of active formatting
  elements</span> always consists of elements in chronological order with the least recently added
  element first and the most recently added element last (except for while steps 7 to 10 of the
  above algorithm are being executed, of course).</p>

  <p>When the steps below require the UA to <dfn>clear the list of active formatting elements up to
  the last marker</dfn>, the UA must perform the following steps:</p>

  <ol>
   <li><p>Let <var>entry</var> be the last (most recently added) entry in the <span>list of
   active formatting elements</span>.</p></li>

   <li><p>Remove <var>entry</var> from the <span>list of active formatting
   elements</span>.</p></li>

   <li><p>If <var>entry</var> was a <span data-x="concept-parser-marker">marker</span>,
   then stop the algorithm at this point. The list has been cleared up to the last <span
   data-x="concept-parser-marker">marker</span>.</p></li>

   <li><p>Go to step 1.</p></li>
  </ol>


  <h5>The element pointers</h5>

  <p>Initially, the <dfn><code data-x="">head</code> element pointer</dfn> and the <dfn><code
  data-x="">form</code> element pointer</dfn> are both null.</p>

  <p>Once a <code>head</code> element has been parsed (whether implicitly or explicitly) the
  <span><code data-x="">head</code> element pointer</span> gets set to point to this node.</p>

  <p>The <span><code data-x="">form</code> element pointer</span> points to the last
  <code>form</code> element that was opened and whose end tag has not yet been seen. It is used to
  make form controls associate with forms in the face of dramatically bad markup, for historical
  reasons. It is ignored inside <code>template</code> elements.</p>


  <h5>Other parsing state flags</h5>

  <p>The <dfn>scripting flag</dfn> is set to "enabled" if <span data-x="concept-n-script">scripting
  was enabled</span> for the <code>Document</code> with which the parser is associated when the
  parser was created, and "disabled" otherwise.</p>

  <p class="note">The <span>scripting flag</span> can be enabled even when the parser was created as
  part of the <span>HTML fragment parsing algorithm</span>, even though <code>script</code> elements
  don't execute in that case.</p>

  <p>The <dfn>frameset-ok flag</dfn> is set to "ok" when the parser is created. It is set to "not
  ok" after certain tokens are seen.</p>

  </div>


  <div w-nodev>

  <h4><dfn>Tokenization</dfn></h4>

  <p>Implementations must act as if they used the following state machine to tokenize HTML. The
  state machine must start in the <span>data state</span>. Most states consume a single character,
  which may have various side-effects, and either switches the state machine to a new state to
  <span>reconsume</span> the <span>current input character</span>, or switches it to a new state to
  consume the <span data-x="next input character">next character</span>, or stays in the same state
  to consume the next character. Some states have more complicated behavior and can consume several
  characters before switching to another state. In some cases, the tokenizer state is also changed
  by the tree construction stage.</p>

  <p>When a state says to <dfn>reconsume</dfn> a matched character in a specified state, that means
  to switch to that state, but when it attempts to consume the <span>next input character</span>,
  provide it with the <span>current input character</span> instead.</p>

  <p>The exact behavior of certain states depends on the <span>insertion mode</span> and the
  <span>stack of open elements</span>. Certain states also use a <dfn><var data-x="temporary
  buffer">temporary buffer</var></dfn> to track progress, and the <span>character reference
  state</span> uses a <dfn><var data-x="return state">return state</var></dfn> to return to the
  state it was invoked from.</p>

  <p>The output of the tokenization step is a series of zero or more of the following tokens:
  DOCTYPE, start tag, end tag, comment, character, end-of-file. DOCTYPE tokens have a name, a public
  identifier, a system identifier, and a <dfn><i>force-quirks flag</i></dfn>. When a DOCTYPE token
  is created, its name, public identifier, and system identifier must be marked as missing (which is
  a distinct state from the empty string), and the <i data-x="force-quirks flag">force-quirks flag</i> must be set to
  <i>off</i> (its other state is <i>on</i>). Start and end tag tokens have a tag name, a <dfn
  data-x="self-closing flag">self-closing flag</dfn>, and a list of attributes, each of which has a
  name and a value. When a start or end tag token is created, its <i data-x="self-closing
  flag">self-closing flag</i> must be unset (its other state is that it be set), and its attributes
  list must be empty. Comment and character tokens have data.</p>

  <p>When a token is emitted, it must immediately be handled by the <span>tree construction</span>
  stage. The tree construction stage can affect the state of the tokenization stage, and can insert
  additional characters into the stream. (For example, the <code>script</code> element can result in
  scripts executing and using the <span>dynamic markup insertion</span> APIs to insert characters
  into the stream being tokenized.)</p>

  <p class="note">Creating a token and emitting it are distinct actions. It is possible for a token
  to be created but implicitly abandoned (never emitted), e.g. if the file ends unexpectedly while
  processing the characters that are being parsed into a start tag token.</p>

  <p>When a start tag token is emitted with its <i data-x="self-closing flag">self-closing flag</i> set, if the flag is not
  <dfn data-x="acknowledge self-closing flag">acknowledged</dfn> when it is processed by the tree
  construction stage, that is a <span
  data-x="parse-error-non-void-html-element-start-tag-with-trailing-solidus">non-void-html-element-start-tag-with-trailing-solidus</span>
  <span>parse error</span>.</p>

  <p>When an end tag token is emitted with attributes, that is an <span
  data-x="parse-error-end-tag-with-attributes">end-tag-with-attributes</span> <span>parse
  error</span>.</p>

  <p>When an end tag token is emitted with its <i data-x="self-closing flag">self-closing flag</i>
  set, that is an <span
  data-x="parse-error-end-tag-with-trailing-solidus">end-tag-with-trailing-solidus</span>
  <span>parse error</span>.</p>

  <p>An <dfn>appropriate end tag token</dfn> is an end tag token whose tag name matches the tag name
  of the last start tag to have been emitted from this tokenizer, if any. If no start tag has been
  emitted from this tokenizer, then no end tag token is appropriate.</p>

  <p>A <span data-x="syntax-charref">character reference</span> is said to be <dfn
  data-x="charref-in-attribute">consumed as part of an attribute</dfn> if the <var data-x="return
  state">return state</var> is either <span>attribute value (double-quoted) state</span>,
  <span>attribute value (single-quoted) state</span>, or <span>attribute value (unquoted)
  state</span>.</p>

  <p>When a state says to <dfn>flush code points consumed as a character reference</dfn>, it means
  that for each <span>code point</span> in the <var data-x="temporary buffer">temporary
  buffer</var> (in the order they were added to the buffer) user agent must append the code point
  from the buffer to the current attribute's value if the character reference was <span
  data-x="charref-in-attribute">consumed as part of an attribute</span>, or emit the code point as a
  character token otherwise.</p>

  <p id="check-parser-pause-flag">Before each step of the tokenizer, the user agent must first check
  the <span>parser pause flag</span>. If it is true, then the tokenizer must abort the processing of
  any nested invocations of the tokenizer, yielding control back to the caller.</p>

  <p>The tokenizer state machine consists of the states defined in the following subsections.</p>


  <!-- Order of the lists below is supposed to be non-error then error, by unicode, then EOF, ending
  with "anything else" -->


  <h5><dfn>Data state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0026 AMPERSAND (&amp;)</dt>
   <dd>Set the <var data-x="return state">return state</var> to the <span>data state</span>.
   Switch to the <span>character reference state</span>.</dd>

   <dt>U+003C LESS-THAN SIGN (&lt;)</dt>
   <dd>Switch to the <span>tag open state</span>.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Emit the <span>current input character</span> as a character token.</dd>

   <dt>EOF</dt>
   <dd>Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Emit the <span>current input character</span> as a character token.</dd>
  </dl>


  <h5><dfn>RCDATA state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0026 AMPERSAND (&amp;)</dt>
   <dd>Set the <var data-x="return state">return state</var> to the <span>RCDATA state</span>.
   Switch to the <span>character reference state</span>.</dd>

   <dt>U+003C LESS-THAN SIGN (&lt;)</dt>
   <dd>Switch to the <span>RCDATA less-than sign state</span>.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Emit a U+FFFD REPLACEMENT CHARACTER character token.</dd>

   <dt>EOF</dt>
   <dd>Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Emit the <span>current input character</span> as a character token.</dd>
  </dl>


  <h5><dfn>RAWTEXT state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+003C LESS-THAN SIGN (&lt;)</dt>
   <dd>Switch to the <span>RAWTEXT less-than sign state</span>.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Emit a U+FFFD REPLACEMENT CHARACTER character token.</dd>

   <dt>EOF</dt>
   <dd>Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Emit the <span>current input character</span> as a character token.</dd>
  </dl>


  <h5><dfn>Script data state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+003C LESS-THAN SIGN (&lt;)</dt>
   <dd>Switch to the <span>script data less-than sign state</span>.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Emit a U+FFFD REPLACEMENT CHARACTER character token.</dd>

   <dt>EOF</dt>
   <dd>Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Emit the <span>current input character</span> as a character token.</dd>
  </dl>


  <h5><dfn>PLAINTEXT state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Emit a U+FFFD REPLACEMENT CHARACTER character token.</dd>

   <dt>EOF</dt>
   <dd>Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Emit the <span>current input character</span> as a character token.</dd>
  </dl>


  <h5><dfn>Tag open state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0021 EXCLAMATION MARK (!)</dt>
   <dd>Switch to the <span>markup declaration open state</span>.</dd>

   <dt>U+002F SOLIDUS (/)</dt>
   <dd>Switch to the <span>end tag open state</span>.</dd>

   <dt><span>ASCII alpha</span></dt>
   <dd>Create a new start tag token, set its tag name to the empty string. <span>Reconsume</span> in
   the <span>tag name state</span>.

   <dt>U+003F QUESTION MARK (?)</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-question-mark-instead-of-tag-name">unexpected-question-mark-instead-of-tag-name</span>
   <span>parse error</span>. Create a comment token whose data is the empty string.
   <span>Reconsume</span> in the <span>bogus comment state</span>.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-before-tag-name">eof-before-tag-name</span>
   <span>parse error</span>. Emit a U+003C LESS-THAN SIGN character token and an end-of-file
   token.</dd>

   <dt>Anything else</dt>
   <dd>This is an <span
   data-x="parse-error-invalid-first-character-of-tag-name">invalid-first-character-of-tag-name</span>
   <span>parse error</span>. Emit a U+003C LESS-THAN SIGN character token. <span>Reconsume</span> in
   the <span>data state</span>.</dd>
  </dl>

  <h5><dfn>End tag open state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt><span>ASCII alpha</span></dt>
   <dd>Create a new end tag token, set its tag name to the empty string. <span>Reconsume</span> in
   the <span>tag name state</span>.

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>This is a <span data-x="parse-error-missing-end-tag-name">missing-end-tag-name</span>
   <span>parse error</span>. Switch to the <span>data state</span>.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-before-tag-name">eof-before-tag-name</span>
   <span>parse error</span>. Emit a U+003C LESS-THAN SIGN character token, a U+002F SOLIDUS
   character token and an end-of-file token.

   <dt>Anything else</dt>
   <dd>This is an <span
   data-x="parse-error-invalid-first-character-of-tag-name">invalid-first-character-of-tag-name</span>
   <span>parse error</span>. Create a comment token whose data is the empty string.
   <span>Reconsume</span> in the <span>bogus comment state</span>.</dd>
  </dl>


  <h5><dfn>Tag name state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>Switch to the <span>before attribute name state</span>.</dd>

   <dt>U+002F SOLIDUS (/)</dt>
   <dd>Switch to the <span>self-closing start tag state</span>.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>Switch to the <span>data state</span>. Emit the current tag token.</dd>

   <dt><span>ASCII upper alpha</span></dt>
   <dd>Append the lowercase version of the <span>current input character</span> (add 0x0020 to the
   character's code point) to the current tag token's tag name.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Append a U+FFFD REPLACEMENT CHARACTER character to the current tag token's tag
   name.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-tag">eof-in-tag</span> <span>parse error</span>.
   Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Append the <span>current input character</span> to the current tag token's tag name.</dd>
  </dl>


  <h5><dfn>RCDATA less-than sign state</dfn></h5>
  <!-- identical to the RAWTEXT less-than sign state, except s/RAWTEXT/RCDATA/g -->

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002F SOLIDUS (/)</dt>
   <dd>Set the <var data-x="temporary buffer">temporary buffer</var> to the empty string. Switch to
   the <span>RCDATA end tag open state</span>.</dd>

   <dt>Anything else</dt>
   <dd>Emit a U+003C LESS-THAN SIGN character token. <span>Reconsume</span> in the <span>RCDATA
   state</span>.</dd>
  </dl>


  <h5><dfn>RCDATA end tag open state</dfn></h5>
  <!-- identical to the RAWTEXT (and Script data) end tag open state, except s/RAWTEXT/RCDATA/g -->

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt><span>ASCII alpha</span></dt>
   <dd>Create a new end tag token, set its tag name to the empty string. <span>Reconsume</span> in
   the <span>RCDATA end tag name state</span>.

   <dt>Anything else</dt>
   <dd>Emit a U+003C LESS-THAN SIGN character token and a U+002F SOLIDUS character token.
   <span>Reconsume</span> in the <span>RCDATA state</span>.</dd>
  </dl>


  <h5><dfn>RCDATA end tag name state</dfn></h5>
  <!-- identical to the RAWTEXT (and Script data) end tag name state, except s/RAWTEXT/RCDATA/g -->

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>If the current end tag token is an <span>appropriate end tag token</span>, then switch to the
   <span>before attribute name state</span>. Otherwise, treat it as per the "anything else" entry
   below.</dd>

   <dt>U+002F SOLIDUS (/)</dt>
   <dd>If the current end tag token is an <span>appropriate end tag token</span>, then switch to the
   <span>self-closing start tag state</span>. Otherwise, treat it as per the "anything else" entry
   below.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>If the current end tag token is an <span>appropriate end tag token</span>, then switch to the
   <span>data state</span> and emit the current tag token. Otherwise, treat it as per the "anything
   else" entry below.</dd>

   <dt><span>ASCII upper alpha</span></dt>
   <dd>Append the lowercase version of the <span>current input character</span> (add 0x0020 to the
   character's code point) to the current tag token's tag name. Append the <span>current input
   character</span> to the <var data-x="temporary buffer">temporary buffer</var>.</dd>

   <dt><span>ASCII lower alpha</span></dt>
   <dd>Append the <span>current input character</span> to the current tag token's tag name. Append
   the <span>current input character</span> to the <var data-x="temporary buffer">temporary
   buffer</var>.</dd>

   <dt>Anything else</dt>
   <dd>Emit a U+003C LESS-THAN SIGN character token, a U+002F SOLIDUS character token, and a
   character token for each of the characters in the <var data-x="temporary buffer">temporary
   buffer</var> (in the order they were added to the buffer). <span>Reconsume</span> in the
   <span>RCDATA state</span>.</dd>
  </dl>


  <h5><dfn>RAWTEXT less-than sign state</dfn></h5>
  <!-- identical to the RCDATA less-than sign state, except s/RCDATA/RAWTEXT/g -->

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002F SOLIDUS (/)</dt>
   <dd>Set the <var data-x="temporary buffer">temporary buffer</var> to the empty string. Switch to
   the <span>RAWTEXT end tag open state</span>.</dd>

   <dt>Anything else</dt>
   <dd>Emit a U+003C LESS-THAN SIGN character token. <span>Reconsume</span> in the <span>RAWTEXT
   state</span>.</dd>
  </dl>


  <h5><dfn>RAWTEXT end tag open state</dfn></h5>
  <!-- identical to the RCDATA (and Script data) end tag open state, except s/RCDATA/RAWTEXT/g -->

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt><span>ASCII alpha</span></dt>
   <dd>Create a new end tag token, set its tag name to the empty string. <span>Reconsume</span> in
   the <span>RAWTEXT end tag name state</span>.

   <dt>Anything else</dt>
   <dd>Emit a U+003C LESS-THAN SIGN character token and a U+002F SOLIDUS character token.
   <span>Reconsume</span> in the <span>RAWTEXT state</span>.</dd>
  </dl>


  <h5><dfn>RAWTEXT end tag name state</dfn></h5>
  <!-- identical to the RCDATA (and Script data) end tag name state, except s/RCDATA/RAWTEXT/g -->

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>If the current end tag token is an <span>appropriate end tag token</span>, then switch to the
   <span>before attribute name state</span>. Otherwise, treat it as per the "anything else" entry
   below.</dd>

   <dt>U+002F SOLIDUS (/)</dt>
   <dd>If the current end tag token is an <span>appropriate end tag token</span>, then switch to the
   <span>self-closing start tag state</span>. Otherwise, treat it as per the "anything else" entry
   below.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>If the current end tag token is an <span>appropriate end tag token</span>, then switch to the
   <span>data state</span> and emit the current tag token. Otherwise, treat it as per the "anything
   else" entry below.</dd>

   <dt><span>ASCII upper alpha</span></dt>
   <dd>Append the lowercase version of the <span>current input character</span> (add 0x0020 to the
   character's code point) to the current tag token's tag name. Append the <span>current input
   character</span> to the <var data-x="temporary buffer">temporary buffer</var>.</dd>

   <dt><span>ASCII lower alpha</span></dt>
   <dd>Append the <span>current input character</span> to the current tag token's tag name. Append
   the <span>current input character</span> to the <var data-x="temporary buffer">temporary
   buffer</var>.</dd>

   <dt>Anything else</dt>
   <dd>Emit a U+003C LESS-THAN SIGN character token, a U+002F SOLIDUS character token, and a
   character token for each of the characters in the <var data-x="temporary buffer">temporary
   buffer</var> (in the order they were added to the buffer). <span>Reconsume</span> in the
   <span>RAWTEXT state</span>.</dd>
  </dl>


  <h5><dfn>Script data less-than sign state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002F SOLIDUS (/)</dt>
   <dd>Set the <var data-x="temporary buffer">temporary buffer</var> to the empty string. Switch to
   the <span>script data end tag open state</span>.</dd>

   <dt>U+0021 EXCLAMATION MARK (!)</dt>
   <dd>Switch to the <span>script data escape start state</span>. Emit a U+003C LESS-THAN SIGN
   character token and a U+0021 EXCLAMATION MARK character token.</dd>

   <dt>Anything else</dt>
   <dd>Emit a U+003C LESS-THAN SIGN character token. <span>Reconsume</span> in the <span>script data
   state</span>.</dd>
  </dl>


  <h5><dfn>Script data end tag open state</dfn></h5>
  <!-- identical to the RCDATA (and RAWTEXT) end tag open state, except s/RCDATA/Script data/g -->

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt><span>ASCII alpha</span></dt>
   <dd>Create a new end tag token, set its tag name to the empty string. <span>Reconsume</span> in
   the <span>script data end tag name state</span>.

   <dt>Anything else</dt>
   <dd>Emit a U+003C LESS-THAN SIGN character token and a U+002F SOLIDUS character token.
   <span>Reconsume</span> in the <span>script data state</span>.</dd>
  </dl>


  <h5><dfn>Script data end tag name state</dfn></h5>
  <!-- identical to the RCDATA (and RAWTEXT) end tag name state, except s/RCDATA/Script data/g -->

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>If the current end tag token is an <span>appropriate end tag token</span>, then switch to the
   <span>before attribute name state</span>. Otherwise, treat it as per the "anything else" entry
   below.</dd>

   <dt>U+002F SOLIDUS (/)</dt>
   <dd>If the current end tag token is an <span>appropriate end tag token</span>, then switch to the
   <span>self-closing start tag state</span>. Otherwise, treat it as per the "anything else" entry
   below.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>If the current end tag token is an <span>appropriate end tag token</span>, then switch to the
   <span>data state</span> and emit the current tag token. Otherwise, treat it as per the "anything
   else" entry below.</dd>

   <dt><span>ASCII upper alpha</span></dt>
   <dd>Append the lowercase version of the <span>current input character</span> (add 0x0020 to the
   character's code point) to the current tag token's tag name. Append the <span>current input
   character</span> to the <var data-x="temporary buffer">temporary buffer</var>.</dd>

   <dt><span>ASCII lower alpha</span></dt>
   <dd>Append the <span>current input character</span> to the current tag token's tag name. Append
   the <span>current input character</span> to the <var data-x="temporary buffer">temporary
   buffer</var>.</dd>

   <dt>Anything else</dt>
   <dd>Emit a U+003C LESS-THAN SIGN character token, a U+002F SOLIDUS character token, and a
   character token for each of the characters in the <var data-x="temporary buffer">temporary
   buffer</var> (in the order they were added to the buffer). <span>Reconsume</span> in the
   <span>script data state</span>.</dd>
  </dl>


  <h5><dfn>Script data escape start state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002D HYPHEN-MINUS (-)</dt>
   <dd>Switch to the <span>script data escape start dash state</span>. Emit a U+002D HYPHEN-MINUS
   character token.</dd>

   <dt>Anything else</dt>
   <dd><span>Reconsume</span> in the <span>script data state</span>.</dd>
  </dl>


  <h5><dfn>Script data escape start dash state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002D HYPHEN-MINUS (-)</dt>
   <dd>Switch to the <span>script data escaped dash dash state</span>. Emit a U+002D HYPHEN-MINUS
   character token.</dd>

   <dt>Anything else</dt>
   <dd><span>Reconsume</span> in the <span>script data state</span>.</dd>
  </dl>


  <h5><dfn>Script data escaped state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002D HYPHEN-MINUS (-)</dt>
   <dd>Switch to the <span>script data escaped dash state</span>. Emit a U+002D HYPHEN-MINUS
   character token.</dd>

   <dt>U+003C LESS-THAN SIGN (&lt;)</dt>
   <dd>Switch to the <span>script data escaped less-than sign state</span>.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Emit a U+FFFD REPLACEMENT CHARACTER character token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span
   data-x="parse-error-eof-in-script-html-comment-like-text">eof-in-script-html-comment-like-text</span>
   <span>parse error</span>. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Emit the <span>current input character</span> as a character token.</dd>
  </dl>


  <h5><dfn>Script data escaped dash state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002D HYPHEN-MINUS (-)</dt>
   <dd>Switch to the <span>script data escaped dash dash state</span>. Emit a U+002D HYPHEN-MINUS
   character token.</dd>

   <dt>U+003C LESS-THAN SIGN (&lt;)</dt>
   <dd>Switch to the <span>script data escaped less-than sign state</span>.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Switch to the <span>script data escaped state</span>. Emit a U+FFFD REPLACEMENT
   CHARACTER character token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span
   data-x="parse-error-eof-in-script-html-comment-like-text">eof-in-script-html-comment-like-text</span>
   <span>parse error</span>. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Switch to the <span>script data escaped state</span>. Emit the <span>current input
   character</span> as a character token.</dd>
  </dl>


  <h5><dfn>Script data escaped dash dash state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002D HYPHEN-MINUS (-)</dt>
   <dd>Emit a U+002D HYPHEN-MINUS character token.</dd>

   <dt>U+003C LESS-THAN SIGN (&lt;)</dt>
   <dd>Switch to the <span>script data escaped less-than sign state</span>.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>Switch to the <span>script data state</span>. Emit a U+003E GREATER-THAN SIGN character
   token.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Switch to the <span>script data escaped state</span>. Emit a U+FFFD REPLACEMENT
   CHARACTER character token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span
   data-x="parse-error-eof-in-script-html-comment-like-text">eof-in-script-html-comment-like-text</span>
   <span>parse error</span>. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Switch to the <span>script data escaped state</span>. Emit the <span>current input
   character</span> as a character token.</dd>
  </dl>


  <h5><dfn>Script data escaped less-than sign state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002F SOLIDUS (/)</dt>
   <dd>Set the <var data-x="temporary buffer">temporary buffer</var> to the empty string. Switch to
   the <span>script data escaped end tag open state</span>.</dd>

   <dt><span>ASCII alpha</span></dt>
   <dd>Set the <var data-x="temporary buffer">temporary buffer</var> to the empty string. Emit a
   U+003C LESS-THAN SIGN character token. <span>Reconsume</span> in the <span>script data double
   escape start state</span>.</dd>

   <dt>Anything else</dt>
   <dd>Emit a U+003C LESS-THAN SIGN character token. <span>Reconsume</span> in the <span>script data
   escaped state</span>.</dd>
  </dl>


  <h5><dfn>Script data escaped end tag open state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt><span>ASCII alpha</span></dt>
   <dd>Create a new end tag token, set its tag name to the empty string. <span>Reconsume</span> in
   the <span>script data escaped end tag name state</span>.</dd>

   <dt>Anything else</dt>
   <dd>Emit a U+003C LESS-THAN SIGN character token and a U+002F SOLIDUS character token.
   <span>Reconsume</span> in the <span>script data escaped state</span>.</dd>
  </dl>


  <h5><dfn>Script data escaped end tag name state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>If the current end tag token is an <span>appropriate end tag token</span>, then switch to the
   <span>before attribute name state</span>. Otherwise, treat it as per the "anything else" entry
   below.</dd>

   <dt>U+002F SOLIDUS (/)</dt>
   <dd>If the current end tag token is an <span>appropriate end tag token</span>, then switch to the
   <span>self-closing start tag state</span>. Otherwise, treat it as per the "anything else" entry
   below.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>If the current end tag token is an <span>appropriate end tag token</span>, then switch to the
   <span>data state</span> and emit the current tag token. Otherwise, treat it as per the "anything
   else" entry below.</dd>

   <dt><span>ASCII upper alpha</span></dt>
   <dd>Append the lowercase version of the <span>current input character</span> (add 0x0020 to the
   character's code point) to the current tag token's tag name. Append the <span>current input
   character</span> to the <var data-x="temporary buffer">temporary buffer</var>.</dd>

   <dt><span>ASCII lower alpha</span></dt>
   <dd>Append the <span>current input character</span> to the current tag token's tag name. Append
   the <span>current input character</span> to the <var data-x="temporary buffer">temporary
   buffer</var>.</dd>

   <dt>Anything else</dt>
   <dd>Emit a U+003C LESS-THAN SIGN character token, a U+002F SOLIDUS character token, and a
   character token for each of the characters in the <var data-x="temporary buffer">temporary buffer
   </var> (in the order they were added to the buffer). <span>Reconsume</span> in the <span>script
   data escaped state</span>.</dd>
  </dl>


  <h5><dfn>Script data double escape start state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dt>U+002F SOLIDUS (/)</dt>
   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>If the <var data-x="temporary buffer">temporary buffer</var> is the string "<code
   data-x="">script</code>", then switch to the <span>script data double escaped state</span>.
   Otherwise, switch to the <span>script data escaped state</span>. Emit the <span>current input
   character</span> as a character token.</dd>

   <dt><span>ASCII upper alpha</span></dt>
   <dd>Append the lowercase version of the <span>current input character</span> (add 0x0020 to the
   character's code point) to the <var data-x="temporary buffer">temporary buffer</var>. Emit the
   <span>current input character</span> as a character token.</dd>

   <dt><span>ASCII lower alpha</span></dt>
   <dd>Append the <span>current input character</span> to the <var data-x="temporary
   buffer">temporary buffer</var>. Emit the <span>current input character</span> as a character
   token.</dd>

   <dt>Anything else</dt>
   <dd><span>Reconsume</span> in the <span>script data escaped state</span>.</dd>
  </dl>


  <h5><dfn>Script data double escaped state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002D HYPHEN-MINUS (-)</dt>
   <dd>Switch to the <span>script data double escaped dash state</span>. Emit a U+002D HYPHEN-MINUS
   character token.</dd>

   <dt>U+003C LESS-THAN SIGN (&lt;)</dt>
   <dd>Switch to the <span>script data double escaped less-than sign state</span>. Emit a U+003C
   LESS-THAN SIGN character token.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Emit a U+FFFD REPLACEMENT CHARACTER character token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span
   data-x="parse-error-eof-in-script-html-comment-like-text">eof-in-script-html-comment-like-text</span>
   <span>parse error</span>. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Emit the <span>current input character</span> as a character token.</dd>
  </dl>


  <h5><dfn>Script data double escaped dash state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002D HYPHEN-MINUS (-)</dt>
   <dd>Switch to the <span>script data double escaped dash dash state</span>. Emit a U+002D
   HYPHEN-MINUS character token.</dd>

   <dt>U+003C LESS-THAN SIGN (&lt;)</dt>
   <dd>Switch to the <span>script data double escaped less-than sign state</span>. Emit a U+003C
   LESS-THAN SIGN character token.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Switch to the <span>script data double escaped state</span>. Emit a U+FFFD
   REPLACEMENT CHARACTER character token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span
   data-x="parse-error-eof-in-script-html-comment-like-text">eof-in-script-html-comment-like-text</span>
   <span>parse error</span>. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Switch to the <span>script data double escaped state</span>. Emit the <span>current input
   character</span> as a character token.</dd>
  </dl>


  <h5><dfn>Script data double escaped dash dash state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002D HYPHEN-MINUS (-)</dt>
   <dd>Emit a U+002D HYPHEN-MINUS character token.</dd>

   <dt>U+003C LESS-THAN SIGN (&lt;)</dt>
   <dd>Switch to the <span>script data double escaped less-than sign state</span>. Emit a U+003C
   LESS-THAN SIGN character token.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>Switch to the <span>script data state</span>. Emit a U+003E GREATER-THAN SIGN character
   token.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Switch to the <span>script data double escaped state</span>. Emit a U+FFFD
   REPLACEMENT CHARACTER character token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span
   data-x="parse-error-eof-in-script-html-comment-like-text">eof-in-script-html-comment-like-text</span>
   <span>parse error</span>. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Switch to the <span>script data double escaped state</span>. Emit the <span>current input
   character</span> as a character token.</dd>
  </dl>


  <h5><dfn>Script data double escaped less-than sign state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002F SOLIDUS (/)</dt>
   <dd>Set the <var data-x="temporary buffer">temporary buffer</var> to the empty string. Switch to
   the <span>script data double escape end state</span>. Emit a U+002F SOLIDUS character token.</dd>

   <dt>Anything else</dt>
   <dd><span>Reconsume</span> in the <span>script data double escaped state</span>.</dd>
  </dl>


  <h5><dfn>Script data double escape end state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dt>U+002F SOLIDUS (/)</dt>
   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>If the <var data-x="temporary buffer">temporary buffer</var> is the string "<code
   data-x="">script</code>", then switch to the <span>script data escaped state</span>. Otherwise,
   switch to the <span>script data double escaped state</span>. Emit the <span>current input
   character</span> as a character token.</dd>

   <dt><span>ASCII upper alpha</span></dt>
   <dd>Append the lowercase version of the <span>current input character</span> (add 0x0020 to the
   character's code point) to the <var data-x="temporary buffer">temporary buffer</var>. Emit the
   <span>current input character</span> as a character token.</dd>

   <dt><span>ASCII lower alpha</span></dt>
   <dd>Append the <span>current input character</span> to the <var data-x="temporary
   buffer">temporary buffer</var>. Emit the <span>current input character</span> as a character
   token.</dd>

   <dt>Anything else</dt>
   <dd><span>Reconsume</span> in the <span>script data double escaped state</span>.</dd>
  </dl>


  <h5><dfn>Before attribute name state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>Ignore the character.</dd>

   <dt>U+002F SOLIDUS (/)</dt>
   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dt>EOF</dt>
   <dd><span>Reconsume</span> in the <span>after attribute name state</span>.</dd>

   <dt>U+003D EQUALS SIGN (=)</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-equals-sign-before-attribute-name">unexpected-equals-sign-before-attribute-name</span>
   <span>parse error</span>. Start a new attribute in the current tag token. Set that attribute's
   name to the <span>current input character</span>, and its value to the empty string. Switch to
   the <span>attribute name state</span>.</dd>

   <dt>Anything else</dt>
   <dd>Start a new attribute in the current tag token. Set that attribute name and value to the
   empty string. <span>Reconsume</span> in the <span>attribute name state</span>.</dd>
  </dl>


  <h5><dfn>Attribute name state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dt>U+002F SOLIDUS (/)</dt>
   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dt>EOF</dt>
   <dd><span>Reconsume</span> in the <span>after attribute name state</span>.</dd>

   <dt>U+003D EQUALS SIGN (=)</dt>
   <dd>Switch to the <span>before attribute value state</span>.</dd>

   <dt><span>ASCII upper alpha</span></dt>
   <dd>Append the lowercase version of the <span>current input character</span> (add 0x0020 to the
   character's code point) to the current attribute's name.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Append a U+FFFD REPLACEMENT CHARACTER character to the current attribute's
   name.</dd>

   <dt>U+0022 QUOTATION MARK (&quot;)</dt>
   <dt>U+0027 APOSTROPHE (')</dt>
   <dt>U+003C LESS-THAN SIGN (&lt;)</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-character-in-attribute-name">unexpected-character-in-attribute-name</span>
   <span>parse error</span>. Treat it as per the "anything else" entry below.</dd>

   <dt>Anything else</dt>
   <dd>Append the <span>current input character</span> to the current attribute's name.</dd>
  </dl>

  <p>When the user agent leaves the attribute name state (and before emitting the tag token, if
  appropriate), the complete attribute's name must be compared to the other attributes on the same
  token; if there is already an attribute on the token with the exact same name, then this is a
  <span data-x="parse-error-duplicate-attribute">duplicate-attribute</span> <span>parse error</span>
  and the new attribute must be removed from the token.</p>

  <p class="note">If an attribute is so removed from a token, it, and the value that gets associated
  with it, if any, are never subsequently used by the parser, and are therefore effectively
  discarded. Removing the attribute in this way does not change its status as the "current
  attribute" for the purposes of the tokenizer, however.</p>


  <h5><dfn>After attribute name state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>Ignore the character.</dd>

   <dt>U+002F SOLIDUS (/)</dt>
   <dd>Switch to the <span>self-closing start tag state</span>.</dd>

   <dt>U+003D EQUALS SIGN (=)</dt>
   <dd>Switch to the <span>before attribute value state</span>.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>Switch to the <span>data state</span>. Emit the current tag token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-tag">eof-in-tag</span>
   <span>parse error</span>. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Start a new attribute in the current tag token. Set that attribute name and value to the
   empty string. <span>Reconsume</span> in the <span>attribute name state</span>.</dd>
  </dl>


  <h5><dfn>Before attribute value state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>Ignore the character.</dd>

   <dt>U+0022 QUOTATION MARK (&quot;)</dt>
   <dd>Switch to the <span>attribute value (double-quoted) state</span>.</dd>

   <dt>U+0027 APOSTROPHE (')</dt>
   <dd>Switch to the <span>attribute value (single-quoted) state</span>.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>This is a <span data-x="parse-error-missing-attribute-value">missing-attribute-value</span>
   <span>parse error</span>. Switch to the <span>data state</span>. Emit the current tag token.</dd>

   <dt>Anything else</dt>
   <dd><span>Reconsume</span> in the <span>attribute value (unquoted) state</span>.</dd>
  </dl>


  <h5><dfn>Attribute value (double-quoted) state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0022 QUOTATION MARK (&quot;)</dt>
   <dd>Switch to the <span>after attribute value (quoted) state</span>.</dd>

   <dt>U+0026 AMPERSAND (&amp;)</dt>
   <dd>Set the <var data-x="return state">return state</var> to the <span>attribute value
   (double-quoted) state</span>. Switch to the <span>character reference state</span>.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Append a U+FFFD REPLACEMENT CHARACTER character to the current attribute's
   value.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-tag">eof-in-tag</span> <span>parse error</span>.
   Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Append the <span>current input character</span> to the current attribute's value.</dd>
  </dl>


  <h5><dfn>Attribute value (single-quoted) state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0027 APOSTROPHE (')</dt>
   <dd>Switch to the <span>after attribute value (quoted) state</span>.</dd>

   <dt>U+0026 AMPERSAND (&amp;)</dt>
   <dd>Set the <var data-x="return state">return state</var> to the <span>attribute value
   (single-quoted) state</span>. Switch to the <span>character reference state</span>.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Append a U+FFFD REPLACEMENT CHARACTER character to the current attribute's
   value.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-tag">eof-in-tag</span> <span>parse error</span>.
   Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Append the <span>current input character</span> to the current attribute's value.</dd>
  </dl>


  <h5><dfn>Attribute value (unquoted) state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>Switch to the <span>before attribute name state</span>.</dd>

   <dt>U+0026 AMPERSAND (&amp;)</dt>
   <dd>Set the <var data-x="return state">return state</var> to the <span>attribute value
   (unquoted) state</span>. Switch to the <span>character reference state</span>.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>Switch to the <span>data state</span>. Emit the current tag token.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Append a U+FFFD REPLACEMENT CHARACTER character to the current attribute's
   value.</dd>

   <dt>U+0022 QUOTATION MARK (&quot;)</dt>
   <dt>U+0027 APOSTROPHE (')</dt>
   <dt>U+003C LESS-THAN SIGN (&lt;)</dt>
   <dt>U+003D EQUALS SIGN (=)</dt>
   <dt>U+0060 GRAVE ACCENT (`)</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-character-in-unquoted-attribute-value">unexpected-character-in-unquoted-attribute-value</span>
   <span>parse error</span>. Treat it as per the "anything else" entry below.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-tag">eof-in-tag</span>
   <span>parse error</span>. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Append the <span>current input character</span> to the current attribute's value.</dd>
  </dl>


  <h5><dfn>After attribute value (quoted) state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>Switch to the <span>before attribute name state</span>.</dd>

   <dt>U+002F SOLIDUS (/)</dt>
   <dd>Switch to the <span>self-closing start tag state</span>.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>Switch to the <span>data state</span>. Emit the current tag token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-tag">eof-in-tag</span> <span>parse error</span>.
   Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>This is a <span
   data-x="parse-error-missing-whitespace-between-attributes">missing-whitespace-between-attributes</span>
   <span>parse error</span>. <span>Reconsume</span> in the <span>before attribute name state</span>.
   </dd>
  </dl>


  <h5><dfn>Self-closing start tag state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>Set the <i data-x="self-closing flag">self-closing flag</i> of the current tag token. Switch
   to the <span>data state</span>. Emit the current tag token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-tag">eof-in-tag</span>
   <span>parse error</span>. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-solidus-in-tag">unexpected-solidus-in-tag</span> <span>parse
   error</span>. <span>Reconsume</span> in the <span>before attribute name state</span>.</dd>
  </dl>


  <h5><dfn>Bogus comment state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>Switch to the <span>data state</span>. Emit the current comment token.</dd>

   <dt>EOF</dt>
   <dd>Emit the comment. Emit an end-of-file token.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Append a U+FFFD REPLACEMENT CHARACTER character to the comment token's data.</dd>

   <dt>Anything else</dt>
   <dd>Append the <span>current input character</span> to the comment token's data.</dd>
  </dl>


  <h5><dfn>Markup declaration open state</dfn></h5>

  <p>If the next few characters are:</p>

  <dl class="switch">

   <dt>Two U+002D HYPHEN-MINUS characters (-)</dt>

   <dd>Consume those two characters, create a comment token whose data is the empty string, and
   switch to the <span>comment start state</span>.</dd>

   <dt><span>ASCII case-insensitive</span> match for the word "DOCTYPE"</dt>
   <dd>Consume those characters and switch to the <span>DOCTYPE state</span>.</dd>

   <dt>The string "[CDATA[" (the five uppercase letters "CDATA" with a U+005B LEFT SQUARE BRACKET
   character before and after)</dt>
   <dd>Consume those characters. If there is an <span>adjusted current node</span> and it is not
   an element in the <span>HTML namespace</span>, then switch to the <span>CDATA section
   state</span>. Otherwise, this is a <span
   data-x="parse-error-cdata-in-html-content">cdata-in-html-content</span> <span>parse
   error</span>. Create a comment token whose data is the "[CDATA[" string. Switch to the
   <span>bogus comment state</span>.</dd>

   <dt>Anything else</dt>
   <dd>This is an <span
   data-x="parse-error-incorrectly-opened-comment">incorrectly-opened-comment</span>
   <span>parse error</span>. Create a comment token whose data is the empty string. Switch to the
   <span>bogus comment state</span> (don't consume anything in the current state).</dd>
  </dl>


  <h5><dfn>Comment start state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002D HYPHEN-MINUS (-)</dt>
   <dd>Switch to the <span>comment start dash state</span>.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>This is an <span
   data-x="parse-error-abrupt-closing-of-empty-comment">abrupt-closing-of-empty-comment</span> <span>parse
   error</span>. Switch to the <span>data state</span>. Emit the current comment token.</dd>

   <dt>Anything else</dt>
   <dd><span>Reconsume</span> in the <span>comment state</span>.</dd>
  </dl>


  <h5><dfn>Comment start dash state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002D HYPHEN-MINUS (-)</dt>
   <dd>Switch to the <span>comment end state</span>.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>This is an <span
   data-x="parse-error-abrupt-closing-of-empty-comment">abrupt-closing-of-empty-comment</span> <span>parse
   error</span>. Switch to the <span>data state</span>. Emit the current comment token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-comment">eof-in-comment</span> <span>parse
   error</span>. Emit the current comment token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Append a U+002D HYPHEN-MINUS character (-) to the comment token's data.
   <span>Reconsume</span> in the <span>comment state</span>.</dd>
  </dl>


  <h5><dfn id="comment">Comment state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+003C LESS-THAN SIGN (&lt;)</dt>
   <dd>Append the <span>current input character</span> to the comment token's data. Switch to the
   <span>comment less-than sign state</span>.</dd>

   <dt>U+002D HYPHEN-MINUS (-)</dt>
   <dd>Switch to the <span>comment end dash state</span>.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Append a U+FFFD REPLACEMENT CHARACTER character to the comment token's data.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-comment">eof-in-comment</span> <span>parse
   error</span>. Emit the current comment token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Append the <span>current input character</span> to the comment token's data.</dd>
  </dl>


  <h5><dfn>Comment less-than sign state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0021 EXCLAMATION MARK (!)</dt>
   <dd>Append the <span>current input character</span> to the comment token's data. Switch to the
   <span>comment less-than sign bang state</span>.</dd>

   <dt>U+003C LESS-THAN SIGN (&lt;)</dt>
   <dd>Append the <span>current input character</span> to the comment token's data.</dd>

   <dt>Anything else</dt>
   <dd><span>Reconsume</span> in the <span>comment state</span>.</dd>
  </dl>


  <h5><dfn>Comment less-than sign bang state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002D HYPHEN-MINUS (-)</dt>
   <dd>Switch to the <span>comment less-than sign bang dash state</span>.</dd>

   <dt>Anything else</dt>
   <dd><span>Reconsume</span> in the <span>comment state</span>.</dd>
  </dl>


  <h5><dfn>Comment less-than sign bang dash state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002D HYPHEN-MINUS (-)</dt>
   <dd>Switch to the <span>comment less-than sign bang dash dash state</span>.</dd>

   <dt>Anything else</dt>
   <dd><span>Reconsume</span> in the <span>comment end dash state</span>.</dd>
  </dl>


  <h5><dfn>Comment less-than sign bang dash dash state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dt>EOF</dt>
   <dd><span>Reconsume</span> in the <span>comment end state</span>.</dd>

   <dt>Anything else</dt>
   <dd>This is a <span data-x="parse-error-nested-comment">nested-comment</span> <span>parse
   error</span>. <span>Reconsume</span> in the <span>comment end state</span>.</dd>
  </dl>


  <h5><dfn>Comment end dash state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002D HYPHEN-MINUS (-)</dt>
   <dd>Switch to the <span>comment end state</span>.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-comment">eof-in-comment</span> <span>parse
   error</span>. Emit the current comment token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Append a U+002D HYPHEN-MINUS character (-) to the comment token's data.
   <span>Reconsume</span> in the <span>comment state</span>.</dd>
  </dl>


  <h5><dfn>Comment end state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>Switch to the <span>data state</span>. Emit the current comment token.</dd>

   <dt>U+0021 EXCLAMATION MARK (!)</dt>
   <dd>Switch to the <span>comment end bang state</span>.</dd>

   <dt>U+002D HYPHEN-MINUS (-)</dt>
   <dd>Append a U+002D HYPHEN-MINUS character (-) to the comment token's data.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-comment">eof-in-comment</span> <span>parse
   error</span>. Emit the current comment token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Append two U+002D HYPHEN-MINUS characters (-) to the comment token's data.
   <span>Reconsume</span> in the <span>comment state</span>.</dd>
  </dl>


  <h5><dfn>Comment end bang state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+002D HYPHEN-MINUS (-)</dt>
   <dd>Append two U+002D HYPHEN-MINUS characters (-) and a U+0021 EXCLAMATION MARK character (!) to
   the comment token's data. Switch to the <span>comment end dash state</span>.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>This is an <span
   data-x="parse-error-incorrectly-closed-comment">incorrectly-closed-comment</span> <span>parse
   error</span>. Switch to the <span>data state</span>. Emit the current comment token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-comment">eof-in-comment</span> <span>parse
   error</span>. Emit the current comment token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Append two U+002D HYPHEN-MINUS characters (-) and a U+0021 EXCLAMATION MARK character (!) to
   the comment token's data. <span>Reconsume</span> in the <span>comment state</span>.</dd>
  </dl>


  <h5><dfn>DOCTYPE state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>Switch to the <span>before DOCTYPE name state</span>.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd><span>Reconsume</span> in the <span>before DOCTYPE name state</span>.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-doctype">eof-in-doctype</span> <span>parse
   error</span>. Create a new DOCTYPE token. Set its <i data-x="force-quirks flag">force-quirks
   flag</i> to <i>on</i>. Emit the current token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>This is a <span
   data-x="parse-error-missing-whitespace-before-doctype-name">missing-whitespace-before-doctype-name</span>
   <span>parse error</span>. <span>Reconsume</span> in the <span>before DOCTYPE name state</span>.
   </dd>
  </dl>


  <h5><dfn>Before DOCTYPE name state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>Ignore the character.</dd>

   <dt><span>ASCII upper alpha</span></dt>
   <dd>Create a new DOCTYPE token. Set the token's name to the lowercase version of the
   <span>current input character</span> (add 0x0020 to the character's code point). Switch to the
   <span>DOCTYPE name state</span>.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Create a new DOCTYPE token. Set the token's name to a U+FFFD REPLACEMENT CHARACTER
   character. Switch to the <span>DOCTYPE name state</span>.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>This is a <span data-x="parse-error-missing-doctype-name">missing-doctype-name</span>
   <span>parse error</span>. Create a new DOCTYPE token. Set its <i
   data-x="force-quirks flag">force-quirks flag</i> to <i>on</i>. Switch to the <span>data
   state</span>. Emit the current token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-doctype">eof-in-doctype</span> <span>parse
   error</span>. Create a new DOCTYPE token. Set its <i data-x="force-quirks flag"> force-quirks
   flag</i> to <i>on</i>. Emit the current token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Create a new DOCTYPE token. Set the token's name to the <span>current input character</span>.
   Switch to the <span>DOCTYPE name state</span>.</dd>
  </dl>


  <h5><dfn>DOCTYPE name state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>Switch to the <span>after DOCTYPE name state</span>.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>Switch to the <span>data state</span>. Emit the current DOCTYPE token.</dd>

   <dt><span>ASCII upper alpha</span></dt>
   <dd>Append the lowercase version of the <span>current input character</span> (add 0x0020 to the
   character's code point) to the current DOCTYPE token's name.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Append a U+FFFD REPLACEMENT CHARACTER character to the current DOCTYPE token's
   name.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-doctype">eof-in-doctype</span> <span>parse
   error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks flag</i> to
   <i>on</i>. Emit the current DOCTYPE token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Append the <span>current input character</span> to the current DOCTYPE token's name.</dd>
  </dl>


  <h5><dfn>After DOCTYPE name state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>Ignore the character.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>Switch to the <span>data state</span>. Emit the current DOCTYPE token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-doctype">eof-in-doctype</span> <span>parse
   error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks flag</i> to
   <i>on</i>. Emit the current DOCTYPE token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>
    <p>If the six characters starting from the <span>current input character</span> are an
    <span>ASCII case-insensitive</span> match for the word "PUBLIC", then consume those characters
    and switch to the <span>after DOCTYPE public keyword state</span>.</p>

    <p>Otherwise, if the six characters starting from the <span>current input character</span> are
    an <span>ASCII case-insensitive</span> match for the word "SYSTEM", then consume those
    characters and switch to the <span>after DOCTYPE system keyword state</span>.</p>

    <p>Otherwise, this is an <span
    data-x="parse-error-invalid-character-sequence-after-doctype-name">invalid-character-sequence-after-doctype-name</span>
    <span>parse error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks
    flag</i> to <i>on</i>. <span>Reconsume</span> in the <span>bogus DOCTYPE state</span>.</p>
   </dd>
  </dl>


  <h5><dfn>After DOCTYPE public keyword state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>Switch to the <span>before DOCTYPE public identifier state</span>.</dd>

   <dt>U+0022 QUOTATION MARK (&quot;)</dt>
   <dd>This is a <span
   data-x="parse-error-missing-whitespace-after-doctype-public-keyword">missing-whitespace-after-doctype-public-keyword</span>
   <span>parse error</span>. Set the current DOCTYPE token's public identifier to the empty string (not
   missing), then switch to the <span>DOCTYPE public identifier (double-quoted) state</span>.</dd>

   <dt>U+0027 APOSTROPHE (')</dt>
   <dd>This is a <span
   data-x="parse-error-missing-whitespace-after-doctype-public-keyword">missing-whitespace-after-doctype-public-keyword</span>
   <span>parse error</span>. Set the current DOCTYPE token's public identifier to the empty string (not
   missing), then switch to the <span>DOCTYPE public identifier (single-quoted) state</span>.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>This is a <span
   data-x="parse-error-missing-doctype-public-identifier">missing-doctype-public-identifier</span> <span>parse
   error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks flag</i> to
   <i>on</i>. Switch to the <span>data state</span>. Emit the current DOCTYPE token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-doctype">eof-in-doctype</span> <span>parse
   error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks flag</i> to
   <i>on</i>. Emit the current DOCTYPE token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>This is a <span
   data-x="parse-error-missing-quote-before-doctype-public-identifier">missing-quote-before-doctype-public-identifier</span>
   <span>parse error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks
   flag</i> to <i>on</i>. <span>Reconsume</span> in the <span>bogus DOCTYPE state</span>.</dd>
  </dl>


  <h5><dfn>Before DOCTYPE public identifier state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>Ignore the character.</dd>

   <dt>U+0022 QUOTATION MARK (&quot;)</dt>
   <dd>Set the current DOCTYPE token's public identifier to the empty string (not missing), then switch to
   the <span>DOCTYPE public identifier (double-quoted) state</span>.</dd>

   <dt>U+0027 APOSTROPHE (')</dt>
   <dd>Set the current DOCTYPE token's public identifier to the empty string (not missing), then switch to
   the <span>DOCTYPE public identifier (single-quoted) state</span>.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>This is a <span
   data-x="parse-error-missing-doctype-public-identifier">missing-doctype-public-identifier</span>
   <span>parse error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks
   flag</i> to <i>on</i>. Switch to the <span>data state</span>. Emit the current DOCTYPE token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-doctype">eof-in-doctype</span>
   <span>parse error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks
   flag</i> to <i>on</i>. Emit the current DOCTYPE token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>This is a <span
   data-x="parse-error-missing-quote-before-doctype-public-identifier">missing-quote-before-doctype-public-identifier</span>
   <span>parse error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks
   flag</i> to <i>on</i>. <span>Reconsume</span> in the <span>bogus DOCTYPE state</span>.</dd>
  </dl>


  <h5><dfn>DOCTYPE public identifier (double-quoted) state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0022 QUOTATION MARK (&quot;)</dt>
   <dd>Switch to the <span>after DOCTYPE public identifier state</span>.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Append a U+FFFD REPLACEMENT CHARACTER character to the current DOCTYPE token's
   public identifier.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>This is an <span
   data-x="parse-error-abrupt-doctype-public-identifier">abrupt-doctype-public-identifier</span>
   <span>parse error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks
   flag</i> to <i>on</i>. Switch to the <span>data state</span>. Emit the current DOCTYPE token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-doctype">eof-in-doctype</span> <span>parse
   error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks flag</i> to
   <i>on</i>. Emit the current DOCTYPE token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Append the <span>current input character</span> to the current DOCTYPE token's public
   identifier.</dd>
  </dl>


  <h5><dfn>DOCTYPE public identifier (single-quoted) state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0027 APOSTROPHE (')</dt>
   <dd>Switch to the <span>after DOCTYPE public identifier state</span>.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Append a U+FFFD REPLACEMENT CHARACTER character to the current DOCTYPE token's
   public identifier.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>This is an <span
   data-x="parse-error-abrupt-doctype-public-identifier">abrupt-doctype-public-identifier</span>
   <span>parse error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks
   flag</i> to <i>on</i>. Switch to the <span>data state</span>. Emit the current DOCTYPE token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-doctype">eof-in-doctype</span> <span>parse
   error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks flag</i> to
   <i>on</i>. Emit the current DOCTYPE token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Append the <span>current input character</span> to the current DOCTYPE token's public
   identifier.</dd>
  </dl>


  <h5><dfn>After DOCTYPE public identifier state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>Switch to the <span>between DOCTYPE public and system identifiers state</span>.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>Switch to the <span>data state</span>. Emit the current DOCTYPE token.</dd>

   <dt>U+0022 QUOTATION MARK (&quot;)</dt>
   <dd>This is a <span
   data-x="parse-error-missing-whitespace-between-doctype-public-and-system-identifiers">missing-whitespace-between-doctype-public-and-system-identifiers</span>
   <span>parse error</span>. Set the current DOCTYPE token's system identifier to the empty string (not
   missing), then switch to the <span>DOCTYPE system identifier (double-quoted) state</span>.</dd>

   <dt>U+0027 APOSTROPHE (')</dt>
   <dd>This is a <span
   data-x="parse-error-missing-whitespace-between-doctype-public-and-system-identifiers">missing-whitespace-between-doctype-public-and-system-identifiers</span>
   <span>parse error</span>. Set the current DOCTYPE token's system identifier to the empty string (not
   missing), then switch to the <span>DOCTYPE system identifier (single-quoted) state</span>.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-doctype">eof-in-doctype</span> <span>parse
   error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks flag</i> to
   <i>on</i>. Emit the current DOCTYPE token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>This is a <span
   data-x="parse-error-missing-quote-before-doctype-system-identifier">missing-quote-before-doctype-system-identifier</span>
   <span>parse error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks
   flag</i> to <i>on</i>. <span>Reconsume</span> in the <span>bogus DOCTYPE state</span>.</dd>
  </dl>


  <h5><dfn>Between DOCTYPE public and system identifiers state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>Ignore the character.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>Switch to the <span>data state</span>. Emit the current DOCTYPE token.</dd>

   <dt>U+0022 QUOTATION MARK (&quot;)</dt>
   <dd>Set the current DOCTYPE token's system identifier to the empty string (not missing), then switch to
   the <span>DOCTYPE system identifier (double-quoted) state</span>.</dd>

   <dt>U+0027 APOSTROPHE (')</dt>
   <dd>Set the current DOCTYPE token's system identifier to the empty string (not missing), then switch to
   the <span>DOCTYPE system identifier (single-quoted) state</span>.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-doctype">eof-in-doctype</span> <span>parse
   error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks flag</i> to
   <i>on</i>. Emit the current DOCTYPE token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>This is a <span
   data-x="parse-error-missing-quote-before-doctype-system-identifier">missing-quote-before-doctype-system-identifier</span>
   <span>parse error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks
   flag</i> to <i>on</i>. <span>Reconsume</span> in the <span>bogus DOCTYPE state</span>.</dd>
  </dl>


  <h5><dfn>After DOCTYPE system keyword state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>Switch to the <span>before DOCTYPE system identifier state</span>.</dd>

   <dt>U+0022 QUOTATION MARK (&quot;)</dt>
   <dd>This is a <span
   data-x="parse-error-missing-whitespace-after-doctype-system-keyword">missing-whitespace-after-doctype-system-keyword</span>
   <span>parse error</span>. Set the current DOCTYPE token's system identifier to the empty string (not
   missing), then switch to the <span>DOCTYPE system identifier (double-quoted) state</span>.</dd>

   <dt>U+0027 APOSTROPHE (')</dt>
   <dd>This is a <span
   data-x="parse-error-missing-whitespace-after-doctype-system-keyword">missing-whitespace-after-doctype-system-keyword</span>
   <span>parse error</span>. Set the current DOCTYPE token's system identifier to the empty string (not
   missing), then switch to the <span>DOCTYPE system identifier (single-quoted) state</span>.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>This is a <span
   data-x="parse-error-missing-doctype-system-identifier">missing-doctype-system-identifier</span>
   <span>parse error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks
   flag</i> to <i>on</i>. Switch to the <span>data state</span>. Emit the current DOCTYPE token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-doctype">eof-in-doctype</span> <span>parse
   error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks flag</i> to
   <i>on</i>. Emit the current DOCTYPE token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>This is a <span
   data-x="parse-error-missing-quote-before-doctype-system-identifier">missing-quote-before-doctype-system-identifier</span>
   <span>parse error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks
   flag</i> to <i>on</i>. <span>Reconsume</span> in the <span>bogus DOCTYPE state</span>.</dd>
  </dl>


  <h5><dfn>Before DOCTYPE system identifier state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>Ignore the character.</dd>

   <dt>U+0022 QUOTATION MARK (&quot;)</dt>
   <dd>Set the current DOCTYPE token's system identifier to the empty string (not missing), then switch to
   the <span>DOCTYPE system identifier (double-quoted) state</span>.</dd>

   <dt>U+0027 APOSTROPHE (')</dt>
   <dd>Set the current DOCTYPE token's system identifier to the empty string (not missing), then switch to
   the <span>DOCTYPE system identifier (single-quoted) state</span>.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>This is a <span
   data-x="parse-error-missing-doctype-system-identifier">missing-doctype-system-identifier</span> <span>parse
   error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks flag</i> to
   <i>on</i>. Switch to the <span>data state</span>. Emit the current DOCTYPE token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-doctype">eof-in-doctype</span> <span>parse
   error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks flag</i> to
   <i>on</i>. Emit the current DOCTYPE token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>This is a <span
   data-x="parse-error-missing-quote-before-doctype-system-identifier">missing-quote-before-doctype-system-identifier</span>
   <span>parse error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks
   flag</i> to <i>on</i>. <span>Reconsume</span> in the <span>bogus DOCTYPE state</span>.</dd>
  </dl>


  <h5><dfn>DOCTYPE system identifier (double-quoted) state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0022 QUOTATION MARK (&quot;)</dt>
   <dd>Switch to the <span>after DOCTYPE system identifier state</span>.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Append a U+FFFD REPLACEMENT CHARACTER character to the current DOCTYPE token's
   system identifier.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>This is an <span
   data-x="parse-error-abrupt-doctype-system-identifier">abrupt-doctype-system-identifier</span>
   <span>parse error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks
   flag</i> to <i>on</i>. Switch to the <span>data state</span>. Emit the current DOCTYPE token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-doctype">eof-in-doctype</span> <span>parse
   error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks flag</i> to
   <i>on</i>. Emit the current DOCTYPE token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Append the <span>current input character</span> to the current DOCTYPE token's system
   identifier.</dd>
  </dl>


  <h5><dfn>DOCTYPE system identifier (single-quoted) state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0027 APOSTROPHE (')</dt>
   <dd>Switch to the <span>after DOCTYPE system identifier state</span>.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Append a U+FFFD REPLACEMENT CHARACTER character to the current DOCTYPE token's
   system identifier.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>This is an <span
   data-x="parse-error-abrupt-doctype-system-identifier">abrupt-doctype-system-identifier</span>
   <span>parse error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks
   flag</i> to <i>on</i>. Switch to the <span>data state</span>. Emit the current DOCTYPE token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-doctype">eof-in-doctype</span> <span>parse
   error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks flag</i> to
   <i>on</i>. Emit the current DOCTYPE token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Append the <span>current input character</span> to the current DOCTYPE token's system
   identifier.</dd>
  </dl>


  <h5><dfn>After DOCTYPE system identifier state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+0009 CHARACTER TABULATION (tab)</dt>
   <dt>U+000A LINE FEED (LF)</dt>
   <dt>U+000C FORM FEED (FF)</dt>
   <!--<dt>U+000D CARRIAGE RETURN (CR)</dt>-->
   <dt>U+0020 SPACE</dt>
   <dd>Ignore the character.</dd>

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>Switch to the <span>data state</span>. Emit the current DOCTYPE token.</dd>

   <dt>EOF</dt>
   <dd>This is an <span data-x="parse-error-eof-in-doctype">eof-in-doctype</span> <span>parse
   error</span>. Set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks flag</i> to
   <i>on</i>. Emit the current DOCTYPE token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-character-after-doctype-system-identifier">unexpected-character-after-doctype-system-identifier</span>
   <span>parse error</span>. <span>Reconsume</span> in the <span>bogus DOCTYPE state</span>. (This
   does <em>not</em> set the current DOCTYPE token's <i data-x="force-quirks flag">force-quirks flag</i> to
   <i>on</i>.)</dd>
  </dl>


  <h5><dfn>Bogus DOCTYPE state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt>U+003E GREATER-THAN SIGN (&gt;)</dt>
   <dd>Switch to the <span>data state</span>. Emit the DOCTYPE token.</dd>

   <dt>U+0000 NULL</dt>
   <dd>This is an <span
   data-x="parse-error-unexpected-null-character">unexpected-null-character</span> <span>parse
   error</span>. Ignore the character.</dd>

   <dt>EOF</dt>
   <dd>Emit the DOCTYPE token. Emit an end-of-file token.</dd>

   <dt>Anything else</dt>
   <dd>Ignore the character.</dd>
  </dl>


  <h5><dfn>CDATA section state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

    <dt>U+005D RIGHT SQUARE BRACKET (])</dt>
    <dd>Switch to the <span>CDATA section bracket state</span>.</dd>

    <dt>EOF</dt>
    <dd>This is an <span data-x="parse-error-eof-in-cdata">eof-in-cdata</span> <span>parse
    error</span>. Emit an end-of-file token.</dd>

    <dt>Anything else</dt>
    <dd>Emit the <span>current input character</span> as a character token.</dd>
  </dl>

  <p class="note">U+0000 NULL characters are handled in the tree construction stage, as part of the
  <span data-x="insertion mode: in foreign content">in foreign content</span> insertion mode, which
  is the only place where CDATA sections can appear.</p>

  <h5><dfn>CDATA section bracket state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

    <dt>U+005D RIGHT SQUARE BRACKET (])</dt>
    <dd>Switch to the <span>CDATA section end state</span>.</dd>

    <dt>Anything else</dt>
    <dd>Emit a U+005D RIGHT SQUARE BRACKET character token. <span>Reconsume</span> in the
    <span>CDATA section state</span>.</dd>
  </dl>


  <h5><dfn>CDATA section end state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

    <dt>U+005D RIGHT SQUARE BRACKET (])</dt>
    <dd>Emit a U+005D RIGHT SQUARE BRACKET character token.</dd>

    <dt>U+003E GREATER-THAN SIGN (>)</dt>
    <dd>Switch to the <span>data state</span>.</dd>

    <dt>Anything else</dt>
    <dd>Emit two U+005D RIGHT SQUARE BRACKET character tokens. <span>Reconsume</span> in the
    <span>CDATA section state</span>.</dd>
  </dl>



  <h5><dfn>Character reference state</dfn></h5>

  <p>Set the <var data-x="temporary buffer">temporary buffer</var> to the empty string. Append
   a U+0026 AMPERSAND (&amp;) character to the <var data-x="temporary buffer">temporary
   buffer</var>. Consume the <span>next input character</span>:</p>

  <dl class="switch">

   <dt><span data-x="ASCII alphanumeric">ASCII alphanumeric</span></dt>
   <dd><span>Reconsume</span> in the <span>named character reference state</span>.</dd>

   <dt>U+0023 NUMBER SIGN (#)</dt>
   <dd>Append the <span>current input character</span> to the
   <var data-x="temporary buffer">temporary buffer</var>. Switch to the <span>numeric character
   reference state</span>.</dd>

   <dt>Anything else</dt>
   <dd><span>Flush code points consumed as a character reference</span>. <span>Reconsume</span> in
   the <var data-x="return state">return state</var>.</dd>
  </dl>


  <h5><dfn>Named character reference state</dfn></h5>

  <p>Consume the maximum number of characters possible, where the consumed characters are one of the
  identifiers in the first column of the <span>named character references</span> table. Append each
  character to the <var data-x="temporary buffer">temporary buffer</var> when it's consumed.</p>

  <dl class="switch">

   <dt>If there is a match</dt>

   <dd>
    <p>If the character reference was <span data-x="charref-in-attribute">consumed as part of an
    attribute</span>, and the last character matched is not a U+003B SEMICOLON character (;), and
    the <span>next input character</span> is either a U+003D EQUALS SIGN character (=) or an
    <span>ASCII alphanumeric</span>, then, for historical reasons, <span>flush code points consumed
    as a character reference</span> and switch to the <var data-x="return state">return state</var>.
    </p>
    <!-- "=" added because of https://www.w3.org/Bugs/Public/show_bug.cgi?id=9207#c5 -->

    <p>Otherwise:</p>

    <ol>
     <li><p>If the last character matched is not a U+003B SEMICOLON character (;), then this is a
     <span
     data-x="parse-error-missing-semicolon-after-character-reference">missing-semicolon-after-character-reference</span>
     <span>parse error</span>.</p></li>

     <li><p>Set the <var data-x="temporary buffer">temporary buffer</var> to the empty string.
     Append one or two characters corresponding to the character reference name (as given by the
     second column of the <span>named character references</span> table) to the <var
     data-x="temporary buffer">temporary buffer</var>.</p></li>

     <li><span>Flush code points consumed as a character reference</span>. Switch to the <var
     data-x="return state">return state</var>.</li>
    </ol>
   </dd>

   <dt>Otherwise</dt>

   <dd><span>Flush code points consumed as a character reference</span>. Switch to the
   <span>ambiguous ampersand state</span>.</dd>
  </dl>

  <div class="example">

    <p>If the markup contains (not in an attribute) the string <code data-x="">I'm &amp;notit; I
    tell you</code>, the character reference is parsed as "not", as in, <code data-x="">I'm &not;it;
    I tell you</code> (and this is a parse error). But if the markup was <code data-x="">I'm
    &amp;notin; I tell you</code>, the character reference would be parsed as "notin;", resulting
    in <code data-x="">I'm &notin; I tell you</code> (and no parse error).</p>

    <p>However, if the markup contains the string <code data-x="">I'm &amp;notit; I tell you</code>
    in an attribute, no character reference is parsed and string remains intact (and there is no
    parse error).</p>

  </div>


  <h5><dfn>Ambiguous ampersand state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

    <dt><span data-x="ASCII alphanumeric">ASCII alphanumeric</span></dt>
    <dd>If the character reference was <span data-x="charref-in-attribute">consumed as part of an
    attribute</span>, then append the <span>current input character</span> to the current
    attribute's value. Otherwise, emit the <span>current input character</span> as a character
    token.</dd>

    <dt>U+003B SEMICOLON (;)</dt>
    <dd>This is an <span
    data-x="parse-error-unknown-named-character-reference">unknown-named-character-reference</span>
    <span>parse error</span>. <span>Reconsume</span> in the <var data-x="return state">return
    state</var>.

    <dt>Anything else</dt>
    <dd><span>Reconsume</span> in the <var data-x="return state">return state</var>.</dd>
  </dl>

  <h5><dfn>Numeric character reference state</dfn></h5>

  <p>Set the <dfn><var data-x="character reference code">character reference code</var></dfn> to
  zero (0).</p>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

    <dt>U+0078 LATIN SMALL LETTER X</dt>
    <dt>U+0058 LATIN CAPITAL LETTER X</dt>
    <dd>Append the <span>current input character</span> to the <var data-x="temporary
    buffer">temporary buffer</var>. Switch to the <span>hexadecimal character reference start
    state</span>.</dd>

    <dt>Anything else</dt>
    <dd><span>Reconsume</span> in the <span>decimal character reference start state</span>.</dd>
  </dl>


  <h5><dfn>Hexadecimal character reference start state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

    <dt><span data-x="ASCII hex digits">ASCII hex digit</span></dt>
    <dd><span>Reconsume</span> in the <span>hexadecimal character reference state</span>.</dd>

    <dt>Anything else</dt>
    <dd>This is an <span
    data-x="parse-error-absence-of-digits-in-numeric-character-reference">absence-of-digits-in-numeric-character-reference</span>
    <span>parse error</span>. <span>Flush code points consumed as a character reference</span>.
    <span>Reconsume</span> in the <var data-x="return state">return state</var>.</dd>
  </dl>


  <h5><dfn>Decimal character reference start state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

    <dt><span data-x="ASCII digits">ASCII digit</span></dt>
    <dd><span>Reconsume</span> in the <span>decimal character reference state</span>.</dd>

    <dt>Anything else</dt>
    <dd>This is an <span
    data-x="parse-error-absence-of-digits-in-numeric-character-reference">absence-of-digits-in-numeric-character-reference</span>
    <span>parse error</span>. <span>Flush code points consumed as a character reference</span>.
    <span>Reconsume</span> in the <var data-x="return state">return state</var>.</dd>
  </dl>


  <h5><dfn>Hexadecimal character reference state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

    <dt><span data-x="ASCII digits">ASCII digit</span></dt>
    <dd>Multiply the <var data-x="character reference code">character reference code</var> by 16.
    Add a numeric version of the <span>current input character</span> (subtract 0x0030 from the
    character's code point) to the <var data-x="character reference code">character reference
    code</var>.</dd>

    <dt><span>ASCII upper hex digit</span></dt>
    <dd>Multiply the <var data-x="character reference code">character reference code</var> by 16.
    Add a numeric version of the <span>current input character</span> as a hexadecimal digit
    (subtract 0x0037 from the character's code point) to the <var data-x="character reference
    code">character reference code</var>.</dd>

    <dt><span>ASCII lower hex digit</span></dt>
    <dd>Multiply the <var data-x="character reference code">character reference code</var> by 16.
    Add a numeric version of the <span>current input character</span> as a hexadecimal digit
    (subtract 0x0057 from the character's code point) to the <var data-x="character reference
    code">character reference code</var>.</dd>

    <dt>U+003B SEMICOLON (;)</dt>
    <dd>Switch to the <span>numeric character reference end state</span>.</dd>

    <dt>Anything else</dt>
    <dd>This is a <span
    data-x="parse-error-missing-semicolon-after-character-reference">missing-semicolon-after-character-reference</span>
    <span>parse error</span>. <span>Reconsume</span> in the <span>numeric character reference end
    state</span>.</dd>
  </dl>


  <h5><dfn>Decimal character reference state</dfn></h5>

  <p>Consume the <span>next input character</span>:</p>

  <dl class="switch">

    <dt><span data-x="ASCII digits">ASCII digit</span></dt>
    <dd>Multiply the <var data-x="character reference code">character reference code</var> by 10.
    Add a numeric version of the <span>current input character</span> (subtract 0x0030 from the
    character's code point) to the <var data-x="character reference code">character reference
    code</var>.</dd>

    <dt>U+003B SEMICOLON (;)</dt>
    <dd>Switch to the <span>numeric character reference end state</span>.</dd>

    <dt>Anything else</dt>
    <dd>This is a <span
    data-x="parse-error-missing-semicolon-after-character-reference">missing-semicolon-after-character-reference</span>
    <span>parse error</span>. <span>Reconsume</span> in the <span>numeric character reference end
    state</span>.</dd>
  </dl>


  <h5><dfn>Numeric character reference end state</dfn></h5>

  <p>Check the <var data-x="character reference code">character reference code</var>:</p>

  <ul>
    <li><p>If the number is 0x00, then this is a <span
    data-x="parse-error-null-character-reference">null-character-reference</span> <span>parse
    error</span>. Set the <var data-x="character reference code">character reference code</var> to
    0xFFFD.</p></li>

    <li><p>If the number is greater than 0x10FFFF, then this is a <span
    data-x="parse-error-character-reference-outside-unicode-range">character-reference-outside-unicode-range</span>
    <span>parse error</span>. Set the <var data-x="character reference code">character reference
    code</var> to 0xFFFD.</p></li>

    <li><p>If the number is a <span data-x="surrogate">surrogate</span>, then this is a <span
    data-x="parse-error-surrogate-character-reference">surrogate-character-reference</span>
    <span>parse error</span>. Set the <var data-x="character reference code">character reference
    code</var> to 0xFFFD.</p></li>

    <li><p>If the number is a <span data-x="noncharacter">noncharacter</span>, then this is a
    <span data-x="parse-error-noncharacter-character-reference">noncharacter-character-reference</span>
    <span>parse error</span>.</p></li>

    <li><p>If the number is 0x0D<!-- CR is not allowed -->, or a
    <span data-x="control">control</span> that's not <span>ASCII whitespace</span>, then this is a
    <span data-x="parse-error-control-character-reference">control-character-reference</span>
    <span>parse error</span>. If the number is one of the numbers in the first column of the
    following table, then find the row with that number in the first column, and set the <var
    data-x="character reference code">character reference code</var> to the number in the second
    column of that row.</p>
    <!-- these are Unicode C1 control characters -->

    <table id="table-charref-overrides">
      <thead>
      <tr><th>Number <th colspan=2>Code point
      <tbody>
      <!-- <tr><td>0x0D <td>0x000D <td>CARRIAGE RETURN (CR) -->
      <tr><td>0x80 <td>0x20AC <td>EURO SIGN (&#x20AC;)
      <!-- <tr><td>0x81 <td>0x0081 <td>&lt;control> -->
      <tr><td>0x82 <td>0x201A <td>SINGLE LOW-9 QUOTATION MARK (&#x201A;)
      <tr><td>0x83 <td>0x0192 <td>LATIN SMALL LETTER F WITH HOOK (&#x0192;)
      <tr><td>0x84 <td>0x201E <td>DOUBLE LOW-9 QUOTATION MARK (&#x201E;)
      <tr><td>0x85 <td>0x2026 <td>HORIZONTAL ELLIPSIS (&#x2026;)
      <tr><td>0x86 <td>0x2020 <td>DAGGER (&#x2020;)
      <tr><td>0x87 <td>0x2021 <td>DOUBLE DAGGER (&#x2021;)
      <tr><td>0x88 <td>0x02C6 <td>MODIFIER LETTER CIRCUMFLEX ACCENT (&#x02C6;)
      <tr><td>0x89 <td>0x2030 <td>PER MILLE SIGN (&#x2030;)
      <tr><td>0x8A <td>0x0160 <td>LATIN CAPITAL LETTER S WITH CARON (&#x0160;)
      <tr><td>0x8B <td>0x2039 <td>SINGLE LEFT-POINTING ANGLE QUOTATION MARK (&#x2039;)
      <tr><td>0x8C <td>0x0152 <td>LATIN CAPITAL LIGATURE OE (&#x0152;)
      <!-- <tr><td>0x8D <td>0x008D <td>&lt;control> -->
      <tr><td>0x8E <td>0x017D <td>LATIN CAPITAL LETTER Z WITH CARON (&#x017D;)
      <!-- <tr><td>0x8F <td>0x008F <td>&lt;control> -->
      <!-- <tr><td>0x90 <td>0x0090 <td>&lt;control> -->
      <tr><td>0x91 <td>0x2018 <td>LEFT SINGLE QUOTATION MARK (&#x2018;)
      <tr><td>0x92 <td>0x2019 <td>RIGHT SINGLE QUOTATION MARK (&#x2019;)
      <tr><td>0x93 <td>0x201C <td>LEFT DOUBLE QUOTATION MARK (&#x201C;)
      <tr><td>0x94 <td>0x201D <td>RIGHT DOUBLE QUOTATION MARK (&#x201D;)
      <tr><td>0x95 <td>0x2022 <td>BULLET (&#x2022;)
      <tr><td>0x96 <td>0x2013 <td>EN DASH (&#x2013;)
      <tr><td>0x97 <td>0x2014 <td>EM DASH (&#x2014;)
      <tr><td>0x98 <td>0x02DC <td>SMALL TILDE (&#x02DC;)
      <tr><td>0x99 <td>0x2122 <td>TRADE MARK SIGN (&#x2122;)
      <tr><td>0x9A <td>0x0161 <td>LATIN SMALL LETTER S WITH CARON (&#x0161;)
      <tr><td>0x9B <td>0x203A <td>SINGLE RIGHT-POINTING ANGLE QUOTATION MARK (&#x203A;)
      <tr><td>0x9C <td>0x0153 <td>LATIN SMALL LIGATURE OE (&#x0153;)
      <!-- <tr><td>0x9D <td>0x009D <td>&lt;control> -->
      <tr><td>0x9E <td>0x017E <td>LATIN SMALL LETTER Z WITH CARON (&#x017E;)
      <tr><td>0x9F <td>0x0178 <td>LATIN CAPITAL LETTER Y WITH DIAERESIS (&#x0178;)
    </table></li>
  </ul>

  <p>Set the <var data-x="temporary buffer">temporary buffer</var> to the empty string. Append a
  code point equal to the <var data-x="character reference code">character reference code</var> to
  the <var data-x="temporary buffer">temporary buffer</var>. <span>Flush code points consumed as a
  character reference</span>. Switch to the <var data-x="return state">return state</var>.</p>

  </div>


  <div w-nodev>

  <!-- v2: One thing that this doesn't define is handling deeply nested documents. There are
  compatibility requirements around that: you can't throw away the elements altogether, consider Tux
  made only with opening <font> elements, one per character. Seems that the best thing to do is to
  close some formatting elements from the middle of the stack when you hit a limit, or something.
  -->

  <h4><dfn>Tree construction</dfn></h4>

  <p>The input to the tree construction stage is a sequence of tokens from the
  <span>tokenization</span> stage. The tree construction stage is associated with a DOM
  <code>Document</code> object when a parser is created. The "output" of this stage consists of
  dynamically modifying or extending that document's DOM tree.</p>

  <p>This specification does not define when an interactive user agent has to render the
  <code>Document</code> so that it is available to the user, or when it has to begin accepting user
  input.</p>

  <hr>

  <p>As each token is emitted from the tokenizer, the user agent must follow the appropriate steps
  from the following list, known as the <dfn>tree construction dispatcher</dfn>:</p>

  <dl class="switch">

   <dt>If the <span>stack of open elements</span> is empty</dt>
   <dt>If the <span>adjusted current node</span> is an element in the <span>HTML namespace</span></dt>
   <dt>If the <span>adjusted current node</span> is a <span>MathML text integration point</span> and the token is a start tag whose tag name is neither "mglyph" nor "malignmark"</dt>
   <dt>If the <span>adjusted current node</span> is a <span>MathML text integration point</span> and the token is a character token</dt>
   <dt>If the <span>adjusted current node</span> is a <span>MathML <code>annotation-xml</code></span> element and the token is a start tag whose tag name is "svg"</dt>
   <dt>If the <span>adjusted current node</span> is an <span>HTML integration point</span> and the token is a start tag</dt>
   <dt>If the <span>adjusted current node</span> is an <span>HTML integration point</span> and the token is a character token</dt>
   <dt>If the token is an end-of-file token</dt>

   <dd>Process the token according to the rules given in the section corresponding to the current
   <span>insertion mode</span> in HTML content.</dd>

   <dt>Otherwise</dt>

   <dd>Process the token according to the rules given in the section for parsing tokens <span
   data-x="insertion mode: in foreign content">in foreign content</span>.</dd>
  </dl>

  <p>The <dfn>next token</dfn> is the token that is about to be processed by the <span>tree
  construction dispatcher</span> (even if the token is subsequently just ignored).</p>

  <p>A node is a <dfn>MathML text integration point</dfn> if it is one of the following
  elements:</p>

  <ul class="brief">
   <li>A <span>MathML <code>mi</code></span> element</li>
   <li>A <span>MathML <code>mo</code></span> element</li>
   <li>A <span>MathML <code>mn</code></span> element</li>
   <li>A <span>MathML <code>ms</code></span> element</li>
   <li>A <span>MathML <code>mtext</code></span> element</li>
  </ul>

  <p>A node is an <dfn>HTML integration point</dfn> if it is one of the following elements:</p>

  <ul class="brief">
   <li>A <span>MathML <code>annotation-xml</code></span> element whose start tag token had an
   attribute with the name "encoding" whose value was an <span>ASCII case-insensitive</span> match
   for the string "<code data-x="">text/html</code>"</li>
   <li>A <span>MathML <code>annotation-xml</code></span> element whose start tag token had an
   attribute with the name "encoding" whose value was an <span>ASCII case-insensitive</span> match
   for the string "<code data-x="">application/xhtml+xml</code>"</li>
   <li>An <span>SVG <code>foreignObject</code></span> element</li>
   <li>An <span>SVG <code>desc</code></span> element</li>
   <li>An <span>SVG <code>title</code></span> element</li>
  </ul>

  <p class="note">If the node in question is the <var
  data-x="concept-frag-parse-context">context</var> element passed to the <span>HTML fragment
  parsing algorithm</span>, then the start tag token for that element is the "fake" token created
  during by that <span>HTML fragment parsing algorithm</span>.</p>

  <hr>

  <p class="note">Not all of the tag names mentioned below are conformant tag names in this
  specification; many are included to handle legacy content. They still form part of the algorithm
  that implementations are required to implement to claim conformance.</p>

  <p class="note">The algorithm described below places no limit on the depth of the DOM tree
  generated, or on the length of tag names, attribute names, attribute values, <code>Text</code>
  nodes, etc. While implementers are encouraged to <a
  href="https://infra.spec.whatwg.org/#algorithm-limits">avoid arbitrary limits</a>, it is
  recognized that practical concerns will likely force user agents to impose nesting depth
  constraints.</p>


  <h5>Creating and inserting nodes</h5>

  <p>While the parser is processing a token, it can enable or disable <dfn data-x="foster
  parent">foster parenting</dfn>. This affects the following algorithm.</p>

  <p>The <dfn>appropriate place for inserting a node</dfn>, optionally using a particular
  <i>override target</i>, is the position in an element returned by running the following steps:</p>

  <ol>
   <li>
    <p>If there was an <i>override target</i> specified, then let <var>target</var> be the
    <i>override target</i>.</p>

    <p>Otherwise, let <var>target</var> be the <span>current node</span>.</p>
   </li>

   <li>
    <p>Determine the <var>adjusted insertion location</var> using the first matching steps
    from the following list:</p>

    <dl class="switch">

     <dt>If <span data-x="foster parent">foster parenting</span> is enabled and <var>target</var> is a <code>table</code>, <code>tbody</code>, <code>tfoot</code>,
     <code>thead</code>, or <code>tr</code> element</dt> <!-- same list as handling of characters
     in "in table" mode -->

     <dd>
      <p class="note">Foster parenting happens when content is misnested in tables.</p>

      <p>Run these substeps:</p>

      <ol>
       <li><p>Let <var>last template</var> be the last <code>template</code> element in the
       <span>stack of open elements</span>, if any.</p>

       <li><p>Let <var>last table</var> be the last <code>table</code> element in the
       <span>stack of open elements</span>, if any.</p>

       <li><p>If there is a <var>last template</var> and either there is no <var>last table</var>, or there is one, but <var>last template</var> is lower
       (more recently added) than <var>last table</var> in the <span>stack of open
       elements</span>, then: let <var>adjusted insertion location</var> be inside <var>last template</var>'s <span>template contents</span>, after its last child (if any),
       and abort these steps.</p></li>

       <li><p>If there is no <var>last table</var>, <!-- there's also implicitly no last
       template, since we didn't hit the previous step --> then let <var>adjusted insertion
       location</var> be inside the first element in the <span>stack of open elements</span> (the
       <code>html</code> element), after its last child (if any), and abort these steps.
       (<span>fragment case</span>)</p>

       <!-- if we get here, we know there's a last table, and if there's a last template, it's older
       than the last table. -->

       <li><p>If <var>last table</var> has a parent node, then let <var>adjusted insertion location</var> be inside <var>last table</var>'s parent
       node, immediately before <var>last table</var>, and abort these steps.</p></li>

       <!-- if we get here, we know there's a last table, but it has no parent, and if there's a
       last template, it's older than the last table. -->

       <li><p>Let <var>previous element</var> be the element immediately above <var>last table</var> in the <span>stack of open elements</span>.</p></li>

       <li><p>Let <var>adjusted insertion location</var> be inside <var>previous
       element</var>, after its last child (if any).</p></li>
      </ol>

      <p class="note">These steps are involved in part because it's possible for elements, the
      <code>table</code> element in this case in particular, to have been moved by a script around
      in the DOM, or indeed removed from the DOM entirely, after the element was inserted by the
      parser.</p>
     </dd>

     <dt>Otherwise</dt>

     <dd>
      <p>Let <var>adjusted insertion location</var> be inside <var>target</var>,
      after its last child (if any).</p>
     </dd>
    </dl>
   </li>

   <li>
    <p>If the <var>adjusted insertion location</var> is inside a <code>template</code>
    element, let it instead be inside the <code>template</code> element's <span>template
    contents</span>, after its last child (if any).</p>
   </li>

   <li>
    <p>Return the <var>adjusted insertion location</var>.</p>
   </li>
  </ol>

  <hr>

  <p>When the steps below require the UA to <dfn data-x="create an element for the token">create an
  element for a token</dfn> in a particular <var>given namespace</var> and with a
  particular <var>intended parent</var>, the UA must run the following steps:</p>

  <ol>
   <li><p>If the <span>active speculative HTML parser</span> is not null, then return the result of
   <span data-x="create a speculative mock element">creating a speculative mock element</span>
   given <var>given namespace</var>, the tag name of the given token, and the attributes of the
   given token.</p></li>

   <li>
    <p>Otherwise, optionally <span>create a speculative mock element</span> given <var>given
    namespace</var>, the tag name of the given token, and the attributes of the given token.</p>

    <p class="note">The result is not used. This step allows for a <span>speculative fetch</span> to
    be initiated from non-speculative parsing. The fetch is still speculative at this point,
    because, for example, by the time the element is inserted, <var>intended parent</var> might
    have been removed from the document.</p>
   </li>

   <li><p>Let <var>document</var> be <var>intended parent</var>'s <span>node document</span>.</li>

   <li><p>Let <var>local name</var> be the tag name of the token.</p></li>

   <li><p>Let <var>is</var> be the value of the "<code data-x="attr-is">is</code>" attribute in the
   given token, if such an attribute exists, or null otherwise.</p></li>

   <li><p>Let <var>definition</var> be the result of <span data-x="look up a custom element
   definition">looking up a custom element definition</span> given <var>document</var>, <var>given
   namespace</var>, <var>local name</var>, and <var>is</var>.</p></li>

   <li><p>Let <var>willExecuteScript</var> be true if <var>definition</var> is non-null and the
   parser was not created as part of the <span>HTML fragment parsing algorithm</span>; otherwise
   false.</p></li>

   <li>
    <p>If <var>willExecuteScript</var> is true:</p>

    <ol>
     <li><p>Increment <var>document</var>'s <span>throw-on-dynamic-markup-insertion
     counter</span>.</p></li>

     <li><p>If the <span>JavaScript execution context stack</span> is empty, then <span>perform a
     microtask checkpoint</span>.</p></li>

     <li><p>Push a new <span>element queue</span> onto <var>document</var>'s <span>relevant
     agent</span>'s <span>custom element reactions stack</span>.</li>
    </ol>
   </li>

   <li>
    <p>Let <var>element</var> be the result of <span data-x="create an element">creating an
    element</span> given <var>document</var>, <var>localName</var>, <var>given namespace</var>,
    null, <var>is</var>, and <var>willExecuteScript</var>.</p>

    <p class="note">This will cause <span data-x="custom element constructor">custom element
    constructors</span> to run, if <var>willExecuteScript</var> is true. However, since we
    incremented the <span>throw-on-dynamic-markup-insertion counter</span>, this cannot cause <span
    data-x="dom-document-write">new characters to be inserted into the tokenizer</span>, or <span
    data-x="dom-document-open">the document to be blown away</span>.</p>
   </li>

   <li>
    <p><span data-x="concept-element-attributes-append">Append</span> each attribute in the given
    token to <var>element</var>.</p>

    <p class="note">This can <span>enqueue a custom element callback reaction</span> for the
    <code data-x="">attributeChangedCallback</code>, which might run immediately (in the next
    step).</p>

    <p class="note">Even though the <code data-x="attr-is">is</code> attribute governs the <span
    data-x="create an element">creation</span> of a <span>customized built-in element</span>, it is
    not present during the execution of the relevant <span>custom element constructor</span>; it is
    appended in this step, along with all other attributes.</p>
   </li>

   <li>
    <p>If <var>willExecuteScript</var> is true:</p>

    <ol>
     <li><p>Let <var>queue</var> be the result of popping from <var>document</var>'s <span>relevant
     agent</span>'s <span>custom element reactions stack</span>. (This will be the same
     <span>element queue</span> as was pushed above.)</p></li>

     <li><p><span>Invoke custom element reactions</span> in <var>queue</var>.</p></li>

     <li><p>Decrement <var>document</var>'s <span>throw-on-dynamic-markup-insertion
     counter</span>.</p></li>
    </ol>
   </li>

   <li><p>If <var>element</var> has an <code data-x="">xmlns</code> attribute <em>in the <span>XMLNS
   namespace</span></em> whose value is not exactly the same as the element's namespace, that is a
   <span>parse error</span>. Similarly, if <var>element</var> has an <code
   data-x="">xmlns:xlink</code> attribute in the <span>XMLNS namespace</span> whose value is not the
   <span>XLink Namespace</span>, that is a <span>parse error</span>.</p></li>

   <li><p>If <var>element</var> is a <span data-x="category-reset">resettable element</span> and not
   a <span>form-associated custom element</span>, then invoke its <span
   data-x="concept-form-reset-control">reset algorithm</span>. (This initializes the element's <span
   data-x="concept-fe-value">value</span> and <span data-x="concept-fe-checked">checkedness</span>
   based on the element's attributes.)</p></li>

   <li><p>If <var>element</var> is a <span>form-associated element</span> and not a
   <span>form-associated custom element</span>, the
   <span><code data-x="">form</code> element pointer</span> is not null, there is no
   <code>template</code> element on the <span>stack of open elements</span>, <var>element</var> is
   either not <span data-x="category-listed">listed</span> or doesn't have a <code
   data-x="attr-fae-form">form</code> attribute, and the <var>intended parent</var> is in the same
   <span>tree</span> as the element pointed to by the <span><code data-x="">form</code> element
   pointer</span>, then <span data-x="concept-form-association">associate</span> <var>element</var>
   with the <code>form</code> element pointed to by the <span><code data-x="">form</code> element
   pointer</span> and set <var>element</var>'s <span>parser inserted flag</span>.</p></li>

   <li><p>Return <var>element</var>.</p></li>
  </ol>

  <hr>

  <p>To <dfn>insert an element at the adjusted insertion location</dfn> with an element
  <var>element</var>:</p>

  <ol>
   <li><p>Let the <var>adjusted insertion location</var> be the <span>appropriate place for
   inserting a node</span>.</p></li>

   <li><p>If it is not possible to insert <var>element</var> at the <var>adjusted insertion
   location</var>, abort these steps.</p></li>

   <li><p>If the parser was not created as part of the <span>HTML fragment parsing
   algorithm</span>, then push a new <span>element queue</span> onto <var>element</var>'s
   <span>relevant agent</span>'s <span>custom element reactions stack</span>.</p></li>

   <li><p>Insert <var>element</var> at the <var>adjusted insertion location</var>.</p></li>

   <li><p>If the parser was not created as part of the <span>HTML fragment parsing
   algorithm</span>, then pop the <span>element queue</span> from <var>element</var>'s
   <span>relevant agent</span>'s <span>custom element reactions stack</span>, and <span>invoke
   custom element reactions</span> in that queue.</p></li>
  </ol>

  <p class="note">If the <var>adjusted insertion location</var> cannot accept more elements, e.g.,
  because it's a <code>Document</code> that already has an element child, then <var>element</var> is
  dropped on the floor.</p>

  <!-- The names of these algorithms are kinda confusing; e.g. see the confusion in
         https://www.w3.org/Bugs/Public/show_bug.cgi?id=18367
       Not sure what we could call them instead, though... -->

  <p>When the steps below require the user agent to <dfn>insert a foreign element</dfn> for a token
  in a given namespace and with a boolean <var>onlyAddToElementStack</var>, the user agent must run
  these steps:</p>

  <ol>
   <li><p>Let the <var>adjusted insertion location</var> be the <span>appropriate place for
   inserting a node</span>.</p></li>

   <!-- this insertion stuff should be cleaned up: https://github.com/whatwg/html/issues/1706 -->

   <li><p>Let <var>element</var> be the result of <span data-x="create an element for the
   token">creating an element for the token</span> in the given namespace, with the intended
   parent being the element in which the <var>adjusted insertion location</var> finds
   itself.</p></li>

   <li><p>If <var>onlyAddToElementStack</var> is false, then run <span>insert an element at the
   adjusted insertion location</span> with <var>element</var>.</p></li>

   <li><p>Push <var>element</var> onto the <span>stack of open elements</span> so that it is the new
   <span>current node</span>.</p></li>

   <li><p>Return <var>element</var>.</p></li>
  </ol>

  <p>When the steps below require the user agent to <dfn>insert an HTML element</dfn> for a token,
  the user agent must <span>insert a foreign element</span> for the token, with the <span>HTML
  namespace</span> and false.</p>

  <hr>

  <p>When the steps below require the user agent to <dfn>adjust MathML attributes</dfn> for a token,
  then, if the token has an attribute named <code data-x="">definitionurl</code>, change its name to
  <code data-x="">definitionURL</code> (note the case difference).</p>

  <p>When the steps below require the user agent to <dfn>adjust SVG attributes</dfn> for a token,
  then, for each attribute on the token whose attribute name is one of the ones in the first column
  of the following table, change the attribute's name to the name given in the corresponding cell in
  the second column. (This fixes the case of SVG attributes that are not all lowercase.)</p>

  <table>
   <thead>
    <tr> <th> Attribute name on token <th> Attribute name on element
   <tbody>
    <tr> <td> <code data-x="">attributename</code> <td> <code data-x="">attributeName</code>
    <tr> <td> <code data-x="">attributetype</code> <td> <code data-x="">attributeType</code>
    <tr> <td> <code data-x="">basefrequency</code> <td> <code data-x="">baseFrequency</code>
    <tr> <td> <code data-x="">baseprofile</code> <td> <code data-x="">baseProfile</code>
    <tr> <td> <code data-x="">calcmode</code> <td> <code data-x="">calcMode</code>
    <tr> <td> <code data-x="">clippathunits</code> <td> <code data-x="">clipPathUnits</code>
    <tr> <td> <code data-x="">diffuseconstant</code> <td> <code data-x="">diffuseConstant</code>
    <tr> <td> <code data-x="">edgemode</code> <td> <code data-x="">edgeMode</code>
    <tr> <td> <code data-x="">filterunits</code> <td> <code data-x="">filterUnits</code>
    <tr> <td> <code data-x="">glyphref</code> <td> <code data-x="">glyphRef</code>
    <tr> <td> <code data-x="">gradienttransform</code> <td> <code data-x="">gradientTransform</code>
    <tr> <td> <code data-x="">gradientunits</code> <td> <code data-x="">gradientUnits</code>
    <tr> <td> <code data-x="">kernelmatrix</code> <td> <code data-x="">kernelMatrix</code>
    <tr> <td> <code data-x="">kernelunitlength</code> <td> <code data-x="">kernelUnitLength</code>
    <tr> <td> <code data-x="">keypoints</code> <td> <code data-x="">keyPoints</code>
    <tr> <td> <code data-x="">keysplines</code> <td> <code data-x="">keySplines</code>
    <tr> <td> <code data-x="">keytimes</code> <td> <code data-x="">keyTimes</code>
    <tr> <td> <code data-x="">lengthadjust</code> <td> <code data-x="">lengthAdjust</code>
    <tr> <td> <code data-x="">limitingconeangle</code> <td> <code data-x="">limitingConeAngle</code>
    <tr> <td> <code data-x="">markerheight</code> <td> <code data-x="">markerHeight</code>
    <tr> <td> <code data-x="">markerunits</code> <td> <code data-x="">markerUnits</code>
    <tr> <td> <code data-x="">markerwidth</code> <td> <code data-x="">markerWidth</code>
    <tr> <td> <code data-x="">maskcontentunits</code> <td> <code data-x="">maskContentUnits</code>
    <tr> <td> <code data-x="">maskunits</code> <td> <code data-x="">maskUnits</code>
    <tr> <td> <code data-x="">numoctaves</code> <td> <code data-x="">numOctaves</code>
    <tr> <td> <code data-x="">pathlength</code> <td> <code data-x="">pathLength</code>
    <tr> <td> <code data-x="">patterncontentunits</code> <td> <code data-x="">patternContentUnits</code>
    <tr> <td> <code data-x="">patterntransform</code> <td> <code data-x="">patternTransform</code>
    <tr> <td> <code data-x="">patternunits</code> <td> <code data-x="">patternUnits</code>
    <tr> <td> <code data-x="">pointsatx</code> <td> <code data-x="">pointsAtX</code>
    <tr> <td> <code data-x="">pointsaty</code> <td> <code data-x="">pointsAtY</code>
    <tr> <td> <code data-x="">pointsatz</code> <td> <code data-x="">pointsAtZ</code>
    <tr> <td> <code data-x="">preservealpha</code> <td> <code data-x="">preserveAlpha</code>
    <tr> <td> <code data-x="">preserveaspectratio</code> <td> <code data-x="">preserveAspectRatio</code>
    <tr> <td> <code data-x="">primitiveunits</code> <td> <code data-x="">primitiveUnits</code>
    <tr> <td> <code data-x="">refx</code> <td> <code data-x="">refX</code>
    <tr> <td> <code data-x="">refy</code> <td> <code data-x="">refY</code>
    <tr> <td> <code data-x="">repeatcount</code> <td> <code data-x="">repeatCount</code>
    <tr> <td> <code data-x="">repeatdur</code> <td> <code data-x="">repeatDur</code>
    <tr> <td> <code data-x="">requiredextensions</code> <td> <code data-x="">requiredExtensions</code>
    <tr> <td> <code data-x="">requiredfeatures</code> <td> <code data-x="">requiredFeatures</code>
    <tr> <td> <code data-x="">specularconstant</code> <td> <code data-x="">specularConstant</code>
    <tr> <td> <code data-x="">specularexponent</code> <td> <code data-x="">specularExponent</code>
    <tr> <td> <code data-x="">spreadmethod</code> <td> <code data-x="">spreadMethod</code>
    <tr> <td> <code data-x="">startoffset</code> <td> <code data-x="">startOffset</code>
    <tr> <td> <code data-x="">stddeviation</code> <td> <code data-x="">stdDeviation</code>
    <tr> <td> <code data-x="">stitchtiles</code> <td> <code data-x="">stitchTiles</code>
    <tr> <td> <code data-x="">surfacescale</code> <td> <code data-x="">surfaceScale</code>
    <tr> <td> <code data-x="">systemlanguage</code> <td> <code data-x="">systemLanguage</code>
    <tr> <td> <code data-x="">tablevalues</code> <td> <code data-x="">tableValues</code>
    <tr> <td> <code data-x="">targetx</code> <td> <code data-x="">targetX</code>
    <tr> <td> <code data-x="">targety</code> <td> <code data-x="">targetY</code>
    <tr> <td> <code data-x="">textlength</code> <td> <code data-x="">textLength</code>
    <tr> <td> <code data-x="">viewbox</code> <td> <code data-x="">viewBox</code>
    <tr> <td> <code data-x="">viewtarget</code> <td> <code data-x="">viewTarget</code>
    <tr> <td> <code data-x="">xchannelselector</code> <td> <code data-x="">xChannelSelector</code>
    <tr> <td> <code data-x="">ychannelselector</code> <td> <code data-x="">yChannelSelector</code>
    <tr> <td> <code data-x="">zoomandpan</code> <td> <code data-x="">zoomAndPan</code>
  </table>

  <p>When the steps below require the user agent to <dfn>adjust foreign attributes</dfn> for a
  token, then, if any of the attributes on the token match the strings given in the first column of
  the following table, let the attribute be a namespaced attribute, with the prefix being the string
  given in the corresponding cell in the second column, the local name being the string given in the
  corresponding cell in the third column, and the namespace being the namespace given in the
  corresponding cell in the fourth column. (This fixes the use of namespaced attributes, in
  particular <span data-x="attr-xml-lang"><code data-x="">lang</code> attributes in the <span>XML
  namespace</span></span>.)</p>

  <table>
   <thead>
    <tr> <th> Attribute name <th> Prefix <th> Local name <th> Namespace
   <tbody>
    <tr> <td> <code data-x="">xlink:actuate</code> <td> <code data-x="">xlink</code> <td> <code data-x="">actuate</code> <td> <span>XLink namespace</span>
    <tr> <td> <code data-x="">xlink:arcrole</code> <td> <code data-x="">xlink</code> <td> <code data-x="">arcrole</code> <td> <span>XLink namespace</span>
    <tr> <td> <code data-x="">xlink:href</code> <td> <code data-x="">xlink</code> <td> <code data-x="">href</code> <td> <span>XLink namespace</span>
    <tr> <td> <code data-x="">xlink:role</code> <td> <code data-x="">xlink</code> <td> <code data-x="">role</code> <td> <span>XLink namespace</span>
    <tr> <td> <code data-x="">xlink:show</code> <td> <code data-x="">xlink</code> <td> <code data-x="">show</code> <td> <span>XLink namespace</span>
    <tr> <td> <code data-x="">xlink:title</code> <td> <code data-x="">xlink</code> <td> <code data-x="">title</code> <td> <span>XLink namespace</span>
    <tr> <td> <code data-x="">xlink:type</code> <td> <code data-x="">xlink</code> <td> <code data-x="">type</code> <td> <span>XLink namespace</span>
    <tr> <td> <code data-x="">xml:lang</code> <td> <code data-x="">xml</code> <td> <code data-x="">lang</code> <td> <span>XML namespace</span>
    <tr> <td> <code data-x="">xml:space</code> <td> <code data-x="">xml</code> <td> <code data-x="">space</code> <td> <span>XML namespace</span>
    <tr> <td> <code data-x="">xmlns</code> <td> (none) <td> <code data-x="">xmlns</code> <td> <span>XMLNS namespace</span>
    <tr> <td> <code data-x="">xmlns:xlink</code> <td> <code data-x="">xmlns</code> <td> <code data-x="">xlink</code> <td> <span>XMLNS namespace</span>
  </table>

  <hr>

  <p>When the steps below require the user agent to <dfn>insert a character</dfn> while processing a
  token, the user agent must run the following steps:</p>

  <ol>
   <li><p>Let <var>data</var> be the characters passed to the algorithm, or, if no
   characters were explicitly specified, the character of the character token being
   processed.</p></li>

   <li><p>Let the <var>adjusted insertion location</var> be the <span>appropriate
   place for inserting a node</span>.</p></li>

   <li>
    <p>If the <var>adjusted insertion location</var> is in a <code>Document</code> node,
    then return.

    <p class="note">The DOM will not let <code>Document</code> nodes have <code>Text</code> node
    children, so they are dropped on the floor.</p>
   </li>

   <li>
    <p>If there is a <code>Text</code> node immediately before the <var>adjusted insertion
    location</var>, then append <var>data</var> to that <code>Text</code> node's <span
    data-x="concept-cd-data">data</span>.</p>

    <p>Otherwise, create a new <code>Text</code> node whose <span
    data-x="concept-cd-data">data</span> is <var>data</var> and
    whose <span>node document</span> is the same as that of the
    element in which the <var>adjusted insertion location</var> finds itself, and insert
    the newly created node at the <var>adjusted insertion location</var>.</p>
   </li>
  </ol>

  <div class="example">

   <p>Here are some sample inputs to the parser and the corresponding number of <code>Text</code>
   nodes that they result in, assuming a user agent that executes scripts.</p>

   <table>
    <thead>
     <tr>
      <th>Input <th>Number of <code>Text</code> nodes
    <tbody>
     <tr>
      <td><pre><code class="html">A&lt;script>
var&nbsp;script&nbsp;=&nbsp;document.getElementsByTagName('script')[0];
document.body.removeChild(script);
&lt;/script>B</code></pre>
      <td>One <code>Text</code> node in the document, containing "AB".
     <tr>
      <td><pre><code class="html">A&lt;script>
var&nbsp;text&nbsp;=&nbsp;document.createTextNode('B');
document.body.appendChild(text);
&lt;/script>C</code></pre>
      <td>Three <code>Text</code> nodes; "A" before the script, the script's contents, and "BC" after the script (the parser appends to the <code>Text</code> node created by the script).
     <tr>
      <td><pre><code class="html">A&lt;script>
var&nbsp;text&nbsp;=&nbsp;document.getElementsByTagName('script')[0].firstChild;
text.data&nbsp;=&nbsp;'B';
document.body.appendChild(text);
&lt;/script>C</code></pre>
      <td>Two adjacent <code>Text</code> nodes in the document, containing "A" and "BC".
     <tr>
      <td><pre><code class="html">A&lt;table>B&lt;tr>C&lt;/tr>D&lt;/table></code></pre>
      <td>One <code>Text</code> node before the table, containing "ABCD". (This is caused by <span data-x="foster parent">foster parenting</span>.)
     <tr>
      <td><pre><code class="html">A&lt;table>&lt;tr>&nbsp;B&lt;/tr>&nbsp;C&lt;/table></code></pre>
      <td>One <code>Text</code> node before the table, containing "A&nbsp;B&nbsp;C" (A-space-B-space-C). (This is caused by <span data-x="foster parent">foster parenting</span>.)
     <tr>
      <td><pre><code class="html">A&lt;table>&lt;tr>&nbsp;B&lt;/tr>&nbsp;&lt;/em>C&lt;/table></code></pre>
      <td>One <code>Text</code> node before the table, containing "A&nbsp;BC" (A-space-B-C), and one <code>Text</code> node inside the table (as a child of a <code>tbody</code>) with a single space character. (Space characters separated from non-space characters by non-character tokens are not affected by <span data-x="foster parent">foster parenting</span>, even if those other tokens then get ignored.)
   </table>

  </div>

  <hr>

  <p>When the steps below require the user agent to <dfn>insert a comment</dfn> while processing a
  comment token, optionally with an explicit insertion position <var>position</var>, the
  user agent must run the following steps:</p>

  <ol>
   <li><p>Let <var>data</var> be the data given in the comment token being
   processed.</p></li>

   <li><p>If <var>position</var> was specified, then let the <var>adjusted
   insertion location</var> be <var>position</var>. Otherwise, let <var>adjusted
   insertion location</var> be the <span>appropriate place for inserting a node</span>.</p></li>

   <li><p>Create a <code>Comment</code> node whose <code data-x="">data</code> attribute is set to
   <var>data</var> and whose <span>node document</span> is
   the same as that of the node in which the <var>adjusted insertion location</var> finds
   itself.</p>

   <li><p>Insert the newly created node at the <var>adjusted insertion
   location</var>.</p></li>
  </ol>

  <h5>Parsing elements that contain only text</h5>

  <p>The <dfn>generic raw text element parsing algorithm</dfn> and the <dfn>generic RCDATA element
  parsing algorithm</dfn> consist of the following steps. These algorithms are always invoked in
  response to a start tag token.</p>

  <ol>
   <li><p><span>Insert an HTML element</span> for the token.</p></li>

   <li><p>If the algorithm that was invoked is the <span>generic raw text element parsing
   algorithm</span>, switch the tokenizer to the <span>RAWTEXT state</span>; otherwise the algorithm
   invoked was the <span>generic RCDATA element parsing algorithm</span>, switch the tokenizer to
   the <span>RCDATA state</span>.</p></li>

   <li><p>Let the <span>original insertion mode</span> be the current <span>insertion
   mode</span>.</p>

   <li><p>Then, switch the <span>insertion mode</span> to "<span data-x="insertion mode:
   text">text</span>".</p></li>
  </ol>


  <h5>Closing elements that have implied end tags</h5>

  <p>When the steps below require the UA to <dfn>generate implied end tags</dfn>, then, while the
  <span>current node</span> is a <code>dd</code> element, a <code>dt</code> element, an
  <code>li</code> element, an <code>optgroup</code> element, an <code>option</code> element, a
  <code>p</code> element, an <code>rb</code> element, an <code>rp</code> element, an <code>rt</code>
  element, or an <code>rtc</code> element, the UA must pop the <span>current node</span> off the
  <span>stack of open elements</span>.</p>

  <p>If a step requires the UA to generate implied end tags but lists an element to exclude from the
  process, then the UA must perform the above steps as if that element was not in the above
  list.</p>

  <p>When the steps below require the UA to <dfn>generate all implied end tags thoroughly</dfn>,
  then, while the <span>current node</span> is a <code>caption</code> element, a
  <code>colgroup</code> element, a <code>dd</code> element, a <code>dt</code> element, an
  <code>li</code> element, an <code>optgroup</code> element, an <code>option</code> element, a
  <code>p</code> element, an <code>rb</code> element, an <code>rp</code> element, an <code>rt</code>
  element, an <code>rtc</code> element, a <code>tbody</code> element, a <code>td</code> element, a
  <code>tfoot</code> element, a <code>th</code> element, a <code>thead</code> element, or a
  <code>tr</code> element, the UA must pop the <span>current node</span> off the
  <span>stack of open elements</span>.</p>



  <h5 id="parsing-main-inhtml">The rules for parsing tokens in HTML content</h5>


  <h6>The "<dfn data-x="insertion mode: initial">initial</dfn>" insertion mode</h6>

  <p>A <code>Document</code> object has an associated <dfn>parser cannot change the mode flag</dfn>
  (a boolean). It is initially false.</p>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode:
  initial">initial</span>" <span>insertion mode</span>, the user agent must handle the token as
  follows:</p>

  <dl class="switch">

   <dt>A character token that is one of U+0009 CHARACTER
   TABULATION, U+000A LINE FEED (LF), U+000C FORM FEED (FF),
   U+000D CARRIAGE RETURN (CR), or U+0020 SPACE</dt>
   <dd>
    <p>Ignore the token.</p>
   </dd>

   <dt>A comment token</dt>
   <dd>
    <p><span>Insert a comment</span> as the last child of the <code>Document</code> object.</p>
   </dd>

   <dt>A DOCTYPE token</dt>
   <dd>
    <p>If the DOCTYPE token's name is not "<code data-x="">html</code>", or the token's public
    identifier is not missing, or the token's system identifier is neither missing nor
    "<code>about:legacy-compat</code>", then there is a <span>parse error</span>.</p>

    <p>Append a <code>DocumentType</code> node to the <code>Document</code> node, with its <span
    data-x="concept-doctype-name">name</span> set to the name given in the DOCTYPE token, or the
    empty string if the name was missing; its <span data-x="concept-doctype-publicid">public
    ID</span> set to the public identifier given in the DOCTYPE token, or the empty string if the
    public identifier was missing; and its <span data-x="concept-doctype-systemid">system ID</span>
    set to the system identifier given in the DOCTYPE token, or the empty string if the system
    identifier was missing.</p>

    <p class="note">This also ensures that the <code>DocumentType</code> node is returned as the
    value of the <code data-x="dom-Document-doctype">doctype</code> attribute of the
    <code>Document</code> object.</p>

    <p id="quirks-mode-doctypes">Then, if the document is <em>not</em> <span>an <code>iframe</code>
    <code data-x="attr-iframe-srcdoc">srcdoc</code> document</span>, and the <span>parser cannot
    change the mode flag</span> is false, and the DOCTYPE token matches one of the conditions in the
    following list, then set the <code>Document</code> to <span>quirks mode</span>:</p>

    <ul class="brief">
     <li> The <i data-x="force-quirks flag">force-quirks flag</i> is set to <i>on</i>. </li>
     <li> The name is not "<code data-x="">html</code>". </li>
     <li> The public identifier is set to: "<code data-x="">-//W3O//DTD W3 HTML Strict 3.0//EN//</code>" </li>
     <li> The public identifier is set to: "<code data-x="">-/W3C/DTD HTML 4.0 Transitional/EN</code>" </li>
     <li> The public identifier is set to: "<code data-x="">HTML</code>" </li>
     <li> The system identifier is set to: "<code data-x="">http://www.ibm.com/data/dtd/v11/ibmxhtml1-transitional.dtd</code>" </li>
     <li> The public identifier starts with: "<code data-x="">+//Silmaril//dtd html Pro v0r11 19970101//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//AS//DTD HTML 3.0 asWedit + extensions//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//AdvaSoft Ltd//DTD HTML 3.0 asWedit + extensions//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML 2.0 Level 1//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML 2.0 Level 2//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML 2.0 Strict Level 1//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML 2.0 Strict Level 2//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML 2.0 Strict//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML 2.0//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML 2.1E//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML 3.0//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML 3.2 Final//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML 3.2//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML 3//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML Level 0//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML Level 1//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML Level 2//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML Level 3//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML Strict Level 0//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML Strict Level 1//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML Strict Level 2//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML Strict Level 3//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML Strict//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//IETF//DTD HTML//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//Metrius//DTD Metrius Presentational//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//Microsoft//DTD Internet Explorer 2.0 HTML Strict//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//Microsoft//DTD Internet Explorer 2.0 HTML//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//Microsoft//DTD Internet Explorer 2.0 Tables//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//Microsoft//DTD Internet Explorer 3.0 HTML Strict//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//Microsoft//DTD Internet Explorer 3.0 HTML//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//Microsoft//DTD Internet Explorer 3.0 Tables//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//Netscape Comm. Corp.//DTD HTML//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//Netscape Comm. Corp.//DTD Strict HTML//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//O'Reilly and Associates//DTD HTML 2.0//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//O'Reilly and Associates//DTD HTML Extended 1.0//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//O'Reilly and Associates//DTD HTML Extended Relaxed 1.0//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//SQ//DTD HTML 2.0 HoTMetaL + extensions//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//SoftQuad Software//DTD HoTMetaL PRO 6.0::19990601::extensions to HTML 4.0//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//SoftQuad//DTD HoTMetaL PRO 4.0::19971010::extensions to HTML 4.0//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//Spyglass//DTD HTML 2.0 Extended//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//Sun Microsystems Corp.//DTD HotJava HTML//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//Sun Microsystems Corp.//DTD HotJava Strict HTML//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//W3C//DTD HTML 3 1995-03-24//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//W3C//DTD HTML 3.2 Draft//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//W3C//DTD HTML 3.2 Final//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//W3C//DTD HTML 3.2//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//W3C//DTD HTML 3.2S Draft//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//W3C//DTD HTML 4.0 Frameset//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//W3C//DTD HTML 4.0 Transitional//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//W3C//DTD HTML Experimental 19960712//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//W3C//DTD HTML Experimental 970421//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//W3C//DTD W3 HTML//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//W3O//DTD W3 HTML 3.0//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//WebTechs//DTD Mozilla HTML 2.0//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//WebTechs//DTD Mozilla HTML//</code>" </li>
     <li> The system identifier is missing and the public identifier starts with: "<code data-x="">-//W3C//DTD HTML 4.01 Frameset//</code>" </li>
     <li> The system identifier is missing and the public identifier starts with: "<code data-x="">-//W3C//DTD HTML 4.01 Transitional//</code>" </li>
    </ul>

    <p>Otherwise, if the document is <em>not</em> <span>an <code>iframe</code> <code
    data-x="attr-iframe-srcdoc">srcdoc</code> document</span>, and the <span>parser cannot change
    the mode flag</span> is false, and the DOCTYPE token matches one of the conditions in the
    following list, then set the <code>Document</code> to <span>limited-quirks mode</span>:</p>

    <ul class="brief">
     <li> The public identifier starts with: "<code data-x="">-//W3C//DTD XHTML 1.0 Frameset//</code>" </li>
     <li> The public identifier starts with: "<code data-x="">-//W3C//DTD XHTML 1.0 Transitional//</code>" </li>
     <li> The system identifier is not missing and the public identifier starts with: "<code data-x="">-//W3C//DTD HTML 4.01 Frameset//</code>" </li>
     <li> The system identifier is not missing and the public identifier starts with: "<code data-x="">-//W3C//DTD HTML 4.01 Transitional//</code>" </li>
    </ul>

    <p>The system identifier and public identifier strings must be compared to the values given in
    the lists above in an <span>ASCII case-insensitive</span> manner. A system identifier whose
    value is the empty string is not considered missing for the purposes of the conditions
    above.</p>

    <p>Then, switch the <span>insertion mode</span> to "<span data-x="insertion mode: before
    html">before html</span>".</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <p>If the document is <em>not</em> <span>an <code>iframe</code> <code
    data-x="attr-iframe-srcdoc">srcdoc</code> document</span>, then this is a <span>parse
    error</span>; if the <span>parser cannot change the mode flag</span> is false, set the
    <code>Document</code> to <span>quirks mode</span>.</p>

    <p>In any case, switch the <span>insertion mode</span> to "<span data-x="insertion mode: before
    html">before html</span>", then reprocess the token.</p>
   </dd>
  </dl>


  <h6>The "<dfn data-x="insertion mode: before html">before html</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: before
  html">before html</span>" <span>insertion mode</span>, the user agent must handle the token as
  follows:</p>

  <dl class="switch">

   <dt>A DOCTYPE token</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>A comment token</dt>
   <dd>
    <p><span>Insert a comment</span> as the last child of the <code>Document</code> object.</p>
   </dd>

   <dt>A character token that is one of U+0009 CHARACTER TABULATION, U+000A LINE FEED (LF), U+000C
   FORM FEED (FF), U+000D CARRIAGE RETURN (CR), or U+0020 SPACE</dt>
   <dd>
    <p>Ignore the token.</p>
   </dd>

   <dt>A start tag whose tag name is "html"</dt>
   <dd>
    <p><span>Create an element for the token</span> in the <span>HTML namespace</span>, with the
    <code>Document</code> as the intended parent. Append it to the <code>Document</code> object. Put
    this element in the <span>stack of open elements</span>.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: before head">before
    head</span>".</p>
   </dd>

   <dt>An end tag whose tag name is one of: "head", "body", "html", "br"</dt>
   <dd>
    <p>Act as described in the "anything else" entry below.</p>
   </dd>

   <dt>Any other end tag</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <p>Create an <code>html</code> element whose <span>node document</span> is the <code>Document</code> object. Append
    it to the <code>Document</code> object. Put this element in the <span>stack of open
    elements</span>.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: before head">before
    head</span>", then reprocess the token.</p>
   </dd>
  </dl>

  <p>The <span>document element</span> can end up being removed from the <code>Document</code>
  object, e.g. by scripts; nothing in particular happens in such cases, content continues being
  appended to the nodes as described in the next section.</p>


  <h6>The "<dfn data-x="insertion mode: before head">before head</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: before
  head">before head</span>" <span>insertion mode</span>, the user agent must handle the token as
  follows:</p>

  <dl class="switch">

   <dt>A character token that is one of U+0009 CHARACTER
   TABULATION, U+000A LINE FEED (LF), U+000C FORM FEED (FF),
   U+000D CARRIAGE RETURN (CR), or U+0020 SPACE</dt>
   <dd>
    <p>Ignore the token.</p> <!-- :-( -->
   </dd>

   <dt>A comment token</dt>
   <dd>
    <p><span>Insert a comment</span>.</p>
   </dd>

   <dt>A DOCTYPE token</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>A start tag whose tag name is "html"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>A start tag whose tag name is "head"</dt>
   <dd>
    <p><span>Insert an HTML element</span> for the token.</p>

    <p>Set the <span><code data-x="">head</code> element pointer</span> to the newly created
    <code>head</code> element.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in head">in
    head</span>".</p>
   </dd>

   <dt>An end tag whose tag name is one of: "head", "body", "html", "br"</dt>
   <dd>
    <p>Act as described in the "anything else" entry below.</p>
   </dd>

   <dt>Any other end tag</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <!-- fake <head> -->

    <p><span>Insert an HTML element</span> for a "head" start tag token with no attributes.</p>

    <p>Set the <span><code data-x="">head</code> element pointer</span> to the newly created
    <code>head</code> element.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in head">in
    head</span>".</p>

    <!-- end of fake <head> -->

    <p>Reprocess the current token.</p>
   </dd>
  </dl>


  <h6 id="parsing-main-inhead">The "<dfn data-x="insertion mode: in head">in head</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: in head">in
  head</span>" <span>insertion mode</span>, the user agent must handle the token as follows:</p>

  <dl class="switch">

   <dt>A character token that is one of U+0009 CHARACTER
   TABULATION, U+000A LINE FEED (LF), U+000C FORM FEED (FF),
   U+000D CARRIAGE RETURN (CR), or U+0020 SPACE</dt>
   <dd>
    <p><span data-x="insert a character">Insert the character</span>.</p>
   </dd>

   <dt>A comment token</dt>
   <dd>
    <p><span>Insert a comment</span>.</p>
   </dd>

   <dt>A DOCTYPE token</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>A start tag whose tag name is "html"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>A start tag whose tag name is one of: "base", "basefont",
   "bgsound", "link"</dt>
   <dd>
    <p><span>Insert an HTML element</span> for the token. Immediately pop the <span>current
    node</span> off the <span>stack of open elements</span>.</p>

    <p><span data-x="acknowledge self-closing flag">Acknowledge the token's <i data-x="self-closing flag">self-closing
    flag</i></span>, if it is set.</p>
   </dd>

   <dt>A start tag whose tag name is "meta"</dt>
   <dd>
    <p><span>Insert an HTML element</span> for the token. Immediately pop the <span>current
    node</span> off the <span>stack of open elements</span>.</p>

    <p><span data-x="acknowledge self-closing flag">Acknowledge the token's <i data-x="self-closing flag">self-closing
    flag</i></span>, if it is set.</p>

    <p>If the <span>active speculative HTML parser</span> is null, then:</p>

    <ol>
     <li><p id="meta-charset-during-parse">If the element has a <code
     data-x="attr-meta-charset">charset</code> attribute, and <span>getting an encoding</span> from
     its value results in an <span>encoding</span>, and the
     <span data-x="concept-encoding-confidence">confidence</span> is currently <i>tentative</i>,
     then <span>change the encoding</span> to the resulting encoding.</p></li>

     <li><p>Otherwise, if the element has an <code data-x="attr-meta-http-equiv">http-equiv</code>
     attribute whose value is an <span>ASCII case-insensitive</span> match for the string "<code
     data-x="">Content-Type</code>", and the element has a <code
     data-x="attr-meta-content">content</code> attribute, and applying the <span>algorithm for
     extracting a character encoding from a <code>meta</code> element</span> to that attribute's
     value returns an <span>encoding</span>, and the
     <span data-x="concept-encoding-confidence">confidence</span> is currently <i>tentative</i>,
     then <span>change the encoding</span> to the extracted encoding.</p></li>
    </ol>

    <p class="note">The <span>speculative HTML parser</span> doesn't speculatively apply character
    encoding declarations in order to reduce implementation complexity.</p>
   </dd>

   <dt>A start tag whose tag name is "title"</dt>
   <dd>
    <p>Follow the <span>generic RCDATA element parsing algorithm</span>.</p>
   </dd>

   <dt>A start tag whose tag name is "noscript", if the <span>scripting flag</span> is enabled</dt>
   <dt>A start tag whose tag name is one of: "noframes", "style"</dt>
   <dd>
    <p>Follow the <span>generic raw text element parsing algorithm</span>.</p>
   </dd>

   <dt>A start tag whose tag name is "noscript", if the <span>scripting flag</span> is disabled</dt>
   <dd>
    <p><span>Insert an HTML element</span> for the token.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in head noscript">in
    head noscript</span>".</p>
   </dd>


   <dt id="scriptTag">A start tag whose tag name is "script"</dt>
   <dd>
    <p>Run these steps:</p>

    <ol>
     <li><p>Let the <var>adjusted insertion location</var> be the <span>appropriate place
     for inserting a node</span>.</p></li>

     <li><p><span>Create an element for the token</span> in the <span>HTML namespace</span>, with
     the intended parent being the element in which the <var>adjusted insertion
     location</var> finds itself.</p></li>

     <li>
      <p>Set the element's <span>parser document</span> to the <code>Document</code>, and set the
      element's <span data-x="script-force-async">force async</span> to false.</p>

      <p class="note">This ensures that, if the script is external, any <code
      data-x="dom-document-write">document.write()</code> calls in the script will execute in-line,
      instead of blowing the document away, as would happen in most other cases. It also prevents
      the script from executing until the end tag is seen.</p>
     </li>

     <li><p>If the parser was created as part of the <span>HTML fragment parsing algorithm</span>,
     then set the <code>script</code> element's <span>already started</span> to true.
     (<span>fragment case</span>)</p></li>

     <li id="document-written-scripts-intervention"><p>If the parser was invoked via the <code
     data-x="dom-document-write">document.write()</code> or <code
     data-x="dom-document-writeln">document.writeln()</code> methods, then optionally set the
     <code>script</code> element's <span>already started</span> to true. (For example, the user
     agent might use this clause to prevent execution of <span data-x="origin">cross-origin</span>
     scripts inserted via <code data-x="dom-document-write">document.write()</code> under slow
     network conditions, or when the page has already taken a long time to load.)</p></li>

     <li><p>Insert the newly created element at the <var>adjusted insertion
     location</var>.</p></li>

     <li><p>Push the element onto the <span>stack of open elements</span> so that it is the new
     <span>current node</span>.</p></li>

     <li><p>Switch the tokenizer to the <span>script data state</span>.</p></li>

     <li><p>Let the <span>original insertion mode</span> be the current <span>insertion
     mode</span>.</p>

     <li><p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode:
     text">text</span>".</p></li>
    </ol>
   </dd>

   <dt>An end tag whose tag name is "head"</dt>
   <dd>
    <p>Pop the <span>current node</span> (which will be the <code>head</code> element) off the
    <span>stack of open elements</span>.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: after head">after
    head</span>".</p>
   </dd>

   <dt>An end tag whose tag name is one of: "body", "html", "br"</dt>
   <dd>
    <p>Act as described in the "anything else" entry below.</p>
   </dd>

   <dt>A start tag whose tag name is "template"</dt>
   <dd>
    <p>Let <var>template start tag</var> be the start tag.</p>

    <p>Insert a <span data-x="concept-parser-marker">marker</span> at the end of the <span>list of
    active formatting elements</span>.</p>

    <p>Set the <span>frameset-ok flag</span> to "not ok".</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in template">in
    template</span>".</p>

    <p>Push "<span data-x="insertion mode: in template">in template</span>" onto the <span>stack of
    template insertion modes</span> so that it is the new <span>current template insertion
    mode</span>.</p>

    <p>Let the <var>adjusted insertion location</var> be the <span>appropriate place for inserting
    a node</span>.</p>

    <p>Let <var>intended parent</var> be the element in which the <var>adjusted insertion
    location</var> finds itself.</p>

    <p>Let <var>document</var> be <var>intended parent</var>'s <span>node document</span>.</p>

    <p>If any of the following are false:</p>

    <ul>
     <li><var>template start tag</var>'s
     <code data-x="attr-template-shadowrootmode">shadowrootmode</code> is not in
     the <span data-x="attr-shadowrootmode-none-state">none</span> state;</li>

     <li><var>document</var>'s <span data-x="concept-document-allow-declarative-shadow-roots">allow
     declarative shadow roots</span> is true; or</li>

     <li>the <span>adjusted current node</span> is not the topmost element in the <span>stack of
     open elements</span>,</li>
    </ul>

    <p>then <span>insert an HTML element</span> for the token.</p>

    <p>Otherwise:</p>

    <ol>
     <li><p>Let <var>declarative shadow host element</var> be <span>adjusted current
     node</span>.</p></li>

     <li><p>Let <var>template</var> be the result of <span>insert a foreign element</span>
     for <var>template start tag</var>, with <span>HTML namespace</span> and true.</p></li>

     <li><p>Let <var>mode</var> be <var>template start tag</var>'s <code
     data-x="attr-template-shadowrootmode">shadowrootmode</code> attribute's value.</p></li>

     <li><p>Let <var>clonable</var> be true if <var>template start tag</var> has a <code
     data-x="attr-template-shadowrootclonable">shadowrootclonable</code> attribute; otherwise
     false.</p></li>

     <li><p>Let <var>serializable</var> be true if <var>template start tag</var> has a <code
     data-x="attr-template-shadowrootserializable">shadowrootserializable</code> attribute;
     otherwise false.</p></li>

     <li><p>Let <var>delegatesFocus</var> be true if <var>template start tag</var> has a <code
     data-x="attr-template-shadowrootdelegatesfocus">shadowrootdelegatesfocus</code> attribute;
     otherwise false.</p></li>

     <li><p>If <var>declarative shadow host element</var> is a <span>shadow host</span>, then
     <span>insert an element at the adjusted insertion location</span> with
     <var>template</var>.</p></li>

     <li>
      <p>Otherwise:

      <ol>
       <li>
        <p><span data-x="concept-attach-a-shadow-root">Attach a shadow root</span> with
        <var>declarative shadow host element</var>, <var>mode</var>, <var>clonable</var>,
        <var>serializable</var>, <var>delegatesFocus</var>, and "<code data-x="">named</code>".</p>

        <p>If an exception is thrown, then catch it and:</p>

        <ol>
         <li><p><span>Insert an element at the adjusted insertion location</span> with
         <var>template</var>.</p></li>

         <li><p>The user agent may report an error to the developer console.</p></li>

         <li><p>Return.</p></li>
        </ol>
       </li>

       <li><p>Let <var>shadow</var> be <var>declarative shadow host element</var>'s
       <span data-x="concept-element-shadow-root">shadow root</span>.</p></li>

       <li><p>Set <var>shadow</var>'s
       <span data-x="concept-shadow-root-declarative">declarative</span> to true.</p></li>

       <li><p>Set <span>template</span>'s <span>template contents</span> property to
       <var>shadow</var>.</p></li>

       <li><p>Set <var>shadow</var>'s <span>available to element internals</span> to true.</p></li>
      </ol>
     </li>
    </ol>
   </dd>

   <dt>An end tag whose tag name is "template"</dt>
   <dd>
    <p>If there is no <code>template</code> element on the <span>stack of open elements</span>, then
    this is a <span>parse error</span>; ignore the token.</p>

    <p>Otherwise, run these steps:</p>

    <ol>
     <li><p><span>Generate all implied end tags thoroughly</span>.</p></li>

     <li><p>If the <span>current node</span> is not a <code>template</code> element, then this is a
     <span>parse error</span>.</p></li>

     <li><p>Pop elements from the <span>stack of open elements</span> until a <code>template</code>
     element has been popped from the stack.</p></li>

     <li><span>Clear the list of active formatting elements up to the last marker</span>.</li>

     <li><p>Pop the <span>current template insertion mode</span> off the <span>stack of template
     insertion modes</span>.</p>

     <li><p><span>Reset the insertion mode appropriately</span>.</p></li>
    </ol>
   </dd>

   <dt>A start tag whose tag name is "head"</dt>
   <dt>Any other end tag</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <!-- can't get here with an EOF and a fragment case -->

    <!-- start of fake </head> -->
    <p>Pop the <span>current node</span> (which will be the <code>head</code> element) off the
    <span>stack of open elements</span>.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: after head">after
    head</span>".</p>
    <!-- end of fake </head> -->

    <p>Reprocess the token.</p>
   </dd>
  </dl>


  <h6 id="parsing-main-inheadnoscript">The "<dfn data-x="insertion mode: in head noscript">in head noscript</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: in head
  noscript">in head noscript</span>" <span>insertion mode</span>, the user agent must handle the
  token as follows:</p>

  <dl class="switch">

   <dt>A DOCTYPE token</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>A start tag whose tag name is "html"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>An end tag whose tag name is "noscript"</dt>
   <dd>
    <p>Pop the <span>current node</span> (which will be a <code>noscript</code> element) from the
    <span>stack of open elements</span>; the new <span>current node</span> will be a
    <code>head</code> element.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in head">in
    head</span>".</p>
   </dd>

   <dt>A character token that is one of U+0009 CHARACTER TABULATION, U+000A LINE FEED (LF), U+000C
   FORM FEED (FF), U+000D CARRIAGE RETURN (CR), or U+0020 SPACE</dt>
   <dt>A comment token</dt>
   <dt>A start tag whose tag name is one of: "basefont", "bgsound", "link", "meta", "noframes",
   "style"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    head">in head</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>An end tag whose tag name is "br"</dt>
   <dd>
    <p>Act as described in the "anything else" entry below.</p>
   </dd>

   <dt>A start tag whose tag name is one of: "head", "noscript"</dt>
   <dt>Any other end tag</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <!-- can't get here with an EOF and a fragment case -->

    <p><span>Parse error</span>.</p>

    <!-- fake </noscript> -->
    <p>Pop the <span>current node</span> (which will be a <code>noscript</code> element) from the
    <span>stack of open elements</span>; the new <span>current node</span> will be a
    <code>head</code> element.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in head">in
    head</span>".</p>
    <!-- end fake </noscript> -->

    <p>Reprocess the token.</p>
   </dd>
  </dl>


  <h6>The "<dfn data-x="insertion mode: after head">after head</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: after
  head">after head</span>" <span>insertion mode</span>, the user agent must handle the token as
  follows:</p>

  <dl class="switch">

   <dt>A character token that is one of U+0009 CHARACTER
   TABULATION, U+000A LINE FEED (LF), U+000C FORM FEED (FF),
   U+000D CARRIAGE RETURN (CR), or U+0020 SPACE</dt>
   <dd>
    <p><span data-x="insert a character">Insert the character</span>.</p>
   </dd>

   <dt>A comment token</dt>
   <dd>
    <p><span>Insert a comment</span>.</p>
   </dd>

   <dt>A DOCTYPE token</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>A start tag whose tag name is "html"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>A start tag whose tag name is "body"</dt>
   <dd>
    <p><span>Insert an HTML element</span> for the token.</p>

    <p>Set the <span>frameset-ok flag</span> to "not ok".</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in body">in
    body</span>".</p>
   </dd>

   <dt>A start tag whose tag name is "frameset"</dt>
   <dd>
    <p><span>Insert an HTML element</span> for the token.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in frameset">in
    frameset</span>".</p>
   </dd>

   <dt>A start tag whose tag name is one of: "base", "basefont", "bgsound", "link", "meta",
   "noframes", "script", "style", "template", "title"</dt>
   <dd>
    <p><span>Parse error</span>.</p>

    <p>Push the node pointed to by the <span><code data-x="">head</code> element pointer</span> onto
    the <span>stack of open elements</span>.</p>

    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    head">in head</span>" <span>insertion mode</span>.</p>

    <p>Remove the node pointed to by the <span><code data-x="">head</code> element pointer</span>
    from the <span>stack of open elements</span>. (It might not be the <span>current node</span> at
    this point.)</p>

    <p class="note">The <span><code data-x="">head</code> element pointer</span> cannot be null at
    this point.</p>
   </dd>

   <dt>An end tag whose tag name is "template"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    head">in head</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>An end tag whose tag name is one of: "body", "html", "br"</dt>
   <dd>
    <p>Act as described in the "anything else" entry below.</p>
   </dd>

   <dt>A start tag whose tag name is "head"</dt>
   <dt>Any other end tag</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <!-- fake <body>, but without resetting frameset-ok -->
    <p><span>Insert an HTML element</span> for a "body" start tag token with no attributes.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in body">in
    body</span>".</p>
    <!-- end fake <body> -->

    <p>Reprocess the current token.</p>
   </dd>
  </dl>


  <h6 id="parsing-main-inbody">The "<dfn data-x="insertion mode: in body">in body</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: in body">in
  body</span>" <span>insertion mode</span>, the user agent must handle the token as follows:</p>

  <dl class="switch">

   <dt>A character token that is U+0000 NULL</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>

    <!-- The D-Link DSL-G604T ADSL router has a zero byte in its
         configuration UI before a <frameset>, which is why U+0000 is
         special-cased here.
         refs: https://bugzilla.mozilla.org/show_bug.cgi?id=563526
               https://www.w3.org/Bugs/Public/show_bug.cgi?id=9659
    -->
   </dd>

   <dt>A character token that is one of U+0009 CHARACTER TABULATION,
   U+000A LINE FEED (LF), U+000C FORM FEED (FF), U+000D CARRIAGE
   RETURN (CR), or U+0020 SPACE</dt>
   <dd>
    <p><span>Reconstruct the active formatting elements</span>, if any.</p>

    <p><span data-x="insert a character">Insert the token's character</span>.</p>
   </dd>

   <dt>Any other character token</dt>
   <dd>
    <p><span>Reconstruct the active formatting elements</span>, if any.</p>

    <p><span data-x="insert a character">Insert the token's character</span>.</p>

    <p>Set the <span>frameset-ok flag</span> to "not ok".</p>
   </dd>

   <dt>A comment token</dt>
   <dd>
    <p><span>Insert a comment</span>.</p>
   </dd>

   <dt>A DOCTYPE token</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>A start tag whose tag name is "html"</dt>
   <dd>
    <p><span>Parse error</span>.</p>

    <p>If there is a <code>template</code> element on the <span>stack of open elements</span>, then
    ignore the token.</p>

    <p>Otherwise, for each attribute on the token, check to see if the attribute is already present
    on the top element of the <span>stack of open elements</span>. If it is not, add the attribute
    and its corresponding value to that element.</p>
   </dd>

   <dt>A start tag whose tag name is one of: "base", "basefont", "bgsound", "link", "meta",
   "noframes", "script", "style", "template", "title"</dt>
   <dt>An end tag whose tag name is "template"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    head">in head</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>A start tag whose tag name is "body"</dt>
   <dd>
    <p><span>Parse error</span>.</p>

    <p>If the <span>stack of open elements</span> has only one node on it, if the second element
    on the <span>stack of open elements</span> is not a <code>body</code> element, or if there is a
    <code>template</code> element on the <span>stack of open elements</span>, then ignore the token.
    (<span>fragment case</span> or there is a <code>template</code> element on the stack)</p>

    <p>Otherwise, set the <span>frameset-ok flag</span> to "not ok"; then, for each attribute on the
    token, check to see if the attribute is already present on the <code>body</code> element (the
    second element) on the <span>stack of open elements</span>, and if it is not, add the attribute
    and its corresponding value to that element.</p>
   </dd>

   <dt>A start tag whose tag name is "frameset"</dt>
   <dd>
    <p><span>Parse error</span>.</p>

    <p>If the <span>stack of open elements</span> has only one node on it, or if the second element
    on the <span>stack of open elements</span> is not a <code>body</code> element, then ignore the
    token. (<span>fragment case</span> or there is a <code>template</code> element on the
    stack)</p>

    <p>If the <span>frameset-ok flag</span> is set to "not ok", ignore the token.</p>

    <p>Otherwise, run the following steps:</p>

    <ol>
     <li><p>Remove the second element on the <span>stack of open elements</span> from its parent
     node, if it has one.</p></li>

     <li><p>Pop all the nodes from the bottom of the <span>stack of open elements</span>, from the
     <span>current node</span> up to, but not including, the root <code>html</code> element.</p>

     <li><p><span>Insert an HTML element</span> for the token.</p></li>

     <li><p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in frameset">in
     frameset</span>".</p>
    </ol>
   </dd>

   <dt>An end-of-file token</dt>
   <dd>
    <p>If the <span>stack of template insertion modes</span> is not empty, then process the token
    <span>using the rules for</span> the "<span data-x="insertion mode: in template">in
    template</span>" <span>insertion mode</span>.</p>
    <!-- this is needed to handle <head><template>...[EOF] - otherwise we don't construct the <body> element -->

    <p>Otherwise, follow these steps:</p>

    <ol>
     <li><p>If there is a node in the <span>stack of open elements</span> that is not either a
     <code>dd</code> element, a <code>dt</code> element, an <code>li</code> element, an
     <code>optgroup</code> element, an <code>option</code> element, a <code>p</code> element, an
     <code>rb</code> element, an <code>rp</code> element, an <code>rt</code> element, an
     <code>rtc</code> element, a <code>tbody</code> element, a <code>td</code> element, a
     <code>tfoot</code> element, a <code>th</code> element, a <code>thead</code> element, a
     <code>tr</code> element, the <code>body</code> element, or the <code>html</code> element, then
     this is a <span>parse error</span>.</p></li> <!-- (some of those are fragment cases) -->

     <li><p><span>Stop parsing</span>.</p></li>
    </ol>
   </dd>

   <dt>An end tag whose tag name is "body"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in
    scope">have a <code>body</code> element in scope</span>, this is a <span>parse error</span>;
    ignore the token.</p>

    <!-- if we get here, the insertion mode here is forcibly "in body". -->

    <p>Otherwise, if there is a node in the <span>stack of open elements</span> that is not either a
    <code>dd</code> element, a <code>dt</code> element, an <code>li</code> element, an
    <code>optgroup</code> element, an <code>option</code> element, a <code>p</code> element, an
    <code>rb</code> element, an <code>rp</code> element, an <code>rt</code> element, an
    <code>rtc</code> element, a <code>tbody</code> element, a <code>td</code> element, a
    <code>tfoot</code> element, a <code>th</code> element, a <code>thead</code> element, a
    <code>tr</code> element, the <code>body</code> element, or the <code>html</code> element, then
    this is a <span>parse error</span>.</p> <!-- (some of those are fragment cases, e.g., for
    <tbody> you'd have hit the first paragraph since the <body> wouldn't be in scope, unless it was
    a fragment case) -->

    <!-- If we ever change the frameset-ok flag to an insertion mode, then we'd have to somehow keep
    track of its state when we switch to after-body. -->

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: after body">after
    body</span>".</p>
   </dd>

   <dt>An end tag whose tag name is "html"</dt>
   <dd>
    <!-- fake </body> -->
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in
    scope">have a <code>body</code> element in scope</span>, this is a <span>parse error</span>;
    ignore the token.</p>

    <!-- if we get here, the insertion mode here is forcibly "in body". -->

    <p>Otherwise, if there is a node in the <span>stack of open elements</span> that is not either a
    <code>dd</code> element, a <code>dt</code> element, an <code>li</code> element, an
    <code>optgroup</code> element, an <code>option</code> element, a <code>p</code> element, an
    <code>rb</code> element, an <code>rp</code> element, an <code>rt</code> element, an
    <code>rtc</code> element, a <code>tbody</code> element, a <code>td</code> element, a
    <code>tfoot</code> element, a <code>th</code> element, a <code>thead</code> element, a
    <code>tr</code> element, the <code>body</code> element, or the <code>html</code> element, then
    this is a <span>parse error</span>.</p> <!-- (some of those are fragment cases, e.g., for
    <tbody> you'd have hit the first paragraph since the <body> wouldn't be in scope, unless it was
    a fragment case) -->

    <!-- If we ever change the frameset-ok flag to an insertion mode, then we'd have to somehow keep
    track of its state when we switch to after-body. -->

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: after body">after
    body</span>".</p>
    <!-- end fake </body> -->

    <p>Reprocess the token.</p>
   </dd>

   <!-- start tags for non-phrasing flow content elements -->

   <!-- the normal ones -->
   <dt>A start tag whose tag name is one of: "address", "article", "aside", "blockquote", "center",
   "details", "dialog", "dir", "div", "dl", "fieldset", "figcaption", "figure", "footer", "header",
   "hgroup", "main", "menu", "nav", "ol", "p", "search", "section", "summary", "ul"</dt>
   <dd>
    <!-- As of May 2008 this doesn't match any browser exactly, but is as close to what IE does as I
    can get without doing the non-tree DOM nonsense, and thus should actually afford better
    compatibility when implemented by the other browsers. -->

    <p>If the <span>stack of open elements</span> <span data-x="has an element in button scope">has a
    <code>p</code> element in button scope</span>, then <span>close a <code>p</code>
    element</span>.</p>

    <p><span>Insert an HTML element</span> for the token.</p>
   </dd>

   <!-- as normal, but close h1-h6 if it's the current node -->
   <dt>A start tag whose tag name is one of: "h1", "h2", "h3", "h4",
   "h5", "h6"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> <span data-x="has an element in button scope">has a
    <code>p</code> element in button scope</span>, then <span>close a <code>p</code>
    element</span>.</p>

    <p>If the <span>current node</span> is an <span data-x="HTML elements">HTML element</span> whose
    tag name is one of "h1", "h2", "h3", "h4", "h5", or "h6", then this is a <span>parse
    error</span>; pop the <span>current node</span> off the <span>stack of open elements</span>.</p>
    <!-- See https://bugs.webkit.org/show_bug.cgi?id=12646 -->

    <p><span>Insert an HTML element</span> for the token.</p>
   </dd>

   <!-- as normal, but drops leading newline -->
   <dt>A start tag whose tag name is one of: "pre", "listing"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> <span data-x="has an element in button scope">has
    a <code>p</code> element in button scope</span>, then <span>close a <code>p</code>
    element</span>.</p>

    <p><span>Insert an HTML element</span> for the token.</p>

    <p>If the <span>next token</span> is a U+000A LINE FEED (LF) character token, then ignore that
    token and move on to the next one. (Newlines at the start of <code>pre</code> blocks are ignored
    as an authoring convenience.)</p>

    <!-- <pre>[CR]X will eat the [CR], <pre>&#x10;X will eat the
    &#x10;, but <pre>&#x13;X will not eat the &#x13;. -->

    <p>Set the <span>frameset-ok flag</span> to "not ok".</p>
   </dd>

   <!-- as normal, but interacts with the form element pointer -->
   <dt>A start tag whose tag name is "form"</dt>
   <dd>
    <p>If the <span><code data-x="form">form</code> element pointer</span> is not null, and there is
    no <code>template</code> element on the <span>stack of open elements</span>, then this is a
    <span>parse error</span>; ignore the token.</p>

    <p>Otherwise:</p>

    <p>If the <span>stack of open elements</span> <span data-x="has an element in button scope">has
    a <code>p</code> element in button scope</span>, then <span>close a <code>p</code>
    element</span>.</p>

    <p><span>Insert an HTML element</span> for the token, and, if there is no <code>template</code>
    element on the <span>stack of open elements</span>, set the <span><code
    data-x="form">form</code> element pointer</span> to point to the element created.</p>
   </dd>

   <!-- as normal, but imply </li> when there's another <li> open in weird cases -->
   <dt>A start tag whose tag name is "li"</dt>
   <dd>
    <p>Run these steps:</p>

    <ol>
     <li><p>Set the <span>frameset-ok flag</span> to "not ok".</p></li>

     <li><p>Initialize <var>node</var> to be the <span>current
     node</span> (the bottommost node of the stack).</p></li>

     <li>
      <p><i>Loop</i>: If <var>node</var> is an <code>li</code> element, then run these
      substeps:</p>

      <ol>
       <!-- fake </li> -->
       <li><p><span>Generate implied end tags</span>, except for <code>li</code> elements.</p></li>

       <li><p>If the <span>current node</span> is not an <code>li</code> element, then this is a
       <span>parse error</span>.</p></li>

       <li><p>Pop elements from the <span>stack of open elements</span> until an <code>li</code>
       element has been popped from the stack.</p></li>
       <!-- end of fake </li> -->

       <li><p>Jump to the step labeled <i>done</i> below.</p></li>
      </ol>
     </li>

     <li><p>If <var>node</var> is in the <span>special</span> category, but is not an
     <code>address</code>, <code>div</code>, or <code>p</code> element, then jump to the step
     labeled <i>done</i> below.</p></li>
     <!-- an element <foo> is in this list if the following markup:

         <!DOCTYPE html><body><ol><li><foo><li>
     ...results in the second <li> not being (in any way) a descendant of the first <li>, or if
     <foo> is a formatting element that gets reopened later. -->

     <li><p>Otherwise, set <var>node</var> to the previous entry in the <span>stack of open
     elements</span> and return to the step labeled <i>loop</i>.</p></li>

     <li><p><i>Done</i>: If the <span>stack of open elements</span> <span data-x="has an element in
     button scope">has a <code>p</code> element in button scope</span>, then <span>close a
     <code>p</code> element</span>.</p></li>

     <li><p>Finally, <span>insert an HTML element</span> for the token.</p></li>
    </ol>
   </dd>

   <!-- as normal, but imply </dt> or </dd> when there's another <dt> or <dd> open in weird cases  -->
   <dt>A start tag whose tag name is one of: "dd", "dt"</dt>
   <dd>
    <p>Run these steps:</p>

    <ol>
     <li><p>Set the <span>frameset-ok flag</span> to "not ok".</p></li>

     <li><p>Initialize <var>node</var> to be the <span>current
     node</span> (the bottommost node of the stack).</p></li>

     <li>
      <p><i>Loop</i>: If <var>node</var> is a <code>dd</code> element, then run these
      substeps:</p>

      <ol>
       <!-- fake </dd> -->
       <li><p><span>Generate implied end tags</span>, except for <code>dd</code> elements.</p></li>

       <li><p>If the <span>current node</span> is not a <code>dd</code> element, then this is a
       <span>parse error</span>.</p></li>

       <li><p>Pop elements from the <span>stack of open elements</span> until a <code>dd</code>
       element has been popped from the stack.</p></li>
       <!-- end of fake </dd> -->

       <li><p>Jump to the step labeled <i>done</i> below.</p></li>
      </ol>
     </li>

     <li>
      <p>If <var>node</var> is a <code>dt</code> element, then run these substeps:</p>

      <ol>
       <!-- fake </dt> -->
       <li><p><span>Generate implied end tags</span>, except for <code>dt</code> elements.</p></li>

       <li><p>If the <span>current node</span> is not a <code>dt</code> element, then this is a
       <span>parse error</span>.</p></li>

       <li><p>Pop elements from the <span>stack of open elements</span> until a <code>dt</code>
       element has been popped from the stack.</p></li>
       <!-- end of fake </dt> -->

       <li><p>Jump to the step labeled <i>done</i> below.</p></li>
      </ol>
     </li>

     <li><p>If <var>node</var> is in the <span>special</span> category, but is not an
     <code>address</code>, <code>div</code>, or <code>p</code> element, then jump to the step
     labeled <i>done</i> below.</p></li>
     <!-- an element <foo> is in this list if the following markup:

         <!DOCTYPE html><body><dl><dt><foo><dt>

     ...results in the second <dt> not being (in any way) a descendant of the first <dt>, or if
     <foo> is a formatting element that gets reopened later. -->

     <li><p>Otherwise, set <var>node</var> to the previous entry in the <span>stack of open
     elements</span> and return to the step labeled <i>loop</i>.</p></li>

     <li><p><i>Done</i>: If the <span>stack of open elements</span> <span data-x="has an element in
     button scope">has a <code>p</code> element in button scope</span>, then <span>close a
     <code>p</code> element</span>.</p></li>

     <li><p>Finally, <span>insert an HTML element</span> for the token.</p></li>
    </ol>
   </dd>

   <!-- same as normal, but effectively ends parsing -->
   <dt>A start tag whose tag name is "plaintext"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> <span data-x="has an element in button scope">has a
    <code>p</code> element in button scope</span>, then <span>close a <code>p</code>
    element</span>.</p>

    <p><span>Insert an HTML element</span> for the token.</p>

    <p>Switch the tokenizer to the <span>PLAINTEXT state</span>.</p>

    <p class="note">Once a start tag with the tag name "plaintext" has been seen, all remaining
    tokens will be character tokens (and a final end-of-file token) because there is no way to
    switch the tokenizer out of the <span>PLAINTEXT state</span>. However, as the tree builder
    remains in its existing insertion mode, it might <span>reconstruct the active formatting
    elements</span> while processing those character tokens. This means that the parser can
    insert other elements into the <code>plaintext</code> element.</p>
   </dd>

   <!-- button is a hybrid -->
   <dt>A start tag whose tag name is "button"</dt>
   <dd>
    <ol>
     <li>
      <p>If the <span>stack of open elements</span> <span data-x="has an element in scope">has a
      <code>button</code> element in scope</span>, then run these substeps:</p>

      <ol>
       <li><p><span>Parse error</span>.</p></li>

       <li><p><span>Generate implied end tags</span>.</p></li>

       <li><p>Pop elements from the <span>stack of open elements</span> until a <code>button</code>
       element has been popped from the stack.</p></li>
      </ol>
     </li>

     <li><p><span>Reconstruct the active formatting elements</span>, if any.</p></li>

     <li><p><span>Insert an HTML element</span> for the token.</p></li>

     <li><p>Set the <span>frameset-ok flag</span> to "not ok".</p></li>
    </ol>
   </dd>

   <!-- end tags for non-phrasing flow content elements (and button) -->

   <!-- the normal ones -->
   <dt>An end tag whose tag name is one of: "address", "article", "aside", "blockquote", "button",
   "center", "details", "dialog", "dir", "div", "dl", "fieldset", "figcaption", "figure", "footer",
   "header", "hgroup", "listing", "main", "menu", "nav", "ol", "pre", "search", "section",
   "summary", "ul"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in
    scope">have an element in scope</span> that is an <span data-x="HTML elements">HTML
    element</span> with the same tag name as that of the token, then this is a <span>parse
    error</span>; ignore the token.</p>

    <p>Otherwise, run these steps:</p>

    <ol>
     <li><p><span>Generate implied end tags</span>.</p></li>

     <li><p>If the <span>current node</span> is not an <span data-x="HTML elements">HTML element</span> with
     the same tag name as that of the token, then this is a
     <span>parse error</span>.</p></li>

     <li><p>Pop elements from the <span>stack of open elements</span> until an <span data-x="HTML
     elements">HTML element</span> with the same tag name as the token has been popped from the
     stack.</p></li>
    </ol>
   </dd>

   <!-- removes the form element pointer instead of the matching node -->
   <dt>An end tag whose tag name is "form"</dt>
   <dd>
    <p>If there is no <code>template</code> element on the <span>stack of open elements</span>, then
    run these substeps:</p>

    <ol>
     <li><p>Let <var>node</var> be the element that the <span><code data-x="">form</code>
     element pointer</span> is set to, or null if it is not set to an element.</p></li>

     <li><p>Set the <span><code data-x="">form</code> element pointer</span> to null.</p></li>

     <li><p>If <var>node</var> is null or if the <span>stack of open elements</span> does
     not <span data-x="has an element in scope">have <var>node</var> in scope</span>, then
     this is a <span>parse error</span>; return and ignore the token.</p></li>

     <li><p><span>Generate implied end tags</span>.</p></li>

     <li><p>If the <span>current node</span> is not <var>node</var>, then this is a
     <span>parse error</span>.</p></li>

     <li><p>Remove <var>node</var> from the <span>stack of open elements</span>.</p></li>
    </ol>

    <p>If there <em>is</em> a <code>template</code> element on the <span>stack of open
    elements</span>, then run these substeps instead:</p>

    <ol>
     <li><p>If the <span>stack of open elements</span> does not <span data-x="has an element in
     scope">have a <code>form</code> element in scope</span>, then this is a <span>parse
     error</span>; return and ignore the token.</p></li>

     <li><p><span>Generate implied end tags</span>.</p></li>

     <li><p>If the <span>current node</span> is not a <code>form</code> element, then this is a
     <span>parse error</span>.</p></li>

     <li><p>Pop elements from the <span>stack of open elements</span> until a <code>form</code>
     element has been popped from the stack.</p></li>
    </ol>
   </dd>

   <!-- as normal, except </p> implies <p> if there's no <p> in scope, and needs care as the elements have optional tags -->
   <dt>An end tag whose tag name is "p"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in button
    scope">have a <code>p</code> element in button scope</span>, then this is a <span>parse
    error</span>; <span>insert an HTML element</span> for a "p" start tag token with no
    attributes.</p>

    <p><span>Close a <code>p</code> element</span>.</p>
   </dd>

   <!-- as normal, but needs care as the elements have optional tags, and are further scoped by <ol>/<ul> -->
   <dt>An end tag whose tag name is "li"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in list item
    scope">have an <code>li</code> element in list item scope</span>, then this is a <span>parse
    error</span>; ignore the token.</p>

    <p>Otherwise, run these steps:</p>

    <ol>
     <li><p><span>Generate implied end tags</span>, except for <code>li</code> elements.</p></li>

     <li><p>If the <span>current node</span> is not an <code>li</code> element, then this is a
     <span>parse error</span>.</p></li>

     <li><p>Pop elements from the <span>stack of open elements</span> until an <code>li</code>
     element has been popped from the stack.</p></li>
    </ol>
   </dd>

   <!-- as normal, but needs care as the elements have optional tags -->
   <dt>An end tag whose tag name is one of: "dd", "dt"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in
    scope">have an element in scope</span> that is an <span data-x="HTML elements">HTML
    element</span> with the same tag name as that of the token, then this is a <span>parse
    error</span>; ignore the token.</p>

    <p>Otherwise, run these steps:</p>

    <ol>
     <li><p><span>Generate implied end tags</span>, except for <span>HTML elements</span> with the
     same tag name as the token.</p></li>

     <li><p>If the <span>current node</span> is not an <span data-x="HTML elements">HTML
     element</span> with the same tag name as that of the token, then this is a <span>parse
     error</span>.</p></li>

     <li><p>Pop elements from the <span>stack of open elements</span> until an <span data-x="HTML
     elements">HTML element</span> with the same tag name as the token has been popped from the
     stack.</p></li>
    </ol>
   </dd>

   <!-- as normal, except acts as a closer for any of the h1-h6 elements -->
   <dt>An end tag whose tag name is one of: "h1", "h2", "h3", "h4", "h5", "h6"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in
    scope">have an element in scope</span> that is an <span data-x="HTML elements">HTML
    element</span> and whose tag name is one of "h1", "h2", "h3", "h4", "h5", or "h6", then this is
    a <span>parse error</span>; ignore the token.</p>

    <p>Otherwise, run these steps:</p>

    <ol>
     <li><p><span>Generate implied end tags</span>.</p></li>

     <li><p>If the <span>current node</span> is not an <span data-x="HTML elements">HTML
     element</span> with the same tag name as that of the token, then this is a <span>parse
     error</span>.</p></li>

     <li><p>Pop elements from the <span>stack of open elements</span> until an <span data-x="HTML
     elements">HTML element</span> whose tag name is one of "h1", "h2", "h3", "h4", "h5", or "h6"
     has been popped from the stack.</p></li>
    </ol>
   </dd>

   <!-- see also applet/marquee/object lower down -->

   <dt>An end tag whose tag name is "sarcasm"</dt>
   <dd>
    <p>Take a deep breath, then act as described in the "any other end
    tag" entry below.</p>
   </dd>

   <!-- ADOPTION AGENCY ELEMENTS
        Mozilla-only: bdo blink del ins sub sup q
        Safari-only: code dfn kbd nobr samp var wbr
        Both: a b big em font i s small strike strong tt u -->

   <dt>A start tag whose tag name is "a"</dt>
   <dd>
    <p>If the <span>list of active formatting elements</span> contains an <code>a</code> element
    between the end of the list and the last <span data-x="concept-parser-marker">marker</span> on
    the list (or the start of the list if there is no <span
    data-x="concept-parser-marker">marker</span> on the list), then this is a <span>parse
    error</span>; run the <span>adoption agency algorithm</span> for the token, then remove that
    element from the <span>list of active formatting elements</span> and the <span>stack of open
    elements</span> if the <span>adoption agency algorithm</span> didn't already remove it (it might
    not have if the element is not <span data-x="has an element in table scope">in table
    scope</span>).</p>

    <p class="example">In the non-conforming stream
    <code data-x="">&lt;a&nbsp;href="a">a&lt;table>&lt;a&nbsp;href="b">b&lt;/table>x</code>, the first
    <code>a</code> element would be closed upon seeing the second one, and the "x" character would
    be inside a link to "b", not to "a". This is despite the fact that the outer <code>a</code>
    element is not in table scope (meaning that a regular <code data-x="">&lt;/a></code> end tag at the start
    of the table wouldn't close the outer <code>a</code> element). The result is that the two
    <code>a</code> elements are indirectly nested inside each other &mdash; non-conforming markup
    will often result in non-conforming DOMs when parsed.</p>

    <p><span>Reconstruct the active formatting elements</span>, if any.</p>

    <p><span>Insert an HTML element</span> for the token. <span>Push onto the list of active
    formatting elements</span> that element.</p>
   </dd>

   <dt>A start tag whose tag name is one of: "b", "big", "code", "em",
   "font", "i", "s", "small", "strike", "strong", "tt", "u"</dt>
   <dd>
    <p><span>Reconstruct the active formatting elements</span>, if any.</p>

    <p><span>Insert an HTML element</span> for the token. <span>Push onto the list of active
    formatting elements</span> that element.</p>
   </dd>

   <dt>A start tag whose tag name is "nobr"</dt>
   <dd>
    <p><span>Reconstruct the active formatting elements</span>, if any.</p>

    <p>If the <span>stack of open elements</span> <span data-x="has an element in scope">has a
    <code>nobr</code> element in scope</span>, then this is a <span>parse error</span>; run the
    <span>adoption agency algorithm</span> for the token, then once again <span>reconstruct the
    active formatting elements</span>, if any.</p>

    <p><span>Insert an HTML element</span> for the token. <span>Push onto the list of active
    formatting elements</span> that element.</p>
   </dd>

   <dt>An end tag whose tag name is one of: "a",
   "b", "big", "code", "em", "font", "i", "nobr", "s", "small",
   "strike", "strong", "tt", "u"</dt>
   <dd>
    <p>Run the <span>adoption agency algorithm</span> for the token.</p>
   </dd>

   <dt>A start tag whose tag name is one of: "applet", "marquee", "object"</dt>
   <dd>
    <p><span>Reconstruct the active formatting elements</span>, if any.</p>

    <p><span>Insert an HTML element</span> for the token.</p>

    <p>Insert a <span data-x="concept-parser-marker">marker</span> at the end of the <span>list of
    active formatting elements</span>.</p>

    <p>Set the <span>frameset-ok flag</span> to "not ok".</p>
   </dd>

   <dt>An end tag token whose tag name is one of: "applet", "marquee", "object"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in
    scope">have an element in scope</span> that is an <span data-x="HTML elements">HTML
    element</span> with the same tag name as that of the token, then this is a <span>parse
    error</span>; ignore the token.</p>

    <p>Otherwise, run these steps:</p>

    <ol>
     <li><p><span>Generate implied end tags</span>.</p></li>

     <li><p>If the <span>current node</span> is not an <span data-x="HTML elements">HTML
     element</span> with the same tag name as that of the token, then this is a <span>parse
     error</span>.</p></li>

     <li><p>Pop elements from the <span>stack of open elements</span> until an <span data-x="HTML
     elements">HTML element</span> with the same tag name as the token has been popped from the
     stack.</p></li>

     <li><span>Clear the list of active formatting elements up to the last marker</span>.</li>
    </ol>
   </dd>

   <dt>A start tag whose tag name is "table"</dt>
   <dd>
    <p>If the <code>Document</code> is <em>not</em> set to <span>quirks mode</span>, and the
    <span>stack of open elements</span> <span data-x="has an element in button scope">has a
    <code>p</code> element in button scope</span>, then <span>close a <code>p</code>
    element</span>.</p> <!-- i hate myself (this quirk was basically caused by acid2; if i'd
    realized we could change the specs when i wrote acid2, we could have avoided having any
    parsing-mode quirks) -Hixie -->

    <p><span>Insert an HTML element</span> for the token.</p>

    <p>Set the <span>frameset-ok flag</span> to "not ok".</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in table">in
    table</span>".</p>
   </dd>

   <dt>An end tag whose tag name is "br"</dt>
   <dd>
    <p><span>Parse error</span>. Drop the attributes from the token, and act as described in the
    next entry; i.e. act as if this was a "br" start tag token with no attributes, rather than the
    end tag token that it actually is.</p>
   </dd>

   <!-- do not insert things here, since the previous entry refers to the next entry -->

   <dt>A start tag whose tag name is one of: "area", "br", "embed",
   "img", "keygen", "wbr"</dt>
   <dd>
    <p><span>Reconstruct the active formatting elements</span>, if any.</p>

    <p><span>Insert an HTML element</span> for the token. Immediately pop the <span>current
    node</span> off the <span>stack of open elements</span>.</p>

    <p><span data-x="acknowledge self-closing flag">Acknowledge the token's <i data-x="self-closing flag">self-closing
    flag</i></span>, if it is set.</p>

    <p>Set the <span>frameset-ok flag</span> to "not ok".</p>
    <!-- shouldn't really do this for <area> -->
   </dd>

   <dt>A start tag whose tag name is "input"</dt>
   <dd>
    <p><span>Reconstruct the active formatting elements</span>, if any.</p>

    <p><span>Insert an HTML element</span> for the token. Immediately pop the <span>current
    node</span> off the <span>stack of open elements</span>.</p>

    <p><span data-x="acknowledge self-closing flag">Acknowledge the token's <i data-x="self-closing flag">self-closing
    flag</i></span>, if it is set.</p>

    <p>If the token does not have an attribute with the name "type", or if it does, but that
    attribute's value is not an <span>ASCII case-insensitive</span> match for the string "<code
    data-x="">hidden</code>", then: set the <span>frameset-ok flag</span> to "not ok".</p>
   </dd>

   <dt>A start tag whose tag name is one of: "param", "source", "track"</dt>
   <dd>
    <p><span>Insert an HTML element</span> for the token. Immediately pop the <span>current
    node</span> off the <span>stack of open elements</span>.</p>

    <p><span data-x="acknowledge self-closing flag">Acknowledge the token's <i data-x="self-closing flag">self-closing
    flag</i></span>, if it is set.</p>
   </dd>

   <dt>A start tag whose tag name is "hr"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> <span data-x="has an element in button scope">has a
    <code>p</code> element in button scope</span>, then <span>close a <code>p</code>
    element</span>.</p>

    <p><span>Insert an HTML element</span> for the token. Immediately pop the <span>current
    node</span> off the <span>stack of open elements</span>.</p>

    <p><span data-x="acknowledge self-closing flag">Acknowledge the token's <i data-x="self-closing flag">self-closing
    flag</i></span>, if it is set.</p>

    <p>Set the <span>frameset-ok flag</span> to "not ok".</p>
   </dd>

   <dt>A start tag whose tag name is "image"</dt>
   <dd>
    <!-- As of 2005-12, studies showed that around 0.2% of pages used the <image> element. -->
    <p><span>Parse error</span>. Change the token's tag name to "img" and reprocess it. (Don't
    ask.)</p>
   </dd>

   <dt>A start tag whose tag name is "textarea"</dt>
   <dd>
    <p>Run these steps:</p>

    <ol>
     <li><p><span>Insert an HTML element</span> for the token.</p></li>

     <li><p>If the <span>next token</span> is a U+000A LINE FEED (LF) character token, then ignore
     that token and move on to the next one. (Newlines at the start of <code>textarea</code>
     elements are ignored as an authoring convenience.)</p></li>

     <!-- see comment in <pre> start tag bit -->

     <li><p>Switch the tokenizer to the <span>RCDATA state</span>.</p></li>

     <li><p>Let the <span>original insertion mode</span> be the current <span>insertion
     mode</span>.</p>

     <li><p>Set the <span>frameset-ok flag</span> to "not ok".</p></li>

     <li><p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode:
     text">text</span>".</p></li>
    </ol>
   </dd>

   <dt>A start tag whose tag name is "xmp"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> <span data-x="has an element in button scope">has a
    <code>p</code> element in button scope</span>, then <span>close a <code>p</code>
    element</span>.</p>

    <p><span>Reconstruct the active formatting elements</span>, if any.</p>

    <p>Set the <span>frameset-ok flag</span> to "not ok".</p>

    <p>Follow the <span>generic raw text element parsing algorithm</span>.</p>
   </dd>

   <dt>A start tag whose tag name is "iframe"</dt>
   <dd>
    <p>Set the <span>frameset-ok flag</span> to "not ok".</p>

    <p>Follow the <span>generic raw text element parsing algorithm</span>.</p>
   </dd>

   <dt>A start tag whose tag name is "noembed"</dt>
   <dt>A start tag whose tag name is "noscript", if the <span>scripting flag</span> is enabled</dt>
   <dd>
    <p>Follow the <span>generic raw text element parsing algorithm</span>.</p>
   </dd>

   <dt>A start tag whose tag name is "select"</dt>
   <dd>
    <p><span>Reconstruct the active formatting elements</span>, if any.</p>

    <p><span>Insert an HTML element</span> for the token.</p>

    <p>Set the <span>frameset-ok flag</span> to "not ok".</p>

    <p>If the <span>insertion mode</span> is one of "<span data-x="insertion mode: in table">in
    table</span>", "<span data-x="insertion mode: in caption">in caption</span>", "<span
    data-x="insertion mode: in table body">in table body</span>", "<span data-x="insertion mode: in
    row">in row</span>", or "<span data-x="insertion mode: in cell">in cell</span>", then switch the
    <span>insertion mode</span> to "<span data-x="insertion mode: in select in table">in select in
    table</span>". Otherwise, switch the <span>insertion mode</span> to "<span data-x="insertion
    mode: in select">in select</span>".</p>
   </dd>

   <dt>A start tag whose tag name is one of: "optgroup", "option"</dt>
   <dd>
    <p>If the <span>current node</span> is an <code>option</code> element, then pop the
    <span>current node</span> off the <span>stack of open elements</span>.</p>

    <p><span>Reconstruct the active formatting elements</span>, if any.</p>

    <p><span>Insert an HTML element</span> for the token.</p>
   </dd>

   <dt>A start tag whose tag name is one of: "rb", "rtc"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> <span data-x="has an element in scope">has a
    <code>ruby</code> element in scope</span>, then <span>generate implied end tags</span>. If the
    <span>current node</span> is not now a <code>ruby</code> element, this is a
    <span>parse error</span>.</p>

    <p><span>Insert an HTML element</span> for the token.</p>
   </dd>

   <dt>A start tag whose tag name is one of: "rp", "rt"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> <span data-x="has an element in scope">has a
    <code>ruby</code> element in scope</span>, then <span>generate implied end tags</span>, except
    for <code>rtc</code> elements. If the <span>current node</span> is not now a <code>rtc</code>
    element or a <code>ruby</code> element, this is a <span>parse error</span>.</p>

    <p><span>Insert an HTML element</span> for the token.</p>
   </dd>

   <dt>A start tag whose tag name is "math"</dt>
   <dd>
    <p><span>Reconstruct the active formatting elements</span>, if any.</p>

    <p><span>Adjust MathML attributes</span> for the token. (This fixes the case of MathML
    attributes that are not all lowercase.)</p>

    <p><span>Adjust foreign attributes</span> for the token. (This fixes the use of namespaced
    attributes, in particular XLink.)</p>

    <p><span>Insert a foreign element</span> for the token, with <span>MathML namespace</span>
    and false.</p>

    <!-- If we ever change the frameset-ok flag to an insertion mode, the following change would be
    implied, except we'd have to do it even in the face of a self-closed tag:
    <p>Set the <span>frameset-ok flag</span> to "not ok".</p>
    -->

    <p>If the token has its <i data-x="self-closing flag">self-closing flag</i> set, pop the <span>current node</span> off the
    <span>stack of open elements</span> and <span data-x="acknowledge self-closing flag">acknowledge
    the token's <i data-x="self-closing flag">self-closing flag</i></span>.</p>
   </dd>

   <dt>A start tag whose tag name is "svg"</dt>
   <dd>
    <p><span>Reconstruct the active formatting elements</span>, if any.</p>

    <p><span>Adjust SVG attributes</span> for the token. (This fixes the case of SVG attributes that
    are not all lowercase.)</p>

    <p><span>Adjust foreign attributes</span> for the token. (This fixes the use of namespaced
    attributes, in particular XLink in SVG.)</p>

    <p><span>Insert a foreign element</span> for the token, with <span>SVG namespace</span>
    and false.</p>

    <!-- If we ever change the frameset-ok flag to an insertion mode, the following change would be
    implied, except we'd have to do it even in the face of a self-closed tag:
    <p>Set the <span>frameset-ok flag</span> to "not ok".</p>
    -->

    <p>If the token has its <i data-x="self-closing flag">self-closing flag</i> set, pop the <span>current node</span> off the
    <span>stack of open elements</span> and <span data-x="acknowledge self-closing flag">acknowledge
    the token's <i data-x="self-closing flag">self-closing flag</i></span>.</p>
   </dd>

   <dt>A start <!--or end--> tag whose tag name is one of: "caption", "col", "colgroup", "frame",
   "head", "tbody", "td", "tfoot", "th", "thead", "tr"</dt>
   <!--<dt>An end tag whose tag name is one of: "area", "base", "basefont", "bgsound", "embed",
   "hr", "iframe", "image", "img", "input", "keygen", "link", "meta", "noembed", "noframes",
   "param", "script", "select", "source", "style", "table", "textarea", "title", "track",
   "wbr"</dt>-->
   <!--<dt>An end tag whose tag name is "noscript", if the <span>scripting flag</span> is
   enabled</dt>-->
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>

    <!-- end tags are commented out because since they can never end up on the stack anyway, the
    default end tag clause will automatically handle them. we don't want to have text in the spec
    that is just an optimization, as that detracts from the spec itself -->
   </dd>

   <dt>Any other start tag</dt>
   <dd>
    <p><span>Reconstruct the active formatting elements</span>, if any.</p>

    <p><span>Insert an HTML element</span> for the token.</p>

    <p class="note">This element will be an <span>ordinary</span> element. With one exception: if
    the <span>scripting flag</span> is disabled, it can also be a <code>noscript</code> element.</p>
   </dd>

   <dt>Any other end tag</dt>
   <dd>
    <p>Run these steps:</p>

    <ol>
     <li><p>Initialize <var>node</var> to be the <span>current node</span> (the bottommost
     node of the stack).</p></li>

     <li><p><i>Loop</i>: If <var>node</var> is an <span data-x="HTML elements">HTML
     element</span> with the same tag name as the token, then:</p>

      <ol>
       <li><p><span>Generate implied end tags</span>, except for <span>HTML elements</span> with the
       same tag name as the token.</p></li>

       <li><p>If <var>node</var> is not the <span>current node</span>, then this is a
       <span>parse error</span>.</p></li>

       <li><p>Pop all the nodes from the <span>current node</span> up to <var>node</var>,
       including <var>node</var>, then stop these steps.</p></li>
      </ol>
     </li>

     <li><p>Otherwise, if <var>node</var> is in the <span>special</span> category, then
     this is a <span>parse error</span>; ignore the token, and return.</p></li>

     <li><p>Set <var>node</var> to the previous entry in the <span>stack of open
     elements</span>.</p></li>

     <li><p>Return to the step labeled <i>loop</i>.</p></li>
    </ol>
   </dd>
  </dl>

  <p>When the steps above say the user agent is to <dfn>close a <code>p</code> element</dfn>, it
  means that the user agent must run the following steps:</p>

  <ol> <!-- prereq: p in scope -->

   <li><p><span>Generate implied end tags</span>, except for <code>p</code> elements.</p></li>

   <li><p>If the <span>current node</span> is not a <code>p</code> element, then this is a
   <span>parse error</span>.</p></li>

   <li><p>Pop elements from the <span>stack of open elements</span> until a <code>p</code> element
   has been popped from the stack.</p></li>
  </ol>

  <!-- AAA -->
  <p id="adoptionAgency">The <dfn>adoption agency algorithm</dfn>, which takes as its only argument
  a token <var>token</var> for which the algorithm is being run, consists of the following
  steps:</p>

  <ol>
   <!-- don't forget about the noah's ark clause when looking at this algorithm! -->

   <li><p>Let <var>subject</var> be <var>token</var>'s tag name.</p></li>

   <li><p>If the <span>current node</span> is an <span data-x="HTML elements">HTML element</span>
   whose tag name is <var>subject</var>, and the <span>current node</span> is not in the
   <span>list of active formatting elements</span>, then pop the <span>current node</span> off the
   <span>stack of open elements</span> and return.</p></li>

   <li><p>Let <var>outerLoopCounter</var> be 0.</p></li>

   <li>
    <p>While true:</p>

    <ol>
     <li><p>If <var>outerLoopCounter</var> is greater than or equal to 8, then return.</p></li>

     <li><p>Increment <var>outerLoopCounter</var> by 1.</p></li>

     <li>
      <p>Let <var>formattingElement</var> be the last element in the <span>list of active
      formatting elements</span> that:</p>

      <ul>
       <li>is between the end of the list and the last <span
       data-x="concept-parser-marker">marker</span> in the list, if any, or the start of the list
       otherwise, and</li>

       <li>has the tag name <var>subject</var>.</li>
      </ul>

      <p>If there is no such element, then return and instead act as described in the "any other end
      tag" entry above.</p>
     </li>

     <li><p>If <var>formattingElement</var> is not in the <span>stack of open elements</span>, then
     this is a <span>parse error</span>; remove the element from the list, and return.</p></li>

     <li><p>If <var>formattingElement</var> is in the <span>stack of open elements</span>, but the
     element is not <span data-x="has an element in scope">in scope</span>, then this is a
     <span>parse error</span>; return.</p></li>

     <!-- at this point, <var>formattingElement</var> is in <span data-x="stack of open
          elements">the stack</span> and <span data-x="list of active formatting elements">the
          list</span>, and is <span data-x="has an element in scope">in scope</span>. -->

     <li><p>If <var>formattingElement</var> is not the <span>current node</span>, this is a
     <span>parse error</span>. (But do not return.)</p></li>

     <li><p>Let <var>furthestBlock</var> be the topmost node in the <span>stack of open
     elements</span> that is lower in the stack than <var>formattingElement</var>, and is an
     element in the <span>special</span> category. There might not be one.</p></li>

     <!-- <html> ... <formattingElement> ... <furthestBlock> ... <currentNode> -->

     <li><p>If there is no <var>furthestBlock</var>, then the UA must first pop all the nodes from
     the bottom of the <span>stack of open elements</span>, from the <span>current node</span> up to
     and including <var>formattingElement</var>, then remove <var>formattingElement</var> from the
     <span>list of active formatting elements</span>, and finally return.</p></li>
     <!-- the "reconstruct the active formatting elements" algorithm will rebuild them later -->

     <li><p>Let <var>commonAncestor</var> be the element immediately above
     <var>formattingElement</var> in the <span>stack of open elements</span>.</p></li>

     <!-- <html> ... <commonAncestor> <formattingElement> ... <furthestBlock> ... <currentNode> -->

     <li><p>Let a bookmark note the position of <var>formattingElement</var> in the <span>list of
     active formatting elements</span> relative to the elements on either side of it in the
     list.</p></li>

     <li><p>Let <var>node</var> and <var>lastNode</var> be <var>furthestBlock</var>.</p></li>

     <li><p>Let <var>innerLoopCounter</var> be 0.</p></li>

     <li>
      <p>While true:</p>

      <ol>
       <li><p>Increment <var>innerLoopCounter</var> by 1.</p></li>

       <li><p>Let <var>node</var> be the element immediately above <var>node</var> in the
       <span>stack of open elements</span>, or if <var>node</var> is no longer in the <span>stack of
       open elements</span> (e.g. because it got removed by this algorithm<!-- in particular, the
       step labeled "removal" below -->), the element that was immediately above <var>node</var> in
       the <span>stack of open elements</span> before <var>node</var> was removed.</p></li>

       <li><p>If <var>node</var> is <var>formattingElement</var>, then <span>break</span>.</p></li>

       <li><p>If <var>innerLoopCounter</var> is greater than 3 and <var>node</var> is in the
       <span>list of active formatting elements</span>, then remove <var>node</var> from the
       <span>list of active formatting elements</span>.</p></li>

       <li><p><!-- "removal" step: -->If <var>node</var> is not in the <span>list of active
       formatting elements</span>, then remove <var>node</var> from the <span>stack of open
       elements</span> and <span>continue</span>.</p></li>

       <li><p><span>Create an element for the token</span> for which the element <var>node</var> was
       created, in the <span>HTML namespace</span>, with <var>commonAncestor</var> as the intended
       parent; replace the entry for <var>node</var> in the <span>list of active formatting
       elements</span> with an entry for the new element, replace the entry for <var>node</var> in
       the <span>stack of open elements</span> with an entry for the new element, and let
       <var>node</var> be the new element.</p></li>

       <li><p>If <var>lastNode</var> is <var>furthestBlock</var>, then move the aforementioned
       bookmark to be immediately after the new <var>node</var> in the <span>list of active
       formatting elements</span>.</p></li>

       <li><p><span data-x="concept-node-append">Append</span> <var>lastNode</var> to
       <var>node</var>.</p></li>

       <li><p>Set <var>lastNode</var> to <var>node</var>.</p></li>
      </ol>
     </li>

     <li><p>Insert whatever <var>lastNode</var> ended up being in the previous step at the
     <span>appropriate place for inserting a node</span>, but using <var>commonAncestor</var>
     as the <i>override target</i>.</p></li>

     <li><p><span>Create an element for the token</span> for which <var>formattingElement</var> was created,
     in the <span>HTML namespace</span>, with <var>furthestBlock</var> as the intended parent.</p></li>

     <li><p>Take all of the child nodes of <var>furthestBlock</var> and append them to the
     element created in the last step.</p></li>

     <li><p>Append that new element to <var>furthestBlock</var>.</p></li>

     <li><p>Remove <var>formattingElement</var> from the <span>list of active formatting
     elements</span>, and insert the new element into the <span>list of active formatting
     elements</span> at the position of the aforementioned bookmark.</p></li>

     <li><p>Remove <var>formattingElement</var> from the <span>stack of open
     elements</span>, and insert the new element into the <span>stack of open elements</span>
     immediately below the position of <var>furthestBlock</var> in that stack.</p></li>
    </ol>
   </li>
  </ol>

  <p class="note">This algorithm's name, the "adoption agency algorithm", comes from the way it
  causes elements to change parents, and is in contrast with <a
  href="https://ln.hixie.ch/?start=1037910467&amp;count=1">other possible algorithms</a> for dealing
  with misnested content.</p>





  <h6 id="parsing-main-incdata">The "<dfn data-x="insertion mode: text">text</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode:
  text">text</span>" <span>insertion mode</span>, the user agent must handle the token as
  follows:</p>

  <dl class="switch">

   <dt>A character token</dt>
   <dd>
    <p><span data-x="insert a character">Insert the token's character</span>.</p>

    <p class="note">This can never be a U+0000 NULL character; the tokenizer converts those to
    U+FFFD REPLACEMENT CHARACTER characters.</p>
   </dd>

   <dt>An end-of-file token</dt>
   <dd>
    <!-- can't be the fragment case -->
    <p><span>Parse error</span>.</p>

    <p>If the <span>current node</span> is a <code>script</code> element, then set its <span>already
    started</span> to true.</p>

    <p>Pop the <span>current node</span> off the <span>stack of open elements</span>.</p>

    <p>Switch the <span>insertion mode</span> to the <span>original insertion mode</span> and
    reprocess the token.</p>
   </dd>

   <dt id="scriptEndTag">An end tag whose tag name is "script"</dt>
   <dd>
    <p>If the <span>active speculative HTML parser</span> is null and the <span>JavaScript execution
    context stack</span> is empty, then <span>perform a microtask checkpoint</span>.</p>

    <p>Let <var>script</var> be the <span>current node</span> (which will be a
    <code>script</code> element).</p>

    <p>Pop the <span>current node</span> off the <span>stack of open elements</span>.</p>

    <p>Switch the <span>insertion mode</span> to the <span>original insertion mode</span>.</p>

    <p>Let the <var>old insertion point</var> have the same value as the current
    <span>insertion point</span>. Let the <span>insertion point</span> be just before the <span>next
    input character</span>.</p>

    <p>Increment the parser's <span>script nesting level</span> by one.</p>

    <p>If the <span>active speculative HTML parser</span> is null, then <span>prepare the script
    element</span> <var>script</var>. This might cause some script to execute, which might cause
    <span data-x="dom-document-write">new characters to be inserted into the tokenizer</span>, and
    might cause the tokenizer to output more tokens, resulting in a <a
    href="#nestedParsing">reentrant invocation of the parser</a>.</p>

    <p>Decrement the parser's <span>script nesting level</span> by one. If the parser's <span>script
    nesting level</span> is zero, then set the <span>parser pause flag</span> to false.</p>

    <p>Let the <span>insertion point</span> have the value of the <var>old insertion
    point</var>. (In other words, restore the <span>insertion point</span> to its previous value.
    This value might be the "undefined" value.)</p>

    <p id="scriptTagParserResumes">At this stage, if the <span>pending parsing-blocking
    script</span> is not null, then:</p>

    <dl class="switch">

     <dt>If the <span>script nesting level</span> is not zero:</dt>

     <dd>
      <p>Set the <span>parser pause flag</span> to true, and abort the processing of any nested
      invocations of the tokenizer, yielding control back to the caller. (Tokenization will resume
      when the caller returns to the "outer" tree construction stage.)</p>

      <p class="note">The tree construction stage of this particular parser is <a
      href="#nestedParsing">being called reentrantly</a>, say from a call to <code
      data-x="dom-document-write">document.write()</code>.</p>
     </dd>


     <dt>Otherwise:</dt>

     <dd>
      <p>While the <span>pending parsing-blocking script</span> is not null:</p>

      <ol>
       <li><p>Let <var>the script</var> be the <span>pending parsing-blocking
       script</span>.</p></li>

       <li><p>Set the <span>pending parsing-blocking script</span> to null.</p></li>

       <li><p><span>Start the speculative HTML parser</span> for this instance of the HTML
       parser.</p></li>

       <li><p>Block the <span data-x="tokenization">tokenizer</span> for this instance of the
       <span>HTML parser</span>, such that the <span>event loop</span> will not run <span
       data-x="concept-task">tasks</span> that invoke the <span
       data-x="tokenization">tokenizer</span>.</p></li>

       <li><p>If the parser's <code>Document</code> <span>has a style sheet that is blocking
       scripts</span> or <var>the script</var>'s <span>ready to be parser-executed</span> is false:
       <span>spin the event loop</span> until the parser's <code>Document</code> <span>has no style
       sheet that is blocking scripts</span> and <var>the script</var>'s <span>ready to be
       parser-executed</span> becomes true.</p></li>

       <li>
        <p>If this <span data-x="abort a parser">parser has been aborted</span> in the meantime,
        return.</p>

        <p class="note">This could happen if, e.g., while the <span>spin the event loop</span>
        algorithm is running, the <code>Document</code> gets <span data-x="destroy a
        document">destroyed</span>, or the <code data-x="dom-document-open">document.open()</code>
        method gets invoked on the <code>Document</code>.</p>
       </li>

       <li><p><span>Stop the speculative HTML parser</span> for this instance of the HTML
       parser.</p></li>

       <li><p>Unblock the <span data-x="tokenization">tokenizer</span> for this instance of the
       <span>HTML parser</span>, such that <span data-x="concept-task">tasks</span> that invoke the
       <span data-x="tokenization">tokenizer</span> can again be run.</p></li>

       <li><p>Let the <span>insertion point</span> be just before the <span>next input
       character</span>.</p></li>

       <li><p>Increment the parser's <span>script nesting level</span> by one (it should be zero
       before this step, so this sets it to one).</p></li>

       <li><p><span>Execute the script element</span> <var>the script</var>.</p></li>

       <li><p>Decrement the parser's <span>script nesting level</span> by one. If the parser's
       <span>script nesting level</span> is zero (which it always should be at this point), then set
       the <span>parser pause flag</span> to false.</p>

       <li><p>Let the <span>insertion point</span> be undefined again.</p></li>
      </ol>
     </dd>
    </dl>
   </dd>

   <dt>Any other end tag</dt>
   <dd>
    <p>Pop the <span>current node</span> off the <span>stack of open elements</span>.</p>

    <p>Switch the <span>insertion mode</span> to the <span>original insertion mode</span>.</p>
   </dd>
  </dl>


  <h6 id="parsing-main-intable">The "<dfn data-x="insertion mode: in table">in table</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: in table">in
  table</span>" <span>insertion mode</span>, the user agent must handle the token as follows:</p>

  <dl class="switch">

   <dt>A character token, if the <span>current node</span> is <code>table</code>, <code>tbody</code>, <code>template</code>, <code>tfoot</code>, <code>thead</code>, or <code>tr</code> element</dt> <!-- same list as foster parenting + template -->
   <dd>
     <p>Let the <dfn><var data-x="concept-pending-table-char-tokens">pending table character
     tokens</var></dfn> be an empty list of tokens.</p>

     <p>Let the <span>original insertion mode</span> be the current <span>insertion mode</span>.</p>

     <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in table text">in
     table text</span>" and reprocess the token.</p>
   </dd>

   <dt>A comment token</dt>
   <dd>
    <p><span>Insert a comment</span>.</p>
   </dd>

   <dt>A DOCTYPE token</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>A start tag whose tag name is "caption"</dt>
   <dd>
    <p><span>Clear the stack back to a table context</span>. (See below.)</p>

    <p>Insert a <span data-x="concept-parser-marker">marker</span> at the end of the <span>list of
    active formatting elements</span>.</p>

    <p><span>Insert an HTML element</span> for the token, then switch the <span>insertion
    mode</span> to "<span data-x="insertion mode: in caption">in caption</span>".</p>
   </dd>

   <dt>A start tag whose tag name is "colgroup"</dt>
   <dd>
    <p><span>Clear the stack back to a table context</span>. (See below.)</p>

    <p><span>Insert an HTML element</span> for the token, then switch the <span>insertion
    mode</span> to "<span data-x="insertion mode: in column group">in column group</span>".</p>
   </dd>

   <dt>A start tag whose tag name is "col"</dt>
   <dd>
    <!-- fake <colgroup> -->
    <p><span>Clear the stack back to a table context</span>. (See below.)</p>

    <p><span>Insert an HTML element</span> for a "colgroup" start tag token with no attributes, then
    switch the <span>insertion mode</span> to "<span data-x="insertion mode: in column group">in
    column group</span>".</p>
    <!-- end of fake <colgroup> -->

    <p>Reprocess the current token.</p>
   </dd>

   <dt>A start tag whose tag name is one of: "tbody", "tfoot", "thead"</dt>
   <dd>
    <p><span>Clear the stack back to a table context</span>. (See below.)</p>

    <p><span>Insert an HTML element</span> for the token, then switch the <span>insertion
    mode</span> to "<span data-x="insertion mode: in table body">in table body</span>".</p>
   </dd>

   <dt>A start tag whose tag name is one of: "td", "th", "tr"</dt>
   <dd>
    <!-- fake <colgroup> -->
    <p><span>Clear the stack back to a table context</span>. (See below.)</p>

    <p><span>Insert an HTML element</span> for a "tbody" start tag token with no attributes, then
    switch the <span>insertion mode</span> to "<span data-x="insertion mode: in table body">in table
    body</span>".</p>
    <!-- end of fake <colgroup> -->

    <p>Reprocess the current token.</p>
   </dd>

   <dt>A start tag whose tag name is "table"</dt>
   <dd>
    <p><span>Parse error</span>.</p>

    <!-- fake </table> -->
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in table
    scope">have a <code>table</code> element in table scope</span>, ignore the token.</p>

    <p>Otherwise:</p>

    <p>Pop elements from this stack until a <code>table</code> element has been popped from the
    stack.</p>

    <p><span>Reset the insertion mode appropriately</span>.</p>
    <!-- end of fake </table> -->

    <p>Reprocess the token.</p>
   </dd>

   <dt>An end tag whose tag name is "table"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in table
    scope">have a <code>table</code> element in table scope</span>, this is a <span>parse
    error</span>; ignore the token.</p>

    <p>Otherwise:</p>

    <p>Pop elements from this stack until a <code>table</code> element has been popped from the
    stack.</p>

    <p><span>Reset the insertion mode appropriately</span>.</p>
   </dd>

   <dt>An end tag whose tag name is one of: "body", "caption", "col", "colgroup", "html", "tbody",
   "td", "tfoot", "th", "thead", "tr"</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>A start tag whose tag name is one of: "style", "script", "template"</dt>
   <dt>An end tag whose tag name is "template"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span
    data-x="insertion mode: in head">in head</span>" <span>insertion
    mode</span>.</p>
   </dd>

   <dt>A start tag whose tag name is "input"</dt>
   <dd>
    <p>If the token does not have an attribute with the name "type", or if it does, but that
    attribute's value is not an <span>ASCII case-insensitive</span> match for the string "<code
    data-x="">hidden</code>", then: act as described in the "anything else" entry below.</p>

    <p>Otherwise:</p>

    <p><span>Parse error</span>.</p>

    <p><span>Insert an HTML element</span> for the token.</p>

    <p>Pop that <code>input</code> element off the <span>stack of open elements</span>.</p>

    <p><span data-x="acknowledge self-closing flag">Acknowledge the token's <i data-x="self-closing flag">self-closing
    flag</i></span>, if it is set.</p>
   </dd>

   <dt>A start tag whose tag name is "form"</dt>
   <dd>
    <p><span>Parse error</span>.</p>

    <p>If there is a <code>template</code> element on the <span>stack of open elements</span>, or if
    the <span><code data-x="form">form</code> element pointer</span> is not null, ignore the
    token.</p> <!-- in a <template>, the <form> would have no effect and thus be a waste of time... -->

    <p>Otherwise:</p>

    <p><span>Insert an HTML element</span> for the token, and set the <span><code
    data-x="form">form</code> element pointer</span> to point to the element created.</p>

    <p>Pop that <code>form</code> element off the <span>stack of open elements</span>.</p>
   </dd>

   <!-- "form" end tag falls through to in-body, which does the right thing -->

   <dt>An end-of-file token</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <p><span>Parse error</span>. Enable <span data-x="foster parent">foster parenting</span>, process
    the token <span>using the rules for</span> the "<span data-x="insertion mode: in body">in
    body</span>" <span>insertion mode</span>, and then disable <span data-x="foster parent">foster
    parenting</span>.</p>
   </dd>
  </dl>

  <p>When the steps above require the UA to <dfn>clear the stack back to a table context</dfn>, it
  means that the UA must, while the <span>current node</span> is not a <code>table</code>,
  <code>template</code>, or <code>html</code> element, pop elements from the <span>stack of open
  elements</span>.</p>

  <p class="note">This is the same list of elements as used in the <i data-x="has an element in
  table scope">has an element in table scope</i> steps.</p>

  <p class="note">The <span>current node</span> being an <code>html</code> element after this
  process is a <span>fragment case</span>.</p>



  <h6 id="parsing-main-intabletext">The "<dfn data-x="insertion mode: in table text">in table text</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: in table
  text">in table text</span>" <span>insertion mode</span>, the user agent must handle the token as
  follows:</p>

  <dl class="switch">

   <dt>A character token that is U+0000 NULL</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>


   <dt>Any other character token</dt>
   <dd>
    <p>Append the character token to the <var data-x="concept-pending-table-char-tokens">pending
    table character tokens</var> list.</p>
   </dd>


   <dt>Anything else</dt>
   <dd>
    <!-- this can only be called if the current node is one of the table model elements -->

    <p>If any of the tokens in the <var data-x="concept-pending-table-char-tokens">pending table
    character tokens</var> list are character tokens that are not <span>ASCII whitespace</span>,
    then this is a <span>parse error</span>: reprocess the character tokens in the <var
    data-x="concept-pending-table-char-tokens">pending table character tokens</var> list using the
    rules given in the "anything else" entry in the "<span data-x="insertion mode: in table">in
    table</span>" insertion mode.</p>

    <!-- if there's active formatting elements, it'll recreate those and foster parent the top one
    and then put the text nodes in the formatting elements; otherwise, it'll foster parent the
    character tokens. -->

    <p>Otherwise, <span data-x="insert a character">insert the characters</span> given by the <var
    data-x="concept-pending-table-char-tokens">pending table character tokens</var> list.</p> <!--
    i.e. inter-element whitespace in the table model isn't foster parented -->

    <!-- no need to empty the list, we're leaving the insertion mode and the list is always emptied
    before we reenter the mode -->

    <p>Switch the <span>insertion mode</span> to the <span>original insertion mode</span> and
    reprocess the token.</p>
   </dd>
  </dl>


  <h6 id="parsing-main-incaption">The "<dfn data-x="insertion mode: in caption">in caption</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: in caption">in
  caption</span>" <span>insertion mode</span>, the user agent must handle the token as follows:</p>

  <dl class="switch">

   <dt>An end tag whose tag name is "caption"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in table
    scope">have a <code>caption</code> element in table scope</span>, this is a <span>parse
    error</span>; ignore the token. (<span>fragment case</span>)</p>

    <p>Otherwise:</p>

    <p><span>Generate implied end tags</span>.</p>

    <p>Now, if the <span>current node</span> is not a <code>caption</code> element, then this is a
    <span>parse error</span>.</p>

    <p>Pop elements from this stack until a <code>caption</code> element has been popped from the
    stack.</p>

    <p><span>Clear the list of active formatting elements up to the last marker</span>.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in table">in
    table</span>".</p>
   </dd>

   <dt>A start tag whose tag name is one of: "caption", "col", "colgroup", "tbody", "td", "tfoot",
   "th", "thead", "tr"</dt> <dt>An end tag whose tag name is "table"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in table
    scope">have a <code>caption</code> element in table scope</span>, this is a <span>parse
    error</span>; ignore the token. (<span>fragment case</span>)</p>

    <p>Otherwise:</p>

    <!-- fake </caption> -->
    <p><span>Generate implied end tags</span>.</p>

    <p>Now, if the <span>current node</span> is not a <code>caption</code> element, then this is a
    <span>parse error</span>.</p>

    <p>Pop elements from this stack until a <code>caption</code> element has been popped from the
    stack.</p>

    <p><span>Clear the list of active formatting elements up to the last marker</span>.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in table">in
    table</span>".</p>
    <!-- end of fake </caption> -->

    <p>Reprocess the token.</p>
   </dd>

   <dt>An end tag whose tag name is one of: "body", "col", "colgroup", "html", "tbody", "td",
   "tfoot", "th", "thead", "tr"</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>
  </dl>


  <h6 id="parsing-main-incolgroup">The "<dfn data-x="insertion mode: in column group">in column group</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: in column
  group">in column group</span>" <span>insertion mode</span>, the user agent must handle the token
  as follows:</p>

  <dl class="switch">

   <dt>A character token that is one of U+0009 CHARACTER TABULATION, U+000A LINE FEED (LF), U+000C
   FORM FEED (FF), U+000D CARRIAGE RETURN (CR), or U+0020 SPACE</dt>
   <dd>
    <p><span data-x="insert a character">Insert the character</span>.</p>
   </dd>

   <dt>A comment token</dt>
   <dd>
    <p><span>Insert a comment</span>.</p>
   </dd>

   <dt>A DOCTYPE token</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>A start tag whose tag name is "html"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>A start tag whose tag name is "col"</dt>
   <dd>
    <p><span>Insert an HTML element</span> for the token. Immediately pop the <span>current
    node</span> off the <span>stack of open elements</span>.</p>

    <p><span data-x="acknowledge self-closing flag">Acknowledge the token's <i data-x="self-closing flag">self-closing
    flag</i></span>, if it is set.</p>
   </dd>

   <dt>An end tag whose tag name is "colgroup"</dt>
   <dd>
    <p>If the <span>current node</span> is not a <code>colgroup</code> element, then this is a
    <span>parse error</span>; ignore the token.</p> <!-- e.g. colgroup fragment case, or
    <template><col></colgroup> -->

    <p>Otherwise, pop the <span>current node</span> from the <span>stack of open elements</span>.
    Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in table">in
    table</span>".</p>
   </dd>

   <dt>An end tag whose tag name is "col"</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>A start tag whose tag name is "template"</dt>
   <dt>An end tag whose tag name is "template"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    head">in head</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>An end-of-file token</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <!-- fake </colgroup> -->
    <p>If the <span>current node</span> is not a <code>colgroup</code> element, then this is a
    <span>parse error</span>; ignore the token.</p> <!-- e.g. colgroup fragment case, or
    <template><col></colgroup> -->

    <p>Otherwise, pop the <span>current node</span> from the <span>stack of open
    elements</span>.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in table">in
    table</span>".</p>
    <!-- end of fake </colgroup> -->

    <p>Reprocess the token.</p>
   </dd>
  </dl>


  <h6 id="parsing-main-intbody">The "<dfn data-x="insertion mode: in table body">in table body</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: in table
  body">in table body</span>" <span>insertion mode</span>, the user agent must handle the token as
  follows:</p>

  <dl class="switch">

   <dt>A start tag whose tag name is "tr"</dt>
   <dd>
    <p><span>Clear the stack back to a table body context</span>. (See below.)</p>

    <p><span>Insert an HTML element</span> for the token, then switch the <span>insertion
    mode</span> to "<span data-x="insertion mode: in row">in row</span>".</p>
   </dd>

   <dt>A start tag whose tag name is one of: "th", "td"</dt>
   <dd>
    <p><span>Parse error</span>.</p>

    <!-- fake </tr> -->
    <p><span>Clear the stack back to a table body context</span>. (See below.)</p>

    <p><span>Insert an HTML element</span> for a "tr" start tag token with no attributes, then
    switch the <span>insertion mode</span> to "<span data-x="insertion mode: in row">in
    row</span>".</p>
    <!-- end of fake </tr> -->

    <p>Reprocess the current token.</p>
   </dd>

   <dt>An end tag whose tag name is one of: "tbody", "tfoot",
   "thead"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in table
    scope">have an element in table scope</span> that is an <span data-x="HTML elements">HTML
    element</span> with the same tag name as the token, this is a <span>parse error</span>;
    ignore the token.</p>

    <p>Otherwise:</p>

    <p><span>Clear the stack back to a table body context</span>. (See below.)</p>

    <p>Pop the <span>current node</span> from the <span>stack of open elements</span>. Switch the
    <span>insertion mode</span> to "<span data-x="insertion mode: in table">in table</span>".</p>
   </dd>

   <dt>A start tag whose tag name is one of: "caption", "col",
   "colgroup", "tbody", "tfoot", "thead"</dt>
   <dt>An end tag whose tag name is "table"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in table
    scope">have a <code>tbody</code>, <code>thead</code>, or <code>tfoot</code> element in table
    scope</span>, this is a <span>parse error</span>; ignore the token.</p>

    <p>Otherwise:</p>

    <p><span>Clear the stack back to a table body context</span>. (See below.)</p>

    <!-- fake </tbody>, </tfoot>, or </thead>, whatever is the current node -->

    <p>Pop the <span>current node</span> from the <span>stack of open elements</span>. Switch the
    <span>insertion mode</span> to "<span data-x="insertion mode: in table">in table</span>".</p>

    <!-- end of fake </tbody>, </tfoot>, or </thead>, whatever was the current node -->

    <p>Reprocess the token.</p>
   </dd>

   <dt>An end tag whose tag name is one of: "body", "caption", "col", "colgroup", "html", "td",
   "th", "tr"</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    table">in table</span>" <span>insertion mode</span>.</p>
   </dd>
  </dl>

  <p>When the steps above require the UA to <dfn>clear the stack back to a table body context</dfn>,
  it means that the UA must, while the <span>current node</span> is not a <code>tbody</code>,
  <code>tfoot</code>, <code>thead</code>, <code>template</code>, or <code>html</code> element, pop
  elements from the <span>stack of open elements</span>.</p>

  <p class="note">The <span>current node</span> being an <code>html</code> element after this
  process is a <span>fragment case</span>.</p>


  <h6 id="parsing-main-intr">The "<dfn data-x="insertion mode: in row">in row</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: in row">in
  row</span>" <span>insertion mode</span>, the user agent must handle the token as follows:</p>

  <dl class="switch">

   <dt>A start tag whose tag name is one of: "th", "td"</dt>
   <dd>
    <p><span>Clear the stack back to a table row context</span>. (See below.)</p>

    <p><span>Insert an HTML element</span> for the token, then switch the <span>insertion
    mode</span> to "<span data-x="insertion mode: in cell">in cell</span>".</p>

    <p>Insert a <span data-x="concept-parser-marker">marker</span> at the end of the <span>list of
    active formatting elements</span>.</p>
   </dd>

   <dt>An end tag whose tag name is "tr"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in table
    scope">have a <code>tr</code> element in table scope</span>, this is a <span>parse error</span>;
    ignore the token.</p>

    <p>Otherwise:</p>

    <p><span>Clear the stack back to a table row context</span>. (See below.)</p>

    <p>Pop the <span>current node</span> (which will be a <code>tr</code> element) from the
    <span>stack of open elements</span>. Switch the <span>insertion mode</span> to "<span
    data-x="insertion mode: in table body">in table body</span>".</p>
   </dd>

   <dt>A start tag whose tag name is one of: "caption", "col", "colgroup", "tbody", "tfoot",
   "thead", "tr"</dt>
   <dt>An end tag whose tag name is "table"</dt>
   <dd>
    <!-- fake <tr> -->
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in table
    scope">have a <code>tr</code> element in table scope</span>, this is a <span>parse error</span>;
    ignore the token.</p>

    <p>Otherwise:</p>

    <p><span>Clear the stack back to a table row context</span>. (See below.)</p>

    <p>Pop the <span>current node</span> (which will be a <code>tr</code> element) from the
    <span>stack of open elements</span>. Switch the <span>insertion mode</span> to "<span
    data-x="insertion mode: in table body">in table body</span>".</p>
    <!-- end of fake </tr> -->

    <p>Reprocess the token.</p>
   </dd>

   <dt>An end tag whose tag name is one of: "tbody", "tfoot", "thead"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in table
    scope">have an element in table scope</span> that is an <span data-x="HTML elements">HTML
    element</span> with the same tag name as the token, this is a <span>parse error</span>;
    ignore the token.</p>

    <!-- fake <tr> -->
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in table
    scope">have a <code>tr</code> element in table scope</span>, ignore the token.</p>

    <p>Otherwise:</p>

    <p><span>Clear the stack back to a table row context</span>. (See below.)</p>

    <p>Pop the <span>current node</span> (which will be a <code>tr</code> element) from the
    <span>stack of open elements</span>. Switch the <span>insertion mode</span> to "<span
    data-x="insertion mode: in table body">in table body</span>".</p>
    <!-- end of fake </tr> -->

    <p>Reprocess the token.</p>
   </dd>

   <dt>An end tag whose tag name is one of: "body", "caption", "col", "colgroup", "html", "td",
   "th"</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    table">in table</span>" <span>insertion mode</span>.</p>
   </dd>
  </dl>

  <p>When the steps above require the UA to <dfn>clear the stack back to a table row context</dfn>,
  it means that the UA must, while the <span>current node</span> is not a <code>tr</code>,
  <code>template</code>, or <code>html</code> element, pop elements from the <span>stack of open
  elements</span>.</p>

  <p class="note">The <span>current node</span> being an <code>html</code> element after this
  process is a <span>fragment case</span>.</p>


  <h6 id="parsing-main-intd">The "<dfn data-x="insertion mode: in cell">in cell</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: in cell">in cell</span>" <span>insertion mode</span>, the user agent must handle the token as follows:</p>

  <dl class="switch">

   <dt>An end tag whose tag name is one of: "td", "th"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in table
    scope">have an element in table scope</span> that is an <span data-x="HTML elements">HTML
    element</span> with the same tag name as that of the token, then this is a <span>parse
    error</span>; ignore the token.</p>

    <p>Otherwise:</p>

    <p><span>Generate implied end tags</span>.</p>

    <p>Now, if the <span>current node</span> is not an <span data-x="HTML elements">HTML
    element</span> with the same tag name as the token, then this is a <span>parse error</span>.</p>

    <p>Pop elements from the <span>stack of open elements</span> until an <span data-x="HTML
    elements">HTML element</span> with the same tag name as the token has been popped from the
    stack.</p>

    <p><span>Clear the list of active formatting elements up to the last marker</span>.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in row">in
    row</span>".</p> <!-- current node here will be a <tr> normally; but could be <html> in the
    fragment case, or <template> in the template case -->
   </dd>

   <dt>A start tag whose tag name is one of: "caption", "col",
   "colgroup", "tbody", "td", "tfoot", "th", "thead", "tr"</dt>
   <dd>
    <p><span>Assert</span>: The <span>stack of open elements</span> <span data-x="has an element in
    table scope">has a <code>td</code> or <code>th</code> element in table scope</span>.</p>
    <!-- Because "in row" only switches to "in cell" if a td or th element is on the stack
    and the rules for resetting the insertion mode only yield "in cell" if last is false
    (i.e. not if node is the fragment context). -->

    <p><span>Close the cell</span> (see below) and reprocess the token.</p>
   </dd>

   <dt>An end tag whose tag name is one of: "body", "caption",
   "col", "colgroup", "html"</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>An end tag whose tag name is one of: "table", "tbody",
   "tfoot", "thead", "tr"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in table
    scope">have an element in table scope</span> that is an <span data-x="HTML elements">HTML
    element</span> with the same tag name as that of the token, then this is a <span>parse
    error</span>; ignore the token.</p>

    <p>Otherwise, <span>close the cell</span> (see below) and reprocess the token.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>
  </dl>

  <p>Where the steps above say to <dfn>close the cell</dfn>, they mean to run the following
  algorithm:</p>

  <ol>
   <!-- fake </td> or </th> -->
   <li><p><span>Generate implied end tags</span>.</p></li>

   <li><p>If the <span>current node</span> is not now a <code>td</code> element or a <code>th</code>
   element, then this is a <span>parse error</span>.</p></li>

   <li><p>Pop elements from the <span>stack of open elements</span> until a <code>td</code>
   element or a <code>th</code> element has been popped from the stack.</p></li>

   <li><p><span>Clear the list of active formatting elements up to the last marker</span>.</p></li>

   <li><p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in row">in
   row</span>".</p></li> <!-- current node here will be a <tr> normally; but could be <html> in the
   fragment case, or <template> in the template case -->
   <!-- end of fake </td> or </th> -->
  </ol>

  <p class="note">The <span>stack of open elements</span> cannot have both a <code>td</code> and a
  <code>th</code> element <span data-x="has an element in table scope">in table scope</span> at the
  same time, nor can it have neither when the <span>close the cell</span> algorithm is invoked.</p>


  <h6 id="parsing-main-inselect">The "<dfn data-x="insertion mode: in select">in select</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: in select">in
  select</span>" <span>insertion mode</span>, the user agent must handle the token as follows:</p>

  <dl class="switch">

   <dt>A character token that is U+0000 NULL</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>Any other character token</dt>
   <dd>
    <p><span data-x="insert a character">Insert the token's character</span>.</p>
   </dd>

   <dt>A comment token</dt>
   <dd>
    <p><span>Insert a comment</span>.</p>
   </dd>

   <dt>A DOCTYPE token</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>A start tag whose tag name is "html"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>A start tag whose tag name is "option"</dt>
   <dd>
    <!-- fake </option> (maybe) -->
    <p>If the <span>current node</span> is an <code>option</code> element, pop that node from the
    <span>stack of open elements</span>.</p>
    <!-- end of fake </option> -->

    <p><span>Insert an HTML element</span> for the token.</p>
   </dd>

   <dt>A start tag whose tag name is "optgroup"</dt>
   <dd>
    <!-- fake </option> (maybe) -->
    <p>If the <span>current node</span> is an <code>option</code> element, pop that node from the
    <span>stack of open elements</span>.</p>
    <!-- end of fake </option> -->

    <!-- fake </optgroup> (maybe) -->
    <p>If the <span>current node</span> is an <code>optgroup</code> element, pop that node from the
    <span>stack of open elements</span>.</p>
    <!-- end of fake </optgroup> -->

    <p><span>Insert an HTML element</span> for the token.</p>
   </dd>

   <dt>A start tag whose tag name is "hr"</dt>
   <dd>
    <!-- fake </option> (maybe) -->
    <p>If the <span>current node</span> is an <code>option</code> element, pop that node from the
    <span>stack of open elements</span>.</p>
    <!-- end of fake </option> -->

    <!-- fake </optgroup> (maybe) -->
    <p>If the <span>current node</span> is an <code>optgroup</code> element, pop that node from the
    <span>stack of open elements</span>.</p>
    <!-- end of fake </optgroup> -->

    <p><span>Insert an HTML element</span> for the token. Immediately pop the <span>current
    node</span> off the <span>stack of open elements</span>.</p>

    <p><span data-x="acknowledge self-closing flag">Acknowledge the token's <i data-x="self-closing
    flag">self-closing flag</i></span>, if it is set.</p>
   </dd>

   <dt>An end tag whose tag name is "optgroup"</dt>
   <dd>
    <!-- fake </option> (maybe) -->
    <p>First, if the <span>current node</span> is an <code>option</code> element, and the node
    immediately before it in the <span>stack of open elements</span> is an <code>optgroup</code>
    element, then pop the <span>current node</span> from the <span>stack of open
    elements</span>.</p>
    <!-- end of fake </option> -->

    <p>If the <span>current node</span> is an <code>optgroup</code> element, then pop that node from
    the <span>stack of open elements</span>. Otherwise, this is a <span>parse error</span>; ignore
    the token.</p>
   </dd>

   <dt>An end tag whose tag name is "option"</dt>
   <dd>
    <p>If the <span>current node</span> is an <code>option</code> element, then pop that node from
    the <span>stack of open elements</span>. Otherwise, this is a <span>parse error</span>; ignore
    the token.</p>
   </dd>

   <dt>An end tag whose tag name is "select"</dt>
   <dd>
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in select
    scope">have a <code>select</code> element in select scope</span>, this is a <span>parse
    error</span>; ignore the token. (<span>fragment case</span>)</p>

    <p>Otherwise:</p>

    <p>Pop elements from the <span>stack of open elements</span> until a <code>select</code> element
    has been popped from the stack.</p>

    <p><span>Reset the insertion mode appropriately</span>.</p>
   </dd>

   <dt>A start tag whose tag name is "select"</dt>
   <dd>
    <p><span>Parse error</span>.</p>

    <!-- fake </select> -->
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in select
    scope">have a <code>select</code> element in select scope</span>, ignore the token.
    (<span>fragment case</span>)</p>

    <p>Otherwise:</p>

    <p>Pop elements from the <span>stack of open elements</span> until a <code>select</code> element
    has been popped from the stack.</p>

    <p><span>Reset the insertion mode appropriately</span>.</p>
    <!-- end of fake </select> -->

    <p class="note">It just gets treated like an end tag.</p>
   </dd>

   <dt>A start tag whose tag name is one of: "input", "keygen", "textarea"</dt>
   <dd>
    <p><span>Parse error</span>.</p>

    <!-- fake </select> -->
    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in select
    scope">have a <code>select</code> element in select scope</span>, ignore the token.
    (<span>fragment case</span>)</p>

    <p>Otherwise:</p>

    <p>Pop elements from the <span>stack of open elements</span> until a <code>select</code> element
    has been popped from the stack.</p>

    <p><span>Reset the insertion mode appropriately</span>.</p>
    <!-- end of fake </select> -->

    <p>Reprocess the token.</p>
   </dd>

   <dt>A start tag whose tag name is one of: "script", "template"</dt>
   <dt>An end tag whose tag name is "template"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    head">in head</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>An end-of-file token</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>
  </dl>


  <h6 id="parsing-main-inselectintable">The "<dfn data-x="insertion mode: in select in table">in select in table</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: in select in
  table">in select in table</span>" <span>insertion mode</span>, the user agent must handle the
  token as follows:</p>

  <dl class="switch">

   <dt>A start tag whose tag name is one of: "caption", "table", "tbody", "tfoot", "thead", "tr",
   "td", "th"</dt>
   <dd>
    <p><span>Parse error</span>.</p>

    <!-- fake </select> -->
    <p>Pop elements from the <span>stack of open elements</span> until a <code>select</code> element
    has been popped from the stack.</p>

    <p><span>Reset the insertion mode appropriately</span>.</p>
    <!-- end of fake </select> -->

    <p>Reprocess the token.</p>
   </dd>

   <dt>An end tag whose tag name is one of: "caption", "table", "tbody", "tfoot", "thead", "tr",
   "td", "th"</dt>
   <dd>
    <p><span>Parse error</span>.</p>

    <p>If the <span>stack of open elements</span> does not <span data-x="has an element in table
    scope">have an element in table scope</span> that is an <span data-x="HTML elements">HTML
    element</span> with the same tag name as that of the token, then ignore the token.</p>

    <p>Otherwise:</p>

    <!-- fake </select> -->
    <p>Pop elements from the <span>stack of open elements</span> until a <code>select</code> element
    has been popped from the stack.</p>

    <p><span>Reset the insertion mode appropriately</span>.</p>
    <!-- end of fake </select> -->

    <p>Reprocess the token.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    select">in select</span>" <span>insertion mode</span>.</p>
   </dd>
  </dl>



  <h6 id="parsing-main-intemplate">The "<dfn data-x="insertion mode: in template">in template</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: in template">in
  template</span>" <span>insertion mode</span>, the user agent must handle the token as follows:</p>

  <dl class="switch">

   <!-- first, tokens we always ignore: -->

   <!-- html: ignored in "in body" mode if there's a <template> on the stack -->
   <!-- head: ignored in "in body" mode always -->
   <!-- body: ignored in "in body" mode if there's a <template> on the stack -->
   <!-- frameset: ignored in "in body" mode if frameset-ok is set to not-ok, which <template> sets it to -->


   <!-- second, tokens that are ambiguous (allowed in multiple modes), let's just handle them in a generic way and not pick a mode: -->

   <dt>A character token</dt>
   <dt>A comment token</dt>
   <dt>A DOCTYPE token</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>A start tag whose tag name is one of: "base", "basefont", "bgsound", "link", "meta", "noframes", "script", "style", "template", "title"</dt>
   <dt>An end tag whose tag name is "template"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    head">in head</span>" <span>insertion mode</span>.</p>
   </dd>

   <!-- now, tokens that imply certain modes -->

   <dt>A start tag whose tag name is one of: "caption", "colgroup", "tbody", "tfoot", "thead"</dt>
   <dd>
    <p>Pop the <span>current template insertion mode</span> off the <span>stack of template
    insertion modes</span>.</p>

    <p>Push "<span data-x="insertion mode: in table">in table</span>" onto the <span>stack of
    template insertion modes</span> so that it is the new <span>current template insertion
    mode</span>.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in table">in
    table</span>", and reprocess the token.</p>
   </dd>

   <dt>A start tag whose tag name is "col"</dt>
   <dd>
    <p>Pop the <span>current template insertion mode</span> off the <span>stack of template
    insertion modes</span>.</p>

    <p>Push "<span data-x="insertion mode: in column group">in column group</span>" onto the
    <span>stack of template insertion modes</span> so that it is the new <span>current template
    insertion mode</span>.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in column group">in
    column group</span>", and reprocess the token.</p>
   </dd>

   <dt>A start tag whose tag name is "tr"</dt>
   <dd>
    <p>Pop the <span>current template insertion mode</span> off the <span>stack of template
    insertion modes</span>.</p>

    <p>Push "<span data-x="insertion mode: in table body">in table body</span>" onto the <span>stack
    of template insertion modes</span> so that it is the new <span>current template insertion
    mode</span>.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in table body">in
    table body</span>", and reprocess the token.</p>
   </dd>

   <dt>A start tag whose tag name is one of: "td", "th"</dt>
   <dd>
    <p>Pop the <span>current template insertion mode</span> off the <span>stack of template
    insertion modes</span>.</p>

    <p>Push "<span data-x="insertion mode: in row">in row</span>" onto the <span>stack of template
    insertion modes</span> so that it is the new <span>current template insertion mode</span>.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in row">in
    row</span>", and reprocess the token.</p>
   </dd>

   <!-- default to in-body mode -->

   <dt>Any other start tag</dt>
   <dd>
    <p>Pop the <span>current template insertion mode</span> off the <span>stack of template
    insertion modes</span>.</p>

    <p>Push "<span data-x="insertion mode: in body">in body</span>" onto the <span>stack of template
    insertion modes</span> so that it is the new <span>current template insertion mode</span>.</p>

    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: in body">in
    body</span>", and reprocess the token.</p>
   </dd>

   <dt>Any other end tag</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <!-- EOF -->


   <dt>An end-of-file token</dt>
   <dd>
    <p>If there is no <code>template</code> element on the <span>stack of open elements</span>, then
    <span>stop parsing</span>. (<span>fragment case</span>)</p>

    <p>Otherwise, this is a <span>parse error</span>.</p>

    <!-- fake </template> -->
    <p>Pop elements from the <span>stack of open elements</span> until a <code>template</code>
    element has been popped from the stack.</p>

    <p><span>Clear the list of active formatting elements up to the last marker</span>.</p>

    <p>Pop the <span>current template insertion mode</span> off the <span>stack of template
    insertion modes</span>.</p>

    <p><span>Reset the insertion mode appropriately</span>.</p>
    <!-- end of fake </template> -->

    <p>Reprocess the token.</p>
   </dd>
  </dl>


  <h6 id="parsing-main-afterbody">The "<dfn data-x="insertion mode: after body">after body</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: after body">after body</span>" <span>insertion mode</span>, the user agent must handle the token as follows:</p>

  <dl class="switch">

   <dt>A character token that is one of U+0009 CHARACTER TABULATION, U+000A LINE FEED (LF), U+000C
   FORM FEED (FF), U+000D CARRIAGE RETURN (CR), or U+0020 SPACE</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>A comment token</dt>
   <dd>
    <p><span>Insert a comment</span> as the last child of the first element in the <span>stack of
    open elements</span> (the <code>html</code> element).</p>
   </dd>

   <dt>A DOCTYPE token</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>A start tag whose tag name is "html"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>An end tag whose tag name is "html"</dt>
   <dd>
    <p>If the parser was created as part of the <span>HTML fragment parsing algorithm</span>, this
    is a <span>parse error</span>; ignore the token. (<span>fragment case</span>)</p>

    <p>Otherwise, switch the <span>insertion mode</span> to "<span data-x="insertion mode: after
    after body">after after body</span>".</p>
   </dd>

   <dt>An end-of-file token</dt>
   <dd>
    <p><span>Stop parsing</span>.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <p><span>Parse error</span>. Switch the <span>insertion mode</span> to "<span data-x="insertion
    mode: in body">in body</span>" and reprocess the token.</p>
   </dd>
  </dl>


  <h6 id="parsing-main-inframeset">The "<dfn data-x="insertion mode: in frameset">in frameset</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: in frameset">in
  frameset</span>" <span>insertion mode</span>, the user agent must handle the token as follows:</p>

  <dl class="switch">

   <dt>A character token that is one of U+0009 CHARACTER TABULATION, U+000A LINE FEED (LF), U+000C
   FORM FEED (FF), U+000D CARRIAGE RETURN (CR), or U+0020 SPACE</dt>
   <dd>
    <p><span data-x="insert a character">Insert the character</span>.</p>
   </dd>

   <dt>A comment token</dt>
   <dd>
    <p><span>Insert a comment</span>.</p>
   </dd>

   <dt>A DOCTYPE token</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>A start tag whose tag name is "html"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>A start tag whose tag name is "frameset"</dt>
   <dd>
    <p><span>Insert an HTML element</span> for the token.</p>
   </dd>

   <dt>An end tag whose tag name is "frameset"</dt>
   <dd>
    <p>If the <span>current node</span> is the root <code>html</code> element, then this is a
    <span>parse error</span>; ignore the token. (<span>fragment case</span>)</p>

    <p>Otherwise, pop the <span>current node</span> from the <span>stack of open
    elements</span>.</p>

    <p>If the parser was not created as part of the <span>HTML fragment parsing algorithm</span>
    (<span>fragment case</span>), and the <span>current node</span> is no longer a
    <code>frameset</code> element, then switch the <span>insertion mode</span> to "<span
    data-x="insertion mode: after frameset">after frameset</span>".</p>
   </dd>

   <dt>A start tag whose tag name is "frame"</dt>
   <dd>
    <p><span>Insert an HTML element</span> for the token. Immediately pop the <span>current
    node</span> off the <span>stack of open elements</span>.</p>

    <p><span data-x="acknowledge self-closing flag">Acknowledge the token's <i data-x="self-closing flag">self-closing
    flag</i></span>, if it is set.</p>
   </dd>

   <dt>A start tag whose tag name is "noframes"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    head">in head</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>An end-of-file token</dt>
   <dd>
    <p>If the <span>current node</span> is not the root <code>html</code> element, then this is a
    <span>parse error</span>.</p>

    <p class="note">The <span>current node</span> can only be the root
    <code>html</code> element in the <span>fragment case</span>.</p>

    <p><span>Stop parsing</span>.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>
  </dl>


  <h6 id="parsing-main-afterframeset">The "<dfn data-x="insertion mode: after frameset">after frameset</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: after
  frameset">after frameset</span>" <span>insertion mode</span>, the user agent must handle the token
  as follows:</p>

  <!-- due to rules in the "in frameset" mode, this can't be entered in the fragment case -->
  <dl class="switch">

   <dt>A character token that is one of U+0009 CHARACTER TABULATION, U+000A LINE FEED (LF), U+000C
   FORM FEED (FF), U+000D CARRIAGE RETURN (CR), or U+0020 SPACE</dt>
   <dd>
    <p><span data-x="insert a character">Insert the character</span>.</p>
   </dd>

   <dt>A comment token</dt>
   <dd>
    <p><span>Insert a comment</span>.</p>
   </dd>

   <dt>A DOCTYPE token</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>A start tag whose tag name is "html"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>An end tag whose tag name is "html"</dt>
   <dd>
    <p>Switch the <span>insertion mode</span> to "<span data-x="insertion mode: after after
    frameset">after after frameset</span>".</p>
   </dd>

   <dt>A start tag whose tag name is "noframes"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    head">in head</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>An end-of-file token</dt>
   <dd>
    <p><span>Stop parsing</span>.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>
  </dl>


  <h6>The "<dfn data-x="insertion mode: after after body">after after body</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: after after
  body">after after body</span>" <span>insertion mode</span>, the user agent must handle the token
  as follows:</p>

  <dl class="switch">

   <dt>A comment token</dt>
   <dd>
    <p><span>Insert a comment</span> as the last child of the <code>Document</code> object.</p>
   </dd>

   <dt>A DOCTYPE token</dt>
   <dt>A character token that is one of U+0009 CHARACTER TABULATION, U+000A LINE FEED (LF), U+000C
   FORM FEED (FF), U+000D CARRIAGE RETURN (CR), or U+0020 SPACE</dt>
   <dt>A start tag whose tag name is "html"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>An end-of-file token</dt>
   <dd>
    <p><span>Stop parsing</span>.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <p><span>Parse error</span>. Switch the <span>insertion mode</span> to "<span data-x="insertion
    mode: in body">in body</span>" and reprocess the token.</p>
   </dd>
  </dl>


  <h6>The "<dfn data-x="insertion mode: after after frameset">after after frameset</dfn>" insertion mode</h6>

  <p>When the user agent is to apply the rules for the "<span data-x="insertion mode: after after
  frameset">after after frameset</span>" <span>insertion mode</span>, the user agent must handle the
  token as follows:</p>

  <dl class="switch">

   <dt>A comment token</dt>
   <dd>
    <p><span>Insert a comment</span> as the last child of the <code>Document</code> object.</p>
   </dd>

   <dt>A DOCTYPE token</dt>
   <dt>A character token that is one of U+0009 CHARACTER TABULATION, U+000A LINE FEED (LF), U+000C
   FORM FEED (FF), U+000D CARRIAGE RETURN (CR), or U+0020 SPACE</dt>
   <dt>A start tag whose tag name is "html"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    body">in body</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>An end-of-file token</dt>
   <dd>
    <p><span>Stop parsing</span>.</p>
   </dd>

   <dt>A start tag whose tag name is "noframes"</dt>
   <dd>
    <p>Process the token <span>using the rules for</span> the "<span data-x="insertion mode: in
    head">in head</span>" <span>insertion mode</span>.</p>
   </dd>

   <dt>Anything else</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>
  </dl>



  <h5 id="parsing-main-inforeign">The rules for parsing tokens <dfn data-x="insertion mode: in foreign content">in foreign content</dfn></h5>

  <p>When the user agent is to apply the rules for parsing tokens in foreign content, the user agent
  must handle the token as follows:</p>

  <dl class="switch">

   <dt>A character token that is U+0000 NULL</dt>
   <dd>
    <p><span>Parse error</span>. <span data-x="insert a character">Insert a U+FFFD REPLACEMENT
    CHARACTER character</span>.</p>
   </dd>

   <dt>A character token that is one of U+0009 CHARACTER TABULATION, U+000A LINE FEED (LF), U+000C
   FORM FEED (FF), U+000D CARRIAGE RETURN (CR), or U+0020 SPACE</dt>
   <dd>
    <p><span data-x="insert a character">Insert the token's character</span>.</p>
   </dd>

   <dt>Any other character token</dt>
   <dd>
    <p><span data-x="insert a character">Insert the token's character</span>.</p>

    <p>Set the <span>frameset-ok flag</span> to "not ok".</p>
   </dd>

   <dt>A comment token</dt>
   <dd>
    <p><span>Insert a comment</span>.</p>
   </dd>

   <dt>A DOCTYPE token</dt>
   <dd>
    <p><span>Parse error</span>. Ignore the token.</p>
   </dd>

   <dt>A start tag whose tag name is one of: <!--"a",--> "b", "big", "blockquote", "body"<!--by
   inspection-->, "br", "center", "code", "dd", "div", "dl", "dt"<!-- so that dd and dt can be
   handled uniformly throughout the parser -->, "em", "embed", "h1", "h2", "h3", "h4"<!--for
   completeness-->, "h5", "h6"<!--for completeness-->, "head"<!--by inspection-->, "hr", "i", "img",
   "li", "listing"<!-- so that pre and listing can be handled uniformly throughout the parser -->,
   "menu", "meta", "nobr", "ol"<!-- so that dl, ul, and ol can be handled uniformly throughout the
   parser -->, "p", "pre", "ruby", "s", <!--"script",--> "small", "span", "strong", "strike"<!-- so
   that s and strike can be handled uniformly throughout the parser -->, <!--"style",--> "sub",
   "sup", "table"<!--by inspection-->, "tt", "u", "ul", "var"</dt> <!-- this list was determined
   empirically by studying over 6,000,000,000 pages that were specifically not XML pages -->
   <dt>A start tag whose tag name is "font", if the token has any attributes named "color", "face",
   or "size"</dt> <!-- the attributes here are required so that SVG <font> will go through as SVG
   but legacy <font>s won't -->
   <dt>An end tag whose tag name is "br", "p"</dt>

   <dd>
    <p><span>Parse error</span>.</p>

    <p>While the <span>current node</span> is not a <span>MathML text integration point</span>, an
    <span>HTML integration point</span>, or an element in the <span>HTML namespace</span>, pop
    elements from the <span>stack of open elements</span>.</p>

    <p>Reprocess the token according to the rules given in the section corresponding to the current
    <span>insertion mode</span> in HTML content.</p>
   </dd>

   <dt>Any other start tag</dt>
   <dd>
    <p>If the <span>adjusted current node</span> is an element in the <span>MathML namespace</span>,
    <span>adjust MathML attributes</span> for the token. (This fixes the case of MathML attributes
    that are not all lowercase.)</p>

    <p>If the <span>adjusted current node</span> is an element in the <span>SVG namespace</span>, and the
    token's tag name is one of the ones in the first column of the following table, change the tag
    name to the name given in the corresponding cell in the second column. (This fixes the case of
    SVG elements that are not all lowercase.)</p>

    <table>
     <thead>
      <tr> <th> Tag name <th> Element name
     <tbody>
      <tr> <td> <code data-x="">altglyph</code> <td> <code data-x="">altGlyph</code>
      <tr> <td> <code data-x="">altglyphdef</code> <td> <code data-x="">altGlyphDef</code>
      <tr> <td> <code data-x="">altglyphitem</code> <td> <code data-x="">altGlyphItem</code>
      <tr> <td> <code data-x="">animatecolor</code> <td> <code data-x="">animateColor</code>
      <tr> <td> <code data-x="">animatemotion</code> <td> <code data-x="">animateMotion</code>
      <tr> <td> <code data-x="">animatetransform</code> <td> <code data-x="">animateTransform</code>
      <tr> <td> <code data-x="">clippath</code> <td> <code data-x="">clipPath</code>
      <tr> <td> <code data-x="">feblend</code> <td> <code data-x="">feBlend</code>
      <tr> <td> <code data-x="">fecolormatrix</code> <td> <code data-x="">feColorMatrix</code>
      <tr> <td> <code data-x="">fecomponenttransfer</code> <td> <code data-x="">feComponentTransfer</code>
      <tr> <td> <code data-x="">fecomposite</code> <td> <code data-x="">feComposite</code>
      <tr> <td> <code data-x="">feconvolvematrix</code> <td> <code data-x="">feConvolveMatrix</code>
      <tr> <td> <code data-x="">fediffuselighting</code> <td> <code data-x="">feDiffuseLighting</code>
      <tr> <td> <code data-x="">fedisplacementmap</code> <td> <code data-x="">feDisplacementMap</code>
      <tr> <td> <code data-x="">fedistantlight</code> <td> <code data-x="">feDistantLight</code>
      <tr> <td> <code data-x="">fedropshadow</code> <td> <code data-x="">feDropShadow</code>
      <tr> <td> <code data-x="">feflood</code> <td> <code data-x="">feFlood</code>
      <tr> <td> <code data-x="">fefunca</code> <td> <code data-x="">feFuncA</code>
      <tr> <td> <code data-x="">fefuncb</code> <td> <code data-x="">feFuncB</code>
      <tr> <td> <code data-x="">fefuncg</code> <td> <code data-x="">feFuncG</code>
      <tr> <td> <code data-x="">fefuncr</code> <td> <code data-x="">feFuncR</code>
      <tr> <td> <code data-x="">fegaussianblur</code> <td> <code data-x="">feGaussianBlur</code>
      <tr> <td> <code data-x="">feimage</code> <td> <code data-x="">feImage</code>
      <tr> <td> <code data-x="">femerge</code> <td> <code data-x="">feMerge</code>
      <tr> <td> <code data-x="">femergenode</code> <td> <code data-x="">feMergeNode</code>
      <tr> <td> <code data-x="">femorphology</code> <td> <code data-x="">feMorphology</code>
      <tr> <td> <code data-x="">feoffset</code> <td> <code data-x="">feOffset</code>
      <tr> <td> <code data-x="">fepointlight</code> <td> <code data-x="">fePointLight</code>
      <tr> <td> <code data-x="">fespecularlighting</code> <td> <code data-x="">feSpecularLighting</code>
      <tr> <td> <code data-x="">fespotlight</code> <td> <code data-x="">feSpotLight</code>
      <tr> <td> <code data-x="">fetile</code> <td> <code data-x="">feTile</code>
      <tr> <td> <code data-x="">feturbulence</code> <td> <code data-x="">feTurbulence</code>
      <tr> <td> <code data-x="">foreignobject</code> <td> <code data-x="">foreignObject</code>
      <tr> <td> <code data-x="">glyphref</code> <td> <code data-x="">glyphRef</code>
      <tr> <td> <code data-x="">lineargradient</code> <td> <code data-x="">linearGradient</code>
      <tr> <td> <code data-x="">radialgradient</code> <td> <code data-x="">radialGradient</code>
      <!--<tr> <td> <code data-x="">solidcolor</code> <td> <code data-x="">solidColor</code> (SVG 1.2)-->
      <tr> <td> <code data-x="">textpath</code> <td> <code data-x="">textPath</code>
    </table>

    <p>If the <span>adjusted current node</span> is an element in the <span>SVG namespace</span>,
    <span>adjust SVG attributes</span> for the token. (This fixes the case of SVG attributes that
    are not all lowercase.)</p>

    <p><span>Adjust foreign attributes</span> for the token. (This fixes the use of namespaced
    attributes, in particular XLink in SVG.)</p>

    <p><span>Insert a foreign element</span> for the token, with <span>adjusted current
    node</span>'s namespace and false.</p>

    <p>If the token has its <i data-x="self-closing flag">self-closing flag</i> set, then run the appropriate steps from the
    following list:</p>

    <dl class="switch">

     <dt>If the token's tag name is "script", and the new <span>current node</span> is in the <span>SVG namespace</span></dt>

     <dd>
      <p><span data-x="acknowledge self-closing flag">Acknowledge the token's <i data-x="self-closing flag">self-closing
      flag</i></span>, and then act as described in the steps for a "script" end tag below.</p>
     </dd>

     <dt>Otherwise</dt>

     <dd>
      <p>Pop the <span>current node</span> off the <span>stack of open elements</span> and <span
      data-x="acknowledge self-closing flag">acknowledge the token's <i data-x="self-closing flag">self-closing
      flag</i></span>.</p>
     </dd>
    </dl>
   </dd>

   <dt id="scriptForeignEndTag">An end tag whose tag name is "script", if the <span>current
   node</span> is an <span>SVG <code>script</code></span> element</dt>
   <dd>
    <p>Pop the <span>current node</span> off the <span>stack of open elements</span>.</p>

    <p>Let the <var>old insertion point</var> have the same value as the current
    <span>insertion point</span>. Let the <span>insertion point</span> be just before the <span>next
    input character</span>.</p>

    <p>Increment the parser's <span>script nesting level</span> by one. Set the <span>parser pause
    flag</span> to true.</p>

    <p>If the <span>active speculative HTML parser</span> is null and the user agent supports SVG,
    then <a href="https://www.w3.org/TR/SVGMobile12/script.html#ScriptContentProcessing">Process the
    SVG <code data-x="">script</code> element</a> according to the SVG rules. <ref>SVG</ref></p>

    <p class="note">Even if this causes <span data-x="dom-document-write">new characters to be
    inserted into the tokenizer</span>, the parser will not be executed reentrantly, since the
    <span>parser pause flag</span> is true.</p>

    <p>Decrement the parser's <span>script nesting level</span> by one. If the parser's <span>script
    nesting level</span> is zero, then set the <span>parser pause flag</span> to false.</p>

    <p>Let the <span>insertion point</span> have the value of the <var>old insertion
    point</var>. (In other words, restore the <span>insertion point</span> to its previous value.
    This value might be the "undefined" value.)</p>
   </dd>

   <dt>Any other end tag</dt>

   <dd>
    <p>Run these steps:</p>

    <ol>
     <li><p>Initialize <var>node</var> to be the <span>current node</span> (the bottommost
     node of the stack).</p></li>

     <li><p>If <var>node</var>'s tag name, <span>converted to ASCII lowercase</span>, is
     not the same as the tag name of the token, then this is a <span>parse error</span>.</p></li>

     <li><p><i>Loop</i>: If <var>node</var> is the topmost element in the <span>stack of
     open elements</span>, then return. (<span>fragment case</span>)</p></li>

     <li><p>If <var>node</var>'s tag name, <span>converted to ASCII lowercase</span>, is
     the same as the tag name of the token, pop elements from the <span>stack of open
     elements</span> until <var>node</var> has been popped from the stack, and then return.</p></li>

     <li><p>Set <var>node</var> to the previous entry in the <span>stack of open
     elements</span>.</p></li>

     <li><p>If <var>node</var> is not an element in the <span>HTML namespace</span>, return
     to the step labeled <i>loop</i>.</p></li>

     <li><p>Otherwise, process the token according to the rules given in the section corresponding
     to the current <span>insertion mode</span> in HTML content.</li>
    </ol>
   </dd>
  </dl>

  </div>


  <div w-nodev>

  <h4>The end</h4>

  <p>Once the user agent <dfn data-x="stop parsing">stops parsing</dfn> the document, the user agent
  must run the following steps:</p>

  <ol>
   <!-- this happens as part of one of the tasks that runs the parser -->

   <li><p>If the <span>active speculative HTML parser</span> is not null, then <span>stop the
   speculative HTML parser</span> and return.</p></li>

   <li><p>Set the <span>insertion point</span> to undefined.</p></li>

   <li><p><span>Update the current document readiness</span> to "<code
   data-x="">interactive</code>".</p></li>

   <li><p>Pop <em>all</em> the nodes off the <span>stack of open elements</span>.</p></li>

   <li><p>While the <span>list of scripts that will execute when the document has finished
   parsing</span> is not empty:</p>

    <ol>
     <li><p><span>Spin the event loop</span> until the first <code>script</code> in the <span>list
     of scripts that will execute when the document has finished parsing</span> has its <span>ready
     to be parser-executed</span> set to true <em>and</em> the parser's <code>Document</code>
     <span>has no style sheet that is blocking scripts</span>.</p></li>

     <li><p><span>Execute the script element</span> given by the first <code>script</code> in
     the <span>list of scripts that will execute when the document has finished
     parsing</span>.</p></li>

     <li><p>Remove the first <code>script</code> element from the <span>list of scripts that will
     execute when the document has finished parsing</span> (i.e. shift out the first entry in the
     list).</p></li>
    </ol>
   </li>

   <li>
    <p><span>Queue a global task</span> on the <span>DOM manipulation task source</span> given the
    <code>Document</code>'s <span>relevant global object</span> to run the following substeps:</p>

    <ol>
     <li><p>Set the <code>Document</code>'s <span>load timing info</span>'s <span>DOM content loaded
     event start time</span> to the <span>current high resolution time</span> given the
     <code>Document</code>'s <span>relevant global object</span>.</p></li>

     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-DOMContentLoaded">DOMContentLoaded</code> at the <code>Document</code>
     object, with its <code data-x="dom-Event-bubbles">bubbles</code> attribute initialized to
     true.</p></li>

     <li><p>Set the <code>Document</code>'s <span>load timing info</span>'s <span>DOM content loaded
     event end time</span> to the <span>current high resolution time</span> given the
     <code>Document</code>'s <span>relevant global object</span>.</p></li>

     <li><p>Enable the <span data-x="dfn-client-message-queue">client message queue</span> of the
     <code>ServiceWorkerContainer</code> object whose associated <span
     data-x="serviceworkercontainer-service-worker-client">service worker client</span> is the
     <code>Document</code> object's <span>relevant settings object</span>.</p></li>

     <li><p>Invoke <span>WebDriver BiDi DOM content loaded</span> with the <code>Document</code>'s
     <span data-x="concept-document-bc">browsing context</span>, and a new <span>WebDriver BiDi
     navigation status</span> whose <span data-x="navigation-status-id">id</span> is the
     <code>Document</code> object's <span data-x="concept-document-navigation-id">during-loading
     navigation ID for WebDriver BiDi</span>, <span data-x="navigation-status-status">status</span>
     is "<code data-x="navigation-status-pending" >pending</code>", and <span
     data-x="navigation-status-url">url</span> is the <code>Document</code> object's <span
     data-x="concept-document-url">URL</span>.</p></li>
    </ol>
   </li>

   <li><p><span>Spin the event loop</span> until the <span>set of scripts that will execute as soon
   as possible</span> and the <span>list of scripts that will execute in order as soon as
   possible</span> are empty.</p></li> <!-- this step is not redundant with the next one, since
   <script> nodes delay the load event of the document they are in, but they might change document
   between being added to one document's set/list and executing those scripts, so they might be
   delaying another document but still be in this document's set/list. -->

   <li><p><span>Spin the event loop</span> until there is nothing that <dfn data-x="delay the load
   event">delays the load event</dfn> in the <code>Document</code>.</p></li>

   <li>
    <p><span>Queue a global task</span> on the <span>DOM manipulation task source</span> given the
    <code>Document</code>'s <span>relevant global object</span> to run the following steps:</p>

    <ol>
     <li><p><span>Update the current document readiness</span> to "<code
     data-x="">complete</code>".</p></li>

     <li><p>If the <code>Document</code> object's <span data-x="concept-document-bc">browsing
     context</span> is null, then abort these steps.</p></li>

     <li><p>Let <var>window</var> be the <code>Document</code>'s <span>relevant global
     object</span>.</p></li>

     <li><p>Set the <code>Document</code>'s <span>load timing info</span>'s <span>load event start
     time</span> to the <span>current high resolution time</span> given <var>window</var>.</p></li>

     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-load">load</code> at <var>window</var>, with <var>legacy target override
     flag</var> set.</p></li>

     <li><p>Invoke <span>WebDriver BiDi load complete</span> with the <code>Document</code>'s <span
     data-x="concept-document-bc">browsing context</span>, and a new <span>WebDriver BiDi
     navigation status</span> whose <span data-x="navigation-status-id">id</span> is the
     <code>Document</code> object's <span data-x="concept-document-navigation-id">during-loading
     navigation ID for WebDriver BiDi</span>, <span data-x="navigation-status-status">status</span>
     is "<code data-x="navigation-status-complete">complete</code>", and <span
     data-x="navigation-status-url">url</span> is the <code>Document</code> object's <span
     data-x="concept-document-url">URL</span>.</p></li>

     <li><p>Set the <code>Document</code> object's <span
     data-x="concept-document-navigation-id">during-loading navigation ID for WebDriver BiDi</span>
     to null.</p></li>

     <li><p>Set the <code>Document</code>'s <span>load timing info</span>'s <span>load event end
     time</span> to the <span>current high resolution time</span> given <var>window</var>.</p></li>

     <li><p><span>Assert</span>: <code>Document</code>'s <span>page showing</span> is
     false.</p></li>

     <li><p>Set the <code>Document</code>'s <span>page showing</span> to true.</p></li>

     <li><p><span>Fire a page transition event</span> named <code
     data-x="event-pageshow">pageshow</code> at <var>window</var> with false.</p></li>

     <li><p><span>Completely finish loading</span> the <code>Document</code>.</p></li>

     <li><p><span>Queue the navigation timing entry</span> for the <code>Document</code>.</p></li>
    </ol>
   </li>

   <li><p>If the <code>Document</code>'s <span>print when loaded</span> flag is set, then run the
   <span>printing steps</span>.</p></li>

   <li><p>The <code>Document</code> is now <dfn>ready for post-load tasks</dfn>.</p></li>
  </ol>

  <p>When the user agent is to <dfn>abort a parser</dfn>, it must run the following steps:</p>

  <ol>
   <li><p>Throw away any pending content in the <span>input stream</span>, and discard any future
   content that would have been added to it.</p></li>

   <li><p><span>Stop the speculative HTML parser</span> for this HTML parser.</p></li>

   <li><p><span>Update the current document readiness</span> to "<code
   data-x="">interactive</code>".</p></li>

   <li><p>Pop <em>all</em> the nodes off the <span>stack of open elements</span>.</p></li>

   <li><p><span>Update the current document readiness</span> to "<code
   data-x="">complete</code>".</p></li>

   <!-- anything else? this is things that happen when you call document.open() on a document that's
   still being parsed, or when you navigate a document that's still parsing, or navigate the parent
   of a frame with a document that's still parsing, or the user hits "stop". Should the pending
   scripts be blown away or anything? -->
  </ol>

  </div>


  <div w-nodev>

  <h4>Speculative HTML parsing</h4>

  <p>User agents may implement an optimization, as described in this section, to speculatively fetch
  resources that are declared in the HTML markup while the HTML parser is waiting for a
  <span>pending parsing-blocking script</span> to be fetched and executed, or during normal parsing,
  at the time <span data-x="create an element for the token">an element is created for a token</span>.
  While this optimization is not defined in precise detail, there are some rules to consider for
  interoperability.</p>

  <p>Each <span>HTML parser</span> can have an <dfn export>active speculative HTML parser</dfn>. It
  is initially null.</p>

  <p>The <dfn export>speculative HTML parser</dfn> must act like the normal HTML parser (e.g., the
  tree builder rules apply), with some exceptions:</p>

  <ul>
   <li>
    <p>The state of the normal HTML parser and the document itself must not be affected.</p>

    <p class="example">For example, the <span>next input character</span> or the <span>stack of open
    elements</span> for the normal HTML parser is not affected by the <span>speculative HTML
    parser</span>.</p>
   </li>

   <li>
    <p>Bytes pushed into the HTML parser's <span>input byte stream</span> must also be pushed into
    the speculative HTML parser's <span>input byte stream</span>. Bytes read from the streams must
    be independent.</p>
   </li>

   <li>
    <p>The result of the speculative parsing is primarily a series of <span data-x="speculative
    fetch">speculative fetches</span>. Which kinds of resources to speculatively fetch is
    <span>implementation-defined</span>, but user agents must not speculatively fetch resources that
    would not be fetched with the normal HTML parser, under the assumption that the script that is
    blocking the HTML parser does nothing.</p>

    <p class="note">It is possible that the same markup is seen multiple times from the
    <span>speculative HTML parser</span> and then the normal HTML parser. It is expected that
    duplicated fetches will be prevented by caching rules, which are not yet fully specified.</p>
   </li>
  </ul>

  <p>A <dfn>speculative fetch</dfn> for a <span>speculative mock element</span> <var>element</var>
  must follow these rules:</p>

  <p class="XXX">Should some of these things be applied to the document "for real", even
  though they are found speculatively?</p>

  <ul>
   <li>
    <p>If the <span>speculative HTML parser</span> encounters one of the following elements, then
    act as if that element is processed for the purpose of its effect of subsequent speculative
    fetches.</p>

    <ul class="brief">
     <li>A <code>base</code> element.</li>

     <li>A <code>meta</code> element whose <code data-x="attr-meta-http-equiv">http-equiv</code>
     attribute is in the <span data-x="attr-meta-http-equiv-content-security-policy">Content
     security policy</span> state.</li>

     <li>A <code>meta</code> element whose <code data-x="attr-meta-name">name</code> attribute is an
     <span>ASCII case-insensitive</span> match for "<code
     data-x="meta-referrer">referrer</code>".</li>

     <li>A <code>meta</code> element whose <code data-x="attr-meta-name">name</code> attribute is an
     <span>ASCII case-insensitive</span> match for "<code data-x="">viewport</code>". (This can
     affect whether a media query list <span>matches the environment</span>.) <ref>CSSDEVICEADAPT</ref></li>
    </ul>
   </li>

   <li><p>Let <var>url</var> be the <span>URL</span> that <var>element</var> would fetch if it was
   processed normally. If there is no such <span>URL</span> or if it is the empty string, then do
   nothing. Otherwise, if <span>url</span> is already in the <span>list of speculative fetch
   URLs</span>, then do nothing. Otherwise, fetch <span>url</span> as if the element was processed
   normally, and add <var>url</var> to the <span>list of speculative fetch URLs</span>.</p></li>
  </ul>

  <p>Each <code>Document</code> has a <dfn>list of speculative fetch URLs</dfn>, which is a
  <span>list</span> of <span data-x="URL">URLs</span>, initially empty.</p>

  <p>To <dfn>start the speculative HTML parser</dfn> for an instance of an HTML parser
  <var>parser</var>:</p>

  <ol>
   <li>
    <p>Optionally, return.</p>

    <p class="note">This step allows user agents to opt out of speculative HTML parsing.</p>
   </li>

   <li>
    <p>If <var>parser</var>'s <span>active speculative HTML parser</span> is not null, then
    <span>stop the speculative HTML parser</span> for <var>parser</var>.</p>

    <p class="note">This can happen when <code data-x="dom-document-write">document.write()</code>
    writes another parser-blocking script. For simplicity, this specification always restarts
    speculative parsing, but user agents can implement a more efficient strategy, so long as the end
    result is equivalent.</p>
   </li>

   <li><p>Let <var>speculativeParser</var> be a new <span>speculative HTML parser</span>, with the
   same state as <var>parser</var>.</p></li>

   <li><p>Let <var>speculativeDoc</var> be a new isomorphic representation of <var>parser</var>'s
   <code>Document</code>, where all elements are instead <span data-x="speculative mock
   element">speculative mock elements</span>. Let <var>speculativeParser</var> parse into
   <var>speculativeDoc</var>.</p></li>

   <li><p>Set <var>parser</var>'s <span>active speculative HTML parser</span> to
   <var>speculativeParser</var>.</p></li>

   <li><p><span>In parallel</span>, run <var>speculativeParser</var> until it is stopped or until it
   reaches the end of its <span>input stream</span>.</p></li>
  </ol>


  <p>To <dfn>stop the speculative HTML parser</dfn> for an instance of an HTML parser
  <var>parser</var>:</p>

  <ol>
   <li><p>Let <var>speculativeParser</var> be <var>parser</var>'s <span>active speculative HTML
   parser</span>.</p></li>

   <li><p>If <var>speculativeParser</var> is null, then return.</p></li>

   <li><p>Throw away any pending content in <var>speculativeParser</var>'s <span>input
   stream</span>, and discard any future content that would have been added to it.</p></li>

   <li><p>Set <var>parser</var>'s <span>active speculative HTML parser</span> to null.</p></li>
  </ol>

  <p>The <span>speculative HTML parser</span> will create <span
  data-x="speculative mock element">speculative mock elements</span> instead of normal elements. DOM
  operations that the tree builder normally does on elements are expected to work appropriately on
  speculative mock elements.</p>

  <p>A <dfn>speculative mock element</dfn> is a <span>struct</span> with the following <span
  data-x="struct item">items</span>:</p>

  <ul>
   <li><p>A <span>string</span> <dfn data-x="concept-mock-namespace">namespace</dfn>, corresponding
   to an element's <span data-x="concept-element-namespace">namespace</span>.</p></li>

   <li><p>A <span>string</span> <dfn data-x="concept-mock-local-name">local name</dfn>,
   corresponding to an element's <span data-x="concept-element-local-name">local
   name</span>.</p></li>

   <li><p>A <span>list</span> <dfn data-x="concept-mock-attribute-list">attribute list</dfn>,
   corresponding to an element's <span>attribute list</span>.</p></li>

   <li><p>A <span>list</span> <dfn data-x="concept-mock-children">children</dfn>, corresponding to
   an element's <span data-x="concept-tree-child">children</span>.</p></li>
  </ul>

  <p>To <dfn>create a speculative mock element</dfn> given a <var>namespace</var>,
  <var>tagName</var>, and <var>attributes</var>:</p>

  <ol>
   <li><p>Let <var>element</var> be a new <span>speculative mock element</span>.</p></li>

   <li><p>Set <var>element</var>'s <span data-x="concept-mock-namespace">namespace</span> to
   <var>namespace</var>.</p></li>

   <li><p>Set <var>element</var>'s <span data-x="concept-mock-local-name">local name</span> to
   <var>tagName</var>.</p></li>

   <li><p>Set <var>element</var>'s <span data-x="concept-mock-attribute-list">attribute list</span>
   to <var>attributes</var>.</p></li>

   <li><p>Set <var>element</var>'s <span data-x="concept-mock-children">children</span> to a new
   empty <span>list</span>.</p></li>

   <li><p>Optionally, perform a <span>speculative fetch</span> for <var>element</var>.</p></li>

   <li><p>Return <var>element</var>.</p></li>
  </ol>

  <p>When the tree builder says to insert an element into a <code>template</code> element's
  <span>template contents</span>, if that is a <span>speculative mock element</span>, and the
  <code>template</code> element's <span>template contents</span> is not a
  <code>ShadowRoot</code> node, instead do nothing. URLs found speculatively inside
  non-declarative-shadow-root <code>template</code> elements might themselves be templates, and must
  not be speculatively fetched.</p>

  </div>


  <div w-nodev>

  <h4>Coercing an HTML DOM into an infoset</h4>

  <p>When an application uses an <span>HTML parser</span> in conjunction with an XML pipeline, it is
  possible that the constructed DOM is not compatible with the XML tool chain in certain subtle
  ways. For example, an XML toolchain might not be able to represent attributes with the name <code
  data-x="">xmlns</code>, since they conflict with the Namespaces in XML syntax. There is also some
  data that the <span>HTML parser</span> generates that isn't included in the DOM itself. This
  section specifies some rules for handling these issues.</p>

  <p>If the XML API being used doesn't support DOCTYPEs, the tool may drop DOCTYPEs altogether.</p>

  <p>If the XML API doesn't support attributes in no namespace that are named "<code
  data-x="">xmlns</code>", attributes whose names start with "<code data-x="">xmlns:</code>", or
  attributes in the <span>XMLNS namespace</span>, then the tool may drop such attributes.</p>

  <p>The tool may annotate the output with any namespace declarations required for proper
  operation.</p>

  <p>If the XML API being used restricts the allowable characters in the local names of elements and
  attributes, then the tool may map all element and attribute local names that the API wouldn't
  support to a set of names that <em>are</em> allowed, by replacing any character that isn't
  supported with the uppercase letter U and the six digits of the character's code point when
  expressed in hexadecimal, using digits 0-9 and capital letters A-F as the symbols, in increasing
  numeric order.</p>

  <p class="example">For example, the element name <code data-x="">foo&lt;bar</code>, which can be
  output by the <span>HTML parser</span>, though it is neither a legal HTML element name nor a
  well-formed XML element name, would be converted into <code data-x="">fooU00003Cbar</code>, which
  <em>is</em> a well-formed XML element name (though it's still not legal in HTML by any means).</p>

  <p class="example">As another example, consider the attribute <code data-x="">xlink:href</code>.
  Used on a MathML element, it becomes, after being <span data-x="adjust foreign
  attributes">adjusted</span>, an attribute with a prefix "<code data-x="">xlink</code>" and a local
  name "<code data-x="">href</code>". However, used on an HTML element, it becomes an attribute with
  no prefix and the local name "<code data-x="">xlink:href</code>", which is not a valid NCName, and
  thus might not be accepted by an XML API. It could thus get converted, becoming "<code
  data-x="">xlinkU00003Ahref</code>".</p>

  <p class="note">The resulting names from this conversion conveniently can't clash with any
  attribute generated by the <span>HTML parser</span>, since those are all either lowercase or those
  listed in the <span>adjust foreign attributes</span> algorithm's table.</p>

  <p>If the XML API restricts comments from having two consecutive U+002D HYPHEN-MINUS characters
  (--), the tool may insert a single U+0020 SPACE character between any such offending
  characters.</p>

  <p>If the XML API restricts comments from ending in a U+002D HYPHEN-MINUS character (-), the tool
  may insert a single U+0020 SPACE character at the end of such comments.</p>

  <p>If the XML API restricts allowed characters in character data, attribute values, or comments,
  the tool may replace any U+000C FORM FEED (FF) character with a U+0020 SPACE character, and any
  other literal non-XML character with a U+FFFD REPLACEMENT CHARACTER.</p>

  <p>If the tool has no way to convey out-of-band information, then the tool may drop the following
  information:</p>

  <ul>
   <li>Whether the document is set to <i data-x="no-quirks mode">no-quirks mode</i>, <i data-x="limited-quirks mode">limited-quirks mode</i>, or
   <i data-x="quirks mode">quirks mode</i></li>

   <li>The association between form controls and forms that aren't their nearest <code>form</code>
   element ancestor (use of the <span><code>form</code> element pointer</span> in the parser)</li>

   <li>The <span>template contents</span> of any <code>template</code> elements.</li>
  </ul>

  <p class="note">The mutations allowed by this section apply <em>after</em> the <span>HTML
  parser</span>'s rules have been applied. For example, a <code data-x="">&lt;a::></code> start tag
  will be closed by a <code data-x="">&lt;/a::></code> end tag, and never by a <code
  data-x="">&lt;/aU00003AU00003A></code> end tag, even if the user agent is using the rules above to
  then generate an actual element in the DOM with the name <code data-x="">aU00003AU00003A</code> for
  that start tag.</p>

  </div>


  <div w-nodev>

  <h4>An introduction to error handling and strange cases in the parser</h4>

  <!-- NON-NORMATIVE SECTION -->

  <p>This section examines some erroneous markup and discusses how the <span>HTML parser</span>
  handles these cases.</p>


  <h5>Misnested tags: &lt;b>&lt;i>&lt;/b>&lt;/i></h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>The most-often discussed example of erroneous markup is as follows:</p>

  <pre><code class="html">&lt;p>1&lt;b>2&lt;i>3&lt;/b>4&lt;/i>5&lt;/p></code></pre>

  <p>The parsing of this markup is straightforward up to the "3". At this point, the DOM looks like
  this:</p>

  <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>p</code><ul><li class="t3"><code>#text</code>: <span data-x="">1</span></li><li class="t1"><code>b</code><ul><li class="t3"><code>#text</code>: <span data-x="">2</span></li><li class="t1"><code>i</code><ul><li class="t3"><code>#text</code>: <span data-x="">3</span></li></ul></li></ul></li></ul></li></ul></li></ul></li></ul>

  <p>Here, the <span>stack of open elements</span> has five elements on it: <code>html</code>,
  <code>body</code>, <code>p</code>, <code>b</code>, and <code>i</code>. The <span>list of active
  formatting elements</span> just has two: <code>b</code> and <code>i</code>. The <span>insertion
  mode</span> is "<span data-x="insertion mode: in body">in body</span>".</p>

  <p>Upon receiving the end tag token with the tag name "b", the "<a href="#adoptionAgency">adoption
  agency algorithm</a>" is invoked. This is a simple case, in that the <var>formattingElement</var>
  is the <code>b</code> element, and there is no <var>furthest block</var>.
  Thus, the <span>stack of open elements</span> ends up with just three elements: <code>html</code>,
  <code>body</code>, and <code>p</code>, while the <span>list of active formatting elements</span>
  has just one: <code>i</code>. The DOM tree is unmodified at this point.</p>

  <p>The next token is a character ("4"), triggers the <span data-x="reconstruct the active
  formatting elements">reconstruction of the active formatting elements</span>, in this case just
  the <code>i</code> element. A new <code>i</code> element is thus created for the "4"
  <code>Text</code> node. After the end tag token for the "i" is also received, and the "5"
  <code>Text</code> node is inserted, the DOM looks as follows:</p>

  <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>p</code><ul><li class="t3"><code>#text</code>: <span data-x="">1</span></li><li class="t1"><code>b</code><ul><li class="t3"><code>#text</code>: <span data-x="">2</span></li><li class="t1"><code>i</code><ul><li class="t3"><code>#text</code>: <span data-x="">3</span></li></ul></li></ul></li><li class="t1"><code>i</code><ul><li class="t3"><code>#text</code>: <span data-x="">4</span></li></ul></li><li class="t3"><code>#text</code>: <span data-x="">5</span></li></ul></li></ul></li></ul></li></ul>


  <h5>Misnested tags: &lt;b>&lt;p>&lt;/b>&lt;/p></h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>A case similar to the previous one is the following:</p>

  <pre><code class="html">&lt;b>1&lt;p>2&lt;/b>3&lt;/p></code></pre>

  <p>Up to the "2" the parsing here is straightforward:</p>

  <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>b</code><ul><li class="t3"><code>#text</code>: <span data-x="">1</span></li><li class="t1"><code>p</code><ul><li class="t3"><code>#text</code>: <span data-x="">2</span></li></ul></li></ul></li></ul></li></ul></li></ul>

  <p>The interesting part is when the end tag token with the tag name "b" is parsed.</p>

  <p>Before that token is seen, the <span>stack of open elements</span> has four elements on it:
  <code>html</code>, <code>body</code>, <code>b</code>, and <code>p</code>. The <span>list of active
  formatting elements</span> just has the one: <code>b</code>. The <span>insertion mode</span> is
  "<span data-x="insertion mode: in body">in body</span>".</p>

  <p>Upon receiving the end tag token with the tag name "b", the "<a href="#adoptionAgency">adoption
  agency algorithm</a>" is invoked, as in the previous example. However, in this case, there
  <em>is</em> a <var>furthest block</var>, namely the <code>p</code> element. Thus, this
  time the adoption agency algorithm isn't skipped over.</p>

  <p>The <var>common ancestor</var> is the <code>body</code> element. A conceptual
  "bookmark" marks the position of the <code>b</code> in the <span>list of active formatting
  elements</span>, but since that list has only one element in it, the bookmark won't have much
  effect.</p>

  <p>As the algorithm progresses, <var>node</var> ends up set to the formatting element
  (<code>b</code>), and <var>last node</var> ends up set to the <var>furthest
  block</var> (<code>p</code>).</p>

  <p>The <var>last node</var> gets appended (moved) to the <var>common
  ancestor</var>, so that the DOM looks like:</p>

  <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>b</code><ul><li class="t3"><code>#text</code>: <span data-x="">1</span></li></ul></li><li class="t1"><code>p</code><ul><li class="t3"><code>#text</code>: <span data-x="">2</span></li></ul></li></ul></li></ul></li></ul>

  <p>A new <code>b</code> element is created, and the children of the <code>p</code> element are
  moved to it:</p>

  <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>b</code><ul><li class="t3"><code>#text</code>: <span data-x="">1</span></li></ul></li><li class="t1"><code>p</code></li></ul></li></ul></li></ul>
  <ul class="domTree"><li class="t1"><code>b</code><ul><li class="t3"><code>#text</code>: <span data-x="">2</span></li></ul></li></ul>

  <p>Finally, the new <code>b</code> element is appended to the <code>p</code> element, so that the
  DOM looks like:</p>

  <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>b</code><ul><li class="t3"><code>#text</code>: <span data-x="">1</span></li></ul></li><li class="t1"><code>p</code><ul><li class="t1"><code>b</code><ul><li class="t3"><code>#text</code>: <span data-x="">2</span></li></ul></li></ul></li></ul></li></ul></li></ul>

  <p>The <code>b</code> element is removed from the <span>list of active formatting elements</span>
  and the <span>stack of open elements</span>, so that when the "3" is parsed, it is appended to the
  <code>p</code> element:</p>

  <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>b</code><ul><li class="t3"><code>#text</code>: <span data-x="">1</span></li></ul></li><li class="t1"><code>p</code><ul><li class="t1"><code>b</code><ul><li class="t3"><code>#text</code>: <span data-x="">2</span></li></ul></li><li class="t3"><code>#text</code>: <span data-x="">3</span></li></ul></li></ul></li></ul></li></ul>


  <h5>Unexpected markup in tables</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>Error handling in tables is, for historical reasons, especially strange. For example, consider
  the following markup:</p>

  <pre><code class="html">&lt;table><strong>&lt;b></strong>&lt;tr>&lt;td>aaa&lt;/td>&lt;/tr><strong>bbb</strong>&lt;/table>ccc</code></pre>

  <p>The highlighted <code>b</code> element start tag is not allowed directly inside a table like
  that, and the parser handles this case by placing the element <em>before</em> the table. (This is
  called <i data-x="foster parent">foster parenting</i>.) This can be seen by examining the DOM tree
  as it stands just after the <code>table</code> element's start tag has been seen:</p>

  <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>table</code></li></ul></li></ul></li></ul>

  <p>...and then immediately after the <code>b</code> element start tag has been seen:</p>

  <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>b</code></li><li class="t1"><code>table</code></li></ul></li></ul></li></ul>

  <p>At this point, the <span>stack of open elements</span> has on it the elements
  <code>html</code>, <code>body</code>, <code>table</code>, and <code>b</code> (in that order,
  despite the resulting DOM tree); the <span>list of active formatting elements</span> just has the
  <code>b</code> element in it; and the <span>insertion mode</span> is "<span data-x="insertion mode:
  in table">in table</span>".</p>

  <p>The <code>tr</code> start tag causes the <code>b</code> element to be popped off the stack and
  a <code>tbody</code> start tag to be implied; the <code>tbody</code> and <code>tr</code> elements
  are then handled in a rather straight-forward manner, taking the parser through the "<span
  data-x="insertion mode: in table body">in table body</span>" and "<span data-x="insertion mode: in
  row">in row</span>" insertion modes, after which the DOM looks as follows:</p>

  <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>b</code></li><li class="t1"><code>table</code><ul><li class="t1"><code>tbody</code><ul><li class="t1"><code>tr</code></li></ul></li></ul></li></ul></li></ul></li></ul>

  <p>Here, the <span>stack of open elements</span> has on it the elements <code>html</code>,
  <code>body</code>, <code>table</code>, <code>tbody</code>, and <code>tr</code>; the <span>list of
  active formatting elements</span> still has the <code>b</code> element in it; and the
  <span>insertion mode</span> is "<span data-x="insertion mode: in row">in row</span>".</p>

  <p>The <code>td</code> element start tag token, after putting a <code>td</code> element on the
  tree, puts a <span data-x="concept-parser-marker">marker</span> on the <span>list of active
  formatting elements</span> (it also switches to the "<span data-x="insertion mode: in cell">in
  cell</span>" <span>insertion mode</span>).</p>

  <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>b</code></li><li class="t1"><code>table</code><ul><li class="t1"><code>tbody</code><ul><li class="t1"><code>tr</code><ul><li class="t1"><code>td</code></li></ul></li></ul></li></ul></li></ul></li></ul></li></ul>

  <p>The <span data-x="concept-parser-marker">marker</span> means that when the "aaa" character
  tokens are seen, no <code>b</code> element is created to hold the resulting <code>Text</code>
  node:</p>

  <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>b</code></li><li class="t1"><code>table</code><ul><li class="t1"><code>tbody</code><ul><li class="t1"><code>tr</code><ul><li class="t1"><code>td</code><ul><li class="t3"><code>#text</code>: <span data-x="">aaa</span></li></ul></li></ul></li></ul></li></ul></li></ul></li></ul></li></ul>

  <p>The end tags are handled in a straight-forward manner; after handling them, the <span>stack of
  open elements</span> has on it the elements <code>html</code>, <code>body</code>,
  <code>table</code>, and <code>tbody</code>; the <span>list of active formatting elements</span>
  still has the <code>b</code> element in it (the <span data-x="concept-parser-marker">marker</span>
  having been removed by the "td" end tag token); and the <span>insertion mode</span> is "<span
  data-x="insertion mode: in table body">in table body</span>".</p>

  <p>Thus it is that the "bbb" character tokens are found. These trigger the "<span data-x="insertion
  mode: in table text">in table text</span>" insertion mode to be used (with the <span>original
  insertion mode</span> set to "<span data-x="insertion mode: in table body">in table body</span>").
  The character tokens are collected, and when the next token (the <code>table</code> element end
  tag) is seen, they are processed as a group. Since they are not all spaces, they are handled as
  per the "anything else" rules in the "<span data-x="insertion mode: in table">in table</span>"
  insertion mode, which defer to the "<span data-x="insertion mode: in body">in body</span>"
  insertion mode but with <span data-x="foster parent">foster parenting</span>.</p>

  <p>When <span data-x="reconstruct the active formatting elements">the active formatting elements
  are reconstructed</span>, a <code>b</code> element is created and <span data-x="foster
  parent">foster parented</span>, and then the "bbb" <code>Text</code> node is appended to it:</p>

  <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>b</code></li><li class="t1"><code>b</code><ul><li class="t3"><code>#text</code>: <span data-x="">bbb</span></li></ul></li><li class="t1"><code>table</code><ul><li class="t1"><code>tbody</code><ul><li class="t1"><code>tr</code><ul><li class="t1"><code>td</code><ul><li class="t3"><code>#text</code>: <span data-x="">aaa</span></li></ul></li></ul></li></ul></li></ul></li></ul></li></ul></li></ul>

  <p>The <span>stack of open elements</span> has on it the elements <code>html</code>,
  <code>body</code>, <code>table</code>, <code>tbody</code>, and the new <code>b</code> (again, note
  that this doesn't match the resulting tree!); the <span>list of active formatting elements</span>
  has the new <code>b</code> element in it; and the <span>insertion mode</span> is still "<span
  data-x="insertion mode: in table body">in table body</span>".</p>

  <p>Had the character tokens been only <span>ASCII whitespace</span> instead of "bbb", then that
  <span>ASCII whitespace</span> would just be appended to the <code>tbody</code> element.</p>

  <p>Finally, the <code>table</code> is closed by a "table" end tag. This pops all the nodes from
  the <span>stack of open elements</span> up to and including the <code>table</code> element, but it
  doesn't affect the <span>list of active formatting elements</span>, so the "ccc" character tokens
  after the table result in yet another <code>b</code> element being created, this time after the
  table:</p>

  <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>b</code></li><li class="t1"><code>b</code><ul><li class="t3"><code>#text</code>: <span data-x="">bbb</span></li></ul></li><li class="t1"><code>table</code><ul><li class="t1"><code>tbody</code><ul><li class="t1"><code>tr</code><ul><li class="t1"><code>td</code><ul><li class="t3"><code>#text</code>: <span data-x="">aaa</span></li></ul></li></ul></li></ul></li></ul></li><li class="t1"><code>b</code><ul><li class="t3"><code>#text</code>: <span data-x="">ccc</span></li></ul></li></ul></li></ul></li></ul>


  <h5>Scripts that modify the page as it is being parsed</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>Consider the following markup, which for this example we will assume is the document with
  <span>URL</span> <code data-x="">https://example.com/inner</code>, being rendered as the content of
  an <code>iframe</code> in another document with the <span>URL</span> <code
  data-x="">https://example.com/outer</code>:</p>

  <pre><code class="html">&lt;div id=a>
 &lt;script>
  var div = document.getElementById('a');
  parent.document.body.appendChild(div);
 &lt;/script>
 &lt;script>
  alert(document.URL);
 &lt;/script>
&lt;/div>
&lt;script>
 alert(document.URL);
&lt;/script></code></pre>

  <p>Up to the first "script" end tag, before the script is parsed, the result is relatively
  straightforward:</p>

  <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>div</code> <span data-x="" class="t2"><code class="attribute name" data-x="attr-id">id</code>="<code class="attribute value" data-x="">a</code>"</span><ul><li class="t3"><code>#text</code>: <span data-x="">
 </span></li><li class="t1"><code>script</code><ul><li class="t3"><code>#text</code>: <span data-x="">var div = document.getElementById('a'); &#x23CE; parent.document.body.appendChild(div);</span></li></ul></li></ul></li></ul></li></ul></li></ul>

  <p>After the script is parsed, though, the <code>div</code> element and its child
  <code>script</code> element are gone:</p>

  <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code></li></ul></li></ul>

  <p>They are, at this point, in the <code>Document</code> of the aforementioned outer
  <span>browsing context</span>. However, the <span>stack of open elements</span> <em>still contains
  the <code>div</code> element</em>.</p>

  <p>Thus, when the second <code>script</code> element is parsed, it is inserted <em>into the outer
  <code>Document</code> object</em>.</p>

  <p>Those parsed into different <code>Document</code>s than the one the parser was created for do
  not execute, so the first alert does not show.</p>

  <p>Once the <code>div</code> element's end tag is parsed, the <code>div</code> element is popped
  off the stack, and so the next <code>script</code> element is in the inner
  <code>Document</code>:</p>

  <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>script</code><ul><li class="t3"><code>#text</code>: <span data-x="">alert(document.URL);</span></li></ul></li></ul></li></ul></li></ul>

  <p>This script does execute, resulting in an alert that says "https://example.com/inner".</p>


  <h5>The execution of scripts that are moving across multiple documents</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>Elaborating on the example in the previous section, consider the case where the second
  <code>script</code> element is an external script (i.e. one with a <code
  data-x="attr-script-src">src</code> attribute). Since the element was not in the parser's
  <code>Document</code> when it was created, that external script is not even downloaded.</p>

  <p>In a case where a <code>script</code> element with a <code data-x="attr-script-src">src</code>
  attribute is parsed normally into its parser's <code>Document</code>, but while the external
  script is being downloaded, the element is moved to another document, the script continues to
  download, but does not execute.</p>

  <p class="note">In general, moving <code>script</code> elements between <code>Document</code>s is
  considered a bad practice.</p>



  <h5>Unclosed formatting elements</h5>

  <!-- NON-NORMATIVE SECTION -->

  <p>The following markup shows how nested formatting elements (such as <code>b</code>) get
  collected and continue to be applied even as the elements they are contained in are closed, but
  that excessive duplicates are thrown away.</p>

  <pre><code class="html">&lt;!DOCTYPE html>
&lt;p>&lt;b class=x>&lt;b class=x>&lt;b>&lt;b class=x>&lt;b class=x>&lt;b>X
&lt;p>X
&lt;p>&lt;b>&lt;b class=x>&lt;b>X
&lt;p>&lt;/b>&lt;/b>&lt;/b>&lt;/b>&lt;/b>&lt;/b>X</code></pre>

  <p>The resulting DOM tree is as follows:</p>

  <ul class="domTree"><li class="t10">DOCTYPE: <code data-x="">html</code></li><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>p</code><ul><li class="t1"><code>b</code> <span class="t2" data-x=""><code class="attribute name" data-x="attr-class">class</code>="<code class="attribute value" data-x="">x</code>"</span><ul><li class="t1"><code>b</code> <span class="t2" data-x=""><code class="attribute name" data-x="attr-class">class</code>="<code class="attribute value" data-x="">x</code>"</span><ul><li class="t1"><code>b</code><ul><li class="t1"><code>b</code> <span class="t2" data-x=""><code class="attribute name" data-x="attr-class">class</code>="<code class="attribute value" data-x="">x</code>"</span><ul><li class="t1"><code>b</code> <span class="t2" data-x=""><code class="attribute name" data-x="attr-class">class</code>="<code class="attribute value" data-x="">x</code>"</span><ul><li class="t1"><code>b</code><ul><li class="t3"><code>#text</code>: <span data-x="">X&#X23CE;</span></li></ul></li></ul></li></ul></li></ul></li></ul></li></ul></li></ul></li><li class="t1"><code>p</code><ul><li class="t1"><code>b</code> <span class="t2" data-x=""><code class="attribute name" data-x="attr-class">class</code>="<code class="attribute value" data-x="">x</code>"</span><ul><li class="t1"><code>b</code><ul><li class="t1"><code>b</code> <span class="t2" data-x=""><code class="attribute name" data-x="attr-class">class</code>="<code class="attribute value" data-x="">x</code>"</span><ul><li class="t1"><code>b</code> <span class="t2" data-x=""><code class="attribute name" data-x="attr-class">class</code>="<code class="attribute value" data-x="">x</code>"</span><ul><li class="t1"><code>b</code><ul><li class="t3"><code>#text</code>: <span data-x="">X&#X23CE;</span></li></ul></li></ul></li></ul></li></ul></li></ul></li></ul></li><li class="t1"><code>p</code><ul><li class="t1"><code>b</code> <span class="t2" data-x=""><code class="attribute name" data-x="attr-class">class</code>="<code class="attribute value" data-x="">x</code>"</span><ul><li class="t1"><code>b</code><ul><li class="t1"><code>b</code> <span class="t2" data-x=""><code class="attribute name" data-x="attr-class">class</code>="<code class="attribute value" data-x="">x</code>"</span><ul><li class="t1"><code>b</code> <span class="t2" data-x=""><code class="attribute name" data-x="attr-class">class</code>="<code class="attribute value" data-x="">x</code>"</span><ul><li class="t1"><code>b</code><ul><li class="t1"><code>b</code><ul><li class="t1"><code>b</code> <span class="t2" data-x=""><code class="attribute name" data-x="attr-class">class</code>="<code class="attribute value" data-x="">x</code>"</span><ul><li class="t1"><code>b</code><ul><li class="t3"><code>#text</code>: <span data-x="">X&#X23CE;</span></li></ul></li></ul></li></ul></li></ul></li></ul></li></ul></li></ul></li></ul></li></ul></li><li class="t1"><code>p</code><ul><li class="t3"><code>#text</code>: <span data-x="">X&#X23CE;</span></li></ul></li></ul></li></ul></li></ul>

  <p>Note how the second <code>p</code> element in the markup has no explicit <code>b</code>
  elements, but in the resulting DOM, up to three of each kind of formatting element (in this case
  three <code>b</code> elements with the class attribute, and two unadorned <code>b</code> elements)
  get reconstructed before the element's "X".</p>

  <p>Also note how this means that in the final paragraph only six <code>b</code> end tags are
  needed to completely clear the <span>list of active formatting elements</span>, even though nine
  <code>b</code> start tags have been seen up to this point.</p>


<!--/HTMLPARSER-->


  <!--en-GB--><h3 id="serialising-html-fragments">Serializing HTML fragments</h3>

  <p>For the purposes of the following algorithm, an element <dfn>serializes as void</dfn> if its
  element type is one of the <span>void elements</span>, or is <code>basefont</code>,
  <code>bgsound</code>, <code>frame</code>, <code>keygen</code>, or <code>param</code>.</p>

  <p>The following steps form the <!--en-GB--><dfn id="html-fragment-serialisation-algorithm"
  export>HTML fragment serialization algorithm</dfn>. The algorithm takes as input a DOM
  <code>Element</code>, <code>Document</code>, or <code>DocumentFragment</code> referred to as
  <var>the node</var>, a boolean <var>serializableShadowRoots</var>, and a
  <code data-x="">sequence&lt;ShadowRoot></code> <var>shadowRoots</var>, and returns a string.</p>

  <p class="note">This algorithm serializes the <em>children</em> of the node being serialized, not
  the node itself.</p>

  <ol>
   <li><p>If <var>the node</var> <span>serializes as void</span>, then return the empty
   string.</p></li>

   <li><p>Let <var>s</var> be a string, and initialize it to the empty string.</p></li>

   <li><p>If <var>the node</var> is a <code>template</code> element, then let <var>the node</var> instead be the <code>template</code> element's <span>template
   contents</span> (a <code>DocumentFragment</code> node).</p></li>

   <li>
    <p>If <var>current node</var> is a <span>shadow host</span>, then:</p>

    <ol>
     <li><p>Let <var>shadow</var> be <var>current node</var>'s
     <span data-x="concept-element-shadow-root">shadow root</span>.</p></li>

     <li>
      <p>If one of the following is true:</p>

      <ul>
       <li><p><var>serializableShadowRoots</var> is true and <var>shadow</var>'s
       <span data-x="shadow-serializable">serializable</span> is true; or</p></li>

       <li><p><var>shadowRoots</var> contains <var>shadow</var>,</p></li>
      </ul>

      <p>then:</p>

      <ol>
       <li><p>Append "<code data-x="">&lt;template shadowrootmode=&quot;</code>".</p></li>

       <li><p>If <var>shadow</var>'s <span data-x="concept-shadow-root-mode">mode</span>
       is "<code data-x="">open</code>", then append "<code data-x="">open</code>".
       Otherwise, append "<code data-x="">closed</code>".</p></li>

       <li><p>Append "<code data-x="">&quot;</code>".</p></li>

       <li><p>If <var>shadow</var>'s <span>delegates focus</span> is set, then append
       "<code data-x=""> shadowrootdelegatesfocus=&quot;&quot;</code>".</p></li>

       <li><p>If <var>shadow</var>'s <span>serializable</span> is set, then append
       "<code data-x=""> shadowrootserializable=&quot;&quot;</code>".</p></li>

       <li><p>If <var>shadow</var>'s <span>clonable</span> is set, then append
       "<code data-x=""> shadowrootclonable=&quot;&quot;</code>".</p></li>

       <li><p>Append "<code data-x="">&gt;</code>".</p></li>

       <li><p>Append the value of running the <span>HTML fragment serialization algorithm</span> with
       <var>shadow</var>, <var>serializableShadowRoots</var>, and <var>shadowRoots</var> (thus recursing
       into this algorithm for that element).</p></li>

       <li><p>Append "<code data-x="">&lt;/template&gt;</code>".</p></li>
      </ol>
     </li>
    </ol>
   </li>

   <li>
    <p>For each child node of <var>the node</var>, in <span>tree order</span>, run the
    following steps:</p>

    <ol>
     <li><p>Let <var>current node</var> be the child node being processed.</p></li>

     <li>
      <p>Append the appropriate string from the following list to <var>s</var>:</p>

      <dl class="switch">

       <dt>If <var>current node</var> is an <code>Element</code></dt>

       <dd>
        <p>If <var>current node</var> is an element in the <span>HTML namespace</span>, the
        <span>MathML namespace</span>, or the <span>SVG namespace</span>, then let <var>tagname</var> be <var>current node</var>'s local name. Otherwise, let <var>tagname</var> be <var>current node</var>'s qualified name.</p>

        <p>Append a U+003C LESS-THAN SIGN character (&lt;), followed by <var>tagname</var>.</p>

        <p class="note">For <span>HTML elements</span> created by the <span>HTML parser</span> or
        <code data-x="dom-Document-createElement">createElement()</code>, <var>tagname</var> will be
        lowercase.</p>

        <p>If <var>current node</var>'s <span data-x="concept-element-is-value"><code
        data-x="">is</code> value</span> is not null, and the element does not have an <code
        data-x="attr-is">is</code> attribute in its attribute list, then append the string "<code
        data-x=""> is="</code>", followed by <var>current node</var>'s <span
        data-x="concept-element-is-value"><code data-x="">is</code> value</span> <span
        data-x="escaping a string">escaped as described below</span> in <i>attribute mode</i>,
        followed by a U+0022 QUOTATION MARK character (&quot;).</p>

        <p>For each attribute that the element has, append a U+0020 SPACE character, the <span
        data-x="attribute's serialized name">attribute's serialized name as described below</span>, a
        U+003D EQUALS SIGN character (=), a U+0022 QUOTATION MARK character (&quot;), the
        attribute's value, <span data-x="escaping a string">escaped as described below</span> in
        <i>attribute mode</i>, and a second U+0022 QUOTATION MARK character (&quot;).</p>

        <p>An <!--en-GB--><dfn id="attribute's-serialised-name">attribute's serialized name</dfn>
        for the purposes of the previous paragraph must be determined as follows:</p>

        <dl class="switch">

         <dt>If the attribute has no namespace</dt>

         <dd>
          <p>The attribute's serialized name is the attribute's local name.</p>

          <p class="note">For attributes on <span>HTML elements</span> set by the <span>HTML
          parser</span> or by <code data-x="dom-Element-setAttribute">setAttribute()</code>, the
          local name will be lowercase.</p>
         </dd>


         <dt>If the attribute is in the <span>XML namespace</span></dt>

         <dd><p>The attribute's serialized name is the string "<code data-x="">xml:</code>" followed
         by the attribute's local name.</p></dd>


         <dt>If the attribute is in the <span>XMLNS namespace</span> and the attribute's local name
         is <code data-x="">xmlns</code></dt>

         <dd><p>The attribute's serialized name is the string "<code
         data-x="">xmlns</code>".</p></dd>


         <dt>If the attribute is in the <span>XMLNS namespace</span> and the attribute's local name
         is not <code data-x="">xmlns</code></dt>

         <dd><p>The attribute's serialized name is the string "<code data-x="">xmlns:</code>"
         followed by the attribute's local name.</p></dd>


         <dt>If the attribute is in the <span>XLink namespace</span></dt>

         <dd><p>The attribute's serialized name is the string "<code data-x="">xlink:</code>"
         followed by the attribute's local name.</p></dd>


         <dt>If the attribute is in some other namespace</dt>

         <dd><p>The attribute's serialized name is the attribute's qualified name.</p></dd>
        </dl>

        <p>While the exact order of attributes is <span>implementation-defined</span>, and may
        depend on factors such as the order that the attributes were given in the original markup,
        the sort order must be stable, such that consecutive invocations of this algorithm serialize
        an element's attributes in the same order.</p>

        <p>Append a U+003E GREATER-THAN SIGN character (&gt;).</p>

        <p>If <var>current node</var> <span>serializes as void</span>, then <span>continue</span> on to the next
        child node at this point.</p>

        <p>Append the value of running the <span>HTML fragment serialization algorithm</span> with
        <var>current node</var>, <var>serializableShadowRoots</var>, and <var>shadowRoots</var> (thus
        recursing into this algorithm for that node), followed by a U+003C LESS-THAN SIGN character (&lt;), a
        U+002F SOLIDUS character (/), <var>tagname</var> again, and finally a U+003E GREATER-THAN SIGN
        character (&gt;).</p>
       </dd>


       <dt>If <var>current node</var> is a <code>Text</code> node</dt>

       <dd>
        <p>If the parent of <var>current node</var> is a <code>style</code>,
        <code>script</code>, <code>xmp</code>, <code>iframe</code>, <code>noembed</code>,
        <code>noframes</code>, or <code>plaintext</code> element, or if the parent of <var>current node</var> is a <code>noscript</code> element and <span
        data-x="concept-n-script">scripting is enabled</span> for the node, then append the value of
        <var>current node</var>'s <span data-x="concept-cd-data">data</span> literally.</p>

        <p>Otherwise, append the value of <var>current node</var>'s <span
        data-x="concept-cd-data">data</span>, <span data-x="escaping a string">escaped as described
        below</span>.</p>
       </dd>


       <dt>If <var>current node</var> is a <code>Comment</code></dt>

       <dd>
        <p>Append "<code data-x="">&lt;!--</code>" (U+003C LESS-THAN SIGN, U+0021
        EXCLAMATION MARK, U+002D HYPHEN-MINUS, U+002D HYPHEN-MINUS), followed by the value of
        <var>current node</var>'s <span data-x="concept-cd-data">data</span>, followed by the
        literal string "<code data-x="">--&gt;</code>" (U+002D HYPHEN-MINUS, U+002D HYPHEN-MINUS,
        U+003E GREATER-THAN SIGN).</p>
       </dd>


       <dt>If <var>current node</var> is a <code>ProcessingInstruction</code></dt>

       <dd>
        <p>Append "<code data-x="">&lt;?</code>" (U+003C LESS-THAN SIGN, U+003F
        QUESTION MARK), followed by the value of <var>current node</var>'s <code
        data-x="">target</code> IDL attribute, followed by a single U+0020 SPACE character, followed
        by the value of <var>current node</var>'s <span data-x="concept-cd-data">data</span>,
        followed by a single U+003E GREATER-THAN SIGN character (>).</p>
       </dd>


       <dt>If <var>current node</var> is a <code>DocumentType</code></dt>

       <dd>
        <p>Append "<code data-x="">&lt;!DOCTYPE</code>" (U+003C LESS-THAN SIGN, U+0021
        EXCLAMATION MARK, U+0044 LATIN CAPITAL LETTER D, U+004F LATIN CAPITAL LETTER O, U+0043 LATIN
        CAPITAL LETTER C, U+0054 LATIN CAPITAL LETTER T, U+0059 LATIN CAPITAL LETTER Y, U+0050 LATIN
        CAPITAL LETTER P, U+0045 LATIN CAPITAL LETTER E), followed by a space (U+0020 SPACE),
        followed by the value of <var>current node</var>'s <span
        data-x="concept-doctype-name">name</span>, followed by "<code data-x="">&gt;</code>"
        (U+003E GREATER-THAN SIGN).</p>
       </dd>

      </dl>
     </li>
    </ol>
   </li>

   <li><p>Return <var>s</var>.</p></li>
  </ol>

  <p class="warning">It is possible that the output of this algorithm, if parsed with an <span>HTML
  parser</span>, will not return the original tree structure. Tree structures that do not roundtrip
  a serialize and reparse step can also be produced by the <span>HTML parser</span> itself, although
  such cases are typically non-conforming.</p>

  <div class="example">

   <p>For instance, if a <code>textarea</code> element to which a <code data-x="">Comment</code>
   node has been appended is serialized and the output is then reparsed, the comment will end up
   being displayed in the text control. Similarly, if, as a result of DOM manipulation, an element
   contains a comment that contains "<code data-x="">--&gt;</code>", then when
   the result of serializing the element is parsed, the comment will be truncated at that point and
   the rest of the comment will be interpreted as markup. More examples would be making a
   <code>script</code> element contain a <code>Text</code> node with the text string "<code
   data-x="">&lt;/script></code>", or having a <code>p</code> element that contains a
   <code>ul</code> element (as the <code>ul</code> element's <span data-x="syntax-start-tag">start
   tag</span> would imply the end tag for the <code>p</code>).</p>

   <p>This can enable cross-site scripting attacks. An example of this would be a page that lets the
   user enter some font family names that are then inserted into a CSS <code>style</code> block via
   the DOM and which then uses the <code data-x="dom-element-innerHTML">innerHTML</code> IDL attribute to get
   the HTML serialization of that <code>style</code> element: if the user enters
   "<code data-x="">&lt;/style>&lt;script>attack&lt;/script></code>" as a font family name, <code
   data-x="dom-element-innerHTML">innerHTML</code> will return markup that, if parsed in a different context,
   would contain a <code>script</code> node, even though no <code>script</code> node existed in the
   original DOM.</p>

  </div>

  <div class="example">

   <p>For example, consider the following markup:</p>

   <pre><code class="html">&lt;form id="outer">&lt;div>&lt;/form>&lt;form id="inner">&lt;input></code></pre>

   <p>This will be parsed into:</p>

   <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>form</code> <span class="t2" data-x=""><code class="attribute name" data-x="attr-id">id</code>="<code class="attribute value" data-x="">outer</code>"</span><ul><li class="t1"><code>div</code><ul><li class="t1"><code>form</code> <span class="t2" data-x=""><code class="attribute name" data-x="attr-id">id</code>="<code class="attribute value" data-x="">inner</code>"</span><ul><li class="t1"><code>input</code></li></ul></li></ul></li></ul></li></ul></li></ul></li></ul>

   <p>The <code>input</code> element will be associated with the inner <code>form</code> element.
   Now, if this tree structure is serialized and reparsed, the <code data-x="">&lt;form
   id="inner"></code> start tag will be ignored, and so the <code>input</code> element will be
   associated with the outer <code>form</code> element instead.</p>

   <pre><code class="html">&lt;html>&lt;head>&lt;/head>&lt;body>&lt;form id="outer">&lt;div><mark>&lt;form id="inner"></mark>&lt;input>&lt;/form>&lt;/div>&lt;/form>&lt;/body>&lt;/html></code></pre>

   <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>form</code> <span class="t2" data-x=""><code class="attribute name" data-x="attr-id">id</code>="<code class="attribute value" data-x="">outer</code>"</span><ul><li class="t1"><code>div</code><ul><li class="t1"><code>input</code></li></ul></li></ul></li></ul></li></ul></li></ul>

  </div>

  <div class="example">

   <p>As another example, consider the following markup:</p>

   <pre><code class="html">&lt;a>&lt;table>&lt;a></code></pre>

   <p>This will be parsed into:</p>

   <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>a</code><ul><li class="t1"><code>a</code></li><li class="t1"><code>table</code></li></ul></li></ul></li></ul></li></ul>

   <p>That is, the <code>a</code> elements are nested, because the second <code>a</code> element is
   <span data-x="foster parent">foster parented</span>. After a serialize-reparse roundtrip, the
   <code>a</code> elements and the <code>table</code> element would all be siblings, because the
   second <code data-x="">&lt;a></code> start tag implicitly closes the first <code>a</code>
   element.</p>

   <pre><code class="html">&lt;html>&lt;head>&lt;/head>&lt;body>&lt;a><mark>&lt;a></mark>&lt;/a>&lt;table>&lt;/table>&lt;/a>&lt;/body>&lt;/html></code></pre>

   <ul class="domTree"><li class="t1"><code>html</code><ul><li class="t1"><code>head</code></li><li class="t1"><code>body</code><ul><li class="t1"><code>a</code></li><li class="t1"><code>a</code></li><li class="t1"><code>table</code></li></ul></li></ul></li></ul>

  </div>

  <p>For historical reasons, this algorithm does not round-trip an initial U+000A LINE FEED (LF)
  character in <code>pre</code>, <code>textarea</code>, or <code>listing</code> elements, even
  though (in the first two cases) the markup being round-tripped can be conforming. The <span>HTML
  parser</span> will drop such a character during parsing, but this algorithm does <em>not</em>
  serialize an extra U+000A LINE FEED (LF) character.</p>
  <!-- https://github.com/whatwg/html/issues/944 -->

  <div class="example">
   <p>For example, consider the following markup:</p>

   <pre><code class="html">&lt;pre>

Hello.&lt;/pre></code></pre>

   <p>When this document is first parsed, the <code>pre</code> element's <span>child text
   content</span> starts with a single newline character. After a serialize-reparse roundtrip, the
   <code>pre</code> element's <span>child text content</span> is simply "<code
   data-x="">Hello.</code>".</p>
  </div>

  <p id="attr-is-during-serialization">Because of the special role of the <code
  data-x="attr-is">is</code> attribute in signaling the creation of <span data-x="customized
  built-in element">customized built-in elements</span>, in that it provides a mechanism for parsed
  HTML to set the element's <span data-x="concept-element-is-value"><code data-x="">is</code>
  value</span>, we special-case its handling during serialization. This ensures that an element's
  <span data-x="concept-element-is-value"><code data-x="">is</code> value</span> is preserved
  through serialize-parse roundtrips.</p>

  <div class="example">
   <p>When creating a <span>customized built-in element</span> via the parser, a developer uses the <code
   data-x="attr-is">is</code> attribute directly; in such cases serialize-parse roundtrips work fine.</p>

   <pre><code class="html">&lt;script>
window.SuperP = class extends HTMLParagraphElement {};
customElements.define("super-p", SuperP, { extends: "p" });
&lt;/script>

&lt;div id="container">&lt;p is="super-p">Superb!&lt;/p>&lt;/div>

&lt;script>
console.log(container.innerHTML); // &lt;p is="super-p">
container.innerHTML = container.innerHTML;
console.log(container.innerHTML); // &lt;p is="super-p">
console.assert(container.firstChild instanceof SuperP);
&lt;/script></code></pre>

   <p>But when creating a customized built-in element via its <span data-x="custom element
   constructor">constructor</span> or via <code
   data-x="dom-Document-createElement">createElement()</code>, the <code data-x="attr-is">is</code>
   attribute is not added. Instead, the <span data-x="concept-element-is-value"><code
   data-x="">is</code> value</span> (which is what the custom elements machinery uses) is set
   without intermediating through an attribute.</p>

   <pre><code class="html">&lt;script>
container.innerHTML = "";
const p = document.createElement("p", { is: "super-p" });
container.appendChild(p);

// The is attribute is not present in the DOM:
console.assert(!p.hasAttribute("is"));

// But the element is still a super-p:
console.assert(p instanceof SuperP);
&lt;/script></code></pre>

   <p>To ensure that serialize-parse roundtrips still work, the serialization process explicitly
   writes out the element's <span data-x="concept-element-is-value"><code data-x="">is</code>
   value</span> as an <code data-x="attr-is">is</code> attribute:</p>

   <pre><code class="html">&lt;script>
console.log(container.innerHTML); // &lt;p is="super-p">
container.innerHTML = container.innerHTML;
console.log(container.innerHTML); // &lt;p is="super-p">
console.assert(container.firstChild instanceof SuperP);
&lt;/script></code></pre>
  </div>

  <p><dfn id="escapingString">Escaping a string</dfn> (for the purposes of the algorithm above)
  consists of running the following steps:</p>

  <ol>
   <li><p>Replace any occurrence of the "<code data-x="">&amp;</code>" character by the string "<code
   data-x="">&amp;amp;</code>".</p></li>

   <li><p>Replace any occurrences of the U+00A0 NO-BREAK SPACE character by the string "<code
   data-x="">&amp;nbsp;</code>".</p></li>

   <li><p>If the algorithm was invoked in the <i>attribute mode</i>, replace any occurrences of the
   "<code data-x="">&quot;</code>" character by the string "<code
   data-x="">&amp;quot;</code>".</p></li>

   <li><p>If the algorithm was <em>not</em> invoked in the <i>attribute mode</i>, replace any
   occurrences of the "<code data-x="">&lt;</code>" character by the string "<code
   data-x="">&amp;lt;</code>", and any occurrences of the "<code data-x="">&gt;</code>" character by
   the string "<code data-x="">&amp;gt;</code>".</p></li>
  </ol>


  <h3>Parsing HTML fragments</h3>

  <p>The <dfn>HTML fragment parsing algorithm</dfn>, given an <code>Element</code> node <dfn
  data-x="concept-frag-parse-context"><var>context</var></dfn>, string <var>input</var>, and an
  optional boolean <var>allowDeclarativeShadowRoots</var> (default false) is the following steps.
  They return a list of zero or more nodes.</p>

  <p class="note">Parts marked <dfn>fragment case</dfn> in algorithms in the <span>HTML
  parser</span> section are parts that only occur if the parser was created for the purposes of this
  algorithm. The algorithms have been annotated with such markings for informational purposes only;
  such markings have no normative weight. If it is possible for a condition described as a
  <span>fragment case</span> to occur even when the parser wasn't created for the purposes of
  handling this algorithm, then that is an error in the specification.</p>

  <ol>
   <li><p>Let <var>document</var> be a <code>Document</code> node whose <span
   data-x="concept-document-type">type</span> is "<code data-x="">html</code>".</p></li>

   <li><p>If <var data-x="concept-frag-parse-context">context</var>'s <span>node document</span> is
   in <span>quirks mode</span>, then set <var>document</var>'s <span
   data-x="concept-document-mode">mode</span> to "<code data-x="">quirks</code>".</p></li>

   <li><p>Otherwise, if <var data-x="concept-frag-parse-context">context</var>'s <span>node
   document</span> is in <span>limited-quirks mode</span>, then set <var>document</var>'s <span
   data-x="concept-document-mode">mode</span> to "<code data-x="">limited-quirks</code>".</p></li>

   <li><p>If <var>allowDeclarativeShadowRoots</var> is true, then set <var>document</var>'s <span
   data-x="concept-document-allow-declarative-shadow-roots">allow declarative shadow roots</span> to
   true.</p></li>

   <li><p>Create a new <span>HTML parser</span>, and associate it with <var>document</var>.</p></li>

   <li>
    <p>Set the state of the <span>HTML parser</span>'s <span>tokenization</span> stage as
    follows, switching on the <var data-x="concept-frag-parse-context">context</var> element:</p>

    <dl class="switch">
     <dt><code>title</code></dt>
     <dt><code>textarea</code></dt>
     <dd>Switch the tokenizer to the <span>RCDATA state</span>.</dd>

     <dt><code>style</code></dt>
     <dt><code>xmp</code></dt>
     <dt><code>iframe</code></dt>
     <dt><code>noembed</code></dt>
     <dt><code>noframes</code></dt>
     <dd>Switch the tokenizer to the <span>RAWTEXT state</span>.</dd>

     <dt><code>script</code></dt>
     <dd>Switch the tokenizer to the <span>script data state</span>.</dd>

     <dt><code>noscript</code></dt>
     <dd>If the <span>scripting flag</span> is enabled, switch the tokenizer to the <span>RAWTEXT
     state</span>. Otherwise, leave the tokenizer in the <span>data state</span>.</dd>

     <dt><code>plaintext</code></dt>
     <dd>Switch the tokenizer to the <span>PLAINTEXT state</span>.</dd>

     <dt>Any other element</dt>
     <dd>Leave the tokenizer in the <span>data state</span>.</dd>
    </dl>

    <p class="note">For performance reasons, an implementation that does not report errors and
    that uses the actual state machine described in this specification directly could use the
    PLAINTEXT state instead of the RAWTEXT and script data states where those are mentioned in the
    list above. Except for rules regarding parse errors, they are equivalent, since there is no
    <span>appropriate end tag token</span> in the fragment case, yet they involve far fewer state
    transitions.</p>
   </li>

   <li><p>Let <var>root</var> be the result of <span data-x="create an element">creating an
   element</span> given <var>document</var>, "<code data-x="">html</code>", and the <span>HTML
   namespace</span>.</p></li>

   <li><p><span data-x="concept-node-append">Append</span> <var>root</var> to
   <var>document</var>.</p></li>

   <li><p>Set up the <span>HTML parser</span>'s <span>stack of open elements</span> so that it
   contains just the single element <var>root</var>.</p></li>

   <li><p>If <var data-x="concept-frag-parse-context">context</var> is a <code>template</code>
   element, then push "<span data-x="insertion mode: in template">in template</span>" onto the
   <span>stack of template insertion modes</span> so that it is the new <span>current template
   insertion mode</span>.</p></li>

   <li>
    <p>Create a start tag token whose name is the local name of <var
    data-x="concept-frag-parse-context">context</var> and whose attributes are the attributes of
    <var data-x="concept-frag-parse-context">context</var>.</p>

    <p>Let this start tag token be the start tag token of <var
    data-x="concept-frag-parse-context">context</var>; e.g. for the purposes of determining if it is
    an <span>HTML integration point</span>.</p>
   </li>

   <li>
    <p><span data-x="reset the insertion mode appropriately">Reset the parser's insertion mode
    appropriately</span>.</p>

    <p class="note">The parser will reference the <var
    data-x="concept-frag-parse-context">context</var> element as part of that algorithm.</p>
   </li>

   <li><p>Set the <span>HTML parser</span>'s <span><code>form</code> element pointer</span> to the
   nearest node to <var data-x="concept-frag-parse-context">context</var> that is a
   <code>form</code> element (going straight up the ancestor chain, and including the element
   itself, if it is a <code>form</code> element), if any. (If there is no such <code>form</code>
   element, the <span><code>form</code> element pointer</span> keeps its initial value,
   null.)</p></li>

   <li><p>Place the <var>input</var> into the <span>input stream</span> for the <span>HTML
   parser</span> just created. The encoding <span
   data-x="concept-encoding-confidence">confidence</span> is <i>irrelevant</i>.</p></li>

   <li><p>Start the <span>HTML parser</span> and let it run until it has consumed all the characters
   just inserted into the input stream.</p></li>

   <li><p>Return <var>root</var>'s <span data-x="concept-tree-child">children</span>, in <span>tree
   order</span>.</p></li>
  </ol>

  </div>



  <h3 split-filename="named-characters"><dfn>Named character references</dfn></h3>

  <p>This table lists the <span data-x="syntax-charref">character reference</span> names that are
  supported by HTML, and the code points to which they refer. It is referenced by the previous
  sections.</p>

  <p class="note">It is intentional, for legacy compatibility, that many code points have multiple
  character reference names. For example, some appear both with and without the trailing semicolon,
  or with different capitalizations.</p>

  <div id="named-character-references-table">
   <table>
    <thead>
     <tr> <th> Name </th> <th> Character(s) </th> <th> Glyph </th> </tr>
    </thead>
<!--BOILERPLATE entities.inc-->
   </table>
  <!--
   If we want to add character references, Almorca suggests:
   > I would add &sub1; (character U+2081), &sub2; (character U+2082) and &sub3; (character U+2083).
   > They would are the equivalent to &sup1;, &sup2;, and &sup3;.
   See also: https://www.w3.org/2003/entities/
  -->
  </div>

  <p>This data is also available <a href="/entities.json">as a JSON file</a>.</p>

  <p><i>The glyphs displayed above are non-normative. Refer to Unicode for formal definitions of the
  characters listed above.</i></p>

  <p class="note">The character reference names originate from <cite>XML Entity Definitions for
  Characters</cite>, though only the above is considered normative. <ref>XMLENTITY</ref></p>

  <p class="note">This list is static and <a
  href="https://github.com/whatwg/html/blob/main/FAQ.md#html-should-add-more-named-character-references">will
  not be expanded or changed in the future</a>.</p>


  <h2 split-filename="xhtml" id="the-xhtml-syntax"><dfn id="xhtml">The XML syntax</dfn></h2>

  <p class="note">This section only describes the rules for XML resources. Rules for
  <code>text/html</code> resources are discussed in the section above entitled "<span>The HTML
  syntax</span>".</p>

  <p class="warning" id="xml-syntax-not-recommended">Using the XML syntax is not recommended, for
  reasons which include the fact that there is no specification which defines the rules for how an
  XML parser must map a string of bytes or characters into a <code>Document</code> object, as well
  as the fact that the XML syntax is essentially unmaintained — in that, it’s not expected that any
  further features will ever be added to the XML syntax (even when such features have been added to
  the <span data-x="the HTML syntax">HTML syntax</span>).</p>

  <div w-nodev>

  <h3 id="writing-xhtml-documents">Writing documents in the XML syntax</h3>

  </div>

  <p class="note">The XML syntax for HTML was formerly referred to as "XHTML", but this
  specification does not use that term (among other reasons, because no such term is used for the
  HTML syntaxes of MathML and SVG).</p>

  <p>The syntax for XML is defined in <cite>XML</cite> and <cite>Namespaces in XML</cite>. <ref>XML</ref>
  <ref>XMLNS</ref></p>

  <p>This specification does not define any syntax-level requirements beyond those defined for XML
  proper.</p>

  <p>XML documents may contain a <code data-x="">DOCTYPE</code> if desired, but this is not required
  to conform to this specification. This specification does not define a public or system
  identifier, nor provide a formal DTD.</p>

  <p class="note">According to <cite>XML</cite>, XML processors are not guaranteed to process
  the external DTD subset referenced in the DOCTYPE. This means, for example, that using <a
  href="https://www.w3.org/TR/xml/#dt-entref">entity references</a> for characters in XML documents
  is unsafe if they are defined in an external file (except for <code data-x="">&amp;lt;</code>,
  <code data-x="">&amp;gt;</code>, <code data-x="">&amp;amp;</code>,
  <code data-x="">&amp;quot;</code>, and <code data-x="">&amp;apos;</code>).</p>


  <div w-nodev>

  <h3 id="parsing-xhtml-documents">Parsing XML documents</h3>

  <p>This section describes the relationship between XML and the DOM, with a particular emphasis on
  how this interacts with HTML.</p>

  <p>An <dfn export>XML parser</dfn>, for the purposes of this specification, is a construct that
  follows the rules given in <cite>XML</cite> to map a string of bytes or characters into a
  <code>Document</code> object.</p>

  <p class="note">At the time of writing, no such rules actually exist.</p><!--XMLPARSE-->

  <p>An <span>XML parser</span> is either associated with a <code>Document</code> object when it is
  created, or creates one implicitly.</p>

  <p>This <code>Document</code> must then be populated with DOM nodes that represent the tree
  structure of the input passed to the parser, as defined by <cite>XML</cite>, <cite>Namespaces
  in XML</cite>, and <cite>DOM</cite>. When creating DOM nodes representing elements,
  the <span data-x="create an element for the token">create an element for a token</span> algorithm
  or some equivalent that operates on appropriate XML data structures must be used, to ensure the
  proper <span data-x="element interface">element interfaces</span> are created and that <span
  data-x="custom element">custom elements</span> are set up correctly.</p>

  <p>For the operations that the <span>XML parser</span> performs on the <code>Document</code>'s
  tree, the user agent must act as if elements and attributes were individually appended and set
  respectively so as to trigger rules in this specification regarding what happens when an element
  is inserted into a document or has its attributes set, and <cite>DOM</cite>'s requirements
  regarding <span>mutation observers</span> mean that mutation observers <em>are</em> fired.
  <ref>XML</ref> <ref>XMLNS</ref> <ref>DOM</ref> <ref>UIEVENTS</ref></p>

  <p>Between the time an element's start tag is parsed and the time either the element's end tag is
  parsed or the parser detects a well-formedness error, the user agent must act as if the element
  was in a <span>stack of open elements</span>.</p>

  <p class="note">This is used by various elements to only start certain processes once they are
  popped off of the <span>stack of open elements</span>.</p>

  <p>This specification provides the following additional information that user agents should use
  when retrieving an external entity: the public identifiers given in the following list all
  correspond to <a href="<!--BOILERPLATE entities-dtd.url-->">the URL given by this link</a>. (This
  URL is a DTD containing the <a href="https://www.w3.org/TR/xml/#sec-entity-decl">entity
  declarations</a> for the names listed in the <span>named character references</span> section.) <ref>XML</ref></p>

  <ul class="brief">
   <li><code data-x="">-//W3C//DTD XHTML 1.0 Transitional//EN</code></li>
   <li><code data-x="">-//W3C//DTD XHTML 1.1//EN</code></li>
   <li><code data-x="">-//W3C//DTD XHTML 1.0 Strict//EN</code></li>
   <li><code data-x="">-//W3C//DTD XHTML 1.0 Frameset//EN</code></li>
   <li><code data-x="">-//W3C//DTD XHTML Basic 1.0//EN</code></li>
   <li><code data-x="">-//W3C//DTD XHTML 1.1 plus MathML 2.0//EN</code></li>
   <li><code data-x="">-//W3C//DTD XHTML 1.1 plus MathML 2.0 plus SVG 1.1//EN</code></li>
   <li><code data-x="">-//W3C//DTD MathML 2.0//EN</code></li>
   <li><code data-x="">-//WAPFORUM//DTD XHTML Mobile 1.0//EN</code></li>
  </ul>

  <p>Furthermore, user agents should attempt to retrieve the above external entity's content when
  one of the above public identifiers is used, and should not attempt to retrieve any other external
  entity's content.</p>

  <p class="note">This is not strictly a <span data-x="willful violation">violation</span> of
  <cite>XML</cite>, but it does contradict the spirit of <cite>XML</cite>'s requirements. This is
  motivated by a desire for user agents to all handle entities in an interoperable fashion without
  requiring any network access for handling external subsets. <ref>XML</ref></p>

  <p>XML parsers can be invoked with <dfn export>XML scripting support enabled</dfn> or <dfn
  export>XML scripting support disabled</dfn>. Except where otherwise specified, XML parsers are
  invoked with <span>XML scripting support enabled</span>.</p>

  <p id="scriptTagXML">When an <span>XML parser</span> with <span>XML scripting support
  enabled</span> creates a <code>script</code> element, it must have its <span>parser
  document</span> set and its <span data-x="script-force-async">force async</span> set to false. If
  the parser was created as part of the <span>XML fragment parsing algorithm</span>, then the
  element's <span>already started</span> must be set to true. When the element's end tag is
  subsequently parsed, the user agent must <span>perform a microtask checkpoint</span>, and then
  <span data-x="prepare the script element">prepare</span> the <code>script</code> element. If this
  causes there to be a <span>pending parsing-blocking script</span>, then the user agent must run
  the following steps:</p>


  <ol>
   <li><p>Block this instance of the <span>XML parser</span>, such that the <span>event loop</span>
   will not run <span data-x="concept-task">tasks</span> that invoke it.</p></li>

   <li><p><span>Spin the event loop</span> until the parser's <code>Document</code> <span>has no
   style sheet that is blocking scripts</span> and the <span>pending parsing-blocking
   script</span>'s <span>ready to be parser-executed</span> is true.</p></li>

   <li><p>Unblock this instance of the <span>XML parser</span>, such that <span
   data-x="concept-task">tasks</span> that invoke it can again be run.</p></li>

   <li><p><span>Execute the script element</span> given by the <span>pending parsing-blocking
   script</span>.</p></li>

   <li><p>Set the <span>pending parsing-blocking script</span> to null.</p></li>
  </ol>

  <p class="note">Since the <code data-x="dom-document-write">document.write()</code> API is not
  available for <span>XML documents</span>, much of the complexity in the <span>HTML parser</span>
  is not needed in the <span>XML parser</span>.</p>

  <p class="note">When the <span>XML parser</span> has <span>XML scripting support disabled</span>,
  none of this happens.</p>

  <p id="templateTagXML">When an <span>XML parser</span> would append a node to a
  <code>template</code> element, it must instead append it to the <code>template</code> element's
  <span>template contents</span> (a <code>DocumentFragment</code> node).</p>

  <p class="note">This is a <span>willful violation</span> of <cite>XML</cite>; unfortunately,
  XML is not formally extensible in the manner that is needed for <code>template</code> processing.
  <ref>XML</ref></p>

  <p>When an <span>XML parser</span> creates a <code>Node</code> object, its <span>node document</span>
  must be set to the <span>node document</span> of
  the node into which the newly created node is to be inserted.</p>

  <p>Certain algorithms in this specification <dfn data-x="feed the parser">spoon-feed the
  parser</dfn> characters one string at a time. In such cases, the <span>XML parser</span> must act
  as it would have if faced with a single string consisting of the concatenation of all those
  characters.</p>

  <p>When an <span>XML parser</span> reaches the end of its input, it must <span>stop
  parsing</span>, following the same rules as the <span>HTML parser</span>. An <span>XML
  parser</span> can also be <span data-x="abort a parser">aborted</span>, which must again be done in
  the same way as for an <span>HTML parser</span>.</p>

  <p>For the purposes of conformance checkers, if a resource is determined to be in <span>the XML
  syntax</span>, then it is an <span data-x="XML documents">XML document</span>.</p>


<!--//HTMLPARSER-->


  <!--en-GB--><h3 id="serialising-xhtml-fragments">Serializing XML fragments</h3>

  <p>The <!--en-GB--><dfn id="xml-fragment-serialisation-algorithm">XML fragment serialization
  algorithm</dfn> for a <code>Document</code> or <code>Element</code> node either returns a fragment
  of XML that represents that node or throws an exception.</p>

  <p>For <code>Document</code>s, the algorithm must return a string in the form of a <a
  href="https://www.w3.org/TR/xml/#sec-well-formed">document entity</a>, if none of the error cases
  below apply.</p>

  <p>For <code>Element</code>s, the algorithm must return a string in the form of an <a
  href="https://www.w3.org/TR/xml/#wf-entities">internal general parsed entity</a>, if none of the
  error cases below apply.</p>

  <p>In both cases, the string returned must be XML namespace-well-formed and must be an isomorphic
  serialization of all of that node's <span>relevant child nodes</span>, in <span>tree order</span>.
  User agents may adjust prefixes and namespace declarations in the serialization (and indeed might
  be forced to do so in some cases to obtain namespace-well-formed XML). User agents may use a
  combination of regular text and character references to represent <code>Text</code> nodes in the
  DOM.</p>

  <p>A node's <dfn>relevant child nodes</dfn> are those that apply given the following rules:</p>

  <dl>
   <dt>For <code>template</code> elements</dt>

   <dd>The <span>relevant child nodes</span> are the child nodes of the <code>template</code>
   element's <span>template contents</span>, if any.</dd>

   <dt>For all other nodes</dt>

   <dd>The <span>relevant child nodes</span> are the child nodes of node itself, if any.</dd>
  </dl>

  <p>For <code>Element</code>s, if any of the elements in the serialization are in no namespace, the
  default namespace in scope for those elements must be explicitly declared as the empty string.<!--
  because otherwise round-tripping might break since it'll pick up the surrounding default ns when
  setting --> (This doesn't apply in the <code>Document</code> case.) <ref>XML</ref>
  <ref>XMLNS</ref></p>

  <p>For the purposes of this section, an internal general parsed entity is considered XML
  namespace-well-formed if a document consisting of an element with no namespace declarations whose
  contents are the internal general parsed entity would itself be XML namespace-well-formed.</p>

  <p>If any of the following error cases are found in the DOM subtree being serialized, then the
  algorithm must throw an <span>"<code>InvalidStateError</code>"</span> <code>DOMException</code>
  instead of returning a string:</p>

  <ul>
   <li>A <code>Document</code> node with no child element nodes.</li>

   <li>A <code>DocumentType</code> node that has an external subset public identifier that contains
   characters that are not matched by the XML <code data-x="">PubidChar</code> production. <ref>XML</ref></li>

   <li>A <code>DocumentType</code> node that has an external subset system identifier that contains
   both a U+0022 QUOTATION MARK (") and a U+0027 APOSTROPHE (') or that contains characters that are
   not matched by the XML <code data-x="">Char</code> production. <ref>XML</ref></li>

   <li>A node with a <!--prefix or--> local name containing a U+003A COLON (:).</li> <!--(prefixes
   can get adjusted, so this isn't an excuse) -->

   <li>A node with a <!--prefix or--> local name that does not match the XML <code
   data-x="xml-Name">Name</code> production. <ref>XML</ref></li> <!--(again, prefixes can get
   adjusted, so this isn't an excuse) -->

   <li>An <code>Attr</code> node with no namespace whose local name is the lowercase string "<code
   data-x="">xmlns</code>". <ref>XMLNS</ref></li>

   <li>An <code>Element</code> node with two or more attributes with the same local name and
   namespace.</li>

   <li>An <code>Attr</code> node, <code>Text</code> node, <code>Comment</code> node, or
   <code>ProcessingInstruction</code> node whose data contains characters that are not matched by
   the XML <code data-x="">Char</code> production. <ref>XML</ref></li>

   <li>A <code>Comment</code> node whose data contains two adjacent U+002D HYPHEN-MINUS characters
   (-) or ends with such a character.</li>

   <li>A <code>ProcessingInstruction</code> node whose target name is an <span>ASCII
   case-insensitive</span> match for the string "<code data-x="">xml</code>".</li>

   <li>A <code>ProcessingInstruction</code> node whose target name contains a U+003A COLON (:).</li>

   <li>A <code>ProcessingInstruction</code> node whose data contains the string "<code
   data-x="">?></code>".</li>
  </ul>

  <p class="note">These are the only ways to make a DOM unserialisable. The DOM enforces all the
  other XML constraints; for example, trying to append two elements to a <code>Document</code> node
  will throw a <span>"<code>HierarchyRequestError</code>"</span> <code>DOMException</code>.</p>



  <h3 id="parsing-xhtml-fragments">Parsing XML fragments</h3>

  <p>The <dfn>XML fragment parsing algorithm</dfn> given an <code>Element</code> node <var
  data-x="concept-frag-parse-context">context</var> and a string <var>input</var>, runs the
  following steps. They return a list of nodes.</p>

  <ol>
   <li><p>Create a new <span>XML parser</span>.</p></li>

   <li>
    <p><span>Feed the parser</span> just created the string corresponding to the start tag of <var
    data-x="concept-frag-parse-context">context</var>, declaring all the namespace prefixes that are
    in scope on that element in the DOM, as well as declaring the default namespace (if any) that is
    in scope on that element in the DOM.</p>

    <p>A namespace prefix is in scope if the DOM <code data-x="">lookupNamespaceURI()</code> method
    on the element would return a non-null value for that prefix.</p>

    <p>The default namespace is the namespace for which the DOM <code
    data-x="">isDefaultNamespace()</code> method on the element would return true.</p>

    <p class="note">No <code data-x="">DOCTYPE</code> is passed to the parser, and therefore no
    external subset is referenced, and therefore no entities will be recognized.</p>
   </li>

   <li><p><span>Feed the parser</span> just created the string <var>input</var>.</p></li>

   <li><p><span>Feed the parser</span> just created the string corresponding to the end tag of <var
   data-x="concept-frag-parse-context">context</var>.</p></li>

   <li><p>If there is an XML well-formedness or XML namespace well-formedness error, then throw a
   <span>"<code>SyntaxError</code>"</span> <code>DOMException</code>.</p></li>

   <li><p>If the <span>document element</span> of the resulting <code>Document</code> has any
   sibling nodes, then throw a <span>"<code>SyntaxError</code>"</span>
   <code>DOMException</code>.</p></li>
   <!-- This is technically redundant, but apparently it has gone wrong in the past:
        https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1443 -->

   <li><p>Return the resulting <code>Document</code> node's <span>document element</span>'s <span
   data-x="concept-tree-child">children</span>, in <span>tree order</span>.</p></li>
  </ol>

  </div>



  <div w-nodev>

  <h2 split-filename="rendering" id="rendering">Rendering</h2>

  <p><i>User agents are not required to present HTML documents in any particular way. However, this
  section provides a set of suggestions for rendering HTML documents that, if followed, are likely
  to lead to a user experience that closely resembles the experience intended by the documents'
  authors. So as to avoid confusion regarding the normativity of this section, "must" has not been
  used. Instead, the term "expected" is used to indicate behavior that will lead to this experience.
  For the purposes of conformance for user agents designated as <a href="#renderingUA">supporting
  the suggested default rendering</a>, the term "expected" in this section has the same conformance
  implications as "must".</i></p>


  <h3>Introduction</h3>

  <p>The suggestions in this section are generally expressed in CSS terms. User agents are expected
  to either support CSS, or translate from the CSS rules given in this section to approximations for
  other presentation mechanisms.</p>

  <p>In the absence of style-layer rules to the contrary (e.g. author style sheets), user agents are
  expected to render an element so that it conveys to the user the meaning that the element
  <span>represents</span>, as described by this specification.</p>

  <p>The suggestions in this section generally assume a visual output medium with a resolution of
  96dpi or greater, but HTML is intended to apply to multiple media (it is a
  <i>media-independent</i> language). User agent implementers are encouraged to adapt the
  suggestions in this section to their target media.</p>

  <hr>

  <p>An element is <dfn export>being rendered</dfn> if it has any associated CSS layout boxes, SVG
  layout boxes, or some equivalent in other styling languages.</p>

  <p class="note">Just being off-screen does not mean the element is not <span>being
  rendered</span>. The presence of the <code data-x="attr-hidden">hidden</code> attribute normally
  means the element is not <span>being rendered</span>, though this might be overridden by the style
  sheets.</p>

  <p class="note">The <span>fully active</span> state does not affect whether an element is
  <span>being rendered</span> or not. Even if a document is not <span>fully active</span> and not
  shown at all to the user, elements within it can still qualify as "being rendered".</p>

  <p>An element is said to <dfn>intersect the viewport</dfn> when it is <span>being rendered</span>
  and its associated CSS layout box intersects the <span>viewport</span>.</p>

  <p class="note">Similar to the <span>being rendered</span> state, elements in non-<span>fully
  active</span> documents can still <span>intersect the viewport</span>. The <span>viewport</span>
  is not shared between documents and might not always be shown to the user, so an element in a
  non-<span>fully active</span> document can still intersect the <span>viewport</span> associated
  with its document.</p>

  <p class="note">This specification does not define the precise timing for when the intersection is
  tested, but it is suggested that the timing match that of the Intersection Observer API. <ref>INTERSECTIONOBSERVER</ref></p>

  <p>An element is <dfn>delegating its rendering to its children</dfn> if it is not <span>being
  rendered</span> but its children (if any) could <span data-x="being rendered">be rendered</span>,
  as a result of CSS 'display: contents', or some equivalent in other styling languages.
  <ref>CSSDISPLAY</ref></p>

  <hr>

  <p>User agents that do not honor author-level CSS style sheets are nonetheless expected to act as
  if they applied the CSS rules given in these sections in a manner consistent with this
  specification and the relevant CSS and Unicode specifications. <ref>CSS</ref> <ref>UNICODE</ref>
  <ref>BIDI</ref></p>

  <p class="note">This is especially important for issues relating to the <span>'display'</span>,
  <span>'unicode-bidi'</span>, and <span>'direction'</span> properties.</p>



  <h3>The CSS user agent style sheet and presentational hints</h3>

  <p>The CSS rules given in these subsections are, except where otherwise specified, expected to be
  used as part of the user-agent level style sheet defaults for all documents that contain
  <span>HTML elements</span>.</p>

  <p>Some rules are intended for the author-level zero-specificity presentational hints part of the
  CSS cascade; these are explicitly called out as <dfn data-lt="presentational hint"
  export>presentational hints</dfn>.</p>

  <hr>

  <p>When the text below says that an attribute <var>attribute</var> on an element
  <var>element</var> <dfn>maps to the pixel length property</dfn> (or properties)
  <var>properties</var>, it means that if <var>element</var> has an attribute <var>attribute</var>
  set, and parsing that attribute's value using the <span>rules for parsing non-negative
  integers</span> doesn't generate an error, then the user agent is expected to use the parsed value
  as a pixel length for a <span data-x="presentational hints">presentational hint</span> for
  <var>properties</var>.</p>

  <p>When the text below says that an attribute <var>attribute</var> on an element
  <var>element</var> <dfn>maps to the dimension property</dfn> (or properties)
  <var>properties</var>, it means that if <var>element</var> has an attribute <var>attribute</var>
  set, and parsing that attribute's value using the <span>rules for parsing dimension values</span>
  doesn't generate an error, then the user agent is expected to use the parsed dimension as the
  value for a <span data-x="presentational hints">presentational hint</span> for
  <var>properties</var>, with the value given as a pixel length if the dimension was a length, and
  with the value given as a percentage if the dimension was a percentage.</p>

  <p>When the text below says that an attribute <var>attribute</var> on an element
  <var>element</var> <dfn>maps to the dimension property (ignoring zero)</dfn> (or properties)
  <var>properties</var>, it means that if <var>element</var> has an attribute <var>attribute</var>
  set, and parsing that attribute's value using the <span>rules for parsing nonzero dimension
  values</span> doesn't generate an error, then the user agent is expected to use the parsed
  dimension as the value for a <span data-x="presentational hints">presentational hint</span> for
  <var>properties</var>, with the value given as a pixel length if the dimension was a length, and
  with the value given as a percentage if the dimension was a percentage.</p>

  <p>When the text below says that a pair of attributes <var>w</var> and <var>h</var> on an
  element <var>element</var> <dfn>map to the aspect-ratio property</dfn>, it means that if
  <var>element</var> has both attributes <var>w</var> and <var>h</var>, and parsing those
  attributes' values using the <span>rules for parsing non-negative integers</span> doesn't
  generate an error for either, then the user agent is expected to use the parsed integers as a
  <span data-x="presentational hints">presentational hint</span> for the <span>'aspect-ratio'</span>
  property of the form <code data-x="">auto <var>w</var> / <var>h</var></code>.</p>

  <p>When the text below says that a pair of attributes <var>w</var> and <var>h</var> on an
  element <var>element</var> <dfn>map to the aspect-ratio property (using dimension rules)</dfn>, it
  means that if <var>element</var> has both attributes <var>w</var> and <var>h</var>, and parsing
  those attributes' values using the <span>rules for parsing dimension values</span> doesn't
  generate an error or return a percentage for either, then the user agent is expected to use the
  parsed dimensions as a <span data-x="presentational hints">presentational hint</span> for the
  <span>'aspect-ratio'</span> property of the form <code data-x="">auto
  <var>w</var> / <var>h</var></code>.</p>

  <p>When a user agent is to <dfn>align descendants</dfn> of a node, the user agent is expected to
  align only those descendants that have both their <span>'margin-inline-start'</span> and
  <span>'margin-inline-end'</span> properties computing to a value other than 'auto', that are
  over-constrained and that have one of those two margins with a <span>used value</span> forced to a
  greater value, and that do not themselves have an applicable <code data-x="">align</code>
  attribute. When multiple elements are to <span data-x="align descendants">align</span> a
  particular descendant, the most deeply nested such element is expected to override the others.
  Aligned elements are expected to be aligned by having the <span data-x="used value">used
  values</span> of their margins on the <span>line-left</span> and <span>line-right</span> sides be
  set accordingly. <ref>CSSLOGICAL</ref> <ref>CSSWM</ref></p>



  <h3>Non-replaced elements</h3>


  <h4>Hidden elements</h4>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

area, base, basefont, datalist, head, link, meta, noembed,
noframes, param, rp, script, style, template, title {
  display: none;
}

<span id="hiddenCSS" data-x="">[hidden]:not([hidden=until-found i]):not(embed)</span> {
  display: none;
}

[hidden=until-found i]:not(embed) {
  content-visibility: hidden;
}

embed[hidden] { display: inline; height: 0; width: 0; } <!-- because for legacy reasons it still needs to instantiate the plugin -->

input[type=hidden i] { display: none !important; }

@media (scripting) {
  noscript { display: none !important; }
}</code></pre>


  <h4>The page</h4>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

html, body { display: block; }</code></pre>


  <!-- body { margin: 8px; } -->

  <p>For each property in the table below, given a <code>body</code> element, the first attribute
  that exists <span>maps to the pixel length property</span> on the <code>body</code> element. If
  none of the attributes for a property are found, or if the value of the attribute that was found
  cannot be parsed successfully, then a default value of 8px is expected to be used for that
  property instead.</p>

  <table>
   <thead>
    <tr>
     <th>Property
     <th>Source
   <tbody>
    <tr>
     <td rowspan="3"><span>'margin-top'</span>
     <td>The <code>body</code> element's <code data-x="attr-body-marginheight">marginheight</code> attribute
    <tr>
     <td>The <code>body</code> element's <code data-x="attr-body-topmargin">topmargin</code> attribute
    <tr>
     <td>The <code>body</code> element's <span>container frame element</span>'s <code data-x="attr-iframe-marginheight">marginheight</code> attribute
   <tbody>
    <tr>
     <td rowspan="3"><span>'margin-right'</span>
     <td>The <code>body</code> element's <code data-x="attr-body-marginwidth">marginwidth</code> attribute
    <tr>
     <td>The <code>body</code> element's <code data-x="attr-body-rightmargin">rightmargin</code> attribute
    <tr>
     <td>The <code>body</code> element's <span>container frame element</span>'s <code data-x="attr-iframe-marginwidth">marginwidth</code> attribute
   <tbody>
    <tr>
     <td rowspan="3"><span>'margin-bottom'</span>
     <td>The <code>body</code> element's <code data-x="attr-body-marginheight">marginheight</code> attribute
    <tr>
     <td>The <code>body</code> element's <code data-x="attr-body-bottommargin">bottommargin</code> attribute
    <tr>
     <td>The <code>body</code> element's <span>container frame element</span>'s <code data-x="attr-iframe-marginheight">marginheight</code> attribute
   <tbody>
    <tr>
     <td rowspan="3"><span>'margin-left'</span>
     <td>The <code>body</code> element's <code data-x="attr-body-marginwidth">marginwidth</code> attribute
    <tr>
     <td>The <code>body</code> element's <code data-x="attr-body-leftmargin">leftmargin</code> attribute
    <tr>
     <td>The <code>body</code> element's <span>container frame element</span>'s <code data-x="attr-iframe-marginwidth">marginwidth</code> attribute
  </table>

  <p>If the <code>body</code> element's <span>node document</span>'s <span>node navigable</span> is
  a <span>child navigable</span>, and the <span data-x="nav-container">container</span> of that
  <span>navigable</span> is a <code>frame</code> or <code>iframe</code> element, then the
  <dfn>container frame element</dfn> of the <code>body</code> element is that <code>frame</code> or
  <code>iframe</code> element. Otherwise, there is no <span>container frame element</span>.</p>

  <p class="warning">The above requirements imply that a page can change the margins of another page
  (including one from another <span>origin</span>) using, for example, an <code>iframe</code>. This
  is potentially a security risk, as it might in some cases allow an attack to contrive a situation
  in which a page is rendered not as the author intended, possibly for the purposes of phishing or
  otherwise misleading the user.</p>

  <hr>

  <p>If a <code>Document</code>'s <span>node navigable</span> is a <span>child navigable</span>,
  then it is expected to be positioned and sized to fit inside the <span>content box</span> of the
  <span data-x="nav-container">container</span> of that <span>navigable</span>. If the <span
  data-x="nav-container">container</span> is not <span>being rendered</span>, the
  <span>navigable</span> is expected to have a <span>viewport</span> with zero width and zero
  height.</p>

  <p>If a <code>Document</code>'s <span>node navigable</span> is a <span>child navigable</span>,
  the <span data-x="nav-container">container</span> of that <span>navigable</span> is a
  <code>frame</code> or <code>iframe</code> element, that element has a <code
  data-x="">scrolling</code> attribute, and that attribute's value is an <span>ASCII
  case-insensitive</span> match for the string "<code data-x="">off</code>", "<code
  data-x="">noscroll</code>", or "<code data-x="">no</code>", then the user agent is expected to
  prevent any scrollbars from being shown for the <span>viewport</span> of the
  <code>Document</code>'s <span>node navigable</span>, regardless of the <span>'overflow'</span>
  property that applies to that <span>viewport</span>.</p>

  <hr>

  <p>When a <code>body</code> element has a <code data-x="attr-background">background</code>
  attribute set to a non-empty value, the new value is expected to be <span
  data-x="encoding-parsing-and-serializing a URL">encoding-parsed-and-serialized</span> relative to
  the element's <span>node document</span>, and if that does not return failure, the user agent is
  expected to treat the attribute as a <span data-x="presentational hints">presentational
  hint</span> setting the element's <span>'background-image'</span> property to the return
  value.</p>

  <p>When a <code>body</code> element has a <code data-x="attr-body-bgcolor">bgcolor</code>
  attribute set, the new value is expected to be parsed using the <span>rules for parsing a legacy
  color value</span>, and if that does not return failure, the user agent is expected to treat the
  attribute as a <span data-x="presentational hints">presentational hint</span> setting the
  element's <span>'background-color'</span> property to the resulting color.</p>

  <p>When a <code>body</code> element has a <code data-x="attr-body-text">text</code> attribute, its
  value is expected to be parsed using the <span>rules for parsing a legacy color value</span>, and
  if that does not return failure, the user agent is expected to treat the attribute as a <span
  data-x="presentational hints">presentational hint</span> setting the element's
  <span>'color'</span> property to the resulting color.</p>

  <p>When a <code>body</code> element has a <code data-x="attr-body-link">link</code> attribute, its
  value is expected to be parsed using the <span>rules for parsing a legacy color value</span>, and
  if that does not return failure, the user agent is expected to treat the attribute as a <span
  data-x="presentational hints">presentational hint</span> setting the <span>'color'</span> property
  of any element in the <code>Document</code> matching the <code data-x="selector-link">:link</code>
  <span>pseudo-class</span> to the resulting color.</p>

  <p>When a <code>body</code> element has a <code data-x="attr-body-vlink">vlink</code> attribute,
  its value is expected to be parsed using the <span>rules for parsing a legacy color value</span>,
  and if that does not return failure, the user agent is expected to treat the attribute as a <span
  data-x="presentational hints">presentational hint</span> setting the <span>'color'</span> property
  of any element in the <code>Document</code> matching the <code
  data-x="selector-visited">:visited</code> <span>pseudo-class</span> to the resulting color.</p>

  <p>When a <code>body</code> element has an <code data-x="attr-body-alink">alink</code> attribute,
  its value is expected to be parsed using the <span>rules for parsing a legacy color value</span>,
  and if that does not return failure, the user agent is expected to treat the attribute as a <span
  data-x="presentational hints">presentational hint</span> setting the <span>'color'</span> property
  of any element in the <code>Document</code> matching the
  <code data-x="selector-active">:active</code> <span>pseudo-class</span> and either the <code
  data-x="selector-link">:link</code> <span>pseudo-class</span> or the <code
  data-x="selector-visited">:visited</code> <span>pseudo-class</span> to the resulting color.</p>



  <h4>Flow content</h4>

  <!-- del, ins, and map are inline. -->

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

address, blockquote, center, dialog, div, figure, figcaption, footer, form,
header, hr, legend, listing, main, p, plaintext, pre, search, xmp {
  display: block;<!-- see also unicode-bidi: isolate rules-->
}

blockquote, figure, listing, p, plaintext, pre, xmp {
  margin-block: 1em;
}

blockquote, figure { margin-inline: 40px; }

address { font-style: italic; }
listing, plaintext, pre, xmp {
  font-family: monospace; white-space: pre;
}

dialog:not([open]) { display: none; }
dialog {
  position: absolute;
  inset-inline-start: 0; inset-inline-end: 0;
  width: fit-content;
  height: fit-content;
  margin: auto;
  border: solid;
  padding: 1em;
  background-color: Canvas;
  color: CanvasText;
}
dialog:modal {
  position: fixed;
  overflow: auto;
  inset-block: 0;
  max-width: calc(100% - 6px - 2em);
  max-height: calc(100% - 6px - 2em);
}
dialog::backdrop {
  background: rgba(0,0,0,0.1);
}

[popover]:not(:popover-open):not(dialog[open]) {
  display:none;
}

dialog:popover-open {
  display:block;
}

[popover] {
  position: fixed;
  inset: 0;
  width: fit-content;
  height: fit-content;
  margin: auto;
  border: solid;
  padding: 0.25em;
  overflow: auto;
  color: CanvasText;
  background-color: Canvas;
}

:popover-open::backdrop {
  position: fixed;
  inset: 0;
  pointer-events: none !important;
  background-color: transparent;
}

slot {
  display: contents;
}</code></pre>

  <p>The following rules are also expected to apply, as <span>presentational hints</span>:</p>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

pre[wrap] { white-space: pre-wrap; }</code></pre>

  <p>In <span>quirks mode</span>, the following rules are also expected to apply:</p>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

form { margin-block-end: 1em; }</code></pre>

  <hr>

  <p>The <code>center</code> element, and the <code>div</code> element when it has an <code
  data-x="attr-div-align">align</code> attribute whose value is an <span>ASCII
  case-insensitive</span> match for either the string "<code data-x="">center</code>" or the string
  "<code data-x="">middle</code>", are expected to center text within themselves, as if they had
  their <span>'text-align'</span> property set to 'center' in a <span data-x="presentational
  hints">presentational hint</span>, and to <span>align descendants</span> to the center.</p>

  <p>The <code>div</code> element, when it has an <code data-x="attr-div-align">align</code>
  attribute whose value is an <span>ASCII case-insensitive</span> match for the string "<code
  data-x="">left</code>", is expected to left-align text within itself, as if it had its
  <span>'text-align'</span> property set to 'left' in a <span data-x="presentational
  hints">presentational hint</span>, and to <span>align descendants</span> to the left.</p>

  <p>The <code>div</code> element, when it has an <code data-x="attr-div-align">align</code>
  attribute whose value is an <span>ASCII case-insensitive</span> match for the string "<code
  data-x="">right</code>", is expected to right-align text within itself, as if it had its
  <span>'text-align'</span> property set to 'right' in a <span data-x="presentational
  hints">presentational hint</span>, and to <span>align descendants</span> to the right.</p>

  <p>The <code>div</code> element, when it has an <code data-x="attr-div-align">align</code>
  attribute whose value is an <span>ASCII case-insensitive</span> match for the string "<code
  data-x="">justify</code>", is expected to full-justify text within itself, as if it had its
  <span>'text-align'</span> property set to 'justify' in a <span data-x="presentational
  hints">presentational hint</span>, and to <span>align descendants</span> to the left.</p>

  <h4>Phrasing content</h4>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

cite, dfn, em, i, var { font-style: italic; }
b, strong { font-weight: bolder; }
code, kbd, samp, tt { font-family: monospace; }
big { font-size: larger; }
small { font-size: smaller; }

sub { vertical-align: sub; }
sup { vertical-align: super; }
sub, sup { line-height: normal; font-size: smaller; }

ruby { display: ruby; }
rt { display: ruby-text; }

:link { color: #0000EE; }
:visited { color: #551A8B; }
:link:active, :visited:active { color: #FF0000; }
:link, :visited { text-decoration: underline; cursor: pointer; }

:focus-visible { outline: auto; }

mark { background: yellow; color: black; } /* this color is just a suggestion and can be changed based on implementation feedback */

abbr[title], acronym[title] { text-decoration: dotted underline; }<!-- CSS3 https://drafts.csswg.org/css3-text/#text-decoration-style -->
ins, u { text-decoration: underline; }
del, s, strike { text-decoration: line-through; }

q::before { content: open-quote; }
q::after { content: close-quote; }

<span data-x="" id="br-wbr-content">br { display-outside: newline; } /* <a href="#bidi-rendering">this also has bidi implications</a> */
nobr { white-space: nowrap; }
wbr { display-outside: break-opportunity; } /* <a href="#bidi-rendering">this also has bidi implications</a> */
nobr wbr { white-space: normal; }</span></code></pre>

  <p>The following rules are also expected to apply, as
  <span>presentational hints</span>:</p>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

br[clear=left i] { clear: left; }
br[clear=right i] { clear: right; }
br[clear=all i], br[clear=both i] { clear: both; }</code></pre>

  <p>For the purposes of the CSS ruby model, runs of children of <code>ruby</code> elements that are
  not <code>rt</code> or <code>rp</code> elements are expected to be wrapped in anonymous boxes
  whose <span>'display'</span> property has the value <span>'ruby-base'</span>. <ref>CSSRUBY</ref></p>

  <p>When a particular part of a ruby has more than one annotation, the annotations should be
  distributed on both sides of the base text so as to minimize the stacking of ruby annotations on
  one side.</p>

  <p class="note">When it becomes possible to do so, the preceding requirement will be updated to be
  expressed in terms of CSS ruby. (Currently, CSS ruby does not handle nested <code>ruby</code>
  elements or multiple sequential <code>rt</code> elements, which is how this semantic is
  expressed.)</p>

  <p>User agents that do not support correct ruby rendering are expected to render parentheses
  around the text of <code>rt</code> elements in the absence of <code>rp</code> elements.</p>

  <hr>

  <p>User agents are expected to support the <span>'clear'</span> property on inline elements (in
  order to render <code>br</code> elements with <code data-x="attr-br-clear">clear</code>
  attributes) in the manner described in the non-normative note to this effect in <cite>CSS</cite>.
  </p> <!-- section 9.5.2 of CSS2.1 -->

  <p>The initial value for the <span>'color'</span> property is expected to be black. The initial
  value for the <span>'background-color'</span> property is expected to be 'transparent'. The
  canvas's background is expected to be white.</p>

  <hr>

  <p>When a <code>font</code> element has a <code data-x="attr-font-color" undefined>color</code>
  attribute, its value is expected to be parsed using the <span>rules for parsing a legacy color
  value</span>, and if that does not return failure, the user agent is expected to treat the
  attribute as a <span data-x="presentational hints">presentational hint</span> setting the
  element's <span>'color'</span> property to the resulting color.</p>

  <p>When a <code>font</code> element has a <code data-x="attr-font-face" undefined>face</code>
  attribute, the user agent is expected to treat the attribute as a <span data-x="presentational
  hints">presentational hint</span> setting the element's <span>'font-family'</span> property to the
  attribute's value.</p>

  <!-- (Apparently only IE supports this?) (Note: if you add this back, make sure to define which of
  'size' vs 'pointsize' wins.)

  <p>When a <code>font</code> element has a <code data-x="attr-font-pointsize"
  undefined>pointsize</code> attribute, the user agent is expected to parse that attribute's value
  using the <span>rules for parsing non-negative integers</span>, and if this doesn't generate an
  error, then the user agent is expected to use the parsed value as a <em>point</em> length for a
  <span data-x="presentational hints">presentational hint</span> for the <span>'font-size'</span>
  property on the element.</p>
  -->

  <p>When a <code>font</code> element has a <code data-x="attr-font-size" undefined>size</code>
  attribute, the user agent is expected to use the following steps, known as the <dfn>rules for
  parsing a legacy font size</dfn>, to treat the attribute as a <span data-x="presentational
  hints">presentational hint</span> setting the element's <span>'font-size'</span> property:</p>

  <ol>
   <li><p>Let <var>input</var> be the attribute's value.</p></li>

   <li><p>Let <var>position</var> be a pointer into <var>input</var>, initially pointing at the
   start of the string.</p></li>

   <li><p><span>Skip ASCII whitespace</span> within <var>input</var> given
   <var>position</var>.</p></li>

   <li><p>If <var>position</var> is past the end of <var>input</var>, there is no <span
   data-x="presentational hints">presentational hint</span>. Return.</p></li>

   <li><p>If the character at <var>position</var> is a U+002B PLUS SIGN character (+), then let
   <var>mode</var> be <i>relative-plus</i>, and advance <var>position</var> to the next character.
   Otherwise, if the character at <var>position</var> is a U+002D HYPHEN-MINUS character (-), then
   let <var>mode</var> be <i>relative-minus</i>, and advance <var>position</var> to the next
   character. Otherwise, let <var>mode</var> be <i>absolute</i>.</p></li>

   <li><p><span>Collect a sequence of code points</span> that are <span>ASCII digits</span> from
   <var>input</var> given <var>position</var>, and let the resulting sequence be
   <var>digits</var>.</p></li>

   <li><p>If <var>digits</var> is the empty string, there is no <span data-x="presentational
   hints">presentational hint</span>. Return.</p></li>

   <li><p>Interpret <var>digits</var> as a base-ten integer. Let <var>value</var> be the resulting
   number.</p></li>

   <li>
    <!-- basefont support would go here, but we removed it -->

    <p>If <var>mode</var> is <i>relative-plus</i>, then increment <var>value</var> by 3. If
    <var>mode</var> is <i>relative-minus</i>, then let <var>value</var> be the result of subtracting
    <var>value</var> from 3.</p>
   </li>

   <li><p>If <var>value</var> is greater than 7, let it be 7.</p></li>

   <li><p>If <var>value</var> is less than 1, let it be 1.</p></li>

   <li>
    <p>Set <span>'font-size'</span> to the keyword corresponding to the value of <var>value</var>
    according to the following table:</p>

    <table>
     <thead>
      <tr>
       <th><var>value</var>
       <th><span>'font-size'</span> keyword
     <tbody>
      <tr>
       <td>1
       <td>'x-small'
      <tr>
       <td>2
       <td>'small'
      <tr>
       <td>3
       <td>'medium'
      <tr>
       <td>4
       <td>'large'
      <tr>
       <td>5
       <td>'x-large'
      <tr>
       <td>6
       <td>'xx-large'
      <tr>
       <td>7
       <td>'xxx-large'
    </table>
   </li>
  </ol>


  <h4 id="bidi-rendering">Bidirectional text</h4>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

[dir]:dir(ltr), bdi:dir(ltr), input[type=tel i]:dir(ltr) { direction: ltr; }
[dir]:dir(rtl), bdi:dir(rtl) { direction: rtl; }

address, blockquote, center, div, figure, figcaption, footer, form, header, hr,
legend, listing, main, p, plaintext, pre, summary, xmp, article, aside, h1, h2,
h3, h4, h5, h6, hgroup, nav, section, search, table, caption, colgroup, col,
thead, tbody, tfoot, tr, td, th, dir, dd, dl, dt, menu, ol, ul, li, bdi, output,
[dir=ltr i], [dir=rtl i], [dir=auto i] {
  unicode-bidi: isolate; <!-- anything that's similar to display:block, plus <bdi>, <output>, and dir="" -->
}

bdo, bdo[dir] { unicode-bidi: isolate-override; } <!-- bdo[dir] rule is to override the otherwise higher-specificity attribute selectors in the previous rule -->

input[dir=auto i]:is([type=search i], [type=tel i], [type=url i],
[type=email i]), textarea[dir=auto i], pre[dir=auto i] {
  unicode-bidi: plaintext;
}
/* see prose for input elements whose type attribute is in the Text state */

/* the <a href="#br-wbr-content">rules setting the 'content' property</a> on <code>br</code> and <code>wbr</code> elements also has bidi implications */</code></pre>

  <p>When an <code>input</code> element's <code data-x="attr-dir">dir</code> attribute is in the
  <span data-x="attr-dir-auto">auto</span> state and its <code data-x="attr-input-type">type</code>
  attribute is in the <span data-x="attr-input-type-text">Text</span> state, then the user agent is
  expected to act as if it had a user-agent-level style sheet rule setting the
  <span>'unicode-bidi'</span> property to 'plaintext'.</p>

  <p>Input fields (i.e. <code>textarea</code> elements, and <code>input</code> elements when their
  <code data-x="attr-input-type">type</code> attribute is in the <span
  data-x="attr-input-type-text">Text</span>, <span data-x="attr-input-type-search">Search</span>,
  <span data-x="attr-input-type-tel">Telephone</span>, <span data-x="attr-input-type-url">URL</span>,
  or <span data-x="attr-input-type-email">Email</span> state) are expected to present an editing
  user interface with a directionality that matches the element's <span>'direction'</span>
  property.</p>

  <p>When the document's character encoding is <span>ISO-8859-8</span>, the following rules are
  additionally expected to apply, following those above: <ref>ENCODING</ref></p>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

address, blockquote, center, div, figure, figcaption, footer, form, header, hr,
legend, listing, main, p, plaintext, pre, summary, xmp, article, aside, h1, h2,
h3, h4, h5, h6, hgroup, nav, section, search, table, caption, colgroup, col,
thead, tbody, tfoot, tr, td, th, dir, dd, dl, dt, menu, ol, ul, li, [dir=ltr i],
[dir=rtl i], [dir=auto i], *|* {
  unicode-bidi: bidi-override;
}
input:not([type=submit i]):not([type=reset i]):not([type=button i]),
textarea {
  unicode-bidi: normal;
}</code></pre>


  <h4>Sections and headings</h4>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

article, aside, h1, h2, h3, h4, h5, h6, hgroup, nav, section {
  display: block;<!-- see also unicode-bidi: isolate rules-->
}

h1 { margin-block: 0.67em; font-size: 2.00em; font-weight: bold; }
h2 { margin-block: 0.83em; font-size: 1.50em; font-weight: bold; }
h3 { margin-block: 1.00em; font-size: 1.17em; font-weight: bold; }
h4 { margin-block: 1.33em; font-size: 1.00em; font-weight: bold; }
h5 { margin-block: 1.67em; font-size: 0.83em; font-weight: bold; }
h6 { margin-block: 2.33em; font-size: 0.67em; font-weight: bold; }</code></pre>

  <p>In the following CSS block, <var>x</var> is shorthand for the following selector:
  <code data-x="">:is(article, aside, nav, section)</code></p>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

<var>x</var> h1 { margin-block: 0.83em; font-size: 1.50em; }
<var>x</var> <var>x</var> h1 { margin-block: 1.00em; font-size: 1.17em; }
<var>x</var> <var>x</var> <var>x</var> h1 { margin-block: 1.33em; font-size: 1.00em; }
<var>x</var> <var>x</var> <var>x</var> <var>x</var> h1 { margin-block: 1.67em; font-size: 0.83em; }
<var>x</var> <var>x</var> <var>x</var> <var>x</var> <var>x</var> h1 { margin-block: 2.33em; font-size: 0.67em; }</code></pre>

  <p class="note">The shorthand is used to keep this block at least mildly readable.</p>



  <h4>Lists</h4>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

dir, dd, dl, dt, menu, ol, ul { display: block; }<!-- see also unicode-bidi:isolate rules -->
li { display: list-item; text-align: match-parent; }<!-- see also unicode-bidi:isolate rules -->

dir, dl, menu, ol, ul { margin-block: 1em; }

:is(dir, dl, menu, ol, ul) :is(dir, dl, menu, ol, ul) {
  margin-block: 0;
}

dd { margin-inline-start: 40px; }
dir, menu, ol, ul { padding-inline-start: 40px; }

ol, ul, menu { counter-reset: list-item; }
ol { list-style-type: decimal; }

dir, menu, ul {
  list-style-type: disc;
}
:is(dir, menu, ol, ul) :is(dir, menu, ul) {
  list-style-type: circle;
}
:is(dir, menu, ol, ul) :is(dir, menu, ol, ul) :is(dir, menu, ul) {
  list-style-type: square;
}</code></pre>

  <p id="decohints">The following rules are also expected to apply, as <span>presentational
  hints</span>:</p>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

ol[type="1"], li[type="1"] { list-style-type: decimal; }
ol[type=a <!--grammar-check-override-->s], li[type=a <!--grammar-check-override-->s] { list-style-type: lower-alpha; }
ol[type=A <!--grammar-check-override-->s], li[type=A <!--grammar-check-override-->s] { list-style-type: upper-alpha; }
ol[type=i s], li[type=i s] { list-style-type: lower-roman; }
ol[type=I s], li[type=I s] { list-style-type: upper-roman; }
ul[type=none i], li[type=none i] { list-style-type: none; }
ul[type=disc i], li[type=disc i] { list-style-type: disc; }
ul[type=circle i], li[type=circle i] { list-style-type: circle; }
ul[type=square i], li[type=square i] { list-style-type: square; }</code></pre>

  <p>In <span>quirks mode</span>, the following rules are also expected to apply:</p>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

li { list-style-position: inside; }
li :is(dir, menu, ol, ul) { list-style-position: outside; }
:is(dir, menu, ol, ul) :is(dir, menu, ol, ul, li) { list-style-position: unset; }</code></pre>

  <p>When rendering <code>li</code> elements, non-CSS user agents are expected to use the
  <span>ordinal value</span> of the <code>li</code> element to render the counter in the list item
  marker.</p>

  <p id="css-list-rendering">For CSS user agents, some aspects of rendering <span
  data-x="css-list-item">list items</span> are defined by the <cite>CSS Lists</cite> specification.
  Additionally, the following attribute mappings are expected to apply: <ref>CSSLISTS</ref></p>

  <p>When an <code>li</code> element has a <code
  data-x="attr-li-value">value</code> attribute, and parsing that attribute's value using the
  <span>rules for parsing integers</span> doesn't generate an error, the user agent is expected to
  use the parsed value <var>value</var> as a <span data-x="presentational hints">presentational
  hint</span> for the <span>'counter-set'</span> property of the form <code data-x="">list-item
  <var>value</var></code>.</p>

  <p>When an <code>ol</code> element has a <code data-x="attr-ol-start">start</code>
  attribute or a <code data-x="attr-ol-reversed">reversed</code> attribute, or both, the user agent
  is expected to use the following steps to treat the attributes as a <span data-x="presentational
  hints">presentational hint</span> for the <span>'counter-reset'</span> property:</p>

  <ol>
   <li><p>Let <var>value</var> be null.</p></li>

   <li><p>If the element has a <code data-x="attr-ol-start">start</code> attribute, then set
   <var>value</var> to the result of parsing the attribute's value using the <span>rules for
   parsing integers</span>.</p></li>

   <li>
    <p>If the element has a <code data-x="attr-ol-reversed">reversed</code> attribute, then:</p>

    <ol>
     <li><p>If <var>value</var> is an integer, then increment <var>value</var> by 1 and return
     <code data-x="">reversed(list-item) <var>value</var></code>.</p></li>

     <li>
      <p>Otherwise, return <code data-x="">reversed(list-item)</code>.</p>

      <p class="note">Either the <code data-x="attr-ol-start">start</code> attribute was absent, or
      parsing its value resulted in an error.</p>
     </li>
    </ol>
   </li>

   <li>
    <p>Otherwise:</p>

    <ol>
     <li><p>If <var>value</var> is an integer, then decrement <var>value</var> by 1 and return
     <code data-x="">list-item <var>value</var></code>.</p></li>

     <li><p>Otherwise, there is no <span data-x="presentational hints">presentational
     hint</span>.</p></li>
    </ol>
   </li>
  </ol>


  <h4>Tables</h4>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

table { display: table; }<!-- see also unicode-bidi:isolate rules -->
caption { display: table-caption; }<!-- see also unicode-bidi:isolate rules -->
colgroup, colgroup[hidden] { display: table-column-group; }<!-- see also unicode-bidi:isolate rules -->
col, col[hidden] { display: table-column; }<!-- see also unicode-bidi:isolate rules -->
thead, thead[hidden] { display: table-header-group; }<!-- see also unicode-bidi:isolate rules -->
tbody, tbody[hidden] { display: table-row-group; }<!-- see also unicode-bidi:isolate rules -->
tfoot, tfoot[hidden] { display: table-footer-group; }<!-- see also unicode-bidi:isolate rules -->
tr, tr[hidden] { display: table-row; }<!-- see also unicode-bidi:isolate rules -->
td, th { display: table-cell; }<!-- see also unicode-bidi:isolate rules -->

colgroup[hidden], col[hidden], thead[hidden], tbody[hidden],
tfoot[hidden], tr[hidden] {
  visibility: collapse;
}

table {
  box-sizing: border-box;
  border-spacing: 2px;
  border-collapse: separate;
  text-indent: initial;
}
td, th { padding: 1px; }
th { font-weight: bold; }

caption { text-align: center; }
thead, tbody, tfoot, table > tr { vertical-align: middle; }
tr, td, th { vertical-align: inherit; }

thead, tbody, tfoot, tr { border-color: inherit; }
table[rules=none i], table[rules=groups i], table[rules=rows i],
table[rules=cols i], table[rules=all i], table[frame=void i],
table[frame=above i], table[frame=below i], table[frame=hsides i],
table[frame=lhs i], table[frame=rhs i], table[frame=vsides i],
table[frame=box i], table[frame=border i],
table[rules=none i] > tr > td, table[rules=none i] > tr > th,
table[rules=groups i] > tr > td, table[rules=groups i] > tr > th,
table[rules=rows i] > tr > td, table[rules=rows i] > tr > th,
table[rules=cols i] > tr > td, table[rules=cols i] > tr > th,
table[rules=all i] > tr > td, table[rules=all i] > tr > th,
table[rules=none i] > thead > tr > td, table[rules=none i] > thead > tr > th,
table[rules=groups i] > thead > tr > td, table[rules=groups i] > thead > tr > th,
table[rules=rows i] > thead > tr > td, table[rules=rows i] > thead > tr > th,
table[rules=cols i] > thead > tr > td, table[rules=cols i] > thead > tr > th,
table[rules=all i] > thead > tr > td, table[rules=all i] > thead > tr > th,
table[rules=none i] > tbody > tr > td, table[rules=none i] > tbody > tr > th,
table[rules=groups i] > tbody > tr > td, table[rules=groups i] > tbody > tr > th,
table[rules=rows i] > tbody > tr > td, table[rules=rows i] > tbody > tr > th,
table[rules=cols i] > tbody > tr > td, table[rules=cols i] > tbody > tr > th,
table[rules=all i] > tbody > tr > td, table[rules=all i] > tbody > tr > th,
table[rules=none i] > tfoot > tr > td, table[rules=none i] > tfoot > tr > th,
table[rules=groups i] > tfoot > tr > td, table[rules=groups i] > tfoot > tr > th,
table[rules=rows i] > tfoot > tr > td, table[rules=rows i] > tfoot > tr > th,
table[rules=cols i] > tfoot > tr > td, table[rules=cols i] > tfoot > tr > th,
table[rules=all i] > tfoot > tr > td, table[rules=all i] > tfoot > tr > th {
  border-color: black;
}</code></pre>

  <p>The following rules are also expected to apply, as <span>presentational hints</span>:</p>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

table[align=left i] { float: left; }
table[align=right i] { float: right; }
table[align=center i] { margin-inline: auto; }
thead[align=absmiddle i], tbody[align=absmiddle i], tfoot[align=absmiddle i],
tr[align=absmiddle i], td[align=absmiddle i], th[align=absmiddle i] {
  text-align: center;
}

caption[align=bottom i] { caption-side: bottom; }
p[align=left i], h1[align=left i], h2[align=left i], h3[align=left i],
h4[align=left i], h5[align=left i], h6[align=left i] {
  text-align: left;
}
p[align=right i], h1[align=right i], h2[align=right i], h3[align=right i],
h4[align=right i], h5[align=right i], h6[align=right i] {
  text-align: right;
}
p[align=center i], h1[align=center i], h2[align=center i], h3[align=center i],
h4[align=center i], h5[align=center i], h6[align=center i] {
  text-align: center;
}
p[align=justify i], h1[align=justify i], h2[align=justify i], h3[align=justify i],
h4[align=justify i], h5[align=justify i], h6[align=justify i] {
  text-align: justify;
}
thead[valign=top i], tbody[valign=top i], tfoot[valign=top i],
tr[valign=top i], td[valign=top i], th[valign=top i] {
  vertical-align: top;
}
thead[valign=middle i], tbody[valign=middle i], tfoot[valign=middle i],
tr[valign=middle i], td[valign=middle i], th[valign=middle i] {
  vertical-align: middle;
}
thead[valign=bottom i], tbody[valign=bottom i], tfoot[valign=bottom i],
tr[valign=bottom i], td[valign=bottom i], th[valign=bottom i] {
  vertical-align: bottom;
}
thead[valign=baseline i], tbody[valign=baseline i], tfoot[valign=baseline i],
tr[valign=baseline i], td[valign=baseline i], th[valign=baseline i] {
  vertical-align: baseline;
}

td[nowrap], th[nowrap] { white-space: nowrap; }

table[rules=none i], table[rules=groups i], table[rules=rows i],
table[rules=cols i], table[rules=all i] {
  border-style: hidden;
  border-collapse: collapse;
}
table[border] { border-style: outset; } /* <span>only if border is not equivalent to zero</span> */
table[frame=void i] { border-style: hidden; }
table[frame=above i] { border-style: outset hidden hidden hidden; }
table[frame=below i] { border-style: hidden hidden outset hidden; }
table[frame=hsides i] { border-style: outset hidden outset hidden; }
table[frame=lhs i] { border-style: hidden hidden hidden outset; }
table[frame=rhs i] { border-style: hidden outset hidden hidden; }
table[frame=vsides i] { border-style: hidden outset; }
table[frame=box i], table[frame=border i] { border-style: outset; }

table[border] > tr > td, table[border] > tr > th,
table[border] > thead > tr > td, table[border] > thead > tr > th,
table[border] > tbody > tr > td, table[border] > tbody > tr > th,
table[border] > tfoot > tr > td, table[border] > tfoot > tr > th {
  /* <span>only if border is not equivalent to zero</span> */
  border-width: 1px;
  border-style: inset;
}
table[rules=none i] > tr > td, table[rules=none i] > tr > th,
table[rules=none i] > thead > tr > td, table[rules=none i] > thead > tr > th,
table[rules=none i] > tbody > tr > td, table[rules=none i] > tbody > tr > th,
table[rules=none i] > tfoot > tr > td, table[rules=none i] > tfoot > tr > th,
table[rules=groups i] > tr > td, table[rules=groups i] > tr > th,
table[rules=groups i] > thead > tr > td, table[rules=groups i] > thead > tr > th,
table[rules=groups i] > tbody > tr > td, table[rules=groups i] > tbody > tr > th,
table[rules=groups i] > tfoot > tr > td, table[rules=groups i] > tfoot > tr > th,
table[rules=rows i] > tr > td, table[rules=rows i] > tr > th,
table[rules=rows i] > thead > tr > td, table[rules=rows i] > thead > tr > th,
table[rules=rows i] > tbody > tr > td, table[rules=rows i] > tbody > tr > th,
table[rules=rows i] > tfoot > tr > td, table[rules=rows i] > tfoot > tr > th {
  border-width: 1px;
  border-style: none;
}
table[rules=cols i] > tr > td, table[rules=cols i] > tr > th,
table[rules=cols i] > thead > tr > td, table[rules=cols i] > thead > tr > th,
table[rules=cols i] > tbody > tr > td, table[rules=cols i] > tbody > tr > th,
table[rules=cols i] > tfoot > tr > td, table[rules=cols i] > tfoot > tr > th {
  border-width: 1px;
  border-block-style: none;
  border-inline-style: solid;
}
table[rules=all i] > tr > td, table[rules=all i] > tr > th,
table[rules=all i] > thead > tr > td, table[rules=all i] > thead > tr > th,
table[rules=all i] > tbody > tr > td, table[rules=all i] > tbody > tr > th,
table[rules=all i] > tfoot > tr > td, table[rules=all i] > tfoot > tr > th {
  border-width: 1px;
  border-style: solid;
}

table[rules=groups i] > colgroup {
  border-inline-width: 1px;
  border-inline-style: solid;
}
table[rules=groups i] > thead,
table[rules=groups i] > tbody,
table[rules=groups i] > tfoot {
  border-block-width: 1px;
  border-block-style: solid;
}

table[rules=rows i] > tr, table[rules=rows i] > thead > tr,
table[rules=rows i] > tbody > tr, table[rules=rows i] > tfoot > tr {
  border-block-width: 1px;
  border-block-style: solid;
}</code></pre>

<!--
 Demos that the above (and prose below) must explain:
   https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1191
   https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1194
   https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1195
   https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1196
   https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1197
   https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1199
   https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1200
   https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1201
   https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1209
-->


  <p>In <span>quirks mode</span>, the following rules are also expected to apply:</p>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

table {
  font-weight: initial;
  font-style: initial;
  font-variant: initial;
  font-size: initial;
  line-height: initial;
  white-space: initial;
  text-align: initial;
}</code></pre>

  <hr>

  <p>For the purposes of the CSS table model, the <code>col</code> element is expected to be treated
  as if it was present as many times as its <code data-x="attr-col-span">span</code> attribute <span
  data-x="rules for parsing non-negative integers">specifies</span>.</p>

  <p>For the purposes of the CSS table model, the <code>colgroup</code> element, if it contains no
  <code>col</code> element, is expected to be treated as if it had as many such children as its
  <code data-x="attr-colgroup-span">span</code> attribute <span data-x="rules for parsing non-negative
  integers">specifies</span>.</p>

  <p>For the purposes of the CSS table model, the <code data-x="attr-tdth-colspan">colspan</code> and
  <code data-x="attr-tdth-rowspan">rowspan</code> attributes on <code>td</code> and <code>th</code>
  elements are expected to <span data-x="rules for parsing non-negative integers">provide</span> the
  <i>special knowledge</i> regarding cells spanning rows and columns.</p>

  <p>In <span>HTML documents</span>, the following rules are also expected to apply:</p>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

:is(table, thead, tbody, tfoot, tr) > form { display: none !important; }</code></pre>

  <hr>

  <p>The <code>table</code> element's <code data-x="attr-table-cellspacing">cellspacing</code>
  attribute <span>maps to the pixel length property</span> <span>'border-spacing'</span> on the
  element.</p>

  <p>The <code>table</code> element's <code data-x="attr-table-cellpadding">cellpadding</code>
  attribute <span data-x="maps to the pixel length property">maps to the pixel length
  properties</span> <span>'padding-top'</span>, <span>'padding-right'</span>,
  <span>'padding-bottom'</span>, and <span>'padding-left'</span> of any <code>td</code> and
  <code>th</code> elements that have corresponding <span data-x="concept-cell">cells</span> in the
  <span data-x="concept-table">table</span> corresponding to the <code>table</code> element.</p>

  <p>The <code>table</code> element's <code data-x="attr-table-height">height</code> attribute
  <span>maps to the dimension property</span> <span>'height'</span> on the <code>table</code>
  element.</p>

  <p>The <code>table</code> element's <code data-x="attr-table-width">width</code> attribute
  <span>maps to the dimension property (ignoring zero)</span> <span>'width'</span> on the
  <code>table</code> element.</p>

  <p>The <code>col</code> element's <code data-x="attr-col-width">width</code> attribute <span>maps
  to the dimension property</span> <span>'width'</span> on the <code>col</code>
  element.</p>

  <p>The <code>thead</code>, <code>tbody</code>, and <code>tfoot</code> elements' <code
  data-x="attr-tbody-height">height</code> attribute <span>maps to the dimension property</span>
  <span>'height'</span> on the element.</p>

  <p>The <code>tr</code> element's <code data-x="attr-tr-height">height</code> attribute <span>maps
  to the dimension property</span> <span>'height'</span> on the <code>tr</code> element.</p>

  <p>The <code>td</code> and <code>th</code> elements' <code data-x="attr-tdth-height">height</code>
  attributes <span data-x="maps to the dimension property (ignoring zero)">map to the dimension
  property (ignoring zero)</span> <span>'height'</span> on the element.</p>

  <p>The <code>td</code> and <code>th</code> elements' <code data-x="attr-tdth-width">width</code>
  attributes <span data-x="maps to the dimension property (ignoring zero)">map to the dimension
  property (ignoring zero)</span> <span>'width'</span> on the element.</p>

  <hr>

  <p>The <code>thead</code>, <code>tbody</code>, <code>tfoot</code>, <code>tr</code>,
  <code>td</code>, and <code>th</code> elements, when they have an <code data-x="">align</code>
  attribute whose value is an <span>ASCII case-insensitive</span> match for either the string "<code
  data-x="">center</code>" or the string "<code data-x="">middle</code>", are expected to center
  text within themselves, as if they had their <span>'text-align'</span> property set to 'center' in
  a <span data-x="presentational hints">presentational hint</span>, and to <span>align
  descendants</span> to the center.</p>

  <p>The <code>thead</code>, <code>tbody</code>, <code>tfoot</code>, <code>tr</code>,
  <code>td</code>, and <code>th</code> elements, when they have an <code data-x="">align</code>
  attribute whose value is an <span>ASCII case-insensitive</span> match for the string "<code
  data-x="">left</code>", are expected to left-align text within themselves, as if they had their
  <span>'text-align'</span> property set to 'left' in a <span data-x="presentational
  hints">presentational hint</span>, and to <span>align descendants</span> to the left.</p>

  <p>The <code>thead</code>, <code>tbody</code>, <code>tfoot</code>, <code>tr</code>,
  <code>td</code>, and <code>th</code> elements, when they have an <code data-x="">align</code>
  attribute whose value is an <span>ASCII case-insensitive</span> match for the string "<code
  data-x="">right</code>", are expected to right-align text within themselves, as if they had their
  <span>'text-align'</span> property set to 'right' in a <span data-x="presentational
  hints">presentational hint</span>, and to <span>align descendants</span> to the right.</p>

  <p>The <code>thead</code>, <code>tbody</code>, <code>tfoot</code>, <code>tr</code>,
  <code>td</code>, and <code>th</code> elements, when they have an <code data-x="">align</code>
  attribute whose value is an <span>ASCII case-insensitive</span> match for the string "<code
  data-x="">justify</code>", are expected to full-justify text within themselves, as if they had
  their <span>'text-align'</span> property set to 'justify' in a <span data-x="presentational
  hints">presentational hint</span>, and to <span>align descendants</span> to the left.</p>

  <p>User agents are expected to have a rule in their user agent style sheet that matches
  <code>th</code> elements that have a parent node whose <span>computed value</span> for the
  <span>'text-align'</span> property is its initial value, whose declaration block consists of just
  a single declaration that sets the <span>'text-align'</span> property to the value 'center'.</p>
  <!-- q.v. '-moz-center-or-inherit' -->

  <hr>

  <p>When a <code>table</code>, <code>thead</code>, <code>tbody</code>, <code>tfoot</code>,
  <code>tr</code>, <code>td</code>, or <code>th</code> element has a <code
  data-x="attr-background">background</code> attribute set to a non-empty value, the new value is
  expected to be <span data-x="encoding-parsing-and-serializing a
  URL">encoding-parsed-and-serialized</span> relative to the element's <span>node document</span>,
  and if that does not return failure, the user agent is expected to treat the attribute as a <span
  data-x="presentational hints">presentational hint</span> setting the element's
  <span>'background-image'</span> property to the return value.</p>

  <p>When a <code>table</code>, <code>thead</code>, <code>tbody</code>, <code>tfoot</code>,
  <code>tr</code>, <code>td</code>, or <code>th</code> element has a <code data-x="">bgcolor</code>
  attribute set, the new value is expected to be parsed using the <span>rules for parsing a legacy
  color value</span>, and if that does not return failure, the user agent is expected to treat the
  attribute as a <span data-x="presentational hints">presentational hint</span> setting the element's
  <span>'background-color'</span> property to the resulting color.</p>

  <p>When a <code>table</code> element has a <code data-x="attr-table-bordercolor">bordercolor</code>
  attribute, its value is expected to be parsed using the <span>rules for parsing a legacy color
  value</span>, and if that does not return failure, the user agent is expected to treat the
  attribute as a <span data-x="presentational hints">presentational hint</span> setting the
  element's <span>'border-top-color'</span>, <span>'border-right-color'</span>,
  <span>'border-bottom-color'</span>, and <span>'border-left-color'</span> properties to the
  resulting color.</p>

  <hr>

  <p>The <code>table</code> element's <code data-x="attr-table-border">border</code> attribute <span
  data-x="maps to the pixel length property">maps to the pixel length properties</span>
  <span>'border-top-width'</span>, <span>'border-right-width'</span>,
  <span>'border-bottom-width'</span>, <span>'border-left-width'</span> on the element. If the
  attribute is present but parsing the attribute's value using the <span>rules for parsing
  non-negative integers</span> generates an error, a default value of 1px is expected to be used for
  that property instead.</p>

  <p>Rules marked "<dfn id="magic-border-selector">only if border is not equivalent to zero</dfn>"
  in the CSS block above is expected to only be applied if the <code
  data-x="attr-table-border">border</code> attribute mentioned in the selectors for the rule is not
  only present but, when parsed using the <span>rules for parsing non-negative integers</span>, is
  also found to have a value other than zero or to generate an error.</p>

  <hr>

  <p>In <span>quirks mode</span>, a <code>td</code> element or a <code>th</code> element that has a
  <code data-x="attr-tdth-nowrap">nowrap</code> attribute but also has a <code
  data-x="attr-tdth-width">width</code> attribute whose value, when parsed using the <span>rules for
  parsing nonzero dimension values</span>, is found to be a length (not an error or a number
  classified as a percentage), is expected to have a <span data-x="presentational
  hints">presentational hint</span> setting the element's <span>'white-space'</span> property to
  'normal', overriding the rule in the CSS block above that sets it to 'nowrap'.</p> <!--
  https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=1793 --> <!-- note that the "rules for
  parsing nonzero dimension values" can't return 0, if the value is "0" they treat it as an error
  -->

  <h4>Margin collapsing quirks</h4>

  <p>A node is <dfn data-x="concept-rendering-substantial">substantial</dfn> if it is a text node
  that is not <span>inter-element whitespace</span>, or if it is an element node.</p>

  <p>A node is <dfn data-x="concept-rendering-blank">blank</dfn> if it is an element that contains no
  <span data-x="concept-rendering-substantial">substantial</span> nodes.</p>

  <p>The <dfn data-x="concept-rendering-elements-with-margins">elements with default margins</dfn>
  are the following elements: <code>blockquote</code>, <code>dir</code>, <code>dl</code>,
  <code>h1</code>, <code>h2</code>, <code>h3</code>, <code>h4</code>, <code>h5</code>,
  <code>h6</code>, <code>listing</code>, <code>menu</code>, <code>ol</code>,
  <code>p</code>, <code>plaintext</code>, <code>pre</code>, <code>ul</code>, <code>xmp</code></p>

  <p>In <span>quirks mode</span>, any <span data-x="concept-rendering-elements-with-margins">element
  with default margins</span> that is the <span data-x="concept-tree-child">child</span> of a
  <code>body</code>, <code>td</code>, or <code>th</code> element and has no <span
  data-x="concept-rendering-substantial">substantial</span> previous siblings is expected to have a
  user-agent level style sheet rule that sets its <span>'margin-block-start'</span> property to
  zero.</p>

  <p>In <span>quirks mode</span>, any <span data-x="concept-rendering-elements-with-margins">element
  with default margins</span> that is the <span data-x="concept-tree-child">child</span> of a
  <code>body</code>, <code>td</code>, or <code>th</code> element, has no <span
  data-x="concept-rendering-substantial">substantial</span> previous siblings, and is <span
  data-x="concept-rendering-blank">blank</span>, is expected to have a user-agent level style sheet
  rule that sets its <span>'margin-block-end'</span> property to zero also.</p>

  <p>In <span>quirks mode</span>, any <span data-x="concept-rendering-elements-with-margins">element
  with default margins</span> that is the <span data-x="concept-tree-child">child</span> of a
  <code>td</code> or <code>th</code> element, has no <span
  data-x="concept-rendering-substantial">substantial</span> following siblings, and is <span
  data-x="concept-rendering-blank">blank</span>, is expected to have a user-agent level style sheet
  rule that sets its <span>'margin-block-start'</span> property to zero.</p>

  <p>In <span>quirks mode</span>, any <code>p</code> element that is the <span
  data-x="concept-tree-child">child</span> of a <code>td</code> or <code>th</code> element and has
  no <span data-x="concept-rendering-substantial">substantial</span> following siblings, is expected
  to have a user-agent level style sheet rule that sets its <span>'margin-block-end'</span> property
  to zero.</p>


  <h4>Form controls</h4>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

input, select, button, textarea {
  letter-spacing: initial;
  word-spacing: initial;
  line-height: initial;
  text-transform: initial;
  text-indent: initial;
  text-shadow: initial;
  appearance: auto;
}

input:not([type=image i], [type=range i], [type=checkbox i], [type=radio i]) {
  overflow: clip !important;
  overflow-clip-margin: 0 !important;
}

input, select, textarea {
  text-align: initial;
}

:autofill {
  field-sizing: fixed !important;
}

input:is([type=reset i], [type=button i], [type=submit i]), button {
  text-align: center;
}

input, button {
  display: inline-block;
}

input[type=hidden i], input[type=file i], input[type=image i] {
  appearance: none;
}

input:is([type=radio i], [type=checkbox i], [type=reset i], [type=button i],
[type=submit i], [type=color i], [type=search i]), select, button {
  box-sizing: border-box;
}

textarea { white-space: pre-wrap; }</code></pre>

  <p>In <span>quirks mode</span>, the following rules are also expected to apply:</p>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

input:not([type=image i]), textarea { box-sizing: border-box; }</code></pre>

  <p>Each kind of form control is also described in the <a href="#widgets">Widgets</a> section,
  which describes the look and feel of the control.</p>

  <p>For <code>input</code> elements where the <code data-x="attr-input-type">type</code> attribute
  is not in the <span data-x="attr-input-type-hidden">Hidden</span> state or the <span
  data-x="attr-input-type-image">Image Button</span> state, and that are <span>being
  rendered</span>, are expected to act as follows:</p>

  <ul>
   <li><p>The <span>inner display type</span> is always 'flow-root'.</p></li>
  </ul>


  <h4>The <code>hr</code> element</h4>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

hr {
  color: gray;
  border-style: inset;
  border-width: 1px;
  margin-block: 0.5em;
  margin-inline: auto;
  overflow: hidden;
}</code></pre>

  <p>The following rules are also expected to apply, as <span>presentational hints</span>:</p>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

hr[align=left i] { margin-left: 0; margin-right: auto; }
hr[align=right i] { margin-left: auto; margin-right: 0; }
hr[align=center i] { margin-left: auto; margin-right: auto; }
hr[color], hr[noshade] { border-style: solid; }</code></pre>

  <p>If an <code>hr</code> element has either a <code data-x="attr-hr-color">color</code> attribute
  or a <code data-x="attr-hr-noshade">noshade</code> attribute, and furthermore also has a <code
  data-x="attr-hr-size">size</code> attribute, and parsing that attribute's value using the
  <span>rules for parsing non-negative integers</span> doesn't generate an error, then the user
  agent is expected to use the parsed value divided by two as a pixel length for
  <span>presentational hints</span> for the properties <span>'border-top-width'</span>,
  <span>'border-right-width'</span>, <span>'border-bottom-width'</span>, and
  <span>'border-left-width'</span> on the element.</p>

  <p>Otherwise, if an <code>hr</code> element has neither a <code data-x="attr-hr-color">color</code>
  attribute nor a <code data-x="attr-hr-noshade">noshade</code> attribute, but does have a <code
  data-x="attr-hr-size">size</code> attribute, and parsing that attribute's value using the
  <span>rules for parsing non-negative integers</span> doesn't generate an error, then: if the
  parsed value is one, then the user agent is expected to use the attribute as a <span
  data-x="presentational hints">presentational hint</span> setting the element's
  <span>'border-bottom-width'</span> to 0; otherwise, if the parsed value is greater than one, then
  the user agent is expected to use the parsed value minus two as a pixel length for
  <span>presentational hints</span> for the <span>'height'</span> property on the element.</p>

  <p>The <code data-x="attr-hr-width">width</code> attribute on an <code>hr</code> element <span>maps
  to the dimension property</span> <span>'width'</span> on the element.</p>

  <p>When an <code>hr</code> element has a <code data-x="attr-hr-color">color</code> attribute, its
  value is expected to be parsed using the <span>rules for parsing a legacy color value</span>, and
  if that does not return failure, the user agent is expected to treat the attribute as a <span
  data-x="presentational hints">presentational hint</span> setting the element's
  <span>'color'</span> property to the resulting color.</p>



  <h4>The <code>fieldset</code> and <code>legend</code> elements</h4>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

fieldset {
  display: block;
  margin-inline: 2px;
  border: groove 2px ThreeDFace;
  padding-block: 0.35em 0.625em;
  padding-inline: 0.75em;
  min-inline-size: min-content;
}

legend {
  padding-inline: 2px;
}

legend[align=left i] {
  justify-self: left;
}

legend[align=center i] {
  justify-self: center;
}

legend[align=right i] {
  justify-self: right;
}</code></pre>

  <p>The <code>fieldset</code> element, when it generates a <span>CSS box</span>, is expected to act
  as follows:</p>

  <ul>
   <li><p>The element is expected to establish a new <span>block formatting context</span>.</p></li>

   <li>
    <p>The <span>'display'</span> property is expected to act as follows:</p>

    <ul>
     <li><p>If the computed value of <span>'display'</span> is a value such that the <span>outer
     display type</span> is 'inline', then behave as 'inline-block'.</p></li>

     <li><p>Otherwise, behave as 'flow-root'.</p></li>
    </ul>

    <p class="note">This does not change the computed value.</p>
   </li>

   <li>
    <p>If the element's box has a child box that matches the conditions in the list below, then the
    first such child box is the 'fieldset' element's <dfn>rendered legend</dfn>:</p>

    <ul class="brief">
     <li>The child is a <code>legend</code> element.</li>
     <li>The child's used value of <span>'float'</span> is 'none'.</li>
     <li>The child's used value of <span>'position'</span> is not 'absolute' or 'fixed'.</li>
    </ul>
   </li>

   <li>
    <p>If the element has a <span>rendered legend</span>, then the border is expected to not be
    painted behind the rectangle defined as follows, using the writing mode of the fieldset:</p>

    <ol>
     <li><p>The block-start edge of the rectangle is the smaller of the block-start edge of the
     <span>rendered legend</span>'s margin rectangle at its static position (ignoring transforms),
     and the block-start outer edge of the <code>fieldset</code>'s border.</p></li>

     <li><p>The block-end edge of the rectangle is the larger of the block-end edge of the
     <span>rendered legend</span>'s margin rectangle at its static position (ignoring transforms),
     and the block-end outer edge of the <code>fieldset</code>'s border.</p></li>

     <li><p>The inline-start edge of the rectangle is the smaller of the inline-start edge of the
     <span>rendered legend</span>'s border rectangle at its static position (ignoring transforms),
     and the inline-start outer edge of the <code>fieldset</code>'s border.</p></li>

     <li><p>The inline-end edge of the rectangle is the larger of the inline-end edge of the
     <span>rendered legend</span>'s border rectangle at its static position (ignoring transforms),
     and the inline-end outer edge of the <code>fieldset</code>'s border.</p></li>
    </ol>
   </li>

   <li><p>The space allocated for the element's border on the block-start side is expected to be
   the element's <span>'border-block-start-width'</span> or the <span>rendered legend</span>'s
   margin box size in the <code>fieldset</code>'s block-flow direction, whichever is
   greater.</p></li>

   <li><p>For the purpose of calculating the used <span>'block-size'</span>, if the computed
   <span>'block-size'</span> is not 'auto', the space allocated for the <span>rendered
   legend</span>'s margin box that spills out past the border, if any, is expected to be
   subtracted from the <span>'block-size'</span>. If the content box's block-size would be
   negative, then let the content box's block-size be zero instead.</p></li>

   <li><p>If the element has a <span>rendered legend</span>, then that element is expected to be
   the first child box.</p></li>

   <li><p>The <span>anonymous fieldset content box</span> is expected to appear after the
   <span>rendered legend</span> and is expected to contain the content (including the '::before'
   and '::after' pseudo-elements) of the <code>fieldset</code> element except for the
   <span>rendered legend</span>, if there is one.</p></li>

   <li><p>The used value of the <span>'padding-top'</span>, <span>'padding-right'</span>,
   <span>'padding-bottom'</span>, and <span>'padding-left'</span> properties are expected to be
   zero.</p></li>

   <li><p>For the purpose of calculating the min-content inline size, use the greater of the
   min-content inline size of the <span>rendered legend</span> and the min-content inline size of
   the <span>anonymous fieldset content box</span>.</p></li>

   <li><p>For the purpose of calculating the max-content inline size, use the greater of the
   max-content inline size of the <span>rendered legend</span> and the max-content inline size of
   the <span>anonymous fieldset content box</span>.</p></li>
 </ul>

  <p>A <code>fieldset</code> element's <span>rendered legend</span>, if any, is expected to act as
  follows:</p>

  <ul>
   <li><p>The element is expected to establish a new <span>formatting context</span> for its
   contents. The type of this <span>formatting context</span> is determined by its
   <span>'display'</span> value, as usual.</p></li>

   <li>
    <p>The <span>'display'</span> property is expected to behave as if its computed value was
    blockified.</p>

    <p class="note">This does not change the computed value.</p>
   </li>

   <li><p>If the <span>computed value</span> of <span>'inline-size'</span> is 'auto', then the
   <span>used value</span> is the <span>fit-content inline size</span>.</p></li>

   <li><p>The element is expected to be positioned in the inline direction as is normal for
   blocks (e.g., taking into account margins and the <span>'justify-self'</span>
   property).</p></li>

   <li>
    <p>The element's box is expected to be constrained in the inline direction by the inline
    content size of the <code>fieldset</code> as if it had used its computed inline padding.</p>

    <div class="example">
     <p>For example, if the <code>fieldset</code> has a specified padding of 50px, then the
     <span>rendered legend</span> will be positioned 50px in from the <code>fieldset</code>'s
     border. The padding will further apply to the <span>anonymous fieldset content box</span>
     instead of the <code>fieldset</code> element itself.</p>
    </div>
   </li>

   <li><p>The element is expected to be positioned in the block-flow direction such that its border
   box is centered over the border on the block-start side of the <code>fieldset</code>
   element.</p></li>
  </ul>


  <p>A <code>fieldset</code> element's <dfn>anonymous fieldset content box</dfn> is expected to
  act as follows:</p>

  <ul>
   <li>
    <p>The <span>'display'</span> property is expected to act as follows:</p>

    <ul>
     <li><p>If the computed value of <span>'display'</span> on the <code>fieldset</code> element
     is 'grid' or 'inline-grid', then set the used value to 'grid'.</p></li>

     <li><p>If the computed value of <span>'display'</span> on the <code>fieldset</code> element
     is 'flex' or 'inline-flex', then set the used value to 'flex'.</p></li>

     <li><p>Otherwise, set the used value to 'flow-root'.</p></li>
    </ul>
   </li>

   <li>
    <p>The following properties are expected to inherit from the <code>fieldset</code> element:</p>

    <ul class="brief">
     <li><span>'align-content'</span></li>
     <li><span>'align-items'</span></li>
     <li><span>'border-radius'</span></li>
     <li><span>'column-count'</span></li>
     <li><span>'column-fill'</span></li>
     <li><span>'column-gap'</span></li>
     <li><span>'column-rule'</span></li>
     <li><span>'column-width'</span></li>
     <li><span>'flex-direction'</span></li>
     <li><span>'flex-wrap'</span></li>
     <li><span>'grid-auto-columns'</span></li>
     <li><span>'grid-auto-flow'</span></li>
     <li><span>'grid-auto-rows'</span></li>
     <li><span>'grid-column-gap'</span></li>
     <li><span>'grid-row-gap'</span></li>
     <li><span>'grid-template-areas'</span></li>
     <li><span>'grid-template-columns'</span></li>
     <li><span>'grid-template-rows'</span></li>
     <li><span>'justify-content'</span></li>
     <li><span>'justify-items'</span></li>
     <li><span>'overflow'</span></li>
     <li><span>'padding-bottom'</span></li>
     <li><span>'padding-left'</span></li>
     <li><span>'padding-right'</span></li>
     <li><span>'padding-top'</span></li>
     <li><span>'text-overflow'</span></li>
     <li><span>'unicode-bidi'</span></li>
    </ul>
   </li>

   <li><p>The <span>'block-size'</span> property is expected to be set to '100%'.</p></li>

   <li><p>For the purpose of calculating percentage padding, act as if the padding was calculated
   for the <code>fieldset</code> element.</p></li>
  </ul>

  <div class="note" id="fieldset-layout-model">
   <figure>
    <svg viewBox="0 0 400 270" width="480" height="324" role="img" aria-label="Fieldset layout
    model">

     <defs>
      <marker id='arrow-red' orient='auto' markerWidth='5' markerHeight='4'
              refX='0.1' refY='2'>
       <path d='M0,0 V4 L5,2 Z' fill='red' />
      </marker>
      <marker id='arrow-blue' orient='auto' markerWidth='5' markerHeight='4'
              refX='0.1' refY='2'>
       <path d='M0,0 V4 L5,2 Z' fill='blue' />
      </marker>
     </defs>

     <!-- outer outline -->
     <rect stroke-dasharray="6" stroke-width="1" stroke="black" width="304" height="176" x="48"
     y="48" fill="none"/>

     <line marker-end="url(#arrow-blue)" x1="200" x2="200" y1="48" y2="20" stroke-width="2"
     stroke="blue"/>
     <line marker-end="url(#arrow-blue)" x1="48" x2="20" y1="136" y2="136" stroke-width="2"
     stroke="blue"/>
     <line marker-end="url(#arrow-blue)" x1="352" x2="380" y1="136" y2="136" stroke-width="2"
     stroke="blue"/>
     <line marker-end="url(#arrow-blue)" x1="200" x2="200" y1="224" y2="250" stroke-width="2"
     stroke="blue"/>
     <text x="210" y="35" font-size="12" font-family="sans-serif" font-style="normal"
     fill="blue">fieldset's margin</text>

     <!-- fieldset -->
     <rect stroke-width="2" stroke="blue" width="300" height="116.5" x="50" y="105" fill="white"/>

     <!-- rendered legend -->
     <rect stroke-width="2" stroke="red" width="85" height="60" x="100" y="75" fill="white"/>
     <text x="120" y="105" dominant-baseline="central" fill="red" font-family="serif"
     font-style="normal">legend</text>

     <!-- legend padding -->
     <rect x="110" y="85" width="65" height="40" fill="none" stroke="#eee" stroke-width="18"/>
     <text x="133" y="88" font-size="12" font-family="sans-serif" font-style="normal"
     fill="gray">padding</text>

     <line marker-end="url(#arrow-red)" x1="185" x2="325" y1="90" y2="90" stroke-width="2"
     stroke="red"/>
     <text x="210" y="85" font-size="12" font-family="sans-serif" font-style="normal"
     fill="red">legend's margin</text>

     <line marker-end="url(#arrow-red)" x1="100" x2="75" y1="90" y2="90" stroke-width="2"
     stroke="red"/>
     <line marker-end="url(#arrow-red)" x1="143" x2="143" y1="75" y2="60" stroke-width="2"
     stroke="red"/>
     <line marker-end="url(#arrow-red)" x1="143" x2="143" y1="135" y2="150" stroke-width="2"
     stroke="red"/>

     <!-- fieldset padding -->
     <rect x="60" y="171" width="280" height="40" fill="none" stroke="#eee" stroke-width="18"/>
     <text x="297" y="174" font-size="12" font-family="sans-serif" font-style="normal"
     fill="gray">padding</text>

     <!-- anonymous fieldset content box outline -->
     <rect stroke-dasharray="6" stroke-width="1" stroke="black" width="296" height="58" x="52"
     y="161" fill="none"/>
     <text x="170" y="156" font-size="12" font-family="sans-serif" font-style="normal">anonymous
     fieldset content box</text>
     <text x="70" y="191" dominant-baseline="central" fill="blue" font-family="serif"
     font-style="normal">content</text>

    </svg>

    <figcaption>The legend is rendered over the top border, and the top border area reserves
    vertical space for the legend. The fieldset's top margin starts at the top margin edge of the
    legend. The legend's horizontal margins, or the <span>'justify-self'</span> property, gives
    its horizontal position. The <span>anonymous fieldset content box</span> appears below the
    legend.</figcaption>
   </figure>
  </div>


  <h3>Replaced elements</h3>

  <p class="note">The following elements can be <span data-x="replaced element">replaced
  elements</span>: <code>audio</code>, <code>canvas</code>, <code>embed</code>, <code>iframe</code>,
  <code>img</code>, <code>input</code>, <code>object</code>, and <code>video</code>.</p>


  <h4 id="embedded-content-rendering-rules">Embedded content</h4>

  <p>The <code>embed</code>, <code>iframe</code>, and <code>video</code> elements are expected to be
  treated as <span data-x="replaced element">replaced elements</span>.</p>

  <p>A <code>canvas</code> element that <span>represents</span> <span>embedded content</span> is
  expected to be treated as a <span>replaced element</span>; the contents of such elements are the
  element's bitmap, if any, or else a <span>transparent black</span> bitmap with the same
  <span>natural dimensions</span> as the element. Other <code>canvas</code> elements are expected
  to be treated as ordinary elements in the rendering model.</p>

  <p>An <code>object</code> element that <span>represents</span> an image, plugin, or its
  <span>content navigable</span> is expected to be treated as a <span>replaced element</span>.
  Other <code>object</code> elements are expected to be treated as ordinary elements in the
  rendering model.</p>

  <p>The <code>audio</code> element, when it is <span data-x="expose a user interface to the
  user">exposing a user interface</span>, is expected to be treated as a
  <span>replaced element</span> about one line high, as wide as is necessary to expose the user
  agent's user interface features. When an <code>audio</code> element is not <span data-x="expose a
  user interface to the user">exposing a user interface</span>, the user agent is expected to force
  its <span>'display'</span> property to compute to 'none', irrespective of CSS rules.</p>

  <p>Whether a <code>video</code> element is <span data-x="expose a user interface to the
  user">exposing a user interface</span> is not expected to affect the size of the rendering;
  controls are expected to be overlaid above the page content without causing any layout changes,
  and are expected to disappear when the user does not need them.</p>

  <p>When a <code>video</code> element represents a poster frame or frame of video, the poster frame
  or frame of video is expected to be rendered at the largest size that maintains the aspect ratio
  of that poster frame or frame of video without being taller or wider than the <code>video</code>
  element itself, and is expected to be centered in the <code>video</code> element.</p>

  <p>Any subtitles or captions are expected to be overlaid directly on top of their
  <code>video</code> element, as defined by the relevant rendering rules; for WebVTT, those are the
  <span>rules for updating the display of WebVTT text tracks</span>. <ref>WEBVTT</ref></p>

  <p>When the user agent starts <span data-x="expose a user interface to the user">exposing a user
  interface</span> for a <code>video</code> element, the user agent should run the <span>rules for
  updating the text track rendering</span> of each of the <span data-x="text track">text
  tracks</span> in the <code>video</code> element's <span>list of text tracks</span> that are <span
  data-x="text track showing">showing</span> and whose <span>text track kind</span> is one of <code
  data-x="dom-TextTrack-kind-subtitles">subtitles</code> or <code
  data-x="dom-TextTrack-kind-captions">captions</code> (e.g., for <span data-x="text track">text
  tracks</span> based on WebVTT, the <span>rules for updating the display of WebVTT text
  tracks</span>). <ref>WEBVTT</ref></p>

  <p class="note">Resizing <code>video</code> and <code>canvas</code> elements does not interrupt
  video playback or clear the canvas.</p>

  <hr>

  <p>The following CSS rules are expected to apply:</p>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

iframe { border: 2px inset; }
<span id="video-object-fit">video { object-fit: contain; }</span></code></pre>


  <h4>Images</h4>

  <p>User agents are expected to render <code>img</code> elements and <code>input</code> elements
  whose <code data-x="attr-input-type">type</code> attributes are in the <span
  data-x="attr-input-type-image">Image Button</span> state, according to the first applicable rules
  from the following list:</p>

  <dl class="switch">


   <dt>If the element <span>represents</span> an image</dt>

   <dd>The user agent is expected to treat the element as a <span>replaced element</span> and
   render the image according to the rules for doing so defined in CSS.</dd>


   <dt>

    If the element does not <span data-x="represents">represent</span> an image and either:

    <ul>
     <li>the user agent has reason to believe that the image will become <i
     data-x="img-available">available</i><!--input-img-available also--> and be rendered in due
     course, or

     <li>the element has no <code data-x="">alt</code> attribute<!-- attr-img-alt, attr-input-alt -->, or

     <li>the <code>Document</code> is in <span>quirks mode</span>, and the element already has
     <span>natural dimensions</span> (e.g., from the <span>dimension attributes</span> or CSS
     rules)
    </ul>

   </dt>

   <dd>The user agent is expected to treat the element as a <span>replaced element</span> whose
   content is the text that the element represents, if any, optionally alongside an icon indicating
   that the image is being obtained (if applicable). For <code>input</code> elements, the element
   is expected to appear button-like to indicate that the element is a <span
   data-x="concept-button">button</span>.</dd>


   <!-- no-quirks and limited-quirks modes only-->
   <dt>If the element is an <code>img</code> element that <span>represents</span> some text and the
   user agent does not expect this to change</dt>

   <dd>The user agent is expected to treat the element as a non-replaced phrasing element whose
   content is the text, optionally with an icon indicating that an image is missing, so that the
   user can request the image be displayed or investigate why it is not rendering. In non-graphical
   contexts, such an icon should be omitted.</dd>


   <!-- no-quirks and limited-quirks modes only-->
   <dt>If the element is an <code>img</code> element that <span>represents</span> nothing and the
   user agent does not expect this to change</dt>

   <dd>The user agent is expected to treat the element as a <span>replaced element</span> whose
   <span>natural dimensions</span> are 0. (In the absence of further styles, this will cause the
   element to essentially not be rendered.)</dd>


   <!-- no-quirks and limited-quirks modes only-->
   <dt>If the element is an <code>input</code> element that does not <span
   data-x="represents">represent</span> an image and the user agent does not expect this to change</dt>

   <dd>The user agent is expected to treat the element as a <span>replaced element</span>
   consisting of a button whose content is the element's alternative text. The <span>natural
   dimensions</span> of the button are expected to be about one line in height and whatever width
   is necessary to render the text on one line.</dd>
  </dl>

  <p>The icons mentioned above are expected to be relatively small so as not to disrupt most text
  but be easily clickable. In a visual environment, for instance, icons could be 16 pixels by 16
  pixels square, or 1em by 1em if the images are scalable. In an audio environment, the icon could
  be a short bleep. The icons are intended to indicate to the user that they can be used to get to
  whatever options the UA provides for images, and, where appropriate, are expected to provide
  access to the context menu that would have come up if the user interacted with the actual
  image.</p>

  <hr>

  <p>All animated images with the same <span>absolute URL</span> and the same image data are
  expected to be rendered synchronized to the same timeline as a group, with the timeline starting
  at the time of the least recent addition to the group.</p>

  <p class="note">In other words, when a second image with the same <span>absolute URL</span> and
  animated image data is inserted into a document, it jumps to the point in the animation cycle that
  is currently being displayed by the first image.</p>

  <p>When a user agent is to <dfn>restart the animation</dfn> for an <code>img</code> element
  showing an animated image, all animated images with the same <span>absolute URL</span> and the
  same image data in that <code>img</code> element's <span>node document</span> are expected to restart
  their animation from the beginning.</p>

  <hr>

  <p>The following CSS rules are expected to apply:</p>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

<span id="img-contain-size">img:is([sizes="auto" i], [sizes^="auto," i]) {
  contain: size !important;
  contain-intrinsic-size: 300px 150px;
}</span></code></pre>

  <p>The following CSS rules are expected to apply when the <code>Document</code> is in <span>quirks
  mode</span>:</p>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

img[align=left i] { margin-right: 3px; }
img[align=right i] { margin-left: 3px; }</code></pre>

  </div>


  <div w-nodev>

  <h4>Attributes for embedded content and images</h4>

  <p>The following CSS rules are expected to apply as <span>presentational hints</span>:</p>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

embed[align=left i], iframe[align=left i], img[align=left i],
input[type=image i][align=left i], object[align=left i] {
  float: left;
}

embed[align=right i], iframe[align=right i], img[align=right i],
input[type=image i][align=right i], object[align=right i] {
  float: right;
}

embed[align=top i], iframe[align=top i], img[align=top i],
input[type=image i][align=top i], object[align=top i] {
  vertical-align: top;
}

embed[align=baseline i], iframe[align=baseline i], img[align=baseline i],
input[type=image i][align=baseline i], object[align=baseline i] {
  vertical-align: baseline;
}

embed[align=texttop i], iframe[align=texttop i], img[align=texttop i],
input[type=image i][align=texttop i], object[align=texttop i] {
  vertical-align: text-top;
}

embed[align=absmiddle i], iframe[align=absmiddle i], img[align=absmiddle i],
input[type=image i][align=absmiddle i], object[align=absmiddle i],
embed[align=abscenter i], iframe[align=abscenter i], img[align=abscenter i],
input[type=image i][align=abscenter i], object[align=abscenter i] {
  vertical-align: middle;
}

embed[align=bottom i], iframe[align=bottom i], img[align=bottom i],
input[type=image i][align=bottom i], object[align=bottom i] {
  vertical-align: bottom;
}</code></pre>

  <p>When an <code>embed</code>, <code>iframe</code>, <code>img</code>, or <code>object</code>
  element, or an <code>input</code> element whose <code data-x="attr-input-type">type</code>
  attribute is in the <span data-x="attr-input-type-image">Image Button</span> state, has an <code
  data-x="">align</code> attribute whose value is an <span>ASCII case-insensitive</span> match for
  the string "<code data-x="">center</code>" or the string "<code data-x="">middle</code>", the user
  agent is expected to act as if the element's <span>'vertical-align'</span> property was set to a
  value that aligns the vertical middle of the element with the parent element's baseline.</p>

  <p>The <code data-x="">hspace</code> attribute of <code>embed</code>,
  <code>img</code>, or <code>object</code> elements, and <code>input</code> elements with a <code
  data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-image">Image
  Button</span> state, <span data-x="maps to the dimension property">maps to the dimension
  properties</span> <span>'margin-left'</span> and <span>'margin-right'</span> on the element.</p>

  <p>The <code data-x="">vspace</code> attribute of <code>embed</code>,
  <code>img</code>, or <code>object</code> elements, and <code>input</code> elements with a <code
  data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-image">Image
  Button</span> state, <span data-x="maps to the dimension property">maps to the dimension
  properties</span> <span>'margin-top'</span> and <span>'margin-bottom'</span> on the element.</p>

  <p>When an <code>iframe</code> element has a <code
  data-x="attr-iframe-frameborder">frameborder</code> attribute whose value, when parsed using the
  <span>rules for parsing integers</span>, is zero or an error, the user agent is expected to have
  <span>presentational hints</span> setting the element's <span>'border-top-width'</span>,
  <span>'border-right-width'</span>, <span>'border-bottom-width'</span>, and
  <span>'border-left-width'</span> properties to zero.</p>

  <p>When an <code>img</code> element, <code>object</code> element, or <code>input</code> element
  with a <code data-x="attr-input-type">type</code> attribute in the <span
  data-x="attr-input-type-image">Image Button</span> state has a <code
  data-x="">border</code> attribute whose value, when parsed using the <span>rules for
  parsing non-negative integers</span>, is found to be a number greater than zero, the user agent is
  expected to use the parsed value for eight <span>presentational hints</span>: four setting the
  parsed value as a pixel length for the element's <span>'border-top-width'</span>,
  <span>'border-right-width'</span>, <span>'border-bottom-width'</span>, and
  <span>'border-left-width'</span> properties, and four setting the element's
  <span>'border-top-style'</span>, <span>'border-right-style'</span>,
  <span>'border-bottom-style'</span>, and <span>'border-left-style'</span> properties to the value
  'solid'.</p>

  <p id="dimRendering">The <code data-x="attr-dim-width">width</code> and <code
  data-x="attr-dim-height">height</code> attributes on an <code>img</code> element's <span
  data-x="concept-img-dimension-attribute-source">dimension attribute source</span> <span
  data-x="maps to the dimension property">map to the dimension properties</span>
  <span>'width'</span> and <span>'height'</span> on the <code>img</code> element respectively. They
  similarly <span>map to the aspect-ratio property (using dimension rules)</span> of the
  <code>img</code> element.</p>

  <p>The <code data-x="attr-dim-width">width</code> and <code data-x="attr-dim-height">height</code>
  attributes on <code>embed</code>, <code>iframe</code>, <code>object</code>, and <code>video</code>
  elements, and <code>input</code> elements with a <code data-x="attr-input-type">type</code>
  attribute in the <span data-x="attr-input-type-image">Image Button</span> state and that either
  represents an image or that the user expects will eventually represent an image, <span
  data-x="maps to the dimension property">map to the dimension properties</span>
  <span>'width'</span> and <span>'height'</span> on the element respectively.</p>

  <p>The <code data-x="attr-dim-width">width</code> and <code data-x="attr-dim-height">height</code>
  attributes <span>map to the aspect-ratio property (using dimension rules)</span> on
  <code>img</code> and <code>video</code> elements, and <code>input</code> elements with a <code
  data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-image">Image
  Button</span> state.</p>

  <p>The <code data-x="attr-canvas-width">width</code> and <code
  data-x="attr-canvas-height">height</code> attributes <span>map to the aspect-ratio property</span>
  on <code>canvas</code> elements.</p>
  </div>


  <div w-nodev>

  <h4>Image maps</h4>

  <p>Shapes on an <span>image map</span> are expected to act, for the purpose of the CSS cascade, as
  elements independent of the original <code>area</code> element that happen to match the same style
  rules but inherit from the <code>img</code> or <code>object</code> element.</p>

  <p>For the purposes of the rendering, only the <span>'cursor'</span> property is expected to have
  any effect on the shape.</p>

  <p class="example">Thus, for example, if an <code>area</code> element has a <code
  data-x="attr-style">style</code> attribute that sets the <span>'cursor'</span> property to 'help',
  then when the user designates that shape, the cursor would change to a Help cursor.</p>

  <p class="example">Similarly, if an <code>area</code> element had a CSS rule that set its
  <span>'cursor'</span> property to 'inherit' (or if no rule setting the <span>'cursor'</span>
  property matched the element at all), the shape's cursor would be inherited from the
  <code>img</code> or <code>object</code> element of the <span>image map</span>, not from the parent
  of the <code>area</code> element.</p>

  </div>


  <div w-nodev>

  <h3 id="widgets"><span id="bindings"></span>Widgets</h3>

  <h4>Native appearance</h4>

  <p>The <cite>CSS Basic User Interface</cite> specification calls elements that can have a
  <span>native appearance</span> <span data-x="widget">widgets</span>, and defines whether to use that <span>native
  appearance</span> depending on the <span>'appearance'</span> property. That logic, in
  turn, depends on whether each the element is classified as a <span>devolvable
  widget</span> or <span>non-devolvable widget</span>. This section defines which elements match
  these concepts for HTML, what their <span>native appearance</span> is, and any particularity of
  their <span data-x="devolved widget">devolved</span> state or <span>primitive appearance</span>.
  <ref>CSSUI</ref></p>

  <p>The following elements can have a <span>native appearance</span> for the purpose of the CSS
  <span>'appearance'</span> property.</p>

  <ul class="brief">
   <li><code>button</code></li>
   <li><code>input</code></li>
   <li><code>meter</code></li>
   <li><code>progress</code></li>
   <li><code>select</code></li>
   <li><code>textarea</code></li>
  </ul>

  </div>


  <div w-nodev>

  <h4>Writing mode</h4>

  <p>Several widgets have their rendering controlled by the <span>'writing-mode'</span> CSS
  property. For the purposes of those widgets, we have the following definitions.</p>

  <p>A <dfn>horizontal writing mode</dfn> is when resolving the <span>'writing-mode'</span>
  property of the control results in a computed value of 'horizontal-tb'.</p>

  <p>A <dfn>vertical writing mode</dfn> is when resolving the <span>'writing-mode'</span> property
  of the control results in a computed value of either 'vertical-rl', 'vertical-lr', 'sideways-rl'
  or 'sideways-lr'.</p>

  </div>


  <div w-nodev>

  <h4>Button layout</h4>

  <p>When an element uses <span>button layout</span>, it is a <span>devolvable widget</span>, and
  it's <span>native appearance</span> is that of a button.</p>

  <p><dfn>Button layout</dfn> is as follows:</p>

  <ul>
   <li>
    <p>If the element is a <code>button</code> element, then the <span>'display'</span> property is
    expected to act as follows:</p>

    <ul>
     <li><p>If the computed value of <span>'display'</span> is 'inline-grid', 'grid',
     'inline-flex', 'flex', 'none', or 'contents', then behave as the computed value.</p></li>

     <li><p>Otherwise, if the computed value of <span>'display'</span> is a value such that the
     <span>outer display type</span> is 'inline', then behave as 'inline-block'.</p></li>

     <li><p>Otherwise, behave as 'flow-root'.</p></li>
    </ul>
   </li>

   <li><p>The element is expected to establish a new <span>formatting context</span> for its
   contents. The type of this formatting context is determined by its <span>'display'</span> value,
   as usual.</p></li>

   <li><p>If the element is <span>absolutely-positioned</span>, then for the purpose of the
   <a href="https://drafts.csswg.org/css2/#visuren">CSS visual formatting model</a>, act as if
   the element is a <span>replaced element</span>. <ref>CSS</ref></p></li>

   <li><p>If the <span>computed value</span> of <span>'inline-size'</span> is 'auto', then the
   <span>used value</span> is the <span>fit-content inline size</span>.</p></li>

   <li><p>For the purpose of the 'normal' keyword of the <span>'align-self'</span> property, act
   as if the element is a replaced element.</p></li>

   <li>
    <p>If the element is an <code>input</code> element, or if it is a <code>button</code> element
    and its computed value for <span>'display'</span> is not 'inline-grid', 'grid', 'inline-flex',
    or 'flex', then the element's box has a child <dfn>anonymous button content box</dfn> with the
    following behaviors:</p>

    <ul>
     <li><p>The box is a <span>block-level</span> <span>block container</span> that establishes a
     new <span>block formatting context</span> (i.e., <span>'display'</span> is
     'flow-root').</p></li>

     <li><p>If the box does not overflow in the horizontal axis, then it is centered
     horizontally.</p></li>

     <li><p>If the box does not overflow in the vertical axis, then it is centered
     vertically.</p></li>
    </ul>

    <p>Otherwise, there is no <span>anonymous button content box</span>.</p>
   </li>
  </ul>

  <p class="XXX">Need to define the expected <span>primitive appearance</span>.</p>

  <h4>The <code>button</code> element</h4>

  <p>The <code>button</code> element, when it generates a <span>CSS box</span>, is expected to
  depict a button and to use <span>button layout</span> whose <span>anonymous button content
  box</span>'s contents (if there is an <span>anonymous button content box</span>) are the child
  boxes the element's box would otherwise have.</p>

  </div>

  <div w-nodev>

  <h4>The <code>details</code> and <code>summary</code> elements</h4>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

details, summary {
  display: block;
}
details > summary:first-of-type {
  display: list-item;
  counter-increment: list-item 0;
  list-style: disclosure-closed inside;
}
details[open] > summary:first-of-type {
  list-style-type: disclosure-open;
}</code></pre>

  <p>The <code>details</code> element is expected to have an internal <span>shadow tree</span> with
  three child elements:</p>

  <ol>
   <li>
    <p>The first child element is a <code>slot</code> that is expected to take the
    <code>details</code> element's first <code>summary</code> element child, if any.  This element
    has a single child <code>summary</code> element called the <dfn>default summary</dfn> which has
    text content that is <span>implementation-defined</span> (and probably locale-specific).</p>

    <p>The <code>summary</code> element that this slot <span>represents</span> is expected to allow
    the user to request the details be shown or hidden.</p>
   </li>

   <li>
    <p>The second child element is a <code>slot</code> that is expected to take the
    <code>details</code> element's remaining descendants, if any.  This element has no contents.</p>

    <p>This element is expected to match the <span>'::details-content'</span> pseudo-element.</p>

    <p>This element is expected to have its <code data-x="attr-style">style</code> attribute set to
    "<code data-x="">display: block; content-visibility: hidden;</code>" when the
    <code>details</code> element does not have an <code data-x="attr-details-open">open</code>
    attribute. When it does have the <code data-x="attr-details-open">open</code> attribute, the
    <code data-x="attr-style">style</code> attribute is expected to be set to
    "<code data-x="">display: block;</code>".</p>

    <p class="note">Because the slots are hidden inside a shadow tree, this <code
    data-x="attr-style">style</code> attribute is not directly visible to author code. Its impacts,
    however, are visible. Notably, the choice of <code data-x="">content-visibility: hidden</code>
    instead of, e.g., <code data-x="">display: none</code>, impacts the results of various APIs that
    query layout information.</p>
   </li>

   <li>
    <p>The third child element is either a <code>link</code> or <code>style</code>
    element with the following styles for the <span>default summary</span>:</p>
    <pre><code class="css">:host summary {
  display: list-item;
  counter-increment: list-item 0;
  list-style: disclosure-closed inside;
}
:host([open]) summary {
  list-style-type: disclosure-open;
}</code></pre>
    <p class="note">The position of this child element relative to the other two is not observable.
    This means that implementations might have it in a different order relative to its siblings.
    Implementations might even associate the style with the shadow tree using a mechanism that is
    not an element.</p>
   </li>

  </ol>

  <p class="note">The structure of this shadow tree is observable through the ways that the children
  of the <code>details</code> element and the '::details-content' pseudo-element respond to CSS
  styles.</p>

  <!-- https://mail.gnome.org/archives/usability/2006-June/msg00015.html -->

  </div>

  <div w-nodev>

  <h4>The <code>input</code> element as a text entry widget</h4>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-text">Text</span>, <span
  data-x="attr-input-type-tel">Telephone</span>, <span data-x="attr-input-type-url">URL</span>, or
  <span data-x="attr-input-type-email">Email</span> state, is a <span>devolvable widget</span>. Its
  expected <span>native appearance</span> is to render as an <span>'inline-block'</span> box depicting
  a one-line text control.</p>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-search">Search</span> state is a <span>devolvable widget</span>.
  Its expected <span>native appearance</span> is to render as an <span>'inline-block'</span> box
  depicting a one-line text control. If the <span>computed value</span> of the element's
  <span>'appearance'</span> property is not <code>'textfield'</code>, it may have a distinct style
  indicating that it is a search field.</p>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-password">Password</span> state is a <span>devolvable
  widget</span>. Its expected <span>native appearance</span> is to render as an
  <span>'inline-block'</span> box depicting a one-line text control that obscures data entry.</p>

  <p>For <code>input</code> elements whose <code data-x="attr-input-type">type</code> attribute is
  in one of the above states, the <span>used value</span> of the <span>'line-height'</span> property
  must be a length value that is no smaller than what the <span>used value</span> would be for
  'line-height: normal'.</p>

  <p class="note">The <span>used value</span> will not be the actual keyword 'normal'. Also, this
  rule does not affect the <span>computed value</span>.</p>

  <p>If these text controls provide a text selection, then, when the user changes the current
  selection, the user agent is expected to <span>queue an element task</span> on the <span>user
  interaction task source</span> given the <code>input</code> element to <span
  data-x="concept-event-fire">fire an event</span> named <code data-x="event-select">select</code>
  at the element, with the <code data-x="dom-Event-bubbles">bubbles</code> attribute initialized to
  true.</p>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is
  in one of the above states is an <span>element with default preferred size</span>, and user
  agents are expected to apply the <span>'field-sizing'</span> CSS property to the element. User
  agents are expected to determine the <span>inline size</span> of its <span>intrinsic size</span>
  by the following steps:</p>

  <ol>
   <li>
    <p>If the <span>'field-sizing'</span> property on the element has a <span>computed value</span>
    of <span data-x="field-sizing-content">'content'</span>, the <span>inline size</span> is
    determined by the text which the element shows. The text is either a
    <span data-x="concept-fe-value">value</span> or a short hint specified by the
    <code data-x="attr-input-placeholder">placeholder</code> attribute. User agents may take the
    text caret size into account in the <span>inline size</span>.</p>
   </li>

   <li>
    <p>If the element has a <code data-x="attr-input-size">size</code> attribute, and parsing that
    attribute's value using the <span>rules for parsing non-negative integers</span> doesn't
    generate an error, return the value obtained from applying the <span>converting a character
    width to pixels</span> algorithm to the value of the attribute.</p>
   </li>

   <li>
    <p>Otherwise, return the value obtained from applying the <span>converting a character width to
    pixels</span> algorithm to the number 20.</p>
   </li>
  </ol>

  <p>The <dfn>converting a character width to pixels</dfn> algorithm returns <span data-x="">(<var>size</var>-1)&times;<var>avg</var>&nbsp;+&nbsp;<var>max</var></span>,
  where <var>size</var> is the character width to convert, <var>avg</var> is the
  average character width of the primary font for the element for which the algorithm is being run,
  in pixels, and <var>max</var> is the maximum character width of that same font, also in
  pixels. (The element's <span>'letter-spacing'</span> property does not affect the result.)</p>

  <p>These text controls are expected to be <span data-x="scroll container">scroll
  containers</span> and support scrolling in the <span>inline axis</span>, but not the <span>block
  axis</span>.</p>

  <p class="XXX">Need to detail the expected <span>native appearance</span> and <span>primitive
  appearance</span>.</p>

  </div>


  <div w-nodev>

  <h4>The <code>input</code> element as domain-specific widgets</h4>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-date">Date</span> state is a <span>devolvable widget</span>
  expected to render as an <span>'inline-block'</span> box depicting a date control.</p>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-month">Month</span> state is a <span>devolvable widget</span>
  expected to render as an <span>'inline-block'</span> box depicting a month control.</p>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-week">Week</span> state is a <span>devolvable widget</span>
  expected to render as an <span>'inline-block'</span> box depicting a week control.</p>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-time">Time</span> state is a <span>devolvable widget</span>
  expected to render as an <span>'inline-block'</span> box depicting a time control.</p>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-datetime-local">Local Date and Time</span> state is a
  <span>devolvable widget</span> expected to render as an <span>'inline-block'</span> box depicting
  a local date and time control.</p>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-number">Number</span> state is a <span>devolvable widget</span>
  expected to render as an <span>'inline-block'</span> box depicting a number control.</p>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-number">Number</span> state is an
  <span>element with default preferred size</span>, and user agents are expected to apply the
  <span>'field-sizing'</span> CSS property to the element. The <span>block size</span> of the
  <span>intrinsic size</span> is about one line high. If the <span>'field-sizing'</span> property
  on the element has a <span>computed value</span> of
  <span data-x="field-sizing-content">'content'</span>, the <span>inline size</span> of the
  <span>intrinsic size</span> is expected to be about as wide as necessary to show the current
  <span data-x="concept-fe-value">value</span>. Otherwise, the <span>inline size</span> of the
  <span>intrinsic size</span> is expected to be about as wide as necessary to show the widest
  possible value.</p>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-date">Date</span>,
  <span data-x="attr-input-type-month">Month</span>,
  <span data-x="attr-input-type-week">Week</span>, <span data-x="attr-input-type-time">Time</span>,
  or <span data-x="attr-input-type-datetime-local">Local Date and Time</span> state, is expected to
  be about one line high, and about as wide as necessary to show the widest possible value.</p>

  <p class="XXX">Need to detail the expected <span>native appearance</span> and <span>primitive
  appearance</span>.</p>

  </div>


  <div w-nodev>

  <h4>The <code>input</code> element as a range control</h4>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-range">Range</span> state is a <span>non-devolvable
  widget</span>. Its expected <span>native appearance</span> is to render as an
  <span>'inline-block'</span> box depicting a slider control.</p>

  <p>When this control has a <span>horizontal writing mode</span>, the control is expected to be a
  horizontal slider. Its lowest value is on the right if the <span>'direction'</span> property has a
  <span>computed value</span> of 'rtl', and on the left otherwise. When this control has a
  <span>vertical writing mode</span>, it is expected to be a vertical slider. Its lowest value is on
  the bottom if the <span>'direction'</span> property has a <span>computed value</span> of 'rtl', and
  on the top otherwise.</p>

  <p>Predefined suggested values (provided by the <code data-x="attr-input-list">list</code>
  attribute) are expected to be shown as tick marks on the slider, which the slider can snap to.</p>

  <p class="XXX">Need to detail the expected <span>primitive appearance</span>.</p>
  </div>


  <div w-nodev>

  <!--en-GB--><h4 id="the-input-element-as-a-colour-well">The <code>input</code> element as a color
  well</h4>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-color">Color</span> state is expected to depict a color well,
  which, when activated, provides the user with a color picker (e.g. a color wheel or color
  palette) from which the color can be changed. The element, when it generates a <span>CSS
  box</span>, is expected to use <span>button layout</span>, that has no child boxes of the
  <span>anonymous button content box</span>. The <span>anonymous button content box</span> is
  expected to have a <span data-x="presentational hints">presentational hint</span> setting the
  <span>'background-color'</span> property to the element's <span
  data-x="concept-fe-value">value</span>.</p>

  <p>Predefined suggested values (provided by the <code data-x="attr-input-list">list</code>
  attribute) are expected to be shown in the color picker interface, not on the color well
  itself.</p>

  <p class="XXX">Need to detail the expected <span>native appearance</span> and <span>primitive
  appearance</span>.</p>

  </div>


  <div w-nodev>

  <h4>The <code>input</code> element as a checkbox and radio button widgets</h4>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-checkbox">Checkbox</span> state is a <span>non-devolvable
  widget</span> expected to render as an <span>'inline-block'</span> box containing a single
  checkbox control, with no label.</p>

  <p class="XXX">Need to detail the expected <span>native appearance</span> and <span>primitive
  appearance</span>.</p>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-radio">Radio Button</span> state is a <span>non-devolvable
  widget</span> expected to render as an <span>'inline-block'</span> box containing a single radio
  button control, with no label.</p>

  <p class="XXX">Need to detail the expected <span>native appearance</span> and <span>primitive
  appearance</span>.</p>

  </div>


  <div w-nodev>

  <h4>The <code>input</code> element as a file upload control</h4>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-file">File Upload</span> state, when it generates a <span>CSS
  box</span>, is expected to render as an <span>'inline-block'</span> box containing a span of text
  giving the filename(s) of the <span data-x="concept-input-type-file-selected">selected
  files</span>, if any, followed by a button that, when activated, provides the user with a file
  picker from which the selection can be changed. The button is expected to use <span>button
  layout</span> and match the <span>'::file-selector-button'</span> pseudo-element. The contents of
  its <span>anonymous button content box</span> are expected to be
  <span>implementation-defined</span> (and possibly locale-specific) text, for example "Choose
  file".</p>

  <p>User agents may handle an <code>input</code> element whose
  <code data-x="attr-input-type">type</code> attribute is in the
  <span data-x="attr-input-type-file">File Upload</span> state as an
  <span>element with default preferred size</span>, and user agents may apply the
  <span>'field-sizing'</span> CSS property to the element. If the <span>'field-sizing'</span>
  property on the element has a <span>computed value</span> of
  <span data-x="field-sizing-content">'content'</span>, the <span>intrinsic size</span> of the
  element is expected to depend on its content such as the <span>'::file-selector-button'</span>
  pseudo-element and chosen file names.</p>

  </div>


  <div w-nodev>

  <h4>The <code>input</code> element as a button</h4>

  <p>An <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in
  the <span data-x="attr-input-type-submit">Submit Button</span>, <span
  data-x="attr-input-type-reset">Reset Button</span>, or <span
  data-x="attr-input-type-button">Button</span> state, when it generates a <span>CSS box</span>, is
  expected to depict a button and use <span>button layout</span> and the contents of the
  <span>anonymous button content box</span> are expected to be the text of the element's <code
  data-x="attr-input-value">value</code> attribute, if any, or text derived from the element's <code
  data-x="attr-input-type">type</code> attribute in an <span>implementation-defined</span> (and
  probably locale-specific) fashion, if not.</p>

  </div>


  <div w-nodev>

  <h4>The <code>marquee</code> element</h4>

  <pre><code class="css">@namespace "http://www.w3.org/1999/xhtml";

marquee {
  display: inline-block;
  text-align: initial;
  overflow: hidden !important;
}</code></pre>

  <p>The <code>marquee</code> element, while <span data-x="concept-marquee-on">turned on</span>, is
  expected to render in an animated fashion according to its attributes as follows:</p>

  <dl>
   <dt>If the element's <code data-x="attr-marquee-behavior">behavior</code> attribute is in the
   <span data-x="attr-marquee-behavior-scroll">scroll</span> state</dt>

   <dd>
    <p>Slide the contents of the element in the direction described by the <code
    data-x="attr-marquee-direction">direction</code> attribute as defined below, such that it begins
    off the start side of the <code>marquee</code>, and ends flush with the inner end side.</p>

    <p class="example">For example, if the <code data-x="attr-marquee-direction">direction</code>
    attribute is <span data-x="attr-marquee-direction-left">left</span> (the default), then the
    contents would start such that their left edge are off the side of the right edge of the
    <code>marquee</code>'s <span>content area</span>, and the contents would then slide up to the
    point where the left edge of the contents are flush with the left inner edge of the
    <code>marquee</code>'s <span>content area</span>.</p>

    <p>Once the animation has ended, the user agent is expected to <span>increment the marquee
    current loop index</span>. If the element is still <span data-x="concept-marquee-on">turned
    on</span> after this, then the user agent is expected to restart the animation.</p>
   </dd>

   <dt>If the element's <code data-x="attr-marquee-behavior">behavior</code> attribute is in the
   <span data-x="attr-marquee-behavior-slide">slide</span> state</dt>

   <dd>
    <p>Slide the contents of the element in the direction described by the <code
    data-x="attr-marquee-direction">direction</code> attribute as defined below, such that it begins
    off the start side of the <code>marquee</code>, and ends off the end side of the
    <code>marquee</code>.</p>

    <p class="example">For example, if the <code data-x="attr-marquee-direction">direction</code>
    attribute is <span data-x="attr-marquee-direction-left">left</span> (the default), then the
    contents would start such that their left edge are off the side of the right edge of the
    <code>marquee</code>'s <span>content area</span>, and the contents would then slide up to the
    point where the <em>right</em> edge of the contents are flush with the left inner edge of the
    <code>marquee</code>'s <span>content area</span>.</p>

    <p>Once the animation has ended, the user agent is expected to <span>increment the marquee
    current loop index</span>. If the element is still <span data-x="concept-marquee-on">turned
    on</span> after this, then the user agent is expected to restart the animation.</p>
   </dd>

   <dt>If the element's <code data-x="attr-marquee-behavior">behavior</code> attribute is in the
   <span data-x="attr-marquee-behavior-alternate">alternate</span> state</dt>

   <dd>
    <p>When the <span>marquee current loop index</span> is even (or zero), slide the contents of the
    element in the direction described by the <code data-x="attr-marquee-direction">direction</code>
    attribute as defined below, such that it begins flush with the start side of the
    <code>marquee</code>, and ends flush with the end side of the <code>marquee</code>.</p>

    <p>When the <span>marquee current loop index</span> is odd, slide the contents of the element in
    the opposite direction than that described by the <code
    data-x="attr-marquee-direction">direction</code> attribute as defined below, such that it begins
    flush with the end side of the <code>marquee</code>, and ends flush with the start side of the
    <code>marquee</code>.</p>

    <p class="example">For example, if the <code data-x="attr-marquee-direction">direction</code>
    attribute is <span data-x="attr-marquee-direction-left">left</span> (the default), then the
    contents would with their right edge flush with the right inner edge of the
    <code>marquee</code>'s <span>content area</span>, and the contents would then slide up to the
    point where the <em>left</em> edge of the contents are flush with the left inner edge of the
    <code>marquee</code>'s <span>content area</span>.</p>

    <p>Once the animation has ended, the user agent is expected to <span>increment the marquee
    current loop index</span>. If the element is still <span data-x="concept-marquee-on">turned
    on</span> after this, then the user agent is expected to continue the animation.</p>
   </dd>
  </dl>

  <p>The <code data-x="attr-marquee-direction">direction</code> attribute has the meanings described
  in the following table:</p>

  <table>
   <thead>
    <tr>
     <th><code data-x="attr-marquee-direction">direction</code> attribute state
     <th>Direction of animation
     <th>Start edge
     <th>End edge
     <th>Opposite direction
   <tbody>
    <tr>
     <td><span data-x="attr-marquee-direction-left">left</span>
     <td>&larr; Right to left
     <td>Right
     <td>Left
     <td>&rarr; Left to Right
    <tr>
     <td><span data-x="attr-marquee-direction-right">right</span>
     <td>&rarr; Left to Right
     <td>Left
     <td>Right
     <td>&larr; Right to left
    <tr>
     <td><span data-x="attr-marquee-direction-up">up</span>
     <td>&uarr; Up (Bottom to Top)
     <td>Bottom
     <td>Top
     <td>&darr; Down (Top to Bottom)
    <tr>
     <td><span data-x="attr-marquee-direction-down">down</span>
     <td>&darr; Down (Top to Bottom)
     <td>Top
     <td>Bottom
     <td>&uarr; Up (Bottom to Top)
  </table>

  <p>In any case, the animation should proceed such that there is a delay given by the <span>marquee
  scroll interval</span> between each frame, and such that the content moves at most the distance
  given by the <span>marquee scroll distance</span> with each frame.</p>

  <p>When a <code>marquee</code> element has a <code undefined data-x="attr-marquee-bgcolor">bgcolor</code>
  attribute set, the value is expected to be parsed using the <span>rules for parsing a legacy color
  value</span>, and if that does not return failure, the user agent is expected to treat the
  attribute as a <span data-x="presentational hints">presentational hint</span> setting the element's
  <span>'background-color'</span> property to the resulting color.</p>

  <p>The <code undefined data-x="attr-marquee-width">width</code> and <code
  undefined data-x="attr-marquee-height">height</code> attributes on a <code>marquee</code> element
  <span data-x="maps to the dimension property">map to the dimension properties</span>
  <span>'width'</span> and <span>'height'</span> on the element respectively.</p>

  <p>The <span>natural height</span> of a <code>marquee</code> element with its <code
  data-x="attr-marquee-direction">direction</code> attribute in the <span
  data-x="attr-marquee-direction-up">up</span> or <span
  data-x="attr-marquee-direction-down">down</span> states is 200 <span data-x="'px'">CSS
  pixels</span>.</p>

  <p>The <code undefined data-x="attr-marquee-vspace">vspace</code> attribute of a
  <code>marquee</code> element <span data-x="maps to the dimension property">maps to the dimension
  properties</span> <span>'margin-top'</span> and <span>'margin-bottom'</span> on the element. The
  <code undefined data-x="attr-marquee-hspace">hspace</code> attribute of a <code>marquee</code>
  element <span data-x="maps to the dimension property">maps to the dimension properties</span>
  <span>'margin-left'</span> and <span>'margin-right'</span> on the element.</p>

  </div>


  <div w-nodev>

  <h4>The <code>meter</code> element</h4>

  <pre><code class="css" data-x="">@namespace "http://www.w3.org/1999/xhtml";

meter { appearance: auto; }</code></pre>

  <p>The <code>meter</code> element is a <span>devolvable widget</span>. Its expected <span>native
  appearance</span> is to render as an <span>'inline-block'</span> box with a <span>'block-size'</span>
  of '1em' and a <span>'inline-size'</span> of '5em', a <span>'vertical-align'</span> of '-0.2em', and
  with its contents depicting a gauge.</p>

  <p>When this element has a <span>horizontal writing mode</span>, the depiction is expected to be of
  a horizontal gauge. Its minimum value is on the right if the <span>'direction'</span> property has
  a <span>computed value</span> of 'rtl', and on the left otherwise. When this element has a
  <span>vertical writing mode</span>, it is expected to depict a vertical gauge. Its minimum value is
  on the bottom if the <span>'direction'</span> property has a <span>computed value</span> of 'rtl',
  and on the top otherwise.</p>

  <p>User agents are expected to use a presentation consistent with platform conventions for gauges,
  if any.</p>

  <p class="note">Requirements for what <!--non-normative-->must be depicted in the gauge are
  included in the definition of the <code>meter</code> element.</p>

  <p class="XXX">Need to detail the expected <span>primitive appearance</span>.</p>

  </div>


  <div w-nodev>

  <h4>The <code>progress</code> element</h4>

  <pre><code class="css" data-x="">@namespace "http://www.w3.org/1999/xhtml";

progress { appearance: auto; }</code></pre>

  <p>The <code>progress</code> element is a <span>devolvable widget</span>. Its expected
  <span>native appearance</span> is to render as an <span>'inline-block'</span> box with a
  <span>'block-size'</span> of '1em' and a <span>'inline-size'</span> of '10em', and a
  <span>'vertical-align'</span> of '-0.2em'.</p>

  <!-- https://software.hixie.ch/utilities/js/canvas/?c.clearRect(0%2C%200%2C%20640%2C%20480)%3B%0Ac.save()%3B%0Atry%20%7B%0A%20%20c.fillStyle%20%3D%20%27black%27%3B%0A%20%20c.font%20%3D%20%278px%20sans-serif%27%3B%0A%20%20c.fillText(%27Horizontal%20ltr%27%2C%2030%2C105)%3B%0A%20%20c.fillText(%27Horizontal%20rtl%27%2C125%2C105)%3B%0A%20%20c.fillText(%27Vertical%20ltr%27%2C%20192%2C105)%3B%0A%20%20c.fillText(%27Vertical%20rtl%27%2C%20233%2C105)%3B%0A%20%20c.font%20%3D%20%27700%2010px%20sans-serif%27%3B%0A%20%20c.fillText(%27Progress%20Bars%27%2C%2013%2C30)%3B%0A%20%20c.font%20%3D%20%27100%2010px%20sans-serif%27%3B%0A%20%20c.fillText(%27(80%25)%27%2C%2037%2C45)%3B%0A%0A%20%20c.beginPath()%3B%0A%20%20var%20g%20%3D%20c.createLinearGradient(10%2C0%2C80%2C0)%3B%0A%20%20g.addColorStop(0%2C%20%27%2300FF00%27)%3B%0A%20%20g.addColorStop(0.8%2C%20%27%2300FF00%27)%3B%0A%20%20g.addColorStop(0.9%2C%20%27%23FFFF00%27)%3B%0A%20%20c.fillStyle%20%3D%20g%3B%0A%20%20c.rect(10%2C80%2C80%2C15)%3B%0A%20%20c.fill()%3B%0A%20%20c.strokeStyle%20%3D%20%27black%27%3B%0A%20%20c.stroke()%3B%0A%0A%20%20c.beginPath()%3B%0A%20%20var%20g%20%3D%20c.createLinearGradient(105%2C0%2C175%2C0)%3B%0A%20%20g.addColorStop(0%2C%20%27%23FFFF00%27)%3B%0A%20%20g.addColorStop(0.2%2C%20%27%23FFFF00%27)%3B%0A%20%20g.addColorStop(0.4%2C%20%27%2300FF00%27)%3B%0A%20%20c.fillStyle%20%3D%20g%3B%0A%20%20c.rect(105%2C80%2C80%2C15)%3B%0A%20%20c.fill()%3B%0A%20%20c.strokeStyle%20%3D%20%27black%27%3B%0A%20%20c.stroke()%3B%0A%0A%20%20c.beginPath()%3B%0A%20%20var%20g%20%3D%20c.createLinearGradient(0%2C80%2C0%2C20)%3B%0A%20%20g.addColorStop(0%2C%20%27%2300FF00%27)%3B%0A%20%20g.addColorStop(0.75%2C%20%27%2300FF00%27)%3B%0A%20%20g.addColorStop(0.85%2C%20%27%23FFFF00%27)%3B%0A%20%20c.fillStyle%20%3D%20g%3B%0A%20%20c.rect(203%2C15%2C15%2C80)%3B%0A%20%20c.fill()%3B%0A%20%20c.strokeStyle%20%3D%20%27black%27%3B%0A%20%20c.stroke()%3B%0A%0A%20%20c.beginPath()%3B%0A%20%20var%20g%20%3D%20c.createLinearGradient(0%2C80%2C0%2C20)%3B%0A%20%20g.addColorStop(0%2C%20%27%23FFFF00%27)%3B%0A%20%20g.addColorStop(0.15%2C%20%27%2300FF00%27)%3B%0A%20%20c.fillStyle%20%3D%20g%3B%0A%20%20c.rect(242%2C15%2C15%2C80)%3B%0A%20%20c.fill()%3B%0A%20%20c.strokeStyle%20%3D%20%27black%27%3B%0A%20%20c.stroke()%3B%0A%7D%20finally%20%7B%0A%20%20c.restore()%3B%0A%7D%0A -->

  <p> <img src="/images/sample-progress.png" alt="" width=276 height=115 class="extra"> When the this
  element has a <span>horizontal writing mode</span>, the element is expected to be depicted as a
  horizontal progress bar. The start is on the right and the end is on the left if the
  <span>'direction'</span> property on this element has a <span>computed value</span> of 'rtl', and
  with the start on the left and the end on the right otherwise. When this element has a
  <span>vertical writing mode</span>, it is expected to be depicted as a vertical progress bar. The
  start is on the bottom and the end is on the top if the <span>'direction'</span> property on this
  element has a <span>computed value</span> of 'rtl', and with the start on the top and the end on
  the bottom otherwise.</p>

  <p>User agents are expected to use a presentation consistent with platform conventions for
  progress bars. In particular, user agents are expected to use different presentations for
  determinate and indeterminate progress bars. User agents are also expected to vary the
  presentation based on the dimensions of the element.</p>

  <p class="note">Requirements for how to determine if the progress bar is determinate or
  indeterminate, and what progress a determinate progress bar is to show, are included in the
  definition of the <code>progress</code> element.</p>

  <p class="XXX">Need to detail the expected <span>primitive appearance</span>.</p>

  </div>


  <div w-nodev>

  <h4>The <code>select</code> element</h4>

  <p>The <code>select</code> element is an <span>element with default preferred size</span>, and
  user agents are expected to apply the <span>'field-sizing'</span> CSS property to
  <code>select</code> elements.

  <p>A <code>select</code> element is either a <dfn export for="select">list box</dfn> or a <dfn
  export for="select">drop-down box</dfn>, depending on its attributes.</p>

  <p>A <code>select</code> element whose <code data-x="attr-select-multiple">multiple</code>
  attribute is present is expected to render as a multi-select <span>list box</span>.</p>

  <p>A <code>select</code> element whose <code data-x="attr-select-multiple">multiple</code>
  attribute is absent, and whose <span data-x="concept-select-size">display size</span> is greater
  than 1, is expected to render as a single-select <span>list box</span>.</p>

  <p>When the element renders as a <span>list box</span>, it is a <span>devolvable widget</span>
  expected to render as an <span>'inline-block'</span> box. The <span>inline size</span> of its
  <span>intrinsic size</span> is the <span>width of the <code>select</code>'s labels</span> plus
  the width of a scrollbar. The <span>block size</span> of its <span>intrinsic size</span> is
  determined by the following steps:</p>

  <ol>
   <li>
    <p>If the <span>'field-sizing'</span> property on the element has a <span>computed value</span>
    of <span data-x="field-sizing-content">'content'</span>, return the height necessary to contain
    all rows for items.</p>
   </li>

   <li>
    <p>If the <code data-x="attr-select-size">size</code> attribute is absent or it has no valid
    value, return the height necessary to contain four rows.</p>
   </li>

   <li>
    <p>Otherwise, return the height necessary to contain as many rows for items as given by the
    element's <span data-x="concept-select-size">display size</span>.</p>
   </li>
  </ol>

  <p>A <code>select</code> element whose <code data-x="attr-select-multiple">multiple</code>
  attribute is absent, and whose <span data-x="concept-select-size">display size</span> is 1, is
  expected to render as an <span>'inline-block'</span> one-line <span>drop-down box</span>.
  The <span>inline size</span> of its <span>intrinsic size</span> is the
  <span>width of the <code>select</code>'s labels</span>. If the <span>'field-sizing'</span>
  property on the element has a <span>computed value</span> of
  <span data-x="field-sizing-content">'content'</span>, the <span>inline size</span> of the
  <span>intrinsic size</span> depends on the shown text. The shown text is typically the label of
  an <code>option</code> of which <span data-x="concept-option-selectedness">selectedness</span> is
  set to true.</p>

  <p>When the element renders as a <span>drop-down box</span>, it is a <span>devolvable
  widget</span>. Its appearance in the devolved state, as well as its appearance when the
  <span>computed value</span> of the element's <span>'appearance'</span> property is
  <code>'menulist-button'</code>, is that of a drop-down box, including a "drop-down button",
  but not necessarily rendered using a native control of the host operating system. In such a state,
  CSS properties such as <span>'color'</span>, <span>'background-color'</span>, and 'border' should
  not be disregarded (as is generally permissible when rendering an element according to its
  <span>native appearance</span>).</p>

  <p>In either case (<span>list box</span> or <span>drop-down box</span>), the element's items are
  expected to be the element's <span data-x="concept-select-option-list">list of options</span>,
  with the element's <code>optgroup</code> element <span data-x="concept-tree-child">children</span>
  providing headers for groups of options where applicable.</p>

  <p>An <code>optgroup</code> element is expected to be rendered by displaying the element's <code
  data-x="attr-optgroup-label">label</code> attribute.</p>

  <p>An <code>option</code> element is expected to be rendered by displaying the element's <span
  data-x="concept-option-label">label</span>, indented under its <code>optgroup</code> element if it
  has one.</p>

  <p>Each sequence of one or more child <code>hr</code> element siblings may be rendered as a single
  separator.</p>

  <p>The <dfn>width of the <code>select</code>'s labels</dfn> is the wider of the width necessary to
  render the widest <code>optgroup</code>, and the width necessary to render the widest
  <code>option</code> element in the element's <span data-x="concept-select-option-list">list of
  options</span> (including its indent, if any).</p>

  <p>If a <code>select</code> element contains a <span>placeholder label option</span>, the user
  agent is expected to render that <code>option</code> in a manner that conveys that it is a label,
  rather than a valid option of the control. This can include preventing the <span>placeholder label
  option</span> from being explicitly selected by the user. When the <span>placeholder label
  option</span>'s <span data-x="concept-option-selectedness">selectedness</span> is true, the control
  is expected to be displayed in a fashion that indicates that no valid option is currently
  selected.</p>

  <p>User agents are expected to render the labels in a <code>select</code> in such a manner that
  any alignment remains consistent whether the label is being displayed as part of the page or in a
  menu control.</p>

  <p class="XXX">Need to detail the expected <span>native appearance</span> and <span>primitive
  appearance</span>.</p>

  </div>


  <div w-nodev>

  <h4>The <code>textarea</code> element</h4>

  <p>The <code>textarea</code> element is a <span>devolvable widget</span> expected to render as an
  <span>'inline-block'</span> box depicting a multiline text control. If this multiline text control
  provides a selection, then, when the user changes the current selection, the user agent is
  expected to <span>queue an element task</span> on the <span>user interaction task source</span>
  given the <code>textarea</code> element to <span data-x="concept-event-fire">fire an event</span>
  named <code data-x="event-select">select</code> at the element, with the <code
  data-x="dom-Event-bubbles">bubbles</code> attribute initialized to true.</p>

  <p>The <code>textarea</code> element is an <span>element with default preferred size</span>, and
  user agents are expected to apply the <span>'field-sizing'</span> CSS property to
  <code>textarea</code> elements.

  <p>If the <span>'field-sizing'</span> property on the element has a <span>computed value</span>
  of <span data-x="field-sizing-content">'content'</span>, the <span>intrinsic size</span> is
  determined from the text which the element shows. The text is either a
  <span data-x="concept-textarea-raw-value">raw value</span> or a short hint specified by the
  <code data-x="attr-textarea-placeholder">placeholder</code> attribute. User agents may take the
  text caret size into account in the <span>intrinsic size</span>. Otherwise, its
  <span>intrinsic size</span> is computed from <span>textarea effective width</span> and
  <span>textarea effective height</span> (as defined below).</p>

  <p>The <dfn>textarea effective width</dfn> of a <code>textarea</code> element is <span
  data-x=""><var>size</var>&times;<var>avg</var>&nbsp;+&nbsp;<var>sbw</var></span>, where
  <var>size</var> is the element's <span data-x="attr-textarea-cols-value">character width</span>,
  <var>avg</var> is the average character width of the primary font of the element, in <span
  data-x="'px'">CSS pixels</span>, and <var>sbw</var> is the width of a scrollbar, in <span
  data-x="'px'">CSS pixels</span>. (The element's <span>'letter-spacing'</span> property does not
  affect the result.)</p>

  <p>The <dfn>textarea effective height</dfn> of a <code>textarea</code> element is the height in
  <span data-x="'px'">CSS pixels</span> of the number of lines specified the element's <span
  data-x="attr-textarea-rows-value">character height</span>, plus the height of a scrollbar in <span
  data-x="'px'">CSS pixels</span>.</p>

  <p>User agents are expected to apply the <span>'white-space'</span> CSS property to
  <code>textarea</code> elements. For historical reasons, if the element has a <code
  data-x="attr-textarea-wrap">wrap</code> attribute whose value is an <span>ASCII
  case-insensitive</span> match for the string "<code data-x="attr-textarea-wrap-off"
  undefined>off</code>", then the user agent is expected to treat the attribute as a <span
  data-x="presentational hints">presentational hint</span> setting the element's
  <span>'white-space'</span> property to 'pre'.</p>

  <p class="XXX">Need to detail the expected <span>native appearance</span> and <span>primitive
  appearance</span>.</p>

  </div>



  <div w-nodev>

  <h3>Frames and framesets</h3>

  <p>User agents are expected to render <code>frameset</code> elements as a box with the height and
  width of the <span>viewport</span>, with a surface rendered according to the following layout
  algorithm:</p>

  <ol>
   <li>
    <p>The <var>cols</var> and <var>rows</var> variables are lists of zero or more pairs consisting
    of a number and a unit, the unit being one of <i>percentage</i>, <i>relative</i>, and
    <i>absolute</i>.</p>

    <p>Use the <span>rules for parsing a list of dimensions</span> to parse the value of the
    element's <code undefined data-x="attr-frameset-cols">cols</code> attribute, if there is one.
    Let <var>cols</var> be the result, or an empty list if there is no such attribute.</p>

    <p>Use the <span>rules for parsing a list of dimensions</span> to parse the value of the
    element's <code undefined data-x="attr-frameset-rows">rows</code> attribute, if there is one.
    Let <var>rows</var> be the result, or an empty list if there is no such attribute.</p>
   </li>

   <li>
    <p>For any of the entries in <var>cols</var> or <var>rows</var> that have the number zero and
    the unit <i>relative</i>, change the entry's number to one.</p>
   </li>

   <li>
    <p>If <var>cols</var> has no entries, then add a single entry consisting of the value 1 and the
    unit <i>relative</i> to <var>cols</var>.</p>

    <p>If <var>rows</var> has no entries, then add a single entry consisting of the value 1 and the
    unit <i>relative</i> to <var>rows</var>.</p>
   </li>

   <li>
    <p>Invoke the algorithm defined below to <span>convert a list of dimensions to a list of pixel
    values</span> using <var>cols</var> as the input list, and the width of the surface that the
    <code>frameset</code> is being rendered into, in <span data-x="'px'">CSS pixels</span>, as the
    input dimension. Let <var>sized cols</var> be the resulting list.</p>

    <p>Invoke the algorithm defined below to <span>convert a list of dimensions to a list of pixel
    values</span> using <var>rows</var> as the input list, and the height of the surface that the
    <code>frameset</code> is being rendered into, in <span data-x="'px'">CSS pixels</span>, as the
    input dimension. Let <var>sized rows</var> be the resulting list.</p>
   </li>

   <li>
    <p>Split the surface into a grid of <var>w</var>&times;<var>h</var>
    rectangles, where <var>w</var> is the number of entries in <var>sized cols</var> and
    <var>h</var> is the number of entries in <var>sized rows</var>.</p>

    <p>Size the columns so that each column in the grid is as many <span data-x="'px'">CSS
    pixels</span> wide as the corresponding entry in the <var>sized cols</var> list.</p>

    <p>Size the rows so that each row in the grid is as many <span data-x="'px'">CSS pixels</span>
    high as the corresponding entry in the <var>sized rows</var> list.</p>
   </li>

   <li>
    <p>Let <var>children</var> be the list of <code>frame</code> and <code>frameset</code> elements
    that are <span data-x="concept-tree-child">children</span> of the <code>frameset</code> element
    for which the algorithm was invoked.</p>
   </li>

   <li>
    <p>For each row of the grid of rectangles created in the previous step, from top to bottom, run
    these substeps:</p>

    <ol>
     <li>
      <p>For each rectangle in the row, from left to right, run these substeps:</p>

      <ol>
       <li>
        <p>If there are any elements left in <var>children</var>, take the first element in the
        list, and assign it to the rectangle.</p>

        <p>If this is a <code>frameset</code> element, then recurse the entire <code>frameset</code>
        layout algorithm for that <code>frameset</code> element, with the rectangle as the
        surface.</p>

        <p>Otherwise, it is a <code>frame</code> element; render its <span>content
        navigable</span>, positioned and sized to fit the rectangle.</p>
       </li>

       <li>
        <p>If there are any elements left in <var>children</var>, remove the first element from
        <var>children</var>.</p>
       </li>
      </ol>
     </li>
    </ol>
   </li>

   <li>
    <p>If the <code>frameset</code> element <span>has a border</span>, draw an outer set of borders
    around the rectangles, using the element's <span>frame border color</span>.</p>

    <p>For each rectangle, if there is an element assigned to that rectangle, and that element
    <span>has a border</span>, draw an inner set of borders around that rectangle, using the
    element's <span>frame border color</span>.</p>

    <p>For each (visible) border that does not abut a rectangle that is assigned a
    <code>frame</code> element with a <code undefined data-x="attr-frame-noresize">noresize</code>
    attribute (including rectangles in further nested <code>frameset</code> elements), the user
    agent is expected to allow the user to move the border, resizing the rectangles within, keeping
    the proportions of any nested <code>frameset</code> grids.</p>

    <p>A <code>frameset</code> or <code>frame</code> element <dfn>has a border</dfn> if the
    following algorithm returns true:</p>

    <ol>
     <li><p>If the element has a <code data-x="">frameborder</code> attribute whose value is not the
     empty string and whose first character is either a U+0031 DIGIT ONE (1) character, a U+0079
     LATIN SMALL LETTER Y character (y), or a U+0059 LATIN CAPITAL LETTER Y character (Y), then
     return true.</p></li>

     <li><p>Otherwise, if the element has a <code data-x="">frameborder</code> attribute, return
     false.</p></li>

     <li><p>Otherwise, if the element has a parent element that is a <code>frameset</code> element,
     then return true if <em>that</em> element <span>has a border</span>, and false if it does
     not.</p></li>

     <li><p>Otherwise, return true.</p></li>
    </ol>

    <p>The <!--en-GB--><dfn id="frame-border-colour">frame border color</dfn> of a
    <code>frameset</code> or <code>frame</code> element is the color obtained from the following
    algorithm:</p>

    <ol>
     <li><p>If the element has a <code data-x="">bordercolor</code> attribute, and applying the
     <span>rules for parsing a legacy color value</span> to that attribute's value does not return
     failure, then return the color so obtained.</p></li>

     <li><p>Otherwise, if the element has a parent element that is a <code>frameset</code> element,
     then return the <span>frame border color</span> of that element.</p>

     <li><p>Otherwise, return gray.</p></li>
    </ol>
   </li>
  </ol>

  <p>The algorithm to <dfn>convert a list of dimensions to a list of pixel values</dfn> consists of
  the following steps:</p>

  <ol>
   <li>
    <p>Let <var>input list</var> be the list of numbers and units passed to the algorithm.</p>

    <p>Let <var>output list</var> be a list of numbers the same length as <var>input list</var>, all
    zero.</p>

    <p>Entries in <var>output list</var> correspond to the entries in <var>input list</var> that
    have the same position.</p>
   </li>

   <li><p>Let <var>input dimension</var> be the size passed to the algorithm.</p>

   <li>
    <p>Let <var>total percentage</var> be the sum of all the numbers in <var>input list</var> whose
    unit is <i>percentage</i>.</p>

    <p>Let <var>total relative</var> be the sum of all the numbers in <var>input list</var> whose
    unit is <i>relative</i>.</p>

    <p>Let <var>total absolute</var> be the sum of all the numbers in <var>input list</var> whose
    unit is <i>absolute</i>.</p>

    <p>Let <var>remaining space</var> be the value of <var>input dimension</var>.</p>
   </li>

   <li>
    <p>If <var>total absolute</var> is greater than <var>remaining space</var>, then for each entry
    in <var>input list</var> whose unit is <i>absolute</i>, set the corresponding value in
    <var>output list</var> to the number of the entry in <var>input list</var> multiplied by
    <var>remaining space</var> and divided by <var>total absolute</var>. Then, set <var>remaining
    space</var> to zero.</p>

    <p>Otherwise, for each entry in <var>input list</var> whose unit is <i>absolute</i>, set the
    corresponding value in <var>output list</var> to the number of the entry in <var>input
    list</var>. Then, decrement <var>remaining space</var> by <var>total absolute</var>.</p>
   </li>

   <li>
    <p>If <var>total percentage</var> multiplied by the <var>input dimension</var> and divided by
    100 is greater than <var>remaining space</var>, then for each entry in <var>input list</var>
    whose unit is <i>percentage</i>, set the corresponding value in <var>output list</var> to the
    number of the entry in <var>input list</var> multiplied by <var>remaining space</var> and
    divided by <var>total percentage</var>. Then, set <var>remaining space</var> to zero.</p>

    <p>Otherwise, for each entry in <var>input list</var> whose unit is <i>percentage</i>, set the
    corresponding value in <var>output list</var> to the number of the entry in <var>input
    list</var> multiplied by the <var>input dimension</var> and divided by 100. Then, decrement
    <var>remaining space</var> by <var>total percentage</var> multiplied by the <var>input
    dimension</var> and divided by 100.</p>
   </li>

   <li>
    <p>For each entry in <var>input list</var> whose unit is <i>relative</i>, set the corresponding
    value in <var>output list</var> to the number of the entry in <var>input list</var> multiplied
    by <var>remaining space</var> and divided by <var>total relative</var>.</p>
   </li>

   <li><p>Return <var>output list</var>.</p></li>
  </ol>

  <p>User agents working with integer values for frame widths (as opposed to user agents that can
  lay frames out with subpixel accuracy) are expected to distribute the remainder first to the last
  entry whose unit is <i>relative</i>, then equally (not proportionally) to each entry whose unit is
  <i>percentage</i>, then equally (not proportionally) to each entry whose unit is <i>absolute</i>,
  and finally, failing all else, to the last entry.</p>

  <hr>

<!--
  <p>The user agent is expected to force the <span>'display'</span> property of <code>frame</code>
  elements to 'block', the <span>'height'</span> property of <code>frame</code> elements to 0, and
  the <span>'width'</span> property of <code>frame</code> elements to 0, irrespective of CSS rules.
  (This only matters for when a <code>frame</code> element is rendered outside a
  <code>frameset</code>.)</p>
  https://software.hixie.ch/utilities/js/live-dom-viewer/?saved=2722 - as of Jan 2014, Chrome did this, but Firefox did not
-->

  <p>The contents of a <code>frame</code> element that does not have a <code>frameset</code> parent
  are expected to be rendered as <span>transparent black</span>; the user agent is expected to not
  render its <span>content navigable</span> in this case, and its <span>content navigable</span> is
  expected to have a <span>viewport</span> with zero width and zero height.</p>

  </div>


  <div w-nodev>

  <h3>Interactive media</h3>

  <h4>Links, forms, and navigation</h4>

  <p>User agents are expected to allow the user to control aspects of <span>hyperlink</span>
  activation and <span>form submission</span>, such as which <span>navigable</span> is to be used
  for the subsequent <span data-x="navigate">navigation</span>.</p>

  <p>User agents are expected to allow users to discover the destination of <span
  data-x="hyperlink">hyperlinks</span> and of <span data-x="form">forms</span> before triggering their
  <span data-x="navigate">navigation</span>.</p>

  <p>User agents are expected to inform the user of whether a <span>hyperlink</span> includes
  <span>hyperlink auditing</span>, and to let them know at a minimum which domains will be contacted
  as part of such auditing.</p>

  <p>User agents may allow users to <span>navigate</span><!--DONAV cite=""--> <span
  data-x="navigable">navigables</span> to the URLs <span data-x="encoding-parsing a
  URL">indicated</span> by the <code data-x="">cite</code> attributes on <code>q</code>,
  <code>blockquote</code>, <code>ins</code>, and <code>del</code> elements.</p>

  <p>User agents may surface <span data-x="hyperlink">hyperlinks</span> created by <code>link</code>
  elements in their user interface, as discussed <a
  href="#providing-users-with-a-means-to-follow-hyperlinks-created-using-the-link-element">previously</a>.</p>




  <h4>The <code data-x="attr-title">title</code> attribute</h4>

  <p>User agents are expected to expose the <span>advisory information</span> of elements upon user
  request, and to make the user aware of the presence of such information.</p>

  <p>On interactive graphical systems where the user can use a pointing device, this could take the
  form of a tooltip. When the user is unable to use a pointing device, then the user agent is
  expected to make the content available in some other fashion, e.g. by making the element a
  <span>focusable area</span> and always displaying the <span>advisory information</span> of the
  currently <span>focused</span> element, or by showing the <span>advisory information</span> of the
  elements under the user's finger on a touch device as the user pans around the screen.</p>

  <p>U+000A LINE FEED (LF) characters are expected to cause line breaks in the tooltip; U+0009
  CHARACTER TABULATION (tab) characters are expected to render as a nonzero horizontal shift that
  lines up the next glyph with the next tab stop, with tab stops occurring at points that are
  multiples of 8 times the width of a U+0020 SPACE character.</p>

  <div class="example">

   <p>For example, a visual user agent could make elements with a <code
   data-x="attr-title">title</code> attribute <span>focusable</span>, and could make
   any <span>focused</span> element with a <code data-x="attr-title">title</code> attribute
   show its tooltip under the element while the element has focus. This would allow a user to tab
   around the document to find all the advisory text.</p>

  </div>

  <div class="example">

   <p>As another example, a screen reader could provide an audio cue when reading an element with a
   tooltip, with an associated key to read the last tooltip for which a cue was played.</p>

  </div>


  <h4>Editing hosts</h4>

  <p>The current text editing caret (i.e. the <span>active range</span>, if it is empty and in an
  <span>editing host</span>), if any, is expected to act like an inline
  <span>replaced element</span> with the vertical dimensions of the caret and with zero width for
  the purposes of the CSS rendering model.</p>

  <p class="note">This means that even an empty block can have the caret inside it, and that when
  the caret is in such an element, it prevents <span data-x="collapsing margins">margins from
  collapsing</span> through the element.</p>



  <h4>Text rendered in native user interfaces</h4>

  <p>User agents are expected to honor the Unicode semantics of text that is exposed in user
  interfaces, for example supporting the bidirectional algorithm in text shown in dialogs, title
  bars, popup menus, and tooltips. Text from the contents of elements is expected to be rendered in
  a manner that honors <span>the directionality</span> of the element from which the text was
  obtained. Text from attributes is expected to be rendered in a manner that honours the
  <span>directionality of the attribute</span>.</p>

  <div class="example">

   <p>Consider the following markup, which has Hebrew text asking for a programming language, the
   languages being text for which a left-to-right direction is important given the punctuation in
   some of their names:</p>

   <pre><code class="html">&lt;p dir="rtl" lang="he">
 &lt;label>
  <span data-x="" dir="rtl" lang="he">&#x5d1;&#x5d7;&#x5e8; &#x5e9;&#x5e4;&#x5ea; &#x5ea;&#x5db;&#x5e0;&#x5d5;&#x5ea;:</span>
  &lt;select>
   &lt;option dir="ltr">C++&lt;/option>
   &lt;option dir="ltr">C#&lt;/option>
   &lt;option dir="ltr">FreePascal&lt;/option>
   &lt;option dir="ltr">F#&lt;/option>
  &lt;/select>
 &lt;/label>
&lt;/p></code></pre>

   <p>If the <code>select</code> element was rendered as a drop down box, a correct rendering would
   ensure that the punctuation was the same both in the drop down, and in the box showing the
   current selection.</p>

   <p><img src="/images/bidiselect.png" width="206" height="105" alt=""></p> <!-- no need for alt
   text, the previous paragraph describes it completely -->

  </div>

  <div class="example">

   <p>The directionality of attributes depends on the attribute and on the element's <code
   data-x="attr-dir">dir</code> attribute, as the following example demonstrates. Consider this
   markup:</p>

   <pre><code class="html"><bdo dir=ltr>&lt;table>
 &lt;tr>
  &lt;th abbr="(&#x05D0;" dir=ltr>A
  &lt;th abbr="(&#x05D0;" dir=rtl>A
  &lt;th abbr="(&#x05D0;" dir=auto>A
&lt;/table></bdo></code></pre>

   <p>If the <code data-x="attr-th-abbr">abbr</code> attributes are rendered, e.g. in a tooltip or
   other user interface, the first will have a left parenthesis (because the direction is 'ltr'),
   the second will have a right parenthesis (because the direction is 'rtl'), and the third will
   have a right parenthesis (because the direction is determined <em>from the attribute value</em>
   to be 'rtl').</p>

   <p>However, if instead the attribute was not a <span>directionality-capable attribute</span>, the
   results would be different:</p>

   <pre><code class="html"><bdo dir=ltr>&lt;table>
 &lt;tr>
  &lt;th data-abbr="(&#x05D0;" dir=ltr>A
  &lt;th data-abbr="(&#x05D0;" dir=rtl>A
  &lt;th data-abbr="(&#x05D0;" dir=auto>A
&lt;/table></bdo></code></pre>

   <p>In this case, if the user agent were to expose the <code data-x="">data-abbr</code> attribute
   in the user interface (e.g. in a debugging environment), the last case would be rendered with a
   <em>left</em> parenthesis, because the direction would be determined from the element's
   contents.</p>

  </div>

  <p>A string provided by a script (e.g. the argument to <code
  data-x="dom-alert">window.alert()</code>) is expected to be treated as an independent set of one or
  more bidirectional algorithm paragraphs when displayed, as defined by the bidirectional algorithm,
  including, for instance, supporting the paragraph-breaking behavior of U+000A LINE FEED (LF)
  characters. For the purposes of determining the paragraph level of such text in the bidirectional
  algorithm, this specification does <em>not</em> provide a higher-level override of rules P2 and
  P3. <ref>BIDI</ref></p>

  <p>When necessary, authors can enforce a particular direction for a given paragraph by starting it
  with the Unicode U+200E LEFT-TO-RIGHT MARK or U+200F RIGHT-TO-LEFT MARK characters.</p>

  <div class="example">

   <p>Thus, the following script:</p>

   <pre><code class="js">alert('\u05DC\u05DE\u05D3 HTML \u05D4\u05D9\u05D5\u05DD!')</code></pre>

   <p>...would always result in a message reading
        "<bdo lang="" dir=rtl>&#x05DC;&#x05DE;&#x05D3;&nbsp;LMTH&nbsp;&#x05D4;&#x05D9;&#x05D5;&#x05DD;!</bdo>"
   (not "<bdo lang="" dir=ltr>&#x05D3;&#x05DE;&#x05DC;&nbsp;HTML&nbsp;&#x05DD;&#x05D5;&#x05D9;&#x05D4;!</bdo>"),
   regardless of the language of the user agent interface or the
   direction of the page or any of its elements.</p>

  </div>

  <div class="example">

   <p>For a more complex example, consider the following script:</p>

   <pre class="bad"><code class="js">/* Warning: this script does not handle right-to-left scripts correctly */
var s;
if (s = prompt('What is your name?')) {
  alert(s + '! Ok, Fred, ' + s + ', and Wilma will get the car.');
}</code></pre>

   <p>When the user enters "<kbd>Kitty</kbd>", the user agent would alert "<samp>Kitty! Ok, Fred,
   Kitty, and Wilma will get the car.</samp>". However, if the user enters "<kbd dir="rtl"
   lang="ar">&#x644;&#x627;&nbsp;&#x623;&#x641;&#x647;&#x645;</kbd>", then the bidirectional
   algorithm will determine that the direction of the paragraph is right-to-left, and so the output
   will be the following unintended mess: "<samp lang=""><bdo
   dir="rtl">&#x644;&#x627;&nbsp;&#x623;&#x641;&#x647;&#x645;!&nbsp;derF&nbsp;,kO,&nbsp;&#x644;&#x627;&nbsp;&#x623;&#x641;&#x647;&#x645;,&nbsp;rac&nbsp;eht&nbsp;teg&nbsp;lliw&nbsp;amliW&nbsp;dna.</bdo></samp>"</p>

   <p>To force an alert that starts with user-provided text (or other text of unknown
   directionality) to render left-to-right, the string can be prefixed with a U+200E LEFT-TO-RIGHT
   MARK character:</p>

   <pre><code class="js">var s;
if (s = prompt('What is your name?')) {
  alert('<strong>\u200E</strong>' + s + '! Ok, Fred, ' + s + ', and Wilma will get the car.');
}</code></pre>

  </div>



  <h3>Print media</h3>

  <p>User agents are expected to allow the user to request the opportunity to <dfn>obtain a physical
  form</dfn> (or a representation of a physical form) of a <code>Document</code>. For example,
  selecting the option to print a page or convert it to PDF format. <ref>PDF</ref></p>

  <p>When the user actually <span data-x="obtain a physical form">obtains a physical form</span> (or
  a representation of a physical form) of a <code>Document</code>, the user agent is expected to
  create a new rendering of the <code>Document</code> for the print media.</p>



  <h3>Unstyled XML documents</h3>

  <!-- https://hixie.ch/tests/evil/xml/ -->

  <p>HTML user agents may, in certain circumstances, find themselves rendering non-HTML documents
  that use vocabularies for which they lack any built-in knowledge. This section provides for a way
  for user agents to handle such documents in a somewhat useful manner.</p>

  <p>While a <code>Document</code> is an <span>unstyled document</span>, the user agent is expected
  to render <span>an unstyled document view</span>.</p>

  <p>A <code>Document</code> is an <dfn>unstyled document</dfn> while it matches the following
  conditions:</p>

  <ul>
   <li>The <code>Document</code> has no author style sheets (whether referenced by HTTP headers, processing instructions, elements like <code>link</code>, inline elements like <code>style</code>, or any other mechanism).
   <li>None of the elements in the <code>Document</code> have any <span>presentational hints</span>.
   <li>None of the elements in the <code>Document</code> have any <span data-x="css-styling-attribute">style attributes</span>.
   <li>None of the elements in the <code>Document</code> are in any of the following namespaces: <span>HTML namespace</span>, <span>SVG namespace</span>, <span>MathML namespace</span>
   <li>The <code>Document</code> has no <span>focusable area</span> (e.g. from XLink) other than the <span>viewport</span>.
   <li>The <code>Document</code> has no <span data-x="hyperlink">hyperlinks</span> (e.g. from XLink).
   <li>There exists no <span data-x="concept-script">script</span> whose <span
   data-x="concept-script-settings-object">settings object</span>'s <span
   data-x="concept-settings-object-global">global object</span> is a <code>Window</code> object with
   this <code>Document</code> as its <span data-x="concept-document-window">associated
   <code>Document</code></span>.
   <li>None of the elements in the <code>Document</code> have any registered event listeners.
  </ul>

  <p><dfn>An unstyled document view</dfn> is one where the DOM is not rendered according to CSS
  (which would, since there are no applicable styles in this context, just result in a wall of
  text), but is instead rendered in a manner that is useful for a developer. This could consist of
  just showing the <code>Document</code> object's source, maybe with syntax highlighting, or it
  could consist of displaying just the DOM tree, or simply a message saying that the page is not a
  styled document.</p>

  <p class="note">If a <code>Document</code> stops being an <span>unstyled document</span>, then the
  conditions above stop applying, and thus a user agent following these requirements will switch to
  using the regular CSS rendering.</p>

  </div>




  <h2 split-filename="obsolete" id="obsolete">Obsolete features</h2>

  <h3>Obsolete but conforming features</h3>

  <p>Features listed in this section will trigger warnings in conformance checkers.</p>

  <p>Authors should not specify a <code data-x="attr-img-border">border</code> attribute on an
  <code>img</code> element. If the attribute is present, its value must be the string "<code
  data-x="">0</code>". CSS should be used instead.</p>

  <p>Authors should not specify a <code data-x="attr-script-charset">charset</code> attribute on a
  <code>script</code> element. If the attribute is present, its value must be an <span>ASCII
  case-insensitive</span> match for "<code data-x="">utf-8</code>". (This has no effect in a
  document that conforms to the requirements elsewhere in this standard of being encoded as
  <span>UTF-8</span>.)</p>

  <p>Authors should not specify a <code data-x="attr-script-language">language</code> attribute on a
  <code>script</code> element. If the attribute is present, its value must be an <span>ASCII
  case-insensitive</span> match for the string "<code data-x="">JavaScript</code>" and either the
  <code data-x="attr-script-type">type</code> attribute must be omitted or its value must be an
  <span>ASCII case-insensitive</span> match for the string "<code data-x="">text/javascript</code>".
  The attribute should be entirely omitted instead (with the value "<code
  data-x="">JavaScript</code>", it has no effect), or replaced with use of the <code
  data-x="attr-script-type">type</code> attribute.</p>

  <p>Authors should not specify a value for the <code data-x="attr-script-type">type</code>
  attribute on <code>script</code> elements that is the empty string or a <span>JavaScript MIME type
  essence match</span>. Instead, they should omit the attribute, which has the same effect.</p>

  <p>Authors should not specify a <code data-x="attr-style-type">type</code> attribute on a
  <code>style</code> element. If the attribute is present, its value must be an <span>ASCII
  case-insensitive</span> match for "<code>text/css</code>".</p>

  <p>Authors should not specify the <code data-x="attr-a-name">name</code> attribute on
  <code>a</code> elements. If the attribute is present, its value must not be the empty string and
  must neither be equal to the value of any of the <span data-x="concept-id">IDs</span> in the
  element's <span>tree</span> other than the element's own <span data-x="concept-id">ID</span>, if
  any, nor be equal to the value of any of the other <code data-x="attr-a-name">name</code>
  attributes on <code>a</code> elements in the element's <span>tree</span>. If this attribute is
  present and the element has an <span data-x="concept-id">ID</span>, then the attribute's value
  must be equal to the element's <span data-x="concept-id">ID</span>. In earlier versions of the
  language, this attribute was intended as a way to specify possible targets for <span
  data-x="concept-url-fragment">fragments</span> in <span data-x="URL">URLs</span>. The <code
  data-x="attr-id">id</code> attribute should be used instead.</p>

  <!-- remove this once type=number is widely supported -->
  <p>Authors should not, but may despite requirements to the contrary elsewhere in this
  specification, specify the <code data-x="attr-input-maxlength">maxlength</code> and <code
  data-x="attr-input-size">size</code> attributes on <code>input</code> elements whose <code
  data-x="attr-input-type">type</code> attributes are in the <span
  data-x="attr-input-type-number">Number</span> state. One valid reason for using these attributes
  regardless is to help legacy user agents that do not support <code>input</code> elements with
  <code data-x="">type="number"</code> to still render the text control with a useful width.</p>


  <div w-nodev>

  <h4>Warnings for obsolete but conforming features</h4>

  <p>To ease the transition from HTML4 Transitional documents to the language defined in
  <em>this</em> specification, and to discourage certain features that are only allowed in very few
  circumstances, conformance checkers must warn the user when the following features are used in a
  document. These are generally old obsolete features that have no effect, and are allowed only to
  distinguish between likely mistakes (regular conformance errors) and mere vestigial markup or
  unusual and discouraged practices (these warnings).</p>

  <p>The following features must be categorized as described
  above:</p>

  <ul><!-- downplayed list -->

   <li><p>The presence of a <code data-x="attr-img-border">border</code> attribute on an
   <code>img</code> element if its value is the string "<code data-x="">0</code>".</p></li>

   <li><p>The presence of a <code data-x="attr-script-charset">charset</code> attribute on a
   <code>script</code> element if its value is an <span>ASCII case-insensitive</span> match for
   "<code data-x="">utf-8</code>".</p></li>

   <li><p>The presence of a <code data-x="attr-script-language">language</code> attribute on a
   <code>script</code> element if its value is an <span>ASCII case-insensitive</span> match for the
   string "<code data-x="">JavaScript</code>" and if there is no <code
   data-x="attr-script-type">type</code> attribute or there is and its value is an <span>ASCII
   case-insensitive</span> match for the string "<code data-x="">text/javascript</code>".</p></li>

   <li><p>The presence of a <code data-x="attr-style-type">type</code> attribute on a
   <code>script</code> element if its value is a <span>JavaScript MIME type essence
   match</span>.</p></li>

   <li><p>The presence of a <code data-x="attr-style-type">type</code> attribute on a
   <code>style</code> element if its value is an <span>ASCII case-insensitive</span> match for
   "<code>text/css</code>".</p></li>

   <li><p>The presence of a <code data-x="attr-a-name">name</code> attribute on an <code>a</code>
   element, if its value is not the empty string.</p></li>

   <li><p>The presence of a <code data-x="attr-input-maxlength">maxlength</code> attribute on an
   <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in the
   <span data-x="attr-input-type-number">Number</span> state.</p></li>

   <li><p>The presence of a <code data-x="attr-input-size">size</code> attribute on an
   <code>input</code> element whose <code data-x="attr-input-type">type</code> attribute is in the
   <span data-x="attr-input-type-number">Number</span> state.</p></li>
  </ul>

  <p>Conformance checkers must distinguish between pages that have no conformance errors and have
  none of these obsolete features, and pages that have no conformance errors but do have some of
  these obsolete features.</p>

  <p class="example">For example, a validator could report some pages as "Valid HTML" and others as
  "Valid HTML with warnings".</p>

  </div>


  <h3>Non-conforming features</h3>

  <p>Elements in the following list are entirely obsolete, and must not be used by authors:</p>

  <dl><!-- alphabetical by first element in the group, except CSS goes last -->

   <dt><dfn element><code>applet</code></dfn></dt>
   <dd><p>Use <code>embed</code> or <code>object</code> instead.</p></dd>

   <dt><dfn element><code>acronym</code></dfn></dt>
   <dd><p>Use <code>abbr</code> instead.</p></dd>

   <dt><dfn element><code>bgsound</code></dfn></dt>
   <dd><p>Use <code>audio</code> instead.</p></dd>

   <dt><dfn element><code>dir</code></dfn></dt>
   <dd><p>Use <code>ul</code> instead.</p></dd>

   <dt><code>frame</code></dt>
   <dt><code>frameset</code></dt>
   <dt><dfn element><code>noframes</code></dfn></dt>
   <dd><p>Either use <code>iframe</code> and CSS instead, or use server-side includes to generate complete pages with the various invariant parts merged in.</p></dd>

   <dt><dfn element><code>isindex</code></dfn></dt>
   <dd><p>Use an explicit <code>form</code> and <span data-x="attr-input-type-text">text control</span> combination instead.</p></dd>

   <dt><dfn element><code>keygen</code></dfn></dt>
   <dd>
    <p>For enterprise device management use cases, use native on-device management capabilities.</p>

    <p>For certificate enrollment use cases, use the Web Cryptography API to generate a keypair for
    the certificate, and then export the certificate and key to allow the user to install them
    manually. <ref>WEBCRYPTO</ref></p>
   </dd>

   <dt><dfn element><code>listing</code></dfn></dt>
   <dd><p>Use <code>pre</code> and <code>code</code> instead.</p></dd>

   <dt><dfn element><code>menuitem</code></dfn></dt>
   <dd><p>To implement a custom context menu, use script to handle the <code
   data-x="event-contextmenu">contextmenu</code> event.</p></dd>

   <dt><dfn element><code>nextid</code></dfn></dt>
   <dd><p>Use GUIDs instead.</p></dd>

   <dt><dfn element><code>noembed</code></dfn></dt>
   <dd><p>Use <code>object</code> instead of <code>embed</code> when fallback is necessary.</p></dd>

   <dt id="the-param-element"><dfn element><code>param</code></dfn></dt>
   <dd><p>Use the <code data-x="attr-object-data">data</code> attribute of the <code>object</code> element to set the
   URL of the external resource.</p></dd>

   <dt><dfn element><code>plaintext</code></dfn></dt>
   <dd><p>Use the "<code>text/plain</code>" <span>MIME type</span> instead.</p></dd>

   <dt><dfn element><code>rb</code></dfn></dt>
   <dt><dfn element><code>rtc</code></dfn></dt>
   <dd><p>Providing the ruby base directly inside the <code>ruby</code> element or using nested
   <code>ruby</code> elements is sufficient.</p></dd>

   <dt><dfn element><code>strike</code></dfn></dt>
   <dd><p>Use <code>del</code> instead if the element is marking an edit, otherwise use <code>s</code> instead.</p></dd>

   <dt><dfn element><code>xmp</code></dfn></dt>
   <dd><p>Use <code>pre</code> and <code>code</code> instead, and escape "<code data-x="">&lt;</code>" and "<code data-x="">&amp;</code>" characters as "<code data-x="">&amp;lt;</code>" and "<code data-x="">&amp;amp;</code>" respectively.</p></dd>

   <dt><dfn element><code>basefont</code></dfn></dt>
   <dt><dfn element><code>big</code></dfn></dt>
   <dt><dfn element><code>blink</code></dfn></dt>
   <dt><dfn element><code>center</code></dfn></dt>
   <dt><dfn element><code>font</code></dfn></dt>
   <dt><code>marquee</code></dt>
   <dt><dfn element><code>multicol</code></dfn></dt>
   <dt><dfn element><code>nobr</code></dfn></dt>
   <dt><dfn element><code>spacer</code></dfn></dt>
   <dt><dfn element><code>tt</code></dfn></dt>

   <dd>
    <p>Use appropriate elements or CSS instead.</p>

    <p>Where the <code>tt</code> element would have been used for marking up keyboard input,
    consider the <code>kbd</code> element; for variables, consider the <code>var</code> element; for
    computer code, consider the <code>code</code> element; and for computer output, consider the
    <code>samp</code> element.</p>

    <p>Similarly, if the <code>big</code> element is being used to denote a heading, consider using
    the <code>h1</code> element; if it is being used for marking up important passages, consider the
    <code>strong</code> element; and if it is being used for highlighting text for reference
    purposes, consider the <code>mark</code> element.</p>

    <p>See also the <a href="#usage-summary">text-level semantics usage summary</a> for more
    suggestions with examples.</p>
   </dd>
  </dl>

  <hr>

  <p>The following attributes are obsolete (though the elements are still part of the language), and
  must not be used by authors:</p>

  <dl><!-- alphabetical by element then attribute of first item in group, except CSS goes last -->

   <dt><dfn element-attr for="a"><code data-x="attr-a-charset">charset</code></dfn> on <code>a</code> elements</dt>
   <dt><dfn element-attr for="link"><code data-x="attr-link-charset">charset</code></dfn> on <code>link</code> elements</dt>
   <dd><p>Use an HTTP `<code>Content-Type</code>` header on the linked resource instead.</p></dd>

   <dt><dfn element-attr for="script"><code data-x="attr-script-charset">charset</code></dfn> on <code>script</code>
   elements (except as noted in the previous section)</dt>
   <dd><p>Omit the attribute. Both documents and scripts are required to use <span>UTF-8</span>, so
   it is redundant to specify it on the <code>script</code> element since it inherits from the
   document.</p></dd>

   <dt><dfn element-attr for="a"><code data-x="attr-a-coords">coords</code></dfn> on <code>a</code> elements</dt>
   <dt><dfn element-attr for="a"><code data-x="attr-a-shape">shape</code></dfn> on <code>a</code> elements</dt>
   <dd><p>Use <code>area</code> instead of <code>a</code> for image maps.</p></dd>

   <dt><dfn element-attr for="a"><code data-x="attr-a-methods">methods</code></dfn> on <code>a</code> elements</dt>
   <dt><dfn element-attr for="link"><code data-x="attr-link-methods">methods</code></dfn> on <code>link</code> elements</dt>
   <dd><p>Use the HTTP OPTIONS feature instead.</p></dd>

   <dt><dfn element-attr for="a"><code data-x="attr-a-name">name</code></dfn> on <code>a</code> elements (except as noted in the previous section)</dt>
   <dt><dfn element-attr for="embed"><code data-x="attr-embed-name">name</code></dfn> on <code>embed</code> elements</dt>
   <dt><dfn element-attr for="img"><code data-x="attr-img-name">name</code></dfn> on <code>img</code> elements</dt>
   <dt><dfn element-attr for="option"><code data-x="attr-option-name">name</code></dfn> on <code>option</code> elements</dt>
   <dd><p>Use the <code data-x="attr-id">id</code> attribute instead.</p></dd>

   <dt><dfn element-attr for="a"><code data-x="attr-a-rev">rev</code></dfn> on <code>a</code> elements</dt>
   <dt><dfn element-attr for="link"><code data-x="attr-link-rev">rev</code></dfn> on <code>link</code> elements</dt>
   <dd><p>Use the <code data-x="attr-hyperlink-rel">rel</code>
   attribute instead, with an opposite term. (For example, instead of
   <code data-x="">rev="made"</code>, use <code
   data-x="">rel="author"</code>.)</p></dd>

   <dt><dfn element-attr for="a"><code data-x="attr-a-urn">urn</code></dfn> on <code>a</code> elements</dt>
   <dt><dfn element-attr for="link"><code data-x="attr-link-urn">urn</code></dfn> on <code>link</code> elements</dt>
   <dd><p>Specify the preferred persistent identifier using the <code data-x="attr-hyperlink-href">href</code> attribute instead.</p></dd>

   <dt><dfn element-attr for="form"><code data-x="attr-form-accept">accept</code></dfn> on <code>form</code> elements</dt>
   <dd><p>Use the <code data-x="attr-input-accept">accept</code> attribute directly on the <code>input</code> elements instead.</p></dd>

   <dt><dfn element-attr for="area"><code data-x="attr-area-hreflang">hreflang</code></dfn> on <code>area</code>
   elements</dt>
   <dt><dfn element-attr for="area"><code data-x="attr-area-type">type</code></dfn> on <code>area</code> elements</dt>
   <dd><p>These attributes do not do anything useful, and for historical reasons there are no
   corresponding IDL attributes on <code>area</code> elements. Omit them altogether.</p></dd>

   <dt><dfn element-attr for="area"><code data-x="attr-area-nohref">nohref</code></dfn> on <code>area</code> elements</dt>
   <dd><p>Omitting the <code data-x="attr-hyperlink-href">href</code>
   attribute is sufficient; the <code
   data-x="attr-area-nohref">nohref</code> attribute is
   unnecessary. Omit it altogether.</p></dd>

   <dt><dfn element-attr for="head"><code data-x="attr-head-profile">profile</code></dfn> on <code>head</code> elements</dt>
   <dd><p>Unnecessary. Omit it altogether.</p></dd>

   <dt><dfn element-attr for="html"><code data-x="attr-html-manifest">manifest</code></dfn> on <code>html</code> elements</dt>
   <dd><p>Use service workers instead. <ref>SW</ref></p></dd>

   <dt><dfn element-attr for="html"><code data-x="attr-html-version">version</code></dfn> on <code>html</code> elements</dt>
   <dd><p>Unnecessary. Omit it altogether.</p></dd>

   <dt><dfn element-attr for="input"><code data-x="attr-input-ismap">ismap</code></dfn> on <code>input</code> elements</dt>
   <dd><p>Unnecessary. Omit it altogether. All <code>input</code> elements with a <code
   data-x="attr-input-type">type</code> attribute in the <span data-x="attr-input-type-image">Image
   Button</span> state are processed as server-side image maps.</p></dd>

   <dt><dfn element-attr for="input"><code data-x="attr-input-usemap">usemap</code></dfn> on <code>input</code> elements</dt>
   <dt><dfn element-attr for="object"><code data-x="attr-object-usemap">usemap</code></dfn> on <code>object</code> elements</dt>
   <dd><p>Use the <code>img</code> element for image maps.</p></dd>

   <dt><dfn element-attr for="iframe"><code data-x="attr-iframe-longdesc">longdesc</code></dfn> on <code>iframe</code> elements</dt>
   <dt><dfn element-attr for="img"><code data-x="attr-img-longdesc">longdesc</code></dfn> on <code>img</code> elements</dt>
   <dd><p>Use a regular <code>a</code> element to link to the
   description, or (in the case of images) use an <span>image
   map</span> to provide a link from the image to the image's
   description.</p></dd>

   <dt><dfn element-attr for="img"><code data-x="attr-img-lowsrc">lowsrc</code></dfn> on <code>img</code> elements</dt>
   <dd><p>Use a progressive JPEG image (given in the <code data-x="attr-img-src">src</code> attribute),
   instead of using two separate images.</p></dd>

   <dt><dfn element-attr for="link"><code data-x="attr-link-target">target</code></dfn> on <code>link</code> elements</dt>
   <dd><p>Unnecessary. Omit it altogether.</p></dd>

   <dt><dfn element-attr for="menu"><code data-x="attr-menu-type">type</code></dfn> on <code>menu</code> elements</dt>
   <dd><p>To implement a custom context menu, use script to handle the <code
   data-x="event-contextmenu">contextmenu</code> event. For toolbar menus, omit the
   attribute.</p></dd>

   <dt><dfn element-attr for="menu"><code data-x="attr-menu-label">label</code></dfn> on <code>menu</code> elements</dt>
   <dt><dfn element-attr><code data-x="attr-contextmenu">contextmenu</code></dfn> on all elements</dt>
   <dt><dfn element-attr><code data-x="handler-onshow">onshow</code></dfn> on all elements</dt>
   <dd><p>To implement a custom context menu, use script to handle the <code
   data-x="event-contextmenu">contextmenu</code> event.</p></dd>

   <dt><dfn element-attr for="meta"><code data-x="attr-meta-scheme">scheme</code></dfn> on <code>meta</code> elements</dt>
   <dd><p>Use only one scheme per field, or make the scheme declaration part of the value.</p></dd>

   <dt><dfn element-attr for="object"><code data-x="attr-object-archive">archive</code></dfn> on <code>object</code> elements</dt>
   <dt><dfn element-attr for="object"><code data-x="attr-object-classid">classid</code></dfn> on <code>object</code> elements</dt>
   <dt><dfn element-attr for="object"><code data-x="attr-object-code">code</code></dfn> on <code>object</code> elements</dt>
   <dt><dfn element-attr for="object"><code data-x="attr-object-codebase">codebase</code></dfn> on <code>object</code> elements</dt>
   <dt><dfn element-attr for="object"><code data-x="attr-object-codetype">codetype</code></dfn> on <code>object</code> elements</dt>
   <dd><p>Use the <code data-x="attr-object-data">data</code> and <code
   data-x="attr-object-type">type</code> attributes to invoke <span
   data-x="plugin">plugins</span>.</p></dd>

   <dt><dfn element-attr for="object"><code data-x="attr-object-declare">declare</code></dfn> on <code>object</code> elements</dt>
   <dd><p>Repeat the <code>object</code> element completely each time the resource is to be reused.</p></dd>

   <dt><dfn element-attr for="object"><code data-x="attr-object-standby">standby</code></dfn> on <code>object</code> elements</dt>
   <dd><p>Optimize the linked resource so that it loads quickly or, at least, incrementally.</p></dd>

   <dt><dfn element-attr for="object"><code data-x="attr-object-typemustmatch">typemustmatch</code></dfn> on <code>object</code> elements</dt>
   <dd><p>Avoid using <code>object</code> elements with untrusted resources.</p></dd>

   <dt><dfn element-attr for="script"><code data-x="attr-script-language">language</code></dfn> on <code>script</code> elements (except as noted in the previous section)</dt>
   <dd><p>Omit the attribute for JavaScript; for <span data-x="data block">data blocks</span>, use
   the <code data-x="attr-script-type">type</code> attribute instead.</p></dd>

   <dt><dfn element-attr for="script"><code data-x="attr-script-event">event</code></dfn> on <code>script</code> elements</dt>
   <dt><dfn element-attr for="script"><code data-x="attr-script-for">for</code></dfn> on <code>script</code> elements</dt>
   <dd><p>Use DOM events mechanisms to register event listeners. <ref>DOM</ref></p></dd>

   <dt><dfn element-attr for="style"><code data-x="attr-style-type">type</code></dfn> on <code>style</code> elements (except as noted in the previous section)</dt>
   <dd><p>Omit the attribute for CSS; for <span data-x="data block">data blocks</span>, use
   <code>script</code> as the container instead of <code>style</code>.</p></dd>

   <dt><dfn element-attr for="table"><code data-x="attr-table-datapagesize">datapagesize</code></dfn> on <code>table</code> elements</dt>
   <dd><p>Unnecessary. Omit it altogether.</p></dd>

   <dt><dfn element-attr for="table"><code data-x="attr-table-summary">summary</code></dfn> on <code>table</code> elements</dt><!-- 2.65% pages -->
   <dd><p>Use one of the <a href="#table-descriptions-techniques">techniques for describing
   tables</a> given in the <code>table</code> section instead.</p></dd>

   <dt><dfn element-attr for="td"><code data-x="attr-td-abbr">abbr</code></dfn> on <code>td</code> elements</dt>
   <dd><p>Use text that begins in an unambiguous and terse manner, and include any more elaborate text after that. The <code data-x="attr-title">title</code> attribute can also be useful in including more detailed text, so that the cell's contents can be made terse. If it's a heading, use <code>th</code> (which has an <code data-x="attr-th-abbr">abbr</code> attribute).</p>

   <dt><dfn element-attr for="td,th"><code data-x="attr-tdth-axis">axis</code></dfn> on <code>td</code> and <code>th</code> elements</dt>
   <dd><p>Use the <code data-x="attr-th-scope">scope</code> attribute on the relevant <code>th</code>.</p>

   <dt><dfn element-attr for="td"><code data-x="attr-td-scope">scope</code></dfn> on <code>td</code> elements</dt>
   <dd><p>Use <code>th</code> elements for heading cells.</p>

   <dt><dfn element-attr><code data-x="attr-datasrc">datasrc</code></dfn> on <code>a</code>, <code>button</code>, <code>div</code>, <code>frame</code>, <code>iframe</code>, <code>img</code>, <code>input</code>, <code>label</code>, <code>legend</code>, <code>marquee</code>, <code>object</code>, <code>option</code>, <code>select</code>, <code>span</code>, <code>table</code>, and <code>textarea</code> elements</dt>
   <dt><dfn element-attr><code data-x="attr-datafld">datafld</code></dfn> on <code>a</code>, <code>button</code>, <code>div</code>, <code>fieldset</code>, <code>frame</code>, <code>iframe</code>, <code>img</code>, <code>input</code>, <code>label</code>, <code>legend</code>, <code>marquee</code>, <code>object</code>, <code>select</code>, <code>span</code>, and <code>textarea</code> elements</dt>
   <dt><dfn element-attr><code data-x="attr-dataformatas">dataformatas</code></dfn> on <code>button</code>, <code>div</code>, <code>input</code>, <code>label</code>, <code>legend</code>, <code>marquee</code>, <code>object</code>, <code>option</code>, <code>select</code>, <code>span</code>, and <code>table</code> elements</dt>
   <dd><p>Use script and a mechanism such as <code>XMLHttpRequest</code> to populate the page dynamically. <ref>XHR</ref></p></dd>

   <dt><dfn element-attr><code data-x="attr-dropzone">dropzone</code></dfn> on all elements</dt>
   <dd><p>Use script to handle the <code data-x="event-dnd-dragenter">dragenter</code> and <code
   data-x="event-dnd-dragover">dragover</code> events instead.</p></dd>

   <dt><dfn element-attr for="body"><code data-x="attr-body-alink">alink</code></dfn> on <code>body</code> elements</dt>
   <dt><dfn element-attr for="body"><code data-x="attr-body-bgcolor">bgcolor</code></dfn> on <code>body</code> elements</dt>
   <dt><dfn element-attr for="body"><code data-x="attr-body-bottommargin">bottommargin</code></dfn> on <code>body</code> elements</dt>
   <dt><dfn element-attr for="body"><code data-x="attr-body-leftmargin">leftmargin</code></dfn> on <code>body</code> elements</dt>
   <dt><dfn element-attr for="body"><code data-x="attr-body-link">link</code></dfn> on <code>body</code> elements</dt>
   <dt><dfn element-attr for="body"><code data-x="attr-body-marginheight">marginheight</code></dfn> on <code>body</code> elements</dt>
   <dt><dfn element-attr for="body"><code data-x="attr-body-marginwidth">marginwidth</code></dfn> on <code>body</code> elements</dt>
   <dt><dfn element-attr for="body"><code data-x="attr-body-rightmargin">rightmargin</code></dfn> on <code>body</code> elements</dt>
   <dt><dfn element-attr for="body"><code data-x="attr-body-text">text</code></dfn> on <code>body</code> elements</dt>
   <dt><dfn element-attr for="body"><code data-x="attr-body-topmargin">topmargin</code></dfn> on <code>body</code> elements</dt>
   <dt><dfn element-attr for="body"><code data-x="attr-body-vlink">vlink</code></dfn> on <code>body</code> elements</dt>
   <dt><dfn element-attr for="br"><code data-x="attr-br-clear">clear</code></dfn> on <code>br</code> elements</dt>
   <dt><dfn element-attr for="caption"><code data-x="attr-caption-align">align</code></dfn> on <code>caption</code> elements</dt>
   <dt><dfn element-attr for="col"><code data-x="attr-col-align">align</code></dfn> on <code>col</code> elements</dt>
   <dt><dfn element-attr for="col"><code data-x="attr-col-char">char</code></dfn> on <code>col</code> elements</dt>
   <dt><dfn element-attr for="col"><code data-x="attr-col-charoff">charoff</code></dfn> on <code>col</code> elements</dt>
   <dt><dfn element-attr for="col"><code data-x="attr-col-valign">valign</code></dfn> on <code>col</code> elements</dt>
   <dt><dfn element-attr for="col"><code data-x="attr-col-width">width</code></dfn> on <code>col</code> elements</dt>
   <dt><dfn element-attr for="div"><code data-x="attr-div-align">align</code></dfn> on <code>div</code> elements</dt>
   <dt><dfn element-attr for="dl"><code data-x="attr-dl-compact">compact</code></dfn> on <code>dl</code> elements</dt>
   <dt><dfn element-attr for="embed"><code data-x="attr-embed-align">align</code></dfn> on <code>embed</code> elements</dt>
   <dt><dfn element-attr for="embed"><code data-x="attr-embed-hspace">hspace</code></dfn> on <code>embed</code> elements</dt>
   <dt><dfn element-attr for="embed"><code data-x="attr-embed-vspace">vspace</code></dfn> on <code>embed</code> elements</dt>
   <dt><dfn element-attr for="hr"><code data-x="attr-hr-align">align</code></dfn> on <code>hr</code> elements</dt>
   <dt><dfn element-attr for="hr"><code data-x="attr-hr-color">color</code></dfn> on <code>hr</code> elements</dt>
   <dt><dfn element-attr for="hr"><code data-x="attr-hr-noshade">noshade</code></dfn> on <code>hr</code> elements</dt>
   <dt><dfn element-attr for="hr"><code data-x="attr-hr-size">size</code></dfn> on <code>hr</code> elements</dt>
   <dt><dfn element-attr for="hr"><code data-x="attr-hr-width">width</code></dfn> on <code>hr</code> elements</dt>
   <dt><dfn element-attr for="h1,h2,h3,h4,h5,h6"><code data-x="attr-hx-align">align</code></dfn> on <code>h1</code>&mdash;<code>h6</code> elements</dt>
   <dt><dfn element-attr for="iframe"><code data-x="attr-iframe-align">align</code></dfn> on <code>iframe</code> elements</dt>
   <dt><dfn element-attr for="iframe"><code data-x="attr-iframe-allowtransparency">allowtransparency</code></dfn> on <code>iframe</code> elements</dt>
   <dt><dfn element-attr for="iframe"><code data-x="attr-iframe-frameborder">frameborder</code></dfn> on <code>iframe</code> elements</dt>
   <dt><dfn element-attr for="iframe"><code data-x="attr-iframe-framespacing">framespacing</code></dfn> on <code>iframe</code> elements</dt>
   <dt><dfn element-attr for="iframe"><code data-x="attr-iframe-hspace">hspace</code></dfn> on <code>iframe</code> elements</dt>
   <dt><dfn element-attr for="iframe"><code data-x="attr-iframe-marginheight">marginheight</code></dfn> on <code>iframe</code> elements</dt>
   <dt><dfn element-attr for="iframe"><code data-x="attr-iframe-marginwidth">marginwidth</code></dfn> on <code>iframe</code> elements</dt>
   <dt><dfn element-attr for="iframe"><code data-x="attr-iframe-scrolling">scrolling</code></dfn> on <code>iframe</code> elements</dt>
   <dt><dfn element-attr for="iframe"><code data-x="attr-iframe-vspace">vspace</code></dfn> on <code>iframe</code> elements</dt>
   <dt><dfn element-attr for="input"><code data-x="attr-input-align">align</code></dfn> on <code>input</code> elements</dt>
   <dt><dfn element-attr for="input"><code data-x="attr-input-border">border</code></dfn> on <code>input</code> elements</dt>
   <dt><dfn element-attr for="input"><code data-x="attr-input-hspace">hspace</code></dfn> on <code>input</code> elements</dt>
   <dt><dfn element-attr for="input"><code data-x="attr-input-vspace">vspace</code></dfn> on <code>input</code> elements</dt>
   <dt><dfn element-attr for="img"><code data-x="attr-img-align">align</code></dfn> on <code>img</code> elements</dt>
   <dt><dfn element-attr for="img"><code data-x="attr-img-border">border</code></dfn> on <code>img</code> elements (except as noted in the previous section)</dt>
   <dt><dfn element-attr for="img"><code data-x="attr-img-hspace">hspace</code></dfn> on <code>img</code> elements</dt>
   <dt><dfn element-attr for="img"><code data-x="attr-img-vspace">vspace</code></dfn> on <code>img</code> elements</dt>
   <dt><dfn element-attr for="legend"><code data-x="attr-legend-align">align</code></dfn> on <code>legend</code> elements</dt>
   <dt><dfn element-attr for="li"><code data-x="attr-li-type">type</code></dfn> on <code>li</code> elements</dt>
   <dt><dfn element-attr for="menu"><code data-x="attr-menu-compact">compact</code></dfn> on <code>menu</code> elements</dt>
   <dt><dfn element-attr for="object"><code data-x="attr-object-align">align</code></dfn> on <code>object</code> elements</dt>
   <dt><dfn element-attr for="object"><code data-x="attr-object-border">border</code></dfn> on <code>object</code> elements</dt>
   <dt><dfn element-attr for="object"><code data-x="attr-object-hspace">hspace</code></dfn> on <code>object</code> elements</dt>
   <dt><dfn element-attr for="object"><code data-x="attr-object-vspace">vspace</code></dfn> on <code>object</code> elements</dt>
   <dt><dfn element-attr for="ol"><code data-x="attr-ol-compact">compact</code></dfn> on <code>ol</code> elements</dt>
   <dt><dfn element-attr for="p"><code data-x="attr-p-align">align</code></dfn> on <code>p</code> elements</dt>
   <dt><dfn element-attr for="pre"><code data-x="attr-pre-width">width</code></dfn> on <code>pre</code> elements</dt>
   <dt><dfn element-attr for="table"><code data-x="attr-table-align">align</code></dfn> on <code>table</code> elements</dt>
   <dt><dfn element-attr for="table"><code data-x="attr-table-bgcolor">bgcolor</code></dfn> on <code>table</code> elements</dt>
   <dt><dfn element-attr for="table"><code data-x="attr-table-border">border</code></dfn> on <code>table</code> elements</dt>
   <dt><dfn element-attr for="table"><code data-x="attr-table-bordercolor">bordercolor</code></dfn> on <code>table</code> elements</dt>
   <dt><dfn element-attr for="table"><code data-x="attr-table-cellpadding">cellpadding</code></dfn> on <code>table</code> elements</dt>
   <dt><dfn element-attr for="table"><code data-x="attr-table-cellspacing">cellspacing</code></dfn> on <code>table</code> elements</dt>
   <dt><dfn element-attr for="table"><code data-x="attr-table-frame">frame</code></dfn> on <code>table</code> elements</dt>
   <dt><dfn element-attr for="table"><code data-x="attr-table-height">height</code></dfn> on <code>table</code> elements</dt>
   <dt><dfn element-attr for="table"><code data-x="attr-table-rules">rules</code></dfn> on <code>table</code> elements</dt>
   <dt><dfn element-attr for="table"><code data-x="attr-table-width">width</code></dfn> on <code>table</code> elements</dt>
   <dt><dfn element-attr for="tbody"><code data-x="attr-tbody-align">align</code></dfn> on <code>tbody</code>, <code>thead</code>, and <code>tfoot</code> elements</dt>
   <dt><dfn element-attr for="tbody"><code data-x="attr-tbody-char">char</code></dfn> on <code>tbody</code>, <code>thead</code>, and <code>tfoot</code> elements</dt>
   <dt><dfn element-attr for="tbody"><code data-x="attr-tbody-charoff">charoff</code></dfn> on <code>tbody</code>, <code>thead</code>, and <code>tfoot</code> elements</dt>
   <dt><dfn element-attr for="tbody"><code data-x="attr-tbody-height">height</code></dfn> on <code>thead</code>, <code>tbody</code>, and <code>tfoot</code> elements</dt>
   <dt><dfn element-attr for="tbody"><code data-x="attr-tbody-vAlign">valign</code></dfn> on <code>tbody</code>, <code>thead</code>, and <code>tfoot</code> elements</dt>
   <dt><dfn element-attr for="td,th"><code data-x="attr-tdth-align">align</code></dfn> on <code>td</code> and <code>th</code> elements</dt>
   <dt><dfn element-attr for="td,th"><code data-x="attr-tdth-bgcolor">bgcolor</code></dfn> on <code>td</code> and <code>th</code> elements</dt>
   <dt><dfn element-attr for="td,th"><code data-x="attr-tdth-char">char</code></dfn> on <code>td</code> and <code>th</code> elements</dt>
   <dt><dfn element-attr for="td,th"><code data-x="attr-tdth-charoff">charoff</code></dfn> on <code>td</code> and <code>th</code> elements</dt>
   <dt><dfn element-attr for="td,th"><code data-x="attr-tdth-height">height</code></dfn> on <code>td</code> and <code>th</code> elements</dt>
   <dt><dfn element-attr for="td,th"><code data-x="attr-tdth-nowrap">nowrap</code></dfn> on <code>td</code> and <code>th</code> elements</dt>
   <dt><dfn element-attr for="td,th"><code data-x="attr-tdth-valign">valign</code></dfn> on <code>td</code> and <code>th</code> elements</dt>
   <dt><dfn element-attr for="td,th"><code data-x="attr-tdth-width">width</code></dfn> on <code>td</code> and <code>th</code> elements</dt>
   <dt><dfn element-attr for="tr"><code data-x="attr-tr-align">align</code></dfn> on <code>tr</code> elements</dt>
   <dt><dfn element-attr for="tr"><code data-x="attr-tr-bgcolor">bgcolor</code></dfn> on <code>tr</code> elements</dt>
   <dt><dfn element-attr for="tr"><code data-x="attr-tr-char">char</code></dfn> on <code>tr</code> elements</dt>
   <dt><dfn element-attr for="tr"><code data-x="attr-tr-charoff">charoff</code></dfn> on <code>tr</code> elements</dt>
   <dt><dfn element-attr for="tr"><code data-x="attr-tr-height">height</code></dfn> on <code>tr</code> elements</dt>
   <dt><dfn element-attr for="tr"><code data-x="attr-tr-valign">valign</code></dfn> on <code>tr</code> elements</dt>
   <dt><dfn element-attr for="ul"><code data-x="attr-ul-compact">compact</code></dfn> on <code>ul</code> elements</dt>
   <dt><dfn element-attr for="ul"><code data-x="attr-ul-type">type</code></dfn> on <code>ul</code> elements</dt>
   <dt><dfn element-attr><code data-x="attr-background">background</code></dfn> on <code>body</code>, <code>table</code>, <code>thead</code>, <code>tbody</code>, <code>tfoot</code>, <code>tr</code>, <code>td</code>, and <code>th</code> elements</dt>
   <dd><p>Use CSS instead.</p></dd>
  </dl>


  <div w-nodev>

  <h3>Requirements for implementations</h3>

  <h4 id="the-marquee-element">The <dfn element><code>marquee</code></dfn> element</h4>

  <p>The <code>marquee</code> element is a presentational element that animates content. CSS
  transitions and animations are a more appropriate mechanism. <ref>CSSANIMATIONS</ref> <ref>CSSTRANSITIONS</ref></p>

  <p>The <code>marquee</code> element must implement the <code>HTMLMarqueeElement</code>
  interface.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLMarqueeElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-marquee-behavior">behavior</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-marquee-bgColor">bgColor</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-marquee-direction">direction</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-marquee-height">height</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-marquee-hspace">hspace</span>;
  [<span>CEReactions</span>] attribute long <span data-x="dom-marquee-loop">loop</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-marquee-scrollamount">scrollAmount</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-marquee-scrollDelay">scrollDelay</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-marquee-trueSpeed">trueSpeed</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-marquee-vspace">vspace</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-marquee-width">width</span>;

  undefined <span data-x="dom-marquee-start">start</span>();
  undefined <span data-x="dom-marquee-stop">stop</span>();
};</code></pre>

  <p>A <code>marquee</code> element can be <dfn data-x="concept-marquee-on">turned on</dfn> or <dfn
  data-x="concept-marquee-off">turned off</dfn>. When it is created, it is <span
  data-x="concept-marquee-on">turned on</span>.</p>

  <p>When the <dfn method for="HTMLMarqueeElement"><code
  data-x="dom-marquee-start">start()</code></dfn> method is called, the <code>marquee</code> element
  must be <span data-x="concept-marquee-on">turned on</span>.</p>

  <p>When the <dfn method for="HTMLMarqueeElement"><code
  data-x="dom-marquee-stop">stop()</code></dfn> method is called, the <code>marquee</code> element
  must be <span data-x="concept-marquee-off">turned off</span>.</p>

  <hr>

  <p>The <dfn element-attr for="marquee"><code data-x="attr-marquee-behavior">behavior</code></dfn>
  content attribute on <code>marquee</code> elements is an <span>enumerated attribute</span> with
  the following keywords and states (all non-conforming):</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
   <tbody>
    <tr>
     <td><code data-x="">scroll</code>
     <td><dfn data-x="attr-marquee-behavior-scroll">scroll</dfn>
    <tr>
     <td><code data-x="">slide</code>
     <td><dfn data-x="attr-marquee-behavior-slide">slide</dfn>
    <tr>
     <td><code data-x="">alternate</code>
     <td><dfn data-x="attr-marquee-behavior-alternate">alternate</dfn>
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <span
  data-x="attr-marquee-behavior-scroll">scroll</span> state.</p>

  <hr>

  <p>The <dfn element-attr for="marquee"><code
  data-x="attr-marquee-direction">direction</code></dfn> content attribute on <code>marquee</code>
  elements is an <span>enumerated attribute</span> with the following keywords and states (all
  non-conforming):</p>

  <table>
   <thead>
    <tr>
     <th>Keyword
     <th>State
   <tbody>
    <tr>
     <td><code data-x="">left</code>
     <td><dfn data-x="attr-marquee-direction-left">left</dfn>
    <tr>
     <td><code data-x="">right</code>
     <td><dfn data-x="attr-marquee-direction-right">right</dfn>
    <tr>
     <td><code data-x="">up</code>
     <td><dfn data-x="attr-marquee-direction-up">up</dfn>
    <tr>
     <td><code data-x="">down</code>
     <td><dfn data-x="attr-marquee-direction-down">down</dfn>
  </table>

  <p>The attribute's <i data-x="missing value default">missing value default</i> and <i
  data-x="invalid value default">invalid value default</i> are both the <span
  data-x="attr-marquee-direction-left">left</span> state.</p>

  <hr>

  <p>The <dfn element-attr for="marquee"><code
  data-x="attr-marquee-truespeed">truespeed</code></dfn> content attribute on <code>marquee</code>
  elements is a <span>boolean attribute</span>.</p>

  <hr>

  <p>A <code>marquee</code> element has a <dfn>marquee scroll interval</dfn>, which is obtained as
  follows:</p>

  <ol>
   <li><p>If the element has a <code undefined data-x="attr-marquee-scrolldelay">scrolldelay</code> attribute,
   and parsing its value using the <span>rules for parsing non-negative integers</span> does not
   return an error, then let <var>delay</var> be the parsed value. Otherwise, let <var>delay</var>
   be 85.</p></li>

   <li><p>If the element does not have a <code data-x="attr-marquee-truespeed">truespeed</code>
   attribute, and the <var>delay</var> value is less than 60, then let <var>delay</var> be 60
   instead.</p></li>

   <li><p>The <span>marquee scroll interval</span> is <var>delay</var>, interpreted in
   milliseconds.</p></li>
  </ol>

  <hr>

  <p>A <code>marquee</code> element has a <dfn>marquee scroll distance</dfn>, which, if the element
  has a <code undefined data-x="attr-marquee-scrollamount">scrollamount</code> attribute, and
  parsing its value using the <span>rules for parsing non-negative integers</span> does not return
  an error, is the parsed value interpreted in <span data-x="'px'">CSS pixels</span>, and otherwise
  is 6 <span data-x="'px'">CSS pixels</span>.</p>

  <hr>

  <p>A <code>marquee</code> element has a <dfn>marquee loop count</dfn>, which, if the element has a
  <dfn element-attr for="marquee"><code data-x="attr-marquee-loop">loop</code></dfn> attribute, and
  parsing its value using the <span>rules for parsing integers</span> does not return an error or a
  number less than 1, is the parsed value, and otherwise is &#x2212;1.</p>

  <p>The <dfn attribute for="HTMLMarqueeElement"><code data-x="dom-marquee-loop">loop</code></dfn>
  IDL attribute, on getting, must return the element's <span>marquee loop count</span>; and on
  setting, if the new value is different than the element's <span>marquee loop count</span> and
  either greater than zero or equal to &#x2212;1, must set the element's <code
  data-x="attr-marquee-loop">loop</code> content attribute (adding it if necessary) to the
  <span>valid integer</span> that represents the new value. (Other values are ignored.)</p>

  <p>A <code>marquee</code> element also has a <dfn>marquee current loop index</dfn>, which is zero
  when the element is created.</p>

  <p>The rendering layer will occasionally <dfn>increment the marquee current loop index</dfn>,
  which must cause the following steps to be run:</p>

  <ol>
   <li><p>If the <span>marquee loop count</span> is &#x2212;1, then return.</p>

   <li><p>Increment the <span>marquee current loop index</span> by one.</p></li>

   <li><p>If the <span>marquee current loop index</span> is now greater than or equal to the
   element's <span>marquee loop count</span>, <span data-x="concept-marquee-off">turn off</span> the
   <code>marquee</code> element.</p></li>
  </ol>

  <hr>

  <p>The <dfn attribute for="HTMLMarqueeElement"><code
  data-x="dom-marquee-behavior">behavior</code></dfn>, <dfn attribute for="HTMLMarqueeElement"><code
  data-x="dom-marquee-direction">direction</code></dfn>, <dfn attribute
  for="HTMLMarqueeElement"><code data-x="dom-marquee-height">height</code></dfn>, <dfn attribute
  for="HTMLMarqueeElement"><code data-x="dom-marquee-hspace">hspace</code></dfn>, <dfn attribute
  for="HTMLMarqueeElement"><code data-x="dom-marquee-vspace">vspace</code></dfn>, and <dfn attribute
  for="HTMLMarqueeElement"><code data-x="dom-marquee-width">width</code></dfn> IDL attributes must
  <span>reflect</span> the respective content attributes of the same name.</p>

  <p>The <dfn attribute for="HTMLMarqueeElement"><code
  data-x="dom-marquee-bgColor">bgColor</code></dfn> IDL attribute must <span>reflect</span> the
  <code undefined data-x="attr-marquee-bgcolor">bgcolor</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLMarqueeElement"><code
  data-x="dom-marquee-scrollAmount">scrollAmount</code></dfn> IDL attribute must
  <span>reflect</span> the <code undefined data-x="attr-marquee-scrollamount">scrollamount</code>
  content attribute. The <span>default value</span> is 6.</p>

  <p>The <dfn attribute for="HTMLMarqueeElement"><code
  data-x="dom-marquee-scrollDelay">scrollDelay</code></dfn> IDL attribute must <span>reflect</span>
  the <code undefined data-x="attr-marquee-scrolldelay">scrolldelay</code> content attribute. The
  <span>default value</span> is 85.</p>

  <p>The <dfn attribute for="HTMLMarqueeElement"><code
  data-x="dom-marquee-trueSpeed">trueSpeed</code></dfn> IDL attribute must <span>reflect</span> the
  <code data-x="attr-marquee-truespeed">truespeed</code> content attribute.</p>


  <h4>Frames</h4>

  <p>The <dfn element><code>frameset</code></dfn> element acts as <span>the body element</span> in
  documents that use frames.</p>

  <p>The <code>frameset</code> element must implement the <code>HTMLFrameSetElement</code>
  interface.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLFrameSetElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-frameset-cols">cols</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-frameset-rows">rows</span>;
};
<span>HTMLFrameSetElement</span> includes <span>WindowEventHandlers</span>;</code></pre>

  <p>The <dfn attribute for="HTMLFrameSetElement"><code data-x="dom-frameset-cols">cols</code></dfn>
  and <dfn attribute for="HTMLFrameSetElement"><code data-x="dom-frameset-rows">rows</code></dfn>
  IDL attributes of the <code>frameset</code> element must <span>reflect</span> the respective
  content attributes of the same name.</p>

  <p>The <code>frameset</code> element exposes as <span>event handler content attributes</span> a
  number of the <span>event handlers</span> of the <code>Window</code> object. It also mirrors their
  <span>event handler IDL attributes</span>.</p>

  <p>The <span>event handlers</span> of the <code>Window</code> object named by the
  <span><code>Window</code>-reflecting body element event handler set</span>, exposed on the
  <code>frameset</code> element, replace the generic <span>event handlers</span> with the same names
  normally supported by <span>HTML elements</span>.</p>

  <hr>

  <p>The <dfn element><code>frame</code></dfn> element has a <span>content navigable</span> similar
  to the <code>iframe</code> element, but rendered within a <code>frameset</code> element.</p>

  <!-- START of section that's very similar to <iframe> -->

  <p>The <code>frame</code> <span>HTML element insertion steps</span>, given
  <var>insertedNode</var>, are:</p>

  <ol>
   <li><p>If <var>insertedNode</var> is not <span>in a document tree</span>, then return.</p></li>

   <li><p>If <var>insertedNode</var>'s <span>root</span>'s <span
   data-x="concept-document-bc">browsing context</span> is null, then return.</p></li>

   <li><p><span>Create a new child navigable</span> for <var>insertedNode</var>.</p></li>

   <li><p><span>Process the <code>frame</code> attributes</span> for <var>insertedNode</var>, with
   <var data-x="process-frame-initial-insertion">initialInsertion</var> set to true.</p></li>
  </ol>

  <p>The <code>frame</code> <span>HTML element removing steps</span>, given <var>removedNode</var>,
  are to <span>destroy a child navigable</span> given <var>removedNode</var>.</p>

  <p>Whenever a <code>frame</code> element with a non-null <span>content navigable</span> has its
  <code undefined data-x="attr-frame-src">src</code> attribute set, changed, or removed, the user
  agent must <span>process the <code>frame</code> attributes</span>.</p> <!-- It doesn't happen
  when the base URL is changed, though. -->

  <p>To <dfn>process the <code>frame</code> attributes</dfn> for an element <var>element</var>, with
  an optional boolean <dfn
  data-x="process-frame-initial-insertion"><var>initialInsertion</var></dfn>:</p>

  <ol>
   <li><p>Let <var>url</var> be the result of running the <span>shared attribute processing steps
   for <code>iframe</code> and <code>frame</code> elements</span> given <var>element</var> and
   <var>initialInsertion</var>.</p></li>

   <li><p>If <var>url</var> is null, then return.</p></li>

   <li>
    <p>If <var>url</var> <span>matches <code>about:blank</code></span> and
    <var>initialInsertion</var> is true, then:</p>

    <ol>
     <li><p><span data-x="concept-event-fire">Fire an event</span> named <code
     data-x="event-load">load</code> at <var>element</var>.</p></li>

     <li><p>Return.</p></li>
    </ol>
   </li>

   <li><p><span>Navigate an <code>iframe</code> or <code>frame</code></span> given
   <var>element</var>, <var>url</var>, and the empty string.</p></li>
  </ol>

  <p>The <code>frame</code> element <span>potentially delays the load event</span>.</p>

  <!-- END of section that's very similar to <iframe> -->

  <p>The <code>frame</code> element must implement the <code>HTMLFrameElement</code> interface.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLFrameElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-frame-name">name</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-frame-scrolling">scrolling</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-frame-src">src</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-frame-frameBorder">frameBorder</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-frame-longDesc">longDesc</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-frame-noResize">noResize</span>;
  readonly attribute <span>Document</span>? <span data-x="dom-frame-contentDocument">contentDocument</span>;
  readonly attribute <span>WindowProxy</span>? <span data-x="dom-frame-contentWindow">contentWindow</span>;

  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-frame-marginHeight">marginHeight</span>;
  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-frame-marginWidth">marginWidth</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLFrameElement"><code data-x="dom-frame-name">name</code></dfn>, <dfn
  attribute for="HTMLFrameElement"><code data-x="dom-frame-scrolling">scrolling</code></dfn>, and
  <dfn attribute for="HTMLFrameElement"><code data-x="dom-frame-src">src</code></dfn> IDL attributes
  of the <code>frame</code> element must <span>reflect</span> the respective content attributes of
  the same name. For the purposes of reflection, the <code>frame</code> element's <code undefined
  data-x="attr-frame-src">src</code> content attribute is defined as containing a
  <span>URL</span>.</p>

  <p>The <dfn attribute for="HTMLFrameElement"><code
  data-x="dom-frame-frameBorder">frameBorder</code></dfn> IDL attribute of the <code>frame</code>
  element must <span>reflect</span> the element's <code undefined
  data-x="attr-frame-frameborder">frameborder</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLFrameElement"><code
  data-x="dom-frame-longDesc">longDesc</code></dfn> IDL attribute of the <code>frame</code> element
  must <span>reflect</span> the element's <code undefined
  data-x="attr-frame-longdesc">longdesc</code> content attribute, which for the purposes of
  reflection is defined as containing a <span>URL</span>.</p>

  <p>The <dfn attribute for="HTMLFrameElement"><code
  data-x="dom-frame-noResize">noResize</code></dfn> IDL attribute of the <code>frame</code> element
  must <span>reflect</span> the element's <code undefined
  data-x="attr-frame-noresize">noresize</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLFrameElement"><code
  data-x="dom-frame-marginHeight">marginHeight</code></dfn> IDL attribute of the <code>frame</code>
  element must <span>reflect</span> the element's <code undefined
  data-x="attr-frame-marginheight">marginheight</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLFrameElement"><code
  data-x="dom-frame-marginWidth">marginWidth</code></dfn> IDL attribute of the <code>frame</code>
  element must <span>reflect</span> the element's <code undefined
  data-x="attr-frame-marginwidth">marginwidth</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLFrameElement"><code
  data-x="dom-frame-contentDocument">contentDocument</code></dfn> getter steps are to return
  <span>this</span>'s <span>content document</span>.</p>

  <p>The <dfn attribute for="HTMLFrameElement"><code
  data-x="dom-frame-contentWindow">contentWindow</code></dfn> getter steps are to return
  <span>this</span>'s <span>content window</span>.</p>


  <h4>Other elements, attributes and APIs</h4> <!-- alphabetical by name and attribute -->

  <p>User agents must treat <code>acronym</code> elements in a manner
  equivalent to <code>abbr</code> elements in terms of semantics and
  for purposes of rendering.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLAnchorElement-partial">HTMLAnchorElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-a-coords">coords</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-a-charset">charset</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-a-name">name</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-a-rev">rev</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-a-shape">shape</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLAnchorElement"><code data-x="dom-a-coords">coords</code></dfn>,
  <dfn attribute for="HTMLAnchorElement"><code data-x="dom-a-charset">charset</code></dfn>, <dfn
  attribute for="HTMLAnchorElement"><code data-x="dom-a-name">name</code></dfn>, <dfn attribute
  for="HTMLAnchorElement"><code data-x="dom-a-rev">rev</code></dfn>, and <dfn attribute
  for="HTMLAnchorElement"><code data-x="dom-a-shape">shape</code></dfn> IDL attributes of the
  <code>a</code> element must <span>reflect</span> the respective content attributes of the same
  name.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLAreaElement-partial">HTMLAreaElement</span> {
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-area-noHref">noHref</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLAreaElement"><code data-x="dom-area-noHref">noHref</code></dfn> IDL
  attribute of the <code>area</code> element must <span>reflect</span> the element's <code
  data-x="attr-area-nohref">nohref</code> content attribute.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLBodyElement-partial">HTMLBodyElement</span> {
  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-body-text">text</span>;
  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-body-link">link</span>;
  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-body-vLink">vLink</span>;
  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-body-aLink">aLink</span>;
  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-body-bgColor">bgColor</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-body-background">background</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLBodyElement"><code data-x="dom-body-text">text</code></dfn> IDL
  attribute of the <code>body</code> element must <span>reflect</span> the element's <code
  data-x="attr-body-text">text</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLBodyElement"><code data-x="dom-body-link">link</code></dfn> IDL
  attribute of the <code>body</code> element must <span>reflect</span> the element's <code
  data-x="attr-body-link">link</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLBodyElement"><code data-x="dom-body-aLink">aLink</code></dfn> IDL
  attribute of the <code>body</code> element must <span>reflect</span> the element's <code
  data-x="attr-body-alink">alink</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLBodyElement"><code data-x="dom-body-vLink">vLink</code></dfn> IDL
  attribute of the <code>body</code> element must <span>reflect</span> the element's <code
  data-x="attr-body-vlink">vlink</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLBodyElement"><code data-x="dom-body-bgColor">bgColor</code></dfn>
  IDL attribute of the <code>body</code> element must <span>reflect</span> the element's <code
  data-x="attr-body-bgcolor">bgcolor</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLBodyElement"><code
  data-x="dom-body-background">background</code></dfn> IDL attribute of the <code>body</code>
  element must <span>reflect</span> the element's <code data-x="attr-background">background</code>
  content attribute. (The <code data-x="attr-background">background</code> content is <em>not</em>
  defined to contain a <span>URL</span>, despite rules regarding its handling in the Rendering
  section above.)</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLBRElement-partial">HTMLBRElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-br-clear">clear</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLBRElement"><code data-x="dom-br-clear">clear</code></dfn> IDL
  attribute of the <code>br</code> element must <span>reflect</span> the content attribute of the
  same name.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLTableCaptionElement-partial">HTMLTableCaptionElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-caption-align">align</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLTableCaptionElement"><code
  data-x="dom-caption-align">align</code></dfn> IDL attribute of the <code>caption</code> element
  must <span>reflect</span> the content attribute of the same name.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLTableColElement-partial">HTMLTableColElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-col-align">align</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-col-ch">ch</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-col-chOff">chOff</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-col-vAlign">vAlign</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-col-width">width</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLTableColElement"><code data-x="dom-col-align">align</code></dfn>
  and <dfn attribute for="HTMLTableColElement"><code data-x="dom-col-width">width</code></dfn> IDL
  attributes of the <code>col</code> element must <span>reflect</span> the respective content
  attributes of the same name.</p>

  <p>The <dfn attribute for="HTMLTableColElement"><code data-x="dom-col-ch">ch</code></dfn> IDL
  attribute of the <code>col</code> element must <span>reflect</span> the element's <code
  data-x="attr-col-char">char</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLTableColElement"><code data-x="dom-col-chOff">chOff</code></dfn>
  IDL attribute of the <code>col</code> element must <span>reflect</span> the element's <code
  data-x="attr-col-charoff">charoff</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLTableColElement"><code data-x="dom-col-vAlign">vAlign</code></dfn>
  IDL attribute of the <code>col</code> element must <span>reflect</span> the element's <code
  data-x="attr-col-valign">valign</code> content attribute.</p>

  <hr>

  <p>User agents must treat <code>dir</code> elements in a manner equivalent to <code>ul</code>
  elements in terms of semantics and for purposes of rendering.</p>

  <p>The <code>dir</code> element must implement the <code>HTMLDirectoryElement</code>
  interface.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLDirectoryElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute boolean <span data-x="dom-dir-compact">compact</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLDirectoryElement"><code
  data-x="dom-dir-compact">compact</code></dfn> IDL attribute of the <code>dir</code> element must
  <span>reflect</span> the content attribute of the same name.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLDivElement-partial">HTMLDivElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-div-align">align</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLDivElement"><code data-x="dom-div-align">align</code></dfn> IDL
  attribute of the <code>div</code> element must <span>reflect</span> the content attribute of the
  same name.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLDListElement-partial">HTMLDListElement</span> {
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-dl-compact">compact</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLDListElement"><code data-x="dom-dl-compact">compact</code></dfn>
  IDL attribute of the <code>dl</code> element must <span>reflect</span> the content attribute of
  the same name.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLEmbedElement-partial">HTMLEmbedElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-embed-align">align</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-embed-name">name</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLEmbedElement"><code data-x="dom-embed-name">name</code></dfn> and
  <dfn attribute for="HTMLEmbedElement"><code data-x="dom-embed-align">align</code></dfn> IDL
  attributes of the <code>embed</code> element must <span>reflect</span> the respective content
  attributes of the same name.</p>

  <hr>

  <p>The <code>font</code> element must implement the <code>HTMLFontElement</code> interface.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLFontElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-font-color">color</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-font-face">face</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-font-size">size</span>; <!-- yes, DOMString, not long (so says DOM2 HTML) -->
};</code></pre>

  <p>The <dfn attribute for="HTMLFontElement"><code data-x="dom-font-color">color</code></dfn>, <dfn
  attribute for="HTMLFontElement"><code data-x="dom-font-face">face</code></dfn>, and <dfn attribute
  for="HTMLFontElement"><code data-x="dom-font-size">size</code></dfn> IDL attributes of the
  <code>font</code> element must <span>reflect</span> the respective content attributes of the same
  name.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLHeadingElement-partial">HTMLHeadingElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-hx-align">align</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLHeadingElement"><code data-x="dom-hx-align">align</code></dfn> IDL
  attribute of the <code>h1</code>&ndash;<code>h6</code> elements must <span>reflect</span> the
  content attribute of the same name.</p>

  <hr>

  <p class="note">The <dfn attribute for="HTMLHeadElement"><code
  data-x="dom-head-profile">profile</code></dfn> IDL attribute on <code>head</code> elements (with
  the <code>HTMLHeadElement</code> interface) is intentionally omitted. Unless so required by <span
  data-x="other applicable specifications">another applicable specification</span>, implementations
  would therefore not support this attribute. (It is mentioned here as it was defined in a previous
  version of <cite>DOM</cite>.)</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLHRElement-partial">HTMLHRElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-hr-align">align</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-hr-color">color</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-hr-noShade">noShade</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-hr-size">size</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-hr-width">width</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLHRElement"><code data-x="dom-hr-align">align</code></dfn>, <dfn
  attribute for="HTMLHRElement"><code data-x="dom-hr-color">color</code></dfn>, <dfn attribute
  for="HTMLHRElement"><code data-x="dom-hr-size">size</code></dfn>, and <dfn attribute
  for="HTMLHRElement"><code data-x="dom-hr-width">width</code></dfn> IDL attributes of the
  <code>hr</code> element must <span>reflect</span> the respective content attributes of the same
  name.</p>

  <p>The <dfn attribute for="HTMLHRElement"><code data-x="dom-hr-noShade">noShade</code></dfn> IDL
  attribute of the <code>hr</code> element must <span>reflect</span> the element's <code
  data-x="attr-hr-noshade">noshade</code> content attribute.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLHtmlElement-partial">HTMLHtmlElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-html-version">version</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLHtmlElement"><code data-x="dom-html-version">version</code></dfn>
  IDL attribute of the <code>html</code> element must <span>reflect</span> the content attribute of
  the same name.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLIFrameElement-partial">HTMLIFrameElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-iframe-align">align</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-iframe-scrolling">scrolling</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-iframe-frameBorder">frameBorder</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-iframe-longDesc">longDesc</span>;

  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-iframe-marginHeight">marginHeight</span>;
  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-iframe-marginWidth">marginWidth</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLIFrameElement"><code data-x="dom-iframe-align">align</code></dfn>
  and <dfn attribute for="HTMLIFrameElement"><code
  data-x="dom-iframe-scrolling">scrolling</code></dfn> IDL attributes of the <code>iframe</code>
  element must <span>reflect</span> the respective content attributes of the same name.</p>

  <p>The <dfn attribute for="HTMLIFrameElement"><code
  data-x="dom-iframe-frameBorder">frameBorder</code></dfn> IDL attribute of the <code>iframe</code>
  element must <span>reflect</span> the element's <code
  data-x="attr-iframe-frameborder">frameborder</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLIFrameElement"><code
  data-x="dom-iframe-longDesc">longDesc</code></dfn> IDL attribute of the <code>iframe</code>
  element must <span>reflect</span> the element's <code
  data-x="attr-iframe-longdesc">longdesc</code> content attribute, which for the purposes of
  reflection is defined as containing a <span>URL</span>.</p>

  <p>The <dfn attribute for="HTMLIFrameElement"><code
  data-x="dom-iframe-marginHeight">marginHeight</code></dfn> IDL attribute of the
  <code>iframe</code> element must <span>reflect</span> the element's <code
  data-x="attr-iframe-marginheight">marginheight</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLIFrameElement"><code
  data-x="dom-iframe-marginWidth">marginWidth</code></dfn> IDL attribute of the <code>iframe</code>
  element must <span>reflect</span> the element's <code
  data-x="attr-iframe-marginwidth">marginwidth</code> content attribute.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLImageElement-partial">HTMLImageElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-img-name">name</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-img-lowsrc">lowsrc</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-img-align">align</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-img-hspace">hspace</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-img-vspace">vspace</span>;
  [<span>CEReactions</span>] attribute USVString <span data-x="dom-img-longDesc">longDesc</span>;

  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-img-border">border</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLImageElement"><code data-x="dom-img-name">name</code></dfn>, <dfn
  attribute for="HTMLImageElement"><code data-x="dom-img-align">align</code></dfn>, <dfn attribute
  for="HTMLImageElement"><code data-x="dom-img-border">border</code></dfn>, <dfn attribute
  for="HTMLImageElement"><code data-x="dom-img-hspace">hspace</code></dfn>, and <dfn attribute
  for="HTMLImageElement"><code data-x="dom-img-vspace">vspace</code></dfn> IDL attributes of the
  <code>img</code> element must <span>reflect</span> the respective content attributes of the same
  name.</p>

  <p>The <dfn attribute for="HTMLImageElement"><code data-x="dom-img-longDesc">longDesc</code></dfn>
  IDL attribute of the <code>img</code> element must <span>reflect</span> the element's <code
  data-x="attr-img-longdesc">longdesc</code> content attribute, which for the purposes of reflection
  is defined as containing a <span>URL</span>.</p>

  <p>The <dfn attribute for="HTMLImageElement"><code data-x="dom-img-lowsrc">lowsrc</code></dfn> IDL
  attribute of the <code>img</code> element must <span>reflect</span> the element's <code
  data-x="attr-img-lowsrc">lowsrc</code> content attribute, which for the purposes of reflection is
  defined as containing a <span>URL</span>.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLInputElement-partial">HTMLInputElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-input-align">align</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-input-useMap">useMap</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLInputElement"><code data-x="dom-input-align">align</code></dfn> IDL
  attribute of the <code>input</code> element must <span>reflect</span> the content attribute of the
  same name.</p>

  <p>The <dfn attribute for="HTMLInputElement"><code data-x="dom-input-useMap">useMap</code></dfn>
  IDL attribute of the <code>input</code> element must <span>reflect</span> the element's <code
  data-x="attr-input-usemap">usemap</code> content attribute.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLLegendElement-partial">HTMLLegendElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-legend-align">align</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLLegendElement"><code data-x="dom-legend-align">align</code></dfn>
  IDL attribute of the <code>legend</code> element must <span>reflect</span> the content attribute
  of the same name.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLLIElement-partial">HTMLLIElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-li-type">type</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLLIElement"><code data-x="dom-li-type">type</code></dfn> IDL
  attribute of the <code>li</code> element must <span>reflect</span> the content attribute of the
  same name.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLLinkElement-partial">HTMLLinkElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-link-charset">charset</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-link-rev">rev</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-link-target">target</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLLinkElement"><code data-x="dom-link-charset">charset</code></dfn>,
  <dfn attribute for="HTMLLinkElement"><code data-x="dom-link-rev">rev</code></dfn>, and <dfn
  attribute for="HTMLLinkElement"><code data-x="dom-link-target">target</code></dfn> IDL attributes
  of the <code>link</code> element must <span>reflect</span> the respective content attributes of
  the same name.</p>

  <hr>

  <p>User agents must treat <code>listing</code> elements in a manner equivalent to <code>pre</code>
  elements in terms of semantics and for purposes of rendering.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLMenuElement-partial">HTMLMenuElement</span> {
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-menu-compact">compact</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLMenuElement"><code data-x="dom-menu-compact">compact</code></dfn>
  IDL attribute of the <code>menu</code> element must <span>reflect</span> the content attribute of
  the same name.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLMetaElement-partial">HTMLMetaElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-meta-scheme">scheme</span>;
};</code></pre>

  <p>User agents may treat the <code data-x="attr-meta-scheme">scheme</code> content attribute on the
  <code>meta</code> element as an extension of the element's <code
  data-x="attr-meta-name">name</code> content attribute when processing a <code>meta</code> element
  with a <code data-x="attr-meta-name">name</code> attribute whose value is one that the user agent
  recognizes as supporting the <code data-x="attr-meta-scheme">scheme</code> attribute.</p>

  <p>User agents are encouraged to ignore the <code data-x="attr-meta-scheme">scheme</code> attribute
  and instead process the value given to the metadata name as if it had been specified for each
  expected value of the <code data-x="attr-meta-scheme">scheme</code> attribute.</p>

  <div class="example">

   <p>For example, if the user agent acts on <code>meta</code> elements with <code
   data-x="attr-meta-name">name</code> attributes having the value "eGMS.subject.keyword", and knows
   that the <code data-x="attr-meta-scheme">scheme</code> attribute is used with this metadata name,
   then it could take the <code data-x="attr-meta-scheme">scheme</code> attribute into account,
   acting as if it was an extension of the <code data-x="attr-meta-name">name</code> attribute. Thus
   the following two <code>meta</code> elements could be treated as two elements giving values for
   two different metadata names, one consisting of a combination of "eGMS.subject.keyword" and
   "LGCL", and the other consisting of a combination of "eGMS.subject.keyword" and "ORLY":</p>

   <pre class="bad"><code class="html">&lt;!-- this markup is invalid -->
&lt;meta name="eGMS.subject.keyword" scheme="LGCL" content="Abandoned vehicles">
&lt;meta name="eGMS.subject.keyword" scheme="ORLY" content="Mah car: kthxbye"></code></pre>

   <p>The suggested processing of this markup, however, would be equivalent to the following:</p>

   <pre><code class="html">&lt;meta name="eGMS.subject.keyword" content="Abandoned vehicles">
&lt;meta name="eGMS.subject.keyword" content="Mah car: kthxbye"></code></pre>

  </div>

  <p>The <dfn attribute for="HTMLMetaElement"><code data-x="dom-meta-scheme">scheme</code></dfn> IDL
  attribute of the <code>meta</code> element must <span>reflect</span> the content attribute of the
  same name.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLObjectElement-partial">HTMLObjectElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-object-align">align</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-object-archive">archive</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-object-code">code</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-object-declare">declare</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-object-hspace">hspace</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-object-standby">standby</span>;
  [<span>CEReactions</span>] attribute unsigned long <span data-x="dom-object-vspace">vspace</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-object-codeBase">codeBase</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-object-codeType">codeType</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-object-useMap">useMap</span>;

  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-object-border">border</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLObjectElement"><code data-x="dom-object-align">align</code></dfn>,
  <dfn attribute for="HTMLObjectElement"><code data-x="dom-object-archive">archive</code></dfn>,
  <dfn attribute for="HTMLObjectElement"><code data-x="dom-object-border">border</code></dfn>, <dfn
  attribute for="HTMLObjectElement"><code data-x="dom-object-code">code</code></dfn>, <dfn attribute
  for="HTMLObjectElement"><code data-x="dom-object-declare">declare</code></dfn>, <dfn attribute
  for="HTMLObjectElement"><code data-x="dom-object-hspace">hspace</code></dfn>, <dfn attribute
  for="HTMLObjectElement"><code data-x="dom-object-standby">standby</code></dfn>, and <dfn attribute
  for="HTMLObjectElement"><code data-x="dom-object-vspace">vspace</code></dfn> IDL attributes of the
  <code>object</code> element must <span>reflect</span> the respective content attributes of the
  same name.</p>

  <p>The <dfn attribute for="HTMLObjectElement"><code
  data-x="dom-object-codeBase">codeBase</code></dfn> IDL attribute of the <code>object</code>
  element must <span>reflect</span> the element's <code
  data-x="attr-object-codebase">codebase</code> content attribute, which for the purposes of
  reflection is defined as containing a <span>URL</span>.</p>

  <p>The <dfn attribute for="HTMLObjectElement"><code
  data-x="dom-object-codeType">codeType</code></dfn> IDL attribute of the <code>object</code>
  element must <span>reflect</span> the element's <code
  data-x="attr-object-codetype">codetype</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLObjectElement"><code data-x="dom-object-useMap">useMap</code></dfn>
  IDL attribute must <span>reflect</span> the <code data-x="attr-hyperlink-usemap">usemap</code>
  content attribute.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLOListElement-partial">HTMLOListElement</span> {
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-ol-compact">compact</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLOListElement"><code data-x="dom-ol-compact">compact</code></dfn>
  IDL attribute of the <code>ol</code> element must <span>reflect</span> the content attribute of
  the same name.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLParagraphElement-partial">HTMLParagraphElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-p-align">align</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLParagraphElement"><code data-x="dom-p-align">align</code></dfn> IDL
  attribute of the <code>p</code> element must <span>reflect</span> the content attribute of the
  same name.</p>

  <hr>

  <p>The <code>param</code> element must implement the <code>HTMLParamElement</code> interface.</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>HTMLParamElement</dfn> : <span>HTMLElement</span> {
  [<span>HTMLConstructor</span>] constructor();

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-param-name">name</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-param-value">value</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-param-type">type</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-param-valueType">valueType</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLParamElement"><code data-x="dom-param-name">name</code></dfn>,
  <dfn attribute for="HTMLParamElement"><code data-x="dom-param-value">value</code></dfn>, and <dfn
  attribute for="HTMLParamElement"><code data-x="dom-param-type">type</code></dfn> IDL attributes
  of the <code>param</code> element must <span>reflect</span> the respective content attributes of
  the same name.</p>

  <p>The <dfn attribute for="HTMLParamElement"><code
  data-x="dom-param-valueType">valueType</code></dfn> IDL attribute of the <code>param</code>
  element must <span>reflect</span> the element's <code data-x="">valuetype</code> content
  attribute.</p>

  <hr>

  <p>User agents must treat <code>plaintext</code> elements in a manner equivalent to
  <code>pre</code> elements in terms of semantics and for purposes of rendering. (The parser has
  special behavior for this element, though.)</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLPreElement-partial">HTMLPreElement</span> {
  [<span>CEReactions</span>] attribute long <span data-x="dom-pre-width">width</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLPreElement"><code data-x="dom-pre-width">width</code></dfn> IDL
  attribute of the <code>pre</code> element must <span>reflect</span> the content attribute of the
  same name.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLStyleElement-partial">HTMLStyleElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-style-type">type</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLStyleElement"><code data-x="dom-style-type">type</code></dfn> IDL
  attribute of the <code>style</code> element must <span>reflect</span> the element's <code
  data-x="attr-style-type">type</code> content attribute.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLScriptElement-partial">HTMLScriptElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-script-charset">charset</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-script-event">event</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-script-htmlFor">htmlFor</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLScriptElement"><code
  data-x="dom-script-charset">charset</code></dfn> and <dfn attribute for="HTMLScriptElement"><code
  data-x="dom-script-event">event</code></dfn> IDL attributes of the <code>script</code> element
  must <span>reflect</span> the respective content attributes of the same name.</p>

  <p>The <dfn attribute for="HTMLScriptElement"><code
  data-x="dom-script-htmlFor">htmlFor</code></dfn> IDL attribute of the <code>script</code> element
  must <span>reflect</span> the element's <code data-x="attr-script-for">for</code> content
  attribute.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLTableElement-partial">HTMLTableElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-table-align">align</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-table-border">border</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-table-frame">frame</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-table-rules">rules</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-table-summary">summary</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-table-width">width</span>;

  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-table-bgColor">bgColor</span>;
  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-table-cellPadding">cellPadding</span>;
  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-table-cellSpacing">cellSpacing</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLTableElement"><code data-x="dom-table-align">align</code></dfn>,
  <dfn attribute for="HTMLTableElement"><code data-x="dom-table-border">border</code></dfn>, <dfn
  attribute for="HTMLTableElement"><code data-x="dom-table-frame">frame</code></dfn>, <dfn attribute
  for="HTMLTableElement"><code data-x="dom-table-summary">summary</code></dfn>, <dfn attribute
  for="HTMLTableElement"><code data-x="dom-table-rules">rules</code></dfn>, and <dfn attribute
  for="HTMLTableElement"><code data-x="dom-table-width">width</code></dfn> IDL attributes of the
  <code>table</code> element must <span>reflect</span> the respective content attributes of the same
  name.</p>

  <p>The <dfn attribute for="HTMLTableElement"><code data-x="dom-table-bgColor">bgColor</code></dfn>
  IDL attribute of the <code>table</code> element must <span>reflect</span> the element's <code
  data-x="attr-table-bgcolor">bgcolor</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLTableElement"><code
  data-x="dom-table-cellPadding">cellPadding</code></dfn> IDL attribute of the <code>table</code>
  element must <span>reflect</span> the element's <code
  data-x="attr-table-cellpadding">cellpadding</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLTableElement"><code
  data-x="dom-table-cellSpacing">cellSpacing</code></dfn> IDL attribute of the <code>table</code>
  element must <span>reflect</span> the element's <code
  data-x="attr-table-cellspacing">cellspacing</code> content attribute.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLTableSectionElement-partial">HTMLTableSectionElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-tbody-align">align</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-tbody-ch">ch</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-tbody-chOff">chOff</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-tbody-vAlign">vAlign</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLTableSectionElement"><code
  data-x="dom-tbody-align">align</code></dfn> IDL attribute of the <code>tbody</code>,
  <code>thead</code>, and <code>tfoot</code> elements must <span>reflect</span> the content
  attribute of the same name.</p>

  <p>The <dfn attribute for="HTMLTableSectionElement"><code data-x="dom-tbody-ch">ch</code></dfn>
  IDL attribute of the <code>tbody</code>, <code>thead</code>, and <code>tfoot</code> elements must
  <span>reflect</span> the elements' <code data-x="attr-tbody-char">char</code> content
  attributes.</p>

  <p>The <dfn attribute for="HTMLTableSectionElement"><code
  data-x="dom-tbody-chOff">chOff</code></dfn> IDL attribute of the <code>tbody</code>,
  <code>thead</code>, and <code>tfoot</code> elements must <span>reflect</span> the elements' <code
  data-x="attr-tbody-charoff">charoff</code> content attributes.</p>

  <p>The <dfn attribute for="HTMLTableSectionElement"><code
  data-x="dom-tbody-vAlign">vAlign</code></dfn> IDL attribute of the <code>tbody</code>,
  <code>thead</code>, and <code>tfoot</code> elements must <span>reflect</span> the elements' <code
  data-x="attr-tbody-valign">valign</code> content attributes.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLTableCellElement-partial">HTMLTableCellElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-tdth-align">align</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-tdth-axis">axis</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-tdth-height">height</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-tdth-width">width</span>;

  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-tdth-ch">ch</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-tdth-chOff">chOff</span>;
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-tdth-noWrap">noWrap</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-tdth-vAlign">vAlign</span>;

  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-tdth-bgColor">bgColor</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLTableCellElement"><code data-x="dom-tdth-align">align</code></dfn>,
  <dfn attribute for="HTMLTableCellElement"><code data-x="dom-tdth-axis">axis</code></dfn>, <dfn
  attribute for="HTMLTableCellElement"><code data-x="dom-tdth-height">height</code></dfn>, and <dfn
  attribute for="HTMLTableCellElement"><code data-x="dom-tdth-width">width</code></dfn> IDL
  attributes of the <code>td</code> and <code>th</code> elements must <span>reflect</span> the
  respective content attributes of the same name.</p>

  <p>The <dfn attribute for="HTMLTableCellElement"><code data-x="dom-tdth-ch">ch</code></dfn> IDL
  attribute of the <code>td</code> and <code>th</code> elements must <span>reflect</span> the
  elements' <code data-x="attr-tdth-char">char</code> content attributes.</p>

  <p>The <dfn attribute for="HTMLTableCellElement"><code data-x="dom-tdth-chOff">chOff</code></dfn>
  IDL attribute of the <code>td</code> and <code>th</code> elements must <span>reflect</span> the
  elements' <code data-x="attr-tdth-charoff">charoff</code> content attributes.</p>

  <p>The <dfn attribute for="HTMLTableCellElement"><code
  data-x="dom-tdth-noWrap">noWrap</code></dfn> IDL attribute of the <code>td</code> and
  <code>th</code> elements must <span>reflect</span> the elements' <code
  data-x="attr-tdth-nowrap">nowrap</code> content attributes.</p>

  <p>The <dfn attribute for="HTMLTableCellElement"><code
  data-x="dom-tdth-vAlign">vAlign</code></dfn> IDL attribute of the <code>td</code> and
  <code>th</code> elements must <span>reflect</span> the elements' <code
  data-x="attr-tdth-valign">valign</code> content attributes.</p>

  <p>The <dfn attribute for="HTMLTableCellElement"><code
  data-x="dom-tdth-bgColor">bgColor</code></dfn> IDL attribute of the <code>td</code> and
  <code>th</code> elements must <span>reflect</span> the elements' <code
  data-x="attr-tdth-bgcolor">bgcolor</code> content attributes.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLTableRowElement-partial">HTMLTableRowElement</span> {
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-tr-align">align</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-tr-ch">ch</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-tr-chOff">chOff</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-tr-vAlign">vAlign</span>;

  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-tr-bgColor">bgColor</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLTableRowElement"><code data-x="dom-tr-align">align</code></dfn> IDL
  attribute of the <code>tr</code> element must <span>reflect</span> the content attribute of the
  same name.</p>

  <p>The <dfn attribute for="HTMLTableRowElement"><code data-x="dom-tr-ch">ch</code></dfn> IDL
  attribute of the <code>tr</code> element must <span>reflect</span> the element's <code
  data-x="attr-tr-char">char</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLTableRowElement"><code data-x="dom-tr-chOff">chOff</code></dfn> IDL
  attribute of the <code>tr</code> element must <span>reflect</span> the element's <code
  data-x="attr-tr-charoff">charoff</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLTableRowElement"><code data-x="dom-tr-vAlign">vAlign</code></dfn>
  IDL attribute of the <code>tr</code> element must <span>reflect</span> the element's <code
  data-x="attr-tr-valign">valign</code> content attribute.</p>

  <p>The <dfn attribute for="HTMLTableRowElement"><code data-x="dom-tr-bgColor">bgColor</code></dfn>
  IDL attribute of the <code>tr</code> element must <span>reflect</span> the element's <code
  data-x="attr-tr-bgcolor">bgcolor</code> content attribute.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="HTMLUListElement-partial">HTMLUListElement</span> {
  [<span>CEReactions</span>] attribute boolean <span data-x="dom-ul-compact">compact</span>;
  [<span>CEReactions</span>] attribute DOMString <span data-x="dom-ul-type">type</span>;
};</code></pre>

  <p>The <dfn attribute for="HTMLUListElement"><code data-x="dom-ul-compact">compact</code></dfn>
  and <dfn attribute for="HTMLUListElement"><code data-x="dom-ul-type">type</code></dfn> IDL
  attributes of the <code>ul</code> element must <span>reflect</span> the respective content
  attributes of the same name.</p>

  <hr>

  <p>User agents must treat <code>xmp</code> elements in a manner equivalent to <code>pre</code>
  elements in terms of semantics and for purposes of rendering. (The parser has special behavior for
  this element though.)</p>

  <hr>

  <pre><code class="idl">partial interface <span id="Document-partial">Document</span> {
  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-document-fgColor">fgColor</span>;
  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-document-linkColor">linkColor</span>;
  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-document-vlinkColor">vlinkColor</span>;
  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-document-alinkColor">alinkColor</span>;
  [<span>CEReactions</span>] attribute [<span>LegacyNullToEmptyString</span>] DOMString <span data-x="dom-document-bgColor">bgColor</span>;

  [SameObject] readonly attribute <span>HTMLCollection</span> <span data-x="dom-document-anchors">anchors</span>;
  [SameObject] readonly attribute <span>HTMLCollection</span> <span data-x="dom-document-applets">applets</span>;

  undefined <span data-x="dom-document-clear">clear</span>();
  undefined <span data-x="dom-document-captureEvents">captureEvents</span>();
  undefined <span data-x="dom-document-releaseEvents">releaseEvents</span>();

  [SameObject] readonly attribute <span>HTMLAllCollection</span> <span data-x="dom-document-all">all</span>;
};</code></pre>

  <p>The attributes of the <code>Document</code> object listed in the first column of the following
  table must <span>reflect</span> the content attribute on <span>the body element</span> with the
  name given in the corresponding cell in the second column on the same row, if <span>the body
  element</span> is a <code>body</code> element (as opposed to a <code>frameset</code> element).
  When there is no <span data-x="the body element">body element</span> or if it is a
  <code>frameset</code> element, the attributes must instead return the empty string on getting and
  do nothing on setting.</p>

  <table>
   <thead>
    <tr>
     <th> IDL attribute
     <th> Content attribute
   <tbody>
    <tr>
     <td><dfn attribute for="Document"><code data-x="dom-document-fgColor">fgColor</code></dfn>
     <td><code data-x="attr-body-text">text</code>
    <tr>
     <td><dfn attribute for="Document"><code data-x="dom-document-linkColor">linkColor</code></dfn>
     <td><code data-x="attr-body-link">link</code>
    <tr>
     <td><dfn attribute for="Document"><code data-x="dom-document-vlinkColor">vlinkColor</code></dfn>
     <td><code data-x="attr-body-vlink">vlink</code>
    <tr>
     <td><dfn attribute for="Document"><code data-x="dom-document-alinkColor">alinkColor</code></dfn>
     <td><code data-x="attr-body-alink">alink</code>
    <tr>
     <td><dfn attribute for="Document"><code data-x="dom-document-bgColor">bgColor</code></dfn>
     <td><code data-x="attr-body-bgcolor">bgcolor</code>
  </table>

  <hr>

  <p>The <dfn attribute for="Document"><code data-x="dom-document-anchors">anchors</code></dfn>
  attribute must return an <code>HTMLCollection</code> rooted at the <code>Document</code> node,
  whose filter matches only <code>a</code> elements with <code data-x="attr-a-name">name</code>
  attributes.</p>

  <p>The <dfn attribute for="Document"><code data-x="dom-document-applets">applets</code></dfn>
  attribute must return an <code>HTMLCollection</code> rooted at the <code>Document</code> node,
  whose filter matches nothing. (It exists for historical reasons.)</p>

  <p>The <dfn method for="Document"><code data-x="dom-document-clear">clear()</code></dfn>,
  <dfn method for="Document"><code data-x="dom-document-captureEvents">captureEvents()</code></dfn>,
  and <dfn method for="Document"><code
  data-x="dom-document-releaseEvents">releaseEvents()</code></dfn> methods must do nothing.</p>

  <hr>

  <p>The <dfn attribute for="Document"><code data-x="dom-document-all">all</code></dfn> attribute
  must return an <code>HTMLAllCollection</code> rooted at the <code>Document</code> node, whose
  filter matches all elements.</p>

  <hr>

  <pre><code class="idl">partial interface <span id="Window-partial">Window</span> {
  undefined <span data-x="dom-window-captureEvents">captureEvents</span>();
  undefined <span data-x="dom-window-releaseEvents">releaseEvents</span>();

  [Replaceable, SameObject] readonly attribute <span>External</span> <span data-x="dom-external">external</span>;
};</code></pre>

  <p>The <dfn method for="Window"><code
  data-x="dom-window-captureEvents">captureEvents()</code></dfn> and <dfn method for="Window"><code
  data-x="dom-window-releaseEvents">releaseEvents()</code></dfn> methods must do nothing.</p>

  <p>The <dfn attribute for="Window"><code data-x="dom-external">external</code></dfn> attribute of
  the <code>Window</code> interface must return an instance of the <code>External</code>
  interface:</p>

  <pre><code class="idl">[Exposed=Window]
interface <dfn interface>External</dfn> {
  undefined <span data-x="dom-external-AddSearchProvider">AddSearchProvider</span>();
  undefined <span data-x="dom-external-IsSearchProviderInstalled">IsSearchProviderInstalled</span>();
};</code></pre>

  <p>The <dfn method for="External"><code
  data-x="dom-external-AddSearchProvider">AddSearchProvider()</code></dfn> and <dfn method
  for="External"><code
  data-x="dom-external-IsSearchProviderInstalled">IsSearchProviderInstalled()</code></dfn> methods
  must do nothing.</p>

  </div>



  <div w-nodev>

  <h2 split-filename="iana" id="iana">IANA considerations</h2>

  <!-- https://www.w3.org/2002/06/registering-mediatype.html -->


  <h3><dfn><code>text/html</code></dfn></h3>

  <p>This registration is for community review and will be submitted to the IESG for review,
  approval, and registration with IANA.</p>

  <!--
   To: ietf-types@iana.org
   Subject: Registration of media type text/html
  -->

  <!--
   Obsoletes:
   https://www.rfc-editor.org/rfc/rfc2854

   Include a request to retire RFC 2854 pursuant to section 6.4 of RFC 2026.
  -->

  <dl>
   <dt>Type name:</dt>
   <dd>text</dd>
   <dt>Subtype name:</dt>
   <dd>html</dd>
   <dt>Required parameters:</dt>
   <dd>No required parameters</dd>
   <dt>Optional parameters:</dt>
   <dd>
    <dl>
     <dt><code data-x="">charset</code></dt>
     <dd>
      <p>The <code data-x="">charset</code> parameter may be provided to specify the
      <span>document's character encoding</span>, overriding any <span data-x="character encoding
      declaration">character encoding declarations</span> in the document other than a Byte Order
      Mark (BOM). The parameter's value must be an <span>ASCII case-insensitive</span> match for the
      string "<code data-x="">utf-8</code>". <ref>ENCODING</ref></p>
     </dd>
    </dl>
   </dd>
   <dt>Encoding considerations:</dt>
   <dd>
    8bit (see the section on <span data-x="character encoding declaration">character encoding
    declarations</span>)
   </dd>
   <dt>Security considerations:</dt>
   <dd>
    <p>Entire novels have been written about the security considerations that apply to HTML
    documents. Many are listed in this document, to which the reader is referred for more details.
    Some general concerns bear mentioning here, however:</p>

    <p>HTML is scripted language, and has a large number of APIs (some of which are described in
    this document). Script can expose the user to potential risks of information leakage, credential
    leakage, cross-site scripting attacks, cross-site request forgeries, and a host of other
    problems. While the designs in this specification are intended to be safe if implemented
    correctly, a full implementation is a massive undertaking and, as with any software, user agents
    are likely to have security bugs.</p>

    <p>Even without scripting, there are specific features in HTML which, for historical reasons,
    are required for broad compatibility with legacy content but that expose the user to unfortunate
    security problems. In particular, the <code>img</code> element can be used in conjunction with
    some other features as a way to effect a port scan from the user's location on the Internet.
    This can expose local network topologies that the attacker would otherwise not be able to
    determine.</p>

    <p>HTML relies on a compartmentalization scheme sometimes known as the <i>same-origin
    policy</i>. An <span>origin</span> in most cases consists of all the pages served from the same
    host, on the same port, using the same protocol.</p>

    <p>It is critical, therefore, to ensure that any untrusted content that forms part of a site be
    hosted on a different <span>origin</span> than any sensitive content on that site. Untrusted
    content can easily spoof any other page on the same origin, read data from that origin, cause
    scripts in that origin to execute, submit forms to and from that origin even if they are
    protected from cross-site request forgery attacks by unique tokens, and make use of any
    third-party resources exposed to or rights granted to that origin.</p>
   </dd>
   <dt>Interoperability considerations:</dt>
   <dd>
    Rules for processing both conforming and non-conforming content
    are defined in this specification.
   </dd>
   <dt>Published specification:</dt>
   <dd id="authors-using-html">
    This document is the relevant specification. Labeling a resource
    with the <code>text/html</code> type asserts that the resource is
    an <span data-x="HTML documents">HTML document</span> using
    <span>the HTML syntax</span>.
   </dd>
   <dt>Applications that use this media type:</dt>
   <dd>
    Web browsers, tools for processing web content, HTML authoring
    tools, search engines, validators.
   </dd>
   <dt>Additional information:</dt>
   <dd>
    <dl>
     <dt>Magic number(s):</dt>
     <dd>No sequence of bytes can uniquely identify an HTML
     document. More information on detecting HTML documents is
     available in <cite>MIME Sniffing</cite>. <ref>MIMESNIFF</ref></dd>
     <dt>File extension(s):</dt>
     <dd>"<code data-x="">html</code>" and "<code data-x="">htm</code>"
     are commonly, but certainly not exclusively, used as the
     extension for HTML documents.</dd>
     <dt>Macintosh file type code(s):</dt>
     <dd><code data-x="">TEXT</code></dd>
    </dl>
   </dd>
   <dt>Person &amp; email address to contact for further information:</dt>
   <dd>Ian Hickson &lt;ian@hixie.ch></dd>
   <dt>Intended usage:</dt>
   <dd>Common</dd>
   <dt>Restrictions on usage:</dt>
   <dd>No restrictions apply.</dd>
   <dt>Author:</dt>
   <dd>Ian Hickson &lt;ian@hixie.ch></dd>
   <dt>Change controller:</dt>
   <dd>W3C</dd>
  </dl>

  <p><span data-x="concept-url-fragment">Fragments</span> used with <code>text/html</code> resources
  either refer to the <span>indicated part</span> of the corresponding <code>Document</code>, or
  provide state information for in-page scripts.</p>



  <h3><dfn><code>multipart/x-mixed-replace</code></dfn></h3>

  <p>This registration is for community review and will be submitted to the IESG for review,
  approval, and registration with IANA.</p>

  <!--
   To: ietf-types@iana.org
   Subject: Registration of media type multipart/x-mixed-replace
  -->

  <dl>
   <dt>Type name:</dt>
   <dd>multipart</dd>
   <dt>Subtype name:</dt>
   <dd>x-mixed-replace</dd>
   <dt>Required parameters:</dt>
   <dd>
    <ul class="brief">
     <li><code data-x="">boundary</code> (defined in RFC2046) <ref>RFC2046</ref>
    </ul>
   </dd>
   <dt>Optional parameters:</dt>
   <dd>No optional parameters.</dd>
   <dt>Encoding considerations:</dt>
   <dd>binary</dd>
   <dt>Security considerations:</dt>
   <dd>
    Subresources of a <code>multipart/x-mixed-replace</code>
    resource can be of any type, including types with non-trivial
    security implications such as <code>text/html</code>.
   </dd>
   <dt>Interoperability considerations:</dt>
   <dd>
    None.
   </dd>
   <dt>Published specification:</dt>
   <dd>
    This specification describes processing rules for web browsers.
    Conformance requirements for generating resources with this type are the same as for <code>multipart/mixed</code>. <ref>RFC2046</ref>
   </dd>
   <dt>Applications that use this media type:</dt>
   <dd>
    This type is intended to be used in resources generated by web servers, for consumption by web browsers.
   </dd>
   <dt>Additional information:</dt>
   <dd>
    <dl>
     <dt>Magic number(s):</dt>
     <dd>No sequence of bytes can uniquely identify a <code>multipart/x-mixed-replace</code> resource.</dd>
     <dt>File extension(s):</dt>
     <dd>No specific file extensions are recommended for this type.</dd>
     <dt>Macintosh file type code(s):</dt>
     <dd>No specific Macintosh file type codes are recommended for this type.</dd>
    </dl>
   </dd>
   <dt>Person &amp; email address to contact for further information:</dt>
   <dd>Ian Hickson &lt;ian@hixie.ch></dd>
   <dt>Intended usage:</dt>
   <dd>Common</dd>
   <dt>Restrictions on usage:</dt>
   <dd>No restrictions apply.</dd>
   <dt>Author:</dt>
   <dd>Ian Hickson &lt;ian@hixie.ch></dd>
   <dt>Change controller:</dt>
   <dd>W3C</dd>
  </dl>

  <p><span data-x="concept-url-fragment">Fragments</span> used with
  <code>multipart/x-mixed-replace</code> resources apply to each body part as defined by the type
  used by that body part.</p>


  <h3><dfn><code>application/xhtml+xml</code></dfn></h3>

  <p>This registration is for community review and will be submitted to the IESG for review,
  approval, and registration with IANA.</p>

  <!--
   To: ietf-types@iana.org
   Subject: Registration of media type application/xhtml+xml
  -->

  <!--
   Obsoletes:
   https://www.rfc-editor.org/rfc/rfc3236
  -->

  <dl>
   <dt>Type name:</dt>
   <dd>application</dd>
   <dt>Subtype name:</dt>
   <dd>xhtml+xml</dd>
   <dt>Required parameters:</dt>
   <dd>Same as for <code>application/xml</code> <ref>RFC7303</ref></dd>
   <dt>Optional parameters:</dt>
   <dd>Same as for <code>application/xml</code> <ref>RFC7303</ref></dd>
   <dt>Encoding considerations:</dt>
   <dd>Same as for <code>application/xml</code> <ref>RFC7303</ref></dd>
   <dt>Security considerations:</dt>
   <dd>Same as for <code>application/xml</code> <ref>RFC7303</ref></dd>
   <dt>Interoperability considerations:</dt>
   <dd>Same as for <code>application/xml</code> <ref>RFC7303</ref></dd>
   <dt id="authors-using-xhtml">Published specification:</dt>
   <dd>Labeling a resource with the <code>application/xhtml+xml</code> type asserts that the
   resource is an XML document that likely has a <span>document element</span> from the <span>HTML
   namespace</span>. Thus, the relevant specifications are <cite>XML</cite>, <cite>Namespaces in
   XML</cite>, and this specification. <ref>XML</ref> <ref>XMLNS</ref></dd>
   <dt>Applications that use this media type:</dt>
   <dd>Same as for <code>application/xml</code> <ref>RFC7303</ref></dd>
   <dt>Additional information:</dt>
   <dd>
    <dl>
     <dt>Magic number(s):</dt>
     <dd>Same as for <code>application/xml</code> <ref>RFC7303</ref></dd>
     <dt>File extension(s):</dt>
     <dd>"<code data-x="">xhtml</code>" and "<code data-x="">xht</code>" are sometimes used as
     extensions for XML resources that have a <span>document element</span> from the <span>HTML
     namespace</span>.</dd>
     <dt>Macintosh file type code(s):</dt>
     <dd><code data-x="">TEXT</code></dd>
    </dl>
   </dd>
   <dt>Person &amp; email address to contact for further information:</dt>
   <dd>Ian Hickson &lt;ian@hixie.ch></dd>
   <dt>Intended usage:</dt>
   <dd>Common</dd>
   <dt>Restrictions on usage:</dt>
   <dd>No restrictions apply.</dd>
   <dt>Author:</dt>
   <dd>Ian Hickson &lt;ian@hixie.ch></dd>
   <dt>Change controller:</dt>
   <dd>W3C</dd>
  </dl>

  <p><span data-x="concept-url-fragment">Fragments</span> used with
  <code>application/xhtml+xml</code> resources have the same semantics as with any
  <span>XML MIME type</span>. <ref>RFC7303</ref></p>


  <h3><dfn><code>text/ping</code></dfn></h3>

  <p>This registration is for community review and will be submitted to the IESG for review,
  approval, and registration with IANA.</p>

  <!--
   To: ietf-types@iana.org
   Subject: Registration of media type text/ping
  -->

  <dl>
   <dt>Type name:</dt>
   <dd>text</dd>
   <dt>Subtype name:</dt>
   <dd>ping</dd>
   <dt>Required parameters:</dt>
   <dd>No parameters</dd>
   <dt>Optional parameters:</dt>
   <dd>
    <dl>
     <dt><code data-x="">charset</code></dt>
     <dd>
      <p>The <code data-x="">charset</code> parameter may be provided. The parameter's value must be
      "<code data-x="">utf-8</code>". This parameter serves no purpose; it is only allowed for
      compatibility with legacy servers.</p>
     </dd>
    </dl>
   </dd>
   <dt>Encoding considerations:</dt>
   <dd>Not applicable.</dd>
   <dt>Security considerations:</dt>
   <dd>
    <p>If used exclusively in the fashion described in the context of
    <span>hyperlink auditing</span>, this type introduces no new
    security concerns.</p>
   </dd>
   <dt>Interoperability considerations:</dt>
   <dd>
    Rules applicable to this type are defined in this specification.
   </dd>
   <dt>Published specification:</dt>
   <dd>
    This document is the relevant specification.
   </dd>
   <dt>Applications that use this media type:</dt>
   <dd>
    Web browsers.
   </dd>
   <dt>Additional information:</dt>
   <dd>
    <dl>
     <dt>Magic number(s):</dt>
     <dd><code>text/ping</code> resources always consist of the four
     bytes 0x50 0x49 0x4E 0x47 (`<code data-x="">PING</code>`).</dd>
     <dt>File extension(s):</dt>
     <dd>No specific file extension is recommended for this type.</dd>
     <dt>Macintosh file type code(s):</dt>
     <dd>No specific Macintosh file type codes are recommended for this type.</dd>
    </dl>
   </dd>
   <dt>Person &amp; email address to contact for further information:</dt>
   <dd>Ian Hickson &lt;ian@hixie.ch></dd>
   <dt>Intended usage:</dt>
   <dd>Common</dd>
   <dt>Restrictions on usage:</dt>
   <dd>Only intended for use with HTTP POST requests generated as part
   of a web browser's processing of the <code
   data-x="attr-hyperlink-ping">ping</code> attribute.</dd>
   <dt>Author:</dt>
   <dd>Ian Hickson &lt;ian@hixie.ch></dd>
   <dt>Change controller:</dt>
   <dd>W3C</dd>
  </dl>

  <p><span data-x="concept-url-fragment">Fragments</span> have no meaning with
  <code>text/ping</code> resources.</p>


  <h3><dfn><code>application/microdata+json</code></dfn></h3>

  <p>This registration is for community review and will be submitted to the IESG for review,
  approval, and registration with IANA.</p>

  <!--
   To: ietf-types@iana.org
   Subject: Registration of media type application/microdata+json
  -->

  <dl>
   <dt>Type name:</dt>
   <dd>application</dd>
   <dt>Subtype name:</dt>
   <dd>microdata+json</dd>
   <dt>Required parameters:</dt>
   <dd>Same as for <code>application/json</code> <ref>JSON</ref></dd>
   <dt>Optional parameters:</dt>
   <dd>Same as for <code>application/json</code> <ref>JSON</ref></dd>
   <dt>Encoding considerations:</dt>
   <dd>8bit (always UTF-8)</dd>
   <dt>Security considerations:</dt>
   <dd>Same as for <code>application/json</code> <ref>JSON</ref></dd>
   <dt>Interoperability considerations:</dt>
   <dd>Same as for <code>application/json</code> <ref>JSON</ref></dd>
   <dt>Published specification:</dt>
   <dd>
    Labeling a resource with the <code>application/microdata+json</code> type asserts that the
    resource is a JSON text that consists of an object with a single entry called "<code
    data-x="">items</code>" consisting of an array of entries, each of which consists of an object
    with an entry called "<code data-x="">id</code>" whose value is a string, an entry called "<code
    data-x="">type</code>" whose value is another string, and an entry called "<code
    data-x="">properties</code>" whose value is an object whose entries each have a value consisting
    of an array of either objects or strings, the objects being of the same form as the objects in
    the aforementioned "<code data-x="">items</code>" entry. Thus, the relevant specifications are
    <cite>JSON</cite> and this specification. <ref>JSON</ref>
   </dd>
   <dt>Applications that use this media type:</dt>
   <dd>
    <p>Applications that transfer data intended for use with HTML's microdata feature, especially in
    the context of drag-and-drop, are the primary application class for this type.</p>
   </dd>
   <dt>Additional information:</dt>
   <dd>
    <dl>
     <dt>Magic number(s):</dt>
     <dd>Same as for <code>application/json</code> <ref>JSON</ref></dd>
     <dt>File extension(s):</dt>
     <dd>Same as for <code>application/json</code> <ref>JSON</ref></dd>
     <dt>Macintosh file type code(s):</dt>
     <dd>Same as for <code>application/json</code> <ref>JSON</ref></dd>
    </dl>
   </dd>
   <dt>Person &amp; email address to contact for further information:</dt>
   <dd>Ian Hickson &lt;ian@hixie.ch></dd>
   <dt>Intended usage:</dt>
   <dd>Common</dd>
   <dt>Restrictions on usage:</dt>
   <dd>No restrictions apply.</dd>
   <dt>Author:</dt>
   <dd>Ian Hickson &lt;ian@hixie.ch></dd>
   <dt>Change controller:</dt>
   <dd>W3C</dd>
  </dl>

  <p><span data-x="concept-url-fragment">Fragments</span> used with
  <code>application/microdata+json</code> resources have the same semantics as when used with
  <code>application/json</code> (namely, at the time of writing, no semantics at all).
  <ref>JSON</ref></p>


  <h3><dfn><code>text/event-stream</code></dfn></h3>

  <p>This registration is for community review and will be submitted to the IESG for review,
  approval, and registration with IANA.</p>

  <!--
   To: ietf-types@iana.org
   Subject: Registration of media type text/event-stream
  -->

  <dl>
   <dt>Type name:</dt>
   <dd>text</dd>
   <dt>Subtype name:</dt>
   <dd>event-stream</dd>
   <dt>Required parameters:</dt>
   <dd>No parameters</dd>
   <dt>Optional parameters:</dt>
   <dd>
    <dl>
     <dt><code data-x="">charset</code></dt>
     <dd>
      <p>The <code data-x="">charset</code> parameter may be provided. The parameter's value must be
      "<code data-x="">utf-8</code>". This parameter serves no purpose; it is only allowed for
      compatibility with legacy servers.</p>
     </dd>
    </dl>
   </dd>
   <dt>Encoding considerations:</dt>
   <dd>8bit (always UTF-8)</dd>
   <dt>Security considerations:</dt>
   <dd>
    <p>An event stream from an origin distinct from the origin of the content consuming the event
    stream can result in information leakage. To avoid this, user agents are required to apply CORS
    semantics. <ref>FETCH</ref></p>

    <p>Event streams can overwhelm a user agent; a user agent is expected to apply suitable
    restrictions to avoid depleting local resources because of an overabundance of information from
    an event stream.</p>

    <p>Servers can be overwhelmed if a situation develops in which the server is causing clients to
    reconnect rapidly. Servers should use a 5xx status code to indicate capacity problems, as this
    will prevent conforming clients from reconnecting automatically.</p>
   </dd>
   <dt>Interoperability considerations:</dt>
   <dd>
    Rules for processing both conforming and non-conforming content are defined in this
    specification.
   </dd>
   <dt>Published specification:</dt>
   <dd>
    This document is the relevant specification.
   </dd>
   <dt>Applications that use this media type:</dt>
   <dd>
    Web browsers and tools using web services.
   </dd>
   <dt>Additional information:</dt>
   <dd>
    <dl>
     <dt>Magic number(s):</dt>
     <dd>No sequence of bytes can uniquely identify an event stream.</dd>
     <dt>File extension(s):</dt>
     <dd>No specific file extensions are recommended for this type.</dd>
     <dt>Macintosh file type code(s):</dt>
     <dd>No specific Macintosh file type codes are recommended for this type.</dd>
    </dl>
   </dd>
   <dt>Person &amp; email address to contact for further information:</dt>
   <dd>Ian Hickson &lt;ian@hixie.ch></dd>
   <dt>Intended usage:</dt>
   <dd>Common</dd>
   <dt>Restrictions on usage:</dt>
   <dd>This format is only expected to be used by dynamic open-ended streams served using HTTP or a
   similar protocol. Finite resources are not expected to be labeled with this type.</dd>
   <dt>Author:</dt>
   <dd>Ian Hickson &lt;ian@hixie.ch></dd>
   <dt>Change controller:</dt>
   <dd>W3C</dd>
  </dl>

  <p><span data-x="concept-url-fragment">Fragments</span> have no meaning with
  <code>text/event-stream</code> resources.</p>


  <h3><dfn><code data-x="scheme-web">web+</code> scheme prefix</dfn></h3>

  <p>This section describes a convention for use with the IANA URI scheme registry. It does not
  itself register a specific scheme. <ref>RFC7595</ref></p>

  <dl>
   <dt>Scheme name:</dt>
   <dd>
    Schemes starting with the four characters "<code
    data-x="">web+</code>" followed by one or more letters in the range
    <code data-x="">a</code>-<code data-x="">z</code>.
   </dd>
   <dt>Status:</dt>
   <dd>Permanent</dd>
   <dt>Scheme syntax:</dt>
   <dd>Scheme-specific.</dd>
   <dt>Scheme semantics:</dt>
   <dd>Scheme-specific.</dd>
   <dt>Encoding considerations:</dt>
   <dd>All "<code data-x="">web+</code>" schemes should use UTF-8 encodings where relevant.</dd>
   <dt>Applications/protocols that use this scheme name:</dt>
   <dd>Scheme-specific.</dd>
   <dt>Interoperability considerations:</dt>
   <dd>The scheme is expected to be used in the context of web applications.</dd>
   <dt>Security considerations:</dt>
   <dd>
    Any web page is able to register a handler for all "<code data-x="">web+</code>" schemes. As
    such, these schemes must not be used for features intended to be core platform features (e.g.,
    HTTP). Similarly, such schemes must not store confidential information in their URLs, such as
    usernames, passwords, personal information, or confidential project names.
   </dd>
   <dt>Contact:</dt>
   <dd>Ian Hickson &lt;ian@hixie.ch></dd>
   <dt>Change controller:</dt>
   <dd>Ian Hickson &lt;ian@hixie.ch></dd>
   <dt>References:</dt>
   <dd>
    <cite>Custom scheme handlers</cite>, HTML Living Standard:
    <a href="#custom-handlers">https://html.spec.whatwg.org/#custom-handlers</a>
   </dd>
  </dl>

  </div>



  <h2 split-filename="indices" id="index" class="no-num">Index</h2>

  <div w-nodev>

  <p>The following sections only cover conforming elements and features.</p>

  </div>

  <h3 class="no-num">Elements</h3>

  <!-- NON-NORMATIVE SECTION -->

  <table>
   <caption>List of elements</caption>
   <thead>
    <tr>
     <th> Element
     <th> Description
     <th> Categories
     <th> Parents&dagger;
     <th> Children
     <th> Attributes
     <th> Interface
   <tbody>

    <tr>
     <th><code>a</code></th>
     <td>Hyperlink</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>*;
         <span data-x="Interactive content">interactive</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span>transparent</span>*</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-hyperlink-href">href</code>;
         <code data-x="attr-hyperlink-target">target</code>;
         <code data-x="attr-hyperlink-download">download</code>;
         <code data-x="attr-hyperlink-ping">ping</code>;
         <code data-x="attr-hyperlink-rel">rel</code>;
         <code data-x="attr-hyperlink-hreflang">hreflang</code>;
         <code data-x="attr-hyperlink-type">type</code>;
         <code data-x="attr-hyperlink-referrerpolicy">referrerpolicy</code></td>
     <td><code>HTMLAnchorElement</code></td>
    </tr>

    <tr>
     <th><code>abbr</code></th>
     <td>Abbreviation</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>address</code></th>
     <td>Contact information for a page or <code>article</code> element</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="Flow content">flow</span>*</td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>area</code></th>
     <td>Hyperlink or dead area on an image map</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span>*</td>
     <td>empty</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-area-alt">alt</code>;
         <code data-x="attr-area-coords">coords</code>;
         <code data-x="attr-area-shape">shape</code>;
         <code data-x="attr-hyperlink-href">href</code>;
         <code data-x="attr-hyperlink-target">target</code>;
         <code data-x="attr-hyperlink-download">download</code>;
         <code data-x="attr-hyperlink-ping">ping</code>;
         <code data-x="attr-hyperlink-rel">rel</code>;
         <code data-x="attr-hyperlink-referrerpolicy">referrerpolicy</code></td>
     <td><code>HTMLAreaElement</code></td>
    </tr>

    <tr>
     <th><code>article</code></th>
     <td>Self-contained syndicatable or reusable composition</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Sectioning content">sectioning</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>aside</code></th>
     <td>Sidebar for tangentially related content</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Sectioning content">sectioning</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>audio</code></th>
     <td>Audio player</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Embedded content">embedded</span>;
         <span data-x="Interactive content">interactive</span>;
         <span data-x="Palpable content">palpable</span>*</td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><code>source</code>*;
         <code>track</code>*;
         <span>transparent</span>*</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-media-src">src</code>;
         <code data-x="attr-media-crossorigin">crossorigin</code>;
         <code data-x="attr-media-preload">preload</code>;
         <code data-x="attr-media-autoplay">autoplay</code>;
         <code data-x="attr-media-loop">loop</code>;
         <code data-x="attr-media-muted">muted</code>;
         <code data-x="attr-media-controls">controls</code></td>
     <td><code>HTMLAudioElement</code></td>
    </tr>

    <tr>
     <th><code>b</code></th>
     <td>Keywords</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>base</code></th>
     <td>Base URL and default target <span>navigable</span> for <span data-x="attr-hyperlink-target">hyperlinks</span> and <span data-x="attr-fs-target">forms</span></td>
     <td><span data-x="Metadata content">metadata</span></td>
     <td><code>head</code></td>
     <td>empty</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-base-href">href</code>;
         <code data-x="attr-base-target">target</code></td>
     <td><code>HTMLBaseElement</code></td>
    </tr>

    <tr>
     <th><code>bdi</code></th>
     <td>Text directionality isolation</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>bdo</code></th>
     <td>Text directionality formatting</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>blockquote</code></th>
     <td>A section quoted from another source</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-blockquote-cite">cite</code></td>
     <td><code>HTMLQuoteElement</code></td>
    </tr>

    <tr>
     <th><code>body</code></th>
     <td>Document body</td>
     <td>none</td>
     <td><code>html</code></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="handler-window-onafterprint">onafterprint</code>;
         <code data-x="handler-window-onbeforeprint">onbeforeprint</code>;
         <code data-x="handler-window-onbeforeunload">onbeforeunload</code>;
         <code data-x="handler-window-onhashchange">onhashchange</code>;
         <code data-x="handler-window-onlanguagechange">onlanguagechange</code>;
         <code data-x="handler-window-onmessage">onmessage</code>;
         <code data-x="handler-window-onmessageerror">onmessageerror</code>;
         <code data-x="handler-window-onoffline">onoffline</code>;
         <code data-x="handler-window-ononline">ononline</code>;
         <code data-x="handler-window-onpageswap">onpageswap</code>;
         <code data-x="handler-window-onpagehide">onpagehide</code>;
         <code data-x="handler-window-onpagereveal">onpagereveal</code>;
         <code data-x="handler-window-onpageshow">onpageshow</code>;
         <code data-x="handler-window-onpopstate">onpopstate</code>;
         <code data-x="handler-window-onrejectionhandled">onrejectionhandled</code>;
         <code data-x="handler-window-onstorage">onstorage</code>;
         <code data-x="handler-window-onunhandledrejection">onunhandledrejection</code>;
         <code data-x="handler-window-onunload">onunload</code></td>
     <td><code>HTMLBodyElement</code></td>
    </tr>

    <tr>
     <th><code>br</code></th>
     <td>Line break, e.g. in poem or postal address</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td>empty</td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLBRElement</code></td>
    </tr>

    <tr>
     <th><code>button</code></th>
     <td>Button control</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Interactive content">interactive</span>;
         <span data-x="category-listed">listed</span>;
         <span data-x="category-label">labelable</span>;
         <span data-x="category-submit">submittable</span>;
         <span data-x="Form-associated element">form-associated</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span>*</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-button-command">command</code>;
         <code data-x="attr-button-commandfor">commandfor</code>;
         <code data-x="attr-fe-disabled">disabled</code>;
         <code data-x="attr-fae-form">form</code>;
         <code data-x="attr-fs-formaction">formaction</code>;
         <code data-x="attr-fs-formenctype">formenctype</code>;
         <code data-x="attr-fs-formmethod">formmethod</code>;
         <code data-x="attr-fs-formnovalidate">formnovalidate</code>;
         <code data-x="attr-fs-formtarget">formtarget</code>;
         <code data-x="attr-fe-name">name</code>;
         <code data-x="attr-popovertarget">popovertarget</code>;
         <code data-x="attr-popovertargetaction">popovertargetaction</code>;
         <code data-x="attr-button-type">type</code>;
         <code data-x="attr-button-value">value</code></td>
     <td><code>HTMLButtonElement</code></td>
    </tr>

    <tr>
     <th><code>canvas</code></th>
     <td>Scriptable bitmap canvas</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Embedded content">embedded</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span>transparent</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-canvas-width">width</code>;
         <code data-x="attr-canvas-height">height</code></td>
     <td><code>HTMLCanvasElement</code></td>
    </tr>

    <tr>
     <th><code>caption</code></th>
     <td>Table caption</td>
     <td>none</td>
     <td><code>table</code></td>
     <td><span data-x="Flow content">flow</span>*</td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLTableCaptionElement</code></td>
    </tr>

    <tr>
     <th><code>cite</code></th>
     <td>Title of a work</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>code</code></th>
     <td>Computer code</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>col</code></th>
     <td>Table column</td>
     <td>none</td>
     <td><code>colgroup</code></td>
     <td>empty</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-col-span">span</code></td>
     <td><code>HTMLTableColElement</code></td>
    </tr>

    <tr>
     <th><code>colgroup</code></th>
     <td>Group of columns in a table</td>
     <td>none</td>
     <td><code>table</code></td>
     <td><code>col</code>*;
         <code>template</code>*</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-colgroup-span">span</code></td>
     <td><code>HTMLTableColElement</code></td>
    </tr>

    <tr>
     <th><code>data</code></th>
     <td>Machine-readable equivalent</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-data-value">value</code></td>
     <td><code>HTMLDataElement</code></td>
    </tr>

    <tr>
     <th><code>datalist</code></th>
     <td>Container for options for <span data-x="attr-input-list">combo box control</span></td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span>*;
         <code>option</code>*;
         <span>script-supporting elements</span>*</td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLDataListElement</code></td>
    </tr>

    <tr>
     <th><code>dd</code></th>
     <td>Content for corresponding <code>dt</code> element(s)</td>
     <td>none</td>
     <td><code>dl</code>;
         <code>div</code>*</td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>del</code></th>
     <td>A removal from the document</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>*;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span>transparent</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-mod-cite">cite</code>;
         <code data-x="attr-mod-datetime">datetime</code></td>
     <td><code>HTMLModElement</code></td>
    </tr>

    <tr>
     <th><code>details</code></th>
     <td>Disclosure control for hiding details</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Interactive content">interactive</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><code>summary</code>*;
         <span data-x="Flow content">flow</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-details-name">name</code>;
         <code data-x="attr-details-open">open</code></td>
     <td><code>HTMLDetailsElement</code></td>
    </tr>

    <tr>
     <th><code>dfn</code></th>
     <td>Defining instance</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span>*</td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>dialog</code></th>
     <td>Dialog box or window</td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-dialog-open">open</code></td>
     <td><code>HTMLDialogElement</code></td>
    </tr>

    <tr>
     <th><code>div</code></th>
     <td>Generic flow container, or container for name-value groups in <code>dl</code> elements</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span>;
         <code>dl</code></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLDivElement</code></td>
    </tr>

    <tr>
     <th><code>dl</code></th>
     <td>Association list consisting of zero or more name-value groups</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><code>dt</code>*;
         <code>dd</code>*;
         <code>div</code>*;
         <span>script-supporting elements</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLDListElement</code></td>
    </tr>

    <tr>
     <th><code>dt</code></th>
     <td>Legend for corresponding <code>dd</code> element(s)</td>
     <td>none</td>
     <td><code>dl</code>;
         <code>div</code>*</td>
     <td><span data-x="Flow content">flow</span>*</td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>em</code></th>
     <td>Stress emphasis</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>embed</code></th>
     <td><span>Plugin</span></td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Embedded content">embedded</span>;
         <span data-x="Interactive content">interactive</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td>empty</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-embed-src">src</code>;
         <code data-x="attr-embed-type">type</code>;
         <code data-x="attr-dim-width">width</code>;
         <code data-x="attr-dim-height">height</code>;
         any*</td>
     <td><code>HTMLEmbedElement</code></td>
    </tr>

    <tr>
     <th><code>fieldset</code></th>
     <td>Group of form controls</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="category-listed">listed</span>;
         <span data-x="Form-associated element">form-associated</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><code>legend</code>*;
         <span data-x="Flow content">flow</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-fieldset-disabled">disabled</code>;
         <code data-x="attr-fae-form">form</code>;
         <code data-x="attr-fe-name">name</code></td>
     <td><code>HTMLFieldSetElement</code></td>
    </tr>

    <tr>
     <th><code>figcaption</code></th>
     <td>Caption for <code>figure</code></td>
     <td>none</td>
     <td><code>figure</code></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>figure</code></th>
     <td>Figure with optional caption</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><code>figcaption</code>*;
         <span data-x="Flow content">flow</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>footer</code></th>
     <td>Footer for a page or section</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="Flow content">flow</span>*</td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>form</code></th>
     <td>User-submittable form</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="Flow content">flow</span>*</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-form-accept-charset">accept-charset</code>;
         <code data-x="attr-fs-action">action</code>;
         <code data-x="attr-form-autocomplete">autocomplete</code>;
         <code data-x="attr-fs-enctype">enctype</code>;
         <code data-x="attr-fs-method">method</code>;
         <code data-x="attr-form-name">name</code>;
         <code data-x="attr-fs-novalidate">novalidate</code>;
         <code data-x="attr-form-rel">rel</code>;
         <code data-x="attr-fs-target">target</code></td>
     <td><code>HTMLFormElement</code></td>
    </tr>

    <tr>
     <th><code>h1</code>, <code>h2</code>, <code>h3</code>, <code>h4</code>, <code>h5</code>, <code>h6</code></th>
     <td>Heading</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Heading content">heading</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><code>legend</code>;
         <code>summary</code>;
         <span data-x="Flow content">flow</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLHeadingElement</code></td>
    </tr>

    <tr>
     <th><code>head</code></th>
     <td>Container for document metadata</td>
     <td>none</td>
     <td><code>html</code></td>
     <td><span data-x="Metadata content">metadata content</span>*</td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLHeadElement</code></td>
    </tr>

    <tr>
     <th><code>header</code></th>
     <td>Introductory or navigational aids for a page or section</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="Flow content">flow</span>*</td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>hgroup</code></th>
     <td>Heading container</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td>
      <code>legend</code>;
      <code>summary</code>;
      <span data-x="Flow content">flow</span></td>
     <td><code>h1</code>;
         <code>h2</code>;
         <code>h3</code>;
         <code>h4</code>;
         <code>h5</code>;
         <code>h6</code>;
         <span>script-supporting elements</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>hr</code></th>
     <td>Thematic break</td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td>empty</td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLHRElement</code></td>
    </tr>

    <tr>
     <th><code>html</code></th>
     <td>Root element</td>
     <td>none</td>
     <td>none*</td>
     <td><code>head</code>*;
         <code>body</code>*</td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLHtmlElement</code></td>
    </tr>

    <tr>
     <th><code>i</code></th>
     <td>Alternate voice</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>iframe</code></th>
     <td><span>Child navigable</span></td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Embedded content">embedded</span>;
         <span data-x="Interactive content">interactive</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td>empty</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-iframe-src">src</code>;
         <code data-x="attr-iframe-srcdoc">srcdoc</code>;
         <code data-x="attr-iframe-name">name</code>;
         <code data-x="attr-iframe-sandbox">sandbox</code>;
         <code data-x="attr-iframe-allow">allow</code>;
         <code data-x="attr-iframe-allowfullscreen">allowfullscreen</code>;
         <code data-x="attr-dim-width">width</code>;
         <code data-x="attr-dim-height">height</code>;
         <code data-x="attr-iframe-referrerpolicy">referrerpolicy</code>;
         <code data-x="attr-iframe-loading">loading</code></td>
     <td><code>HTMLIFrameElement</code></td>
    </tr>

    <tr>
     <th><code>img</code></th>
     <td>Image</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Embedded content">embedded</span>;
         <span data-x="Interactive content">interactive</span>*;
         <span data-x="Form-associated element">form-associated</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span>;
         <code>picture</code></td>
     <td>empty</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-img-alt">alt</code>;
         <code data-x="attr-img-src">src</code>;
         <code data-x="attr-img-srcset">srcset</code>;
         <code data-x="attr-img-sizes">sizes</code>;
         <code data-x="attr-img-crossorigin">crossorigin</code>;
         <code data-x="attr-hyperlink-usemap">usemap</code>;
         <code data-x="attr-img-ismap">ismap</code>;
         <code data-x="attr-dim-width">width</code>;
         <code data-x="attr-dim-height">height</code>;
         <code data-x="attr-img-referrerpolicy">referrerpolicy</code>;
         <code data-x="attr-img-decoding">decoding</code>;
         <code data-x="attr-img-loading">loading</code>;
         <code data-x="attr-img-fetchpriority">fetchpriority</code></td>
     <td><code>HTMLImageElement</code></td>
    </tr>

    <tr>
     <th><code>input</code></th>
     <td>Form control</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Interactive content">interactive</span>*;
         <span data-x="category-listed">listed</span>;
         <span data-x="category-label">labelable</span>;
         <span data-x="category-submit">submittable</span>;
         <span data-x="category-reset">resettable</span>;
         <span data-x="Form-associated element">form-associated</span>;
         <span data-x="Palpable content">palpable</span>*</td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td>empty</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-input-accept">accept</code>;
         <code data-x="attr-input-alpha">alpha</code>;
         <code data-x="attr-input-alt">alt</code>;
         <code data-x="attr-fe-autocomplete">autocomplete</code>;
         <code data-x="attr-input-checked">checked</code>;
         <code data-x="attr-input-colorspace">colorspace</code>;
         <code data-x="attr-fe-dirname">dirname</code>;
         <code data-x="attr-fe-disabled">disabled</code>;
         <code data-x="attr-fae-form">form</code>;
         <code data-x="attr-fs-formaction">formaction</code>;
         <code data-x="attr-fs-formenctype">formenctype</code>;
         <code data-x="attr-fs-formmethod">formmethod</code>;
         <code data-x="attr-fs-formnovalidate">formnovalidate</code>;
         <code data-x="attr-fs-formtarget">formtarget</code>;
         <code data-x="attr-dim-height">height</code>;
         <code data-x="attr-input-list">list</code>;
         <code data-x="attr-input-max">max</code>;
         <code data-x="attr-input-maxlength">maxlength</code>;
         <code data-x="attr-input-min">min</code>;
         <code data-x="attr-input-minlength">minlength</code>;
         <code data-x="attr-input-multiple">multiple</code>;
         <code data-x="attr-fe-name">name</code>;
         <code data-x="attr-input-pattern">pattern</code>;
         <code data-x="attr-input-placeholder">placeholder</code>;
         <code data-x="attr-popovertarget">popovertarget</code>;
         <code data-x="attr-popovertargetaction">popovertargetaction</code>;
         <code data-x="attr-input-readonly">readonly</code>;
         <code data-x="attr-input-required">required</code>;
         <code data-x="attr-input-size">size</code>;
         <code data-x="attr-input-src">src</code>;
         <code data-x="attr-input-step">step</code>;
         <code data-x="attr-input-type">type</code>;
         <code data-x="attr-input-value">value</code>;
         <code data-x="attr-dim-width">width</code></td>
     <td><code>HTMLInputElement</code></td>
    </tr>

    <tr>
     <th><code>ins</code></th>
     <td>An addition to the document</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>*;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span>transparent</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-mod-cite">cite</code>;
         <code data-x="attr-mod-datetime">datetime</code></td>
     <td><code>HTMLModElement</code></td>
    </tr>

    <tr>
     <th><code>kbd</code></th>
     <td>User input</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>label</code></th>
     <td>Caption for a form control</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Interactive content">interactive</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span>*</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-label-for">for</code></td>
     <td><code>HTMLLabelElement</code></td>
    </tr>

    <tr>
     <th><code>legend</code></th>
     <td>Caption for <code>fieldset</code></td>
     <td>none</td>
     <td><code>fieldset</code></td>
     <td><span data-x="Phrasing content">phrasing</span>;
         <span data-x="heading content">heading content</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLLegendElement</code></td>
    </tr>

    <tr>
     <th><code>li</code></th>
     <td>List item</td>
     <td>none</td>
     <td><code>ol</code>;
         <code>ul</code>;
         <code>menu</code>*</td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-li-value">value</code>*</td>
     <td><code>HTMLLIElement</code></td>
    </tr>

    <tr>
     <th><code>link</code></th>
     <td>Link metadata</td>
     <td><span data-x="Metadata content">metadata</span>;
         <span data-x="Flow content">flow</span>*;
         <span data-x="Phrasing content">phrasing</span>*</td>
     <td><code>head</code>;
         <code>noscript</code>*;
         <span data-x="phrasing content">phrasing</span>*</td>
     <td>empty</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-link-href">href</code>;
         <code data-x="attr-link-crossorigin">crossorigin</code>;
         <code data-x="attr-link-rel">rel</code>;
         <code data-x="attr-link-as">as</code>;
         <code data-x="attr-link-media">media</code>;
         <code data-x="attr-link-hreflang">hreflang</code>;
         <code data-x="attr-link-type">type</code>;
         <code data-x="attr-link-sizes">sizes</code>;
         <code data-x="attr-link-imagesrcset">imagesrcset</code>;
         <code data-x="attr-link-imagesizes">imagesizes</code>;
         <code data-x="attr-link-referrerpolicy">referrerpolicy</code>;
         <code data-x="attr-link-integrity">integrity</code>;
         <code data-x="attr-link-blocking">blocking</code>;
         <code data-x="attr-link-color">color</code>;
         <code data-x="attr-link-disabled">disabled</code>;
         <code data-x="attr-link-fetchpriority">fetchpriority</code></td>
     <td><code>HTMLLinkElement</code></td>
    </tr>

    <tr>
     <th><code>main</code></th>
     <td>Container for the dominant contents of the document</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span>*</td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>map</code></th>
     <td><span>Image map</span></td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>*;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span>transparent</span>;
         <code>area</code>*</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-map-name">name</code></td>
     <td><code>HTMLMapElement</code></td>
    </tr>

    <tr>
     <th><code>mark</code></th>
     <td>Highlight</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><span>MathML <code>math</code></span></th>
     <td>MathML root</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Embedded content">embedded</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td>per <ref>MATHML</ref></td>
     <td>per <ref>MATHML</ref></td>
     <td><code>Element</code></td>
    </tr>

    <tr>
     <th><code>menu</code></th>
     <td>Menu of commands</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Palpable content">palpable</span>*</td>
     <td><span data-x="Flow content">flow</span></td>
     <td><code>li</code>;
         <span>script-supporting elements</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLMenuElement</code></td>
    </tr>

    <tr>
     <th><code>meta</code></th>
     <td>Text metadata</td>
     <td><span data-x="Metadata content">metadata</span>;
         <span data-x="Flow content">flow</span>*;
         <span data-x="Phrasing content">phrasing</span>*</td>
     <td><code>head</code>;
         <code>noscript</code>*;
         <span data-x="phrasing content">phrasing</span>*</td>
     <td>empty</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-meta-name">name</code>;
         <code data-x="attr-meta-http-equiv">http-equiv</code>;
         <code data-x="attr-meta-content">content</code>;
         <code data-x="attr-meta-charset">charset</code>;
         <code data-x="attr-meta-media">media</code></td>
     <td><code>HTMLMetaElement</code></td>
    </tr>

    <tr>
     <th><code>meter</code></th>
     <td>Gauge</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="category-label">labelable</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span>*</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-meter-value">value</code>;
         <code data-x="attr-meter-min">min</code>;
         <code data-x="attr-meter-max">max</code>;
         <code data-x="attr-meter-low">low</code>;
         <code data-x="attr-meter-high">high</code>;
         <code data-x="attr-meter-optimum">optimum</code></td>
     <td><code>HTMLMeterElement</code></td>
    </tr>

    <tr>
     <th><code>nav</code></th>
     <td>Section with navigational links</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Sectioning content">sectioning</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>noscript</code></th>
     <td>Fallback content for script</td>
     <td><span data-x="Metadata content">metadata</span>;
         <span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span></td>
     <td><code>head</code>*;
         <span data-x="Phrasing content">phrasing</span>*</td>
     <td>varies*</td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>object</code></th>
     <td>Image, <span>child navigable</span>, or <span>plugin</span></td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Embedded content">embedded</span>;
         <span data-x="Interactive content">interactive</span>*;
         <span data-x="category-listed">listed</span>;
         <span data-x="Form-associated element">form-associated</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span>transparent</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-object-data">data</code>;
         <code data-x="attr-object-type">type</code>;
         <code data-x="attr-object-name">name</code>;
         <code data-x="attr-fae-form">form</code>;
         <code data-x="attr-dim-width">width</code>;
         <code data-x="attr-dim-height">height</code></td>
     <td><code>HTMLObjectElement</code></td>
    </tr>

    <tr>
     <th><code>ol</code></th>
     <td>Ordered list</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Palpable content">palpable</span>*</td>
     <td><span data-x="Flow content">flow</span></td>
     <td><code>li</code>;
         <span>script-supporting elements</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-ol-reversed">reversed</code>;
         <code data-x="attr-ol-start">start</code>;
         <code data-x="attr-ol-type">type</code></td>
     <td><code>HTMLOListElement</code></td>
    </tr>

    <tr>
     <th><code>optgroup</code></th>
     <td>Group of options in a list box</td>
     <td>none</td>
     <td><code>select</code></td>
     <td><code>option</code>;
         <span>script-supporting elements</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-optgroup-disabled">disabled</code>;
         <code data-x="attr-optgroup-label">label</code></td>
     <td><code>HTMLOptGroupElement</code></td>
    </tr>

    <tr>
     <th><code>option</code></th>
     <td>Option in a list box or combo box control</td>
     <td>none</td>
     <td><code>select</code>;
         <code>datalist</code>;
         <code>optgroup</code></td>
     <td><span data-x="text content">text</span>*</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-option-disabled">disabled</code>;
         <code data-x="attr-option-label">label</code>;
         <code data-x="attr-option-selected">selected</code>;
         <code data-x="attr-option-value">value</code></td>
     <td><code>HTMLOptionElement</code></td>
    </tr>

    <tr>
     <th><code>output</code></th>
     <td>Calculated output value</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="category-listed">listed</span>;
         <span data-x="category-label">labelable</span>;
         <span data-x="category-reset">resettable</span>;
         <span data-x="Form-associated element">form-associated</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-output-for">for</code>;
         <code data-x="attr-fae-form">form</code>;
         <code data-x="attr-fe-name">name</code></td>
     <td><code>HTMLOutputElement</code></td>
    </tr>

    <tr>
     <th><code>p</code></th>
     <td>Paragraph</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLParagraphElement</code></td>
    </tr>

    <tr>
     <th><code>picture</code></th>
     <td>Image</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Embedded content">embedded</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><code>source</code>*;
         one <code>img</code>;
         <span>script-supporting elements</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLPictureElement</code></td>
    </tr>

    <tr>
     <th><code>pre</code></th>
     <td>Block of preformatted text</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLPreElement</code></td>
    </tr>

    <tr>
     <th><code>progress</code></th>
     <td>Progress bar</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="category-label">labelable</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span>*</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-progress-value">value</code>;
         <code data-x="attr-progress-max">max</code></td>
     <td><code>HTMLProgressElement</code></td>
    </tr>

    <tr>
     <th><code>q</code></th>
     <td>Quotation</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-q-cite">cite</code></td>
     <td><code>HTMLQuoteElement</code></td>
    </tr>

    <tr>
     <th><code>rp</code></th>
     <td>Parenthesis for ruby annotation text</td>
     <td>none</td>
     <td><code>ruby</code></td>
     <td><span data-x="text content">text</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>rt</code></th>
     <td>Ruby annotation text</td>
     <td>none</td>
     <td><code>ruby</code></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>ruby</code></th>
     <td>Ruby annotation(s)</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span>;
         <code>rt</code>;
         <code>rp</code>*</td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>s</code></th>
     <td>Inaccurate text</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>samp</code></th>
     <td>Computer output</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>script</code></th>
     <td>Embedded script</td>
     <td><span data-x="Metadata content">metadata</span>;
         <span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Script-supporting elements">script-supporting</span></td>
     <td><code>head</code>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Script-supporting elements">script-supporting</span></td>
     <td>script, data, or script documentation*</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-script-src">src</code>;
         <code data-x="attr-script-type">type</code>;
         <code data-x="attr-script-nomodule">nomodule</code>;
         <code data-x="attr-script-async">async</code>;
         <code data-x="attr-script-defer">defer</code>;
         <code data-x="attr-script-crossorigin">crossorigin</code>;
         <code data-x="attr-script-integrity">integrity</code>;
         <code data-x="attr-script-referrerpolicy">referrerpolicy</code>;
         <code data-x="attr-script-blocking">blocking</code>;
         <code data-x="attr-script-fetchpriority">fetchpriority</code></td>
     <td><code>HTMLScriptElement</code></td>
    </tr>

    <tr>
     <th><code>search</code></th>
     <td>Container for search controls</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>section</code></th>
     <td>Generic document or application section</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Sectioning content">sectioning</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>select</code></th>
     <td>List box control</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Interactive content">interactive</span>;
         <span data-x="category-listed">listed</span>;
         <span data-x="category-label">labelable</span>;
         <span data-x="category-submit">submittable</span>;
         <span data-x="category-reset">resettable</span>;
         <span data-x="Form-associated element">form-associated</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><code>option</code>;
         <code>optgroup</code>;
         <span>script-supporting elements</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-fe-autocomplete">autocomplete</code>;
         <code data-x="attr-fe-disabled">disabled</code>;
         <code data-x="attr-fae-form">form</code>;
         <code data-x="attr-select-multiple">multiple</code>;
         <code data-x="attr-fe-name">name</code>;
         <code data-x="attr-select-required">required</code>;
         <code data-x="attr-select-size">size</code></td>
     <td><code>HTMLSelectElement</code></td>
    </tr>

    <tr>
     <th><code>slot</code></th>
     <td>Shadow tree slot</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span>transparent</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-slot-name">name</code></td>
     <td><code>HTMLSlotElement</code></td>
    </tr>

    <tr>
     <th><code>small</code></th>
     <td>Side comment</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>source</code></th>
     <td>Image source for <code>img</code> or media source for <code>video</code> or <code>audio</code></td>
     <td>none</td>
     <td><code>picture</code>;
         <code>video</code>;
         <code>audio</code></td>
     <td>empty</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-source-type">type</code>;
         <code data-x="attr-source-media">media</code>;
         <code data-x="attr-source-src">src</code>;
         <code data-x="attr-source-srcset">srcset</code>;
         <code data-x="attr-source-sizes">sizes</code>;
         <code data-x="attr-dim-width">width</code>;
         <code data-x="attr-dim-height">height</code></td>
     <td><code>HTMLSourceElement</code></td>
    </tr>

    <tr>
     <th><code>span</code></th>
     <td>Generic phrasing container</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLSpanElement</code></td>
    </tr>

    <tr>
     <th><code>strong</code></th>
     <td>Importance</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>style</code></th>
     <td>Embedded styling information</td>
     <td><span data-x="Metadata content">metadata</span></td>
     <td><code>head</code>;
         <code>noscript</code>*</td>
     <td>text*</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-style-media">media</code>;
         <code data-x="attr-style-blocking">blocking</code></td>
     <td><code>HTMLStyleElement</code></td>
    </tr>

    <tr>
     <th><code>sub</code></th>
     <td>Subscript</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>summary</code></th>
     <td>Caption for <code>details</code></td>
     <td>none</td>
     <td><code>details</code></td>
     <td><span data-x="Phrasing content">phrasing</span>;
         <span data-x="Heading content">heading content</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>sup</code></th>
     <td>Superscript</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><span>SVG <code>svg</code></span></th>
     <td>SVG root</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Embedded content">embedded</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td>per <ref>SVG</ref></td>
     <td>per <ref>SVG</ref></td>
     <td><code>SVGSVGElement</code></td>
    </tr>

    <tr>
     <th><code>table</code></th>
     <td>Table</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><code>caption</code>*;
         <code>colgroup</code>*;
         <code>thead</code>*;
         <code>tbody</code>*;
         <code>tfoot</code>*;
         <code>tr</code>*;
         <span>script-supporting elements</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLTableElement</code></td>
    </tr>

    <tr>
     <th><code>tbody</code></th>
     <td>Group of rows in a table</td>
     <td>none</td>
     <td><code>table</code></td>
     <td><code>tr</code>;
         <span>script-supporting elements</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLTableSectionElement</code></td>
    </tr>

    <tr>
     <th><code>td</code></th>
     <td>Table cell</td>
     <td>none</td>
     <td><code>tr</code></td>
     <td><span data-x="Flow content">flow</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-tdth-colspan">colspan</code>;
         <code data-x="attr-tdth-rowspan">rowspan</code>;
         <code data-x="attr-tdth-headers">headers</code></td>
     <td><code>HTMLTableCellElement</code></td>
    </tr>

    <tr>
     <th><code>template</code></th>
     <td>Template</td>
     <td><span data-x="Metadata content">metadata</span>;
         <span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Script-supporting elements">script-supporting</span></td>
     <td><span data-x="Metadata content">metadata</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Script-supporting elements">script-supporting</span>;
         <code>colgroup</code>*</td>
     <td>empty</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-template-shadowrootmode">shadowrootmode</code>;
         <code data-x="attr-template-shadowrootdelegatesfocus">shadowrootdelegatesfocus</code>;
         <code data-x="attr-template-shadowrootclonable">shadowrootclonable</code>;
         <code data-x="attr-template-shadowrootserializable">shadowrootserializable</code></td>
     <td><code>HTMLTemplateElement</code></td>
    </tr>

    <tr>
     <th><code>textarea</code></th>
     <td>Multiline text controls</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Interactive content">interactive</span>;
         <span data-x="category-listed">listed</span>;
         <span data-x="category-label">labelable</span>;
         <span data-x="category-submit">submittable</span>;
         <span data-x="category-reset">resettable</span>;
         <span data-x="Form-associated element">form-associated</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="text content">text</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-fe-autocomplete">autocomplete</code>;
         <code data-x="attr-textarea-cols">cols</code>;
         <code data-x="attr-fe-dirname">dirname</code>;
         <code data-x="attr-fe-disabled">disabled</code>;
         <code data-x="attr-fae-form">form</code>;
         <code data-x="attr-textarea-maxlength">maxlength</code>;
         <code data-x="attr-textarea-minlength">minlength</code>;
         <code data-x="attr-fe-name">name</code>;
         <code data-x="attr-textarea-placeholder">placeholder</code>;
         <code data-x="attr-textarea-readonly">readonly</code>;
         <code data-x="attr-textarea-required">required</code>;
         <code data-x="attr-textarea-rows">rows</code>;
         <code data-x="attr-textarea-wrap">wrap</code></td>
     <td><code>HTMLTextAreaElement</code></td>
    </tr>

    <tr>
     <th><code>tfoot</code></th>
     <td>Group of footer rows in a table</td>
     <td>none</td>
     <td><code>table</code></td>
     <td><code>tr</code>;
         <span>script-supporting elements</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLTableSectionElement</code></td>
    </tr>

    <tr>
     <th><code>th</code></th>
     <td>Table header cell</td>
     <td><span data-x="Interactive content">interactive</span>*</td>
     <td><code>tr</code></td>
     <td><span data-x="Flow content">flow</span>*</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-tdth-colspan">colspan</code>;
         <code data-x="attr-tdth-rowspan">rowspan</code>;
         <code data-x="attr-tdth-headers">headers</code>;
         <code data-x="attr-th-scope">scope</code>;
         <code data-x="attr-th-abbr">abbr</code></td>
     <td><code>HTMLTableCellElement</code></td>
    </tr>

    <tr>
     <th><code>thead</code></th>
     <td>Group of heading rows in a table</td>
     <td>none</td>
     <td><code>table</code></td>
     <td><code>tr</code>;
         <span>script-supporting elements</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLTableSectionElement</code></td>
    </tr>

    <tr>
     <th><code>time</code></th>
     <td>Machine-readable equivalent of date- or time-related data</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-time-datetime">datetime</code></td>
     <td><code>HTMLTimeElement</code></td>
    </tr>

    <tr>
     <th><code>title</code></th>
     <td>Document title</td>
     <td><span data-x="Metadata content">metadata</span></td>
     <td><code>head</code></td>
     <td><span data-x="text content">text</span>*</td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLTitleElement</code></td>
    </tr>

    <tr>
     <th><code>tr</code></th>
     <td>Table row</td>
     <td>none</td>
     <td><code>table</code>;
         <code>thead</code>;
         <code>tbody</code>;
         <code>tfoot</code></td>
     <td><code>th</code>*;
         <code>td</code>;
         <span>script-supporting elements</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLTableRowElement</code></td>
    </tr>

    <tr>
     <th><code>track</code></th>
     <td>Timed text track</td>
     <td>none</td>
     <td><code>audio</code>;
         <code>video</code></td>
     <td>empty</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-track-default">default</code>;
         <code data-x="attr-track-kind">kind</code>;
         <code data-x="attr-track-label">label</code>;
         <code data-x="attr-track-src">src</code>;
         <code data-x="attr-track-srclang">srclang</code></td>
     <td><code>HTMLTrackElement</code></td>
    </tr>

    <tr>
     <th><code>u</code></th>
     <td>Unarticulated annotation</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>ul</code></th>
     <td>List</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Palpable content">palpable</span>*</td>
     <td><span data-x="Flow content">flow</span></td>
     <td><code>li</code>;
         <span>script-supporting elements</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLUListElement</code></td>
    </tr>

    <tr>
     <th><code>var</code></th>
     <td>Variable</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><code>video</code></th>
     <td>Video player</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Embedded content">embedded</span>;
         <span data-x="Interactive content">interactive</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td><code>source</code>*;
         <code>track</code>*;
         <span>transparent</span>*</td>
     <td><span data-x="global attributes">globals</span>;
         <code data-x="attr-media-src">src</code>;
         <code data-x="attr-media-crossorigin">crossorigin</code>;
         <code data-x="attr-video-poster">poster</code>;
         <code data-x="attr-media-preload">preload</code>;
         <code data-x="attr-media-autoplay">autoplay</code>;
         <code data-x="attr-video-playsinline">playsinline</code>;
         <code data-x="attr-media-loop">loop</code>;
         <code data-x="attr-media-muted">muted</code>;
         <code data-x="attr-media-controls">controls</code>;
         <code data-x="attr-dim-width">width</code>;
         <code data-x="attr-dim-height">height</code></td>
     <td><code>HTMLVideoElement</code></td>
    </tr>

    <tr>
     <th><code>wbr</code></th>
     <td>Line breaking opportunity</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span></td>
     <td><span data-x="Phrasing content">phrasing</span></td>
     <td>empty</td>
     <td><span data-x="global attributes">globals</span></td>
     <td><code>HTMLElement</code></td>
    </tr>

    <tr>
     <th><span data-x="autonomous custom element">autonomous custom elements</span></th>
     <td>Author-defined elements</td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span>;
         <span data-x="Palpable content">palpable</span></td>
     <td><span data-x="Flow content">flow</span>;
         <span data-x="Phrasing content">phrasing</span></td>
     <td><span>transparent</span></td>
     <td><span data-x="global attributes">globals</span>; any, as decided by the element's author</td>
     <td>Supplied by the element's author (inherits from <code>HTMLElement</code>)</td>
    </tr>

  </table>

  <p class="tablenote"><small>An asterisk (*) in a cell indicates that the actual rules are more
  complicated than indicated in the table above.</small></p>

  <p class="tablenote"><small>&dagger; Categories in the "Parents" column refer to parents that list
  the given categories in their content model, not to elements that themselves are in those
  categories. For example, the <code>a</code> element's "Parents" column says "phrasing", so any
  element whose content model contains the "phrasing" category could be a parent of an
  <code>a</code> element. Since the "flow" category includes all the "phrasing" elements, that means
  the <code>th</code> element could be a parent to an <code>a</code> element.</small></p>


  <h3 class="no-num">Element content categories</h3>

  <!-- NON-NORMATIVE SECTION -->

  <!-- when updating this also check the category-list <ul>s -->

  <table>
   <caption>List of element content categories</caption>
   <thead>
    <tr>
     <th> Category
     <th> Elements
     <th> Elements with exceptions
   <tbody>
    <tr>
     <td> <span>Metadata content</span>
     <td>
      <code>base</code>;
      <code>link</code>;
      <code>meta</code>;
      <code>noscript</code>;
      <code>script</code>;
      <code>style</code>;
      <code>template</code>;
      <code>title</code>
     <td>
      &mdash;

    <tr>
     <td> <span>Flow content</span>
     <td>
      <code>a</code>;
      <code>abbr</code>;
      <code>address</code>;
      <code>article</code>;
      <code>aside</code>;
      <code>audio</code>;
      <code>b</code>;
      <code>bdi</code>;
      <code>bdo</code>;
      <code>blockquote</code>;
      <code>br</code>;
      <code>button</code>;
      <code>canvas</code>;
      <code>cite</code>;
      <code>code</code>;
      <code>data</code>;
      <code>datalist</code>;
      <code>del</code>;
      <code>details</code>;
      <code>dfn</code>;
      <code>dialog</code>;
      <code>div</code>;
      <code>dl</code>;
      <code>em</code>;
      <code>embed</code>;
      <code>fieldset</code>;
      <code>figure</code>;
      <code>footer</code>;
      <code>form</code>;
      <code>h1</code>;
      <code>h2</code>;
      <code>h3</code>;
      <code>h4</code>;
      <code>h5</code>;
      <code>h6</code>;
      <code>header</code>;
      <code>hgroup</code>;
      <code>hr</code>;
      <code>i</code>;
      <code>iframe</code>;
      <code>img</code>;
      <code>input</code>;
      <code>ins</code>;
      <code>kbd</code>;
      <code>label</code>;
      <code>map</code>;
      <code>mark</code>;
      <span>MathML <code>math</code></span>;
      <code>menu</code>;
      <code>meter</code>;
      <code>nav</code>;
      <code>noscript</code>;
      <code>object</code>;
      <code>ol</code>;
      <code>output</code>;
      <code>p</code>;
      <code>picture</code>;
      <code>pre</code>;
      <code>progress</code>;
      <code>q</code>;
      <code>ruby</code>;
      <code>s</code>;
      <code>samp</code>;
      <code>script</code>;
      <code>search</code>;
      <code>section</code>;
      <code>select</code>;
      <code>slot</code>;
      <code>small</code>;
      <code>span</code>;
      <code>strong</code>;
      <code>sub</code>;
      <code>sup</code>;
      <span>SVG <code>svg</code></span>;
      <code>table</code>;
      <code>template</code>;
      <code>textarea</code>;
      <code>time</code>;
      <code>u</code>;
      <code>ul</code>;
      <code>var</code>;
      <code>video</code>;
      <code>wbr</code>;
      <span data-x="autonomous custom element">autonomous custom elements</span>;
      <span data-x="text content">Text</span>
     <td>
      <code>area</code> (if it is a descendant of a <code>map</code> element);
      <code>link</code> (if it is <span>allowed in the body</span>);
      <code>main</code> (if it is a <span>hierarchically correct <code>main</code> element</span>);
      <code>meta</code> (if the <code data-x="attr-itemprop">itemprop</code> attribute is present)

    <tr>
     <td> <span>Sectioning content</span>
     <td>
      <code>article</code>;
      <code>aside</code>;
      <code>nav</code>;
      <code>section</code>
     <td>
      &mdash;

    <tr>
     <td> <span>Heading content</span>
     <td>
      <code>h1</code>;
      <code>h2</code>;
      <code>h3</code>;
      <code>h4</code>;
      <code>h5</code>;
      <code>h6</code>;
      <code>hgroup</code>
     <td>
      &mdash;

    <tr>
     <td> <span>Phrasing content</span>
     <td>
      <code>a</code>;
      <code>abbr</code>;
      <code>audio</code>;
      <code>b</code>;
      <code>bdi</code>;
      <code>bdo</code>;
      <code>br</code>;
      <code>button</code>;
      <code>canvas</code>;
      <code>cite</code>;
      <code>code</code>;
      <code>data</code>;
      <code>datalist</code>;
      <code>del</code>;
      <code>dfn</code>;
      <code>em</code>;
      <code>embed</code>;
      <code>i</code>;
      <code>iframe</code>;
      <code>img</code>;
      <code>input</code>;
      <code>ins</code>;
      <code>kbd</code>;
      <code>label</code>;
      <code>map</code>;
      <code>mark</code>;
      <span>MathML <code>math</code></span>;
      <code>meter</code>;
      <code>noscript</code>;
      <code>object</code>;
      <code>output</code>;
      <code>picture</code>;
      <code>progress</code>;
      <code>q</code>;
      <code>ruby</code>;
      <code>s</code>;
      <code>samp</code>;
      <code>script</code>;
      <code>select</code>;
      <code>slot</code>;
      <code>small</code>;
      <code>span</code>;
      <code>strong</code>;
      <code>sub</code>;
      <code>sup</code>;
      <span>SVG <code>svg</code></span>;
      <code>template</code>;
      <code>textarea</code>;
      <code>time</code>;
      <code>u</code>;
      <code>var</code>;
      <code>video</code>;
      <code>wbr</code>;
      <span data-x="autonomous custom element">autonomous custom elements</span>;
      <span data-x="text content">Text</span>
     <td>
      <code>area</code> (if it is a descendant of a  <code>map</code> element);
      <code>link</code> (if it is <span>allowed in the body</span>);
      <code>meta</code> (if the <code data-x="attr-itemprop">itemprop</code> attribute is present)

    <tr>
     <td> <span>Embedded content</span>
     <td>
      <code>audio</code>;
      <code>canvas</code>;
      <code>embed</code>;
      <code>iframe</code>;
      <code>img</code>;
      <span>MathML <code>math</code></span>;
      <code>object</code>;
      <code>picture</code>;
      <span>SVG <code>svg</code></span>;
      <code>video</code>
     <td>
      &mdash;

    <tr>
     <td> <span>Interactive content</span>
     <td>
      <code>button</code>;
      <code>details</code>;
      <code>embed</code>;
      <code>iframe</code>;
      <code>label</code>;
      <code>select</code>;
      <code>textarea</code>
     <td>
      <code>a</code> (if the <code data-x="attr-hyperlink-href">href</code> attribute is present);
      <code>audio</code> (if the <code data-x="attr-media-controls">controls</code> attribute is present);
      <code>img</code> (if the <code data-x="attr-hyperlink-usemap">usemap</code> attribute is present);
      <code>input</code> (if the <code data-x="attr-input-type">type</code> attribute is <em>not</em> in the <span data-x="attr-input-type-hidden">Hidden</span> state);
      <code>video</code> (if the <code data-x="attr-media-controls">controls</code> attribute is present)

    <tr>
     <td> <span data-x="form-associated element">Form-associated elements</span>
     <td>
      <code>button</code>;
      <code>fieldset</code>;
      <code>input</code>;
      <code>label</code>;
      <code>object</code>;
      <code>output</code>;
      <code>select</code>;
      <code>textarea</code>;
      <code>img</code>;
      <span data-x="form-associated custom element">form-associated custom elements</span>
     <td>
      &mdash;

    <tr>
     <td> <span data-x="category-listed">Listed elements</span>
     <td>
      <code>button</code>;
      <code>fieldset</code>;
      <code>input</code>;
      <code>object</code>;
      <code>output</code>;
      <code>select</code>;
      <code>textarea</code>;
      <span data-x="form-associated custom element">form-associated custom elements</span>
     <td>
      &mdash;

    <tr>
     <td> <span data-x="category-submit">Submittable elements</span>
     <td>
      <code>button</code>;
      <code>input</code>;
      <code>select</code>;
      <code>textarea</code>;
      <span data-x="form-associated custom element">form-associated custom elements</span>
     <td>
      &mdash;

    <tr>
     <td> <span data-x="category-reset">Resettable elements</span>
     <td>
      <code>input</code>;
      <code>output</code>;
      <code>select</code>;
      <code>textarea</code>;
      <span data-x="form-associated custom element">form-associated custom elements</span>
     <td>
      &mdash;

    <tr>
     <td> <span data-x="category-autocapitalize-and-autocorrect">Autocapitalize-and-autocorrect
     inheriting elements</span>
     <td>
      <code>button</code>;
      <code>fieldset</code>;
      <code>input</code>;
      <code>output</code>;
      <code>select</code>;
      <code>textarea</code>
     <td>
      &mdash;

    <tr>
     <td> <span data-x="category-label">Labelable elements</span>
     <td>
      <code>button</code>;
      <code>input</code>;
      <code>meter</code>;
      <code>output</code>;
      <code>progress</code>;
      <code>select</code>;
      <code>textarea</code>;
      <span data-x="form-associated custom element">form-associated custom elements</span>
     <td>
      &mdash;

    <tr>
     <td> <span>Palpable content</span>
     <td>
      <code>a</code>;
      <code>abbr</code>;
      <code>address</code>;
      <code>article</code>;
      <code>aside</code>;
      <code>b</code>;
      <code>bdi</code>;
      <code>bdo</code>;
      <code>blockquote</code>;
      <code>button</code>;
      <code>canvas</code>;
      <code>cite</code>;
      <code>code</code>;
      <code>data</code>;
      <code>del</code>;
      <code>details</code>;
      <code>dfn</code>;
      <code>div</code>;
      <code>em</code>;
      <code>embed</code>;
      <code>fieldset</code>;
      <code>figure</code>;
      <code>footer</code>;
      <code>form</code>;
      <code>h1</code>;
      <code>h2</code>;
      <code>h3</code>;
      <code>h4</code>;
      <code>h5</code>;
      <code>h6</code>;
      <code>header</code>;
      <code>hgroup</code>;
      <code>i</code>;
      <code>iframe</code>;
      <code>img</code>;
      <code>ins</code>;
      <code>kbd</code>;
      <code>label</code>;
      <code>main</code>;
      <code>map</code>;
      <code>mark</code>;
      <span>MathML <code>math</code></span>;
      <code>meter</code>;
      <code>nav</code>;
      <code>object</code>;
      <code>output</code>;
      <code>p</code>;
      <code>picture</code>;
      <code>pre</code>;
      <code>progress</code>;
      <code>q</code>;
      <code>ruby</code>;
      <code>s</code>;
      <code>samp</code>;
      <code>search</code>;
      <code>section</code>;
      <code>select</code>;
      <code>small</code>;
      <code>span</code>;
      <code>strong</code>;
      <code>sub</code>;
      <code>sup</code>;
      <span>SVG <code>svg</code></span>;
      <code>table</code>;
      <code>textarea</code>;
      <code>time</code>;
      <code>u</code>;
      <code>var</code>;
      <code>video</code>;
      <span data-x="autonomous custom element">autonomous custom elements</span>
     <td>
      <code>audio</code> (if the <code data-x="attr-media-controls">controls</code> attribute is present);
      <code>dl</code> (if the element's children include at least one name-value group);
      <code>input</code> (if the <code data-x="attr-input-type">type</code> attribute is <em>not</em> in the <span data-x="attr-input-type-hidden">Hidden</span> state);
      <code>menu</code> (if the element's children include at least one <code>li</code> element);
      <code>ol</code> (if the element's children include at least one <code>li</code> element);
      <code>ul</code> (if the element's children include at least one <code>li</code> element);
      <span data-x="text content">Text</span> that is not <span>inter-element whitespace</span>

    <tr>
     <td> <span>Script-supporting elements</span>
     <td>
      <code>script</code>;
      <code>template</code>
     <td>
      &mdash;

  </table>


  <h3 class="no-num">Attributes</h3>

  <!-- NON-NORMATIVE SECTION -->

  <table id="attributes-1">
   <caption>List of attributes (excluding event handler content attributes)</caption>
   <thead>
    <tr>
     <th> Attribute
     <th> Element(s)
     <th> Description
     <th> Value
   <tbody>
    <tr>
     <th> <code data-x="">abbr</code>
     <td> <code data-x="attr-th-abbr">th</code>
     <td> Alternative label to use for the header cell when referencing the cell in other contexts
     <td> <a href="#attribute-text">Text</a>*
    <tr>
     <th> <code data-x="">accept</code>
     <td> <code data-x="attr-input-accept">input</code>
     <td> Hint for expected file type in <span data-x="attr-input-type-file">file upload controls</span>
     <td> <span>Set of comma-separated tokens</span>* consisting of <span data-x="valid MIME type string with no parameters">valid MIME type strings with no parameters</span> or <code data-x="">audio/*</code>, <code data-x="">video/*</code>, or <code data-x="">image/*</code>
    <tr>
     <th> <code data-x="">accept-charset</code>
     <td> <code data-x="attr-form-accept-charset">form</code>
     <td> Character encodings to use for <span>form submission</span>
     <td> <span>ASCII case-insensitive</span> match for "<code data-x="">UTF-8</code>"
    <tr>
     <th> <code data-x="">accesskey</code>
     <td> <span data-x="attr-accesskey">HTML elements</span>
     <td> Keyboard shortcut to activate or focus element
     <td> <span>Ordered set of unique space-separated tokens</span>, none of which are <span>identical to</span> another, each consisting of one code point in length
    <tr>
     <th> <code data-x="">action</code>
     <td> <code data-x="attr-fs-action">form</code>
     <td> <span>URL</span> to use for <span>form submission</span>
     <td> <span>Valid non-empty URL potentially surrounded by spaces</span>
    <tr>
     <th> <code data-x="">allow</code>
     <td> <code data-x="attr-iframe-allow">iframe</code>
     <td> <span data-x="concept-permissions-policy">Permissions policy</span> to be applied to the <code>iframe</code>'s contents
     <td> <span data-x="concept-serialized-permissions-policy">Serialized permissions policy</span>
    <tr>
     <th> <code data-x="">allowfullscreen</code>
     <td> <code data-x="attr-iframe-allowfullscreen">iframe</code>
     <td> Whether to allow the <code>iframe</code>'s contents to use <code data-x="dom-element-requestFullscreen">requestFullscreen()</code>
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">alpha</code>
     <td> <code data-x="attr-input-alpha">input</code>
     <td> Allow the color's alpha component to be set
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">alt</code>
     <td> <code data-x="attr-area-alt">area</code>;
          <code data-x="attr-img-alt">img</code>;
          <code data-x="attr-input-alt">input</code>
     <td> Replacement text for use when images are not available
     <td> <a href="#attribute-text">Text</a>*
    <tr>
     <th> <code data-x="">as</code>
     <td> <code data-x="attr-link-as">link</code>
     <td> <span data-x="concept-potential-destination">Potential destination</span> for a preload request (for <code data-x="attr-link-rel">rel</code>="<code data-x="rel-preload">preload</code>" and <code data-x="attr-link-rel">rel</code>="<code data-x="rel-modulepreload">modulepreload</code>")
     <td> <span data-x="concept-potential-destination">Potential destination</span>, for <code data-x="attr-link-rel">rel</code>="<code data-x="rel-preload">preload</code>"; <span data-x="concept-script-like-destination">script-like destination</span>, for <code data-x="attr-link-rel">rel</code>="<code data-x="rel-modulepreload">modulepreload</code>"
    <tr>
     <th> <code data-x="">async</code>
     <td> <code data-x="attr-script-async">script</code>
     <td> Execute script when available, without blocking while fetching
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">autocapitalize</code>
     <td> <span data-x="attr-autocapitalize">HTML elements</span>
     <td> Recommended autocapitalization behavior (for supported input methods)
     <td> "<code data-x="attr-autocapitalize-on">on</code>";
          "<code data-x="attr-autocapitalize-off">off</code>";
          "<code data-x="attr-autocapitalize-none">none</code>";
          "<code data-x="attr-autocapitalize-sentences">sentences</code>";
          "<code data-x="attr-autocapitalize-words">words</code>";
          "<code data-x="attr-autocapitalize-characters">characters</code>"
    <tr>
     <th> <code data-x="">autocomplete</code>
     <td> <code data-x="attr-form-autocomplete">form</code>
     <td> Default setting for autofill feature for controls in the form
     <td> "<code data-x="">on</code>"; "<code data-x="">off</code>"
    <tr>
     <th> <code data-x="">autocomplete</code>
     <td> <code data-x="attr-fe-autocomplete">input</code>;
          <code data-x="attr-fe-autocomplete">select</code>;
          <code data-x="attr-fe-autocomplete">textarea</code>
     <td> Hint for form autofill feature
     <td> <span data-x="autofill field">Autofill field</span> name and related tokens*
    <tr>
     <th> <code data-x="">autocorrect</code>
     <td> <span data-x="attr-autocorrect">HTML elements</span>
     <td> Recommended autocorrection behavior (for supported input methods)
     <td> "<code data-x="attr-autocorrect-on">on</code>";
          "<code data-x="attr-autocorrect-off">off</code>"
    <tr>
     <th> <code data-x="">autofocus</code>
     <td> <span data-x="attr-fe-autofocus">HTML elements</span>
     <td> Automatically focus the element when the page is loaded
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">autoplay</code>
     <td> <code data-x="attr-media-autoplay">audio</code>;
          <code data-x="attr-media-autoplay">video</code>
     <td> Hint that the <span>media resource</span> can be started automatically when the page is loaded
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">blocking</code>
     <td> <code data-x="attr-link-blocking">link</code>;
          <code data-x="attr-script-blocking">script</code>;
          <code data-x="attr-style-blocking">style</code>
     <td> Whether the element is <span>potentially render-blocking</span>
     <td> <span>Unordered set of unique space-separated tokens</span>*
    <tr>
     <th> <code data-x="">charset</code>
     <td> <code data-x="attr-meta-charset">meta</code>
     <td> <span>Character encoding declaration</span>
     <td> "<code data-x="">utf-8</code>"
    <tr>
     <th> <code data-x="">checked</code>
     <td> <code data-x="attr-input-checked">input</code>
     <td> Whether the control is checked
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">cite</code>
     <td> <code data-x="attr-blockquote-cite">blockquote</code>;
          <code data-x="attr-mod-cite">del</code>;
          <code data-x="attr-mod-cite">ins</code>;
          <code data-x="attr-q-cite">q</code>
     <td> Link to the source of the quotation or more information about the edit
     <td> <span>Valid URL potentially surrounded by spaces</span>
    <tr>
     <th> <code data-x="">class</code>
     <td> <span data-x="attr-class">HTML elements</span>
     <td> Classes to which the element belongs
     <td> <span>Set of space-separated tokens</span>
    <tr>
     <th> <code data-x="">closedby</code>
     <td> <code data-x="attr-dialog-closedby">dialog</code>
     <td> Which user actions will close the dialog
     <td> "<code data-x="attr-dialog-closedby-any">any</code>";
          "<code data-x="attr-dialog-closedby-closerequest">closerequest</code>";
          "<code data-x="attr-dialog-closedby-none">none</code>";
    <tr>
     <th> <code data-x="">color</code>
     <td> <code data-x="attr-link-color">link</code>
     <td> Color to use when customizing a site's icon (for <code data-x="attr-link-rel">rel</code>="<code data-x="">mask-icon</code>")
     <td> CSS <span>&lt;color></span>
    <tr>
     <th> <code data-x="">colorspace</code>
     <td> <code data-x="attr-input-colorspace">input</code>
     <td> The color space of the serialized color
     <td> "<code data-x="attr-input-colorspace-limited-srgb">limited-srgb</code>";
          "<code data-x="attr-input-colorspace-display-p3">display-p3</code>"
    <tr>
     <th> <code data-x="">cols</code>
     <td> <code data-x="attr-textarea-cols">textarea</code>
     <td> Maximum number of characters per line
     <td> <span>Valid non-negative integer</span> greater than zero
    <tr>
     <th> <code data-x="">colspan</code>
     <td> <code data-x="attr-tdth-colspan">td</code>;
          <code data-x="attr-tdth-colspan">th</code>
     <td> Number of columns that the cell is to span
     <td> <span>Valid non-negative integer</span> greater than zero
    <tr>
     <th> <code data-x="">command</code>
     <td> <code data-x="attr-button-command">button</code>
     <td> Indicates to the targeted element which action to take.
     <td> "<code data-x="attr-button-command-toggle-popover">toggle-popover</code>";
          "<code data-x="attr-button-command-show-popover">show-popover</code>";
          "<code data-x="attr-button-command-hide-popover">hide-popover</code>";
          "<code data-x="attr-button-command-close">close</code>";
          "<code data-x="attr-button-command-show-modal">show-modal</code>";
          a <span data-x="attr-button-command-custom">custom command keyword</span>
    <tr>
     <th> <code data-x="">commandfor</code>
     <td> <code data-x="attr-button-commandfor">button</code>
     <td> Targets another element to be invoked.
     <td> <span data-x="concept-id">ID</span>*
    <tr>
     <th> <code data-x="">content</code>
     <td> <code data-x="attr-meta-content">meta</code>
     <td> Value of the element
     <td> <a href="#attribute-text">Text</a>*
    <tr>
     <th> <code data-x="">contenteditable</code>
     <td> <span data-x="attr-contenteditable">HTML elements</span>
     <td> Whether the element is editable
     <td> "<code data-x="">true</code>"; "<code data-x="">plaintext-only</code>";
          "<code data-x="">false</code>"
    <tr>
     <th> <code data-x="">controls</code>
     <td> <code data-x="attr-media-controls">audio</code>;
          <code data-x="attr-media-controls">video</code>
     <td> Show user agent controls
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">coords</code>
     <td> <code data-x="attr-area-coords">area</code>
     <td> Coordinates for the shape to be created in an <span>image map</span>
     <td> <span>Valid list of floating-point numbers</span>*
    <tr>
     <th> <code data-x="">crossorigin</code>
     <td> <code data-x="attr-media-crossorigin">audio</code>;
          <code data-x="attr-img-crossorigin">img</code>;
          <code data-x="attr-link-crossorigin">link</code>;
          <code data-x="attr-script-crossorigin">script</code>;
          <code data-x="attr-media-crossorigin">video</code>
     <td> How the element handles crossorigin requests
     <td> "<code data-x="attr-crossorigin-anonymous-keyword">anonymous</code>"; "<code data-x="attr-crossorigin-use-credentials-keyword">use-credentials</code>"
    <tr>
     <th> <code data-x="">data</code>
     <td> <code data-x="attr-object-data">object</code>
     <td> Address of the resource
     <td> <span>Valid non-empty URL potentially surrounded by spaces</span>
    <tr>
     <th> <code data-x="">datetime</code>
     <td> <code data-x="attr-mod-datetime">del</code>;
          <code data-x="attr-mod-datetime">ins</code>
     <td> Date and (optionally) time of the change
     <td> <span>Valid date string with optional time</span>
    <tr>
     <th> <code data-x="">datetime</code>
     <td> <code data-x="attr-time-datetime">time</code>
     <td> Machine-readable value
     <td> <span>Valid month string</span>,
          <span>valid date string</span>,
          <span>valid yearless date string</span>,
          <span>valid time string</span>,
          <span>valid local date and time string</span>,
          <span>valid time-zone offset string</span>,
          <span>valid global date and time string</span>,
          <span>valid week string</span>,
          <span>valid non-negative integer</span>, or
          <span>valid duration string</span>
    <tr>
     <th> <code data-x="">decoding</code>
     <td> <code data-x="attr-img-decoding">img</code>
     <td> Decoding hint to use when processing this image for presentation
     <td> "<code data-x="attr-img-decoding-sync">sync</code>";
          "<code data-x="attr-img-decoding-async">async</code>";
          "<code data-x="attr-img-decoding-auto">auto</code>"
    <tr>
     <th> <code data-x="">default</code>
     <td> <code data-x="attr-track-default">track</code>
     <td> Enable the track if no other <span>text track</span> is more suitable
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">defer</code>
     <td> <code data-x="attr-script-defer">script</code>
     <td> Defer script execution
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">dir</code>
     <td> <span data-x="attr-dir">HTML elements</span>
     <td> <span data-x="the directionality">The text directionality</span> of the element
     <td> "<code data-x="attr-dir-ltr">ltr</code>"; "<code data-x="attr-dir-rtl">rtl</code>"; "<code data-x="attr-dir-auto">auto</code>"
    <tr>
     <th> <code data-x="">dir</code>
     <td> <code>bdo</code>
     <td> <span data-x="the directionality">The text directionality</span> of the element
     <td> "<code data-x="attr-dir-ltr">ltr</code>"; "<code data-x="attr-dir-rtl">rtl</code>"
    <tr>
     <th> <code data-x="">dirname</code>
     <td> <code data-x="attr-fe-dirname">input</code>;
          <code data-x="attr-fe-dirname">textarea</code>
     <td> Name of form control to use for sending the element's <span data-x="the directionality">directionality</span> in <span>form submission</span>
     <td> <a href="#attribute-text">Text</a>*
    <tr>
     <th> <code data-x="">disabled</code>
     <td> <code data-x="attr-fe-disabled">button</code>;
          <code data-x="attr-fe-disabled">input</code>;
          <code data-x="attr-optgroup-disabled">optgroup</code>;
          <code data-x="attr-option-disabled">option</code>;
          <code data-x="attr-fe-disabled">select</code>;
          <code data-x="attr-fe-disabled">textarea</code>;
          <span data-x="attr-fe-disabled">form-associated custom elements</span>
     <td> Whether the form control is disabled
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">disabled</code>
     <td> <code data-x="attr-fieldset-disabled">fieldset</code>
     <td> Whether the descendant form controls, except any inside <code>legend</code>, are disabled
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">disabled</code>
     <td> <code data-x="attr-link-disabled">link</code>
     <td> Whether the link is disabled
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">download</code>
     <td> <code data-x="attr-hyperlink-download">a</code>;
          <code data-x="attr-hyperlink-download">area</code>
     <td> Whether to download the resource instead of navigating to it, and its filename if so
     <td> Text
    <tr>
     <th> <code data-x="">draggable</code>
     <td> <span data-x="attr-draggable">HTML elements</span>
     <td> Whether the element is draggable
     <td> "<code data-x="">true</code>"; "<code data-x="">false</code>"
    <tr>
     <th> <code data-x="">enctype</code>
     <td> <code data-x="attr-fs-enctype">form</code>
     <td> <span>Entry list</span> encoding type to use for <span>form submission</span>
     <td> "<code data-x="attr-fs-enctype-urlencoded">application/x-www-form-urlencoded</code>"; "<code data-x="attr-fs-enctype-formdata">multipart/form-data</code>"; "<code data-x="attr-fs-enctype-text">text/plain</code>"
    <tr>
     <th> <code data-x="">enterkeyhint</code>
     <td> <span data-x="attr-enterkeyhint">HTML elements</span>
     <td> Hint for selecting an enter key action
     <td> "<code data-x="attr-enterkeyhint-keyword-enter">enter</code>";
          "<code data-x="attr-enterkeyhint-keyword-done">done</code>";
          "<code data-x="attr-enterkeyhint-keyword-go">go</code>";
          "<code data-x="attr-enterkeyhint-keyword-next">next</code>";
          "<code data-x="attr-enterkeyhint-keyword-previous">previous</code>";
          "<code data-x="attr-enterkeyhint-keyword-search">search</code>";
          "<code data-x="attr-enterkeyhint-keyword-send">send</code>"
    <tr>
     <th> <code data-x="">fetchpriority</code>
     <td> <code data-x="attr-img-fetchpriority">img</code>;
          <code data-x="attr-link-fetchpriority">link</code>;
          <code data-x="attr-script-fetchpriority">script</code>
     <td> Sets the <span data-x="concept-request-priority">priority</span> for <span data-x="concept-fetch">fetches</span> initiated by the element
     <td> "<code data-x="attr-fetchpriority-auto">auto</code>";
          "<code data-x="attr-fetchpriority-high">high</code>";
          "<code data-x="attr-fetchpriority-low">low</code>"
    <tr>
     <th> <code data-x="">for</code>
     <td> <code data-x="attr-label-for">label</code>
     <td> Associate the label with form control
     <td> <span data-x="concept-id">ID</span>*
    <tr>
     <th> <code data-x="">for</code>
     <td> <code data-x="attr-output-for">output</code>
     <td> Specifies controls from which the output was calculated
     <td> <span>Unordered set of unique space-separated tokens</span> consisting of IDs*
    <tr>
     <th> <code data-x="">form</code>
     <td> <code data-x="attr-fae-form">button</code>;
          <code data-x="attr-fae-form">fieldset</code>;
          <code data-x="attr-fae-form">input</code>;
          <code data-x="attr-fae-form">object</code>;
          <code data-x="attr-fae-form">output</code>;
          <code data-x="attr-fae-form">select</code>;
          <code data-x="attr-fae-form">textarea</code>;
          <span data-x="attr-fae-form">form-associated custom elements</span>
     <td> Associates the element with a <code>form</code> element
     <td> <span data-x="concept-id">ID</span>*
    <tr>
     <th> <code data-x="">formaction</code>
     <td> <code data-x="attr-fs-formaction">button</code>;
          <code data-x="attr-fs-formaction">input</code>
     <td> <span>URL</span> to use for <span>form submission</span>
     <td> <span>Valid non-empty URL potentially surrounded by spaces</span>
    <tr>
     <th> <code data-x="">formenctype</code>
     <td> <code data-x="attr-fs-formenctype">button</code>;
          <code data-x="attr-fs-formenctype">input</code>
     <td> <span>Entry list</span> encoding type to use for <span>form submission</span>
     <td> "<code data-x="attr-fs-enctype-urlencoded">application/x-www-form-urlencoded</code>"; "<code data-x="attr-fs-enctype-formdata">multipart/form-data</code>"; "<code data-x="attr-fs-enctype-text">text/plain</code>"
    <tr>
     <th> <code data-x="">formmethod</code>
     <td> <code data-x="attr-fs-formmethod">button</code>;
          <code data-x="attr-fs-formmethod">input</code>
     <td> Variant to use for <span>form submission</span>
     <td> "<code data-x="">GET</code>";
          "<code data-x="">POST</code>";
          "<code data-x="">dialog</code>"
    <tr>
     <th> <code data-x="">formnovalidate</code>
     <td> <code data-x="attr-fs-formnovalidate">button</code>;
          <code data-x="attr-fs-formnovalidate">input</code>
     <td> Bypass form control validation for <span>form submission</span>
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">formtarget</code>
     <td> <code data-x="attr-fs-formtarget">button</code>;
          <code data-x="attr-fs-formtarget">input</code>
     <td> <span>Navigable</span> for <span>form submission</span>
     <td> <span>Valid navigable target name or keyword</span>
    <tr>
     <th> <code data-x="">headers</code>
     <td> <code data-x="attr-tdth-headers">td</code>;
          <code data-x="attr-tdth-headers">th</code>
     <td> The header cells for this cell
     <td> <span>Unordered set of unique space-separated tokens</span> consisting of IDs*
    <tr>
     <th> <code data-x="">height</code>
     <td> <code data-x="attr-canvas-height">canvas</code>;
          <code data-x="attr-dim-height">embed</code>;
          <code data-x="attr-dim-height">iframe</code>;
          <code data-x="attr-dim-height">img</code>;
          <code data-x="attr-dim-height">input</code>;
          <code data-x="attr-dim-height">object</code>;
          <code data-x="attr-dim-height">source</code> (in <code>picture</code>);
          <code data-x="attr-dim-height">video</code>
     <td> Vertical dimension
     <td> <span>Valid non-negative integer</span>
    <tr>
     <th> <code data-x="">hidden</code>
     <td> <span data-x="attr-hidden">HTML elements</span>
     <td> Whether the element is relevant
     <td> "<code data-x="attr-hidden-until-found">until-found</code>";
          "<code data-x="attr-hidden">hidden</code>";
          the empty string
    <tr>
     <th> <code data-x="">high</code>
     <td> <code data-x="attr-meter-high">meter</code>
     <td> Low limit of high range
     <td> <span>Valid floating-point number</span>*
    <tr>
     <th> <code data-x="">href</code>
     <td> <code data-x="attr-hyperlink-href">a</code>;
          <code data-x="attr-hyperlink-href">area</code>
     <td> Address of the <span>hyperlink</span>
     <td> <span>Valid URL potentially surrounded by spaces</span>
    <tr>
     <th> <code data-x="">href</code>
     <td> <code data-x="attr-link-href">link</code>
     <td> Address of the <span>hyperlink</span>
     <td> <span>Valid non-empty URL potentially surrounded by spaces</span>
    <tr>
     <th> <code data-x="">href</code>
     <td> <code data-x="attr-base-href">base</code>
     <td> <span>Document base URL</span>
     <td> <span>Valid URL potentially surrounded by spaces</span>
    <tr>
     <th> <code data-x="">hreflang</code>
     <td> <code data-x="attr-hyperlink-hreflang">a</code>;
          <code data-x="attr-link-hreflang">link</code>
     <td> Language of the linked resource
     <td> Valid BCP 47 language tag
    <tr>
     <th> <code data-x="">http-equiv</code>
     <td> <code data-x="attr-meta-http-equiv">meta</code>
     <td> Pragma directive
     <td> "<code data-x="attr-meta-http-equiv-keyword-content-type">content-type</code>";
          "<code data-x="attr-meta-http-equiv-keyword-default-style">default-style</code>";
          "<code data-x="attr-meta-http-equiv-keyword-refresh">refresh</code>";
          "<code data-x="attr-meta-http-equiv-keyword-x-ua-compatible">x-ua-compatible</code>";
          "<code data-x="attr-meta-http-equiv-keyword-content-security-policy">content-security-policy</code>"
    <tr>
     <th> <code data-x="">id</code>
     <td> <span data-x="attr-id">HTML elements</span>
     <td> The element's <span data-x="concept-id">ID</span>
     <td> <a href="#attribute-text">Text</a>*
    <tr>
     <th> <code data-x="">imagesizes</code>
     <td> <code data-x="attr-link-imagesizes">link</code>
     <td> Image sizes for different page layouts (for <code data-x="attr-link-rel">rel</code>="<code data-x="rel-preload">preload</code>")
     <td> <span>Valid source size list</span>
    <tr>
     <th> <code data-x="">imagesrcset</code>
     <td> <code data-x="attr-link-imagesrcset">link</code>
     <td> Images to use in different situations, e.g., high-resolution displays, small monitors, etc. (for <code data-x="attr-link-rel">rel</code>="<code data-x="rel-preload">preload</code>")
     <td> Comma-separated list of <span data-x="image candidate string">image candidate strings</span>
    <tr>
     <th> <code data-x="">inert</code>
     <td> <span data-x="attr-inert">HTML elements</span>
     <td> Whether the element is <span>inert</span>.
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">inputmode</code>
     <td> <span data-x="attr-inputmode">HTML elements</span>
     <td> Hint for selecting an input modality
     <td> "<code data-x="attr-inputmode-keyword-none">none</code>";
          "<code data-x="attr-inputmode-keyword-text">text</code>";
          "<code data-x="attr-inputmode-keyword-tel">tel</code>";
          "<code data-x="attr-inputmode-keyword-email">email</code>";
          "<code data-x="attr-inputmode-keyword-url">url</code>";
          "<code data-x="attr-inputmode-keyword-numeric">numeric</code>";
          "<code data-x="attr-inputmode-keyword-decimal">decimal</code>";
          "<code data-x="attr-inputmode-keyword-search">search</code>"
    <tr>
     <th> <code data-x="">integrity</code>
     <td> <code data-x="attr-link-integrity">link</code>;
          <code data-x="attr-script-integrity">script</code>
     <td> Integrity metadata used in <cite>Subresource Integrity</cite> checks <ref>SRI</ref>
     <td> <a href="#attribute-text">Text</a>
    <tr>
     <th> <code data-x="">is</code>
     <td> <span data-x="attr-is">HTML elements</span>
     <td> Creates a <span>customized built-in element</span>
     <td> <span>Valid custom element name</span> of a defined <span>customized built-in element</span>
    <tr>
     <th> <code data-x="">ismap</code>
     <td> <code data-x="attr-img-ismap">img</code>
     <td> Whether the image is a server-side image map
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">itemid</code>
     <td> <span data-x="attr-itemid">HTML elements</span>
     <td> <span>Global identifier</span> for a microdata item
     <td> <span>Valid URL potentially surrounded by spaces</span>
    <tr>
     <th> <code data-x="">itemprop</code>
     <td> <span data-x="attr-itemprop">HTML elements</span>
     <td> <span>Property names</span> of a microdata item
     <td> <span>Unordered set of unique space-separated tokens</span> consisting of <span data-x="absolute URL">valid absolute URLs</span>, <span data-x="defined property name">defined property names</span>, or text*
    <tr>
     <th> <code data-x="">itemref</code>
     <td> <span data-x="attr-itemref">HTML elements</span>
     <td> <span data-x="referenced">Referenced</span> elements
     <td> <span>Unordered set of unique space-separated tokens</span> consisting of IDs*
    <tr>
     <th> <code data-x="">itemscope</code>
     <td> <span data-x="attr-itemscope">HTML elements</span>
     <td> Introduces a microdata item
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">itemtype</code>
     <td> <span data-x="attr-itemtype">HTML elements</span>
     <td> <span>Item types</span> of a microdata item
     <td> <span>Unordered set of unique space-separated tokens</span> consisting of <span data-x="absolute URL">valid absolute URLs</span>*
    <tr>
     <th> <code data-x="">kind</code>
     <td> <code data-x="attr-track-kind">track</code>
     <td> The type of text track
     <td> "<code data-x="attr-track-kind-keyword-subtitles">subtitles</code>";
          "<code data-x="attr-track-kind-keyword-captions">captions</code>";
          "<code data-x="attr-track-kind-keyword-descriptions">descriptions</code>";
          "<code data-x="attr-track-kind-keyword-chapters">chapters</code>";
          "<code data-x="attr-track-kind-keyword-metadata">metadata</code>"
    <tr>
     <th> <code data-x="">label</code>
     <td> <code data-x="attr-optgroup-label">optgroup</code>;
          <code data-x="attr-option-label">option</code>;
          <code data-x="attr-track-label">track</code>
     <td> User-visible label
     <td> <a href="#attribute-text">Text</a>
    <tr>
     <th> <code data-x="">lang</code>
     <td> <span data-x="attr-lang">HTML elements</span>
     <td> <span>Language</span> of the element
     <td> Valid BCP 47 language tag or the empty string
    <tr>
     <th> <code data-x="">list</code>
     <td> <code data-x="attr-input-list">input</code>
     <td> List of autocomplete options
     <td> <span data-x="concept-id">ID</span>*
    <tr>
     <th> <code data-x="">loading</code>
     <td> <code data-x="attr-iframe-loading">iframe</code>;
          <code data-x="attr-img-loading">img</code>
     <td> Used when determining loading deferral
     <td> "<code data-x="attr-loading-lazy">lazy</code>";
          "<code data-x="attr-loading-eager">eager</code>"
    <tr>
     <th> <code data-x="">loop</code>
     <td> <code data-x="attr-media-loop">audio</code>;
          <code data-x="attr-media-loop">video</code>
     <td> Whether to loop the <span>media resource</span>
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">low</code>
     <td> <code data-x="attr-meter-low">meter</code>
     <td> High limit of low range
     <td> <span>Valid floating-point number</span>*
    <tr>
     <th> <code data-x="">max</code>
     <td> <code data-x="attr-input-max">input</code>
     <td> Maximum value
     <td> Varies*
    <tr>
     <th> <code data-x="">max</code>
     <td> <code data-x="attr-meter-max">meter</code>;
          <code data-x="attr-progress-max">progress</code>
     <td> Upper bound of range
     <td> <span>Valid floating-point number</span>*
    <tr>
     <th> <code data-x="">maxlength</code>
     <td> <code data-x="attr-input-maxlength">input</code>;
          <code data-x="attr-textarea-maxlength">textarea</code>
     <td> Maximum <span>length</span> of value
     <td> <span>Valid non-negative integer</span>
    <tr>
     <th> <code data-x="">media</code>
     <td> <code data-x="attr-link-media">link</code>;
          <code data-x="attr-meta-media">meta</code>;
          <code data-x="attr-source-media">source</code>;
          <code data-x="attr-style-media">style</code>
     <td> Applicable media
     <td> <span>Valid media query list</span>
    <tr>
     <th> <code data-x="">method</code>
     <td> <code data-x="attr-fs-method">form</code>
     <td> Variant to use for <span>form submission</span>
     <td> "<code data-x="attr-fs-method-GET-keyword">GET</code>";
          "<code data-x="attr-fs-method-POST-keyword">POST</code>";
          "<code data-x="attr-fs-method-dialog-keyword">dialog</code>"
    <tr>
     <th> <code data-x="">min</code>
     <td> <code data-x="attr-input-min">input</code>
     <td> Minimum value
     <td> Varies*
    <tr>
     <th> <code data-x="">min</code>
     <td> <code data-x="attr-meter-min">meter</code>
     <td> Lower bound of range
     <td> <span>Valid floating-point number</span>*
    <tr>
     <th> <code data-x="">minlength</code>
     <td> <code data-x="attr-input-minlength">input</code>;
          <code data-x="attr-textarea-minlength">textarea</code>
     <td> Minimum <span>length</span> of value
     <td> <span>Valid non-negative integer</span>
    <tr>
     <th> <code data-x="">multiple</code>
     <td> <code data-x="attr-input-multiple">input</code>;
          <code data-x="attr-select-multiple">select</code>
     <td> Whether to allow multiple values
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">muted</code>
     <td> <code data-x="attr-media-muted">audio</code>;
          <code data-x="attr-media-muted">video</code>
     <td> Whether to mute the <span>media resource</span> by default
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">name</code>
     <td> <code data-x="attr-fe-name">button</code>;
          <code data-x="attr-fe-name">fieldset</code>;
          <code data-x="attr-fe-name">input</code>;
          <code data-x="attr-fe-name">output</code>;
          <code data-x="attr-fe-name">select</code>;
          <code data-x="attr-fe-name">textarea</code>;
          <span data-x="attr-fe-name">form-associated custom elements</span>
     <td> Name of the element to use for <span>form submission</span> and in the <code data-x="dom-form-elements">form.elements</code> API <!--or: Name of the element to use in the <code data-x="dom-form-elements">form.elements</code> API. -->
     <td> <a href="#attribute-text">Text</a>*
    <tr>
     <th> <code data-x="">name</code>
     <td> <code data-x="attr-details-name">details</code>
     <td> Name of group of mutually-exclusive <code>details</code> elements
     <td> <a href="#attribute-text">Text</a>*
    <tr>
     <th> <code data-x="">name</code>
     <td> <code data-x="attr-form-name">form</code>
     <td> Name of form to use in the <code data-x="dom-document-forms">document.forms</code> API
     <td> <a href="#attribute-text">Text</a>*
    <tr>
     <th> <code data-x="">name</code>
     <td> <code data-x="attr-iframe-name">iframe</code>;
          <code data-x="attr-object-name">object</code>
     <td> Name of <span>content navigable</span>
     <td> <span>Valid navigable target name or keyword</span>
    <tr>
     <th> <code data-x="">name</code>
     <td> <code data-x="attr-map-name">map</code>
     <td> Name of <span>image map</span> to <span data-x="referenced">reference</span> from the <code data-x="attr-hyperlink-usemap">usemap</code> attribute
     <td> <a href="#attribute-text">Text</a>*
    <tr>
     <th> <code data-x="">name</code>
     <td> <code data-x="attr-meta-name">meta</code>
     <td> Metadata name
     <td> <a href="#attribute-text">Text</a>*
    <tr>
     <th> <code data-x="">name</code>
     <td> <code data-x="attr-slot-name">slot</code>
     <td> Name of shadow tree slot
     <td> <a href="#attribute-text">Text</a>
    <tr>
     <th> <code data-x="">nomodule</code>
     <td> <code data-x="attr-script-nomodule">script</code>
     <td> Prevents execution in user agents that support <span data-x="module script">module scripts</span>
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">nonce</code>
     <td> <span data-x="attr-nonce">HTML elements</span>
     <td> Cryptographic nonce used in <cite>Content Security Policy</cite> checks <ref>CSP</ref>
     <td> <a href="#attribute-text">Text</a>
    <tr>
     <th> <code data-x="">novalidate</code>
     <td> <code data-x="attr-fs-novalidate">form</code>
     <td> Bypass form control validation for <span>form submission</span>
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">open</code>
     <td> <code data-x="attr-details-open">details</code>
     <td> Whether the details are visible
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">open</code>
     <td> <code data-x="attr-dialog-open">dialog</code>
     <td> Whether the dialog box is showing
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">optimum</code>
     <td> <code data-x="attr-meter-optimum">meter</code>
     <td> Optimum value in gauge
     <td> <span>Valid floating-point number</span>*
    <tr>
     <th> <code data-x="">pattern</code>
     <td> <code data-x="attr-input-pattern">input</code>
     <td> Pattern to be matched by the form control's value
     <td> Regular expression matching the JavaScript <i data-x="js-prod-Pattern">Pattern</i> production
    <tr>
     <th> <code data-x="">ping</code>
     <td> <code data-x="attr-hyperlink-ping">a</code>;
          <code data-x="attr-hyperlink-ping">area</code>
     <td> <span data-x="URL">URLs</span> to ping
     <td> <span>Set of space-separated tokens</span> consisting of <span data-x="valid non-empty URL">valid non-empty URLs</span>
    <tr>
     <th> <code data-x="">placeholder</code>
     <td> <code data-x="attr-input-placeholder">input</code>;
          <code data-x="attr-textarea-placeholder">textarea</code>
     <td> User-visible label to be placed within the form control
     <td> <a href="#attribute-text">Text</a>*
    <tr>
     <th> <code data-x="">playsinline</code>
     <td> <code data-x="attr-video-playsinline">video</code>
     <td> Encourage the user agent to display video content within the element's playback area
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">popover</code>
     <td> <span data-x="attr-popover">HTML elements</span>
     <td> Makes the element a <span data-x="attr-popover">popover</span> element
     <td> "<code data-x="attr-popover-auto">auto</code>";
          "<code data-x="attr-popover-manual">manual</code>";
    <tr>
     <th> <code data-x="">popovertarget</code>
     <td> <code data-x="attr-popovertarget">button</code>;
          <code data-x="attr-popovertarget">input</code>
     <td> Targets a popover element to toggle, show, or hide
     <td> <span data-x="concept-id">ID</span>*
    <tr>
     <th> <code data-x="">popovertargetaction</code>
     <td> <code data-x="attr-popovertargetaction">button</code>;
          <code data-x="attr-popovertargetaction">input</code>
     <td> Indicates whether a targeted popover element is to be toggled, shown, or hidden
     <td> "<code data-x="attr-popovertargetaction-toggle">toggle</code>";
          "<code data-x="attr-popovertargetaction-show">show</code>";
          "<code data-x="attr-popovertargetaction-hide">hide</code>"
    <tr>
     <th> <code data-x="">poster</code>
     <td> <code data-x="attr-video-poster">video</code>
     <td> Poster frame to show prior to video playback
     <td> <span>Valid non-empty URL potentially surrounded by spaces</span>
    <tr>
     <th> <code data-x="">preload</code>
     <td> <code data-x="attr-media-preload">audio</code>;
          <code data-x="attr-media-preload">video</code>
     <td> Hints how much buffering the <span>media resource</span> will likely need
     <td> "<code data-x="attr-media-preload-none">none</code>";
          "<code data-x="attr-media-preload-metadata">metadata</code>";
          "<code data-x="attr-media-preload-auto">auto</code>"
    <tr>
     <th> <code data-x="">readonly</code>
     <td> <code data-x="attr-input-readonly">input</code>;
          <code data-x="attr-textarea-readonly">textarea</code>
     <td> Whether to allow the value to be edited by the user
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">readonly</code>
     <td> <span data-x="attr-face-readonly">form-associated custom elements</span>
     <td> Affects <code data-x="dom-ElementInternals-willValidate">willValidate</code>, plus any behavior added by the custom element author
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">referrerpolicy</code>
     <td> <code data-x="attr-hyperlink-referrerpolicy">a</code>;
          <code data-x="attr-hyperlink-referrerpolicy">area</code>;
          <code data-x="attr-iframe-referrerpolicy">iframe</code>;
          <code data-x="attr-img-referrerpolicy">img</code>;
          <code data-x="attr-link-referrerpolicy">link</code>;
          <code data-x="attr-script-referrerpolicy">script</code>
     <td> <span>Referrer policy</span> for <span data-x="concept-fetch">fetches</span> initiated by the element
     <td> <span>Referrer policy</span>
    <tr>
     <th> <code data-x="">rel</code>
     <td> <code data-x="attr-hyperlink-rel">a</code>;
          <code data-x="attr-hyperlink-rel">area</code>
     <td> Relationship between the location in the document containing the <span>hyperlink</span> and the destination resource
     <td> <span>Unordered set of unique space-separated tokens</span>*
    <tr>
     <th> <code data-x="">rel</code>
     <td> <code data-x="attr-link-rel">link</code>
     <td> Relationship between the document containing the <span>hyperlink</span> and the destination resource
     <td> <span>Unordered set of unique space-separated tokens</span>*
    <tr>
     <th> <code data-x="">required</code>
     <td> <code data-x="attr-input-required">input</code>;
          <code data-x="attr-select-required">select</code>;
          <code data-x="attr-textarea-required">textarea</code>
     <td> Whether the control is required for <span>form submission</span>
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">reversed</code>
     <td> <code data-x="attr-ol-reversed">ol</code>
     <td> Number the list backwards
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">rows</code>
     <td> <code data-x="attr-textarea-rows">textarea</code>
     <td> Number of lines to show
     <td> <span>Valid non-negative integer</span> greater than zero
    <tr>
     <th> <code data-x="">rowspan</code>
     <td> <code data-x="attr-tdth-rowspan">td</code>;
          <code data-x="attr-tdth-rowspan">th</code>
     <td> Number of rows that the cell is to span
     <td> <span>Valid non-negative integer</span>
    <tr>
     <th> <code data-x="">sandbox</code>
     <td> <code data-x="attr-iframe-sandbox">iframe</code>
     <td> Security rules for nested content
     <td> <span>Unordered set of unique space-separated tokens</span>, <span>ASCII case-insensitive</span>, consisting of
      <ul class="brief">
       <li>"<code data-x="attr-iframe-sandbox-allow-downloads">allow-downloads</code>"</li>
       <li>"<code data-x="attr-iframe-sandbox-allow-forms">allow-forms</code>"</li>
       <li>"<code data-x="attr-iframe-sandbox-allow-modals">allow-modals</code>"</li>
       <li>"<code
       data-x="attr-iframe-sandbox-allow-orientation-lock">allow-orientation-lock</code>"</li>
       <li>"<code data-x="attr-iframe-sandbox-allow-pointer-lock">allow-pointer-lock</code>"</li>
       <li>"<code data-x="attr-iframe-sandbox-allow-popups">allow-popups</code>"</li>
       <li>"<code
       data-x="attr-iframe-sandbox-allow-popups-to-escape-sandbox">allow-popups-to-escape-sandbox</code>"</li>
       <li>"<code data-x="attr-iframe-sandbox-allow-presentation">allow-presentation</code>"</li>
       <li>"<code data-x="attr-iframe-sandbox-allow-same-origin">allow-same-origin</code>"</li>
       <li>"<code data-x="attr-iframe-sandbox-allow-scripts">allow-scripts</code>"</li>
       <li>"<code data-x="attr-iframe-sandbox-allow-top-navigation">allow-top-navigation</code>"</li>
       <li>"<code
       data-x="attr-iframe-sandbox-allow-top-navigation-by-user-activation">allow-top-navigation-by-user-activation</code>"</li>
       <li>"<code
       data-x="attr-iframe-sandbox-allow-top-navigation-to-custom-protocols">allow-top-navigation-to-custom-protocols</code>"</li>
      </ul>
    <tr>
     <th> <code data-x="">scope</code>
     <td> <code data-x="attr-th-scope">th</code>
     <td> Specifies which cells the header cell applies to
     <td> "<code data-x="attr-th-scope-row">row</code>";
          "<code data-x="attr-th-scope-col">col</code>";
          "<code data-x="attr-th-scope-rowgroup">rowgroup</code>";
          "<code data-x="attr-th-scope-colgroup">colgroup</code>"
    <tr>
     <th> <code data-x="">selected</code>
     <td> <code data-x="attr-option-selected">option</code>
     <td> Whether the option is selected by default
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">shadowrootclonable</code>
     <td> <code data-x="attr-template-shadowrootclonable">template</code>
     <td> Sets <span>clonable</span> on a declarative shadow root
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">shadowrootdelegatesfocus</code>
     <td> <code data-x="attr-template-shadowrootdelegatesfocus">template</code>
     <td> Sets <span>delegates focus</span> on a declarative shadow root
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">shadowrootmode</code>
     <td> <code data-x="attr-template-shadowrootmode">template</code>
     <td> Enables streaming declarative shadow roots
     <td> "<code data-x="">open</code>";
          "<code data-x="">closed</code>"
    <tr>
     <th> <code data-x="">shadowrootserializable</code>
     <td> <code data-x="attr-template-shadowrootserializable">template</code>
     <td> Sets <span>serializable</span> on a declarative shadow root
     <td> <span>Boolean attribute</span>
    <tr>
     <th> <code data-x="">shape</code>
     <td> <code data-x="attr-area-shape">area</code>
     <td> The kind of shape to be created in an <span>image map</span>
     <td> "<code data-x="attr-area-shape-keyword-circle">circle</code>";
          "<code data-x="attr-area-shape-keyword-default">default</code>";
          "<code data-x="attr-area-shape-keyword-poly">poly</code>";
          "<code data-x="attr-area-shape-keyword-rect">rect</code>"
    <tr>
     <th> <code data-x="">size</code>
     <td> <code data-x="attr-input-size">input</code>;
          <code data-x="attr-select-size">select</code>
     <td> Size of the control
     <td> <span>Valid non-negative integer</span> greater than zero
    <tr>
     <th> <code data-x="">sizes</code>
     <td> <code data-x="attr-link-sizes">link</code>
     <td> Sizes of the icons (for <code data-x="attr-link-rel">rel</code>="<code data-x="rel-icon">icon</code>")
     <td> <span>Unordered set of unique space-separated tokens</span>, <span>ASCII case-insensitive</span>, consisting of sizes*
    <tr>
     <th> <code data-x="">sizes</code>
     <td> <code data-x="attr-img-sizes">img</code>;
          <code data-x="attr-source-sizes">source</code>
     <td> Image sizes for different page layouts
     <td> <span>Valid source size list</span>
    <tr>
     <th> <code data-x="">slot</code>
     <td> <span data-x="attr-id">HTML elements</span>
     <td> The element's desired slot
     <td> <a href="#attribute-text">Text</a>
    <tr>
     <th> <code data-x="">span</code>
     <td> <code data-x="attr-col-span">col</code>;
          <code data-x="attr-colgroup-span">colgroup</code>
     <td> Number of columns spanned by the element
     <td> <span>Valid non-negative integer</span> greater than zero
    <tr>
     <th> <code data-x="">spellcheck</code>
     <td> <span data-x="attr-spellcheck">HTML elements</span>
     <td> Whether the element is to have its spelling and grammar checked
     <td> "<code data-x="attr-spellcheck-true">true</code>";
          "<code data-x="attr-spellcheck-false">false</code>";
          the empty string
    <tr>
     <th> <code data-x="">src</code>
     <td> <code data-x="attr-media-src">audio</code>;
          <code data-x="attr-embed-src">embed</code>;
          <code data-x="attr-iframe-src">iframe</code>;
          <code data-x="attr-img-src">img</code>;
          <code data-x="attr-input-src">input</code>;
          <code data-x="attr-script-src">script</code>;
          <code data-x="attr-source-src">source</code> (in <code>video</code> or <code>audio</code>);
          <code data-x="attr-track-src">track</code>;
          <code data-x="attr-media-src">video</code>
     <td> Address of the resource
     <td> <span>Valid non-empty URL potentially surrounded by spaces</span>
    <tr>
     <th> <code data-x="">srcdoc</code>
     <td> <code data-x="attr-iframe-srcdoc">iframe</code>
     <td> A document to render in the <code>iframe</code>
     <td> The source of <span>an <code>iframe</code> <code data-x="attr-iframe-srcdoc">srcdoc</code> document</span>*
    <tr>
     <th> <code data-x="">srclang</code>
     <td> <code data-x="attr-track-srclang">track</code>
     <td> Language of the text track
     <td> Valid BCP 47 language tag
    <tr>
     <th> <code data-x="">srcset</code>
     <td> <code data-x="attr-img-srcset">img</code>;
          <code data-x="attr-source-srcset">source</code>
     <td> Images to use in different situations, e.g., high-resolution displays, small monitors, etc.
     <td> Comma-separated list of <span data-x="image candidate string">image candidate strings</span>
    <tr>
     <th> <code data-x="">start</code>
     <td> <code data-x="attr-ol-start">ol</code>
     <td> <span data-x="concept-ol-start">Starting value</span> of the list
     <td> <span>Valid integer</span>
    <tr>
     <th> <code data-x="">step</code>
     <td> <code data-x="attr-input-step">input</code>
     <td> Granularity to be matched by the form control's value
     <td> <span>Valid floating-point number</span> greater than zero, or "<code data-x="">any</code>"
    <tr>
     <th> <code data-x="">style</code>
     <td> <span data-x="attr-style">HTML elements</span>
     <td> Presentational and formatting instructions
     <td> CSS declarations*
    <tr>
     <th> <code data-x="">tabindex</code>
     <td> <span data-x="attr-tabindex">HTML elements</span>
     <td> Whether the element is <span>focusable</span> and <span>sequentially focusable</span>, and
     the relative order of the element for the purposes of <span>sequential focus navigation</span>
     <td> <span>Valid integer</span>
    <tr>
     <th> <code data-x="">target</code>
     <td> <code data-x="attr-hyperlink-target">a</code>;
          <code data-x="attr-hyperlink-target">area</code>
     <td> <span>Navigable</span> for <span>hyperlink</span> <span data-x="navigate">navigation</span>
     <td> <span>Valid navigable target name or keyword</span>
    <tr>
     <th> <code data-x="">target</code>
     <td> <code data-x="attr-base-target">base</code>
     <td> Default <span>navigable</span> for <span>hyperlink</span> <span data-x="navigate">navigation</span> and <span>form submission</span>
     <td> <span>Valid navigable target name or keyword</span>
    <tr>
     <th> <code data-x="">target</code>
     <td> <code data-x="attr-fs-target">form</code>
     <td> <span>Navigable</span> for <span>form submission</span>
     <td> <span>Valid navigable target name or keyword</span>
    <tr>
     <th> <code data-x="">title</code>
     <td> <span data-x="attr-title">HTML elements</span>
     <td> Advisory information for the element
     <td> <a href="#attribute-text">Text</a>
    <tr>
     <th> <code data-x="">title</code>
     <td> <code data-x="attr-abbr-title">abbr</code>;
          <code data-x="attr-dfn-title">dfn</code>
     <td> Full term or expansion of abbreviation
     <td> <a href="#attribute-text">Text</a>
    <tr>
     <th> <code data-x="">title</code>
     <td> <code data-x="attr-input-title">input</code>
     <td> Description of pattern (when used with <code data-x="attr-input-pattern">pattern</code> attribute)
     <td> <a href="#attribute-text">Text</a>
    <tr>
     <th> <code data-x="">title</code>
     <td> <code data-x="attr-link-title">link</code>
     <td> Title of the link
     <td> <a href="#attribute-text">Text</a>
    <tr>
     <th> <code data-x="">title</code>
     <td> <code data-x="attr-link-title">link</code>;
          <code data-x="attr-style-title">style</code>
     <td> <span>CSS style sheet set name</span>
     <td> <a href="#attribute-text">Text</a>
    <tr>
     <th> <code data-x="">translate</code>
     <td> <span data-x="attr-translate">HTML elements</span>
     <td> Whether the element is to be translated when the page is localized
     <td> "<code data-x="">yes</code>"; "<code data-x="">no</code>"
    <tr>
     <th> <code data-x="">type</code>
     <td> <code data-x="attr-hyperlink-type">a</code>;
          <code data-x="attr-link-type">link</code>
     <td> Hint for the type of the referenced resource
     <td> <span>Valid MIME type string</span>
    <tr>
     <th> <code data-x="">type</code>
     <td> <code data-x="attr-button-type">button</code>
     <td> Type of button
     <td> "<code data-x="attr-button-type-submit">submit</code>";
          "<code data-x="attr-button-type-reset">reset</code>";
          "<code data-x="attr-button-type-button">button</code>"
    <tr>
     <th> <code data-x="">type</code>
     <td> <code data-x="attr-embed-type">embed</code>;
          <code data-x="attr-object-type">object</code>;
          <code data-x="attr-source-type">source</code>
     <td> Type of embedded resource
     <td> <span>Valid MIME type string</span>
    <tr>
     <th> <code data-x="">type</code>
     <td> <code data-x="attr-input-type">input</code>
     <td> Type of form control
     <td> <span data-x="attr-input-type"><code>input</code> type keyword</span>
    <tr>
     <th> <code data-x="">type</code>
     <td> <code data-x="attr-ol-type">ol</code>
     <td> Kind of list marker
     <td> "<code data-x="attr-ol-type-keyword-decimal">1</code>";
          "<code data-x="attr-ol-type-keyword-lower-alpha">a</code>";
          "<code data-x="attr-ol-type-keyword-upper-alpha">A</code>";
          "<code data-x="attr-ol-type-keyword-lower-roman">i</code>";
          "<code data-x="attr-ol-type-keyword-upper-roman">I</code>"
    <tr>
     <th> <code data-x="">type</code>
     <td> <code data-x="attr-script-type">script</code>
     <td> Type of script
     <td> "<code data-x="">module</code>"; a <span>valid MIME type string</span> that is not a <span>JavaScript MIME type essence match</span>
    <tr>
     <th> <code data-x="">usemap</code>
     <td> <code data-x="attr-hyperlink-usemap">img</code>
     <td> Name of <span>image map</span> to use
     <td> <span>Valid hash-name reference</span>*
    <tr>
     <th> <code data-x="">value</code>
     <td> <code data-x="attr-button-value">button</code>;
          <code data-x="attr-option-value">option</code>
     <td> Value to be used for <span>form submission</span>
     <td> <a href="#attribute-text">Text</a>
    <tr>
     <th> <code data-x="">value</code>
     <td> <code data-x="attr-data-value">data</code>
     <td> Machine-readable value
     <td> <a href="#attribute-text">Text</a>*
    <tr>
     <th> <code data-x="">value</code>
     <td> <code data-x="attr-input-value">input</code>
     <td> Value of the form control
     <td> Varies*
    <tr>
     <th> <code data-x="">value</code>
     <td> <code data-x="attr-li-value">li</code>
     <td> <span>Ordinal value</span> of the list item
     <td> <span>Valid integer</span>
    <tr>
     <th> <code data-x="">value</code>
     <td> <code data-x="attr-meter-value">meter</code>;
          <code data-x="attr-progress-value">progress</code>
     <td> Current value of the element
     <td> <span>Valid floating-point number</span>
    <tr>
     <th> <code data-x="">width</code>
     <td> <code data-x="attr-canvas-width">canvas</code>;
          <code data-x="attr-dim-width">embed</code>;
          <code data-x="attr-dim-width">iframe</code>;
          <code data-x="attr-dim-width">img</code>;
          <code data-x="attr-dim-width">input</code>;
          <code data-x="attr-dim-width">object</code>;
          <code data-x="attr-dim-width">source</code> (in <code>picture</code>);
          <code data-x="attr-dim-width">video</code>
     <td> Horizontal dimension
     <td> <span>Valid non-negative integer</span>
    <tr>
     <th> <code data-x="">wrap</code>
     <td> <code data-x="attr-textarea-wrap">textarea</code>
     <td> How the value of the form control is to be wrapped for <span>form submission</span>
     <td> "<code data-x="attr-textarea-wrap-soft">soft</code>";
          "<code data-x="attr-textarea-wrap-hard">hard</code>"
    <tr>
     <th> <code data-x="">writingsuggestions</code>
     <td> <span data-x="attr-writingsuggestions">HTML elements</span>
     <td> Whether the element can offer writing suggestions or not.
     <td> "<code data-x="attr-writingsuggestions-true">true</code>";
          "<code data-x="attr-writingsuggestions-false">false</code>";
          the empty string
  </table>

  <p class="tablenote"><small>An asterisk (*) in a cell indicates that the actual rules are more
  complicated than indicated in the table above.</small></p>

  <hr>

  <table id="ix-event-handlers">
   <caption>List of event handler content attributes</caption>
   <thead>
    <tr>
     <th> Attribute
     <th> Element(s)
     <th> Description
     <th> Value
   <tbody>

    <tr>
     <th id="ix-handler-window-onafterprint"> <code data-x="">onafterprint</code>
     <td> <code data-x="handler-window-onafterprint">body</code>
     <td> <code data-x="event-afterprint">afterprint</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onauxclick"> <code data-x="">onauxclick</code>
     <td> <span data-x="handler-onauxclick">HTML elements</span>
     <td> <code data-x="event-auxclick">auxclick</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onbeforeinput"> <code data-x="">onbeforeinput</code>
     <td> <span data-x="handler-onbeforeinput">HTML elements</span>
     <td> <code data-x="event-beforeinput">beforeinput</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onbeforematch"> <code data-x="">onbeforematch</code>
     <td> <span data-x="handler-onbeforematch">HTML elements</span>
     <td> <code data-x="event-beforematch">beforematch</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-window-onbeforeprint"> <code data-x="">onbeforeprint</code>
     <td> <code data-x="handler-window-onbeforeprint">body</code>
     <td> <code data-x="event-beforeprint">beforeprint</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-window-onbeforeunload"> <code data-x="">onbeforeunload</code>
     <td> <code data-x="handler-window-onbeforeunload">body</code>
     <td> <code data-x="event-beforeunload">beforeunload</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onbeforetoggle"> <code data-x="">onbeforetoggle</code>
     <td> <span data-x="handler-onbeforetoggle">HTML elements</span>
     <td> <code data-x="event-beforetoggle">beforetoggle</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onblur"> <code data-x="">onblur</code>
     <td> <span data-x="handler-onblur">HTML elements</span>
     <td> <code data-x="event-blur">blur</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-oncancel"> <code data-x="">oncancel</code>
     <td> <span data-x="handler-oncancel">HTML elements</span>
     <td> <code data-x="event-cancel">cancel</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-oncanplay"> <code data-x="">oncanplay</code>
     <td> <span data-x="handler-oncanplay">HTML elements</span>
     <td> <code data-x="event-media-canplay">canplay</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-oncanplaythrough"> <code data-x="">oncanplaythrough</code>
     <td> <span data-x="handler-oncanplaythrough">HTML elements</span>
     <td> <code data-x="event-media-canplaythrough">canplaythrough</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onchange"> <code data-x="">onchange</code>
     <td> <span data-x="handler-onchange">HTML elements</span>
     <td> <code data-x="event-change">change</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onclick"> <code data-x="">onclick</code>
     <td> <span data-x="handler-onclick">HTML elements</span>
     <td> <code data-x="event-click">click</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onclose"> <code data-x="">onclose</code>
     <td> <span data-x="handler-onclose">HTML elements</span>
     <td> <code data-x="event-close">close</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-oncommand"> <code data-x="">oncommand</code>
     <td> <span data-x="handler-oncommand">HTML elements</span>
     <td> <code data-x="event-command">command</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-oncontextlost"> <code data-x="">oncontextlost</code>
     <td> <span data-x="handler-oncontextlost">HTML elements</span>
     <td> <code data-x="event-contextlost">contextlost</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-oncontextmenu"> <code data-x="">oncontextmenu</code>
     <td> <span data-x="handler-oncontextmenu">HTML elements</span>
     <td> <code data-x="event-contextmenu">contextmenu</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-oncontextrestored"> <code data-x="">oncontextrestored</code>
     <td> <span data-x="handler-oncontextrestored">HTML elements</span>
     <td> <code data-x="event-contextrestored">contextrestored</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-oncopy"> <code data-x="">oncopy</code>
     <td> <span data-x="handler-oncopy">HTML elements</span>
     <td> <code data-x="event-copy">copy</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-oncuechange"> <code data-x="">oncuechange</code>
     <td> <span data-x="handler-oncuechange">HTML elements</span>
     <td> <code data-x="event-media-cuechange">cuechange</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-oncut"> <code data-x="">oncut</code>
     <td> <span data-x="handler-oncut">HTML elements</span>
     <td> <code data-x="event-cut">cut</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-ondblclick"> <code data-x="">ondblclick</code>
     <td> <span data-x="handler-ondblclick">HTML elements</span>
     <td> <code data-x="event-dblclick">dblclick</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-ondrag"> <code data-x="">ondrag</code>
     <td> <span data-x="handler-ondrag">HTML elements</span>
     <td> <code data-x="event-dnd-drag">drag</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-ondragend"> <code data-x="">ondragend</code>
     <td> <span data-x="handler-ondragend">HTML elements</span>
     <td> <code data-x="event-dnd-dragend">dragend</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-ondragenter"> <code data-x="">ondragenter</code>
     <td> <span data-x="handler-ondragenter">HTML elements</span>
     <td> <code data-x="event-dnd-dragenter">dragenter</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-ondragleave"> <code data-x="">ondragleave</code>
     <td> <span data-x="handler-ondragleave">HTML elements</span>
     <td> <code data-x="event-dnd-dragleave">dragleave</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-ondragover"> <code data-x="">ondragover</code>
     <td> <span data-x="handler-ondragover">HTML elements</span>
     <td> <code data-x="event-dnd-dragover">dragover</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-ondragstart"> <code data-x="">ondragstart</code>
     <td> <span data-x="handler-ondragstart">HTML elements</span>
     <td> <code data-x="event-dnd-dragstart">dragstart</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-ondrop"> <code data-x="">ondrop</code>
     <td> <span data-x="handler-ondrop">HTML elements</span>
     <td> <code data-x="event-dnd-drop">drop</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-ondurationchange"> <code data-x="">ondurationchange</code>
     <td> <span data-x="handler-ondurationchange">HTML elements</span>
     <td> <code data-x="event-media-durationchange">durationchange</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onemptied"> <code data-x="">onemptied</code>
     <td> <span data-x="handler-onemptied">HTML elements</span>
     <td> <code data-x="event-media-emptied">emptied</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onended"> <code data-x="">onended</code>
     <td> <span data-x="handler-onended">HTML elements</span>
     <td> <code data-x="event-media-ended">ended</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onerror"> <code data-x="">onerror</code>
     <td> <span data-x="handler-onerror">HTML elements</span>
     <td> <code data-x="event-error">error</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onfocus"> <code data-x="">onfocus</code>
     <td> <span data-x="handler-onfocus">HTML elements</span>
     <td> <code data-x="event-focus">focus</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onformdata"> <code data-x="">onformdata</code>
     <td> <span data-x="handler-onformdata">HTML elements</span>
     <td> <code data-x="event-formdata">formdata</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-window-onhashchange"> <code data-x="">onhashchange</code>
     <td> <code data-x="handler-window-onhashchange">body</code>
     <td> <code data-x="event-hashchange">hashchange</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-oninput"> <code data-x="">oninput</code>
     <td> <span data-x="handler-oninput">HTML elements</span>
     <td> <code data-x="event-input">input</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-oninvalid"> <code data-x="">oninvalid</code>
     <td> <span data-x="handler-oninvalid">HTML elements</span>
     <td> <code data-x="event-invalid">invalid</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onkeydown"> <code data-x="">onkeydown</code>
     <td> <span data-x="handler-onkeydown">HTML elements</span>
     <td> <code data-x="event-keydown">keydown</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onkeypress"> <code data-x="">onkeypress</code>
     <td> <span data-x="handler-onkeypress">HTML elements</span>
     <td> <code data-x="event-keypress">keypress</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onkeyup"> <code data-x="">onkeyup</code>
     <td> <span data-x="handler-onkeyup">HTML elements</span>
     <td> <code data-x="event-keyup">keyup</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-window-onlanguagechange"> <code data-x="">onlanguagechange</code>
     <td> <code data-x="handler-window-onlanguagechange">body</code>
     <td> <code data-x="event-languagechange">languagechange</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onload"> <code data-x="">onload</code>
     <td> <span data-x="handler-onload">HTML elements</span>
     <td> <code data-x="event-load">load</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onloadeddata"> <code data-x="">onloadeddata</code>
     <td> <span data-x="handler-onloadeddata">HTML elements</span>
     <td> <code data-x="event-media-loadeddata">loadeddata</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onloadedmetadata"> <code data-x="">onloadedmetadata</code>
     <td> <span data-x="handler-onloadedmetadata">HTML elements</span>
     <td> <code data-x="event-media-loadedmetadata">loadedmetadata</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onloadstart"> <code data-x="">onloadstart</code>
     <td> <span data-x="handler-onloadstart">HTML elements</span>
     <td> <code data-x="event-media-loadstart">loadstart</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-window-onmessage"> <code data-x="">onmessage</code>
     <td> <code data-x="handler-window-onmessage">body</code>
     <td> <code data-x="event-message">message</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-window-onmessageerror"> <code data-x="">onmessageerror</code>
     <td> <code data-x="handler-window-onmessageerror">body</code>
     <td> <code data-x="event-messageerror">messageerror</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onmousedown"> <code data-x="">onmousedown</code>
     <td> <span data-x="handler-onmousedown">HTML elements</span>
     <td> <code data-x="event-mousedown">mousedown</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onmouseenter"> <code data-x="">onmouseenter</code>
     <td> <span data-x="handler-onmouseenter">HTML elements</span>
     <td> <code data-x="event-mouseenter">mouseenter</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onmouseleave"> <code data-x="">onmouseleave</code>
     <td> <span data-x="handler-onmouseleave">HTML elements</span>
     <td> <code data-x="event-mouseleave">mouseleave</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onmousemove"> <code data-x="">onmousemove</code>
     <td> <span data-x="handler-onmousemove">HTML elements</span>
     <td> <code data-x="event-mousemove">mousemove</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onmouseout"> <code data-x="">onmouseout</code>
     <td> <span data-x="handler-onmouseout">HTML elements</span>
     <td> <code data-x="event-mouseout">mouseout</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onmouseover"> <code data-x="">onmouseover</code>
     <td> <span data-x="handler-onmouseover">HTML elements</span>
     <td> <code data-x="event-mouseover">mouseover</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onmouseup"> <code data-x="">onmouseup</code>
     <td> <span data-x="handler-onmouseup">HTML elements</span>
     <td> <code data-x="event-mouseup">mouseup</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-window-onoffline"> <code data-x="">onoffline</code>
     <td> <code data-x="handler-window-onoffline">body</code>
     <td> <code data-x="event-offline">offline</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-window-ononline"> <code data-x="">ononline</code>
     <td> <code data-x="handler-window-ononline">body</code>
     <td> <code data-x="event-online">online</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-window-onpagehide"> <code data-x="">onpagehide</code>
     <td> <code data-x="handler-window-onpagehide">body</code>
     <td> <code data-x="event-pagehide">pagehide</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-window-onpagereveal"> <code data-x="">onpagereveal</code>
     <td> <code data-x="handler-window-onpagereveal">body</code>
     <td> <code data-x="event-pagereveal">pagereveal</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-window-onpageshow"> <code data-x="">onpageshow</code>
     <td> <code data-x="handler-window-onpageshow">body</code>
     <td> <code data-x="event-pageshow">pageshow</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-window-onpageswap"> <code data-x="">onpageswap</code>
     <td> <code data-x="handler-window-onpageswap">body</code>
     <td> <code data-x="event-pageswap">pageswap</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onpaste"> <code data-x="">onpaste</code>
     <td> <span data-x="handler-onpaste">HTML elements</span>
     <td> <code data-x="event-paste">paste</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onpause"> <code data-x="">onpause</code>
     <td> <span data-x="handler-onpause">HTML elements</span>
     <td> <code data-x="event-media-pause">pause</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onplay"> <code data-x="">onplay</code>
     <td> <span data-x="handler-onplay">HTML elements</span>
     <td> <code data-x="event-media-play">play</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onplaying"> <code data-x="">onplaying</code>
     <td> <span data-x="handler-onplaying">HTML elements</span>
     <td> <code data-x="event-media-playing">playing</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-window-onpopstate"> <code data-x="">onpopstate</code>
     <td> <code data-x="handler-window-onpopstate">body</code>
     <td> <code data-x="event-popstate">popstate</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onprogress"> <code data-x="">onprogress</code>
     <td> <span data-x="handler-onprogress">HTML elements</span>
     <td> <code data-x="event-media-progress">progress</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onratechange"> <code data-x="">onratechange</code>
     <td> <span data-x="handler-onratechange">HTML elements</span>
     <td> <code data-x="event-media-ratechange">ratechange</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onreset"> <code data-x="">onreset</code>
     <td> <span data-x="handler-onreset">HTML elements</span>
     <td> <code data-x="event-reset">reset</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onresize"> <code data-x="">onresize</code>
     <td> <span data-x="handler-onresize">HTML elements</span>
     <td> <code data-x="event-resize">resize</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-window-onrejectionhandled"> <code data-x="">onrejectionhandled</code>
     <td> <code data-x="handler-window-onrejectionhandled">body</code>
     <td> <code data-x="event-rejectionhandled">rejectionhandled</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onscroll"> <code data-x="">onscroll</code>
     <td> <span data-x="handler-onscroll">HTML elements</span>
     <td> <code data-x="event-scroll">scroll</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onscrollend"> <code data-x="">onscrollend</code>
     <td> <span data-x="handler-onscrollend">HTML elements</span>
     <td> <code data-x="event-scrollend">scrollend</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onsecuritypolicyviolation"> <code data-x="">onsecuritypolicyviolation</code>
     <td> <span data-x="handler-onsecuritypolicyviolation">HTML elements</span>
     <td> <code data-x="event-securitypolicyviolation">securitypolicyviolation</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onseeked"> <code data-x="">onseeked</code>
     <td> <span data-x="handler-onseeked">HTML elements</span>
     <td> <code data-x="event-media-seeked">seeked</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onseeking"> <code data-x="">onseeking</code>
     <td> <span data-x="handler-onseeking">HTML elements</span>
     <td> <code data-x="event-media-seeking">seeking</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onselect"> <code data-x="">onselect</code>
     <td> <span data-x="handler-onselect">HTML elements</span>
     <td> <code data-x="event-select">select</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onslotchange"> <code data-x="">onslotchange</code>
     <td> <span data-x="handler-onslotchange">HTML elements</span>
     <td> <code data-x="event-slotchange">slotchange</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onstalled"> <code data-x="">onstalled</code>
     <td> <span data-x="handler-onstalled">HTML elements</span>
     <td> <code data-x="event-media-stalled">stalled</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-window-onstorage"> <code data-x="">onstorage</code>
     <td> <code data-x="handler-window-onstorage">body</code>
     <td> <code data-x="event-storage">storage</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onsubmit"> <code data-x="">onsubmit</code>
     <td> <span data-x="handler-onsubmit">HTML elements</span>
     <td> <code data-x="event-submit">submit</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onsuspend"> <code data-x="">onsuspend</code>
     <td> <span data-x="handler-onsuspend">HTML elements</span>
     <td> <code data-x="event-media-suspend">suspend</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-ontimeupdate"> <code data-x="">ontimeupdate</code>
     <td> <span data-x="handler-ontimeupdate">HTML elements</span>
     <td> <code data-x="event-media-timeupdate">timeupdate</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-ontoggle"> <code data-x="">ontoggle</code>
     <td> <span data-x="handler-ontoggle">HTML elements</span>
     <td> <code data-x="event-toggle">toggle</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-window-onunhandledrejection"> <code data-x="">onunhandledrejection</code>
     <td> <code data-x="handler-window-onunhandledrejection">body</code>
     <td> <code data-x="event-unhandledrejection">unhandledrejection</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-window-onunload"> <code data-x="">onunload</code>
     <td> <code data-x="handler-window-onunload">body</code>
     <td> <code data-x="event-unload">unload</code> event handler for <code>Window</code> object
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onvolumechange"> <code data-x="">onvolumechange</code>
     <td> <span data-x="handler-onvolumechange">HTML elements</span>
     <td> <code data-x="event-media-volumechange">volumechange</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onwaiting"> <code data-x="">onwaiting</code>
     <td> <span data-x="handler-onwaiting">HTML elements</span>
     <td> <code data-x="event-media-waiting">waiting</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

    <tr>
     <th id="ix-handler-onwheel"> <code data-x="">onwheel</code>
     <td> <span data-x="handler-onwheel">HTML elements</span>
     <td> <code data-x="event-wheel">wheel</code> event handler
     <td> <span data-x="event handler content attributes">Event handler content attribute</span>

  </table>


  <h3 class="no-num">Element interfaces</h3>

  <!-- NON-NORMATIVE SECTION -->

  <table>
   <caption>List of interfaces for elements</caption>
   <thead>
    <tr>
     <th> Element(s)
     <th> Interface(s)
   <tbody>
    <tr>
     <td> <code>a</code>
     <td> <code>HTMLAnchorElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>abbr</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>address</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>area</code>
     <td> <code>HTMLAreaElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>article</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>aside</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>audio</code>
     <td> <code>HTMLAudioElement</code> : <code>HTMLMediaElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>b</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>base</code>
     <td> <code>HTMLBaseElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>bdi</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>bdo</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>blockquote</code>
     <td> <code>HTMLQuoteElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>body</code>
     <td> <code>HTMLBodyElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>br</code>
     <td> <code>HTMLBRElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>button</code>
     <td> <code>HTMLButtonElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>canvas</code>
     <td> <code>HTMLCanvasElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>caption</code>
     <td> <code>HTMLTableCaptionElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>cite</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>code</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>col</code>
     <td> <code>HTMLTableColElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>colgroup</code>
     <td> <code>HTMLTableColElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>data</code>
     <td> <code>HTMLDataElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>datalist</code>
     <td> <code>HTMLDataListElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>dd</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>del</code>
     <td> <code>HTMLModElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>details</code>
     <td> <code>HTMLDetailsElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>dfn</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>dialog</code>
     <td> <code>HTMLDialogElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>div</code>
     <td> <code>HTMLDivElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>dl</code>
     <td> <code>HTMLDListElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>dt</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>em</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>embed</code>
     <td> <code>HTMLEmbedElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>fieldset</code>
     <td> <code>HTMLFieldSetElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>figcaption</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>figure</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>footer</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>form</code>
     <td> <code>HTMLFormElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>h1</code>
     <td> <code>HTMLHeadingElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>h2</code>
     <td> <code>HTMLHeadingElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>h3</code>
     <td> <code>HTMLHeadingElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>h4</code>
     <td> <code>HTMLHeadingElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>h5</code>
     <td> <code>HTMLHeadingElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>h6</code>
     <td> <code>HTMLHeadingElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>head</code>
     <td> <code>HTMLHeadElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>header</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>hgroup</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>hr</code>
     <td> <code>HTMLHRElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>html</code>
     <td> <code>HTMLHtmlElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>i</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>iframe</code>
     <td> <code>HTMLIFrameElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>img</code>
     <td> <code>HTMLImageElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>input</code>
     <td> <code>HTMLInputElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>ins</code>
     <td> <code>HTMLModElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>kbd</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>label</code>
     <td> <code>HTMLLabelElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>legend</code>
     <td> <code>HTMLLegendElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>li</code>
     <td> <code>HTMLLIElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>link</code>
     <td> <code>HTMLLinkElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>main</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>map</code>
     <td> <code>HTMLMapElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>mark</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>menu</code>
     <td> <code>HTMLMenuElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>meta</code>
     <td> <code>HTMLMetaElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>meter</code>
     <td> <code>HTMLMeterElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>nav</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>noscript</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>object</code>
     <td> <code>HTMLObjectElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>ol</code>
     <td> <code>HTMLOListElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>optgroup</code>
     <td> <code>HTMLOptGroupElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>option</code>
     <td> <code>HTMLOptionElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>output</code>
     <td> <code>HTMLOutputElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>p</code>
     <td> <code>HTMLParagraphElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>picture</code>
     <td> <code>HTMLPictureElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>pre</code>
     <td> <code>HTMLPreElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>progress</code>
     <td> <code>HTMLProgressElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>q</code>
     <td> <code>HTMLQuoteElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>rp</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>rt</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>ruby</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>s</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>samp</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>search</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>script</code>
     <td> <code>HTMLScriptElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>section</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>select</code>
     <td> <code>HTMLSelectElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>slot</code>
     <td> <code>HTMLSlotElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>small</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>source</code>
     <td> <code>HTMLSourceElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>span</code>
     <td> <code>HTMLSpanElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>strong</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>style</code>
     <td> <code>HTMLStyleElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>sub</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>summary</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>sup</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>table</code>
     <td> <code>HTMLTableElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>tbody</code>
     <td> <code>HTMLTableSectionElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>td</code>
     <td> <code>HTMLTableCellElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>template</code>
     <td> <code>HTMLTemplateElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>textarea</code>
     <td> <code>HTMLTextAreaElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>tfoot</code>
     <td> <code>HTMLTableSectionElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>th</code>
     <td> <code>HTMLTableCellElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>thead</code>
     <td> <code>HTMLTableSectionElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>time</code>
     <td> <code>HTMLTimeElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>title</code>
     <td> <code>HTMLTitleElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>tr</code>
     <td> <code>HTMLTableRowElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>track</code>
     <td> <code>HTMLTrackElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>u</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>ul</code>
     <td> <code>HTMLUListElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>var</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <code>video</code>
     <td> <code>HTMLVideoElement</code> : <code>HTMLMediaElement</code> : <code>HTMLElement</code>

    <tr>
     <td> <code>wbr</code>
     <td> <code>HTMLElement</code>

    <tr>
     <td> <span data-x="custom element">custom elements</span>
     <td> supplied by the element's author (inherits from <code>HTMLElement</code>)

  </table>


  <h3 class="no-num">All interfaces</h3>

  <!-- NON-NORMATIVE SECTION -->

INSERT INTERFACES HERE


  <h3 class="no-num">Events</h3>

  <!-- NON-NORMATIVE SECTION -->

  <p>The following table lists events fired by this document, excluding those
  already defined in <a href="#mediaevents">media element events</a> and
  <a href="#dndevents">drag-and-drop events</a>.</p>

  <table>
   <caption>List of events</caption>
   <thead>
    <tr>
     <th> Event
     <th> Interface
     <th> Interesting targets
     <th> Description
   <tbody>

    <tr> <!-- DOMContentLoaded -->
     <td> <dfn event for="Document"><code data-x="event-DOMContentLoaded">DOMContentLoaded</code></dfn>
     <td> <code>Event</code>
     <td> <code>Document</code>
     <td> Fired at the <code>Document</code> once the parser has finished

    <tr> <!-- afterprint -->
     <td> <dfn event for="Window"><code data-x="event-afterprint">afterprint</code></dfn>
     <td> <code>Event</code>
     <td> <code>Window</code>
     <td> Fired at the <code>Window</code> after printing

    <tr> <!-- beforeprint -->
     <td> <dfn event for="Window"><code data-x="event-beforeprint">beforeprint</code></dfn>
     <td> <code>Event</code>
     <td> <code>Window</code>
     <td> Fired at the <code>Window</code> before printing

    <tr> <!-- beforematch -->
     <td> <dfn event for="HTMLElement"><code data-x="event-beforematch">beforematch</code></dfn>
     <td> <code>Event</code>
     <td> Elements
     <td> Fired on elements with the <code
     data-x="attr-hidden-until-found-state">hidden=until-found</code> attribute before they are
     revealed.

    <tr> <!-- beforetoggle -->
     <td> <dfn event for="HTMLElement"><code data-x="event-beforetoggle">beforetoggle</code></dfn>
     <td> <code>ToggleEvent</code>
     <td> Elements
     <td> Fired on elements with the <code data-x="attr-popover">popover</code> attribute when they
     are transitioning between showing and hidden

    <tr> <!-- beforeunload -->
     <td> <dfn event for="Window"><code data-x="event-beforeunload">beforeunload</code></dfn>
     <td> <code>BeforeUnloadEvent</code>
     <td> <code>Window</code>
     <td> Fired at the <code>Window</code> when the page is about to be unloaded, in case the page would like to show a warning prompt

    <tr> <!-- blur -->
     <td> <dfn event for="Window,HTMLElement"><code data-x="event-blur">blur</code></dfn>
     <td> <code>Event</code>
     <td> <code>Window</code>, elements
     <td> Fired at nodes when they stop being <span>focused</span>

    <tr> <!-- cancel -->
     <td> <dfn event for="CloseWatcher,HTMLElement"><code data-x="event-cancel">cancel</code></dfn>
     <td> <code>Event</code>
     <td> <code>CloseWatcher</code>, <code>dialog</code> elements, <code>input</code> elements
     <td> Fired at <code>CloseWatcher</code> objects or <code>dialog</code> elements when they receive a <span>close request</span>, or at <code>input</code> elements whose <span data-x="attr-input-type">type</span> attribute is in the <span data-x="attr-input-type-file">File</span> state when the user does not change their selection

    <tr> <!-- change -->
     <td> <dfn event for="HTMLElement"><code data-x="event-change" id="event-input-change">change</code></dfn>
     <td> <code>Event</code>
     <td> Form controls
     <td> Fired at controls when the user commits a value change (see also the <code data-x="event-input">input</code> event)

    <tr> <!-- click -->
     <td> <code data-x="event-click">click</code>
     <td> <code>PointerEvent</code>
     <td> Elements
     <td> Normally a mouse event; also synthetically fired at an element before its <span>activation behavior</span> is run, when an element is activated from a non-pointer input device (e.g. a keyboard)

    <tr> <!-- close -->
     <td> <dfn event for="CloseWatcher,HTMLElement,MessagePort"><code data-x="event-close">close</code></dfn>
     <td> <code>Event</code>
     <td> <code>CloseWatcher</code>, <code>dialog</code> elements, <code>MessagePort</code>
     <td> Fired at <code>CloseWatcher</code> objects or <code>dialog</code> elements when they are closed via a <span>close request</span> or via web developer code, or at <code>MessagePort</code> objects when <span data-x="disentangle">disentangled</span>

    <tr> <!-- command -->
     <td> <dfn event for="HTMLElement"><code data-x="event-command">command</code></dfn>
     <td> <code>CommandEvent</code>
     <td> Elements
     <td> Fired at elements when they handle a user invocation, via a <code
     data-x="attr-button-commandfor">commandfor</code> attribute.

    <tr> <!-- connect -->
     <td> <dfn event for="SharedWorkerGlobalScope"><code data-x="event-WorkerGlobalScope-connect">connect</code></dfn>
     <td> <code>MessageEvent</code>
     <td> <code>SharedWorkerGlobalScope</code>
     <td> Fired at a shared worker's global scope when a new client connects

    <tr> <!-- contextlost -->
     <td> <dfn event for="HTMLElement,OffscreenCanvas"><code data-x="event-contextlost">contextlost</code></dfn>
     <td> <code>Event</code>
     <td> <code>canvas</code> elements, <code>OffscreenCanvas</code> objects
     <td> Fired when the corresponding <code>CanvasRenderingContext2D</code> or <code>OffscreenCanvasRenderingContext2D</code> is lost

    <tr> <!-- contextrestore -->
     <td> <dfn event for="HTMLElement,OffscreenCanvas"><code data-x="event-contextrestored">contextrestored</code></dfn>
     <td> <code>Event</code>
     <td> <code>canvas</code> elements, <code>OffscreenCanvas</code> objects
     <td> Fired when the corresponding <code>CanvasRenderingContext2D</code> or <code>OffscreenCanvasRenderingContext2D</code> is restored after being lost

    <tr> <!-- currententrychange -->
     <td> <dfn event for="Navigation"><code data-x="event-currententrychange">currententrychange</code></dfn>
     <td> <code>NavigationCurrentEntryChangeEvent</code>
     <td> <code>Navigation</code>
     <td> Fired when <code data-x="dom-Navigation-currentEntry">navigation.currentEntry</code> changes

    <tr> <!-- dispose -->
     <td> <dfn event for="NavigationHistoryEntry"><code data-x="event-dispose">dispose</code></dfn>
     <td> <code>Event</code>
     <td> <code>NavigationHistoryEntry</code>
     <td> Fired when the <span>session history entry</span> corresponding to the <code>NavigationHistoryEntry</code> has been permanently evicted from session history and can no longer be traversed to

    <tr> <!-- error -->
     <td> <dfn event for="Window,WorkerGlobalScope,HTMLElement,EventSource,Worker,SharedWorker,AbstractWorker"><code data-x="event-error">error</code></dfn>
     <td> <code>Event</code> or <code>ErrorEvent</code>
     <td> Global scope objects, <code>Worker</code> objects, elements, networking-related objects
     <td> Fired when unexpected errors occur (e.g. networking errors, script errors, decoding errors)

    <tr> <!-- focus -->
     <td> <dfn event for="Window,HTMLElement"><code data-x="event-focus">focus</code></dfn>
     <td> <code>Event</code>
     <td> <code>Window</code>, elements
     <td> Fired at nodes <span data-x="gains focus">gaining focus</span>

    <tr> <!-- formdata -->
     <td> <dfn event for="HTMLElement"><code data-x="event-formdata">formdata</code></dfn>
     <td> <code>FormDataEvent</code>
     <td> <code>form</code> elements
     <td> Fired at a <code>form</code> element when it is <span>constructing the entry list</span>

    <tr> <!-- hashchange -->
     <td> <dfn event for="Window"><code data-x="event-hashchange">hashchange</code></dfn>
     <td> <code>HashChangeEvent</code>
     <td> <code>Window</code>
     <td> Fired at the <code>Window</code> when the <span data-x="concept-url-fragment">fragment</span> part of the document's <span data-x="concept-document-url">URL</span> changes

    <tr> <!-- input -->
     <td> <span><code data-x="event-input">input</code></span>
     <td> <code>Event</code>
     <td> Elements
     <td> Fired when the user changes the <code data-x="attr-contenteditable">contenteditable</code> element's content, or the form control's value. See also the <code data-x="event-change">change</code> event for form controls.

    <tr> <!-- invalid -->
     <td> <dfn event for="HTMLElement"><code data-x="event-invalid">invalid</code></dfn>
     <td> <code>Event</code>
     <td> Form controls
     <td> Fired at controls during form validation if they do not satisfy their constraints

    <tr> <!-- languagechange -->
     <td> <dfn event for="Window,WorkerGlobalScope"><code data-x="event-languagechange">languagechange</code></dfn>
     <td> <code>Event</code>
     <td> Global scope objects
     <td> Fired at the global scope object when the user's preferred languages change

    <tr> <!-- load -->
     <td> <dfn event for="Window,HTMLElement"><code data-x="event-load">load</code></dfn>
     <td> <code>Event</code>
     <td> <code>Window</code>, elements
     <td> Fired at the <code>Window</code> when the document has finished loading; fired at an element containing a resource (e.g. <code>img</code>, <code>embed</code>) when its resource has finished loading

    <tr> <!-- message -->
     <td> <dfn event for="Window,EventSource,MessagePort,BroadcastChannel,DedicatedWorkerGlobalScope,Worker,ServiceWorkerContainer"><code data-x="event-message">message</code></dfn>
     <td> <code>MessageEvent</code>
     <td> <code>Window</code>, <code>EventSource</code>, <code>MessagePort</code>, <code>BroadcastChannel</code>, <code>DedicatedWorkerGlobalScope</code>, <code>Worker</code>, <code>ServiceWorkerContainer</code>
     <td> Fired at an object when it receives a message

    <tr> <!-- messageerror -->
     <td> <dfn event for="Window,MessagePort,BroadcastChannel,DedicatedWorkerGlobalScope,Worker,ServiceWorkerContainer"><code data-x="event-messageerror">messageerror</code></dfn>
     <td> <code>MessageEvent</code>
     <td> <code>Window</code>, <code>MessagePort</code>, <code>BroadcastChannel</code>, <code>DedicatedWorkerGlobalScope</code>, <code>Worker</code>, <code>ServiceWorkerContainer</code>
     <td> Fired at an object when it receives a message that cannot be deserialized

    <tr> <!-- navigate -->
     <td> <dfn event for="Navigation"><code data-x="event-navigate">navigate</code></dfn>
     <td> <code>NavigateEvent</code>
     <td> <code>Navigation</code>
     <td> Fired before the <span>navigable</span> <span data-x="navigate">navigates</span>, <span data-x="reload">reloads</span>, <span data-x="traverse the history by a delta">traverses</span>, or <span data-x="shared history push/replace state steps">otherwise</span> changes its URL

    <tr> <!-- navigateerror -->
     <td> <dfn event for="Navigation"><code data-x="event-navigateerror">navigateerror</code></dfn>
     <td> <code>ErrorEvent</code>
     <td> <code>Navigation</code>
     <td> Fired when a navigation does not complete successfully

    <tr> <!-- navigatesuccess -->
     <td> <dfn event for="Navigation"><code data-x="event-navigatesuccess">navigatesuccess</code></dfn>
     <td> <code>Event</code>
     <td> <code>Navigation</code>
     <td> Fired when a navigation completes successfully

    <tr> <!-- offline -->
     <td> <dfn event for="Window,WorkerGlobalScope"><code data-x="event-offline">offline</code></dfn>
     <td> <code>Event</code>
     <td> Global scope objects
     <td> Fired at the global scope object when the network connections fails

    <tr> <!-- online -->
     <td> <dfn event for="Window,WorkerGlobalScope"><code data-x="event-online">online</code></dfn>
     <td> <code>Event</code>
     <td> Global scope objects
     <td> Fired at the global scope object when the network connections returns

    <tr> <!-- open -->
     <td> <dfn event for="EventSource"><code data-x="event-open">open</code></dfn>
     <td> <code>Event</code>
     <td> <code>EventSource</code>
     <td> Fired at <code>EventSource</code> objects when a connection is established

    <tr> <!-- pageswap -->
     <td> <dfn event for="Window"><code data-x="event-pageswap">pageswap</code></dfn>
     <td> <code>PageSwapEvent</code>
     <td> <code>Window</code>
     <td>Fired at the <code>Window</code> right before a document is <span
     data-x="unload a document">unloaded</span> as a result of a navigation.

    <tr> <!-- pagehide -->
     <td> <dfn event for="Window"><code data-x="event-pagehide">pagehide</code></dfn>
     <td> <code>PageTransitionEvent</code>
     <td> <code>Window</code>
     <td> Fired at the <code>Window</code> when the page's <span>session history entry</span> stops
     being the <span data-x="nav-active-history-entry">active entry</span>

    <tr> <!-- pagereveal -->
     <td> <dfn event for="Window"><code data-x="event-pagereveal">pagereveal</code></dfn>
     <td> <code>PageRevealEvent</code>
     <td> <code>Window</code>
     <td> Fired at the <code>Window</code> when the page begins to render for the first time after
     it has been initialized or <span data-x="reactivate a document">reactivated</span>

    <tr> <!-- pageshow -->
     <td> <dfn event for="Window"><code data-x="event-pageshow">pageshow</code></dfn>
     <td> <code>PageTransitionEvent</code>
     <td> <code>Window</code>
     <td> Fired at the <code>Window</code> when the page's <span>session history entry</span>
     becomes the <span data-x="nav-active-history-entry">active entry</span>

    <tr> <!-- pointercancel -->
     <td> <span><code data-x="event-pointercancel">pointercancel</code></span>
     <td> <code>PointerEvent</code>
     <td> Elements and <code>Text</code> nodes
     <td> Fired at the <span>source node</span> when the user attempts to initiate a drag-and-drop operation

    <tr> <!-- popstate -->
     <td> <dfn event for="Window"><code data-x="event-popstate">popstate</code></dfn>
     <td> <code>PopStateEvent</code>
     <td> <code>Window</code>
     <td> Fired at the <code>Window</code> when in some cases of <span data-x="traverse the history by a delta">session history traversal</span>

    <tr> <!-- readystatechange -->
     <td> <dfn event for="Document"><code data-x="event-readystatechange">readystatechange</code></dfn>
     <td> <code>Event</code>
     <td> <code>Document</code>
     <td> Fired at the <code>Document</code> when it finishes parsing and again when all its subresources have finished loading

    <tr> <!-- rejectionhandled -->
     <td> <dfn event for="Window,WorkerGlobalScope"><code data-x="event-rejectionhandled">rejectionhandled</code></dfn>
     <td> <code>PromiseRejectionEvent</code>
     <td> Global scope objects
     <td> Fired at global scope objects when a previously-unhandled promise rejection becomes handled

    <tr> <!-- reset -->
     <td> <dfn event for="HTMLElement"><code data-x="event-reset">reset</code></dfn>
     <td> <code>Event</code>
     <td> <code>form</code> elements
     <td> Fired at a <code>form</code> element when it is <span data-x="concept-form-reset">reset</span>

    <tr> <!-- select -->
     <td> <dfn event for="HTMLElement"><code data-x="event-select">select</code></dfn>
     <td> <code>Event</code>
     <td> Form controls
     <td> Fired at form controls when their text selection is adjusted (whether by an API or by the user)

    <tr> <!-- storage -->
     <td> <dfn event for="Window"><code data-x="event-storage">storage</code></dfn>
     <td> <code>StorageEvent</code>
     <td> <code>Window</code>
     <td> Fired at <code>Window</code> event when the corresponding <code data-x="dom-localStorage">localStorage</code> or <code data-x="dom-sessionStorage">sessionStorage</code> storage areas change

    <tr> <!-- submit -->
     <td> <dfn event for="HTMLElement"><code data-x="event-submit">submit</code></dfn>
     <td> <code>SubmitEvent</code>
     <td> <code>form</code> elements
     <td> Fired at a <code>form</code> element when it is <span data-x="concept-form-submit">submitted</span>

    <tr> <!-- toggle -->
     <td> <dfn event for="HTMLElement"><code data-x="event-toggle">toggle</code></dfn>
     <td> <code>ToggleEvent</code>
     <td> <code>details</code> and <span data-x="attr-popover">popover</span> elements
     <td> Fired at <code>details</code> elements when they open or close; fired on elements with the
     <code data-x="attr-popover">popover</code> attribute when they are transitioning between
     showing and hidden

    <tr> <!-- unhandledrejection -->
     <td> <dfn event for="Window,WorkerGlobalScope"><code data-x="event-unhandledrejection">unhandledrejection</code></dfn>
     <td> <code>PromiseRejectionEvent</code>
     <td> Global scope objects
     <td> Fired at global scope objects when a promise rejection goes unhandled

    <tr> <!-- unload -->
     <td> <dfn event for="Window"><code data-x="event-unload">unload</code></dfn>
     <td> <code>Event</code>
     <td> <code>Window</code>
     <td> Fired at the <code>Window</code> object when the page is going away

    <tr> <!-- visibilitychange -->
     <td> <dfn event for="Document"><code data-x="event-visibilitychange">visibilitychange</code></dfn>
     <td> <code>Event</code>
     <td> <code>Document</code>
     <td> Fired at the <code>Document</code> object when the page becomes visible or hidden to the
     user

  </table>


  <h3 class="no-num">HTTP headers</h3>

  <!-- NON-NORMATIVE SECTION -->

  <p>The following HTTP request headers are defined by this specification:</p>

  <ul>
   <li>`<code>Last-Event-ID</code>`</li>
   <li>`<code>Ping-From</code>`</li>
   <li>`<code>Ping-To</code>`</li>
  </ul>

  <p>The following HTTP response headers are defined by this specification:</p>

  <ul>
   <li>`<code>Cross-Origin-Embedder-Policy</code>`</li>
   <li>`<code>Cross-Origin-Embedder-Policy-Report-Only</code>`</li>
   <li>`<code>Cross-Origin-Opener-Policy</code>`</li>
   <li>`<code>Cross-Origin-Opener-Policy-Report-Only</code>`</li>
   <li>`<code>Origin-Agent-Cluster</code>`</li>
   <li>`<code>Refresh</code>`</li>
   <li>`<code>X-Frame-Options</code>`</li>
  </ul>


  <h3 class="no-num">MIME types</h3>

  <!-- NON-NORMATIVE SECTION -->

  <p>The following MIME types are mentioned in this specification:</p>

  <dl>
   <dt><dfn><code>application/atom+xml</code></dfn></dt>
   <dd>Atom <ref>ATOM</ref></dd>

   <dt><dfn><code>application/json</code></dfn></dt>
   <dd>JSON <ref>JSON</ref></dd>

   <dt><dfn data-x-href="https://www.rfc-editor.org/rfc/rfc2046#section-4.5.1"><code>application/octet-stream</code></dfn></dt>
   <dd>Generic binary data <ref>RFC2046</ref></dd>

   <dt><code>application/microdata+json</code></dt>
   <dd>Microdata as JSON

   <dt><dfn><code>application/rss+xml</code></dfn></dt>
   <dd>RSS <!--<ref>RSS</ref>-->

   <dt><dfn><code>application/wasm</code></dfn></dt>
   <dd>WebAssembly <ref>WASM</ref></dd>

   <dt><code>application/x-www-form-urlencoded</code></dt>
   <dd>Form submission</dd>

   <dt><code>application/xhtml+xml</code></dt>
   <dd>HTML</dd>

   <dt><dfn><code>application/xml</code></dfn></dt>
   <dd>XML <ref>XML</ref> <ref>RFC7303</ref></dd>

   <dt><dfn><code>image/gif</code></dfn></dt>
   <dd>GIF images <ref>GIF</ref></dd>

   <dt><dfn><code>image/jpeg</code></dfn></dt>
   <dd>JPEG images <ref>JPEG</ref></dd>

   <dt><dfn><code>image/png</code></dfn></dt>
   <dd>PNG images <ref>PNG</ref></dd>

   <dt><dfn><code>image/svg+xml</code></dfn></dt>
   <dd>SVG images <ref>SVG</ref></dd>

   <dt><dfn><code>multipart/form-data</code></dfn></dt>
   <dd>Form submission <ref>RFC7578</ref></dd>

   <dt><dfn><code>multipart/mixed</code></dfn></dt>
   <dd>Generic mixed content <ref>RFC2046</ref></dd>

   <dt><code>multipart/x-mixed-replace</code></dt>
   <dd>Streaming server push</dd>

   <dt><dfn><code>text/css</code></dfn></dt>
   <dd>CSS <ref>CSS</ref></dd>

   <dt><code>text/event-stream</code></dt>
   <dd>Server-sent event streams</dd>

   <dt><dfn><code>text/javascript</code></dfn></dt>
   <dd>JavaScript <ref>JAVASCRIPT</ref> <ref>RFC9239</ref>

   <!--
   <dt><dfn><code>text/javascript;e4x=1</code></dfn></dt>
   <dd>E4X <ref>E4X</ref>
   -->

   <dt><dfn><code>text/json</code></dfn></dt>
   <dd>JSON (legacy type)</dd>

   <dt><dfn data-x-href="https://www.rfc-editor.org/rfc/rfc2046#section-4.1.3"><code>text/plain</code></dfn></dt>
   <dd>Generic plain text <ref>RFC2046</ref> <ref>RFC3676</ref></dd>

   <dt><code>text/html</code></dt>
   <dd>HTML</dd>

   <dt><code>text/ping</code></dt>
   <dd>Hyperlink auditing</dd>

   <dt><dfn><code>text/uri-list</code></dfn></dt>
   <dd>List of URLs <ref>RFC2483</ref>

   <dt><dfn><code>text/vcard</code></dfn></dt>
   <dd>vCard <ref>RFC6350</ref>

   <dt><dfn><code>text/vtt</code></dfn></dt>
   <dd>WebVTT <ref>WEBVTT</ref></dd>

   <dt><dfn><code>text/xml</code></dfn></dt>
   <dd>XML <ref>XML</ref> <ref>RFC7303</ref></dd>

   <dt><dfn data-x-href="https://www.rfc-editor.org/rfc/rfc4337#section-3"><code>video/mp4</code></dfn></dt>
   <dd>MPEG-4 video <ref>RFC4337</ref></dd>

   <dt><dfn><code>video/mpeg</code></dfn></dt>
   <dd>MPEG video <ref>RFC2046</ref></dd>
  </dl>



  <h2 split-filename="references" class="no-num" id="references">References</h2><!--REFS-->

  <p>All references are normative unless marked "Non-normative".</p>

  <!-- Dates are only included for standards older than the web, because the newer ones keep changing. -->

  <dl id="ref-list">

   <dt id="refsABNF">[ABNF]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc5234">Augmented BNF for Syntax Specifications: ABNF</a></cite>, D. Crocker, P. Overell. IETF.</dd>

   <dt id="refsABOUT">[ABOUT]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc6694">The 'about' URI scheme</a></cite>, S. Moonesamy. IETF.</dd>

   <dt id="refsAPNG">[APNG]</dt>
   <dd>(Non-normative) <cite><a href="https://wiki.mozilla.org/APNG_Specification">APNG Specification</a></cite>. S. Parmenter, V. Vukicevic, A. Smith. Mozilla.</dd>

   <dt id="refsARIA">[ARIA]</dt>
   <dd><cite><a href="https://w3c.github.io/aria/">Accessible Rich Internet Applications (WAI-ARIA)</a></cite>, J. Diggs, J. Nurthen, M. Cooper. W3C.</dd>

   <dt id="refsARIAHTML">[ARIAHTML]</dt>
   <dd><cite><a href="https://w3c.github.io/html-aria/">ARIA in HTML</a></cite>, S. Faulkner, S. O'Hara. W3C.</dd>

   <dt id="refsATAG">[ATAG]</dt>
   <dd>(Non-normative) <cite><a href="https://www.w3.org/TR/ATAG20/">Authoring Tool Accessibility Guidelines (ATAG) 2.0</a></cite>, J. Richards, J. Spellman, J. Treviranus. W3C.</dd>

   <dt id="refsATOM">[ATOM]</dt>
   <dd>(Non-normative) <cite><a href="https://www.rfc-editor.org/rfc/rfc4287">The Atom Syndication Format</a></cite>, M. Nottingham, R. Sayre. IETF.</dd>

   <dt id="refsBATTERY">[BATTERY]</dt>
   <dd>(Non-normative) <cite><a href="https://w3c.github.io/battery/">Battery Status API</a></cite>, A. Kostiainen, M. Lamouri. W3C.</dd>

   <dt id="refsBCP47">[BCP47]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/info/bcp47">Tags for Identifying Languages; Matching of Language Tags</a></cite>, A. Phillips, M. Davis. IETF.</dd>

   <dt id="refsBEZIER">[BEZIER]</dt>
   <dd><cite>Courbes &agrave; poles</cite>, P. de Casteljau. INPI, 1959.</dd>

   <dt id="refsBIDI">[BIDI]</dt>
   <dd><cite><a href="https://www.unicode.org/reports/tr9/">UAX #9: Unicode Bidirectional Algorithm</a></cite>, M. Davis. Unicode Consortium.</dd>

   <dt id="refsBOCU1">[BOCU1]</dt>
   <dd>(Non-normative) <cite><a href="https://www.unicode.org/notes/tn6/">UTN #6: BOCU-1: MIME-Compatible Unicode Compression</a></cite>, M. Scherer, M. Davis. Unicode Consortium.</dd>

   <dt id="refsCESU8">[CESU8]</dt>
   <dd>(Non-normative) <cite><a href="https://www.unicode.org/reports/tr26/">UTR #26: Compatibility Encoding Scheme For UTF-16: 8-BIT (CESU-8)</a></cite>, T. Phipps. Unicode Consortium.</dd>

   <dt id="refsCHARMOD">[CHARMOD]</dt>
   <dd>(Non-normative) <cite><a href="https://www.w3.org/TR/charmod/">Character Model for the World Wide Web 1.0: Fundamentals</a></cite>, M. D&uuml;rst, F. Yergeau, R. Ishida, M. Wolf, T. Texin. W3C.</dd>

   <dt id="refsCHARMODNORM">[CHARMODNORM]</dt>
   <dd>(Non-normative) <cite><a href="https://w3c.github.io/charmod-norm/">Character Model for the World Wide Web: String Matching</a></cite>, A. Phillips. W3C.</dd>

   <dt id="refsCLIPBOARD-APIS">[CLIPBOARD-APIS]</dt>
   <dd><cite><a href="https://w3c.github.io/clipboard-apis/">Clipboard API and events</a></cite>, G. Kacmarcik, A. Snigdha. W3C.</dd>

   <dt id="refsCOMPOSITE">[COMPOSITE]</dt>
   <dd><cite><a href="https://drafts.fxtf.org/compositing/">Compositing and Blending</a></cite>, R. Cabanier, N. Andronikos. W3C.</dd>

   <dt id="refsCOMPUTABLE">[COMPUTABLE]</dt>
   <dd>(Non-normative) <cite><a href="http://www.turingarchive.org/browse.php/B/12">On computable numbers, with an application to the Entscheidungsproblem</a></cite>, A. Turing. In <cite>Proceedings of the London Mathematical Society</cite>, series 2, volume 42, pages 230-265. London Mathematical Society, 1937.</dd>

   <dt id="refsCOMPUTEPRESSURE">[COMPUTEPRESSURE]</dt>
   <dd>(Non-normative) <cite><a href="https://w3c.github.io/compute-pressure/">Compute Pressure</a></cite>, K. Christiansen, A. Mandy. W3C.</dd>

   <dt id="refsCONSOLE">[CONSOLE]</dt>
   <dd><cite><a href="https://console.spec.whatwg.org/">Console</a></cite>, T. Stock, R. Kowalski, D. Farolino. WHATWG.</dd>

   <dt id="refsCOOKIES">[COOKIES]</dt>
   <dd><cite><a href="https://httpwg.org/specs/rfc6265.html">HTTP State Management Mechanism</a></cite>, A. Barth. IETF.</dd>

   <dt id="refsCREDMAN">[CREDMAN]</dt>
   <dd><cite><a href="https://w3c.github.io/webappsec-credential-management/">Credential Management</a></cite>, N. Satragno, J. Hodges, M. West. W3C.</dd>

   <dt id="refsCSP">[CSP]</dt>
   <dd><cite><a href="https://w3c.github.io/webappsec-csp/">Content Security Policy</a></cite>, M. West, D. Veditz. W3C.</dd>

   <dt id="refsCSS">[CSS]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css2/">Cascading Style Sheets Level 2 Revision 2</a></cite>, B. Bos, T. &Ccedil;elik, I. Hickson, H. Lie. W3C.</dd>

   <dt id="refsCSSALIGN">[CSSALIGN]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-align/">CSS Box Alignment</a></cite>, E. Etemad, T. Atkins. W3C.</dd>

   <dt id="refsCSSANCHOR">[CSSANCHOR]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-anchor-position/">CSS Anchor Positioning</a></cite>, T. Atkins, E. Etemad, I. Kilpatrick. W3C.</dd>

   <dt id="refsCSSANIMATIONS">[CSSANIMATIONS]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-animations/">CSS Animations</a></cite>, D. Jackson, D. Hyatt, C. Marrin, S. Galineau, L. Baron. W3C.</dd>

   <dt id="refsCSSATTR">[CSSATTR]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-style-attr/">CSS Style Attributes</a></cite>, T. &Ccedil;elik, E. Etemad. W3C.</dd>

   <dt id="refsCSSBG">[CSSBG]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-backgrounds/">CSS Backgrounds and Borders</a></cite>, B. Bos, E. Etemad, B. Kemper. W3C.</dd>

   <dt id="refsCSSBOX">[CSSBOX]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-box-3/">CSS Box Model</a></cite>, E. Etemad. W3C.</dd>

   <dt id="refsCSSCASCADE">[CSSCASCADE]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-cascade/">CSS Cascading and Inheritance</a></cite>, E. Etemad, T. Atkins. W3C.</dd>

   <dt id="refsCSSCONTAIN">[CSSCONTAIN]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-contain/">CSS Containment</a></cite>, T. Atkins, F. Rivoal, V. Levin. W3C.</dd>

   <dt id="refsCSSCOLOR">[CSSCOLOR]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-color/">CSS Color Module</a></cite>, T. &Ccedil;elik, C. Lilley, L. Baron. W3C.</dd>

   <dt id="refsCSSCOLORADJUST">[CSSCOLORADJUST]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-color-adjust/">CSS Color Adjustment Module</a></cite>, E. Etemad, R. Atanassov, R. Lillesveen, T. Atkins. W3C.</dd>

   <dt id="refsCSSDEVICEADAPT">[CSSDEVICEADAPT]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-device-adapt/">CSS Device Adaption</a></cite>, F. Rivoal, M. Rakow. W3C.</dd>

   <dt id="refsCSSDISPLAY">[CSSDISPLAY]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-display/">CSS Display</a></cite>, T. Atkins, E. Etemad. W3C.</dd>

   <dt id="refsCSSFONTLOAD">[CSSFONTLOAD]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-font-loading/">CSS Font Loading</a></cite>, T. Atkins, J. Daggett. W3C.</dd>

   <dt id="refsCSSFONTS">[CSSFONTS]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-fonts/">CSS Fonts</a></cite>, J. Daggett. W3C.</dd>

   <dt id="refsCSSFLEXBOX">[CSSFLEXBOX]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-flexbox/">CSS Flexible Box Layout</a></cite>, T. Atkins, E. Etemad, R. Atanassov. W3C.</dd>

   <dt id="refsCSSGC">[CSSGC]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-content/">CSS Generated Content</a></cite>, H. Lie, E. Etemad, I. Hickson. W3C.</dd>

   <dt id="refsCSSGRID">[CSSGRID]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-grid/">CSS Grid Layout</a></cite>, T. Atkins, E. Etemad, R. Atanassov. W3C.</dd>

   <dt id="refsCSSIMAGES">[CSSIMAGES]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-images/">CSS Images Module</a></cite>, E. Etemad, T. Atkins, L. Verou. W3C.</dd>

   <dt id="refsCSSIMAGES4">[CSSIMAGES4]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-images-4/">CSS Images Module Level 4</a></cite>, E. Etemad, T. Atkins, L. Verou. W3C.</dd>

   <dt id="refsCSSINLINE">[CSSINLINE]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-inline/">CSS Inline Layout</a></cite>, D. Cramer, E. Etemad. W3C.</dd>

   <dt id="refsCSSLISTS">[CSSLISTS]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-lists/">CSS Lists and Counters</a></cite>, T. Atkins. W3C.</dd>

   <dt id="refsCSSLOGICAL">[CSSLOGICAL]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-logical/">CSS Logical Properties</a></cite>, R. Atanassov, E. Etemad. W3C.</dd>

   <dt id="refsCSSMULTICOL">[CSSMULTICOL]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-multicol/">CSS Multi-column Layout</a></cite>, H. Lie, F. Rivoal, R. Andrew. W3C.</dd>

   <dt id="refsCSSOM">[CSSOM]</dt>
   <dd><cite><a href="https://drafts.csswg.org/cssom/">Cascading Style Sheets Object Model (CSSOM)</a></cite>, S. Pieters, G. Adams. W3C.</dd>

   <dt id="refsCSSOMVIEW">[CSSOMVIEW]</dt>
   <dd><cite><a href="https://drafts.csswg.org/cssom-view/">CSSOM View Module</a></cite>, S. Pieters, G. Adams. W3C.</dd>

   <dt id="refsCSSOVERFLOW">[CSSOVERFLOW]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-overflow-3/">CSS Overflow Module</a></cite>, L. Baron, F. Rivoal. W3C.</dd>

   <dt id="refsCSSPAINT">[CSSPAINT]</dt>
   <dd>(Non-normative) <cite><a href="https://drafts.css-houdini.org/css-paint-api/">CSS Painting API</a></cite>, I. Kilpatrick, D. Jackson. W3C.</dd>

   <dt id="refsCSSPOSITION">[CSSPOSITION]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-position/">CSS Positioned Layout</a></cite>, R. Atanassov, A. Eicholz. W3C.</dd>

   <dt id="refsCSSPSEUDO">[CSSPSEUDO]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-pseudo/">CSS Pseudo-Elements</a></cite>, D. Glazman, E. Etemad, A. Stearns. W3C.</dd>

   <dt id="refsCSSRUBY">[CSSRUBY]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-ruby/">CSS3 Ruby Module</a></cite>, R. Ishida. W3C.</dd>

   <dt id="refsCSSSCOPING">[CSSSCOPING]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-scoping/">CSS Scoping Module</a></cite>, T. Atkins. W3C.</dd>

   <dt id="refsCSSSIZING">[CSSSIZING]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-sizing/">CSS Box Sizing Module</a></cite>, T. Atkins, E. Etemad. W3C.</dd>

   <dt id="refsCSSSCROLLANCHORING">[CSSSCROLLANCHORING]</dt>
   <dd>(Non-normative) <cite><a href="https://drafts.csswg.org/css-scroll-anchoring/">CSS Scroll Anchoring</a></cite>, T. Atkins-Bittner. W3C.</dd>

   <dt id="refsCSSSYNTAX">[CSSSYNTAX]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-syntax/">CSS Syntax</a></cite>, T. Atkins, S. Sapin. W3C.</dd>

   <dt id="refsCSSTRANSITIONS">[CSSTRANSITIONS]</dt>
   <dd>(Non-normative) <cite><a href="https://drafts.csswg.org/css-transitions/">CSS Transitions</a></cite>, L. Baron, D. Jackson, B. Birtles. W3C.</dd>

   <dt id="refsCSSTABLE">[CSSTABLE]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-tables/">CSS Table</a></cite>, F. Remy, G. Whitworth. W3C.</dd>

   <dt id="refsCSSTEXT">[CSSTEXT]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-text/">CSS Text</a></cite>, E. Etemad, K. Ishii. W3C.</dd>

   <dt id="refsCSSVALUES">[CSSVALUES]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-values/">CSS3 Values and Units</a></cite>, H. Lie, T. Atkins, E. Etemad. W3C.</dd>

   <dt id="refsCSSVIEWTRANSITIONS">[CSSVIEWTRANSITIONS]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-view-transitions/">CSS View Transitions</a></cite>, T. Atkins Jr.; J. Archibald; K Sagar. W3C.</dd>

   <dt id="refsCSSUI">[CSSUI]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-ui/">CSS3 Basic User Interface Module</a></cite>, F. Rivoal. W3C.</dd>

   <dt id="refsCSSWM">[CSSWM]</dt>
   <dd><cite><a href="https://drafts.csswg.org/css-writing-modes/">CSS Writing Modes</a></cite>, E. Etemad, K. Ishii. W3C.</dd>

   <dt id="refsDASH">[DASH]</dt>
   <dd><cite><a href="https://www.iso.org/standard/65274.html">Dynamic adaptive streaming over HTTP (DASH)</a></cite>. ISO.</dd>

   <dt id="refsDEVICEPOSTURE">[DEVICEPOSTURE]</dt>
   <dd>(Non-normative) <cite><a href="https://w3c.github.io/device-posture/">Device Posture API</a></cite>, D. Gonzalez-Zuniga, K. Christiansen. W3C.</dd>

   <dt id="refsDOM">[DOM]</dt>
   <dd><cite><a href="https://dom.spec.whatwg.org/">DOM</a></cite>, A. van Kesteren, A. Gregor, Ms2ger. WHATWG.</dd>

   <dt id="refsDOMPARSING">[DOMPARSING]</dt>
   <dd><cite><a href="https://w3c.github.io/DOM-Parsing/">DOM Parsing and Serialization</a></cite>, T. Leithead. W3C.</dd>

   <dt id="refsDOT">[DOT]</dt>
   <dd>(Non-normative) <cite><a href="https://graphviz.org/doc/info/lang.html">The DOT Language</a></cite>. Graphviz.</dd>

   <dt id="refsE163">[E163]</dt>
   <dd><cite>Recommendation E.163 &mdash; Numbering Plan for The International Telephone Service</cite>, CCITT Blue Book, Fascicle II.2, pp. 128-134, November 1988.</dd>

   <dt id="refsENCODING">[ENCODING]</dt>
   <dd><cite><a href="https://encoding.spec.whatwg.org/">Encoding</a></cite>, A. van Kesteren, J. Bell. WHATWG.</dd>

   <dt id="refsEXECCOMMAND">[EXECCOMMAND]</dt>
   <dd><cite><a href="https://w3c.github.io/editing/docs/execCommand/">execCommand</a></cite>, J. Wilm, A. Gregor. W3C Editing APIs CG.</dd>

   <dt id="refsEXIF">[EXIF]</dt>
   <dd>(Non-normative) <cite><a href="https://www.jeita.or.jp/cgi-bin/standard_e/list.cgi?cateid=1&subcateid=4">Exchangeable image file format</a></cite>. JEITA.</dd>

   <dt id="refsFETCH">[FETCH]</dt>
   <dd><cite><a href="https://fetch.spec.whatwg.org/">Fetch</a></cite>, A. van Kesteren. WHATWG.</dd>

   <dt id="refsFETCHMETADATA">[FETCH-METADATA]</dt>
   <dd><cite><a href="https://w3c.github.io/webappsec-fetch-metadata/">Fetch Metadata Request Headers</a></cite>, M.West. W3C.</dd>

   <dt id="refsFILEAPI">[FILEAPI]</dt>
   <dd><cite><a href="https://w3c.github.io/FileAPI/">File API</a></cite>, A. Ranganathan. W3C.</dd>

   <dt id="refsFILTERS">[FILTERS]</dt>
   <dd><cite><a href="https://drafts.fxtf.org/filter-effects/">Filter Effects</a></cite>, D. Schulze, D. Jackson, C. Harrelson. W3C.</dd>

   <dt id="refsFULLSCREEN">[FULLSCREEN]</dt>
   <dd><cite><a href="https://fullscreen.spec.whatwg.org/">Fullscreen</a></cite>, A. van Kesteren, T. &Ccedil;elik. WHATWG.</dd>

   <dt id="refsGEOMETRY">[GEOMETRY]</dt>
   <dd><cite><a href="https://drafts.fxtf.org/geometry/">Geometry Interfaces</a></cite>. S. Pieters, D. Schulze, R. Cabanier. W3C.</dd>

   <dt id="refsGIF">[GIF]</dt>
   <dd>(Non-normative) <cite><a href="https://www.w3.org/Graphics/GIF/spec-gif89a.txt">Graphics Interchange Format</a></cite>. CompuServe.</dd>

   <dt id="refsGRAPHICS">[GRAPHICS]</dt>
   <dd>(Non-normative) <cite>Computer Graphics: Principles and Practice in C</cite>, Second Edition, J. Foley, A. van Dam, S. Feiner, J. Hughes. Addison-Wesley. ISBN 0-201-84840-6.</dd>
   <!--
   This book ("Computer Graphics: Principles and Practice in C") apparently does not make any
   references to literature in the bibliographic section to define the "even-odd" rule for polygon
   filling and hit testing. In the absence of such a reference, I guess that this book is the most
   authoritative reference.
   -->

   <dt id="refsGREGORIAN">[GREGORIAN]</dt>
   <dd>(Non-normative) <cite>Inter Gravissimas</cite>, A. Lilius, C. Clavius. Gregory XIII Papal Bull, February 1582.</dd>

   <dt id="refsHRT">[HRT]</dt>
   <dd><cite><a href="https://w3c.github.io/hr-time/">High Resolution Time</a></cite>, I. Grigorik, J. Simonsen, J. Mann. W3C.</dd>

   <dt id="refsHTMLAAM">[HTMLAAM]</dt>
   <dd><cite><a href="https://w3c.github.io/html-aam/">HTML Accessibility API Mappings 1.0</a></cite>, S. Faulkner, A. Surkov, S. O'Hara. W3C.</dd>

   <dt id="refsHTTP">[HTTP]</dt>
   <dd><cite><a href="https://httpwg.org/specs/rfc7230.html">Hypertext Transfer Protocol (HTTP/1.1): Message Syntax and Routing</a></cite>, R. Fielding, J. Reschke. IETF.</dd>
   <dd><cite><a href="https://httpwg.org/specs/rfc7231.html">Hypertext Transfer Protocol (HTTP/1.1): Semantics and Content</a></cite>, R. Fielding, J. Reschke. IETF.</dd>
   <dd><cite><a href="https://httpwg.org/specs/rfc7232.html">Hypertext Transfer Protocol (HTTP/1.1): Conditional Requests</a></cite>, R. Fielding, J. Reschke. IETF.</dd>
   <dd><cite><a href="https://httpwg.org/specs/rfc7233.html">Hypertext Transfer Protocol (HTTP/1.1): Range Requests</a></cite>, R. Fielding, Y. Lafon, J. Reschke. IETF.</dd>
   <dd><cite><a href="https://httpwg.org/specs/rfc7234.html">Hypertext Transfer Protocol (HTTP/1.1): Caching</a></cite>, R. Fielding, M. Nottingham, J. Reschke. IETF.</dd>
   <dd><cite><a href="https://httpwg.org/specs/rfc7235.html">Hypertext Transfer Protocol (HTTP/1.1): Authentication</a></cite>, R. Fielding, J. Reschke. IETF.</dd>

   <dt id="refsINDEXEDDB">[INDEXEDDB]</dt>
   <dd><cite><a href="https://w3c.github.io/IndexedDB/">Indexed Database API</a></cite>, A. Alabbas, J. Bell. W3C.</dd>

   <dt id="refsINBAND">[INBAND]</dt>
   <dd><cite><a href="https://dev.w3.org/html5/html-sourcing-inband-tracks/">Sourcing In-band Media Resource Tracks from Media Containers into HTML</a></cite>, S. Pfeiffer, B. Lund. W3C.</dd>

   <dt id="refsINFRA">[INFRA]</dt>
   <dd><cite><a href="https://infra.spec.whatwg.org/">Infra</a></cite>, A. van Kesteren, D. Denicola. WHATWG.</dd>

   <dt id="refsINTERSECTIONOBSERVER">[INTERSECTIONOBSERVER]</dt>
   <dd><cite><a href="https://w3c.github.io/IntersectionObserver/">Intersection Observer</a></cite>, S. Zager. W3C.</dd>

   <dt id="refsRESIZEOBSERVER">[RESIZEOBSERVER]</dt>
   <dd><cite><a href="https://drafts.csswg.org/resize-observer-1/">Resize Observer</a></cite>, O. Brufau, E. Álvarez. W3C.</dd>

   <dt id="refsISO3166">[ISO3166]</dt>
   <dd><cite><a href="https://www.iso.org/iso-3166-country-codes.html">ISO 3166: Codes for the representation of names of countries and their subdivisions</a></cite>. ISO.</dd>

   <dt id="refsISO4217">[ISO4217]</dt>
   <dd><cite><a href="https://www.iso.org/iso-4217-currency-codes.html">ISO 4217: Codes for the representation of currencies and funds</a></cite>. ISO.</dd>

   <dt id="refsISO8601">[ISO8601]</dt>
   <dd>(Non-normative) <cite><a href="http://isotc.iso.org/livelink/livelink/4021199/ISO_8601_2004_E.zip?func=doc.Fetch&amp;nodeid=4021199">ISO8601: Data elements and interchange formats &mdash; Information interchange &mdash; Representation of dates and times</a></cite>. ISO.</dd>

   <dt id="refsJAVASCRIPT">[JAVASCRIPT]</dt>
   <dd><cite><a href="https://tc39.es/ecma262/">ECMAScript Language Specification</a></cite>. Ecma International.</dd>

   <dt id="refsJLREQ">[JLREQ]</dt>
   <dd><cite><a href="https://www.w3.org/TR/jlreq/">Requirements for Japanese Text Layout</a></cite>. W3C.</dd> <!-- too many editors to list -->

   <dt id="refsJPEG">[JPEG]</dt>
   <dd><cite><a href="https://www.w3.org/Graphics/JPEG/jfif3.pdf">JPEG File Interchange Format</a></cite>, E. Hamilton.</dd>

   <dt id="refsJSERRORSTACKS">[JSERRORSTACKS]</dt>
   <dd>(Non-normative) <cite><a href="https://tc39.es/proposal-error-stacks/">Error Stacks</a></cite>. Ecma International.</dd>

   <dt id="refsJSDYNAMICCODEBRANDCHECKS">[JSDYNAMICCODEBRANDCHECKS]</dt>
   <dd><cite><a href="https://tc39.es/proposal-dynamic-code-brand-checks/">Dynamic code brand checks</a></cite>. Ecma International.</dd>

   <dt id="refsJSINTL">[JSINTL]</dt>
   <dd><cite><a href="https://tc39.es/ecma402/">ECMAScript Internationalization API Specification</a></cite>. Ecma International.</dd>

   <dt id="refsJSON">[JSON]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc8259">The JavaScript Object Notation (JSON) Data Interchange Format</a></cite>, T. Bray. IETF.</dd>

   <dt id="refsJSTEMPORAL">[JSTEMPORAL]</dt>
   <dd><cite><a href="https://tc39.es/proposal-temporal/">Temporal</a></cite>. Ecma International.</dd>

   <dt id="refsLONGTASKS">[LONGTASKS]</dt>
   <dd><cite><a href="https://w3c.github.io/longtasks/">Long Tasks</a></cite>, D. Denicola, I. Grigorik, S. Panicker. W3C.</dd>

   <dt id="refsLONGANIMATIONFRAMES">[LONGANIMATIONFRAMES]</dt>
   <dd><cite><a href="https://w3c.github.io/long-animation-frames/">Long Animation Frames</a></cite>, N. Rosenthal. W3C.</dd>

   <dt id="refsMAILTO">[MAILTO]</dt>
   <dd>(Non-normative) <cite><a href="https://www.rfc-editor.org/rfc/rfc6068">The 'mailto' URI scheme</a></cite>, M. Duerst, L. Masinter, J. Zawinski. IETF.</dd>

   <dt id="refsMANIFEST">[MANIFEST]</dt>
   <dd><cite><a href="https://www.w3.org/TR/appmanifest/">Web App Manifest</a></cite>, M. Caceres, K. Rohde Christiansen, M. Lamouri, A. Kostiainen, M. Giuca, A. Gustafson. W3C.</dd>

   <dt id="refsMATHML">[MATHMLCORE]</dt>
   <dd><cite><a href="https://w3c.github.io/mathml-core/">Mathematical Markup Language (MathML)</a></cite>, D. Carlisle, Frédéric Wang. W3C.</dd>

   <dt id="refsMEDIAFRAG">[MEDIAFRAG]</dt>
   <dd><cite><a href="https://www.w3.org/TR/media-frags/">Media Fragments URI</a></cite>, R. Troncy, E. Mannens, S. Pfeiffer, D. Van Deursen. W3C.</dd>

   <dt id="refsMEDIASOURCE">[MEDIASOURCE]</dt>
   <dd><cite><a href="https://w3c.github.io/media-source/">Media Source Extensions</a></cite>, A. Colwell, A. Bateman, M. Watson. W3C.</dd>

   <dt id="refsMEDIASTREAM">[MEDIASTREAM]</dt>
   <dd><cite><a href="https://w3c.github.io/mediacapture-main/getusermedia.html">Media Capture and Streams</a></cite>, D. Burnett, A. Bergkvist, C. Jennings, A. Narayanan. W3C.</dd>

   <dt id="refsREPORTING">[REPORTING]</dt>
   <dd><cite><a href="https://w3c.github.io/reporting/">Reporting</a></cite>, D. Creager, I. Clelland, M. West. W3C.</dd>

   <dt id="refsMFREL">[MFREL]</dt>
   <dd><cite><a href="https://microformats.org/wiki/existing-rel-values#HTML5_link_type_extensions">Microformats Wiki: existing rel values</a></cite>. Microformats.</dd>

   <dt id="refsMIMESNIFF">[MIMESNIFF]</dt>
   <dd><cite><a href="https://mimesniff.spec.whatwg.org/">MIME Sniffing</a></cite>, G. Hemsley. WHATWG.</dd>

   <dt id="refsMIX">[MIX]</dt>
   <dd><cite><a href="https://w3c.github.io/webappsec-mixed-content/">Mixed Content</a></cite>, M. West. W3C.</dd>

   <dt id="refsMNG">[MNG]</dt>
   <dd><cite><a href="http://www.libpng.org/pub/mng/spec/">MNG (Multiple-image Network Graphics) Format</a></cite>. G. Randers-Pehrson.</dd>

   <dt id="refsMPEG2">[MPEG2]</dt>
   <dd><cite>ISO/IEC 13818-1: Information technology &mdash; Generic coding of moving pictures and associated audio information: Systems</cite>. ISO/IEC.</dd>
   <!-- search for ["bytes are removed from this buffer at a rate defined by sb_leak_rate"] to find it -->

   <dt id="refsMPEG4">[MPEG4]</dt>
   <dd><cite>ISO/IEC 14496-12: ISO base media file format</cite>. ISO/IEC.</dd>
   <!-- search for ["Box Structure was and subsequent clauses were re-organized"] to find it -->

   <dt id="refsMQ">[MQ]</dt>
   <dd><cite><a href="https://drafts.csswg.org/mediaqueries/">Media Queries</a></cite>, H. Lie, T. &Ccedil;elik, D. Glazman, A. van Kesteren. W3C.</dd>

   <dt id="refsMULTIPLEBUFFERING">[MULTIPLEBUFFERING]</dt>
   <dd>(Non-normative) <cite><a href="https://en.wikipedia.org/wiki/Multiple_buffering">Multiple buffering</a></cite>. Wikipedia.</dd>

   <dt id="refsNAVIGATIONTIMING">[NAVIGATIONTIMING]</dt>
   <dd><cite><a href="https://w3c.github.io/navigation-timing/">Navigation Timing</a></cite>, Y. Weiss. W3C.</dd>

   <dt id="refsNPAPI">[NPAPI]</dt>
   <dd>(Non-normative) <cite><a href="https://developer.mozilla.org/en-US/docs/Plugins/Guide">Gecko Plugin API Reference</a></cite>. Mozilla.</dd>

   <dt id="refsOGGSKELETONHEADERS">[OGGSKELETONHEADERS]</dt>
   <dd><cite><a href="https://wiki.xiph.org/SkeletonHeaders">SkeletonHeaders</a></cite>. Xiph.Org.</dd>

   <dt id="refsOPENSEARCH">[OPENSEARCH]</dt>
   <dd><cite><a href="https://github.com/dewitt/opensearch/blob/master/opensearch-1-1-draft-6.md#autodiscovery-in-htmlxhtml">Autodiscovery in HTML/XHTML</a></cite>. In <cite>OpenSearch 1.1 Draft 6</cite>. GitHub.</dd>

   <dt id="refsORIGIN">[ORIGIN]</dt>
   <dd>(Non-normative) <cite><a href="https://www.rfc-editor.org/rfc/rfc6454">The Web Origin Concept</a></cite>, A. Barth. IETF.</dd>

   <dt id="refsPAINTTIMING">[PAINTTIMING]</dt>
   <dd><cite><a href="https://w3c.github.io/paint-timing/">Paint Timing</a></cite>, S. Panicker. W3C.</dd>

   <dt id="refsPAYMENTREQUEST">[PAYMENTREQUEST]</dt>
   <dd><cite><a href="https://w3c.github.io/payment-request/">Payment Request API</a></cite>, M. Cáceres, D. Wang, R. Solomakhin, I. Jacobs. W3C.</dd>

   <dt id="refsPDF">[PDF]</dt>
   <dd>(Non-normative) <cite><a href="https://www.adobe.com/content/dam/Adobe/en/devnet/acrobat/pdfs/PDF32000_2008.pdf">Document management &mdash; Portable document format &mdash; Part 1: PDF</a></cite>. ISO.</dd>

   <dt id="refsPERFORMANCETIMELINE">[PERFORMANCETIMELINE]</dt>
   <dd><cite><a href="https://w3c.github.io/performance-timeline/">Performance Timeline</a></cite>, N. Peña Moreno, W3C.</dd>

   <dt id="refsPERMISSIONSPOLICY">[PERMISSIONSPOLICY]</dt>
   <dd><cite><a href="https://w3c.github.io/webappsec-feature-policy/">Permissions Policy</a></cite>, I. Clelland, W3C.</dd>

   <dt id="refsPICTUREINPICTURE">[PICTUREINPICTURE]</dt>
   <dd>(Non-normative) <cite><a href="https://w3c.github.io/picture-in-picture/">Picture-in-Picture</a></cite>, F. Beaufort, M. Lamouri, W3C</dd>

   <dt id="refsPINGBACK">[PINGBACK]</dt>
   <dd><cite><a href="https://www.hixie.ch/specs/pingback/pingback">Pingback 1.0</a></cite>, S. Langridge, I. Hickson.</dd>

   <dt id="refsPNG">[PNG]</dt>
   <dd><cite><a href="https://www.w3.org/TR/PNG/">Portable Network Graphics (PNG) Specification</a></cite>, D. Duce. W3C.</dd>

   <dt id="refsPOINTEREVENTS">[POINTEREVENTS]</dt>
   <dd><cite><a href="https://w3c.github.io/pointerevents/">Pointer Events</a></cite>, J. Rossi, M. Brubeck, R. Byers, P. H. Lauke. W3C.</dd>

   <dt id="refsPOINTERLOCK">[POINTERLOCK]</dt>
   <dd><cite><a href="https://w3c.github.io/pointerlock/">Pointer Lock</a></cite>, V. Scheib. W3C.</dd>

   <dt id="refsPPUTF8">[PPUTF8]</dt>
   <dd>(Non-normative) <cite><a href="https://www.sw.it.aoyama.ac.jp/2012/pub/IUC11-UTF-8.pdf">The Properties and Promises <!-- Promizes (sic) --> of UTF-8</a></cite>, M. D&uuml;rst. University of Z&uuml;rich. In <cite>Proceedings of the 11th International Unicode Conference</cite>.</dd>

   <dt id="refsPRESENTATION">[PRESENTATION]</dt>
   <dd><cite><a href="https://w3c.github.io/presentation-api/">Presentation API</a></cite>, M. Foltz, D. Röttsches. W3C.</dd>

   <dt id="refsREFERRERPOLICY">[REFERRERPOLICY]</dt>
   <dd><cite><a href="https://w3c.github.io/webappsec-referrer-policy/">Referrer Policy</a></cite>, J. Eisinger, E. Stark. W3C.</dd>

   <dt id="refsREQUESTIDLECALLBACK">[REQUESTIDLECALLBACK]</dt>
   <dd><cite><a href="https://w3c.github.io/requestidlecallback/">Cooperative Scheduling of Background Tasks</a></cite>, R. McIlroy, I. Grigorik. W3C.</dd>

   <dt id="refsRESOURCETIMING">[RESOURCETIMING]</dt>
   <dd><cite><a href="https://w3c.github.io/resource-timing/">Resource Timing</a></cite>, Yoav Weiss; Noam Rosenthal. W3C.</dd>

   <dt id="refsRFC1034">[RFC1034]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc1034">Domain Names - Concepts and Facilities</a></cite>, P. Mockapetris. IETF, November 1987.</dd>

   <dt id="refsRFC1123">[RFC1123]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc1123">Requirements for Internet Hosts -- Application and Support</a></cite>, R. Braden. IETF, October 1989.</dd>

   <dt id="refsRFC2046">[RFC2046]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc2046">Multipurpose Internet Mail Extensions (MIME) Part Two: Media Types</a></cite>, N. Freed, N.  Borenstein. IETF.</dd> <!-- for text/plain and "Internet Media type"; not for definition of "MIME type". -->

   <dt id="refsRFC2397">[RFC2397]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc2397">The "data" URL scheme</a></cite>, L. Masinter. IETF.</dd>

   <dt id="refsRFC5545">[RFC5545]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc5545">Internet Calendaring and Scheduling Core Object Specification (iCalendar)</a></cite>, B. Desruisseaux. IETF.</dd>

   <dt id="refsRFC2483">[RFC2483]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc2483">URI Resolution Services Necessary for URN Resolution</a></cite>, M. Mealling, R. Daniel. IETF.</dd>

   <dt id="refsRFC3676">[RFC3676]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc3676">The Text/Plain Format and DelSp Parameters</a></cite>, R. Gellens. IETF.</dd>

   <dt id="refsRFC9239">[RFC9239]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc9239">Updates to ECMAScript Media Types</a></cite>, M. Miller, M. Borins, M. Bynens, B. Farias. IETF.</dd>

   <dt id="refsRFC4337">[RFC4337]</dt>
   <dd>(Non-normative) <cite><a href="https://www.rfc-editor.org/rfc/rfc4337">MIME Type Registration for MPEG-4</a></cite>, Y. Lim, D. Singer. IETF.</dd>

   <dt id="refsRFC7595">[RFC7595]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc7595">Guidelines and Registration Procedures for URI Schemes</a></cite>, D. Thaler, T. Hansen, T. Hardie. IETF.</dd>

   <dt id="refsRFC5322">[RFC5322]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc5322">Internet Message Format</a></cite>, P. Resnick. IETF.</dd>

   <dt id="refsRFC6381">[RFC6381]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc6381">The 'Codecs' and 'Profiles' Parameters for "Bucket" Media Types</a></cite>, R. Gellens, D. Singer, P. Frojdh. IETF.</dd>

   <dt id="refsRFC6266">[RFC6266]</dt>
   <dd><cite><a href="https://httpwg.org/specs/rfc6266.html">Use of the Content-Disposition Header Field in the Hypertext Transfer Protocol (HTTP)</a></cite>, J. Reschke. IETF.</dd>

   <dt id="refsRFC6350">[RFC6350]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc6350">vCard Format Specification</a></cite>, S. Perreault. IETF.</dd>

   <dt id="refsRFC6596">[RFC6596]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc6596">The Canonical Link Relation</a></cite>, M. Ohye, J. Kupke. IETF.</dd>

   <dt id="refsRFC6903">[RFC6903]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc6903">Additional Link Relation Types</a></cite>, J. Snell. IETF.</dd>

   <dt id="refsRFC7034">[RFC7034]</dt>
   <dd>(Non-normative) <cite><a href="https://www.rfc-editor.org/rfc/rfc7034">HTTP Header Field X-Frame-Options</a></cite>, D. Ross, T. Gondrom. IETF.</dd>

   <dt id="refsRFC7303">[RFC7303]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc7303">XML Media Types</a></cite>, H. Thompson, C. Lilley. IETF.</dd>

   <dt id="refsRFC7578">[RFC7578]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc7578">Returning Values from Forms: multipart/form-data</a></cite>, L. Masinter. IETF.</dd>

   <dt id="refsRFC8297">[RFC8297]</dt>
   <dd><cite><a href="https://httpwg.org/specs/rfc8297.html">An HTTP Status Code for Indicating Hints</a></cite>, K. Oku. IETF.</dd>

   <dt id="refsSCREENORIENTATION">[SCREENORIENTATION]</dt>
   <dd><cite><a href="https://w3c.github.io/screen-orientation/">Screen Orientation</a></cite>, M. Cáceres. W3C.</dd>

   <dt id="refsSCSU">[SCSU]</dt>
   <dd>(Non-normative) <cite><a href="https://www.unicode.org/reports/tr6/">UTR #6: A Standard Compression Scheme For Unicode</a></cite>, M. Wolf, K. Whistler, C. Wicksteed, M. Davis, A. Freytag, M. Scherer. Unicode Consortium.</dd>

   <dt id="refsSECURE-CONTEXTS">[SECURE-CONTEXTS]</dt>
   <dd><cite><a href="https://w3c.github.io/webappsec-secure-contexts/">Secure Contexts</a></cite>, M. West. W3C.</dd>

   <dt id="refsSELECTION">[SELECTION]</dt>
   <dd><cite><a href="https://w3c.github.io/selection-api/">Selection API</a></cite>, R. Niwa. W3C.</dd>

   <dt id="refsSELECTORS">[SELECTORS]</dt>
   <dd><cite><a href="https://drafts.csswg.org/selectors/">Selectors</a></cite>, E. Etemad, T. &Ccedil;elik, D. Glazman, I. Hickson, P. Linss, J. Williams. W3C.</dd>

   <dt id="refsSMS">[SMS]</dt>
   <dd>(Non-normative) <cite><a href="https://www.rfc-editor.org/rfc/rfc5724">URI Scheme for Global System for Mobile Communications (GSM) Short Message Service (SMS)</a></cite>, E. Wilde, A. Vaha-Sipila. IETF.</dd>

   <dt id="refsSTRUCTURED-FIELDS">[STRUCTURED-FIELDS]</dt>
   <dd><cite><a href="https://httpwg.org/specs/rfc8941.html">Structured Field Values for HTTP</a></cite>, M. Nottingham, P-H. Kamp. IETF.</dd>

   <dt id="refsSRI">[SRI]</dt>
   <dd><cite><a href="https://w3c.github.io/webappsec-subresource-integrity/">Subresource Integrity</a></cite>, D. Akhawe, F. Braun, F. Marier, J. Weinberger. W3C.</dd>

   <dt id="refsSTORAGE">[STORAGE]</dt>
   <dd><cite><a href="https://storage.spec.whatwg.org/">Storage</a></cite>, A. van Kesteren. WHATWG.</dd>

   <dt id="refsSVG">[SVG]</dt>
   <dd><cite><a href="https://svgwg.org/svg2-draft/">Scalable Vector Graphics (SVG) 2</a></cite>, N Andronikos, R. Atanassov, T. Bah, B. Birtles, B. Brinza, C. Concolato, E. Dahlström, C. Lilley, C. McCormack, D. Schepers, R. Schwerdtfeger, D. Storey, S. Takagi, J. Watt. W3C.</dd>

   <dt id="refsSW">[SW]</dt>
   <dd><cite><a href="https://w3c.github.io/ServiceWorker/">Service Workers</a></cite>, A. Russell, J. Song, J. Archibald. W3C.</dd>

   <dt id="refsTOR">[TOR]</dt>
   <dd>(Non-normative) <cite><a href="https://www.torproject.org/">Tor</a></cite>.</dd>

   <dt id="refsTOUCH">[TOUCH]</dt>
   <dd><cite><a href="https://w3c.github.io/touch-events/">Touch Events</a></cite>, D. Schepers, S. Moon, M. Brubeck, A. Barstow, R. Byers. W3C.</dd>

   <dt id="refsTRUSTED-TYPES">[TRUSTED-TYPES]</dt>
   <dd><cite><a href="https://w3c.github.io/trusted-types/dist/spec/">Trusted Types</a></cite>, K. Kotowicz, M. West. W3C.</dd>

   <dt id="refsTZDATABASE">[TZDATABASE]</dt>
   <dd>(Non-normative) <cite><a href="https://www.iana.org/time-zones">Time Zone Database</a></cite>. IANA.</dd>

   <dt id="refsUAAG">[UAAG]</dt>
   <dd>(Non-normative) <cite><a href="https://www.w3.org/TR/UAAG20/">User Agent Accessibility Guidelines (UAAG) 2.0</a></cite>, J. Allan, K. Ford, J. Richards, J. Spellman. W3C.</dd>

   <dt id="refsUIEVENTS">[UIEVENTS]</dt>
   <dd><cite><a href="https://w3c.github.io/uievents/">UI Events Specification</a></cite>, G. Kacmarcik, T. Leithead. W3C.</dd>

<!--(the above references to this are commented out)
   <dt id="refsUNDO">[UNDO]</dt>
   <dd><cite><a href="https://dvcs.w3.org/hg/undomanager/raw-file/tip/undomanager.html">UndoManager and DOM Transaction</a></cite>, R. Niwa.</dd>
-->

   <dt id="refsUNICODE">[UNICODE]</dt>
   <dd><cite><a href="https://www.unicode.org/versions/">The Unicode Standard</a></cite>. Unicode Consortium.</dd>

   <dt id="refsUNIVCHARDET">[UNIVCHARDET]</dt>
   <dd>(Non-normative) <cite><a href="https://www-archive.mozilla.org/projects/intl/UniversalCharsetDetection.html">A composite approach to language/encoding detection</a></cite>, S. Li, K. Momoi. Netscape. In <cite>Proceedings of the 19th International Unicode Conference</cite>.</dd>

   <dt id="refsURL">[URL]</dt>
   <dd><cite><a href="https://url.spec.whatwg.org/">URL</a></cite>, A. van Kesteren. WHATWG.</dd>

   <dt id="refsURN">[URN]</dt>
   <dd><cite><a href="https://www.rfc-editor.org/rfc/rfc2141">URN Syntax</a></cite>, R. Moats. IETF.</dd>

   <dt id="refsUTF7">[UTF7]</dt>
   <dd>(Non-normative) <cite><a href="https://www.rfc-editor.org/rfc/rfc2152">UTF-7: A Mail-Safe Transformation Format of Unicode</a></cite>, D. Goldsmith, M. Davis. IETF.</dd>

   <dt id="refsUTF8DET">[UTF8DET]</dt>
   <dd>(Non-normative) <cite><a href="https://www.w3.org/International/questions/qa-forms-utf-8">Multilingual form encoding</a></cite>, M. D&uuml;rst. W3C.</dd>

   <dt id="refsUTR36">[UTR36]</dt>
   <dd>(Non-normative) <cite><a href="https://www.unicode.org/reports/tr36/">UTR #36: Unicode Security Considerations</a></cite>, M. Davis, M. Suignard. Unicode Consortium.</dd>

   <dt id="refsWASM">[WASM]</dt>
   <dd><cite><a href="https://webassembly.github.io/spec/core/bikeshed/index.html">WebAssembly Core Specification</a></cite>, A. Rossberg. W3C.</dd>

   <dt id="refsWASMESM">[WASMESM]</dt>
   <dd><cite><a href="https://webassembly.github.io/esm-integration/js-api/index.html">WebAssembly JavaScript Interface: ESM Integration</a></cite>, L. Clark, D. Ehrenberg., A. Takikawa., G. Bedford. W3C.</dd>

   <dt id="refsWASMJS">[WASMJS]</dt>
   <dd>(Non-normative) <cite><a href="https://webassembly.github.io/spec/js-api/">WebAssembly JavaScript Interface</a></cite>, D. Ehrenberg. W3C.</dd>

   <dt id="refsWCAG">[WCAG]</dt>
   <dd>(Non-normative) <cite><a href="https://w3c.github.io/wcag/guidelines/">Web Content Accessibility Guidelines (WCAG)</a></cite>, A. Kirkpatrick, J. O Connor, A. Campbell, M. Cooper. W3C.</dd>

   <dt id="refsWEBANIMATIONS">[WEBANIMATIONS]</dt>
   <dd><cite><a href="https://drafts.csswg.org/web-animations-1/">Web Animations</a></cite>, B. Birtles, S. Stephens, D. Stockwell. W3C.</dd>

   <dt id="refsWEBAUDIO">[WEBAUDIO]</dt>
   <dd>(Non-normative) <cite><a href="https://webaudio.github.io/web-audio-api/">Web Audio API</a></cite>, P. Adenot, H. Choi. W3C.</dd>

   <dt id="refsWEBAUTHN">[WEBAUTHN]</dt>
   <dd><cite><a href="https://w3c.github.io/webauthn">Web Authentication: An API for accessing Public Key Credentials</a></cite>, M. Jones, A. Kumar, E. Lundberg, D. Balfanz, V. Bharadwaj, A. Birgisson, A. Czeskis, J. Hodges, J.C. Jones, H. Le Van Gong, A. Liao, R. Lindemann, J. Bradley, C. Brand, T. Cappalli, A. Langley, G. Mandyam, M. Miller, N. Satragno, N. Steele, J. Tan, S. Weeden, M. West, J. Yasskin. W3C.</dd>

   <dt id="refsWEBCODECS">[WEBCODECS]</dt>
   <dd><cite><a href="https://w3c.github.io/webcodecs/">WebCodecs API</a></cite>, C. Cunningham, P. Adenot, B. Aboba. W3C.</dd>

   <dt id="refsWEBCRYPTO">[WEBCRYPTO]</dt>
   <dd><cite><a href="https://w3c.github.io/webcrypto/Overview.html">Web Cryptography API</a></cite>, D. Huigens. W3C.</dd>

   <dt id="refsWEBDRIVER">[WEBDRIVER]</dt>
   <dd><cite><a href="https://w3c.github.io/webdriver/">WebDriver</a></cite>, S. Stewart, D. Burns. W3C.</dd>

   <dt id="refsWEBDRIVERBIDI">[WEBDRIVERBIDI]</dt>
   <dd><cite><a href="https://w3c.github.io/webdriver-bidi/">WebDriver BiDi</a></cite>. W3C</dd>

   <dt id="refsWEBGL">[WEBGL]</dt>
   <dd><cite><a href="https://www.khronos.org/registry/webgl/specs/latest/">WebGL Specifications</a></cite>, D. Jackson, J. Gilbert. Khronos Group.</dd>

   <dt id="refsWEBGPU">[WEBGPU]</dt>
   <dd><cite><a href="https://gpuweb.github.io/gpuweb/">WebGPU</a></cite>, D. Malyshau, K. Ninomiya. W3C.</dd>

   <dt id="refsWEBIDL">[WEBIDL]</dt>
   <dd><cite><a href="https://webidl.spec.whatwg.org/">Web IDL</a></cite>, E. Chen, T. Gu. WHATWG.</dd>

   <dt id="refsWEBLINK">[WEBLINK]</dt>
   <dd><cite><a href="https://httpwg.org/specs/rfc8288.html">Web Linking</a></cite>, M. Nottingham. IETF.</dd>

   <dt id="refsWEBLOCKS">[WEBLOCKS]</dt>
   <dd>(Non-normative) <cite><a href="https://w3c.github.io/web-locks/">Web Locks API</a></cite>, J. Bell, K. Rosylight. W3C.</dd>

   <dt id="refsWEBMCG">[WEBMCG]</dt>
   <dd><cite><a href="https://www.webmproject.org/docs/container/">WebM Container Guidelines</a></cite>. The WebM Project.</dd>

   <dt id="refsWEBNFC">[WEBNFC]</dt>
   <dd>(Non-normative) <cite><a href="https://w3c.github.io/web-nfc/">Web NFC</a></cite>, F. Beaufort, K. Christiansen, Z. Kis. W3C.</dd>

   <dt id="refsWEBRTC">[WEBRTC]</dt>
   <dd>(Non-normative) <cite><a href="https://w3c.github.io/webrtc-pc/">Web RTC</a></cite>, C. Jennings, F. Castelli, H. Boström, J. Bruaroey. W3C.</dd>

   <dt id="refsWEBSOCKETS">[WEBSOCKETS]</dt>
   <dd><cite><a href="https://websockets.spec.whatwg.org/">WebSockets</a></cite>, A. Rice. WHATWG.</dd>

   <dt id="refsWEBTRANSPORT">[WEBTRANSPORT]</dt>
   <dd><cite><a href="https://w3c.github.io/webtransport/">WebTransport</a></cite>, B. Aboba, N. Jaju, V. Vasiliev. W3C.</dd>

   <dt id="refsWEBVTT">[WEBVTT]</dt>
   <dd><cite><a href="https://w3c.github.io/webvtt/">WebVTT</a></cite>, S. Pieters. W3C.</dd>

   <dt id="refsWHATWGWIKI">[WHATWGWIKI]</dt>
   <dd><cite><a href="https://wiki.whatwg.org/">The WHATWG Wiki</a></cite>. WHATWG.</dd>

   <dt id="refsX121">[X121]</dt>
   <dd><cite>Recommendation X.121 &mdash; International Numbering Plan for Public Data Networks</cite>, CCITT Blue Book, Fascicle VIII.3, pp. 317-332.</dd>

   <dt id="refsXFN">[XFN]</dt>
   <dd><cite><a href="https://gmpg.org/xfn/11">XFN 1.1 profile</a></cite>, T. &Ccedil;elik, M. Mullenweg, E. Meyer. GMPG.</dd>

   <dt id="refsXHR">[XHR]</dt>
   <dd><cite><a href="https://xhr.spec.whatwg.org/"><code>XMLHttpRequest</code></a></cite>, A. van Kesteren. WHATWG.</dd>

   <dt id="refsXKCD1288">[XKCD1288]</dt>
   <dd>(Non-normative) <cite><a href="https://xkcd.com/1288/">Substitutions</a></cite>, Randall Munroe. xkcd.</dd>

   <dt id="refsXML">[XML]</dt>
   <dd><cite><a href="https://www.w3.org/TR/xml/">Extensible Markup Language</a></cite>, T. Bray, J. Paoli, C. Sperberg-McQueen, E. Maler, F. Yergeau. W3C.</dd>

   <dt id="refsXMLENTITY">[XMLENTITY]</dt>
   <dd>(Non-normative) <cite><a href="https://www.w3.org/2003/entities/2007doc/">XML Entity Definitions for Characters</a></cite>, D. Carlisle, P. Ion. W3C.</dd>

   <dt id="refsXMLNS">[XMLNS]</dt>
   <dd><cite><a href="https://www.w3.org/TR/xml-names/">Namespaces in XML</a></cite>, T. Bray, D. Hollander, A. Layman, R. Tobin. W3C.</dd>

   <dt id="refsXMLSSPI">[XMLSSPI]</dt>
   <dd><cite><a href="https://www.w3.org/TR/xml-stylesheet/">Associating Style Sheets with XML documents</a></cite>, J. Clark, S. Pieters, H. Thompson. W3C.</dd>

   <dt id="refsXPATH10">[XPATH10]</dt>
   <dd><cite><a href="https://www.w3.org/TR/1999/REC-xpath-19991116/">XML Path Language (XPath) Version 1.0</a></cite>, J. Clark, S. DeRose. W3C.</dd>

   <dt id="refsXSLT10">[XSLT10]</dt>
   <dd>(Non-normative) <cite><a href="https://www.w3.org/TR/1999/REC-xslt-19991116">XSL Transformations (XSLT) Version 1.0</a></cite>, J. Clark. W3C.</dd>

   <dt id="refsXSLTP">[XSLTP]</dt>
   <dd>(Non-normative) <cite><a href="https://wiki.whatwg.org/wiki/DOM_XSLTProcessor">DOM XSLTProcessor</a></cite>, WHATWG Wiki. WHATWG.</dd>
  </dl>


  <h2 split-filename="acknowledgements" id="acknowledgments" class="no-num">Acknowledgments</h2>
  <!-- ACKS (the filename is in en-GB for historical reasons) -->

  <p>Thanks to Tim Berners-Lee for inventing HTML, without which none of this would exist.</p>

  <p>Thanks to

  Aankhen,
  Aaqa Ishtyaq,
  Aaron Boodman,
  Aaron Leventhal,
  Aaron Krajeski,
  Abhishek Ghaskata,
  Abhishek Gupta, <!-- a1626 on GitHub -->
  Adam Barth,
  Adam de Boor,
  Adam Hepton,
  Adam Klein,
  Adam Rice,
  Adam Roben,
  Addison Phillips,
  Adele Peterson,
  Adrian Bateman,
  Adrian Roselli,
  Adrian Sutton,
  Agust&iacute;n Fern&aacute;ndez,
  Aharon (Vladimir) Lanin,
  Ajai Tirumali,
  Ajay Poshak, <!-- ajayposhak on GitHub -->
  Akash Balenalli,
  Akatsuki Kitamura,
  Alan Jeffrey, <!-- asajeffrey on GitHub -->
  Alan Plum,
  Alastair Campbell,
  Alejandro G. Castro,
  Alex Bishop,
  Alex Nicolaou,
  Alex Nozdriukhin,
  Alex Rousskov,
  Alex Soncodi,
  Alexander Farkas,
  Alexander J. Vincent,
  Alexander Kalenik,
  Alexandre Dieulot,
  Alexandre Morgaut,
  Alexey Feldgendler,
  &#1040;&#1083;&#1077;&#1082;&#1089;&#1077;&#1081; &#1055;&#1088;&#1086;&#1089;&#1082;&#1091;&#1088;&#1103;&#1082;&#1086;&#1074; (Alexey Proskuryakov),
  Alexey Shvayka,
  Alexis Deveria,
  Alfred Agrell, <!-- Alcaro on GitHub -->
  Ali Juma,
  Alice Boxhall,
  Alice Wonder,
  Allan Clements,
  Allen Wirfs-Brock,
  Alex Komoroske,
  Alex Russell,
  Alphan Chen,
  Aman Ansari, <!-- aman-atg on Github -->
  Ami Fischman,
  Amos Jeffries,
  Amos Lim,
  Anders Carlsson,
  André Bargull,
  Andr&eacute; E. Veltstra,
  Andrea Rendine,
  Andreas<!-- mqmq87 -->,
  Andreas Deuschlinger, <!-- AndyOGo on GitHub -->
  Andreas Farre,
  Andreas Kling,
  Andrei Popescu,
  Andres Gomez,
  Andres Rios,
  Andreu Botella,
  Andrew Barfield,
  Andrew Clover,
  Andrew Gove,
  Andrew Grieve,
  Andrew Kaster,
  Andrew Macpherson, <!-- fuzzbomb on GitHub -->
  Andrew Oakley,
  Andrew Paseltiner,
  Andrew Simons,
  Andrew Smith,
  Andrew W. Hagen,
  Andrew Williams,
  Andrey V. Lukyanov,
  Andry Rendy,
  Andy Davies,
  Andy Earnshaw,
  Andy Heydon,
  Andy Paicu,
  Andy Palay,
  Anjana Vakil,
  Ankur Kaushal,
  Anna Belle Leiserson,
  Anna Sidwell,
  Anthony Boyd,
  Anthony Bryan,
  Anthony Hickson,
  Anthony Ramine,
  Anthony Ricaud,
  Anton Vayvod,
  Antonio Sartori,
  Antti Koivisto,
  Arfat Salman,
  Arkadiusz Michalski,
  Arne Thomassen,
  Aron Spohr,
  Arphen Lin,
  Arthur Hemery,
  Arthur Sonzogni, <!-- ArthurSonzogni on GitHub -->
  Arthur Stolyar,
  Arun Patole,
  Aryeh Gregor,
  Asanka Herath,
  Asbj&oslash;rn Ulsberg,
  Ashley Gullen,
  Ashley Sheridan,
  Asumu Takikawa,
  Atsushi Takayama,
  Attila Haraszti,
  Aurelien Levy,
  Ave Wrigley,
  Avi Drissman,
  Axel Dahmen,
  방성범 (Bang Seongbeom), <!-- bangseongbeom on GitHub -->
  Barry Pollard, <!-- tunetheweb on GitHub -->
  Ben Boyle,
  Ben Godfrey,
  Ben Golightly, <!-- golightlyb on GitHub -->
  Ben Kelly,
  Ben Lerner,
  Ben Leslie,
  Ben Meadowcroft,
  Ben Millard,
  Benjamin Carl Wiley Sittler,
  Benjamin Hawkes-Lewis,
  Benji Bilheimer, <!-- benjibee on GitHub -->
  Benoit Ren,
  Bert Bos,
  Bijan Parsia,
  Bil Corry,
  Bill Mason,
  Bill McCoy,
  Billy Wong,
  Billy Woods,
  Bjartur Thorlacius,
  Bj&ouml;rn H&ouml;hrmann,
  Blake Frantz,
  Bob Lund,
  Bob Owen,
  Bobby Holley,
  Boris Zbarsky,
  Brad Fults,
  Brad Neuberg,
  Brad Spencer,
  Bradley Meck,
  Brady Eidson,
  Brandon Jones,
  Brendan Eich,
  Brenton Simpson,
  Brett Wilson,
  Brett Zamir,
  Brian Birtles,
  Brian Blakely,
  Brian Campbell,
  Brian Korver,
  Brian Kuhn,
  Brian M. Dube,
  Brian Ryner,
  Brian Smith,
  Brian Wilson,
  Bryan Sullivan,
  Bruce Bailey,
  Bruce D'Arcus,
  Bruce Lawson,
  Bruce Miller,
  Bugs Nash,
  C. Scott Ananian,
  C. Williams,
  Cameron McCormack,
  Cameron Zemek,
  Cao Yipeng,
  Carlos Amengual,
  Carlos Gabriel Cardona,
  Carlos Ibarra L&oacute;pez, <!-- carlosjoan91 on GitHub -->
  Carlos Perell&oacute; Mar&iacute;n,
  Carolyn MacLeod,
  Casey Leask,
  Cătălin Badea,
  C&#x103;t&#x103;lin Mari&#x219;,
  Cem Turesoy, <!-- ctur on GitHub -->
  ceving, <!-- GitHub -->
  Chao Cai,
  &#xc724;&#xc11d;&#xcc2c; (Channy Yun),
  Charl van Niekerk,
  Charlene Wright,
  Charles Iliya Krempeaux,
  Charles McCathie Nevile,
  Charlie Reis,
  白丞祐 (Cheng-You Bai), <!-- CYBAI on GitHub -->
  Chris Apers,
  Chris Cressman,
  Chris Dumez,
  Chris Evans,
  Chris Harrelson,
  Chris Markiewicz,
  Chris Morris,
  Chris Nardi,
  Chris Needham,
  Chris Pearce,
  Chris Peterson,
  Chris Rebert,
  Chris Weber,
  Chris Wilson,
  Christian Biesinger,
  Christian Johansen,
  Christian Schmidt,
  Christoph P&auml;per,
  Christophe Dumez,
  Christopher Aillon,
  Christopher Cameron,
  Christopher Ferris,
  Chriswa,
  Clark Buehler,
  Cole Robison,
  Colin Fine,
  Collin Jackson,
  Corey Farwell,
  Corprew Reed,
  Craig Cockburn,
  Csaba Gabor,
  Csaba Marton,
  Cynthia Shelly,
  Cyrille Tuzi,
  Daksh Shah,
  Dan Callahan,
  Dan Yoder,
  Dane Foster, <!-- studdugie on GitHub -->
  Daniel Barclay,
  Daniel Bratell,
  Daniel Brooks,
  Daniel Brumbaugh Keeney,
  Daniel Buchner,
  Daniel Cheng,
  Daniel Clark, <!-- dandclark on GitHub -->
  Daniel Davis,
  Daniel Ehrenberg,
  Daniel Ethridge, <!-- wlib on GitHub -->
  Daniel Glazman,
  Daniel Holbert,
  Daniel Peng,
  Daniel Schattenkirchner,
  Daniel Sp&aring;ng,
  Daniel Steinberg,
  Daniel Tan,
  Daniel Trebbien,
  Daniel Vogelheim,
  Danny Sullivan,
  Daphne Preston-Kendal,
  Darien Maillet Valentine,
  Darin Adler,
  Darin Fisher,
  Darxus,
  Dave Camp,
  Dave Cramer,
  Dave Hodder,
  Dave Lampton,
  Dave Singer,
  Dave Tapuska,
  Dave Townsend<!-- Mossop on moz irc -->,
  David Baron,
  David Bloom,
  David Bokan,
  David Bruant,
  David Carlisle,
  David E. Cleary,
  David Egan Evans,
  David Fink,
  David Flanagan,
  David Gerard,
  David Grogan,
  David Hale,
  David H&aring;s&auml;ther,
  David Hyatt,
  David I. Lehn,
  David John Burrowes,
  David Matja,
  David Remahl,
  David Resseguie,
  David Smith,
  David Storey,
  David Vest,
  David Woolley,
  David Zbarsky,
  Dave Methvin,
  DeWitt Clinton,
  Dean Edridge,
  Dean Edwards,
  Dean Jackson,
  Debanjana Sarkar, <!-- debanjana-a11y on GitHub -->
  Debi Orton,
  Delan Azabani, <!-- delan on GitHub -->
  Derek Featherstone,
  Derek Guenther,
  Devarshi Pant,
  Devdatta,
  Devin Mullins,
  Devin Rousso,
  Di Zhang,
  Diego Ferreiro Val,
  Diego Gonz&aacute;lez Z&uacute;&ntilde;iga, <!-- diekus on GitHub -->
  Diego Ponce de León,
  Dimitri Glazkov,
  Dimitry Golubovsky,
  Dirk Pranke,
  Dirk Schulze,
  Dirkjan Ochtman,
  Divya Manian,
  Dmitry Lazutkin,
  Dmitry Titov,
  dolphinling,
  Dominic Cooney,
  Dominique Haza&euml;l-Massieux,
  Don Brutzman,
  Donovan Glover,
  Doron Rosenberg,
  Doug Kramer,
  Doug Simpkinson,
  Drew Wilson,
  Edgar Chen,
  Edmund Lai,
  Eduard Pascual,
  Eduardo Vela,
  Edward Welbourne,
  Edward Z. Yang,
  Ehsan Akhgari,
  Ehsan Karamad,
  Eira Monstad,
  Eitan Adler,
  Eli Friedman,
  Eli Grey,
  Eliot Graff,
  Elisabeth Robson,
  Elizabeth Castro,
  Elliott Sprehn,
  Elliotte Harold,
  Emilio Cobos Álvarez,
  Emily Stark,
  Eric Carlson,
  Eric Casler,
  Eric Lawrence,
  Eric Portis,
  Eric Rescorla,
  Eric Semling,
  Eric Shepherd, <!-- a2sheppy on GitHub -->
  Eric Willigers,
  Erik Arvidsson,
  Erik Charlebois,
  Erik Rose,
  栗本 英理子 (Eriko Kurimoto),
  espretto, <!-- GitHub -->
  Evan Jacobs,
  Evan Martin,
  Evan Prodromou,
  Evan Stade,
  Evert,
  Evgeny Kapun,
  ExE-Boss, <!-- GitHub -->
  Ezequiel Garz&oacute;n,
  fantasai,
  Félix Sanz,
  Felix Sasaki,
  Fernando Altomare Serboncini,
  Forbes Lindesay,
  Francesco Schwarz,
  Francis Brosnan Blazquez,
  Franck 'Shift' Qu&eacute;lain,
  Fran&ccedil;ois Marier,
  Frank Barchard,
  Frank Liberato,
  Franklin Shirley,
  Frederik Braun,
  Fredrik S&ouml;derquist,
  &#40284;&#39164;&#25991;&#25935; (Fumitoshi Ukai),
  Futomi Hatano,
  Gavin Carothers,
  Gavin Kistner,
  Gareth Rees,
  Garrett Smith,
  Gary Blackwood,
  Gary Kacmarcik,
  Gary Katsevman,
  Geoff Richards,
  Geoffrey Garen,
  Georg Neis,
  George Lund,
  Gianmarco Armellin,
  Giovanni Campagna,
  Giuseppe Pascale,
  Glenn Adams,
  Glenn Maynard,
  Graham Klyne,
  Greg Botten,
  Greg Houston,
  Greg Wilkins,
  Gregg Tavares,
  Gregory J. Rosmaita,
  Gregory Terzian,
  Grey,
  guest271314,
  Guilherme Johansson Tramontina,
  Guy Bedford,
  Gytis Jakutonis,
  H&aring;kon Wium Lie,
  Habib Virji,
  Hajime Morrita,
  Hallvord Reiar Michaelsen Steen,
  Hanna Laakso,
  Hans S. T&oslash;mmerhalt,
  Hans Stimer,
  Harald Alvestrand,
  Hayato Ito,
  何志翔 (HE Zhixiang),
  Henri Sivonen,
  Henrik Lied,
  Henrik Lievonen,
  Henry Lewis,
  Henry Mason,
  Henry Story,
  Hermann Donfack Zeufack,
  中川博貴 (Hiroki Nakagawa), <!-- nhiroki on GitHub -->
  Hiroshige Hayashizaki,
  Hiroyuki USHITO,
  Hitoshi Yoshida,
  Hongchan Choi, <!-- hoch on GitHub -->
  王华 (Hua Wang), <!-- PaperStrike on GitHub -->
  Hugh Bellamy,
  Hugh Guiney,
  Hugh Winkler,
  Ian Bicking,
  Ian Clelland,
  Ian Davis,
  Ian Fette,
  Ian Henderson,
  Ian Kilpatrick,
  Ibrahim Ahmed,
  Ido Green,
  Ignacio Javier,
  Igor Oliveira,
  安次嶺 一功 (Ikko Ashimine), <!-- eltociear on GitHub -->
  Ilya Grigorik,
  Ingvar Stepanyan,
  isonmad,
  Iurii Kucherov,
  Ivan Enderlin,
  Ivan Nikulin, <!-- inikulin on GitHub -->
  Ivan Panchenko,
  Ivo Emanuel Gon&ccedil;alves,
  J. King,
  J.C. Jones,
  Jackson Ray Hamilton,
  Jacob Davies,
  Jacques Distler,
  Jake Archibald,
  Jake Verbaten,
  Jakub Vrána,
  Jakub &#321;opusza&#324;ski,
  Jakub Wilk,
  James Craig,
  James Graham,
  James Greene,
  James Justin Harrell,
  James Kozianski,
  James M Snell,
  James Perrett,
  James Robinson,
  Jamie Liu,
  Jamie Lokier,
  Jamie Mansfield,
  Jan K&uuml;hle, <!-- frigus02 on GitHub -->
  Jan Miksovsky,
  Janice Shiu,
  Janusz Majnert,
  Jan-Ivar Bruaroey,
  Jan-Klaas Kollhof,
  Jared Jacobs,
  Jason Duell,
  Jason Kersey,
  Jason Lustig,
  Jason Orendorff,
  Jason White,
  Jasper Bryant-Greene,
  Jasper St. Pierre,
  Jatinder Mann,
  Jay Henry Kao,
  Jean-Yves Avenard,
  Jed Hartman,
  Jeff Balogh,
  Jeff Cutsinger,
  Jeff Gilbert,
  Jeff "=JeffH" Hodges,
  Jeff Schiller,
  Jeff Walden,
  Jeffrey Yasskin,
  Jeffrey Zeldman,
  &#32993;&#24935;&#37586; (Jennifer Braithwaite),
  Jellybean Stonerfish,
  Jennifer Apacible,
  Jens Bannmann,
  Jens Fendler,
  Jens Oliver Meiert,
  Jens Widell,
  Jer Noble,
  Jeremey Hustman,
  Jeremy Keith,
  Jeremy Orlow,
  Jeremy Roman,
  Jeroen van der Meer,
  Jerry Smith,
  Jesse Ren&eacute;e Beach,
  Jessica Jong,
  jfkthame, <!-- GitHub -->
  Jian Li,
  Jihye Hong,
  Jim Jewett,
  Jim Ley,
  Jim Meehan,
  Jim Michaels,
  Jinho Bang,
  Jinjiang (勾三股四), <!-- GitHub -->
  Jirka Kosek,
  Jjgod Jiang,
  Joaquim Medeiros,
  Jo&atilde;o Eiras,
  Jochen Eisinger,
  Joe Clark,
  Joe Gregorio,
  Joel Spolsky,
  Joel Verhagen,
  Joey Arhar,
  Johan Herland,
  Johanna Herman,
  John Boyer,
  John Bussjaeger,
  John Carpenter,
  John Daggett,
  John Fallows,
  John Foliot,
  John Harding,
  John Keiser,
  John Law,
  John Musgrave,
  John Snyders,
  John Stockton,
  John-Mark Bell,
  Johnny Stenback,
  Jon Coppeard,
  Jon Ferraiolo,
  Jon Gibbins,
  Jon Jensen,
  Jon Perlow,
  Jonas Sicking,
  Jonathan Cook,
  Jonathan Kew,
  Jonathan Neal,
  Jonathan Oddy,
  Jonathan Rees,
  Jonathan Watt,
  Jonathan Worent,
  Jonny Axelsson,
  Joram Schrijver,
  Jordan Tucker,
  Jorgen Horstink,
  Joris van der Wel,
  Jorunn Danielsen Newth,
  Joseph Kesselman,
  Joseph Mansfield,
  Joseph Pecoraro,
  Josh Aas,
  Josh Hart,
  Josh Juran,
  Josh Levenberg,
  Josh Matthews,
  Joshua Bell,
  Joshua Chen, <!-- Josh-Cena on GitHub -->
  Joshua Randall,
  Juan Olvera, <!-- thinkxl on GitHub -->
  Juanmi Huertas,
  Jukka K. Korpela,
  Jules Cl&eacute;ment-Ripoche,
  Julian Reschke,
  Julio Lopez,
  小勝 純 (Jun Kokatsu),
  Jun Yang (harttle),
  Jungkee Song,
  J&uuml;rgen Jeka,
  Justin Lebar,
  Justin Novosad,
  Justin Rogers,
  Justin Schuh,
  Justin Sinclair,
  Juuso Lapinlampi,
  Ka-Sing Chou,
  Kagami Sascha Rosylight,
  Kai Hendry,
  Kamishetty Sreeja,
  &#x5442;&#x5eb7;&#x8c6a; (KangHao Lu)<!-- Kenny, kennyluck-->,
  Karl Dubost,
  Karl Tomlinson,
  Kartik Arora,
  Kartikaya Gupta,
  Kathy Walton,
  河童エクマ(Kawarabe Ecma)<!-- heppokofrontend on GitHub -->
  Keith Cirkel, <!-- keithamus on GitHub -->
  Keith Rollin,
  Keith Yeung,
  Kelly Ford,
  Kelly Norton,
  Ken Russell,
  Kenji Baheux,
  Kevin Benson,
  Kevin Cole,
  Kevin Gadd,
  Kevin McNee,
  Kevin Venkiteswaran,
  Khushal Sagar,
  Kinuko Yasuda,
  Koji Ishii,
  Korn&eacute;l P&aacute;l,
  Kornel Lesinski,
  上野 康平 (UENO, Kouhei),
  Kris Northfield,
  Kristian Spangsege,
  Kristof Zelechovski,
  Krzysztof Maczy&#x0144;ski,
  &#x9ed2;&#x6fa4;&#x525b;&#x5fd7; (Kurosawa Takeshi),
  Kyle Barnhart,
  Kyle Hofmann<!-- Ozob -->,
  Kyle Huey,
  L&eacute;onard Bouchet,
  L&eacute;onie Watson,
  Lachlan Hunt,
  Larry Masinter,
  Larry Page,
  Lars Gunther<!-- Keryx Web -->,
  Lars Solberg,
  Laura Carlson,
  Laura Granka,
  Laura L. Carlson,
  Laura Wisewell,
  Laurens Holst,
  Lawrence Forooghian,
  Lee Kowalkowski,
  Leif Halvard Silli,
  Leif Kornstaedt,
  Lenny Domnitser,
  Leonard Rosenthol,
  Leons Petrazickis,
  Liviu Tinta,
  Lobotom Dysmon,
  Logan<!-- on moz irc -->,
  Logan Moore,
  Loune,
  Lucas Gadani,
  &#x0141;ukasz Pilorz,
  Luke Kenneth Casson Leighton,
  Luke Warlow,
  Luke Wilde,
  Maciej Stachowiak,
  Magne Andersson,
  Magnus Kristiansen<!-- Dashiva -->,
  Maik Merten,
  Majid Valipour,
  Maksim Sadym,
  Malcolm Rowe,
  Manish Goregaokar,
  Manish Tripathi,
  Manuel Martinez-Almeida,
  Manuel Rego Casasnovas,
  Marc Hoyois,
  Marc-André Choquette, <!-- eel-in-a-hovercraft on GitHub -->
  Marc-André Lafortune,
  Marco Zehe,
  Marcus Bointon,
  Marcus Otterström,
  Marijn Kruisselbrink,
  Mark Amery,
  Mark Birbeck,
  Mark Davis,
  Mark Green,
  Mark Miller,
  Mark Nottingham,
  Mark Pilgrim,
  Mark Rogers, <!-- dd8 on GitHub -->
  Mark Rowe<!--bdash-->,
  Mark Schenk,
  Mark Vickers,
  Mark Wilton-Jones,
  Markus Cadonau,
  Markus Stange,
  Martijn van der Ven,
  Martijn Wargers,
  Martin Atkins,
  Martin Chaov,
  Martin D&uuml;rst,
  Martin Honnen,
  Martin Janecke,
  Martin Kutschker,
  Martin Nilsson,
  Martin Thomson,
  Masataka Yakura,
  Masatoshi Kimura,
  Mason Freed, <!-- mfreed7 on GitHub -->
  Mason Mize,
  Mathias Bynens,
  Mathieu Henri,
  Matias Larsson,
  Matt Brubeck,
  Matt Di Pasquale,
  Matt Falkenhagen,
  Matt Giuca, <!-- mgiuca on GitHub -->
  Matt Harding,
  Matt Schmidt,
  Matt Wright,
  Matthew Gaudet, <!-- mgaudet on GitHub -->
  Matthew Gregan,
  Matthew Mastracci,
  Matthew Noorenberghe,
  Matthew Raymond,
  Matthew Thomas,
  Matthew Tylee Atkinson, <!-- matatk on GitHub -->
  Mattias Waldau,
  Max Romantschuk,
  Maxim Tsoy, <!-- muodov on GitHub -->
  Mayeul Cantan,
  Menachem Salomon,
  Menno van Slooten,
  Micah Dubinko,
  Micah Nerren,
  Michael 'Ratt' Iannarelli,
  Michael A. Nachbaur,
  Michael A. Puls II<!--Shadow2531-->,
  Michael Carter,
  Michael Daskalov,
  Michael Day,
  Michael Dyck,
  Michael Enright,
  Michael Ficarra,
  Michael Gratton,
  Michael Kohler,
  Michael McKelvey,
  Michael Nordman,
  Michael Powers,
  Michael Rakowski,
  Michael(tm) Smith,
  Michael Walmsley,
  Michal Zalewski,
  Michel Buffa,
  Michel Fortin,
  Michelangelo De Simone,
  Michiel van der Blonk,
  Miguel Casas-Sanchez<!-- yellowdoge on GitHub -->,
  Mihai &#x015E;ucan<!-- from ROBO Design -->,
  Mihai Parparita,
  Mike Brown,
  Mike Dierken<!-- S. Mike Dierken -->,
  Mike Dixon,
  Mike Hearn,
  Mike Pennisi,
  Mike Schinkel,
  Mike Shaver,
  Mikko Rantalainen,
  Mingye Wang,
  Mirko Brodesser,
  Mohamed Zergaoui<!-- Innovimax SARL -->,
  Mohammad Al Houssami,
  Mohammad Reza Zakerinasab,
  Momdo Nakamura,
  Morten Stenshorne,
  Mounir Lamouri,
  Ms2ger,
  mtrootyy,
  邱慕安 (Mu-An<!--grammar-check-override--> Chiou),
  Mukilan Thiyagarajan,
  Mustaq Ahmed,
  Myles Borins,
  Nadia Heninger,
  Nate Chapin,
  NARUSE Yui,
  Navid Zolghadr,
  Neil Deakin,
  Neil Rashbrook,
  Neil Soiffer,
  Nereida Rondon,
  networkException,
  Nicholas Shanks,
  Nicholas Stimpson,
  Nicholas Zakas,
  Nickolay Ponomarev,
  Nicolas Gallagher,
  Nicolas Pena Moreno,
  Nicolò Ribaudo,
  Nidhi Jaju,
  Nikki Bee,
  Niklas Gögge,
  Nina Satragno,
  Noah Mendelsohn,
  Noah Slater,
  Noam Rosenthal,
  Noel Gordon,
  Nolan Waite,
  NoozNooz42,
  Norbert Lindenberg,
  Oisín Nolan, <!-- oisinnolan on GitHub -->
  Ojan Vafai,
  Olaf Hoffmann,
  Olav Junker Kj&aelig;r,
  Old&#345;ich Vete&#353;n&#237;k,
  Oli Studholme,
  Oliver Hunt,
  Oliver Rigby,
  Olivia (Xiaoni) Lai, <!-- xlai-o on GitHub -->
  Olivier Gendrin,
  Olli Pettay,
  Ondřej Žára, <!-- ondras on GitHub -->
  Ori Avtalion,
  Oriol Brufau,
  oSand, <!-- from reddit -->
  Pablo Flouret,
  Patrick Dark,
  Patrick Garies,
  Patrick H. Lauke,
  Patrik Persson,
  Paul Adenot,
  Paul Lewis,
  Paul Norman,
  Per-Erik Brodin,
  一丝 (percyley), <!-- yisibl on GitHub -->
  Perry Smith,
  Peter Beverloo,
  Peter Karlsson,
  Peter Kasting,
  Peter Moulder,
  Peter Occil,
  Peter Stark,
  Peter Van der Beken,
  Peter van der Zee,
  Peter-Paul Koch,
  Phil Pickering,
  Philip Ahlberg,
  Philip Brembeck, <!-- philipbrembeck on GitHub -->
  Philip Taylor,
  Philip TAYLOR<!-- a different one -->,
  Philippe De Ryck,
  Pierre-Arnaud Allumé,
  Pierre-Marie Dartus,
  Pierre-Yves Gérardy,
  Piers Wombwell, <!-- pwombwell on GitHub -->
  Pooja Sanklecha,
  Prashant Hiremath,
  Prashanth Chandra,
  Prateek Rungta,
  Pravir Gupta,
  Prayag Verma,
  &#x674e;&#x666e;&#x541b; (Pujun Li)<!-- masa jack -->,
  Rachid Finge,
  Rafael Weinstein,
  Rafa&#x0142; Mi&#x0142;ecki,
  Rahim Abdi,
  Rahul Purohit,
  Raj Doshi,
  Rajas Moonka,
  Rakina Zata Amni,
  Ralf Stoltze,
  Ralph Giles,
  Raphael Champeimont,
  Rebecca Star,
  Remci Mizkur,
  Remco,
  Remy Sharp,
  Rene Saarsoo,
  Rene Stach,
  Ric Hardacre,
  Rich Clark,
  Rich Doughty,
  Richa Rupela,
  Richard Gibson,
  Richard Ishida,
  Richard Torres<!-- richardtorres314 on GitHub -->,
  Ricky Mondello,
  Rigo Wenning,
  Rikkert Koppes,
  Rimantas Liubertas,
  Riona Macnamara,
  Rob Buis,
  Rob Ennals,
  Rob Jellinghaus,
  Rob S,
  Rob Smith,
  Robert Blaut,
  Robert Collins,
  Robert Hogan,
  Robert Kieffer,
  Robert Linder,
  Robert Millan,
  Robert O'Callahan,
  Robert Sayre,
  Robin Berjon,
  Robin Schaufler,
  Rodger Combs,
  Roland Steiner,
  Roma Matusevich,
  Romain Deltour,
  Roman Ivanov,
  Roy Fielding,
  Rune Lillesveen,
  Russell Bicknell,
  Ruud Steltenpool,
  Ryan King,
  Ryan Landay,
  Ryan Sleevi,
  Ryo Kajiwara, <!-- sylph01 on GitHub -->
  Ryo Kato,
  Ryosuke Niwa,
  S. Mike Dierken,
  Salvatore Loreto,
  Sam Atkins, <!-- atkinssj on GitHub -->
  Sam Dutton,
  Sam Kuper,
  Sam Ruby,
  Sam Sneddon, <!-- gsnedders on GitHub -->
  Sam Weinig,
  Samikshya Chand,
  Samuel Bronson,
  Samy Kamkar,
  Sander van Lambalgen,
  Sanjoy Pal,
  Sanket Joshi,
  Sarah Gebauer,
  Sarven Capadisli,
  Satrujit Behera,
  Sayan Sivakumaran,
  Schalk Neethling,
  Scott Beardsley,
  Scott Gonz&aacute;lez,
  Scott Hess,
  Scott Miles,
  Scott O'Hara,
  Sean B. Palmer, <!-- sbp on GitHub -->
  Sean Feng,
  Sean Fraser,
  Sean Hayes,
  Sean Hogan,
  Sean Knapp,
  Sebastian Markb&aring;ge,
  Sebastian Schnitzenbaumer,
  Sendil Kumar N,
  Seth Call,
  Seth Dillingham,
  Shannon Moeller,
  Shanti Rao,
  Shaun Inman,
  Shiino Yuki, <!-- yuki3 on GitHub -->
  贺师俊 (HE Shi-Jun), <!-- hax on GitHub -->
  Shiki Okasaka,
  Shivani Sharma,
  shreyateeza, <!-- GitHub -->
  Shubheksha Jalan,
  Sidak Singh Aulakh,
  Sierk Bornemann,
  Sigbj&oslash;rn Finne,
  Sigbj&oslash;rn Vik,
  Silver Ghost, <!-- see bug 19614 -->
  Silvia Pfeiffer,
  &#x160;ime Vidas,
  Simon Fraser,
  Simon Montagu,
  Simon Sapin,
  Yu Han, <!-- yuzhe-han on GitHub -->
  Simon Spiegel,
  Simon Wülker, <!-- Wuelle on GitHub -->
  Siye Liu,
  skeww, <!-- on reddit -->
  Smylers,
  Srirama Chandra Sekhar Mogali,
  Stanton McCandlish,
  stasoid, <!-- GitHub -->
  Stefan H&aring;kansson,
  Stefan Haustein,
  Stefan Santesson,
  Stefan Schumacher,
  Ştefan Vargyas,
  Stefan Weiss,
  Steffen Meschkat,
  Stephen Chenney,
  Stephen Ma,
  Stephen Stewart,
  Stephen White,
  Steve Comstock,
  Steve Faulkner,
  Steve Fink,
  Steve Orvell,
  Steve Runyon,
  Steven Bennett,
  Steven Bingler,
  Steven Garrity,
  Steven Tate,
  Stewart Brodie,
  Stuart Ballard,
  Stuart Langridge,
  Stuart Parmenter,
  Subramanian Peruvemba,
  Sudhanshu Jaiswal,
  sudokus999, <!-- GitHub -->
  Sunava Dutta,
  Surma,
  Susan Borgrink,
  Susan <!--G.--> Lesch,
  Sylvain Pasche,
  T.J. Crowder, <!-- tjcrowder and farsightsoftware on GitHub -->
  Tab Atkins-Bittner,
  Taiju Tsuiki,
  Takashi Toyoshima,
  Takayoshi Kochi,
  Takeshi Yoshino,
  <span data-x="" lang="tr">Tantek &Ccedil;elik</span>,
  田村健人 (Kent TAMURA),
  Tawanda Moyo, <!-- tawandamoyo on GitHub-->
  Taylor Hunt,
  Ted Mielczarek,
  Terence Eden, <!-- edent on GitHub -->
  Terrence Wood,
  Tetsuharu OHZEKI,
  Theresa O'Connor,
  Thijs van der Vossen,
  Thomas Broyer,
  Thomas Koetter,
  Thomas O'Connor,
  Tim Altman,
  Tim Dresser,
  Tim Flynn, <!-- trflynn89 on GitHub -->
  Tim Johansson,
  Tim Nguyen, <!-- nt1m on Github -->
  Tim Perry,
  Tim van der Lippe,
  TJ VanToll,
  Tobias Schneider,
  Tobie Langel,
  Toby Inkster,
  Todd Moody,
  Tom Baker,
  Tom Pike,
  Tom Schuster,
  Tom ten Thij,
  Tomasz Jakut, <!-- Comandeer on GitHub -->
  Tomek Wytrębowicz,
  Tommy Thorsen,
  Tony Ross,
  Tooru Fujisawa,
  Toru Kobayashi,
  Traian Captan,
  Travis Leithead,
  Trevor Rowbotham,
  Trevor Saunders,
  Trey Eckels,
  triple-underscore, <!-- GitHub -->
  Tristan Fraipont, <!-- Kaiido on GitHub -->
  Tristan Parisot,  <!-- quadristan on GitHub -->
  保呂 毅 (Tsuyoshi Horo),
  Tyler Close,
  Valentin Gosu,
  Vardhan Gupta,
  Vas Sudanagunta,
  Veli Şenol,
  Victor Carbune,
  Victor Costan,
  Vipul Snehadeep Chawathe,
  Vitya Muhachev,
  Vlad Levin,
  Vladimir Katardjiev,
  Vladimir Vuki&#x0107;evi&#x0107;,
  Vyacheslav Aristov,
  voracity,
  Walter Steiner,
  Wakaba,
  Wayne Carr,
  Wayne Pollock,
  Wellington Fernando de Macedo,
  Wenson Hsieh,
  Weston Ruter,
  Wilhelm Joys Andersen,
  Will Levine,
  Will Ray,
  William Chen,
  William Swanson,
  Willy Martin Aguirre Rodriguez,
  Wladimir Palant,
  Wojciech Mach,
  Wolfram Kriesing,
  Xan Gregg,
  xenotheme,
  XhmikosR,
  Xida Chen,
  Xidorn Quan,
  Xue Fuqiao,
  Yang Chen,
  Yao Xiao,
  Yash Handa,
  Yay295,
  Ye-Kui Wang,
  Yehuda Katz,
  Yi Xu,
  Yi-An Huang,
  Yngve Nysaeter Pettersen,
  Yoav Weiss,
  Yonathan Randolph,
  Yu Huojiang,
  Yuki Okushi, <!-- JohnTitor on GitHub -->
  Yury Delendik,
  平野裕 (Yutaka Hirano),
  Yuzo Fujishima,
  西條柚 (Yuzu Saijo),
  Zhenbin Xu,
  &#24352;&#26234;&#24378; (Zhiqiang Zhang),
  Zoltan Herczeg,
  Zyachel,
  and
  &Oslash;istein E. Andersen,

  for their useful comments, both large and small, that have led to changes to this specification
  over the years.</p>

  <p>Thanks also to everyone who has ever posted about HTML to their blogs, public mailing lists, or
  forums, including all the contributors to the <a href="https://www.w3.org/html/wg/lists/">various
  W3C HTML WG lists</a> and the <a href="https://whatwg.org/mailing-list">various WHATWG
  lists</a>.

  <p>Special thanks to Richard Williamson for creating the first implementation of
  <code>canvas</code> in Safari, from which the canvas feature was designed.</p>

  <p>Special thanks also to the Microsoft employees who first implemented the event-based
  drag-and-drop mechanism, <code data-x="attr-contenteditable">contenteditable</code>, and other
  features first widely deployed by the Windows Internet Explorer browser.</p>

  <p w-nodev>Special thanks and $10,000 to David Hyatt who came up with a broken implementation
  of the <a href="#adoptionAgency">adoption agency algorithm</a> that the editor had to reverse
  engineer and fix before using it in the parsing section.</p>

  <p>Thanks to the participants of the microdata usability study for allowing us to use their
  mistakes as a guide for designing the microdata feature.</p>

  <p>Thanks to the many sources that provided inspiration for the examples used in the
  specification.</p>

  <p>Thanks also to the Microsoft blogging community for some ideas, to the attendees of the W3C
  Workshop on Web Applications and Compound Documents for inspiration, to the #mrt crew, the #mrt.no
  crew, and the #whatwg crew, and to Pillar and Hedral for their ideas and support.</p>

  <p>Thanks to Igor Zhbanov for generating PDF versions of the specification.</p>

  <p>Special thanks to the <a href="https://www.w3.org/community/respimg/">RICG</a> for developing
  the <code>picture</code> element and related features; in particular thanks to Adrian Bateman,
  Bruce Lawson, David Newton, Ilya Grigorik, John Schoenick, Leon de Rijke, Mat Marquis, Marcos
  Cáceres, Tab Atkins, Theresa O'Connor, and Yoav Weiss for their contributions.</p>

  <p>Special thanks to the <a href="https://www.w3.org/WebPlatform/WG/">WPWG</a> for incubating the
  <a href="#custom-elements">custom elements</a> feature. In particular, thanks to David Hyatt and
  Ian Hickson for their influence through the XBL specifications, Dimitri Glazkov for the first
  draft of the custom elements specification, and to
  Alex Komoroske,
  Alex Russell,
  Andres Rios,
  Boris Zbarsky,
  Brian Kardell,
  Daniel Buchner,
  Dominic Cooney,
  Erik Arvidsson,
  Elliott Sprehn,
  Hajime Morrita,
  Hayato Ito,
  Jan Miksovsky,
  Jonas Sicking,
  Olli Pettay,
  Rafael Weinstein,
  Roland Steiner,
  Ryosuke Niwa,
  Scott Miles,
  Steve Faulkner,
  Steve Orvell,
  Tab Atkins,
  Theresa O'Connor,
  Tim Perry, and
  William Chen
  for their contributions.</p>

  <p>Special thanks to the <a href="https://www.w3.org/Style/CSS/members">CSSWG</a> for developing
  the <a href="#worklets">worklets</a>. In particular, thanks to Ian Kilpatrick for his work as
  editor of the original worklets specification.</p>

  <p>For about ten years starting in 2003, this standard was almost entirely written by Ian Hickson
  (<a href="https://www.google.com/">Google</a>, <a
  href="mailto:ian@hixie.ch">ian@hixie.ch</a>).</p>

  <p>Starting in 2015, the editor group expanded. It is currently maintained by
  <a href="https://annevankesteren.nl/">Anne van Kesteren</a> (<a href="https://www.apple.com/">Apple</a>, <a href="mailto:annevk@annevk.nl">annevk@annevk.nl</a>),
  <a href="https://domenic.me/">Domenic Denicola</a> (<a href="https://www.google.com/">Google</a>, <a href="mailto:d@domenic.me">d@domenic.me</a>)
  <a href="https://domfarolino.com">Dominic Farolino</a> (<a href="https://www.google.com/">Google</a>, <a href="mailto:domfarolino@gmail.com">domfarolino@gmail.com</a>),
  <a href="https://foolip.org/">Philip J&auml;genstedt</a> (<a href="https://www.google.com/">Google</a>, <a href="mailto:philip@foolip.org">philip@foolip.org</a>), and
  Simon Pieters (<a href="https://www.mozilla.org/">Mozilla</a>, <a href="mailto:zcorpan@gmail.com">zcorpan@gmail.com</a>).</p>

  <!-- Hopefully Kam and Tess won't notice they're covered by these acknowledgments three times! -->


  <h2 id="ipr" class="no-num">Intellectual property rights</h2>

  <div w-nodev itemscope itemtype="http://n.whatwg.org/work">
   <p>The image in the introduction is based on <a itemprop="work" href="https://www.flickr.com/photos/wonderlane/2986252088/">a photo</a>
   by <a itemprop="https://creativecommons.org/ns#attributionURL" href="https://www.flickr.com/photos/wonderlane/">Wonderlane</a>.
   (<a itemprop="license" href="https://creativecommons.org/licenses/by/2.0/">CC BY 2.0</a>)
  </div>

  <div itemscope itemtype="http://n.whatwg.org/work">
   <p>The image of the wolf in the embedded content introduction is based on <a itemprop="work" href="https://commons.wikimedia.org/wiki/File:WolfRunningInSnow.jpg">a photo</a>
   by <a itemprop="http://creativecommons.org/ns#attributionURL" href="https://commons.wikimedia.org/wiki/File:WolfRunningInSnow.jpg">Barry O'Neill</a>.
   (<a itemprop="license" href="https://en.wikipedia.org/wiki/Public_domain">Public domain</a>)
  </div>

  <div itemscope itemtype="http://n.whatwg.org/work">
   <p>The image of the kettlebell swing in the embedded content introduction is based on <a itemprop="work" href="https://pixabay.com/en/functional-mobility-articular-606568/">a photo</a>
   by <a itemprop="http://creativecommons.org/ns#attributionURL" href="https://pixabay.com/en/users/506563-506563/">kokkarina</a>.
   (<a itemprop="license" href="https://creativecommons.org/publicdomain/zero/1.0/">CC0 1.0</a>)
  </div>

  <div itemscope itemtype="http://n.whatwg.org/work">
   <p>The Blue Robot Player sprite used in the canvas demo is based on
   <a itemprop="work" href="https://johncolburn.deviantart.com/art/Blue-Robot-Player-Sprite-323813997">a work</a> by
   <a itemprop="https://creativecommons.org/ns#attributionURL" href="https://johncolburn.deviantart.com/">JohnColburn</a>.
   (<a itemprop="license" href="https://creativecommons.org/licenses/by-sa/3.0/">CC BY-SA 3.0</a>)</p>
  </div>

  <div itemscope itemtype="http://n.whatwg.org/work">
   <p>The photograph of robot 148 climbing the tower at the FIRST Robotics Competition 2013 Silicon Valley Regional is based on
   <a itemprop="work" href="https://www.flickr.com/photos/lenore-m/8631391979/">a work</a> by
   <a itemprop="https://creativecommons.org/ns#attributionURL" href="https://www.flickr.com/photos/lenore-m/">Lenore Edman</a>.
   (<a itemprop="license" href="https://creativecommons.org/licenses/by/2.0/">CC BY 2.0</a>)</p>
  </div>

  <div itemscope itemtype="http://n.whatwg.org/work">
   <p>The diagram showing how <code data-x="attr-script-async">async</code> and <code
   data-x="attr-script-defer">defer</code> impact <code>script</code> loading is based on a
   similar diagram from <a itemprop="work"
   href="https://peter.sh/experiments/asynchronous-and-deferred-javascript-execution-explained/">a
   blog post</a> by <a itemprop="https://creativecommons.org/ns#attributionURL"
   href="https://peter.sh/about/">Peter Beverloo</a>.
   (<a itemprop="license" href="https://creativecommons.org/publicdomain/zero/1.0/">CC0 1.0</a>)</p>
  </div>

  <div itemscope itemtype="http://n.whatwg.org/work">
   <p>The image decoding demo used to demonstrate module-based workers draws on some example code
   from <a itemprop="work" href="https://www.html5rocks.com/en/tutorials/canvas/imagefilters/">a
   tutorial</a> by <a itemprop="https://creativecommons.org/ns#attributionURL"
   href="https://www.fhtr.net/">Ilmari Heikkinen</a>.
   (<a itemprop="license" href="https://creativecommons.org/licenses/by/3.0/">CC BY 3.0</a>)</p>
  </div>

  <div itemscope itemtype="http://n.whatwg.org/work">
   <p>The <code data-x="">&lt;flag-icon></code> example was inspired by <a itemprop="work"
   href="https://github.com/stevenrskelton/flag-icon">a custom element</a> by <a
   itemprop="https://creativecommons.org/ns#attributionURL" href="http://stevenskelton.ca/">Steven
   Skelton</a>. (<a itemprop="license" href="https://opensource.org/licenses/MIT">MIT</a>)</p>
  </div>

  <p itemscope itemtype="http://n.whatwg.org/work">Part of the revision history of the
  <code>picture</code> element and related features can be found in the <a
  href="https://github.com/ResponsiveImagesCG/picture-element"><code>ResponsiveImagesCG/picture-element</code>
  repository</a>, which is available under the <a itemprop="license"
  href="https://www.w3.org/Consortium/Legal/2015/copyright-software-and-document">W3C Software and
  Document License</a>.</p>

  <p itemscope itemtype="http://n.whatwg.org/work">Part of the revision history of the <code
  data-x="meta-theme-color">theme-color</code> metadata name can be found in the <a
  href="https://github.com/whatwg/meta-theme-color"><code>whatwg/meta-theme-color</code>
  repository</a>, which is available under <a itemprop="license"
  href="https://creativecommons.org/publicdomain/zero/1.0/">CC0</a>.</p>

  <p itemscope itemtype="http://n.whatwg.org/work">Part of the revision history of the <a
  href="#custom-elements">custom elements</a> feature can be found in the <a
  href="https://github.com/w3c/webcomponents"><code>w3c/webcomponents</code> repository</a>, which
  is available under the <a itemprop="license"
  href="https://www.w3.org/Consortium/Legal/2015/copyright-software-and-document">W3C Software and
  Document License</a>.</p>

  <p itemscope itemtype="http://n.whatwg.org/work">Part of the revision history of the <code
  data-x="dom-innerText">innerText</code> getter and setter can be found in the <a
  href="https://github.com/rocallahan/innerText-spec"><code>rocallahan/innerText-spec</code>
  repository</a>, which is available under <a itemprop="license"
  href="https://creativecommons.org/publicdomain/zero/1.0/">CC0</a>.</p>

  <p itemscope itemtype="http://n.whatwg.org/work">Part of the revision history of the <a
  href="#worklets">worklets</a> feature can be found in the <a
  href="https://github.com/w3c/css-houdini-drafts"><code>w3c/css-houdini-drafts</code>
  repository</a>, which is available under the <a itemprop="license"
  href="https://www.w3.org/Consortium/Legal/2015/copyright-software-and-document">W3C Software and
  Document License</a>.</p>

  <p itemscope itemtype="http://n.whatwg.org/work">Part of the revision history of the <a
  href="#import-maps">import maps</a> feature can be found in the <a
  href="https://github.com/WICG/import-maps"><code>WICG/import-maps</code>
  repository</a>, which is available under the <a itemprop="license"
  href="https://www.w3.org/Consortium/Legal/2015/copyright-software-and-document">W3C Software and
  Document License</a>.</p>

  <p itemscope itemtype="http://n.whatwg.org/work">Part of the revision history of the <a
  href="#navigation-api">navigation API</a> feature can be found in the <a
  href="https://github.com/WICG/navigation-api"><code>WICG/navigation-api</code>
  repository</a>, which is available under the <a itemprop="license"
  href="https://www.w3.org/Consortium/Legal/2015/copyright-software-and-document">W3C Software and
  Document License</a>.</p>

  <p itemscope itemtype="http://n.whatwg.org/work">Part of the revision history of the <a
  href="#close-requests-and-close-watchers">Close requests and close watchers</a> section can be
  found in the <a href="https://github.com/WICG/close-watcher"><code>WICG/close-watcher</code>
  repository</a>, which is available under the <a itemprop="license"
  href="https://www.w3.org/Consortium/Legal/2015/copyright-software-and-document">W3C Software and
  Document License</a>.</p>

  <p>Copyright © WHATWG (Apple, Google, Mozilla, Microsoft). This work is licensed under a <a
  rel="license" href="https://creativecommons.org/licenses/by/4.0/">Creative Commons Attribution 4.0
  International License</a>. To the extent portions of it are incorporated into source code, such
  portions in the source code are licensed under the <a rel="license"
  href="https://opensource.org/licenses/BSD-3-Clause">BSD 3-Clause License</a> instead.</p>

  <p w-nodev w-noreview>This is the Living Standard. Those interested in the patent-review version
  should view the <a href="/review-drafts/2025-01/">Living Standard Review Draft</a>.</p>
 </body>
</html>