16

Basically I need to know when the window.getSelection() has changed and bind a handler to this event. Ideas?

OBS: Please note that I'm not looking to bind a selection change on a INPUT or TEXTAREA. I'm talking about any selection in the window.

Brian Tompsett - 汤莱恩
  • 5,753
  • 72
  • 57
  • 129
Karl Mendes
  • 649
  • 4
  • 9
  • possible duplicate of [How to monitor window selection change event in javascript](http://stackoverflow.com/questions/5139623/how-to-monitor-window-selection-change-event-in-javascript) – Felix Kling Jan 24 '12 at 18:01
  • 2
    not a duplicate because I need to monitor the window selection event, not for a input – Karl Mendes Jan 24 '12 at 21:18
  • Actually I think it is a duplicate, as the other question does seem to be asking the same thing rather imprecisely, but the answers there are not useful. A better duplicate is this one: http://stackoverflow.com/questions/8442158/selection-change-event-in-contenteditable – Tim Down Jan 27 '12 at 14:56

3 Answers3

19

2019 update

All major browsers now support the selectionchange event, which does the job. Firefox was the last browser to get it, and it has had it without a configuration flag since version 52 (released in March 2017).

Original answer

There is no cross-browser way of detecting changes to the selection. IE (since version 5.5, I think) and WebKit/Blink browsers (Chrome, Safari and Opera version from the last couple of years, for example) support a selectionchange event on the document which does exactly what you want.

Firefox and pre-Blink Opera have no such event and all you can do is detect selection changes made via keyboard and mouse events, which is unsatisfactory (there is no way of detecting "Select All" from context or edit menus, for example), or simply poll the Selection object returned by window.getSelection() (checking the selection's anchorNode, anchorOffset, focusNode and focusOffset properties against their previous values should be sufficient).

Tim Down
  • 318,141
  • 75
  • 454
  • 536
  • 1
    Update: Opera now supports selectionchange (since the switch to Blink). For work, I recently wrote a lightweight selectionchange polyfill for Firefox. https://github.com/2is10/selectionchange-polyfill The Firefox feature request, for tracking purposes: https://bugzilla.mozilla.org/show_bug.cgi?id=571294 – jaredjacobs Jul 09 '14 at 20:36
  • @jaredjacobs: Thanks. I've incorporated that into my answer. – Tim Down Jul 10 '14 at 22:54
  • 1
    It got into FF43, but you still need to change a flag. Should be available soon then. https://developer.mozilla.org/en-US/docs/Web/Events/selectionchange#Browser_compatibility – neo post modern Jan 01 '16 at 16:14
  • 1
    In FF52, it's no longer behind a flag. https://developer.mozilla.org/en-US/docs/Web/API/Document/selectionchange_event FF52 was released 2 years ago, we should be good. – Josh Powlison Jun 15 '19 at 18:18
  • 1
    @JoshPowlison: That's good news. I've incorporated that into my answer. Thanks very much. – Tim Down Jun 17 '19 at 11:07
9

There is no cross-browser event for that.

However, there does exist an event called selectionchange, which trigger on every change in a selection in the document, but it is only supported in IE and recent WebKit (Chrome/Safari), so no Firefox/Opera.

You can use the selectionchange event like this:

$(document).on('selectionchange', function(e) {
    console.log('selectionchange', e.originalEvent); 
});

jsfiddle example

Sindre Sorhus
  • 62,972
  • 39
  • 168
  • 232
  • ^ This. However, it is extremely greedy and may fire many times a second, similar to `window.onresize`. – Kafoso Dec 02 '16 at 09:19
0

I'd argue that the selectionchange event is too greedy, and you want to watch the mouse events. The handler is the easy part. Returning the selected nodes is the important part! Adapting Tim Down's answer here this is what I came up with:

    $(document).on('mouseup', function(e) { 
    
            //cool! jQuery object containing all nodes (elements) that were just selected
            var $selNodes = $().getSelectedNodes();
            if ($selNodes.length)
                     alert('user selected something');
            
            //so now I can do work on just the elements I am interested in that were selected
            if($selNodes.filter('.justTheClassIwant').length)
                alert('user selected the elements I want to watch');
        });
    
    jQuery.fn.extend({
    getSelectedNodes: function() {
        if (window.getSelection) {
            var sel = window.getSelection();
            if (!sel.isCollapsed) {
                var range = sel.getRangeAt(0);
                var node = range.startContainer;
                var endNode = range.endContainer;

                // Special case for a range that is contained within a single node
                if (node == endNode) {
                    return [node];
                }

                // Iterate nodes until we hit the end container
                var rangeNodes = [];
                while (node && node != endNode) {
                    if (node.hasChildNodes()) {
                        node = node.firstChild;
                    } 
                    else {
                        while (node && !node.nextSibling) {
                            node = node.parentNode;
                        }
                        node = node.nextSibling;
                    }
    
                    rangeNodes.push(node);
                }

                // Add partially selected nodes at the start of the range
                node = range.startContainer;
                while (node && node != range.commonAncestorContainer) {
                    rangeNodes.unshift(node);
                    node = node.parentNode;
                }

                return jQuery(rangeNodes);
            }
        }
        return jQuery([]);
    }
});
mike
  • 2,149
  • 20
  • 29