/** * A DragTracker listens for drag events on an Element and fires events at the start and end of the drag, * as well as during the drag. This is useful for components such as {@link Ext.slider.Multi}, where there is * an element that can be dragged around to change the Slider's value. * * DragTracker provides a series of template methods that should be overridden to provide functionality * in response to detected drag operations. These are onBeforeStart, onStart, onDrag, onCancel and onEnd. * See {@link Ext.slider.Multi}'s initEvents function for an example implementation. */Ext.define('Ext.dd.DragTracker', { uses: ['Ext.util.Region'], mixins: { observable: 'Ext.util.Observable' }, /** * @property {Boolean} active * Indicates whether the user is currently dragging this tracker. * @readonly */ active: false, /** * @property {HTMLElement} dragTarget * The element being dragged. * * Only valid during drag operations. * * If the {@link #delegate} option is used, this will be the delegate element which was mousedowned. * @readonly */ /** * @cfg {Boolean} trackOver * Set to true to fire mouseover and mouseout events when the mouse enters or leaves the target element. * * This is implicitly set when an {@link #overCls} is specified. * * If the {@link #delegate} option is used, these events fire only when a delegate element is entered of left. */ trackOver: false, /** * @cfg {String} overCls * A CSS class to add to the DragTracker's target element when the element (or, if the {@link #delegate} * option is used, when a delegate element) is mouseovered. * * If the {@link #delegate} option is used, these events fire only when a delegate element is entered of left. */ /** * @cfg {Ext.util.Region/Ext.dom.Element} constrainTo * A {@link Ext.util.Region Region} (Or an element from which a Region measurement will be read) * which is used to constrain the result of the {@link #getOffset} call. * * This may be set any time during the DragTracker's lifecycle to set a dynamic constraining region. */ /** * @cfg {Number} tolerance * Number of pixels the drag target must be moved before dragging is * considered to have started. */ tolerance: 5, /** * @cfg {Boolean/Number} autoStart * Specify `true` to defer trigger start by 1000 ms. * Specify a Number for the number of milliseconds to defer trigger start. */ autoStart: false, /** * @cfg {Ext.dom.Element/HTMLElement/String} el * The target element or ID of the element on which the DragTracker will be initialized. */ /** * @cfg {String} delegate * A CSS selector which identifies child elements within the DragTracker's encapsulating * Element which are the tracked elements. This limits tracking to only begin when the matching elements are mousedowned. * * This may also be a specific child element within the DragTracker's encapsulating element to use as the tracked element. */ /** * @cfg {Boolean} [preventDefault=true] * Specify `false` to enable default actions on onMouseDown events. */ /** * @cfg {Boolean} [stopEvent=false] * Specify `true` to stop the `mousedown` event from bubbling to outer listeners from the target element (or its delegates). */ /** * @event mouseover * Fires when the mouse enters the DragTracker's target element (or if {@link #delegate} is * used, when the mouse enters a delegate element). * * **Only available when {@link #trackOver} is `true`** * * @param {Object} this * @param {Object} e event object * @param {HTMLElement} target The element mouseovered. */ /** * @event mouseout * Fires when the mouse exits the DragTracker's target element (or if {@link #delegate} is * used, when the mouse exits a delegate element). * * **Only available when {@link #trackOver} is `true`** * * @param {Object} this * @param {Object} e event object */ /** * @event mousedown * Fires when the mouse button is pressed down, but before a drag operation begins. The * drag operation begins after either the mouse has been moved by {@link #tolerance} pixels, * or after the {@link #autoStart} timer fires. * * Return `false` to veto the drag operation. * * @param {Object} this * @param {Object} e event object */ /** * @event mouseup * @param {Object} this * @param {Object} e event object */ /** * @event mousemove * Fired when the mouse is moved. Returning false cancels the drag operation. * @param {Object} this * @param {Object} e event object */ /** * @event beforedragstart * @param {Object} this * @param {Object} e event object */ /** * @event dragstart * @param {Object} this * @param {Object} e event object */ /** * @event dragend * @param {Object} this * @param {Object} e event object */ /** * @event drag * @param {Object} this * @param {Object} e event object */ constructor : function(config){ var me = this; Ext.apply(me, config); me.dragRegion = new Ext.util.Region(0,0,0,0); if (me.el) { me.initEl(me.el); } // Dont pass the config so that it is not applied to 'this' again me.mixins.observable.constructor.call(me); if (me.disabled) { me.disable(); } if (Ext.supports.Touch) { Ext.getWin().on({ touchstart: 'onWindowTouchStart', scope: me, capture: true }); } }, /** * Initializes the DragTracker on a given element. * @param {Ext.dom.Element/HTMLElement/String} el The element or element ID */ initEl: function(el) { var me = this, delegate = me.delegate, elCmp, touchScrollable, unselectable; me.el = el = Ext.get(el); // Disable drag to select. We must take over any drag selecting gestures. // The delegate option may also be an element on which to listen if (delegate) { if (delegate.isElement) { me.handle = delegate; unselectable = delegate; } } else { unselectable = el; } // Only make the element unselectable if we have a known delegate, or this item is to be dragged. // Otherwise it's too wide of a net to cast, callers will need to apply unselectable to the appropriate // delegated elements to get the same effect. if (unselectable) { unselectable.addCls(Ext.baseCSSPrefix + 'unselectable'); } // If delegate specified an actual element to listen on, we do not use the delegate listener option me.delegate = me.handle ? undefined : me.delegate; // See if the handle or delegates are inside the scrolling part of the component. // If they are, we will need to use longpress to trigger the dragstart. if (Ext.supports.Touch) { elCmp = Ext.Component.from(el); touchScrollable = elCmp && elCmp.getScrollable(); if (touchScrollable) { elCmp = touchScrollable.getElement(); if (me.handle && !elCmp.contains(me.handle)) { touchScrollable = false; } else if (me.delegate && !elCmp.down(me.delegate)) { touchScrollable = false; } else { touchScrollable = touchScrollable.getX() || touchScrollable.getY(); } } } if (!me.handle) { me.handle = el; } // Add a mousedown listener which reacts only on the elements targeted by the delegate config. // We process mousedown to begin tracking. me.handleListeners = { scope: me, delegate: me.delegate, dragstart: me.onDragStart }; // If the element is part of a component which is scrollable by touch // then we have to use a longpress to trigger drag. // In this case, we also use untranslated mousedown because of multi input platforms. if (touchScrollable) { me.handleListeners.longpress = me.onMouseDown; me.handleListeners.mousedown = { fn: me.onMouseDown, delegate: me.delegate, translate: false }; me.handleListeners.contextmenu = function(e) { e.stopEvent(); }; } else { me.handleListeners.mousedown = me.onMouseDown; } // If configured to do so, track mouse entry and exit into the target (or delegate). // The mouseover and mouseout CANNOT be replaced with mouseenter and mouseleave // because delegate cannot work with those pseudoevents. Entry/exit checking is done in the handler. if (!Ext.supports.TouchEvents && (me.trackOver || me.overCls)) { Ext.apply(me.handleListeners, { mouseover: me.onMouseOver, mouseout: me.onMouseOut }); } me.mon(me.handle, me.handleListeners); // Accessibility me.keyNav = new Ext.util.KeyNav({ target: el, up: me.onResizeKeyDown, left: me.onResizeKeyDown, right: me.onResizeKeyDown, down: me.onResizeKeyDown, scope: me }); }, disable: function() { this.disabled = true; }, enable: function() { this.disabled = false; }, destroy: function() { // endDrag has a mandatory event parameter this.endDrag({}); Ext.destroy(this.keyNav); this.callParent(); }, onWindowTouchStart: function(e) { if (this.mouseIsDown) { // on devices that support multi-touch the second touch terminates drag this.onMouseUp(e); } }, // When the pointer enters a tracking element, fire a mouseover if the mouse entered from outside. // This is mouseenter functionality, but we cannot use mouseenter because we are using "delegate" to filter mouse targets onMouseOver: function(e, target) { var me = this, handleCls, el, i, len, cls; if (!me.disabled) { // Note that usually `delegate` is the same as `handleCls` just with a preceding '.' // Also, we're now adding the classes directly to the resizer el rather than to an ancestor since this // caused unwanted scrollbar flickering in IE 9 and less (both quirks and standards) when the panel // contained a textarea with auto overflow. It would cause an unwanted recalc as the ancestor had classes // added and removed. See EXTJS-11673. if (e.within(e.target, true, true) || me.delegate) { handleCls = me.handleCls; me.mouseIsOut = false; if (handleCls) { for (i = 0, len = me.handleEls.length; i < len; i++) { el = me.handleEls[i]; cls = el.delegateCls; if (!cls) { cls = el.delegateCls = [handleCls, '-', el.region, '-over'].join(''); } el.addCls([cls, me.overCls]); } } me.fireEvent('mouseover', me, e, me.delegate ? e.getTarget(me.delegate, target) : me.handle); } } }, // When the pointer exits a tracking element, fire a mouseout. // This is mouseleave functionality, but we cannot use mouseleave because we are using "delegate" to filter mouse targets onMouseOut: function(e) { var me = this, el, i, len; if (me.mouseIsDown) { me.mouseIsOut = true; } else { if (me.handleCls) { for (i = 0, len = me.handleEls.length; i < len; i++) { el = me.handleEls[i]; el.removeCls([el.delegateCls, me.overCls]); } } me.fireEvent('mouseout', me, e); } }, onMouseDown: function(e, target){ var me = this, // If this is a translated event, the event object is chained, so // we need to track on the parentEvent if it exists. trackEvent = e.parentEvent || e; // If this is disabled, or the mousedown has been processed by an upstream DragTracker, return if (me.disabled || trackEvent.dragTracked) { return; } // This information should be available in mousedown listener and onBeforeStart implementations me.dragTarget = me.delegate ? target : me.handle.dom; me.startXY = me.lastXY = e.getXY(); me.startRegion = Ext.fly(me.dragTarget).getRegion(); if (me.fireEvent('mousedown', me, e) === false || me.fireEvent('beforedragstart', me, e) === false || me.onBeforeStart(e) === false) { return; } // Track when the mouse is down so that mouseouts while the mouse is down are not processed. // The onMouseOut method will only ever be called after mouseup. me.mouseIsDown = true; // Flag for downstream DragTracker instances that the mouse is being tracked. trackEvent.dragTracked = true; // See Ext.dd.DragDropManager::handleMouseDown //<feature legacyBrowser> me.el.setCapture(); //</feature> e.stopPropagation(); if (me.preventDefault !== false || e.pointerType === 'touch') { e.preventDefault(); } Ext.getDoc().on({ scope: me, capture: true, mouseup: me.onMouseUp, mousemove: me.onMouseMove, selectstart: me.stopSelect }); // Flag for the onMouseMove method. // If endDrag is called while active via some other code such as a timer, or key event // then it sets dragEnded to indicate to any subsequent mousemove event that it should not proceed. me.dragEnded = false; if (!me.tolerance) { me.triggerStart(); } else if (me.autoStart) { me.timer = Ext.defer(me.triggerStart, me.autoStart === true ? 1000 : me.autoStart, me, [e]); } }, onMouseMove: function(e, target){ var me = this, xy = e.getXY(), s = me.startXY; e.stopPropagation(); if (me.preventDefault !== false) { e.preventDefault(); } // If, during a drag, some other action (eg a keystroke) hides or destroys the target, // endDrag will be called and the mousemove listener removed. But is the mouse is down // events continue to be delivered to the handler. If this happens, active will be false here. if (me.dragEnded) { return; } me.lastXY = xy; if (!me.active) { if (Math.max(Math.abs(s[0]-xy[0]), Math.abs(s[1]-xy[1])) > me.tolerance) { me.triggerStart(e); } else { return; } } // Returning false from a mousemove listener deactivates if (me.fireEvent('mousemove', me, e) === false) { me.onMouseUp(e); } else { me.onDrag(e); me.fireEvent('drag', me, e); } }, onMouseUp: function(e) { var me = this; // Clear the flag which ensures onMouseOut fires only after the mouse button // is lifted if the mouseout happens *during* a drag. me.mouseIsDown = false; // If we mouseouted the el *during* the drag, the onMouseOut method will not have fired. Ensure that it gets processed. if (me.mouseIsOut) { me.mouseIsOut = false; me.onMouseOut(e); } if (me.preventDefault !== false) { e.preventDefault(); } // See Ext.dd.DragDropManager::handleMouseDown if (Ext.isIE && document.releaseCapture) { document.releaseCapture(); } me.fireEvent('mouseup', me, e); me.endDrag(e); }, /** * @private * Stop the drag operation, and remove active mouse listeners. */ endDrag: function(e) { var me = this, wasActive = me.active; Ext.getDoc().un({ mousemove: me.onMouseMove, mouseup: me.onMouseUp, selectstart: me.stopSelect, capture: true, scope: me }); me.clearStart(); me.active = false; me.dragEnded = true; if (wasActive) { me.onEnd(e); me.fireEvent('dragend', me, e); } else { me.onCancel(e); } // Private property calculated when first required and only cached during a drag me._constrainRegion = null; }, triggerStart: function(e) { var me = this; me.clearStart(); me.active = true; me.onStart(e); me.fireEvent('dragstart', me, e); }, clearStart : function() { var timer = this.timer; if (timer) { clearTimeout(timer); this.timer = null; } }, stopSelect : function(e) { e.stopEvent(); return false; }, /** * Template method which should be overridden by each DragTracker instance. Called when the user first clicks and * holds the mouse button down. Return false to disallow the drag * @param {Ext.event.Event} e The event object * @template */ onBeforeStart : function(e) { }, /** * Template method which should be overridden by each DragTracker instance. Called when a drag operation starts * (e.g. the user has moved the tracked element beyond the specified tolerance) * @param {Ext.event.Event} e The event object * @template */ onStart : function(xy) { }, /** * Template method which should be overridden by each DragTracker instance. Called whenever a drag has been detected. * @param {Ext.event.Event} e The event object * @template */ onDrag : function(e) { }, /** * Template method which mey be overridden by each DragTracker instance. Called when a mouseup gesture is detected * but the onStart has not yet been reached. To clear things up that may have been set up on {@link #onBeforeStart}. * @param {Ext.event.Event} e The event object * @template */ onCancel : function(e) { }, /** * Template method which should be overridden by each DragTracker instance. Called when a drag operation has been completed * (e.g. the user clicked and held the mouse down, dragged the element and then released the mouse button) * @param {Ext.event.Event} e The event object * @template */ onEnd : function(e) { }, /** * Returns the drag target. This is usually the DragTracker's encapsulating element. * * If the {@link #delegate} option is being used, this may be a child element which matches the * {@link #delegate} selector. * * @return {Ext.dom.Element} The element currently being tracked. */ getDragTarget : function(){ return this.dragTarget; }, /** * @private * @return {Ext.dom.Element} The DragTracker's encapsulating element. */ getDragCt : function(){ return this.el; }, /** * @private * Return the Region into which the drag operation is constrained. * Either the XY pointer itself can be constrained, or the dragTarget element * The private property _constrainRegion is cached until onMouseUp */ getConstrainRegion: function() { var me = this; if (me.constrainTo) { if (me.constrainTo instanceof Ext.util.Region) { return me.constrainTo; } if (!me._constrainRegion) { me._constrainRegion = Ext.fly(me.constrainTo).getViewRegion(); } } else { if (!me._constrainRegion) { me._constrainRegion = me.getDragCt().getViewRegion(); } } return me._constrainRegion; }, getXY : function(constrain){ return constrain ? this.constrainModes[constrain](this, this.lastXY) : this.lastXY; }, /** * Returns the X, Y offset of the current mouse position from the mousedown point. * * This method may optionally constrain the real offset values, and returns a point coerced in one * of two modes: * * - `point` * The current mouse position is coerced into the constrainRegion and the resulting position is returned. * - `dragTarget` * The new {@link Ext.util.Region Region} of the {@link #getDragTarget dragTarget} is calculated * based upon the current mouse position, and then coerced into the constrainRegion. The returned * mouse position is then adjusted by the same delta as was used to coerce the region. * * @param {String} constrainMode (Optional) If omitted the true mouse position is returned. May be passed * as `point` or `dragTarget`. See above. * @return {Number[]} The `X, Y` offset from the mousedown point, optionally constrained. */ getOffset : function(constrain){ var xy = this.getXY(constrain), s = this.startXY; return [xy[0]-s[0], xy[1]-s[1]]; }, onDragStart: function(e) { e.stopPropagation(); }, constrainModes: { // Constrain the passed point to within the constrain region point: function(me, xy) { var dr = me.dragRegion, constrainTo = me.getConstrainRegion(); // No constraint if (!constrainTo) { return xy; } dr.x = dr.left = dr[0] = dr.right = xy[0]; dr.y = dr.top = dr[1] = dr.bottom = xy[1]; dr.constrainTo(constrainTo); return [dr.left, dr.top]; }, // Constrain the dragTarget to within the constrain region. Return the passed xy adjusted by the same delta. dragTarget: function(me, xy) { var s = me.startXY, dr = me.startRegion.copy(), constrainTo = me.getConstrainRegion(), adjust; // No constraint if (!constrainTo) { return xy; } // See where the passed XY would put the dragTarget if translated by the unconstrained offset. // If it overflows, we constrain the passed XY to bring the potential // region back within the boundary. dr.translateBy(xy[0]-s[0], xy[1]-s[1]); // Constrain the X coordinate by however much the dragTarget overflows if (dr.right > constrainTo.right) { xy[0] += adjust = (constrainTo.right - dr.right); // overflowed the right dr.left += adjust; } if (dr.left < constrainTo.left) { xy[0] += (constrainTo.left - dr.left); // overflowed the left } // Constrain the Y coordinate by however much the dragTarget overflows if (dr.bottom > constrainTo.bottom) { xy[1] += adjust = (constrainTo.bottom - dr.bottom); // overflowed the bottom dr.top += adjust; } if (dr.top < constrainTo.top) { xy[1] += (constrainTo.top - dr.top); // overflowed the top } return xy; } }});