Hand Holding Puzzle Piece — by Nicole De Khors from Burst

4 Ways to Add HTML Elements with JavaScript

Alexandre Alves
Bits and Pieces
Published in
5 min readMar 22, 2022

Often, websites need to inject dynamic HTML parts using JavaScript to show items such as blog posts, usernames, or pieces of news in groups.

In this article, I’ll list parts of the code of a small game using HTML, CSS, and JavaScript, with four different approaches to adding HTML elements to a page on the client side.

These four approaches are:

  1. document.createElement
  2. Template HTML tag
  3. EJS templating language
  4. EJS + fetch

Let’s play a game

The challenge is to find the magic integer number out between 0 and 100 using attempts as little as possible. For instance, if the user enters 10, the website answers it is “bigger than 10”. User tries 50. Then, the page prints the magic number is less than 50, and so on. User wins when the attempt is equal to the magic number.

The website must show an input field that receives an integer, a list of attempts, and the magic number when finishing the game. It finishes when the user finds the number.
Game’s page

Approaches

In this article, the focus is on the method called addAttemptElement on the script.js file. This method receives the user’s number, verifies if its value matches the magic number, and prints a message on the page.

Complete solutions are on GitHub.

1. document.createElement

It is an HTML document method, which creates the HTML element specified by tag name.

In the index.html file, we have just the attempts container:

<section id="attempts"></section>

In the script.js file:

const userNumberElements = document.querySelector('#attempts');function addAttemptElement(value) {
const attempt = document.createElement('span');
attempt.textContent = value > magicNumber ? `Less than ${value}` : `Bigger than ${value}`;
attempt.classList.add("attempt");
userNumberElements.appendChild(attempt);
}

With createElement, we need to create each element as well as indicate its properties using JavaScript, for instance the class attribute. It can turn the code extensive and complex but creating a new element like that is more performative than recreating its container via innerHTML.

2. Template HTML tag

The Content Template element is a mechanism for holding an HTML excerpt without rendering it immediately when a page is loaded.

In index.html:

<section id="attempts">
<template>
<span class="attempt"></span>
</template>
</section>

In script.js:

const userNumberElements = document.querySelector('#attempts');
const userNumberElementTemplate = document.querySelector('template');
function addAttemptElement(value) {
const elementClone = userNumberElementTemplate.content.cloneNode(true)
const attempt = elementClone.querySelector('span');
attempt.textContent = value > magicNumber ? `Less than ${value}` : `Bigger than ${value}`;
userNumberElements.appendChild(attempt);
}

Using template tag, all HTML code is on index file. Furthermore, the task includes cloning the template and selecting elements to attribute value via JavaScript. Although, here the element attempt can be added to DOM as HTML element too like createElement.

3. EJS templating language

EJS is a simple templating language, which generates HTML markup with JavaScript.

Templating languages can improve your developer experience adding legibility and reusability to your code. The most common use of this kind of template language is to add libraries to inject HTML on the server side, but you can use it on the client side as well.

Other common languages like that are: Handlebars, Pug, and Mustache.

In index.html:

<section id="attempts"></section>
...
<script src="ejs.min.js"></script>
<script id="ejs-template" type="text/template">
<span class="attempt">
<%=value > magicNumber ? `Less than ${value}` : `Bigger than ${value}`%>
</span>
</script>

In script.js:

const userNumberElements = document.querySelector('#attempts');
const userNumberElementTemplate = document.querySelector('#ejs-template').innerHTML;
function addAttemptElement(value) {
const attempt = ejs.compile(userNumberElementTemplate)({ magicNumber, value });
userNumberElements.innerHTML += attempt;
}

Despite of having less code in the addAttemptElement function, it’s necessary linking an external library ejs.min.js and adding new elements via string concatenation with this approach. A performance issue can be found here when recreating the whole container with innerHTML.

Note: the logic for comparing the user’s number and the magic number is in the EJS code, on the index.html.

4. EJS + fetch

Using the global fetch() method of Fetch API, resources can be loaded from several places. Based on the previous approach, EJS templates are placed on specific files.

In index.html:

<section id="attempts"></section>
...
<script src="ejs.min.js"></script>

In ejs-template.html:

<span class="attempt">
<%=value > magicNumber ? `Less than ${value}` : `Bigger than ${value}`%>
</span>

In script.js:

let userNumberElementTemplate = '';
async function getTemplate(url) {
if (userNumberElementTemplate) return userNumberElementTemplate
let res = await fetch(url);
return res.text();
}
function addAttemptElement(value) {
getTemplate('./ejs-template.html')
.then(userNumberElementTemplate=>{
const attempt = ejs.compile(userNumberElementTemplate)({ magicNumber, value });
userNumberElements.innerHTML += attempt;
})
}

This approach has lots of changes. The variable userNumberElementTemplate has a singleton concept, getTemplate fetches template files, and finally, addAttemptElement injects the HTML code on the page.

Conclusion

The game is simple; however, it can be created with these 4 ways to inject dynamic HTML elements on the page. Different challenges need different approaches depending on performance, scalability, and other requirements. Each one of them has pros and cons as seen above.

I hope you have found this article useful. If so, be sure to like and share.

If you have suggestions when inserting code into an HTML file, feel free to comment below.

Build Apps with reusable components, just like Lego

Bit’s open-source tool help 250,000+ devs to build apps with components.

Turn any UI, feature, or page into a reusable component — and share it across your applications. It’s easier to collaborate and build faster.

Learn more

Split apps into components to make app development easier, and enjoy the best experience for the workflows you want:

Micro-Frontends

Design System

Code-Sharing and reuse

Monorepo

Sign up to discover human stories that deepen your understanding of the world.

Free

Distraction-free reading. No ads.

Organize your knowledge with lists and highlights.

Tell your story. Find your audience.

Membership

Read member-only stories

Support writers you read most

Earn money for your writing

Listen to audio narrations

Read offline with the Medium app

Published in Bits and Pieces

Insightful articles, step-by-step tutorials, and the latest news on full-stack composable software development

Written by Alexandre Alves

Full-Stack Developer in Canada, NASA Datanauts Member, NASA Space Apps Ambassador, and challenge lover

No responses yet

What are your thoughts?