google / incremental-dom

An in-place DOM diffing library
http://google.github.io/incremental-dom/
Apache License 2.0
3.54k stars 180 forks source link

Extended custom element creation #419

Open dy opened 5 years ago

dy commented 5 years ago

For now elementOpen supports custom element tag names, but what if it's required to create custom element, extended from built-in?

Like in this example from MDN:

let el = document.createElement('ul', { is: 'expanding-list' })

That seems to be not implemented: https://github.com/google/incremental-dom/blob/9a78593b62a6000a2942c36f92025d7af81bb8aa/src/nodes.ts#L70

Is there any possible workaround or solution to that? Tricks in notifications.nodesAdded?

Any hope that will ever get added?

sparhami commented 5 years ago

Incremental DOM's current API does not support is directly, though a new one could be added in the future. While this is standard, I think the API is still a bit contentious, and I would like to see all major browsers adopt this before making an API change to support this functionality first class.

There are currently two options if you need to create such a custom element. Given a custom element definition:

class MyList extends HTMLUListElement {
  constructor() {
    super();
  }
}
customElements.define('my-list', MyList , { extends: "ul" });
  1. If you have the element definition, pass the constructor to Incremental DOM:
elementOpen(MyList, key, statics);
...
elementClose(MyList);
  1. If you don't have the element definition (e.g. need to asynchronously load the definition):
function MyListConstructor() {
  return document.createElement('ul', { is: 'my-list' });
}

elementOpen(MyListConstructor, key, statics);
...
elementClose(MyListConstructor);

The second is a little inefficient in that it creates an extra object that gets thrown away immediately (when Incremental DOM uses new), but the cost is likely not a big deal compared to the element creation itself.

dy commented 5 years ago

The second case is actually a blocker, if we persist that extended custom element state. With the next patching the state is lost.

sparhami commented 5 years ago

The second case is actually a blocker, if we persist that extended custom element state. With the next patching the state is lost.

Ah, it doesn't pick it up as a matching node. You can do:

function MyListConstructor() {
  return document.createElement('ul', { is: 'my-list' });
}

MyListConstructor.prototype.toString = function() {
  return 'ul';
};

Unfortunately, there doesn't seem to be a way to distinguish this from a plain <ul> as far as DOM diffing goes. So if you had a <ul> adjacent, they could get reused depending on the control flow. As a result, I would strongly recommend enforcing usage of a key if you are going to use this pattern. Note that keys do not need to be unique anymore, so all your instances of the extended custom element can share the same key.

sparhami commented 5 years ago

Actually, looking at it again, the toString is only needed if you server-side render the extended element. If you client-side render it, it should reuse the element correctly.

If that is not the case, do you have an example where the element is not being reused?

dy commented 5 years ago

My bad, I overlooked that I should’ve passed the same function each render call. It works like a charm, thanks for your reply.