/** * This class is a container used by the {@link Ext.dataview.plugin.ListSwiper listswiper} * plugin to display information and controls when an item is swiped. */Ext.define('Ext.dataview.listswiper.Stepper', { extend: 'Ext.dataview.listswiper.Item', xtype: 'listswiperstepper', requires: [ 'Ext.fx.easing.EaseOut', 'Ext.util.translatable.CssTransform' ], config: { /** * @cfg {String} iconCls * One or more space separated CSS classes to be applied to the icon element. * See {@link Ext.Button#iconCls} for details. */ iconCls: null, /** * @cfg {String} text * The swipe action text. */ text: null, /** * @cfg {Object} undo * A config object for the undo button. */ undo: { docked: 'right', ui: 'listswiperstepper-trigger' }, /** * @private */ step: null, /** * @private */ side: null, /** * @cfg {Boolean/Object} animation * `true` for the default animation (`{ duration: 500, easing: 'ease-out' }`) or * a standard animation config object to be used for default swipe animations. */ animation: true }, classCls: Ext.baseCSSPrefix + 'listswiperstepper', layout: { type: 'hbox', align: 'center' }, scrollDock: null, sideCls: { left: Ext.baseCSSPrefix + 'side-left', right: Ext.baseCSSPrefix + 'side-right' }, tpl: '<div class="' + Ext.baseCSSPrefix + 'listswiperstepper-text">{text}</div>', template: [{ reference: 'bodyElement', cls: Ext.baseCSSPrefix + 'body-el', uiCls: 'body-el', children: [{ reference: 'iconWrapElement', cls: Ext.baseCSSPrefix + 'icon-wrap-el', uiCls: 'icon-wrap-el', children: [{ reference: 'iconElement', cls: Ext.baseCSSPrefix + 'icon-el ' + Ext.baseCSSPrefix + 'font-icon' }] }, { reference: 'innerElement', cls: Ext.baseCSSPrefix + 'inner-el', uiCls: 'inner-el' }] }], initialize: function() { this.callParent(); this.bodyElement.on('tap', 'onTap', this); }, onRender: function() { this.steps = this.buildSteps(); }, applyAnimation: function(animation) { if (animation === true) { animation = { duration: 500, easing: { type: 'ease-out' } }; } return animation; }, updateTranslationTarget: function(target) { this.translatable = Ext.Factory.translatable({ element: target }, 'csstransform'); }, revert: function(animate) { var me = this, action = me.getAction(); me.invokeAction(action, 'revert'); me.finalize(animate); }, /** * Dismisses the pending action by triggering the `dismiss` event. * See {@link Ext.dataview.plugin.ListSwiper#actionOnDismiss} for details. */ dismiss: function(animate) { var me = this, action = me.getAction(), state = me.getState(); if (state === 'undo') { me.invokeAction(action, 'commit'); } me.finalize(animate); }, sortFn: function(a, b) { return b.x - a.x; }, /** * Builds a lookup tables with effective thresholds (in pixel) to save some calculations * during the multiple drag events. This tables should be invalidated every time the list * is horizontally resized (which should not happen during swipe interactions). * @private */ buildSteps: function() { var me = this, item = me.ownerCmp, el = item.el, left = me.getLeftActions() || {}, right = me.getRightActions() || {}, width = el.getWidth(), steps = { r: [], l: [] }, totalThreshold = 0, fn = function(side, index, action) { var threshold = Ext.util.Format.defaultValue(action.threshold, '25%'), number = parseInt(threshold, 10); if (isNaN(number)) { return; // skip this action! } if (typeof threshold === 'string' && threshold.indexOf('%') !== -1) { number = width * number / 100; } totalThreshold += number; steps[side].push({ action: action, side: side === 'r' ? 'right' : 'left', tx: side === 'r' ? -width : width, x: totalThreshold, key: action.key || index }); }; Ext.Object.each(left, fn.bind(this, 'l')); totalThreshold = 0; Ext.Object.each(right, fn.bind(this, 'r')); return steps; }, findStep: function(dx, force) { var me = this, res = { step: null, active: true }, steps = me.steps[dx > 0 ? 'l' : 'r'], ilen = steps.length, absDx = Math.abs(dx), step, i; for (i = ilen - 1; !res.step && i >= 0; --i) { step = steps[i]; if (step.x < absDx) { res.step = step; } } if (!res.step && force && ilen > 0) { res.step = steps[0]; res.active = false; } return res; }, updateStep: function(step, oldStep) { var me = this, action = (step && step.action), oldAction = (oldStep && oldStep.action), actionCls = action && action.cls, oldActionCls = oldAction && oldAction.cls, actionKeyCls = step && ('swipe-action-' + step.key), oldActionKeyCls = oldStep && ('swipe-action-' + oldStep.key); if (step) { me.setSide(step.side); } me.replaceCls(oldActionCls, actionCls); me.replaceCls(oldActionKeyCls, actionKeyCls, Ext.baseCSSPrefix); me.syncStep(); }, updateSide: function(side, oldSide) { var me = this, classes = me.sideCls, layout = me.getLayout(); me.replaceCls(classes[oldSide], classes[side]); if (layout.setPack) { layout.setPack(side === 'right' ? 'end' : 'start'); } }, onDragStart: function(evt) { evt.stopPropagation(); }, onDragMove: function(evt) { var me = this, plugin = me.owner, directionLock = plugin.getDirectionLock(), state = me.getState(), step = me.getStep(), translatable = me.translatable, dx = evt.deltaX, res; if (state === 'undo') { return; } if (state === 'consumed') { me.setState('reaquired'); translatable.stopAnimation(); } res = me.findStep(dx, true); // Direction Lock causes a friction pull if (directionLock && (res.step && step) && (res.step.side !== step.side)) { me.setState('overdrag'); res.step = null; } else { me.setState(res.step ? res.active ? 'active' : 'peek' : 'overdrag'); me.setStep(res.step || null); } // if there is no action for the current gesture, we still want to allow the user // to swipe the item but with friction to let him know that no action is available. translatable.translateAxis('x', res.step ? dx : dx * 0.1); evt.stopPropagation(); }, onDragEnd: function(evt) { var me = this, state = me.getState(), step = me.getStep(), dx = evt.deltaX, res; if (state === 'undo' || state === 'consumed') { return; } evt.stopPropagation(); res = me.findStep(dx, false); if (!res.step || res.step.side !== step.side) { me.finalize(true); return; } me.setStep(res.step); me.commit(true); }, commit: function(animate) { var me = this, step = me.getStep(), action = step.action, plugin = me.owner, translatable = me.translatable, delay, precommitResult, undo; me.setAction(action); precommitResult = me.invokeAction(action, 'precommit'); if (action.undoable) { me.setState('undo'); undo = me.add(me.getUndo()); undo.setHandler(me.onUndoTap.bind(me)); me.setSide(undo.getDocked() === 'left' ? 'right' : 'left'); translatable.translateAxis('x', step.tx, me.getAnimation()); delay = plugin.getCommitDelay(); if (delay) { if (precommitResult && precommitResult.then) { precommitResult.then(function() { plugin.dismissAllTask.delay(delay); }); } else { plugin.dismissAllTask.delay(delay); } } } else { if (precommitResult && precommitResult.then) { precommitResult.then(me.invokeAction.bind(me, action, 'commit')).then( me.finalize.bind(me, animate) ); } else { me.invokeAction(action, 'commit'); me.finalize(animate); } } }, finalize: function(animate) { var me = this, animation = me.getAnimation(), translatable = me.translatable; translatable.stopAnimation(); me.setState('consumed'); if (!animate) { me.doFinalize(); return; } if (translatable.x !== 0) { translatable.on({ animationend: 'doFinalize', single: true, scope: me }); translatable.translateAxis('x', 0, animate && animation); } }, doFinalize: function() { var me = this, plugin = me.owner, item = me.ownerCmp, state = me.getState(), translatable = me.translatable; // if the state is not consumed the user has picked up the swiper // before the close animation finished. if (state === 'consumed') { translatable.translateAxis('x', 0, false); if (!me.destroyed && item) { plugin.destroyItem(item); } } }, syncStep: function() { var me = this, item = me.ownerCmp, record = item.getRecord(), step = me.getStep(), ui = null, iconCls = '', text = '', action, data; if (step) { action = step.action; if (action) { ui = action.ui; iconCls = action.iconCls; text = action.text; data = action.data; } data = Ext.apply({ text: text }, data, record ? record.getData(true) : {}); this.setUi(ui); this.setIconCls(iconCls); me.setData(data); } else { this.setUi(null); this.setIconCls(null); me.setData(null); } }, updateIconCls: function(iconCls, oldIconCls) { this.iconElement.replaceCls(oldIconCls, iconCls); }, privates: { getRenderTarget: function() { return this.innerElement; }, onTap: function(evt) { var me = this, plugin = me.owner, dimissOnTap = plugin.getDismissOnTap(); evt.stopPropagation(); if (dimissOnTap) { me.dismiss(); } }, onUndoTap: function(button, evt) { evt.stopPropagation(); this.revert(); } }});