diff --git a/source b/source index 0639bb4821c..b68d610128d 100644 --- a/source +++ b/source @@ -1773,8 +1773,9 @@ a.setAttribute('href', 'https://example.com/'); // change the content attribute data-x="control">controls or U+0020 SPACE).

An HTML element can have specific HTML element insertion steps, HTML element - post-connection steps, and HTML element removing steps, all defined for the - element's local name.

+ post-connection steps, HTML element removing steps, and HTML element moving + steps all defined for the element's local + name.

The insertion steps for the HTML Standard, given insertedNode, are defined as the following:

@@ -1861,6 +1862,29 @@ a.setAttribute('href', 'https://example.com/'); // change the content attribute popover algorithm given removedNode, false, false, and false.

+

The moving steps for the HTML Standard, given + movedNode, are defined as the following:

+ +
    +
  1. If movedNode is an element whose namespace is the HTML namespace, and this + standard defines HTML element moving steps for movedNode's local name, then run the corresponding HTML + element moving steps given movedNode.

  2. + +
  3. +

    If movedNode is a form-associated element or the ancestor of a + form-associated element, then:

    + +
      +
    1. If the form-associated element has a form owner and the + form-associated element and its form owner are no longer in the same + tree, then reset the form owner of the form-associated + element.

    2. +
    +
  4. +
+

A node is inserted into a document when the insertion steps are invoked @@ -3224,6 +3248,7 @@ a.setAttribute('href', 'https://example.com/'); // change the content attribute

  • append() method
  • appendChild() method
  • cloneNode() method
  • +
  • moveBefore() method
  • importNode() method
  • preventDefault() method
  • id attribute
  • @@ -3263,6 +3288,7 @@ a.setAttribute('href', 'https://example.com/'); // change the content attribute
  • The insertion steps,
  • The post-connection steps, removing steps, + moving steps, adopting steps, and children changed steps hooks for elements
  • The change, append, remove, replace, get an attribute by namespace and local name, set value, and remove an attribute by namespace and local name algorithms for attributes
  • @@ -29160,8 +29186,8 @@ interface HTMLSourceElement : HTMLElement {
    -

    The source HTML element insertion - steps, given insertedNode, are:

    +

    The source HTML element insertion steps, given + insertedNode, are:

    1. If insertedNode's parent is a media element that has no HTMLSourceElement : HTMLElement { mutation for the img element.

    -

    The source HTML element removing - steps, given removedNode and oldParent, are:

    +

    The source HTML element moving steps, given movedNode and + oldParent, are:

    + +
      +
    1. If movedNode's next sibling was an img element and + oldParent is a picture element, then, count this as a relevant mutation for the img element.

    2. +
    + +

    The source HTML element removing steps, given removedNode + and oldParent, are:

    1. If removedNode's next sibling was an img element and @@ -29437,8 +29472,8 @@ interface HTMLImageElement : HTMLElement { page layout from shifting around after the image loads.

    -

    The img HTML element insertion - steps, given insertedNode, are:

    +

    The img HTML element insertion steps, given insertedNode, + are:

    1. If insertedNode's parent is a picture element, then, count this as @@ -29446,8 +29481,16 @@ interface HTMLImageElement : HTMLElement { insertedNode.

    -

    The img HTML element removing - steps, given removedNode and oldParent, are:

    +

    The img HTML element moving steps, given movedNode and + oldParent, are:

    + +
      +
    1. If oldParent is a picture element, then, count this as a relevant mutation for movedNode.

    2. +
    + +

    The img HTML element removing steps, given removedNode and + oldParent, are:

    1. If oldParent is a picture element, then, count this as a @@ -30948,9 +30991,9 @@ was an English <a href="/wiki/Music_hall">music hall</a> singer, ...

      The element's referrerpolicy attribute's state is changed.

    2. -
    3. The img or source HTML element insertion steps or - HTML element removing steps count the mutation as a relevant mutation.

    4. +
    5. The img or source HTML element insertion steps, + HTML element removing steps, and HTML element moving steps count the + mutation as a relevant mutation.

    6. The element's parent is a picture element and a source element that is a previous sibling has its srcset, MediaError { other node is the node after pointer. Initially, let pointer be the position between the candidate node and the next node, if there are any, or the end of the list, if it is the last node.

      -

      As nodes are inserted and removed into the media element, +

      As nodes are inserted, removed, and moved into the media element, pointer must be updated as follows:

      -
      If a new node is inserted between the two - nodes that define pointer
      +
      If a new node is inserted or moved between the two nodes that define pointer
      Let pointer be the point between the node before pointer and the new node. In other words, insertions at pointer go after pointer.
      @@ -53835,6 +53878,17 @@ interface HTMLSelectElement : HTMLElement { select element's selectedness setting algorithm.

    +

    The option HTML element moving steps, given movedNode and + oldParent, are:

    + +
      +
    1. Run the option HTML element removing steps given + movedNode and oldParent.

    2. + +
    3. Run the option HTML element insertion steps given + movedNode.

    4. +
    +

    The optgroup HTML element removing steps, given removedNode and oldParent, are:

    @@ -53844,6 +53898,14 @@ interface HTMLSelectElement : HTMLElement { algorithm.

    +

    The optgroup HTML element moving steps, given movedNode + and oldParent, are:

    + +
      +
    1. Run the optgroup HTML element removing steps given + movedNode and oldParent.

    2. +
    +

    If an option element in the list of options asks for a reset, then run that select element's selectedness setting algorithm.

    @@ -56344,12 +56406,11 @@ interface HTMLLegendElement : HTMLElement { form attribute and an element with an ID is inserted into or removed from the - Document, then the user agent must reset the form owner of that - form-associated element.

    + Document, or its HTML element moving steps are run, then the user agent + must reset the form owner of that form-associated element.

    -

    The form owner is also reset by the HTML Standard's insertion steps and removing steps.

    +

    The form owner is also reset by the HTML element insertion steps, + HTML element removing steps, and HTML element moving steps.

    To reset the form owner of a form-associated element element:

    @@ -72066,7 +72127,8 @@ console.log(plasticButton.outerHTML); // will output '<button is="plastic-but responsibility of conveying button semantics (including the ability to be disabled) might look something like this:

    -
    class TacoButton extends HTMLElement {
    +  
    class TacoButton extends HTMLElement {
       static observedAttributes = ["disabled"];
     
       constructor() {
    @@ -72379,6 +72441,70 @@ document.body.append(parent);
     // CChild connectedCallback: isConnected = false
    +
    Preserving custom element state when moved
    + + + +

    When manipulating the DOM tree, an element can be moved in the tree while connected. This applies to custom + elements as well. By default, the "disconnectedCallback" and "connectedCallback" would be called on the element, one after the other. This is + done to maintain compatibility with existing custom elements that predate the moveBefore() method. This means that by default, custom + elements reset their state as if they were removed and re-inserted. In the example above, the impact would be that the + observer would be disconnected and re-connected, and the tab index would be reset.

    + +

    To opt in to a state-preserving behavior while moving, the author can implement a "connectedMoveCallback". The existence of this callback, even if empty, would + supercede the default behavior of calling "disconnectedCallback" and "connectedCallback". "connectedMoveCallback" can also be an + appropriate place to execute logic that depends on the element's ancestors. For example:

    + +
    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();
    +  }
    +}
    +

    Core concepts

    A custom element is an element that is A collection of lifecycle callbacks -

    A map, whose keys are the strings "connectedCallback", - "disconnectedCallback", "adoptedCallback", - "attributeChangedCallback", - "formAssociatedCallback", - "formDisabledCallback", - "formResetCallback", and "formStateRestoreCallback". - The corresponding values are either a Web IDL Function - callback function type value, or null. By default the value of each entry is null.
    +
    A map, whose keys are the strings "connectedCallback", "disconnectedCallback", "adoptedCallback", "connectedMoveCallback", "attributeChangedCallback", + "formAssociatedCallback", "formDisabledCallback", + "formResetCallback", and "formStateRestoreCallback". The corresponding values are either a Web IDL Function callback function type value, or null. By default the value + of each entry is null.
    A construction stack
    @@ -72816,6 +72942,7 @@ dictionary ElementDefinitionOptions {
  • Let lifecycleCallbacks be the ordered map «[ "connectedCallback" → null, "disconnectedCallback" → null, "adoptedCallback" → null, "connectedMoveCallback" → null, "attributeChangedCallback" → null ]».

  • @@ -73263,6 +73390,9 @@ customElements.define("x-foo", class extends HTMLElement {
  • When it becomes disconnected, its disconnectedCallback is called, with no arguments.

  • +
  • When it is moved, its connectedMoveCallback is called, with no arguments.

  • +
  • When it is adopted into a new document, its adoptedCallback is called, given the old document and new document as arguments.

  • @@ -73405,10 +73535,40 @@ customElements.define("x-foo", class extends HTMLElement { data-x="concept-custom-element-definition-lifecycle-callbacks">lifecycle callbacks with key callbackName.

    +
  • +

    If callbackName is "connectedMoveCallback" and + callback is null: + +

      +
    1. Let disconnectedCallback be the value of the entry in definition's + lifecycle callbacks + with key "disconnectedCallback".

    2. + +
    3. Let connectedCallback be the value of the entry in definition's + lifecycle callbacks + with key "connectedCallback".

    4. + +
    5. If connectedCallback and disconnectedCallback are null, then + return.

    6. + +
    7. +

      Set callback to the following steps:

      + +
        +
      1. If disconnectedCallback is not null, then call + disconnectedCallback with no arguments.

      2. + +
      3. If connectedCallback is not null, then call connectedCallback + with no arguments.

      4. +
      +
    8. +
    +
  • +
  • If callback is null, then return.

  • -

    If callbackName is "attributeChangedCallback", then:

    +

    If callbackName is "attributeChangedCallback":

    1. Let attributeName be the first element of args.