/** * @class Ext.scroll.Scroller * @author Jacky Nguyen <jacky@sencha.com> * * Momentum scrolling is one of the most important part of the framework's UI layer. In Sencha Touch there are * several scroller implementations so we can have the best performance on all mobile devices and browsers. * * Scroller settings can be changed using the {@link Ext.Container#scrollable scrollable} configuration in * {@link Ext.Container}. Anything you pass to that method will be passed to the scroller when it is * instantiated in your container. * * Please note that the {@link Ext.Container#getScrollable} method returns an instance of {@link Ext.scroll.View}. * So if you need to get access to the scroller after your container has been instantiated, you must use the * {@link Ext.scroll.View#getScroller} method. * * // lets assume container is a container you have * // created which is scrollable * container.getScrollable().getScroller().setFps(10); * * ## Example * * Here is a simple example of how to adjust the scroller settings when using a {@link Ext.Container} (or anything * that extends it). * * @example * var container = Ext.create('Ext.Container', { * fullscreen: true, * html: 'This container is scrollable!', * scrollable: { * direction: 'vertical' * } * }); * * As you can see, we are passing the {@link #direction} configuration into the scroller instance in our container. * * You can pass any of the configs below in that {@link Ext.Container#scrollable scrollable} configuration and it will * just work. * * Go ahead and try it in the live code editor above! */Ext.define('Ext.scroll.Scroller', { extend: 'Ext.Evented', requires: [ 'Ext.fx.easing.BoundMomentum', 'Ext.fx.easing.EaseOut', 'Ext.util.Translatable' ], /** * @event maxpositionchange * Fires whenever the maximum position has changed. * @param {Ext.scroll.Scroller} this * @param {Number} maxPosition The new maximum position. */ /** * @event refresh * Fires whenever the Scroller is refreshed. * @param {Ext.scroll.Scroller} this */ /** * @event scrollstart * Fires whenever the scrolling is started. * @param {Ext.scroll.Scroller} this * @param {Number} x The current x position. * @param {Number} y The current y position. */ /** * @event scrollend * Fires whenever the scrolling is ended. * @param {Ext.scroll.Scroller} this * @param {Number} x The current x position. * @param {Number} y The current y position. */ /** * @event scroll * Fires whenever the Scroller is scrolled. * @param {Ext.scroll.Scroller} this * @param {Number} x The new x position. * @param {Number} y The new y position. */ config: { /** * @cfg element * @private */ element: null, /** * @cfg {String} direction * Possible values: 'auto', 'vertical', 'horizontal', or 'both'. * @accessor */ direction: 'auto', /** * @cfg fps * @private */ fps: 'auto', /** * @cfg {Boolean} disabled * Whether or not this component is disabled. * @accessor */ disabled: null, /** * @cfg {Boolean} directionLock * `true` to lock the direction of the scroller when the user starts scrolling. * This is useful when putting a scroller inside a scroller or a {@link Ext.Carousel}. * @accessor */ directionLock: false, /** * @cfg {Object} momentumEasing * A valid config for {@link Ext.fx.easing.BoundMomentum}. The default value is: * * { * momentum: { * acceleration: 30, * friction: 0.5 * }, * bounce: { * acceleration: 30, * springTension: 0.3 * } * } * * Note that supplied object will be recursively merged with the default object. For example, you can simply * pass this to change the momentum acceleration only: * * { * momentum: { * acceleration: 10 * } * } * * @accessor */ momentumEasing: { momentum: { acceleration: 30, friction: 0.5 }, bounce: { acceleration: 30, springTension: 0.3 }, minVelocity: 1 }, /** * @cfg bounceEasing * @private */ bounceEasing: { duration: 400 }, /** * @cfg outOfBoundRestrictFactor * @private */ outOfBoundRestrictFactor: 0.5, /** * @cfg startMomentumResetTime * @private */ startMomentumResetTime: 300, /** * @cfg maxAbsoluteVelocity * @private */ maxAbsoluteVelocity: 6, /** * @cfg containerSize * @private */ containerSize: 'auto', /** * @cfg size * @private */ size: 'auto', /** * @cfg autoRefresh * @private */ autoRefresh: true, /** * @cfg {Object/Number} initialOffset * The initial scroller position. When specified as Number, * both x and y will be set to that value. */ initialOffset: { x: 0, y: 0 }, /** * @cfg {Number/Object} slotSnapSize * The size of each slot to snap to in 'px', can be either an object with `x` and `y` values, i.e: * * { * x: 50, * y: 100 * } * * or a number value to be used for both directions. For example, a value of `50` will be treated as: * * { * x: 50, * y: 50 * } * * @accessor */ slotSnapSize: { x: 0, y: 0 }, /** * @cfg slotSnapOffset * @private */ slotSnapOffset: { x: 0, y: 0 }, slotSnapEasing: { duration: 150 }, translatable: { translationMethod: 'auto', useWrapper: false }, /** * @cfg {Boolean} mouseEnabled * @private * `true` to enable the scrolling using the mouse. */ mouseEnabled: false, /** * @private * May be set to `false` to inhibit constraining scroll position to within element size boundaries. * For example, during a drag operation which "overscrolls" an element in order to make room * for dropping below content. */ snapToBoundary: true }, statics: { instances: {} }, cls: Ext.baseCSSPrefix + 'scroll-scroller', containerCls: Ext.baseCSSPrefix + 'scroll-container', dragStartTime: 0, dragEndTime: 0, isDragging: false, isAnimating: false, isMouseEvent: { mousedown: 1, mousemove: 1, mouseup: 1 }, /** * @private * @constructor * @chainable */ constructor: function(config) { var me = this, element = config && config.element, onEvent = 'onEvent'; me.listenerMap = { touchstart: 'onTouchStart', touchend: 'onTouchEnd', dragstart: 'onDragStart', drag: 'onDrag', dragend: 'onDragEnd' }; me.listeners = { touchstart: onEvent, touchend: onEvent, dragstart: onEvent, drag: onEvent, dragend: onEvent, scope: me }; me.minPosition = { x: 0, y: 0 }; me.startPosition = { x: 0, y: 0 }; me.lastMaxPosition = { x: 0, y: 0 }; me.velocity = { x: 0, y: 0 }; me.isAxisEnabledFlags = { x: false, y: false }; me.flickStartPosition = { x: 0, y: 0 }; me.flickStartTime = { x: 0, y: 0 }; me.lastDragPosition = { x: 0, y: 0 }; me.dragDirection = { x: 0, y: 0}; me.initialConfig = config; if (element) { me.setElement(element); } me.self.instances[me.id] = me; return this; }, initialize: function() { var me = this; if (!me.initialized) { me.mixins.observable.constructor.call(me, me.initialConfig); me.initialized = true; } }, /** * @private */ applyElement: function(element) { if (!element) { return; } return Ext.get(element); }, /** * @private * @chainable */ updateElement: function(element) { var me = this; me.initialize(); if (!me.FixedHBoxStretching) { element.addCls(me.cls); } if (!me.getDisabled()) { me.attachListeners(); } var fn = function () { me.refreshMaxPosition(); // maintain indirection to name }, tee = Ext.Function.createSequence, updateContainerSize = me.updateContainerSize, updateSize = me.updateSize; me.updateContainerSize = updateContainerSize ? tee(updateContainerSize, fn) : fn; me.updateSize = updateSize ? tee(updateSize, fn) : fn; me.on({ maxpositionchange: 'snapToBoundary', minpositionchange: 'snapToBoundary', scope: me }); return me; }, applyTranslatable: function(config, translatable) { return Ext.factory(config, Ext.util.Translatable, translatable); }, updateTranslatable: function(translatable) { translatable.setConfig({ element: this.getElement(), listeners: { animationframe: 'onAnimationFrame', animationend: 'onAnimationEnd', scope: this } }); }, updateFps: function(fps) { if (fps !== 'auto') { this.getTranslatable().setFps(fps); } }, /** * @private */ attachListeners: function() { this.getContainer().on(this.listeners); }, /** * @private */ detachListeners: function() { this.getContainer().un(this.listeners); }, /** * @private */ updateDisabled: function(disabled) { if (disabled) { this.detachListeners(); } else { this.attachListeners(); } }, updateInitialOffset: function(initialOffset) { if (typeof initialOffset == 'number') { initialOffset = { x: initialOffset, y: initialOffset }; } this.getTranslatable().translate(-initialOffset.x, -initialOffset.y); }, /** * @private * @return {String} */ applyDirection: function(direction) { var minPosition = this.getMinPosition(), maxPosition = this.getMaxPosition(), isHorizontal, isVertical; this.givenDirection = direction; if (direction === 'auto') { isHorizontal = maxPosition.x > minPosition.x; isVertical = maxPosition.y > minPosition.y; if (isHorizontal && isVertical) { direction = 'both'; } else if (isHorizontal) { direction = 'horizontal'; } else { direction = 'vertical'; } } return direction; }, /** * @private */ updateDirection: function(direction, oldDirection) { var isAxisEnabledFlags = this.isAxisEnabledFlags, verticalCls = this.cls + '-vertical', horizontalCls = this.cls + '-horizontal', element = this.getElement(); if (oldDirection === 'both' || oldDirection === 'horizontal') { element.removeCls(horizontalCls); } if (oldDirection === 'both' || oldDirection === 'vertical') { element.removeCls(verticalCls); } isAxisEnabledFlags.x = isAxisEnabledFlags.y = false; if (direction === 'both' || direction === 'horizontal') { isAxisEnabledFlags.x = true; element.addCls(horizontalCls); } if (direction === 'both' || direction === 'vertical') { isAxisEnabledFlags.y = true; element.addCls(verticalCls); } }, /** * Returns `true` if a specified axis is enabled. * @param {String} axis The axis to check (`x` or `y`). * @return {Boolean} `true` if the axis is enabled. */ isAxisEnabled: function(axis) { this.getDirection(); return this.isAxisEnabledFlags[axis]; }, /** * @private * @return {Object} */ applyMomentumEasing: function(easing) { var defaultClass = Ext.fx.easing.BoundMomentum; return { x: Ext.factory(easing, defaultClass), y: Ext.factory(easing, defaultClass) }; }, /** * @private * @return {Object} */ applyBounceEasing: function(easing) { var defaultClass = Ext.fx.easing.EaseOut; return { x: Ext.factory(easing, defaultClass), y: Ext.factory(easing, defaultClass) }; }, updateBounceEasing: function(easing) { this.getTranslatable().setEasingX(easing.x).setEasingY(easing.y); }, /** * @private * @return {Object} */ applySlotSnapEasing: function(easing) { var defaultClass = Ext.fx.easing.EaseOut; return { x: Ext.factory(easing, defaultClass), y: Ext.factory(easing, defaultClass) }; }, /** * @private * @return {Object} */ getPosition: function() { return this.getTranslatable().getPosition(); }, /** * @private * @return {Object} */ getMinPosition: function() { var minPosition = this.minPosition; if (!minPosition) { this.minPosition = minPosition = { x: 0, y: 0 }; this.fireEvent('minpositionchange', this, minPosition); } return minPosition; }, /** * @private * @return {Object} */ getMaxPosition: function() { var me = this, maxPosition = me.maxPosition, size, containerSize; if (!maxPosition) { size = me.getSize(); containerSize = me.getContainerSize(); me.maxPosition = maxPosition = { x: Math.max(0, size.x - containerSize.x), y: Math.max(0, size.y - containerSize.y) }; // Only fire the event if either maximum changed. if (maxPosition.x !== me.lastMaxPosition.x || maxPosition.y !== me.lastMaxPosition.y) { me.fireEvent('maxpositionchange', me, maxPosition); me.lastMaxPosition.x = maxPosition.x; me.lastMaxPosition.y = maxPosition.y; } } return maxPosition; }, /** * @private */ refreshMaxPosition: function() { this.maxPosition = null; this.getMaxPosition(); }, /** * @private * @return {Object} */ applyContainerSize: function(size) { var containerDom = this.getContainer().dom, x, y; if (!containerDom) { return; } this.givenContainerSize = size; if (size === 'auto') { x = containerDom.clientWidth; y = containerDom.clientHeight; } else { x = size.x; y = size.y; } return { x: x, y: y }; }, /** * @private * @param {String/Object} size * @return {Object} */ applySize: function(size) { var dom = this.getElement().dom, containerDom = this.getContainer().dom, x, y; if (!dom) { return; } this.givenSize = size; if (size === 'auto') { x = Math.max(dom.offsetWidth, containerDom.clientWidth); y = Math.max(dom.offsetHeight, containerDom.clientHeight); } else if (typeof size == 'number') { x = size; y = size; } else { x = size.x; y = size.y; } return { x: x, y: y }; }, /** * @private */ updateAutoRefresh: function(autoRefresh) { this.getElement().toggleListener(autoRefresh, 'resize', 'onElementResize', this); this.getContainer().toggleListener(autoRefresh, 'resize', 'onContainerResize', this); }, applySlotSnapSize: function(snapSize) { if (typeof snapSize == 'number') { return { x: snapSize, y: snapSize }; } return snapSize; }, applySlotSnapOffset: function(snapOffset) { if (typeof snapOffset == 'number') { return { x: snapOffset, y: snapOffset }; } return snapOffset; }, /** * @private * Returns the container for this scroller */ getContainer: function() { var container = this.container, element; if (!container) { element = this.getElement().getParent(); this.container = container = this.FixedHBoxStretching ? element.getParent() : element; //<debug error> if (!container) { Ext.Logger.error("Making an element scrollable that doesn't have any container"); } //</debug> container.addCls(this.containerCls); } return container; }, /** * @private * @return {Ext.scroll.Scroller} this * @chainable */ refresh: function() { this.stopAnimation(); this.getTranslatable().refresh(); this.setSize(this.givenSize); this.setContainerSize(this.givenContainerSize); this.setDirection(this.givenDirection); this.fireEvent('refresh', this); return this; }, onElementResize: function(element, info) { this.setSize({ x: info.width, y: info.height }); this.refresh(); }, onContainerResize: function(container, info) { this.setContainerSize({ x: info.width, y: info.height }); this.refresh(); }, /** * Scrolls to the given location. * * @param {Number} x The scroll position on the x axis. * @param {Number} y The scroll position on the y axis. * @param {Boolean/Object} animation (optional) Whether or not to animate the scrolling to the new position. * * @return {Ext.scroll.Scroller} this * @chainable */ scrollTo: function(x, y, animation) { var me = this, fireScrollCallback; if (me.isDestroyed) { return me; } //<deprecated product=touch since=2.0> if (typeof x != 'number' && arguments.length === 1) { //<debug warn> Ext.Logger.deprecate("Calling scrollTo() with an object argument is deprecated, " + "please pass x and y arguments instead", me); //</debug> y = x.y; x = x.x; } //</deprecated> var translatable = me.getTranslatable(), position = me.getPosition(), positionChanged = false, translationX, translationY; if (me.isAxisEnabled('x')) { if (isNaN(x) || typeof x != 'number') { x = position.x; } else { if (position.x !== x) { positionChanged = true; } } translationX = this.convertX(-x); } if (me.isAxisEnabled('y')) { if (isNaN(y) || typeof y != 'number') { y = position.y; } else { if (position.y !== y) { positionChanged = true; } } translationY = -y; } if (positionChanged) { if (animation !== undefined && animation !== false) { // We need a callback to fire it after the animation fireScrollCallback = function() { me.fireEvent('scroll', me, x, y); }; // If they passed a boolean, create an object to hold the callback. if (animation === true) { animation = { callback: fireScrollCallback }; } // They want a callback, so we need to create a sequence on it. else if (animation.callback) { animation.callback = Ext.Function.createSequence(animation.callback, fireScrollCallback); } // We can just use the callback for our own purpose else { animation.callback = fireScrollCallback; } translatable.translateAnimated(translationX, translationY, animation); } else { translatable.translate(translationX, translationY); me.fireEvent('scroll', me, x, y); } } return this; }, /** * @private * @return {Ext.scroll.Scroller} this * @chainable */ scrollToTop: function(animation) { var initialOffset = this.getInitialOffset(); return this.scrollTo(initialOffset.x, initialOffset.y, animation); }, /** * Scrolls to the end of the scrollable view. * @return {Ext.scroll.Scroller} this * @chainable */ scrollToEnd: function(animation) { var size = this.getSize(), cntSize = this.getContainerSize(); return this.scrollTo(size.x - cntSize.x, size.y - cntSize.y, animation); }, /** * Change the scroll offset by the given amount. * @param {Number} x The offset to scroll by on the x axis. * @param {Number} y The offset to scroll by on the y axis. * @param {Boolean/Object} animation (optional) Whether or not to animate the scrolling to the new position. * @return {Ext.scroll.Scroller} this * @chainable */ scrollBy: function(x, y, animation) { var position = this.getPosition(); x = (typeof x == 'number') ? x + position.x : null; y = (typeof y == 'number') ? y + position.y : null; return this.scrollTo(x, y, animation); }, onEvent: function(e) { // use browserEvent to get the "real" type of DOM event that was fired, not a // potentially translated (or recognized) type var me = this, browserEvent = e.browserEvent; // prevents scrolling in response to mouse input on multi-input devices such // as windows 8 laptops with touch screens. If we are not on a device that // supports touch input, then do not prevent scrolling as scrolling is often // enabled on these devices for debugging purposes. // TODO: this should be handled by the event system once EXTJSIV-12840 is implemented if (!Ext.supports.Touch || me.getMouseEnabled() || (!me.isMouseEvent[browserEvent.type] && browserEvent.pointerType !== 'mouse')) { me[me.listenerMap[e.type]](e); } }, /** * @private */ onTouchStart: function() { this.isTouching = this.self.isTouching = true; this.stopAnimation(); }, /** * @private */ onTouchEnd: function() { var position = this.getPosition(); this.isTouching = this.self.isTouching = false; if (!this.isDragging && this.snapToSlot()) { this.fireEvent('scrollstart', this, position.x, position.y); } }, /** * @private */ onDragStart: function(e) { var direction = this.getDirection(), absDeltaX = e.absDeltaX, absDeltaY = e.absDeltaY, directionLock = this.getDirectionLock(), startPosition = this.startPosition, flickStartPosition = this.flickStartPosition, flickStartTime = this.flickStartTime, lastDragPosition = this.lastDragPosition, currentPosition = this.getPosition(), dragDirection = this.dragDirection, x = currentPosition.x, y = currentPosition.y, now = Ext.Date.now(); this.isDragging = true; if (directionLock && direction !== 'both') { if ((direction === 'horizontal' && absDeltaX > absDeltaY) || (direction === 'vertical' && absDeltaY > absDeltaX)) { e.stopPropagation(); } else { this.isDragging = false; return; } } lastDragPosition.x = x; lastDragPosition.y = y; flickStartPosition.x = x; flickStartPosition.y = y; startPosition.x = x; startPosition.y = y; flickStartTime.x = now; flickStartTime.y = now; dragDirection.x = 0; dragDirection.y = 0; this.dragStartTime = now; this.isDragging = true; this.fireEvent('scrollstart', this, x, y); }, /** * @private */ onAxisDrag: function(axis, delta) { if (!this.isAxisEnabled(axis)) { return; } var flickStartPosition = this.flickStartPosition, flickStartTime = this.flickStartTime, lastDragPosition = this.lastDragPosition, dragDirection = this.dragDirection, old = this.getPosition()[axis], min = this.getMinPosition()[axis], max = this.getMaxPosition()[axis], start = this.startPosition[axis], last = lastDragPosition[axis], current = start - delta, lastDirection = dragDirection[axis], restrictFactor = this.getOutOfBoundRestrictFactor(), startMomentumResetTime = this.getStartMomentumResetTime(), now = Ext.Date.now(), distance; if (current < min) { current *= restrictFactor; } else if (current > max) { distance = current - max; current = max + distance * restrictFactor; } if (current > last) { dragDirection[axis] = 1; } else if (current < last) { dragDirection[axis] = -1; } if ((lastDirection !== 0 && (dragDirection[axis] !== lastDirection)) || (now - flickStartTime[axis]) > startMomentumResetTime) { flickStartPosition[axis] = old; flickStartTime[axis] = now; } lastDragPosition[axis] = current; }, /** * @private */ onDrag: function(e) { if (!this.isDragging) { return; } var lastDragPosition = this.lastDragPosition; this.onAxisDrag('x', this.convertX(e.deltaX)); this.onAxisDrag('y', e.deltaY); this.scrollTo(lastDragPosition.x, lastDragPosition.y); }, /** * @private */ onDragEnd: function(e) { var easingX, easingY; if (!this.isDragging) { return; } this.dragEndTime = Ext.Date.now(); this.onDrag(e); this.isDragging = false; easingX = this.getAnimationEasing('x', e); easingY = this.getAnimationEasing('y', e); if (easingX || easingY) { this.getTranslatable().animate(easingX, easingY); } else { this.onScrollEnd(); } }, /** * @private */ getAnimationEasing: function(axis, e) { if (!this.isAxisEnabled(axis)) { return null; } var currentPosition = this.getPosition()[axis], minPosition = this.getMinPosition()[axis], maxPosition = this.getMaxPosition()[axis], maxAbsVelocity = this.getMaxAbsoluteVelocity(), boundValue = null, dragEndTime = this.dragEndTime, velocity = e.flick.velocity[axis], isX = axis === 'x', easingConfig, easing; if (currentPosition < minPosition) { boundValue = minPosition; } else if (currentPosition > maxPosition) { boundValue = maxPosition; } if (isX) { currentPosition = this.convertX(currentPosition); boundValue = this.convertX(boundValue); } // Out of bound, to be pulled back if (boundValue !== null) { easing = this.getBounceEasing()[axis]; easing.setConfig({ startTime: dragEndTime, startValue: -currentPosition, endValue: -boundValue }); return easing; } if (velocity === 0) { return null; } if (velocity < -maxAbsVelocity) { velocity = -maxAbsVelocity; } else if (velocity > maxAbsVelocity) { velocity = maxAbsVelocity; } if (Ext.browser.is.IE) { velocity *= 2; } easing = this.getMomentumEasing()[axis]; easingConfig = { startTime: dragEndTime, startValue: -currentPosition, startVelocity: velocity * 1.5, minMomentumValue: -maxPosition, maxMomentumValue: 0 }; if (isX) { this.convertEasingConfig(easingConfig); } easing.setConfig(easingConfig); return easing; }, /** * @private */ onAnimationFrame: function(translatable, x, y) { var position = this.getPosition(); this.fireEvent('scroll', this, this.convertX(position.x), position.y); }, /** * @private */ onAnimationEnd: function() { this.snapToBoundary(); this.onScrollEnd(); }, /** * @private * Stops the animation of the scroller at any time. */ stopAnimation: function() { this.getTranslatable().stopAnimation(); }, /** * @private */ onScrollEnd: function() { var position = this.getPosition(); if (this.isTouching || !this.snapToSlot()) { this.fireEvent('scrollend', this, position.x, position.y); } }, /** * @private * @return {Boolean} */ snapToSlot: function() { var snapX = this.getSnapPosition('x'), snapY = this.getSnapPosition('y'), easing = this.getSlotSnapEasing(); if (snapX !== null || snapY !== null) { this.scrollTo(snapX, snapY, { easingX: easing.x, easingY: easing.y }); return true; } return false; }, /** * @private * @return {Number/null} */ getSnapPosition: function(axis) { var snapSize = this.getSlotSnapSize()[axis], snapPosition = null, position, snapOffset, maxPosition, mod; if (snapSize !== 0 && this.isAxisEnabled(axis)) { position = this.getPosition()[axis]; snapOffset = this.getSlotSnapOffset()[axis]; maxPosition = this.getMaxPosition()[axis]; mod = Math.floor((position - snapOffset) % snapSize); if (mod !== 0) { if (position !== maxPosition) { if (Math.abs(mod) > snapSize / 2) { snapPosition = Math.min(maxPosition, position + ((mod > 0) ? snapSize - mod : mod - snapSize)); } else { snapPosition = position - mod; } } else { snapPosition = position - mod; } } } return snapPosition; }, /** * @private */ snapToBoundary: function() { // Snapping to boundary may be temporarily inhibited. // For example, during a drag operation which "overscrolls" an element in order to make room // for dropping below content. if (!this.getSnapToBoundary()) { return; } var position = this.getPosition(), minPosition = this.getMinPosition(), maxPosition = this.getMaxPosition(), minX = minPosition.x, minY = minPosition.y, maxX = maxPosition.x, maxY = maxPosition.y, x = Math.round(position.x), y = Math.round(position.y); if (x < minX) { x = minX; } else if (x > maxX) { x = maxX; } if (y < minY) { y = minY; } else if (y > maxY) { y = maxY; } this.scrollTo(x, y); }, destroy: function() { var element = this.getElement(), sizeMonitors = this.sizeMonitors, container; if (sizeMonitors) { sizeMonitors.element.destroy(); sizeMonitors.container.destroy(); } if (element && !element.isDestroyed) { element.removeCls(this.cls); container = this.getContainer(); if (container && !container.isDestroyed) { container.removeCls(this.containerCls); } } Ext.destroy(this.getTranslatable()); delete this.self.instances[this.id]; this.callParent(arguments); }, // hook for rtl mode to convert an x coordinate to RTL space. convertX: function(x) { return x; }, // overridden in RTL mode to swap min/max momentum values convertEasingConfig: function(config) { return config; } }, function() { //<deprecated product=touch since=2.0> this.override({ constructor: function(config) { var element, acceleration, slotSnapOffset, friction, springTension, minVelocity; if (!config) { config = {}; } if (typeof config == 'string') { config = { direction: config }; } if (arguments.length == 2) { //<debug warn> Ext.Logger.deprecate("Passing element as the first argument is deprecated, pass it as the " + "'element' property of the config object instead"); //</debug> element = config; config = arguments[1]; if (!config) { config = {}; } config.element = element; } /** * @cfg {Number} acceleration A higher acceleration gives the scroller more initial velocity. * @deprecated 2.0.0 Please use {@link #momentumEasing}.momentum.acceleration and {@link #momentumEasing}.bounce.acceleration instead. */ if (config.hasOwnProperty('acceleration')) { acceleration = config.acceleration; delete config.acceleration; //<debug warn> Ext.Logger.deprecate("'acceleration' config is deprecated, set momentumEasing.momentum.acceleration and momentumEasing.bounce.acceleration configs instead"); //</debug> Ext.merge(config, { momentumEasing: { momentum: { acceleration: acceleration }, bounce: { acceleration: acceleration } } }); } if (config.hasOwnProperty('snap')) { config.slotSnapOffset = config.snap; //<debug warn> Ext.Logger.deprecate("'snap' config is deprecated, please use the 'slotSnapOffset' config instead"); //</debug> } /** * @cfg {Number} friction The friction of the scroller. By raising this value the length that momentum scrolls * becomes shorter. This value is best kept between 0 and 1. * @deprecated 2.0.0 Please set the {@link #momentumEasing}.momentum.friction configuration instead */ if (config.hasOwnProperty('friction')) { friction = config.friction; delete config.friction; //<debug warn> Ext.Logger.deprecate("'friction' config is deprecated, set momentumEasing.momentum.friction config instead"); //</debug> Ext.merge(config, { momentumEasing: { momentum: { friction: friction } } }); } if (config.hasOwnProperty('springTension')) { springTension = config.springTension; delete config.springTension; //<debug warn> Ext.Logger.deprecate("'springTension' config is deprecated, set momentumEasing.momentum.springTension config instead"); //</debug> Ext.merge(config, { momentumEasing: { momentum: { springTension: springTension } } }); } if (config.hasOwnProperty('minVelocityForAnimation')) { minVelocity = config.minVelocityForAnimation; delete config.minVelocityForAnimation; //<debug warn> Ext.Logger.deprecate("'minVelocityForAnimation' config is deprecated, set momentumEasing.minVelocity config instead"); //</debug> Ext.merge(config, { momentumEasing: { minVelocity: minVelocity } }); } this.callOverridden(arguments); }, scrollToAnimated: function(x, y, animation) { //<debug warn> Ext.Logger.deprecate("scrollToAnimated() is deprecated, please use `scrollTo()` and pass 'animation' as " + "the third argument instead"); //</debug> return this.scrollTo.apply(this, arguments); }, scrollBy: function(x, y, animation) { if (Ext.isObject(x)) { //<debug warn> Ext.Logger.deprecate("calling `scrollBy()` with an object of `x` and `y` properties is no longer supported. " + "Please pass `x` and `y` values as two separate arguments instead"); //</debug> y = x.y; x = x.x; } return this.callOverridden([x, y, animation]); }, /** * Sets the offset of this scroller. * @param {Object} offset The offset to move to. * @param {Number} offset.x The x-axis offset. * @param {Number} offset.y The y-axis offset. * @deprecated 2.0.0 Please use `{@link #scrollTo}` instead. * @return {Ext.scroll.Scroller} this * @chainable */ setOffset: function(offset) { return this.scrollToAnimated(-offset.x, -offset.y); } }); /** * @method updateBoundary * Updates the boundary information for this scroller. * @return {Ext.scroll.Scroller} this * @removed 2.0.0 Please use {@link #method-refresh} instead. * @chainable */// Ext.deprecateClassMethod('updateBoundary', 'refresh'); //</deprecated> });