HTML doesn't use boolean values for boolean attributes, surprisingly.
In HTML, boolean attributes are specified by either merely adding the attribute name, or (especially in XHTML) by using the attribute name as its value.
<input type="checkbox" disabled> <!-- HTML -->
<input type="checkbox" disabled /> <!-- Also HTML -->
<input type="checkbox" disabled="disabled" /> <!-- XHTML and HTML -->
This is documented in the HTML specification: http://www.w3.org/TR/html5/infrastructure.html#boolean-attribute
A number of attributes are boolean attributes. The presence of a boolean attribute on an element represents the true value, and the absence of the attribute represents the false value.
The values "true" and "false" are not allowed on boolean attributes. To represent a false value, the attribute has to be omitted altogether.
To add to the confusion, in the DOM these boolean attributes are specified with boolean values, for example:
/** @type HTMLInputElement */
const inputElement = document.createElement('input');
inputElement.disabled = true; // <-- The DOM *does* use a proper boolean value here.
console.log( inputElement.disabled ); // Prints "true"
inputElement.disabled = false;
console.log( inputElement.disabled ); // Prints "false"
...to add even more confusion - due to JavaScript's falsyness - using string values with the property will not work as you'd expect:
inputElement.disabled = 'true';
console.log( inputElement.disabled ); // Prints "true"
inputElement.disabled = 'false';
console.log( inputElement.disabled ); // *Still* prints "true"
(This is because the JavaScript string 'false'
is not type-coerced into the JavaScript boolean value false
).
Also, some HTML attributes do have true
and false
as possible values, such as contentEditable
(which also has inherit
as a third option), also consider <form autocomplete="">
which can be on
and off
(and many other values), which might trip some people up too. I think some legacy (Internet Explorer 4.0-era) extensions like <object>
and <applet>
may have had boolean attributes, and definitely had booleans in their child <param value="">
attributes, that's just an historical curiosity at this point).