| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360 | 
							- /* Flot plugin for selecting regions of a plot.
 
- Copyright (c) 2007-2014 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 be
 
- specified. "color" is color of the selection (if you need to change the color
 
- later 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 can
 
- be customized to determine the smallest size a selection can be and still
 
- have the selection rectangle be displayed. When customizing this value, the
 
- fact 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 1
 
- minute, setting "minSize" to 1 will not make the minimum selection size 1
 
- minute, but rather 1 pixel. Note also that setting "minSize" to 0 will prevent
 
- "plotunselected" events from being fired when the user clicks the mouse without
 
- dragging.
 
- When selection support is enabled, a "plotselected" event will be emitted on
 
- the DOM element you passed into the plot function. The event handler gets a
 
- parameter 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 the
 
- selection. A "plotselecting" event is fired during the process with the same
 
- parameters as the "plotselected" event, in case you want to know what's
 
- happening while it's happening,
 
- A "plotunselected" event with no arguments is emitted when the user clicks the
 
- mouse to remove the selection. As stated above, setting "minSize" to 0 will
 
- destroy 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);
 
 
  |