| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360 | /* Flot plugin for selecting regions of a plot.Copyright (c) 2007-2013 IOLA and Ole Laursen.Licensed under the MIT license.The plugin supports these options:selection: {	mode: null or "x" or "y" or "xy",	color: color,	shape: "round" or "miter" or "bevel",	minSize: number of pixels}Selection support is enabled by setting the mode to one of "x", "y" or "xy".In "x" mode, the user will only be able to specify the x range, similarly for"y" mode. For "xy", the selection becomes a rectangle where both ranges can bespecified. "color" is color of the selection (if you need to change the colorlater on, you can get to it with plot.getOptions().selection.color). "shape"is the shape of the corners of the selection."minSize" is the minimum size a selection can be in pixels. This value canbe customized to determine the smallest size a selection can be and stillhave the selection rectangle be displayed. When customizing this value, thefact that it refers to pixels, not axis units must be taken into account.Thus, for example, if there is a bar graph in time mode with BarWidth set to 1minute, setting "minSize" to 1 will not make the minimum selection size 1minute, but rather 1 pixel. Note also that setting "minSize" to 0 will prevent"plotunselected" events from being fired when the user clicks the mouse withoutdragging.When selection support is enabled, a "plotselected" event will be emitted onthe DOM element you passed into the plot function. The event handler gets aparameter with the ranges selected on the axes, like this:	placeholder.bind( "plotselected", function( event, ranges ) {		alert("You selected " + ranges.xaxis.from + " to " + ranges.xaxis.to)		// similar for yaxis - with multiple axes, the extra ones are in		// x2axis, x3axis, ...	});The "plotselected" event is only fired when the user has finished making theselection. A "plotselecting" event is fired during the process with the sameparameters as the "plotselected" event, in case you want to know what'shappening while it's happening,A "plotunselected" event with no arguments is emitted when the user clicks themouse to remove the selection. As stated above, setting "minSize" to 0 willdestroy this behavior.The plugin allso adds the following methods to the plot object:- setSelection( ranges, preventEvent )  Set the selection rectangle. The passed in ranges is on the same form as  returned in the "plotselected" event. If the selection mode is "x", you  should put in either an xaxis range, if the mode is "y" you need to put in  an yaxis range and both xaxis and yaxis if the selection mode is "xy", like  this:	setSelection({ xaxis: { from: 0, to: 10 }, yaxis: { from: 40, to: 60 } });  setSelection will trigger the "plotselected" event when called. If you don't  want that to happen, e.g. if you're inside a "plotselected" handler, pass  true as the second parameter. If you are using multiple axes, you can  specify the ranges on any of those, e.g. as x2axis/x3axis/... instead of  xaxis, the plugin picks the first one it sees.- clearSelection( preventEvent )  Clear the selection rectangle. Pass in true to avoid getting a  "plotunselected" event.- getSelection()  Returns the current selection in the same format as the "plotselected"  event. If there's currently no selection, the function returns null.*/(function ($) {    function init(plot) {        var selection = {                first: { x: -1, y: -1}, second: { x: -1, y: -1},                show: false,                active: false            };        // FIXME: The drag handling implemented here should be        // abstracted out, there's some similar code from a library in        // the navigation plugin, this should be massaged a bit to fit        // the Flot cases here better and reused. Doing this would        // make this plugin much slimmer.        var savedhandlers = {};        var mouseUpHandler = null;                function onMouseMove(e) {            if (selection.active) {                updateSelection(e);                                plot.getPlaceholder().trigger("plotselecting", [ getSelection() ]);            }        }        function onMouseDown(e) {            if (e.which != 1)  // only accept left-click                return;                        // cancel out any text selections            document.body.focus();            // prevent text selection and drag in old-school browsers            if (document.onselectstart !== undefined && savedhandlers.onselectstart == null) {                savedhandlers.onselectstart = document.onselectstart;                document.onselectstart = function () { return false; };            }            if (document.ondrag !== undefined && savedhandlers.ondrag == null) {                savedhandlers.ondrag = document.ondrag;                document.ondrag = function () { return false; };            }            setSelectionPos(selection.first, e);            selection.active = true;            // this is a bit silly, but we have to use a closure to be            // able to whack the same handler again            mouseUpHandler = function (e) { onMouseUp(e); };                        $(document).one("mouseup", mouseUpHandler);        }        function onMouseUp(e) {            mouseUpHandler = null;                        // revert drag stuff for old-school browsers            if (document.onselectstart !== undefined)                document.onselectstart = savedhandlers.onselectstart;            if (document.ondrag !== undefined)                document.ondrag = savedhandlers.ondrag;            // no more dragging            selection.active = false;            updateSelection(e);            if (selectionIsSane())                triggerSelectedEvent();            else {                // this counts as a clear                plot.getPlaceholder().trigger("plotunselected", [ ]);                plot.getPlaceholder().trigger("plotselecting", [ null ]);            }            return false;        }        function getSelection() {            if (!selectionIsSane())                return null;                        if (!selection.show) return null;            var r = {}, c1 = selection.first, c2 = selection.second;            $.each(plot.getAxes(), function (name, axis) {                if (axis.used) {                    var p1 = axis.c2p(c1[axis.direction]), p2 = axis.c2p(c2[axis.direction]);                     r[name] = { from: Math.min(p1, p2), to: Math.max(p1, p2) };                }            });            return r;        }        function triggerSelectedEvent() {            var r = getSelection();            plot.getPlaceholder().trigger("plotselected", [ r ]);            // backwards-compat stuff, to be removed in future            if (r.xaxis && r.yaxis)                plot.getPlaceholder().trigger("selected", [ { x1: r.xaxis.from, y1: r.yaxis.from, x2: r.xaxis.to, y2: r.yaxis.to } ]);        }        function clamp(min, value, max) {            return value < min ? min: (value > max ? max: value);        }        function setSelectionPos(pos, e) {            var o = plot.getOptions();            var offset = plot.getPlaceholder().offset();            var plotOffset = plot.getPlotOffset();            pos.x = clamp(0, e.pageX - offset.left - plotOffset.left, plot.width());            pos.y = clamp(0, e.pageY - offset.top - plotOffset.top, plot.height());            if (o.selection.mode == "y")                pos.x = pos == selection.first ? 0 : plot.width();            if (o.selection.mode == "x")                pos.y = pos == selection.first ? 0 : plot.height();        }        function updateSelection(pos) {            if (pos.pageX == null)                return;            setSelectionPos(selection.second, pos);            if (selectionIsSane()) {                selection.show = true;                plot.triggerRedrawOverlay();            }            else                clearSelection(true);        }        function clearSelection(preventEvent) {            if (selection.show) {                selection.show = false;                plot.triggerRedrawOverlay();                if (!preventEvent)                    plot.getPlaceholder().trigger("plotunselected", [ ]);            }        }        // function taken from markings support in Flot        function extractRange(ranges, coord) {            var axis, from, to, key, axes = plot.getAxes();            for (var k in axes) {                axis = axes[k];                if (axis.direction == coord) {                    key = coord + axis.n + "axis";                    if (!ranges[key] && axis.n == 1)                        key = coord + "axis"; // support x1axis as xaxis                    if (ranges[key]) {                        from = ranges[key].from;                        to = ranges[key].to;                        break;                    }                }            }            // backwards-compat stuff - to be removed in future            if (!ranges[key]) {                axis = coord == "x" ? plot.getXAxes()[0] : plot.getYAxes()[0];                from = ranges[coord + "1"];                to = ranges[coord + "2"];            }            // auto-reverse as an added bonus            if (from != null && to != null && from > to) {                var tmp = from;                from = to;                to = tmp;            }                        return { from: from, to: to, axis: axis };        }                function setSelection(ranges, preventEvent) {            var axis, range, o = plot.getOptions();            if (o.selection.mode == "y") {                selection.first.x = 0;                selection.second.x = plot.width();            }            else {                range = extractRange(ranges, "x");                selection.first.x = range.axis.p2c(range.from);                selection.second.x = range.axis.p2c(range.to);            }            if (o.selection.mode == "x") {                selection.first.y = 0;                selection.second.y = plot.height();            }            else {                range = extractRange(ranges, "y");                selection.first.y = range.axis.p2c(range.from);                selection.second.y = range.axis.p2c(range.to);            }            selection.show = true;            plot.triggerRedrawOverlay();            if (!preventEvent && selectionIsSane())                triggerSelectedEvent();        }        function selectionIsSane() {            var minSize = plot.getOptions().selection.minSize;            return Math.abs(selection.second.x - selection.first.x) >= minSize &&                Math.abs(selection.second.y - selection.first.y) >= minSize;        }        plot.clearSelection = clearSelection;        plot.setSelection = setSelection;        plot.getSelection = getSelection;        plot.hooks.bindEvents.push(function(plot, eventHolder) {            var o = plot.getOptions();            if (o.selection.mode != null) {                eventHolder.mousemove(onMouseMove);                eventHolder.mousedown(onMouseDown);            }        });        plot.hooks.drawOverlay.push(function (plot, ctx) {            // draw selection            if (selection.show && selectionIsSane()) {                var plotOffset = plot.getPlotOffset();                var o = plot.getOptions();                ctx.save();                ctx.translate(plotOffset.left, plotOffset.top);                var c = $.color.parse(o.selection.color);                ctx.strokeStyle = c.scale('a', 0.8).toString();                ctx.lineWidth = 1;                ctx.lineJoin = o.selection.shape;                ctx.fillStyle = c.scale('a', 0.4).toString();                var x = Math.min(selection.first.x, selection.second.x) + 0.5,                    y = Math.min(selection.first.y, selection.second.y) + 0.5,                    w = Math.abs(selection.second.x - selection.first.x) - 1,                    h = Math.abs(selection.second.y - selection.first.y) - 1;                ctx.fillRect(x, y, w, h);                ctx.strokeRect(x, y, w, h);                ctx.restore();            }        });                plot.hooks.shutdown.push(function (plot, eventHolder) {            eventHolder.unbind("mousemove", onMouseMove);            eventHolder.unbind("mousedown", onMouseDown);                        if (mouseUpHandler)                $(document).unbind("mouseup", mouseUpHandler);        });    }    $.plot.plugins.push({        init: init,        options: {            selection: {                mode: null, // one of null, "x", "y" or "xy"                color: "#e8cfac",                shape: "round", // one of "round", "miter", or "bevel"                minSize: 5 // minimum number of pixels            }        },        name: 'selection',        version: '1.1'    });})(jQuery);
 |