DOM

What is the Document Object Model (DOM) and how does it work?

The Document Object Model (DOM) is a crucial concept in web development, serving as a bridge between HTML and the interactive features that JavaScript can add to websites. Without the DOM, websites would remain static, lacking the dynamic interactivity that users have come to expect. This article explores what the DOM is, how it works, and why it’s foundational for modern web development.

1. Understanding the DOM: Definition and Purpose

At its core, the DOM is a programming interface provided by browsers to represent a webpage’s content, structure, and style. It’s essentially an organized, hierarchical representation of a document, typically structured in HTML or XML. The DOM allows programming languages, notably JavaScript, to manipulate the content, structure, and styling of a webpage dynamically.

When a web page loads, the browser parses the HTML and CSS and creates a live model of the page that can be modified using JavaScript. This model is the DOM. This approach lets developers change elements, attributes, and CSS properties on the fly without reloading the page, leading to smoother, more interactive user experiences.

The DOM is also cross-platform and language-independent. It’s not tied to JavaScript, even though JavaScript is the most commonly used language to manipulate it. Technically, any language that supports the DOM API (Application Programming Interface) can interact with the document.

2. The Structure of the DOM

images?q=tbn:ANd9GcQ gb1JFLO4FoR9g2zUXbT0pUdS7YbnWKPkSw&s

The DOM is often visualized as a tree structure, where each HTML or XML element is represented as a “node” in this tree. Here’s how it generally breaks down:

  • Document Node: This is the root of the DOM tree and represents the entire document. It’s the entry point through which scripts can access and manipulate the document content.
  • Element Nodes: Each HTML element, such as <div>, <p>, <h1>, etc., is an element node in the tree. These nodes are nested within one another based on the HTML structure.
  • Attribute Nodes: Attributes like class, id, and href associated with HTML elements are attribute nodes. They provide additional properties for element nodes.
  • Text Nodes: These nodes represent the actual text within HTML elements. For example, in <p>Hello World</p>, “Hello World” is a text node inside the <p> element node.

The structure of the DOM tree is hierarchical, with each element potentially containing other elements or text, creating a parent-child relationship. Here’s an example of a simple HTML document and how it would be structured in the DOM:

				
					<!DOCTYPE html>
<html>
  <head>
    <title>Sample Page</title>
  </head>
  <body>
    <h1><span class="ez-toc-section" id="Welcome_to_the_DOM"></span>Welcome to the DOM<span class="ez-toc-section-end"></span></h1>
    <p>This is a simple example.</p> <script data-no-optimize="1">!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?module.exports=e():"function"==typeof define&&define.amd?define(e):(t="undefined"!=typeof globalThis?globalThis:t||self).LazyLoad=e()}(this,function(){"use strict";function e(){return(e=Object.assign||function(t){for(var e=1;e<arguments.length;e++){var n,a=arguments[e];for(n in a)Object.prototype.hasOwnProperty.call(a,n)&&(t[n]=a[n])}return t}).apply(this,arguments)}function i(t){return e({},it,t)}function o(t,e){var n,a="LazyLoad::Initialized",i=new t(e);try{n=new CustomEvent(a,{detail:{instance:i}})}catch(t){(n=document.createEvent("CustomEvent")).initCustomEvent(a,!1,!1,{instance:i})}window.dispatchEvent(n)}function l(t,e){return t.getAttribute(gt+e)}function c(t){return l(t,bt)}function s(t,e){return function(t,e,n){e=gt+e;null!==n?t.setAttribute(e,n):t.removeAttribute(e)}(t,bt,e)}function r(t){return s(t,null),0}function u(t){return null===c(t)}function d(t){return c(t)===vt}function f(t,e,n,a){t&&(void 0===a?void 0===n?t(e):t(e,n):t(e,n,a))}function _(t,e){nt?t.classList.add(e):t.className+=(t.className?" ":"")+e}function v(t,e){nt?t.classList.remove(e):t.className=t.className.replace(new RegExp("(^|\\s+)"+e+"(\\s+|$)")," ").replace(/^\s+/,"").replace(/\s+$/,"")}function g(t){return t.llTempImage}function b(t,e){!e||(e=e._observer)&&e.unobserve(t)}function p(t,e){t&&(t.loadingCount+=e)}function h(t,e){t&&(t.toLoadCount=e)}function n(t){for(var e,n=[],a=0;e=t.children[a];a+=1)"SOURCE"===e.tagName&&n.push(e);return n}function m(t,e){(t=t.parentNode)&&"PICTURE"===t.tagName&&n(t).forEach(e)}function a(t,e){n(t).forEach(e)}function E(t){return!!t[st]}function I(t){return t[st]}function y(t){return delete t[st]}function A(e,t){var n;E(e)||(n={},t.forEach(function(t){n[t]=e.getAttribute(t)}),e[st]=n)}function k(a,t){var i;E(a)&&(i=I(a),t.forEach(function(t){var e,n;e=a,(t=i[n=t])?e.setAttribute(n,t):e.removeAttribute(n)}))}function L(t,e,n){_(t,e.class_loading),s(t,ut),n&&(p(n,1),f(e.callback_loading,t,n))}function w(t,e,n){n&&t.setAttribute(e,n)}function x(t,e){w(t,ct,l(t,e.data_sizes)),w(t,rt,l(t,e.data_srcset)),w(t,ot,l(t,e.data_src))}function O(t,e,n){var a=l(t,e.data_bg_multi),i=l(t,e.data_bg_multi_hidpi);(a=at&&i?i:a)&&(t.style.backgroundImage=a,n=n,_(t=t,(e=e).class_applied),s(t,ft),n&&(e.unobserve_completed&&b(t,e),f(e.callback_applied,t,n)))}function N(t,e){!e||0<e.loadingCount||0<e.toLoadCount||f(t.callback_finish,e)}function C(t,e,n){t.addEventListener(e,n),t.llEvLisnrs[e]=n}function M(t){return!!t.llEvLisnrs}function z(t){if(M(t)){var e,n,a=t.llEvLisnrs;for(e in a){var i=a[e];n=e,i=i,t.removeEventListener(n,i)}delete t.llEvLisnrs}}function R(t,e,n){var a;delete t.llTempImage,p(n,-1),(a=n)&&--a.toLoadCount,v(t,e.class_loading),e.unobserve_completed&&b(t,n)}function T(o,r,c){var l=g(o)||o;M(l)||function(t,e,n){M(t)||(t.llEvLisnrs={});var a="VIDEO"===t.tagName?"loadeddata":"load";C(t,a,e),C(t,"error",n)}(l,function(t){var e,n,a,i;n=r,a=c,i=d(e=o),R(e,n,a),_(e,n.class_loaded),s(e,dt),f(n.callback_loaded,e,a),i||N(n,a),z(l)},function(t){var e,n,a,i;n=r,a=c,i=d(e=o),R(e,n,a),_(e,n.class_error),s(e,_t),f(n.callback_error,e,a),i||N(n,a),z(l)})}function G(t,e,n){var a,i,o,r,c;t.llTempImage=document.createElement("IMG"),T(t,e,n),E(c=t)||(c[st]={backgroundImage:c.style.backgroundImage}),o=n,r=l(a=t,(i=e).data_bg),c=l(a,i.data_bg_hidpi),(r=at&&c?c:r)&&(a.style.backgroundImage='url("'.concat(r,'")'),g(a).setAttribute(ot,r),L(a,i,o)),O(t,e,n)}function D(t,e,n){var a;T(t,e,n),a=e,e=n,(t=It[(n=t).tagName])&&(t(n,a),L(n,a,e))}function V(t,e,n){var a;a=t,(-1<yt.indexOf(a.tagName)?D:G)(t,e,n)}function F(t,e,n){var a;t.setAttribute("loading","lazy"),T(t,e,n),a=e,(e=It[(n=t).tagName])&&e(n,a),s(t,vt)}function j(t){t.removeAttribute(ot),t.removeAttribute(rt),t.removeAttribute(ct)}function P(t){m(t,function(t){k(t,Et)}),k(t,Et)}function S(t){var e;(e=At[t.tagName])?e(t):E(e=t)&&(t=I(e),e.style.backgroundImage=t.backgroundImage)}function U(t,e){var n;S(t),n=e,u(e=t)||d(e)||(v(e,n.class_entered),v(e,n.class_exited),v(e,n.class_applied),v(e,n.class_loading),v(e,n.class_loaded),v(e,n.class_error)),r(t),y(t)}function $(t,e,n,a){var i;n.cancel_on_exit&&(c(t)!==ut||"IMG"===t.tagName&&(z(t),m(i=t,function(t){j(t)}),j(i),P(t),v(t,n.class_loading),p(a,-1),r(t),f(n.callback_cancel,t,e,a)))}function q(t,e,n,a){var i,o,r=(o=t,0<=pt.indexOf(c(o)));s(t,"entered"),_(t,n.class_entered),v(t,n.class_exited),i=t,o=a,n.unobserve_entered&&b(i,o),f(n.callback_enter,t,e,a),r||V(t,n,a)}function H(t){return t.use_native&&"loading"in HTMLImageElement.prototype}function B(t,i,o){t.forEach(function(t){return(a=t).isIntersecting||0<a.intersectionRatio?q(t.target,t,i,o):(e=t.target,n=t,a=i,t=o,void(u(e)||(_(e,a.class_exited),$(e,n,a,t),f(a.callback_exit,e,n,t))));var e,n,a})}function J(e,n){var t;et&&!H(e)&&(n._observer=new IntersectionObserver(function(t){B(t,e,n)},{root:(t=e).container===document?null:t.container,rootMargin:t.thresholds||t.threshold+"px"}))}function K(t){return Array.prototype.slice.call(t)}function Q(t){return t.container.querySelectorAll(t.elements_selector)}function W(t){return c(t)===_t}function X(t,e){return e=t||Q(e),K(e).filter(u)}function Y(e,t){var n;(n=Q(e),K(n).filter(W)).forEach(function(t){v(t,e.class_error),r(t)}),t.update()}function t(t,e){var n,a,t=i(t);this._settings=t,this.loadingCount=0,J(t,this),n=t,a=this,Z&&window.addEventListener("online",function(){Y(n,a)}),this.update(e)}var Z="undefined"!=typeof window,tt=Z&&!("onscroll"in window)||"undefined"!=typeof navigator&&/(gle|ing|ro)bot|crawl|spider/i.test(navigator.userAgent),et=Z&&"IntersectionObserver"in window,nt=Z&&"classList"in document.createElement("p"),at=Z&&1<window.devicePixelRatio,it={elements_selector:".lazy",container:tt||Z?document:null,threshold:300,thresholds:null,data_src:"src",data_srcset:"srcset",data_sizes:"sizes",data_bg:"bg",data_bg_hidpi:"bg-hidpi",data_bg_multi:"bg-multi",data_bg_multi_hidpi:"bg-multi-hidpi",data_poster:"poster",class_applied:"applied",class_loading:"litespeed-loading",class_loaded:"litespeed-loaded",class_error:"error",class_entered:"entered",class_exited:"exited",unobserve_completed:!0,unobserve_entered:!1,cancel_on_exit:!0,callback_enter:null,callback_exit:null,callback_applied:null,callback_loading:null,callback_loaded:null,callback_error:null,callback_finish:null,callback_cancel:null,use_native:!1},ot="src",rt="srcset",ct="sizes",lt="poster",st="llOriginalAttrs",ut="loading",dt="loaded",ft="applied",_t="error",vt="native",gt="data-",bt="ll-status",pt=[ut,dt,ft,_t],ht=[ot],mt=[ot,lt],Et=[ot,rt,ct],It={IMG:function(t,e){m(t,function(t){A(t,Et),x(t,e)}),A(t,Et),x(t,e)},IFRAME:function(t,e){A(t,ht),w(t,ot,l(t,e.data_src))},VIDEO:function(t,e){a(t,function(t){A(t,ht),w(t,ot,l(t,e.data_src))}),A(t,mt),w(t,lt,l(t,e.data_poster)),w(t,ot,l(t,e.data_src)),t.load()}},yt=["IMG","IFRAME","VIDEO"],At={IMG:P,IFRAME:function(t){k(t,ht)},VIDEO:function(t){a(t,function(t){k(t,ht)}),k(t,mt),t.load()}},kt=["IMG","IFRAME","VIDEO"];return t.prototype={update:function(t){var e,n,a,i=this._settings,o=X(t,i);{if(h(this,o.length),!tt&&et)return H(i)?(e=i,n=this,o.forEach(function(t){-1!==kt.indexOf(t.tagName)&&F(t,e,n)}),void h(n,0)):(t=this._observer,i=o,t.disconnect(),a=t,void i.forEach(function(t){a.observe(t)}));this.loadAll(o)}},destroy:function(){this._observer&&this._observer.disconnect(),Q(this._settings).forEach(function(t){y(t)}),delete this._observer,delete this._settings,delete this.loadingCount,delete this.toLoadCount},loadAll:function(t){var e=this,n=this._settings;X(t,n).forEach(function(t){b(t,e),V(t,n,e)})},restoreAll:function(){var e=this._settings;Q(e).forEach(function(t){U(t,e)})}},t.load=function(t,e){e=i(e);V(t,e)},t.resetStatus=function(t){r(t)},Z&&function(t,e){if(e)if(e.length)for(var n,a=0;n=e[a];a+=1)o(t,n);else o(t,e)}(t,window.lazyLoadOptions),t});!function(e,t){"use strict";function a(){t.body.classList.add("litespeed_lazyloaded")}function n(){console.log("[LiteSpeed] Start Lazy Load Images"),d=new LazyLoad({elements_selector:"[data-lazyloaded]",callback_finish:a}),o=function(){d.update()},e.MutationObserver&&new MutationObserver(o).observe(t.documentElement,{childList:!0,subtree:!0,attributes:!0})}var d,o;e.addEventListener?e.addEventListener("load",n,!1):e.attachEvent("onload",n)}(window,document);</script></body>
</html>

				
			

This document would be represented in the DOM as a hierarchy, with html as the root node, head and body as its children, and so on. Each node can have children (nested elements) or siblings (elements on the same level).

3. How the DOM Works: Parsing and Rendering

The process begins with the parsing of the HTML and CSS files by the browser to create the DOM and CSSOM (CSS Object Model) trees, respectively. These two trees are then combined to form the Render Tree, which defines the visual display of elements on the screen.

  • Step 1: HTML Parsing – The browser parses the HTML document and builds a tree-like structure (DOM Tree) that represents the document’s structure.
  • Step 2: CSS Parsing – Simultaneously, CSS files are parsed to build the CSSOM. This tree specifies the styles associated with elements.
  • Step 3: Render Tree Construction – The DOM and CSSOM are combined to form the Render Tree, which dictates how elements will be visually presented.
  • Step 4: Layout and Paint – The browser calculates the exact positions and dimensions of each element (layout) and then paints the elements on the screen.

Any JavaScript that modifies the DOM or CSSOM after this process triggers a reflow (layout recalculation) or repaint (visual update). Minimizing these operations is essential for maintaining performance, particularly on complex, dynamic web pages.

4. Manipulating the DOM with JavaScript

One of the primary reasons the DOM is so powerful is because it allows for dynamic interaction with JavaScript. JavaScript can add, remove, modify, or even completely rearrange elements in the DOM. Here are some common methods and properties used to interact with the DOM:

  • Accessing Elements:

    • document.getElementById(): Selects an element by its ID.
    • document.getElementsByClassName(): Selects elements by their class.
    • document.getElementsByTagName(): Selects elements by their tag name.
    • document.querySelector() and document.querySelectorAll(): Select elements using CSS selectors.
  • Creating and Adding Elements:

    • document.createElement(): Creates a new element.
    • parentNode.appendChild(newNode): Appends a new child element to a parent node.
    • parentNode.insertBefore(newNode, referenceNode): Inserts an element before a specified reference node.
  • Removing Elements:

    • parentNode.removeChild(childNode): Removes a specified child from a parent node.
  • Modifying Attributes and Styles:

    • element.setAttribute(): Sets an attribute on an element.
    • element.style: Modifies inline styles directly.
  • Event Handling:

    • The DOM also supports event handling, allowing developers to add interactivity. For instance, element.addEventListener('click', callback) allows you to specify a function to be run when an element is clicked.

Example:

				
					// Changing text in a <p> element with id="demo"
document.getElementById("demo").innerText = "Hello, DOM!";

				
			

This example dynamically changes the text inside a <p> element. Similarly, the DOM API offers methods to make elements disappear, change colors, or trigger animations in response to user actions.

5. The DOM and Performance

Interacting with the DOM can be computationally expensive, especially if the document is large or complex. Here are a few performance tips:

  • Minimize Reflows and Repaints: Adding or removing nodes frequently can cause reflows and repaints, which may slow down page performance. Batch DOM updates or use CSS classes to apply multiple style changes at once.
  • Use Document Fragments: When adding multiple elements to the DOM, use DocumentFragment to add them all at once, which is more efficient.
  • Optimize Event Handling: Avoid adding event listeners to many individual elements; instead, use event delegation to add a single listener to a parent element.

6. The DOM in Modern JavaScript Frameworks

dom

JavaScript frameworks like React, Vue, and Angular interact with the DOM differently by using something called the Virtual DOM. The Virtual DOM is a lightweight copy of the DOM that exists purely in memory. Instead of updating the DOM directly, these frameworks update the Virtual DOM. They then compare it to the actual DOM and only apply changes where necessary. This approach greatly improves performance by reducing the number of direct DOM manipulations.

For example, React’s Virtual DOM allows it to perform “diffing,” a process where it determines the minimal number of updates needed and only changes those parts of the actual DOM. This makes applications faster and more efficient.

7. Conclusion

The Document Object Model (DOM) is a foundational concept in web development. It provides a structured, programmable interface for web content, making it possible for developers to create interactive, dynamic websites. Through the DOM, JavaScript can manipulate web pages in real time, responding to user interactions, changing content, and altering the page’s style.

Understanding how the DOM works, how to manipulate it efficiently, and its implications for performance are essential skills for web developers. As JavaScript frameworks evolve and improve, knowing the basics of the DOM remains crucial, even in the context of advanced technologies like the Virtual DOM. By mastering DOM manipulation, developers unlock the potential to create more interactive, responsive, and engaging user experiences.

Leave a Reply