aaaidan aaaidan - 2 months ago 26
HTML Question

How to deal with browser differences with indeterminate checkbox

Primer: An HTML checkbox can be set as

, which displays it as neither checked nor unchecked. Even in this indeterminate state, there is still an underlying boolean

Mac OS X checkboxes in various states

When an indeterminate checkbox is clicked, it loses its
state. Depending on the browser (Firefox), it can additionally toggle the

This jsfiddle illustrates the situation. In Firefox, clicking either of the checkboxes once causes them to toggle their initial underlying
state. In IE, the
property is left alone for the first click.

I would like all browsers to behave the same, even if this means additional javascript. Unfortunately, the
property is set to false before the
handler (or
and jquery
) is called, so I can't detect whether it's called for a click on an
checkbox or not.

(for spacebar toggle) events show the prior
state, but I'd rather not be that specific: it seems fragile.

I could maintain a separate property on the checkbox (
or similar), but I wanted to know if there's a simple solution I'm missing, and/or if other people are having similar issues.


If you would like to have an inderterminate checkbox which becomes checked on all browsers on click (or at least on IE, Chrome and FF5+), you need to initialise the checked attribute correctly, as shown here I have written the following functions to help you:

/// Gives a checkbox the inderminate state and the right
/// checked state so that it becomes checked on click
/// on click on IE, Chrome and Firefox 5+
function makeIndeterminate(checkbox)
    checkbox.checked = getCheckedStateForIndeterminate();
    checkbox.indeterminate = true;

and the interesting function which relies on feature detection:

/// Determine the checked state to give to a checkbox
/// with indeterminate state, so that it becomes checked
/// on click on IE, Chrome and Firefox 5+
function getCheckedStateForIndeterminate()
    // Create a unchecked checkbox with indeterminate state
    var test = document.createElement("input");
    test.type = "checkbox";
    test.checked = false;
    test.indeterminate = true;

    // Try to click the checkbox
    var body = document.body;
    body.appendChild(test); // Required to work on FF;
    body.removeChild(test); // Required to work on FF

    // Check if the checkbox is now checked and cache the result
    if (test.checked)
        getCheckedStateForIndeterminate = function () { return false; };
        return false;
        getCheckedStateForIndeterminate = function () { return true; };
        return true;

No image tricks, no jQuery, no extra attributes and no event handling involved. This relies only on simple JavaScript initialisation (note that the "indeterminate" attribute cannot be set in the HTML markup, so JavaScript initialisation would have been required anyway).