4 <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
5 <title>The source code</title>
6 <link href="../prettify/prettify.css" type="text/css" rel="stylesheet" />
7 <script type="text/javascript" src="../prettify/prettify.js"></script>
8 <style type="text/css">
9 .highlight { display: block; background-color: #ddd; }
11 <script type="text/javascript">
12 function highlight() {
13 document.getElementById(location.hash.replace(/#/, "")).className = "highlight";
17 <body onload="prettyPrint(); highlight();">
18 <pre class="prettyprint lang-js"><span id='Ext-dd-DragTracker'>/**
19 </span> * @class Ext.dd.DragTracker
20 * A DragTracker listens for drag events on an Element and fires events at the start and end of the drag,
21 * as well as during the drag. This is useful for components such as {@link Ext.slider.Multi}, where there is
22 * an element that can be dragged around to change the Slider's value.
23 * DragTracker provides a series of template methods that should be overridden to provide functionality
24 * in response to detected drag operations. These are onBeforeStart, onStart, onDrag and onEnd.
25 * See {@link Ext.slider.Multi}'s initEvents function for an example implementation.
27 Ext.define('Ext.dd.DragTracker', {
29 uses: ['Ext.util.Region'],
32 observable: 'Ext.util.Observable'
35 <span id='Ext-dd-DragTracker-property-active'> /**
36 </span> * @property active
38 * Read-only property indicated whether the user is currently dragging this
43 <span id='Ext-dd-DragTracker-property-dragTarget'> /**
44 </span> * @property dragTarget
46 * <p><b>Only valid during drag operations. Read-only.</b></p>
47 * <p>The element being dragged.</p>
48 * <p>If the {@link #delegate} option is used, this will be the delegate element which was mousedowned.</p>
51 <span id='Ext-dd-DragTracker-cfg-trackOver'> /**
52 </span> * @cfg {Boolean} trackOver
53 * <p>Defaults to <code>false</code>. Set to true to fire mouseover and mouseout events when the mouse enters or leaves the target element.</p>
54 * <p>This is implicitly set when an {@link #overCls} is specified.</p>
55 * <b>If the {@link #delegate} option is used, these events fire only when a delegate element is entered of left.</b>.
59 <span id='Ext-dd-DragTracker-cfg-overCls'> /**
60 </span> * @cfg {String} overCls
61 * <p>A CSS class to add to the DragTracker's target element when the element (or, if the {@link #delegate} option is used,
62 * when a delegate element) is mouseovered.</p>
63 * <b>If the {@link #delegate} option is used, these events fire only when a delegate element is entered of left.</b>.
66 <span id='Ext-dd-DragTracker-cfg-constrainTo'> /**
67 </span> * @cfg {Ext.util.Region/Element} constrainTo
68 * <p>A {@link Ext.util.Region Region} (Or an element from which a Region measurement will be read) which is used to constrain
69 * the result of the {@link #getOffset} call.</p>
70 * <p>This may be set any time during the DragTracker's lifecycle to set a dynamic constraining region.</p>
73 <span id='Ext-dd-DragTracker-cfg-tolerance'> /**
74 </span> * @cfg {Number} tolerance
75 * Number of pixels the drag target must be moved before dragging is
76 * considered to have started. Defaults to <code>5</code>.
80 <span id='Ext-dd-DragTracker-cfg-autoStart'> /**
81 </span> * @cfg {Boolean/Number} autoStart
82 * Defaults to <code>false</code>. Specify <code>true</code> to defer trigger start by 1000 ms.
83 * Specify a Number for the number of milliseconds to defer trigger start.
87 <span id='Ext-dd-DragTracker-cfg-delegate'> /**
88 </span> * @cfg {String} delegate
89 * Optional. <p>A {@link Ext.DomQuery DomQuery} selector which identifies child elements within the DragTracker's encapsulating
90 * Element which are the tracked elements. This limits tracking to only begin when the matching elements are mousedowned.</p>
91 * <p>This may also be a specific child element within the DragTracker's encapsulating element to use as the tracked element.</p>
94 <span id='Ext-dd-DragTracker-cfg-preventDefault'> /**
95 </span> * @cfg {Boolean} preventDefault
96 * Specify <code>false</code> to enable default actions on onMouseDown events. Defaults to <code>true</code>.
99 <span id='Ext-dd-DragTracker-cfg-stopEvent'> /**
100 </span> * @cfg {Boolean} stopEvent
101 * Specify <code>true</code> to stop the <code>mousedown</code> event from bubbling to outer listeners from the target element (or its delegates). Defaults to <code>false</code>.
104 constructor : function(config){
105 Ext.apply(this, config);
107 <span id='Ext-dd-DragTracker-event-mouseover'> /**
108 </span> * @event mouseover <p><b>Only available when {@link #trackOver} is <code>true</code></b></p>
109 * <p>Fires when the mouse enters the DragTracker's target element (or if {@link #delegate} is
110 * used, when the mouse enters a delegate element).</p>
111 * @param {Object} this
112 * @param {Object} e event object
113 * @param {HtmlElement} target The element mouseovered.
117 <span id='Ext-dd-DragTracker-event-mouseout'> /**
118 </span> * @event mouseout <p><b>Only available when {@link #trackOver} is <code>true</code></b></p>
119 * <p>Fires when the mouse exits the DragTracker's target element (or if {@link #delegate} is
120 * used, when the mouse exits a delegate element).</p>
121 * @param {Object} this
122 * @param {Object} e event object
126 <span id='Ext-dd-DragTracker-event-mousedown'> /**
127 </span> * @event mousedown <p>Fires when the mouse button is pressed down, but before a drag operation begins. The
128 * drag operation begins after either the mouse has been moved by {@link #tolerance} pixels, or after
129 * the {@link #autoStart} timer fires.</p>
130 * <p>Return false to veto the drag operation.</p>
131 * @param {Object} this
132 * @param {Object} e event object
136 <span id='Ext-dd-DragTracker-event-mouseup'> /**
137 </span> * @event mouseup
138 * @param {Object} this
139 * @param {Object} e event object
143 <span id='Ext-dd-DragTracker-event-mousemove'> /**
144 </span> * @event mousemove Fired when the mouse is moved. Returning false cancels the drag operation.
145 * @param {Object} this
146 * @param {Object} e event object
150 <span id='Ext-dd-DragTracker-event-beforestart'> /**
151 </span> * @event beforestart
152 * @param {Object} this
153 * @param {Object} e event object
157 <span id='Ext-dd-DragTracker-event-dragstart'> /**
158 </span> * @event dragstart
159 * @param {Object} this
160 * @param {Object} e event object
164 <span id='Ext-dd-DragTracker-event-dragend'> /**
165 </span> * @event dragend
166 * @param {Object} this
167 * @param {Object} e event object
171 <span id='Ext-dd-DragTracker-event-drag'> /**
172 </span> * @event drag
173 * @param {Object} this
174 * @param {Object} e event object
179 this.dragRegion = Ext.create('Ext.util.Region', 0,0,0,0);
182 this.initEl(this.el);
185 // Dont pass the config so that it is not applied to 'this' again
186 this.mixins.observable.constructor.call(this);
193 <span id='Ext-dd-DragTracker-method-initEl'> /**
194 </span> * Initializes the DragTracker on a given element.
195 * @param {Ext.core.Element/HTMLElement} el The element
197 initEl: function(el) {
198 this.el = Ext.get(el);
200 // The delegate option may also be an element on which to listen
201 this.handle = Ext.get(this.delegate);
203 // If delegate specified an actual element to listen on, we do not use the delegate listener option
204 this.delegate = this.handle ? undefined : this.delegate;
207 this.handle = this.el;
210 // Add a mousedown listener which reacts only on the elements targeted by the delegate config.
211 // We process mousedown to begin tracking.
212 this.mon(this.handle, {
213 mousedown: this.onMouseDown,
214 delegate: this.delegate,
218 // If configured to do so, track mouse entry and exit into the target (or delegate).
219 // The mouseover and mouseout CANNOT be replaced with mouseenter and mouseleave
220 // because delegate cannot work with those pseudoevents. Entry/exit checking is done in the handler.
221 if (this.trackOver || this.overCls) {
222 this.mon(this.handle, {
223 mouseover: this.onMouseOver,
224 mouseout: this.onMouseOut,
225 delegate: this.delegate,
231 disable: function() {
232 this.disabled = true;
236 this.disabled = false;
239 destroy : function() {
240 this.clearListeners();
244 // When the pointer enters a tracking element, fire a mouseover if the mouse entered from outside.
245 // This is mouseenter functionality, but we cannot use mouseenter because we are using "delegate" to filter mouse targets
246 onMouseOver: function(e, target) {
249 if (Ext.EventManager.contains(e) || me.delegate) {
250 me.mouseIsOut = false;
252 me.el.addCls(me.overCls);
254 me.fireEvent('mouseover', me, e, me.delegate ? e.getTarget(me.delegate, target) : me.handle);
259 // When the pointer exits a tracking element, fire a mouseout.
260 // This is mouseleave functionality, but we cannot use mouseleave because we are using "delegate" to filter mouse targets
261 onMouseOut: function(e) {
262 if (this.mouseIsDown) {
263 this.mouseIsOut = true;
266 this.el.removeCls(this.overCls);
268 this.fireEvent('mouseout', this, e);
272 onMouseDown: function(e, target){
273 // If this is disabled, or the mousedown has been processed by an upstream DragTracker, return
274 if (this.disabled ||e.dragTracked) {
278 // This information should be available in mousedown listener and onBeforeStart implementations
279 this.dragTarget = this.delegate ? target : this.handle.dom;
280 this.startXY = this.lastXY = e.getXY();
281 this.startRegion = Ext.fly(this.dragTarget).getRegion();
283 if (this.fireEvent('mousedown', this, e) === false ||
284 this.fireEvent('beforedragstart', this, e) === false ||
285 this.onBeforeStart(e) === false) {
289 // Track when the mouse is down so that mouseouts while the mouse is down are not processed.
290 // The onMouseOut method will only ever be called after mouseup.
291 this.mouseIsDown = true;
293 // Flag for downstream DragTracker instances that the mouse is being tracked.
294 e.dragTracked = true;
296 if (this.preventDefault !== false) {
301 mouseup: this.onMouseUp,
302 mousemove: this.onMouseMove,
303 selectstart: this.stopSelect
305 if (this.autoStart) {
306 this.timer = Ext.defer(this.triggerStart, this.autoStart === true ? 1000 : this.autoStart, this, [e]);
310 onMouseMove: function(e, target){
311 // BrowserBug: IE hack to see if button was released outside of window.
312 // Needed in IE6-9 in quirks and strictmode
313 if (this.active && Ext.isIE && !e.browserEvent.button) {
325 if (Math.max(Math.abs(s[0]-xy[0]), Math.abs(s[1]-xy[1])) > this.tolerance) {
326 this.triggerStart(e);
332 // Returning false from a mousemove listener deactivates
333 if (this.fireEvent('mousemove', this, e) === false) {
337 this.fireEvent('drag', this, e);
341 onMouseUp: function(e) {
342 // Clear the flag which ensures onMouseOut fires only after the mouse button
343 // is lifted if the mouseout happens *during* a drag.
344 this.mouseIsDown = false;
346 // If we mouseouted the el *during* the drag, the onMouseOut method will not have fired. Ensure that it gets processed.
347 if (this.mouseIsOut) {
348 this.mouseIsOut = false;
352 this.fireEvent('mouseup', this, e);
356 <span id='Ext-dd-DragTracker-method-endDrag'> /**
358 * Stop the drag operation, and remove active mouse listeners.
360 endDrag: function(e) {
361 var doc = Ext.getDoc(),
362 wasActive = this.active;
364 doc.un('mousemove', this.onMouseMove, this);
365 doc.un('mouseup', this.onMouseUp, this);
366 doc.un('selectstart', this.stopSelect, this);
371 this.fireEvent('dragend', this, e);
373 // Private property calculated when first required and only cached during a drag
374 delete this._constrainRegion;
376 // Remove flag from event singleton. Using "Ext.EventObject" here since "endDrag" is called directly in some cases without an "e" param
377 delete Ext.EventObject.dragTracked;
380 triggerStart: function(e) {
384 this.fireEvent('dragstart', this, e);
387 clearStart : function() {
389 clearTimeout(this.timer);
394 stopSelect : function(e) {
399 <span id='Ext-dd-DragTracker-method-onBeforeStart'> /**
400 </span> * Template method which should be overridden by each DragTracker instance. Called when the user first clicks and
401 * holds the mouse button down. Return false to disallow the drag
402 * @param {Ext.EventObject} e The event object
404 onBeforeStart : function(e) {
408 <span id='Ext-dd-DragTracker-method-onStart'> /**
409 </span> * Template method which should be overridden by each DragTracker instance. Called when a drag operation starts
410 * (e.g. the user has moved the tracked element beyond the specified tolerance)
411 * @param {Ext.EventObject} e The event object
413 onStart : function(xy) {
417 <span id='Ext-dd-DragTracker-method-onDrag'> /**
418 </span> * Template method which should be overridden by each DragTracker instance. Called whenever a drag has been detected.
419 * @param {Ext.EventObject} e The event object
421 onDrag : function(e) {
425 <span id='Ext-dd-DragTracker-method-onEnd'> /**
426 </span> * Template method which should be overridden by each DragTracker instance. Called when a drag operation has been completed
427 * (e.g. the user clicked and held the mouse down, dragged the element and then released the mouse button)
428 * @param {Ext.EventObject} e The event object
430 onEnd : function(e) {
434 <span id='Ext-dd-DragTracker-method-getDragTarget'> /**
435 </span> * </p>Returns the drag target. This is usually the DragTracker's encapsulating element.</p>
436 * <p>If the {@link #delegate} option is being used, this may be a child element which matches the
437 * {@link #delegate} selector.</p>
438 * @return {Ext.core.Element} The element currently being tracked.
440 getDragTarget : function(){
441 return this.dragTarget;
444 <span id='Ext-dd-DragTracker-method-getDragCt'> /**
446 * @returns {Element} The DragTracker's encapsulating element.
448 getDragCt : function(){
452 <span id='Ext-dd-DragTracker-method-getConstrainRegion'> /**
454 * Return the Region into which the drag operation is constrained.
455 * Either the XY pointer itself can be constrained, or the dragTarget element
456 * The private property _constrainRegion is cached until onMouseUp
458 getConstrainRegion: function() {
459 if (this.constrainTo) {
460 if (this.constrainTo instanceof Ext.util.Region) {
461 return this.constrainTo;
463 if (!this._constrainRegion) {
464 this._constrainRegion = Ext.fly(this.constrainTo).getViewRegion();
467 if (!this._constrainRegion) {
468 this._constrainRegion = this.getDragCt().getViewRegion();
471 return this._constrainRegion;
474 getXY : function(constrain){
475 return constrain ? this.constrainModes[constrain](this, this.lastXY) : this.lastXY;
478 <span id='Ext-dd-DragTracker-method-getOffset'> /**
479 </span> * <p>Returns the X, Y offset of the current mouse position from the mousedown point.</p>
480 * <p>This method may optionally constrain the real offset values, and returns a point coerced in one
481 * of two modes:</p><ul>
482 * <li><code>point</code><div class="sub-desc">The current mouse position is coerced into the
483 * {@link #constrainRegion}, and the resulting position is returned.</div></li>
484 * <li><code>dragTarget</code><div class="sub-desc">The new {@link Ext.util.Region Region} of the
485 * {@link #getDragTarget dragTarget} is calculated based upon the current mouse position, and then
486 * coerced into the {@link #constrainRegion}. The returned mouse position is then adjusted by the
487 * same delta as was used to coerce the region.</div></li>
489 * @param constrainMode {String} Optional. If omitted the true mouse position is returned. May be passed
490 * as <code>'point'</code> or <code>'dragTarget'. See above.</code>.
491 * @returns {Array} The <code>X, Y</code> offset from the mousedown point, optionally constrained.
493 getOffset : function(constrain){
494 var xy = this.getXY(constrain),
497 return [xy[0]-s[0], xy[1]-s[1]];
501 // Constrain the passed point to within the constrain region
502 point: function(me, xy) {
503 var dr = me.dragRegion,
504 constrainTo = me.getConstrainRegion();
511 dr.x = dr.left = dr[0] = dr.right = xy[0];
512 dr.y = dr.top = dr[1] = dr.bottom = xy[1];
513 dr.constrainTo(constrainTo);
515 return [dr.left, dr.top];
518 // Constrain the dragTarget to within the constrain region. Return the passed xy adjusted by the same delta.
519 dragTarget: function(me, xy) {
521 dr = me.startRegion.copy(),
522 constrainTo = me.getConstrainRegion(),
530 // See where the passed XY would put the dragTarget if translated by the unconstrained offset.
531 // If it overflows, we constrain the passed XY to bring the potential
532 // region back within the boundary.
533 dr.translateBy(xy[0]-s[0], xy[1]-s[1]);
535 // Constrain the X coordinate by however much the dragTarget overflows
536 if (dr.right > constrainTo.right) {
537 xy[0] += adjust = (constrainTo.right - dr.right); // overflowed the right
540 if (dr.left < constrainTo.left) {
541 xy[0] += (constrainTo.left - dr.left); // overflowed the left
544 // Constrain the Y coordinate by however much the dragTarget overflows
545 if (dr.bottom > constrainTo.bottom) {
546 xy[1] += adjust = (constrainTo.bottom - dr.bottom); // overflowed the bottom
549 if (dr.top < constrainTo.top) {
550 xy[1] += (constrainTo.top - dr.top); // overflowed the top