l337 Web Component

 

Let’s make a Web Component that transforms the text inside to l337 5p34k.

Here’s a working example. Type to translate to l337 5p34k:

The quick red fox jumps over the lazy brown dog.

Starting with the implementation, this is what we want the markup to look like:

<l337-5p34k>The quick red fox jumps over the lazy brown dog.</l337-5p34k>

To define a Custom Element, we must use window.customElements, specifically window.customElements.define:

class L3375P34K {
    // will define below
}

customElements.define('l337-5p34k', L3375P34K);

One thing to note is that the name for a new Custom Element must contain a hyphen. This forces us to namespace our new component.

Now we create the class that defines the Custom Element:

const L337MAPPING = {
    a: 4,
    e: 3,
    g: 6,
    o: 0,
    s: 5,
    t: 7
};

class L3375P34K extends HTMLElement {
    constructor() {
        super();

        this.shadowTextNode = document.createTextNode('');
        this.textNode = null;
        this.observer = new MutationObserver((e) => {
            const text = e[0].addedNodes[0].textContent;

            this._onTranslate(text);
        });

        this.observer.observe(this, {
            childList: true
        });

        const shadow = this.attachShadow({
            mode: 'open'
        });

        shadow.appendChild(this.shadowTextNode);

        this._onLoad = this._onLoad.bind(this);
    }

    connectedCallback() {
        document.addEventListener('readystatechange', this._onLoad, true);
        this._onLoad();
    }

    adoptedCallback() {
        this._onLoad();
    }

    disconnectedCallback() {
        document.removeEventListener('readystatechange', this._onLoad, true);

        this.observer.disconnect();
    }

    _onLoad() {
        if (document.readyState !== 'complete') {
            return;
        }

        this.textNode = Array.prototype.slice.call(this.childNodes)
            .filter((node) => {
                return node.nodeType === 3;
            })
            .shift();

        if (!this.textNode) {
            this.textNode = document.createTextNode('');

            this.appendChild(this.textNode);
        }

        this._onTranslate()
    }

    _onTranslate(_text) {
        const text = (_text || (this.textNode || {}).textContent || '')
            .trim()
            .toLowerCase();
        const translated = text
            .split('')
            .map((chr) => {
                return L337MAPPING[chr] !== undefined ? L337MAPPING[chr] : chr;
            })
            .join('');

        this.shadowTextNode.textContent = translated;
    }
}

if (window.customElements) {
    customElements.define('l337-5p34k', L3375P34K);
}

Let’s break that down:

const L337MAPPING = {
    a: 4,
    e: 3,
    g: 6,
    o: 0,
    s: 5,
    t: 7
};

This is our very simple Latin -> l337 mapping. a becomes 4, o becomes 0, etc.

class L3375P34K extends HTMLElement {
    constructor() {
        super();

        this.shadowTextNode = document.createTextNode('');
        this.textNode = null;
        this.observer = new MutationObserver((e) => {
            const text = e[0].addedNodes[0].textContent;

            this._onTranslate(text);
        });

        this.observer.observe(this, {
            childList: true
        });

        const shadow = this.attachShadow({
            mode: 'open'
        });

        shadow.appendChild(this.shadowTextNode);

        this._onLoad = this._onLoad.bind(this);
    }

    // ...
}

Here’s our class and constructor. Note that our class extends HTMLElement. We create a textNode that will be attached to the Shadow DOM of our Custom Element. After that, we just bind a few callbacks that will be used later.

We also apply a listener to our Custom Element, waiting for a childList event. This is the event we will hook into to re-translate the text if the text is changed, since otherwise we are really only waiting for an element to be attached to the DOM. After the element loads (and/or after the text is changed), we trigger the _onTranslate() function.

// ...

    connectedCallback() {
        document.addEventListener('readystatechange', this._onLoad, true);
        this._onLoad();
    }

    adoptedCallback() {
        this._onLoad();
    }

    disconnectedCallback() {
        document.removeEventListener('readystatechange', this._onLoad, true);

        this.observer.disconnect();
    }

// ...

Here we have our Custom Element lifecycle methods.

  • connectedCallback() will be called when the element is connected to the DOM
    • We wait for the document to be loaded. If it’s already loaded, we move forward. This helps us instantiate the Custom Element if the page loads with it as opposed to adding it later dynamically.
  • adoptedCallback() will be called when the element is moved to a new document
    • We trigger another load because something may have changed while the element was moving
  • disconnectedCallback() will be called when the element is removed/disconnected from the DOM
    • We clean up some callbacks
// ...

    _onLoad() {
        if (document.readyState !== 'complete') {
            return;
        }

        this.textNode = Array.prototype.slice.call(this.childNodes)
            .filter((node) => {
                return node.nodeType === 3;
            })
            .shift();

        if (!this.textNode) {
            this.textNode = document.createTextNode('');

            this.appendChild(this.textNode);
        }

        this._onTranslate()
    }

// ...

Here we try to load the Custom Element. If the document is not ready, we do not load.

Next we grab the first textNode inside the <l337-5p34k> element, if one does not exist, we create one and append it to the element.

// ...

    _onTranslate(_text) {
        const text = (_text || (this.textNode || {}).textContent || '')
            .trim()
            .toLowerCase();
        const translated = text
            .split('')
            .map((chr) => {
                return L337MAPPING[chr] !== undefined ? L337MAPPING[chr] : chr;
            })
            .join('');

        this.shadowTextNode.textContent = translated;
    }

// ...

The translate function is probably the simpliest part of the entire Custom Element. It takes the textContent from the elements textNode, **map()**s over all the characters, replaces the appropriate ones with the L337MAPPING defined earlier, and assigns the translated text to the Shadow DOM textNode.

Last thing to do is the actual definition:

if (window.customElements) {
    customElements.define('l337-5p34k', L3375P34K);
}

If Custom Elements are not supported, then we won’t try to define ours. Again, our Custom Element’s name must be hyphenated, but it cannot start with a number, stopping us from naming it “1337” (which would’ve been much more legit).

And that’s it!

Nice things that could be in the future:

  • We could add custom styling to our Custom Element
  • We could add CSS or JS functionality to animate characters so it looks like they’re being translated before our eyes