Web Components: The secret weapon to help you power the web
Kevin Schaaf of the Polymer Project and Caridy Patino of Salesforce had talked about the state of web components at Google I/O 2019. They had told that the how popular are web components? If you are using the web today, then you will be probably using web components. According to count, somewhere between 5% and 8% of all page loads today use one or more web components. It makes web components one of the most successful new web platform features shipped in the last five years. You can find web components on sites you probably use every day, like YouTube, Content Sites, Design Systems, GitHub, etc. They're also used on many publishing and news sites built with AMP - AMP components are also web components. Now, many enterprises are also adopting web components. According to the report, Web Components are going to fundamentally change the nature of HTML.
At first glance, you may seem them like a very complicated set of new technologies, but Web Components are built around a simple premise. Developers should be free to act like browser vendors, extending and vocabulary of HTML itself. If you haven't used these technologies yet, then this article has a very simple message for you. If you're already familiar with DOM APIs and HTML elements, then I think you are already an expert at Web Components. To know why web components are so important, we need to know how we've hacked around the lack of Web Components. To understand, We will through the process of consuming a third-party widget.
First, I will include the widget's CSS and JavaScript:
<link rel="stylesheet" type="text/css" href="widget.css" /> <script src="widget.js"></script>
Now, I need to add placeholder elements to the page where our widgets will be inserted.
<div data-widget></div>
Finally, when the DOM will be ready, we will reach back into the document, now, we find the placeholder elements and will instantiate our widgets:
$(function() { $('[data-widget]').widget(); });
Now, we have introduced a custom widget to the page, but it is not aware of the browser's element lifecycle. It will become clear if we will update the DOM:
el.innerHTML = '<div data-widget></div>';
Since it is not a typical element, now we will manually instantiate any new widgets as we have updated our document:
$(el).find('[data-widget]').widget();
We can avoid this constant two-step process to completely abstract away DOM interaction. Unfortunately, it is a pretty heavy-handed solution that will usually result in widgets being tide to particular libraries or frameworks.
Get started with web components
There are a lot of great ways to get started with web components.
If you're building a web app, then try to use available web components. Here are just a few examples:
- As Google vends its own Material design system as web components: Material Web Components.
- The Wired Elements are a cool set of web components that feature a sketchy, hand-drawn look.
- In the Market, there are great special-purpose Web Components, which you can drop into any app to add 3D content.
If you're developing a design system for your company, or you're vending a single component or library that you want to be usable in any environment, consider authoring your components using web components. Also, you can use the native web components APIs, but they're pretty low-level, so to make the process easier there are a number of libraries available.
Component Soup
Once, we have instantiated our widgets, our placeholder elements will have been filled with third-party markup:
<div data-widget> <div class="widget-foobar"> <input type="text" class="widget-text" /> <button class="widget-button">Go</button> </div> </div>
This markup will now sit in the same context as our application markup. Its internals will be visible when we will traverse the DOM, and the styles for this widget will exist in the same global context as our styles, leading to high risk of style clashes. All of their classes must be carefully namespaces with widget- (or something similar) to avoid naming collisions. Our code will now all mixed up with the third-party code, with no clean separation between the two. Basically, there will be no encapsulation.
Web Components to the rescue
With Web Components, it has become clear what we are missing.
<!-- Import it: --> <link rel="import" href="widget.html" /> <!-- Use it: --> <widget />
In this case, we will import a new custom element with a single import and will use it immediately. More importantly, since <widget /> is an actual element, it will hook into the browser’s element lifecycle, will allow us to add a new instance to the page as we would with any native widget:
el.innerHTML = '<widget />'; // The widget is now instantiated
When we will inspect this element, now we are able to see that it is a single element. However, if we will enable Shadow DOM in our developer tools, we will see something very interesting. This inside hidden element is its private implementation details, in the form of a document fragment:
#document-fragment <div> <input type="text" /> <button>Go</button> </div>
While these elements will be visible to the naked eye, they will be hidden from us when we will traverse the DOM or will write CSS selectors. To the outside world, even when instantiated, our custom widget will still just a single element. Finally, we have a simple, encapsulated widget that will behave exactly like a standard HTML element.
In the interest of time
When we talk about Web Components, we will not be talking about a single technology. We will be talking about a suite of new tools that are each useful in their own right: Custom Elements, Shadow DOM, HTML Templates, HTML Imports, and Decorators. The primary goal of Web Components has to give us the encapsulation we will be missing. Luckily, this goal will be achieved purely with Custom Elements and Shadow DOM. So, in the interest of time, we’ll begin by focusing on these two technologies. Rather than immediately jumping into a list of new browser features, I found it helpful for us to first reacquaint ourselves with what we already know about the native elements we’ve been consuming for years. After all, it doesn’t hurt for us to understand what we’re trying to build.
What we already know about elements
We all know that elements can be instantiated through markup or JavaScript:
<input type="text" />
Next,
document.createElement('input'); el.innerHTML = '<input type="text" />';
We will know that elements are instances:
document.createElement('input') instanceof HTMLInputElement; // true document.createElement('div') instanceof HTMLDivElement; // true
Now, we will know that elements perform their own initialization:
// If we create an input with a value attribute defined... el.innerHTML = '<input type="text" value="foobar" />'; // ...the value *property* is already in sync el.querySelector('input').value;
Next, we will know that elements can respond to attribute changes:
// If we change the value *attribute*... input.setAttribute('value', 'Foobar'); // ...the value *property* updates accordingly input.value === 'Foobar'; // true
Now, we will know that elements can have hidden internal DOM structures:
<!-- A single 'input' provides a complex calendar --> <input type="date" /> // Despite its complexity, to us it's still just a single element dateInput.children.length === 0; // true
Now, we will know that elements have access to child elements:
<!-- We can provide as many 'option' tags as we like --> <select> <option>1</option> <option>2</option> <option>3</option> </select>
Next, we know that elements can provide style hooks to their internals:
dialog::backdrop { background: rgba(0, 0, 0, 0.5); }
Finally, we have known that elements can have their own private styles. Unlike the custom widgets of today, we will never need to manually include CSS for the browser’s native widgets. By understanding all of this, we’ll be well on our way to understanding Web Components. With Custom Elements and Shadow DOM, now we can recreate all of this standard behaviour in our widgets.
Custom elements
Registering a new element can be as simple as this:
var MyElement = document.register('my-element'); // 'document.register' returns a constructor
Now, you might have noticed that our element name contains a hyphen. This will be an important requirement for Custom Elements to ensure our tag names don’t clash with current or future elements. This element will now work like any other native element:
<my-element />
It means that our element will work with all the standard DOM APIs:
document.create('my-element'); el.innerHTML = '<my-element />'; document.create('my-element') instanceof MyElement; // true
Breathing life into our custom element
Currently, It will be a pretty useless element. Let’s give it some context:
// We'll now provide the second argument to 'document.register' document.register('my-element', { prototype: Object.create(HTMLElement.prototype, { createdCallback: { value: function() { this.innerHTML = '<h1>ELEMENT CREATED!</h1>'; } } }) });
In this example, we'll set up the prototype for our custom element, using Object.create to make a new object that will inherit from the HTMLElement prototype. We will define a createdCallback function which will run every time a new instance of the element is created. We can also optionally define attributeChangedCallback, enteredViewCallback and leftViewCallback. Inside our callback, we will modify our new element however we like. In this case, we’ve set its innerHTML. So far we’ll be able to dynamically modify the contents of our custom element, but this will not too different from the custom widgets of today. In order to complete the picture, we will need a way to provide encapsulation to our new element by hiding its internals.
Encapsulation with Shadow DOM
Now, we will modify our createdCallback a bit. This time, we are gonna instead of setting the innerHTML directly on our custom element, we will do something quite different:
createdCallback: { value: function() { var shadow = this.createShadowRoot(); shadow.innerHTML = '<h1>SHADOW DOM!</h1>'; } }
In this example, you can see the words ‘SHADOW DOM!’ when looking at the page, but inspecting the DOM will reveal a single, empty <my-element /> tag. Instead of modifying the containing page, we will create a new shadow root inside our custom element using this.createShadowRoot(). Anything inside of this shadow root, while visible to the naked eye, will be hidden from DOM APIs and CSS selectors in the containing page, maintaining the illusion that this widget will be only a single element. If we are writing a custom calendar widget, the shadow root will be where our complex calendar markup would go, allowing us to expose a single tag as a simple interface to its hidden complexity.
Accessing the “light DOM”
So far, our custom element will be just an empty tag, but what happens if elements will be nested inside our new component? We may want a widget with similar flexibility to the <select> tag, which can contain many <option> tags. As a working example, let’s assume the following markup.
<my-element> This is the light DOM. <i>hello</i> <i>world</i> </my-element>
As soon as a new shadow root will be created against this custom element, its child nodes will no longer visible. We will refer to these hidden child nodes as the “light DOM”. If we will inspect the page or traverse the DOM we will be able to see these hidden nodes, but the end-user would have no clue these elements exist at all. Without shadow DOM, this example will simply appear as ‘This is the light DOM. hello world’.
When we will set up the shadow DOM inside the createdCallback function, we can use the new content tag to distribute elements from the light DOM into the shadow DOM:
createdCallback: { value: function() { var shadow = this.createShadowRoot(); // The child nodes, including 'i' tags, have now disappeared shadow.innerHTML = 'The "i" tags from the light DOM are: ' + '<content select="i" />'; // Now, only the 'i' tags are visible inside the shadow DOM } }
With shadow DOM and the <content> tag, this will now appear as ‘The “i” tags from the light DOM are: helloworld’. Note that the <i> tags have rendered next to each other with no whitespace.
Encapsulating styles
What’s important to understand about Shadow DOM is that we’ll create a clean separation between our widget’s markup and the outside world, will be known as the shadow boundary. One powerful feature of Shadow DOM will that styles declared inside do not leak outside of the shadow boundary.
createdCallback: { value: function() { var shadow = this.createShadowRoot(); shadow.innerHTML = "<style>span { color: green }</style>" + "<span>I'm green</span>"; } }
Even though a very generic span style will be defined in the shadow DOM, it will have no effect on <span> tags in the containing page:
<my-element /> <span>I'm not green</span>
If we will be distributing elements from the light DOM into the shadow DOM, like in our <i> example earlier, it will be important to understand that these nodes will not technically belong to our widget. Distributed nodes will still belong to the containing page, meaning that we can’t style these elements by simply writing a standard selector. Instead it, we must style these distributed elements within the ::content pseudo-element:
::content i { color: blue; }
Which, in the context of a component, will look something like this:
createdCallback: { value: function() { var shadow = this.createShadowRoot(); shadow.innerHTML = '<style>::content i { color: blue; }</style>' + 'The "i" tags from the light DOM are: ' + '<content select="i" />'; } }
Exposing style hooks
When we will hide the internal markup of our custom element, it will still sometimes desirable to allow certain aspects of our element to be re-styled from outside. For example, if we will be writing a custom calendar widget, we might want to allow end-users to style the buttons, without giving them access to the entirety of our widget’s markup. This is where the part attribute and pseudo-element will come in:
createdCallback: { value: function() { var shadow = this.createShadowRoot(); shadow.innerHTML = 'Hello <em part="world">World</em>'; } }
The ::part() pseudo-element will allow us to style any element with a part attribute:
my-element::part(world) { color: green; }
This part contract will be essential in maintaining encapsulation. In the previous example, we had styled the word “World”, but users of our widget will have no idea that it will actually an em tag under the hood. One important benefit of this system will that we will be free to dramatically change the markup inside our widget between versions, so long as the “part” attributes will be still in place.
Just the beginning
Web Components finally will give us a way to achieve simple, consistent, reusable, encapsulated and composable widgets, but we will only just getting started. It will be a great time to start experimenting. Before you can begin, you will need to make sure your browser has the relevant features enabled. If you will use Chrome, head to chrome://flags and enable “experimental Web Platform features”. To target browsers that won't have these features enabled, you can use Google’s Polymer, or Mozilla’s X-Tag.
Time to experiment
All of the functionality presented in this article will be simply an exercise in emulating standard browser behaviour. We’ve been working with the browser’s native widgets for a long time, so taking the step towards writing our own will not be as difficult as it might seem. If you haven’t created a component before, I will urge you to open up the console and experiment. Now, you should try making a custom element, then try to create a shadow root (against any element, not just Custom Elements).
If you will do this then it will naturally raise questions about topics not fully discussed in this article. You will have a qus, Do we have to use strings to define the markup in our Shadow DOM? My ans, No, this is where HTML Templates come in. Next, you can have, Can we bundle an HTML template with our component’s JavaScript? Yes, with HTML Imports.
Even if it’s too early to use this stuff in production, it will never too early to be prepared for the future of the web.
Javascript Developer | Problem Solver
4yNice one bro!
Research Associate @ Robert Koch Institute, Berlin, Germany | Doctoral Candidate , Teaching Assistant (TA) @ Freie Universität Berlin | Member @ APHA, USA | National Winner @ EXL EQ-22
4yWell done man.
Technical Project Manager | Social Impact Tech Solutions | Aspiring Product Manager | BIT Mesra '21
4yGoing Great Mate! ✨