Search code examples
javascriptaddeventlistenerselectedindex

Is it possible to listen for selectedIndex change?


I am looking for a way (or a workaround) to subscribe for selectedIndex changing (on <select> element) when the change done by a simple assignment (like myElement.selectedIndex=1).

<select id = "mySelect" onchange="myListener()">
    <option>0</option>
    <option>1</option>
</select>

<script>
    function myListener() {
        console.log('Yes, I hear...'); // doesn't work on selectedIndex assignment
    }
    document.getElementById('mySelect').selectedIndex = 1;
</script>

Seems it's not possible, but maybe there is some workaround.
dispatchEvent is not an option (the listener attachment must be done from outside).


My only solution so far, very similar to @jren510 (but I like his solution more).

function myListener() {
    console.log('Yes, I hear...');
}

const originalPropDescriptor = Object.getOwnPropertyDescriptor(HTMLSelectElement.prototype, 'selectedIndex');

Object.defineProperty(HTMLSelectElement.prototype, 'originalSelectedIndex', originalPropDescriptor);

Object.defineProperty(HTMLSelectElement.prototype, 'selectedIndex', {
    get() {
        return this.originalSelectedIndex;
    },

    set(value) {
        myListener();
        this.originalSelectedIndex = value;
    }
});

I would like to avoid overriding native methods, by so far it's the only way I see.


Solution

  • Maybe not the best answer, but you did mention workarounds..

    You can try overriding the setter for the object for which you are trying to detect the change in.

    let mySelect = document.getElementById("mySelect");
    
    const callback = () => {
        console.log("Yes, I hear...");
    };
    
    const original = Object.getOwnPropertyDescriptor(
        Object.getPrototypeOf(mySelect),
        "selectedIndex"
    );
    Object.defineProperty(mySelect, "selectedIndex", {
        set: function (t) {
            callback();
            return original.set.apply(this, arguments);
        },
        get: function () {
            return original.get.apply(this);
        }
    });
    
    let myButton = document.getElementById("myButton");
    myButton.addEventListener("click", () => {
        mySelect.selectedIndex = 1;
    });
    <select id="mySelect">
        <option>0</option>
        <option>1</option>
    </select>
    
    <button id="myButton"> change selector value to 1 </button>

    It's worth mentioning that the callback does not run when manually changing the select here.. if you want to preserve that behavior, you can add an event listener that calls callback as usual, though there may exist a better solution that captures both cases.