Properties Vs Attributes in Javascript

We've all used properties and attributes in our code. But for a long time, I really didn’t understand the difference. In short, attributes set initial values, while properties reflect an element’s dynamic state after interaction or manipulation. Simple enough, but how does that relate to data-attributes? What about custom elements? Let’s take a deep dive and figure out the key differences.

Attributes vs. Properties

Attributes are Static, Properties are Dynamic

Attributes are defined in the HTML and serve as the initial state. However, once the page loads, JavaScript properties take over and can change independently.

<input id="myInput" type="text" value="Hello" />
<script>
  const input = document.getElementById("myInput");

  console.log(input.getAttribute("value")); // "Hello" (Attribute)
  console.log(input.value); // "Hello" (Property)

  input.value = "Goodbye";
  console.log(input.getAttribute("value")); // Still "Hello" (Attribute)
  console.log(input.value); // "Goodbye" (Property)
</script>

Changing input.value updates the displayed text but does not update the value attribute in the DOM. However, if you manually reset the form or use setAttribute(), the attribute will be restored.

Custom Elements & Observed Attributes

Built-in elements, like <input>, do not automatically sync properties with attributes. However, custom elements can listen for attribute changes using observedAttributes().

<script>
  class MyElement extends HTMLElement {
    static get observedAttributes() {
      return ["title"];
    }

    attributeChangedCallback(name, oldValue, newValue) {
      console.log(`Attribute ${name} changed from ${oldValue} to ${newValue}`);
    }
  }

  customElements.define("my-element", MyElement);
</script>

<my-element title="Hello"></my-element>

<script>
  const el = document.querySelector("my-element");
  el.setAttribute("title", "Updated Title");
  // Logs: "Attribute title changed from Hello to Updated Title"
</script>

This behavior is opt-in for custom elements, but not available for built-in ones. For example, changing input.value = "New" will not trigger an attribute change.

Attributes Live in the DOM, Properties Live in Memory

Attributes are stored in the DOM itself (setAttribute() modifies them). Properties are stored in JavaScript memory and may or may not sync with attributes.

<img id="logo" src="logo.png" />
<script>
  const img = document.getElementById("logo");
  console.log(img.getAttribute("src")); // "logo.png"
  console.log(img.src); // Full URL (e.g., "http://example.com/logo.png")

  img.src = "new-logo.png";
  console.log(img.getAttribute("src")); // Still "logo.png"
</script>

This happens because properties often represent computed values (like img.src expanding to an absolute URL).

Keeping Attributes & Properties in Sync

If you want to keep attributes and properties in sync, you need to explicitly update them.

<input id="syncInput" value="Initial" />
<script>
  const input = document.getElementById("syncInput");

  // Keep value attribute in sync with property
  input.addEventListener("input", () => {
    input.setAttribute("value", input.value);
  });
</script>

This ensures that when a user types, both the property (input.value) and attribute (value="...") remain updated.

How do Data Attributes Work?

Native elements provide specific attributes by default, such as the src attribute on an img or the type attribute on an input. However, you can also define your own custom attributes using the data- prefix, which is commonly referred to as "data attributes." These custom data attributes are accessible in both your CSS and JavaScript, providing a flexible way to store extra information.

Unlike the standard attributes like src or value, data attributes allow you to store custom data directly on HTML elements without affecting the rendering of the page. They’re especially useful for adding extra metadata or managing state within your app. Let’s see how they work in practice:

Example: Setting a Custom Data Attribute for Theming

Imagine you want to store the theme preference in your HTML element. Here's how you'd do it with a data attribute:

<body data-theme="dark">
  <!-- Content -->
</body>

<script>
  // Get Data
  const theme = document.body.dataset.theme; // Note the use of dataset
  console.log(theme); // "dark"

  // Set Data
  document.body.dataset.theme = "light";
  console.log(document.body.dataset.theme); // "light"
</script>

In the example above, we added a data-theme attribute to the <body> tag. This attribute can be accessed in JavaScript via dataset.theme. Notice that when you access or set data attributes in JavaScript, you use the dataset object, which automatically converts data-* attributes into camelCase.

While data attributes are easy to use and versatile, it’s important to note that they should primarily be used for storing extra data that isn't critical to your page's layout or style. They help keep your HTML clean and provide a way to pass information around without cluttering the DOM or your CSS. I'll go into a deeper discussion on data attributes in a future article.

TL;DR

  • Attributes (set in HTML) provide the initial values.
  • Properties (set in JavaScript) represent the current state.
  • Changing a property does not update the attribute (unless manually synced).
  • Custom elements can observe attribute changes with observedAttributes().
  • Attributes live in the DOM, while properties live in JavaScript memory.
  • Data attributes (data-*) allow you to store custom data in HTML elements that are accessible in both CSS and JavaScript.

Understanding this distinction is key when working with forms, custom elements, or even React props vs. attributes. So next time you're debugging, be sure to check both—is it an attribute or a property?