- Background
- Introducing CHTML - One More Technology Suite, Side-By-Side With Web Components
- Examples
- Conclusion
After over many years now of trying to make success of web components, it has still been necessary to propose additional platform features to better support modern UI development. We find a good number of proposals at WICG and GitHub, and important posts out there in the community in this problem domain. In this document, I have explored having a new technology suite that brings platform support for modern UI development paradigms - bindings, reactivity, component distribution, and lots more.
While we could go on and on talking over additions to Web Components, this proposal is pursuant to empowering the platform as a whole, not just a subset of it. As we will see, the problems dicussed here aren't just deficiencies in Web Components; these are gaps at the language level that we've got to address at the language level!
* My personal conclusions from exploring this path brings us close, by coincidence, to some of the disparate ideas from across the web. This is more than enough to establish the problems in front of us. I have been able to reference a few of these related ideas down the line and can see each of them finding their fulfillment in this one suite. It seems we can finally call it a day!
-
A Roles API. We've historically lacked a way to access elements by roles and subroles. Web Components still did not provide a way to access components and subcomponents by their roles. We see our markup somewhat following a roles model but lack a standard way to draw the role-to-role relationships. We resort to CSS class-naming conventions (like BEM) to try to achieve this but acquire more of a problem than a solution; and we lose so much time in the process!
A good case study is this dialog-modal example on WAI-ARIA Practices. Here we find a collection of widgets each with the role
role="dialog"
and structural class names likeclass="dialog_label"
,class="dialog_form"
,class="dialog_form_item"
,class="label_text"
,class="dialog_desc"
,class="dialog_form_actions"
, etc. With some effort, we can figure out the role structure of the first dialog to be:dialog - <div role="dialog"> |-label - <h2 id="dialog1_label" class="dialog_label"> |-form - <div class="dialog_form"> | |-item (Street) - <div class="dialog_form_item"> | |-item (City) - <div class="dialog_form_item"> | |-item (State) - <div class="dialog_form_item"> | |-item (Zip) - <div class="dialog_form_item"> | |-item (Special instructions) - <div class="dialog_form_item"> |-form_actions - <div class="dialog_form_actions">
Now, we find the associated JavaScript to be tightly-coupled to the implementation details of the component (with queries like
element.parentNode
,getElementById()
), and here comes the specificity wars and volatility associated with this approach. Contrast this with a more bankable roles API (in a form likedialog->label
,dialog->form
, etc) that lets a component hide its implementation details. (See the point with Stuart P.'s Parts and Walls proposal from 2015.)So far, Web Components has seen the point with exposing a component's internals by name. We have:
- Shadow Parts - for styling purposes;
- Slots - for composition purposes.
What's missing everywhere is a structural API for applications.
But, should this be considered as an addition to Custom Elements? Here is where we've got to look at the bigger picture of the HTML language as a whole, and this I come bringing in this suite as Scoped HTML.
-
Bindings. Our idea of a UI component is of a UI block that is bound to a corresponding part of an application. We want to be able to keep the UI in sync with application state without having to manually track and apply changes. This magic has swept modern UI develipment so much that every framework out there is featuring the joint concept of bindings and reactivity. This is where these frameworks have got to really outshine the platform. This becomes a critically-needed feature on the platform if we must natively drive UI development. (I found this early-stage idea for a template syntax by Jonathan Kingston all the way from 2014. See it come back in this proposal from 2017.)
While these first wave of ideas have considered a new
{{
syntax}}
on top of HTML, we find that the platform already has what it takes to unlock this feature, considering the language's existing provision for logic. I explored this possibility of using standard scripts in HTML for reactve presentational logic and come bringing it as Scoped JS.
-
Slots-Based Composition. Slots-based composition is one killer feature to come to HTML. Unfortunately, it was imagined for use only in the Shadow DOM, whereas, in fact, the usecase of composition is everywhere as long as the UI is concerned. Even as it is, the whole idea of slots-based composition falls apart for apps that have to render server-side as the Shadow DOM still can't be serialized for client-side hydration. And since the Shadow DOM is all we've got for this, we're now having to think counter-intuitive to its concept of encapsulation, just so we could have it for a more general usecase!
For a moment, how about slots-based composition for HTML generally? That could save us from looking to the Shadow DOM for something it wasn't designed for! This is the prospect for us in the HTML Partials feature of this suite.
-
Scoping. We've long ago realized the need for scoping in HTML - a way to keep things out of the browser's global scope. We tasted Scoped CSS but were forced to drop it in favour the Shadow DOM's encapsulation. What we've realized at this point is that the Shadow DOM doesn't fit the generic usecase of being able to scope a stylesheet. We're now back to the selector wars as we style in the open HTML.
The time is now, more than ever, right to return Scoped CSS as we look to the platform to better support modern UI development. Why, even frameworks out there can see this as a great feature for the UI. Making this a general language feature, and not just a feature of the Shadow DOM, is everyone's position.
At the same time, we find the problem we're trying to solve with Scoped CSS also present with scripts (usually presentational logic) that have to use CSS selectors to manipulate the document. And while selecting by IDs would give perfect specificity, a terrible challenge lies in writing collision-free names, as IDs, like CSS selectors, also share one global namespace - a problem that has been underdiscussed. This has given us three things to wresttle with at the global level: CSS selectors, IDs, and, by extension, scripts. This proposal soughts to address all three scoping issues with three new language features: Scoped HTML, Scoped CSS, and Scoped JS!
CHTML is a suite of new DOM features that brings language support for modern UI development paradigms: a component-based architecture, data binding, and reactivity. It aims to make it possible to build functional user interfaces out of language primitives and native APIs. This will be helping us bank more on the platform and less on abstractions.
Now, instead of introducing totally new ideas, CHTML chooses to look within to find new possibilities with existing platform features. It is designed to work side-by-side with Web Components and to bring some of Shadow DOM's exclusive features to the open HTML.
* I'm excited to say that CHTML is already a working prototype today and obtainable from the Web-Native project. By simply including the polyfill on a page, the ideas discussed here can be seen. In fact, the Web-Native website is a live example of CHTML at work.
This feature let's an element establish its own naming context for descendant elements. It makes it possible to keep IDs out of HTML's global namespace and gives us a document that is structured as a hierarchy of scopes and subscopes.
Scopes are designated with the root
Boolean attribute.
<div root>
<div id="some-id"></div>
</div>
Scopes and Subscopes
Below is a hierarchy of scopes.
<body>
<article root id="continents">
<b>Continents</b>
<section root id="europe">
<b>Europe</b>
<div id="about">
<b>About Europe</b>
<ul>
<li>Fact1</li>
<li>Fact2</li>
</ul>
</div>
<div id="countries">
<b>Countries in Europe</b>
<ul>
<li>Country1</li>
<li>Country2</li>
</ul>
</div>
</section>
<section root id="africa">
<b>Africa</b>
<div id="about">
<b>About Africa</b>
<ul>
<li>Fact1</li>
<li>Fact2</li>
</ul>
</div>
<div id="countries">
<b>Countries in Africa</b>
<ul>
<li>Country1</li>
<li>Country2</li>
</ul>
</div>
</section>
</article>
</body>
Scoped HTML turns out the ideal, clean, modular naming convention compared to current class-based alternatives like we saw earlier.
Here, what we get is simply a hierarchy of scopes:
continents
|- europe
| |- about
| |- countries
|- africa
|- about
|- countries
A hierarchy of scopes with scoped IDs makes it possible to have a roles API, or a UI component model, that an application can bank on. Scoped HTML exposes a new DOM property idrefs
for accessing scope trees.
// The regular querySelector() function would give us the #article element
let continents = document.querySelector('#continents');
// The new idrefs DOM property would give us the structural parts
let europe = continents.idrefs.europe;
let africa = continents.idrefs.africa;
// Deeply-nested...
let aboutAfrica = continents.idrefs.africa.idrefs.about;
Now, much naming wars, structural guesswork, and inefficient DOM queries can be eliminated with a simple, bankable API.
An element's .idrefs
property is implemented as a live object that reflects the element's scope tree in real time. CHTML also supports the Observer API for change detection; Obs.observe()
can thus be used to observe additions and removals on the scope tree.
Obs.observe(element.idrefs, changes => {
console.log(changes.map(change => change.name));
});
With the introduction of Scoped HTML, a few backwards-compatible changes will now be neccessary:
- The regular ID selector
#
can be made to respect scope boudaries. - The forward slash
/
can be used to denote a scope boundary. Querying deeply-scoped IDs would now look like:#continents / #europe / #about
. - Two new query selectors (
scopeSelector()
andscopeSelectorAll()
) would now be created, or the regularquerySelector()
andquerySelectorAll()
DOM methods can be upgraded to support scope boudaries. - URL fragment identifiers would now need to be path-based to reference an element deep in the scope hierarchy. (We find that discussions are already underway here, and probabbly elswhere too, to reform the nature of fragment identifiers; this becomes a good time to bake-in path notation.)
All of Scoped HTML, excluding Scope Selectors, is currently implemented in the CHTML at Web-Native. This implementation makes use of mutation observers. It is also making do with the scoped:id
attribute instead of the actual id
attribute - to respect the current validition of HTML documents.
On the feasibility of a native implementation of scoped IDs, we find that this can come no risk to pre-CHTML websites as their lack of roots can forever keep their IDs scoped to the document root.
Read the full Scoped HTML docs
CHTML reproposes the ability to scope a stylesheet as a language feature and not just a Shadow DOM feature. With component-oriented HTML in mind, we now have a better use-case for Scoped CSS.
With support for Scope Selectors, Scoped CSS could look like this:
<div>
<style scoped>
:root {
color: red;
}
#title {
font-weight: bold;
}
#content {
font-weight: normal;
}
#content / #sub-content {
font-style: italics;
}
</style>
<div>
<div id="title"></div>
<div root id="content">
<div id="sub-content"></div>
</div>
</div>
</div>
This feature makes it possible for scripts to be scoped to their containing element and completely out of the global browser scope. Scoped scripts have their this
variable implicitly bound to their immediate container element. They are defined with the scoped
Boolean attribute.
<div>
<div class="message">This task is now complete!</div>
<div class="exit" title="Close this message.">X</div>
<script scoped>
this.querySelector('.exit').addEventListener('click', () => {
this.remove();
});
</script>
</div>
Other variables in a scoped script are to be explicitly-bound from external values. Variables are bound by name, as in the message
variable below.
<body>
<div id="alert">
<div class="message"></div>
<div class="exit" title="Close this message.">X</div>
<script scoped>
let messageEl = this.querySelector('.message');
messageEl.innerHTML = message;
</script>
</div>
<script>
let alertEl = document.querySelector('#alert');
alertEl.bind({
message: 'This task is now complete!',
});
</script>
</body>
Scoped JS gives us this special ability to bring DOM manipulation logic closer to their targets and away from an application. It has presentational logic as its sole responsibility and thus helps us keep the main application layer void of the implementation details of the UI. As shown above, an application simply binds its hard-earned values and is freeeeee!
Scoped JS is to HTML what a template syntax is to a UI component framework. But by coming in a script tag, as opposed to being in text-based string interpolation, we get to avoid repurposing HTML's plain text content for logic, or the required compile step that must sniff those tokens for special interpretation. Overall, we can now finally use the web's exact language for logic rightly for logic! (Contrast this with Apple's proposal from 2017 for the problem domain.)
Scoped JS follows the normal top-down execution of a script. Calling the .bind()
method with different variable-bindings reruns the script top-down. But as a UI binding langauge, it also features Selective Execution where we update a variable to rerun only the corresponding statements within the script that depend on the update - skipping the other statements. This makes for the most-efficient way to keep a block of the UI in sync with little updates from an application.
To update a variable or multiple variables, call .bind()
with false
as a second paremeter.
alertEl.bind({
variable2: 'New value',
variable5: 'New value',
}, false);
Also, Scoped JS exposes a new DOM property .bindings
for selectively updating an element's bindings.
alertEl.bindings.variable5 = 'New value',
This is illustrated in the clock below.
<body>
<div id="clock">
<div class="greeting"></div>
<div class="current-time"></div>
<script scoped>
this.querySelector('.greeting').innerHTML = greeting;
this.querySelector('.current-time').innerHTML = currentTime;
</script>
</div>
<script>
let clockEl = document.querySelector('#clock');
clockEl.bind({
greeting: 'Good Afternoon!',
currentTime: '00:00:00',
});
// Clock ticks
setInterval(() => {
clockEl.bindings.currentTime = (new Date).toLocaleString();
}, 100);
</script>
</body>
Scoped JS also supports the Observer API for object observability. With Observer, Scoped JS is able to respond to mutations made directly on the bound data object. So the clock above could be driven by direct updates to the data object.
<script>
let clockState = {
greeting: 'Good Afternoon!',
currentTime: '00:00:00',
};
document.querySelector('#clock').bind(clockState);
// Clock ticks
setInterval(() => {
Obs.set(clockState, 'currentTime', (new Date).toLocaleString());
}, 100);
</script>
Scoped JS is also able to pick up deep mutations on the bound object for statements that reference deep into the object, as in clock.currentTime
.
<body>
<div id="clock">
<div class="greeting"></div>
<div class="current-time"></div>
<script scoped>
this.querySelector('.greeting').innerHTML = clock.greeting;
this.querySelector('.current-time').innerHTML = clock.currentTime;
</script>
</div>
<script>
let state = {
clock: {
greeting: 'Good Afternoon!',
currentTime: '00:00:00',
},
};
document.querySelector('#clock').bind(state);
// Clock ticks
setTimeout(() => {
Obs.set(state.clock, 'currentTime', (new Date).toLocaleString());
}, 100);
</script>
</body>
On updating a variable, the dependency chain within the script is followed even when broken into local variables. Below, a change to clock.currentTime
will still propagate through variable1
and variable2
. (While the first and last statements in the script are left untouched, as expected.)
<body>
<div id="clock">
<div class="greeting"></div>
<div class="current-time"></div>
<script scoped>
this.querySelector('.greeting').innerHTML = clock.greeting;
let variable1 = clock.currentTime;
this.style.backgroundColor = 'yellow';
let variable2 = variable1;
this.querySelector('.current-time').innerHTML = variable2;
this.style.color = 'blue';
</script>
</div>
</body>
From a high-level view, we would now be striking new cords:
<style scoped>
(for styling) -<script scoped>
(for logic).element.style
(for styling) -element.bindings
(for logic).
By default, scoped scripts have no access to anything besides what is explicitly bound into the scope. But they also have an idea of a global scope - that is, bindings seen by every scoped script. This global scope is created by binding on the document
object itself, using a new document.bind()
method.
document.bind({
greeting: 'Good Afternoon!',
});
Providing false
as a second parameter to this method performs Selective Execution.
There is also the document.bindings
property for selectively updating globals.
document.bindings.greeting = 'Good Evening!';
By design, Scoped JS parses scoped scripts immediately they land on the DOM, but runs them only after the global scope has been initialized with document.bind()
or the document.bindings
property. Newer scipts are run immediately after this global runtime initilization. But the runtime of an individual script will begin before the global one on calling the element's .bind()
method or assigning to its .bindings
property.
Alternatively, the autorun=true
directive may be set on the CHTML META tag. The autorun
Boolean attribute may also be set on individual script elements.
Also, an element may receive bindings before its scoped script is appended or is ready to run. The element's runtime begins the first time both are available.
Scoped JS features a way to handle syntax or reference errors that may occur with scoped scripts. Normally, these are shown in the console as warnings. But they can be silenced by setting a directive on the CHTML META tag. Induvidual scripts may also be given a directive, to override whatever the global directive is.
<html>
<head>
<meta name="chtml" content="script-errors=0;" />
</head>
<body>
<h1></h1>
<script scoped errors="1">
this.querySelectorSelectorSelector('h1').innerHTML = headline;
</script>
</body>
</html>
* The trailing semi-colon (;) in the CHTML META tag is optional.
The script tag of a scoped script is not always needed for the lifetime of the page. They are discarded by default after parsing. But when a page is rendered on the server and has to be hydrated by the browser, it becomes necessary to retain these scripts for revival on the browser. This feature is designed to be explicitly turned on with a directive on the CHTML META tag.
<html>
<head>
<meta name="chtml" content="isomorphic=true;" />
</head>
<body>
<h1></h1>
<script scoped>
this.querySelector('h1').innerHTML = headline;
</script>
</body>
</html>
Now, running the code document.bind({headline: 'Hello World'})
both on the server and on the browser should give us the same result.
* The trailing semi-colon (;) in the CHTML META tag is optional.
Environment-Specific Bindings
Sometimes, we want certain bindings to apply only on the server; sometimes, only on the browser. For example, animation is only a thing in the browser. This is the perfect use-case for conditionals.
<div>
<script scoped>
if (condition) {
this.animate(...);
}
</script>
</div>
Above, condition
could be a simple question about the current environment, and this is acheivable by simply exposing a global env
variable: document.bind({env:'server', headline: 'Hello World'})
.
<div>
<script scoped>
if (env !== 'server') {
this.anumate([
{color:'red'},
{color:'blue'},
], {duration:600,});
}
</script>
</div>
All of Scoped JS is currently implemented in the CHTML at Web-Native, but with the use of a custom MIME type for the script tag: <script type="scoped"></script>
. A custom MIME type helps exclude the script from normal browser processing. Native implementation may want to really use the scoped
Boolean attribute as in <script scoped></script>
, to correspond with <style scoped></style>
and to retain the role of the type
attribute for scoped scripts.
The implementation is based on the JSEN library - an experimental implementation of a subset of the JavaScript language.
This feature brings the ability to define, extend, import/export reusable HTML snippets using the template, partials, and slots paradigm.
A template is a collection of independent partials that can be consumed from anywhere in the main document.
<head>
<template name="template1">
<div id="partial-1"></div>
<div id="partial-2"></div>
</template>
</head>
An element in the main document, called the implementation block or the composition area, can define slots, and then, point to a template to have the template's partials each mapped to a slot.
<html>
<head>
<template name="template1">
<div id="partial-2" slot="slot-1"></div>
<div id="partial-2" slot="slot-2"></div>
</template>
</head>
<body>
<div template="template1">
<h2>I have slots</h2>
<slot name="slot-1"></slot>
<span>
<slot name="slot-2"></slot>
</span>
</div>
</body>
</html>
Composition takes place and the slots are replaced by the template's partials. The block is said to have implemented the template.
<html>
<head>
<template name="template1">
<div id="partial-2" slot="slot-1"></div>
<div id="partial-2" slot="slot-2"></div>
</template>
</head>
<body>
<div template="template1">
<h2>I have slots</h2>
<div id="partial-2" slot="slot-1"></div>
<span>
<div id="partial-2" slot="slot-2"></div>
</span>
</div>
</body>
</html>
An implementation block can implement another template by simply pointing to it; slots are disposed off of their previous slotted contents and recomposed from the new template.
The <slot>
element, even though replaced, is never really destroyed. It returns to its exact position whenever the last of its slotted elements get deleted, or whenever the slot has no corresponding partial in the next implemented template.
Now, a template is to the composition block what the Light DOM of a custom element is to the Shadow DOM - providing slottable contents for slots; called slottables in Web Components, partials in CHTML.
HTML Partials also supports Default Slots. Here, a template's direct children without an explicit slot
attribute are slotted into the Default Slot in the implementation block.
Unscoped Slots
By default, slots are scoped to their containing implementation block. But the <slot>
element may also be used independent of an implementation block to point to its own template.
<html>
<head>
<template name="template1">
<div slot="slot-1"></div>
<div slot="slot-2"></div>
</template>
<template name="template2">
<div slot="slot-1"></div>
<div slot="slot-2"></div>
</template>
</head>
<body>
<div template="template1">
<h2>I have slots</h2>
<slot name="slot-1"></slot>
<span>
<slot name="slot-1" template="template2"></slot>
</span>
</div>
<slot name="slot-2" template="template1"></slot>
</body>
</html>
In HTML Partials, slots may be defined with extra properties that a slotted element can inherit. Every element slotted in its place will take on these properties.
Both attributes and content can be inheritted this way.
Attributes
A slot's attributes, other than the slot-exclusive name
and template
attributes, are inheritted by every slotted element.
When a partial inherits attributes from a slot, inheritted attributes are made to take priority over any existing attributes. On inheriting single-value attributes, like the id
attribute, any such attribute is replaced on the slotted element. On inheriting space-delimitted attributes, like the class
attribute, new and non-duplicate values are placed after any existing values. On inheriting key/value attributes, like the style
attribute, new declarations are placed after any existing declarations, making CSS cascading work on the style
attribute.
Below, we are using Slot Attributes to recompose the same partial differently for each usecase.
<html>
<head>
<template name="template1">
<div slot="slot-1"></div>
<div slot="slot-2"></div>
</template>
</head>
<body>
<div template="template1">
<slot name="slot-1" id="headline" style="color:red"></slot>
</div>
<slot name="slot-1" template="template1" style="color:blue"></slot>
</body>
</html>
Content
Normally, a slot can have default content that renders before slotting takes place. But this content can instead be defined as a new set of partials that can be implemented by slotted elements. This time, the slot element gets to act as the template and the slotted element as the implementation block. (In the light/shadow paradigm, this is the slot element acting as an element's Light DOM and the slotted element as its Shadow DOM.)
To implement a slot, a partial would set its template
attribute to the keyword @slot
instead of pointing to an actual template.
<html>
<head>
<template name="template1">
<!-- I am a recomposable partial. My ideal slot provides the partials for me -->
<div slot="slot-1" template="@slot">
<slot name="slot-1-1"></slot>
</div>
<!-- I am a regular partial -->
<div slot="slot-2"></div>
</template>
</head>
<body>
<div template="template1">
<!-- I am an implementable slot. My ideal partial defines slots -->
<slot name="slot-1">
<div slot="slot-1-1"></div>
</slot>
</div>
</body>
</html>
Templates may be nested for organizational purposes.
<template name="template1">
<div slot="slot1"></div>
<div slot="slot2"></div>
<template name="nested1">
<div slot="slot3"></div>
<div slot="slot4"></div>
</template>
<template name="nested2">
<div slot="slot5"></div>
<div slot="slot6"></div>
</template>
</template>
Nested templates are referenced using a path notation:
<div template="template1/nested1">
</div>
Templates may reference remote content using the src
attribute. (We can even find similar ideas here and here, and probbably elsewhere.)
Remote file: http://localhost/templates.html
<div slot="slot-1"></div>
<div slot="slot-2"></div>
<template name="nested1">
<div slot="slot3"></div>
<div slot="slot4"></div>
</template>
<template name="nested2">
<div slot="slot5"></div>
<div slot="slot6"></div>
</template>
<p></p>
Document: http://localhost
<head>
<template name="template1" src="/templates.html"></template>
</head>
Where remote templates are detected in a document, slots are resolved after all templates have loaded their content.
HTML Partials introduces a few new DOM properties for working with composition.
For the document object:
-
.templatesReadyState
- This property reflects the document's loading status of remote templates:loading
- This is the initial value of this property.complete
- This is the value of this property when templates are done loading, or when there are no remote templates at all. For this state, thetemplatesreadystatechange
event is fired on the document object.
-
.templates
- This property represents the list of templates in the document. Templates are exposed here by name. Sodocument.templates.template1
should return the template element used in the examples above.Very interestingly,
document.templates
has even been proposed here!
For the <template>
element:
-
.partials
- This property represents the list of partials defined by the template. It is an object holding a reference to partials by name. Unnamed partials are treated as having the name default. So, for the template below,<template name="template1"> <div slot="one"></div> <div slot="two"></div> <div slot="default"></div> <p></p> </template>
accessing
document.templates.template1.partials.one
should return an array containing the first<div>
; whiledocument.templates.template1.partials.default
should return an array containing the last<div>
and<p>
. -
.templates
- This property represents the list of templates nested within the template. It is an object holding a reference to templates by name.<template name="template1"> <template name="nested1"></template> <template name="nested2"> <div slot="one"></div> </template> </template>
accessing
document.templates.template1.templates.nested1
should return the first nested template, whiledocument.templates.template1.templates.nested2
the second nested template. And the nesting can go on as much as code organization requires.
For every element:
.template
- This property represents a copy of the<template>
element referenced by an element. So if an element implements a template as in<div template="html/temp"></div>
, thenelement.template
should be a copy of the<template>
at themodule/temp
namespace;element.template.partials.default
should thus return an array like the above.
For the <slot>
element:
.slottedElements
- This property represents the list of partials slotted into a slot. (Much like theHTMLSlotElement.assignedElements()
method.).resolve()
- This method, without arguments, is used to programatically resolve a slot from the appropriate partial in the template given in context..empty([silently = false])
- This method is used to programatically empty the slot of its partials, thereby triggering the restoration of the slot element itself. To empty the slot silently without restoring the original slot element, providetrue
on the first parameter.
For slotted elements:
.slotReference
- This property gives a reference to the slot element an element was assigned to. (Much like theSlottable.assignedSlot
property.)
Persistent slots is the promise of slots-based composition; placeholders must never really lose their place. This promise is easy to keep on a live DOM, as slot positions can be easily maintained - even after a slot is replaced. Where the challenge lies is when rendering happens on the server and slots replaced but all have to be hydrated on the browser. How would be the browser figure out the positions of replaced slots?
HTML Partials addresses this by serializing slot elements as comment nodes with a view to recreating the original slot elements from these comments on getting to the browser. This way, composition is able to continue. Now in the browser, deleting a server-slotted element, for example, should trigger the restoration of the original slot element; changing the template
attribute of any element should dispose off all its server-slotted elements and recompose the block from the new referenced template.
Before Rendering on the Server
<html>
<head>
<template name="template1">
<div slot="slot-1"></div>
<div slot="slot-2"></div>
</template>
</head>
<body>
<div template="template1">
<slot name="slot-1" id="headline" style="color:red">Default Headline</slot>
</div>
<slot template="template1" name="slot-1" style="color:blue"></slot>
</body>
</html>
After Rendering on the Server
<html>
<head>
<template name="template1">
<div slot="slot-1"></div>
<div slot="slot-2"></div>
</template>
</head>
<body>
<div template="template1">
<div slot="slot-1" id="headline" style="color:red"></div>
<!-- <slot name="slot-1" id="headline" style="color:red">Default Headline</slot> -->
</div>
<div slot="slot-1" style="color:blue"></div>
<!-- <slot template="template1" name="slot-1" style="color:blue"></slot> -->
</body>
</html>
Now on the Browser
Find and delete the server-slotted element with ID #headline
. The original slot element should now be restored and ready to replaced on the next composition.
<html>
<head>
<template name="template1">
<div slot="slot-1"></div>
<div slot="slot-2"></div>
</template>
</head>
<body>
<div template="template1">
<slot name="slot-1" id="headline" style="color:red">Default Headline</slot>
<!-- <slot name="slot-1" id="headline" style="color:red">Default Headline</slot> -->
</div>
<div slot="slot-1" style="color:blue"></div>
<!-- <slot template="template1" name="slot-1" style="color:blue"></slot> -->
</body>
</html>
Enabliing Slots Serialization
Since slots serialization is only necessary for isomorphic pages, this feature is designed to be explicitly turned on on the CHTML META tag.
<html>
<head>
<meta name="chtml" content="isomorphic=true;" />
</head>
<body></body>
</html>
* The trailing semi-colon (;) in the CHTML META tag is optional.
All of HTML Partials is currently implemented in the CHTML at Web-Native, but with mutation observers and a little more verbosed attribute and element names. The <slot>
element is implemented as <partials-slot>
. The slot
attribute used by partials in a <template>
is implemented as partials-slot
.
Read the full HTML Partials docs
Here are some of CHTML's possibilities. To try the examples below with the current implementation of CHTML, please visit docs.web-native.dev/chtml to find the corresponding sample code in the examples given.
Below is a TODO list composed from a JavaScript array using Scoped HTML, Scoped JS in combination with the HTML Partials API.
<html>
<head>
<template name="items">
<li>
<script scoped>this.innerHTML = desc;</script>
</li>
</template>
</head>
<body>
<div root id="todo" template="items">
<h2 id="title"></h2>
<ul id="items"></ul>
<script scoped>
this.idrefs.title.innerHTML = title;
items.forEach(itemBinding => {
let itemElement = this.template.partials.default[0].cloneNode(true);
itemElement.bind(itemBinding);
this.idrefs.items.append(itemElement);
});
</script>
</div>
<script>
document.querySelector('#todo').bind({
title: 'My TODOs',
items: [
{desc: 'TODO-1'},
{desc: 'TODO-2'},
{desc: 'TODO-3'},
],
});
</script>
</body>
</html>
We could even add the ability to add/remove items. For the remove feature, we'd add a click event listener to the item element definition. For the add feature, we'd add a button to the TODO container that calls the addItem()
method of the TODO application.
<html>
<head>
<title>A TODO Example</title>
<template name="items">
<li root>
<span id="desc"></span>
<button id="remover">Remove</button>
<script scoped>
this.idrefs.desc.innerHTML = desc;
this.idrefs.remover.addEventListener('click', () => this.remove());
</script>
</li>
</template>
</head>
<body>
<div root id="todo">
<h2 id="title"></h2>
<ol id="items" template="items"></ol>
<button id="adder">Add</button>
<script scoped>
this.idrefs.title.innerHTML = title;
$(this.idrefs.items).itemize(items, (el, data) => el.bind(data));
this.idrefs.adder.addEventListener('click', () => addItem());
</script>
</div>
<script src="//unpkg.com/@web-native-js/observer/dist/main.js"></script>
<script src="//unpkg.com/@web-native-js/play-ui/dist/main.js"></script>
<script>
// Declare our tools
let Obs = window.WebNative.Observer;
let $ = window.WebNative.PlayUI;
// Create the app
let todo = {
$,
title: 'My TODOsrr',
items: [
{desc: 'Task-1'},
{desc: 'Task-2'},
{desc: 'Task-3'},
],
addItem() {
window.todoItems.push({desc: prompt('Task description'),});
},
};
// Bind the app to the UI
document.querySelector('#todo').bind(todo);
// Make the items available globally
// so that we can always manipulate them
window.todoItems = Obs.proxy(todo.items);
</script>
</body>
</html>
This example makes an SPA of templates and slots composition. Below, we're using the two template elements to each represent a route - each holding partials that are unique to a route. Then we point the body element to implement the template whose namespace matches the current URL.
<html>
<head>
<template name="route">
<template name="home">
<h1 slot="headline">
Welcome Home!
</h1>
<p slot="content">
<a href="#/about">About Me</a>
</p>
</template>
<template name="about">
<h1 slot="headline">
About Me!
</h1>
<p slot="content">
<a href="#/home">Back to Home</a>
</p>
</template>
</template>
</head>
<body template="route/home">
<header></header>
<main>
<div id="banner">
<slot name="headline">404</slot>
</div>
<div>
<slot name="content">Page not Found!</slot>
</div>
</main>
<footer></footer>
<script>
window.addEventListener('popstate', e => {
let path = document.location.hash.substr(1);
document.body.setAttribute('template', 'route' + path);
});
</script>
</body>
</html>
Navigate to a route that does not begin with #/home
or #/about
, you should see the default content showing 404.
That CHTML is a foundational technology just gives us every room to bring our own tooling. This example shows how we could make a DOM abstraction API, like jQuery, available to scoped scripts.
Below, we're simply binding the $
variable globally for use in every scoped script.
<body>
<div root id="alert">
<div id="message"></div>
<script scoped>
$(this.idrefs.message).html(message);
</script>
</div>
<script src="https://ajax.googleapis.com/ajax/libs/jquery/3.5.1/jquery.min.js"></script>
<script>
document.bind({$: window.jQuery});
document.querySelector('#alert').bind({
message: 'This task is now complete!',
});
</script>
</body>
Tooling can also save the day on the efficiency of DOM manipulation. Generally, surgically updating the DOM may have performance implications on the UI, as arising from layout thrashing (see this article on Web Fundamentals). But we also don't need as much as a Virtual DOM. A technique like that of fast DOM could just suffice. The upcoming PlayUI library has a design that brings this technique (what we like to call async DOM) in the syntax-sugar of jQuery.
<body>
<div root id="alert">
<div id="message"></div>
<script scoped>
$(this.idrefs.message).html(message).then(() => {
// Do something next
});
</script>
</div>
<script src="https://ajax.googleapis.com/ajax/libs/jquery/3.5.1/jquery.min.js"></script>
<script src="//unpkg.com/@web-native-js/play-ui/dist/main.js"></script>
<script>
document.bind({$: window.WebNative.PlayUI});
document.querySelector('#alert').bind({
message: 'This task is now complete!',
});
</script>
</body>
Until now, we've suffered the consequences of the lack of platform support for a component-based HTML. It seemed that the days were gone for doing anything in vanilla HTML, CSS and JavaScript. But with the CHTML suite, we can now finally #justUseThePlatform!
* It seems that the letters "CHTML" should read "Component-Oriented HTML", "Composable HTML", or something else altogether that describes language-wide features for a component-based UI.
At a time when native web authoring languages HTML, CSS, and JavaScript have been put behind a compiler; everything has left its place! So here comes one of the biggest motivations for CHTML - to have a modern UI technology that supports the natural use of the web's authoring languages. With platform support, we can collectively put much dependencies behind us and get back to writing code that hits the ground running.