Форк Rambox
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 

643 lines
21 KiB

/**
* 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 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 beforestart
* @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();
}
},
/**
* 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;
me.el = el = Ext.get(el);
// The delegate option may also be an element on which to listen
if (delegate && delegate.isElement) {
me.handle = delegate;
}
// If delegate specified an actual element to listen on, we do not use the delegate listener option
me.delegate = me.handle ? undefined : me.delegate;
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,
mousedown: me.onMouseDown,
dragstart: me.onDragStart
};
// 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() {
var me = this;
// endDrag has a mandatory event parameter
me.endDrag({});
me.clearListeners();
me.el = me.handle = null;
},
// 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 disabled, or the mousedown has been processed by an upstream DragTracker, return
if (me.disabled ||e.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.
e.dragTracked = true;
// See Ext.dd.DragDropManager::handleMouseDown
//<feature legacyBrowser>
me.el.setCapture();
//</feature>
e.stopPropagation();
if (me.preventDefault !== false) {
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;
if (wasActive) {
me.dragEnded = true;
me.onEnd(e);
me.fireEvent('dragend', me, 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 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;
}
}
});