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.

938 lines
29 KiB

This file contains ambiguous Unicode characters!

This file contains ambiguous Unicode characters that may be confused with others in your current locale. If your use case is intentional and legitimate, you can safely ignore this warning. Use the Escape button to highlight these characters.

(function($, window, document, undefined) {
var CLASS_SCROLL = $.className('scroll');
var CLASS_SCROLLBAR = $.className('scrollbar');
var CLASS_INDICATOR = $.className('scrollbar-indicator');
var CLASS_SCROLLBAR_VERTICAL = CLASS_SCROLLBAR + '-vertical';
var CLASS_SCROLLBAR_HORIZONTAL = CLASS_SCROLLBAR + '-horizontal';
var CLASS_ACTIVE = $.className('active');
var ease = {
quadratic: {
style: 'cubic-bezier(0.25, 0.46, 0.45, 0.94)',
fn: function(k) {
return k * (2 - k);
}
},
circular: {
style: 'cubic-bezier(0.1, 0.57, 0.1, 1)',
fn: function(k) {
return Math.sqrt(1 - (--k * k));
}
},
outCirc: {
style: 'cubic-bezier(0.075, 0.82, 0.165, 1)'
},
outCubic: {
style: 'cubic-bezier(0.165, 0.84, 0.44, 1)'
}
}
var Scroll = $.Class.extend({
init: function(element, options) {
this.wrapper = this.element = element;
this.scroller = this.wrapper.children[0];
this.scrollerStyle = this.scroller && this.scroller.style;
this.stopped = false;
this.options = $.extend(true, {
scrollY: true, //是否竖向滚动
scrollX: false, //是否横向滚动
startX: 0, //初始化时滚动至x
startY: 0, //初始化时滚动至y
indicators: true, //是否显示滚动条
stopPropagation: false,
hardwareAccelerated: true,
fixedBadAndorid: false,
preventDefaultException: {
tagName: /^(INPUT|TEXTAREA|BUTTON|SELECT|VIDEO)$/
},
momentum: true,
snapX: 0.5, //横向切换距离(以当前容器宽度为基准)
snap: false, //图片轮播,拖拽式选项卡
bounce: true, //是否启用回弹
bounceTime: 500, //回弹动画时间
bounceEasing: ease.outCirc, //回弹动画曲线
scrollTime: 500,
scrollEasing: ease.outCubic, //轮播动画曲线
directionLockThreshold: 5,
parallaxElement: false, //视差元素
parallaxRatio: 0.5
}, options);
this.x = 0;
this.y = 0;
this.translateZ = this.options.hardwareAccelerated ? ' translateZ(0)' : '';
this._init();
if (this.scroller) {
this.refresh();
// if (this.options.startX !== 0 || this.options.startY !== 0) { //需要判断吗?后续根据实际情况再看看
this.scrollTo(this.options.startX, this.options.startY);
// }
}
},
_init: function() {
this._initParallax();
this._initIndicators();
this._initEvent();
},
_initParallax: function() {
if (this.options.parallaxElement) {
this.parallaxElement = document.querySelector(this.options.parallaxElement);
this.parallaxStyle = this.parallaxElement.style;
this.parallaxHeight = this.parallaxElement.offsetHeight;
this.parallaxImgStyle = this.parallaxElement.querySelector('img').style;
}
},
_initIndicators: function() {
var self = this;
self.indicators = [];
if (!this.options.indicators) {
return;
}
var indicators = [],
indicator;
// Vertical scrollbar
if (self.options.scrollY) {
indicator = {
el: this._createScrollBar(CLASS_SCROLLBAR_VERTICAL),
listenX: false
};
this.wrapper.appendChild(indicator.el);
indicators.push(indicator);
}
// Horizontal scrollbar
if (this.options.scrollX) {
indicator = {
el: this._createScrollBar(CLASS_SCROLLBAR_HORIZONTAL),
listenY: false
};
this.wrapper.appendChild(indicator.el);
indicators.push(indicator);
}
for (var i = indicators.length; i--;) {
this.indicators.push(new Indicator(this, indicators[i]));
}
},
_initSnap: function() {
this.currentPage = {};
this.pages = [];
var snaps = this.snaps;
var length = snaps.length;
var m = 0;
var n = -1;
var x = 0;
var leftX = 0;
var rightX = 0;
var snapX = 0;
for (var i = 0; i < length; i++) {
var snap = snaps[i];
var offsetLeft = snap.offsetLeft;
var offsetWidth = snap.offsetWidth;
if (i === 0 || offsetLeft <= snaps[i - 1].offsetLeft) {
m = 0;
n++;
}
if (!this.pages[m]) {
this.pages[m] = [];
}
x = this._getSnapX(offsetLeft);
snapX = Math.round((offsetWidth) * this.options.snapX);
leftX = x - snapX;
rightX = x - offsetWidth + snapX;
this.pages[m][n] = {
x: x,
leftX: leftX,
rightX: rightX,
pageX: m,
element: snap
}
if (snap.classList.contains(CLASS_ACTIVE)) {
this.currentPage = this.pages[m][0];
}
if (x >= this.maxScrollX) {
m++;
}
}
this.options.startX = this.currentPage.x || 0;
},
_getSnapX: function(offsetLeft) {
return Math.max(Math.min(0, -offsetLeft + (this.wrapperWidth / 2)), this.maxScrollX);
},
_gotoPage: function(index) {
this.currentPage = this.pages[Math.min(index, this.pages.length - 1)][0];
for (var i = 0, len = this.snaps.length; i < len; i++) {
if (i === index) {
this.snaps[i].classList.add(CLASS_ACTIVE);
} else {
this.snaps[i].classList.remove(CLASS_ACTIVE);
}
}
this.scrollTo(this.currentPage.x, 0, this.options.scrollTime);
},
_nearestSnap: function(x) {
if (!this.pages.length) {
return {
x: 0,
pageX: 0
};
}
var i = 0;
var length = this.pages.length;
if (x > 0) {
x = 0;
} else if (x < this.maxScrollX) {
x = this.maxScrollX;
}
for (; i < length; i++) {
var nearestX = this.direction === 'left' ? this.pages[i][0].leftX : this.pages[i][0].rightX;
if (x >= nearestX) {
return this.pages[i][0];
}
}
return {
x: 0,
pageX: 0
};
},
_initEvent: function(detach) {
var action = detach ? 'removeEventListener' : 'addEventListener';
window[action]('orientationchange', this);
window[action]('resize', this);
this.scroller[action]('webkitTransitionEnd', this);
this.wrapper[action]($.EVENT_START, this);
this.wrapper[action]($.EVENT_CANCEL, this);
this.wrapper[action]($.EVENT_END, this);
this.wrapper[action]('drag', this);
this.wrapper[action]('dragend', this);
this.wrapper[action]('flick', this);
this.wrapper[action]('scrollend', this);
if (this.options.scrollX) {
this.wrapper[action]('swiperight', this);
}
var segmentedControl = this.wrapper.querySelector($.classSelector('.segmented-control'));
if (segmentedControl) { //靠这个bug排查了一下午阻止hash跳转一旦hash跳转会导致可拖拽选项卡的tab不见
mui(segmentedControl)[detach ? 'off' : 'on']('click', 'a', $.preventDefault);
}
this.wrapper[action]('scrollstart', this);
this.wrapper[action]('refresh', this);
},
_handleIndicatorScrollend: function() {
this.indicators.map(function(indicator) {
indicator.fade();
});
},
_handleIndicatorScrollstart: function() {
this.indicators.map(function(indicator) {
indicator.fade(1);
});
},
_handleIndicatorRefresh: function() {
this.indicators.map(function(indicator) {
indicator.refresh();
});
},
handleEvent: function(e) {
if (this.stopped) {
this.resetPosition();
return;
}
switch (e.type) {
case $.EVENT_START:
this._start(e);
break;
case 'drag':
this.options.stopPropagation && e.stopPropagation();
this._drag(e);
break;
case 'dragend':
case 'flick':
this.options.stopPropagation && e.stopPropagation();
this._flick(e);
break;
case $.EVENT_CANCEL:
case $.EVENT_END:
this._end(e);
break;
case 'webkitTransitionEnd':
this.transitionTimer && this.transitionTimer.cancel();
this._transitionEnd(e);
break;
case 'scrollstart':
this._handleIndicatorScrollstart(e);
break;
case 'scrollend':
this._handleIndicatorScrollend(e);
this._scrollend(e);
e.stopPropagation();
break;
case 'orientationchange':
case 'resize':
this._resize();
break;
case 'swiperight':
e.stopPropagation();
break;
case 'refresh':
this._handleIndicatorRefresh(e);
break;
}
},
_start: function(e) {
this.moved = this.needReset = false;
this._transitionTime();
if (this.isInTransition) {
this.needReset = true;
this.isInTransition = false;
var pos = $.parseTranslateMatrix($.getStyles(this.scroller, 'webkitTransform'));
this.setTranslate(Math.round(pos.x), Math.round(pos.y));
// this.resetPosition(); //reset
$.trigger(this.scroller, 'scrollend', this);
// e.stopPropagation();
e.preventDefault();
}
this.reLayout();
$.trigger(this.scroller, 'beforescrollstart', this);
},
_getDirectionByAngle: function(angle) {
if (angle < -80 && angle > -100) {
return 'up';
} else if (angle >= 80 && angle < 100) {
return 'down';
} else if (angle >= 170 || angle <= -170) {
return 'left';
} else if (angle >= -35 && angle <= 10) {
return 'right';
}
return null;
},
_drag: function(e) {
// if (this.needReset) {
// e.stopPropagation(); //disable parent drag(nested scroller)
// return;
// }
var detail = e.detail;
if (this.options.scrollY || detail.direction === 'up' || detail.direction === 'down') { //如果是竖向滚动或手势方向是上或下
//ios8 hack
if ($.os.ios && parseFloat($.os.version) >= 8) { //多webview时离开当前webview会导致后续touch事件不触发
var clientY = detail.gesture.touches[0].clientY;
//下拉刷新 or 上拉加载
if ((clientY + 10) > window.innerHeight || clientY < 10) {
this.resetPosition(this.options.bounceTime);
return;
}
}
}
var isPreventDefault = isReturn = false;
var direction = this._getDirectionByAngle(detail.angle);
if (detail.direction === 'left' || detail.direction === 'right') {
if (this.options.scrollX) {
isPreventDefault = true;
if (!this.moved) { //识别角度(该角度导致轮播不灵敏)
// if (direction !== 'left' && direction !== 'right') {
// isReturn = true;
// } else {
$.gestures.session.lockDirection = true; //锁定方向
$.gestures.session.startDirection = detail.direction;
// }
}
} else if (this.options.scrollY && !this.moved) {
isReturn = true;
}
} else if (detail.direction === 'up' || detail.direction === 'down') {
if (this.options.scrollY) {
isPreventDefault = true;
// if (!this.moved) { //识别角度,竖向滚动似乎没必要进行小角度验证
// if (direction !== 'up' && direction !== 'down') {
// isReturn = true;
// }
// }
if (!this.moved) {
$.gestures.session.lockDirection = true; //锁定方向
$.gestures.session.startDirection = detail.direction;
}
} else if (this.options.scrollX && !this.moved) {
isReturn = true;
}
} else {
isReturn = true;
}
if (this.moved || isPreventDefault) {
e.stopPropagation(); //阻止冒泡(scroll类嵌套)
detail.gesture && detail.gesture.preventDefault();
}
if (isReturn) { //禁止非法方向滚动
return;
}
if (!this.moved) {
$.trigger(this.scroller, 'scrollstart', this);
} else {
e.stopPropagation(); //move期间阻止冒泡(scroll嵌套)
}
var deltaX = 0;
var deltaY = 0;
if (!this.moved) { //start
deltaX = detail.deltaX;
deltaY = detail.deltaY;
} else { //move
deltaX = detail.deltaX - $.gestures.session.prevTouch.deltaX;
deltaY = detail.deltaY - $.gestures.session.prevTouch.deltaY;
}
var absDeltaX = Math.abs(detail.deltaX);
var absDeltaY = Math.abs(detail.deltaY);
if (absDeltaX > absDeltaY + this.options.directionLockThreshold) {
deltaY = 0;
} else if (absDeltaY >= absDeltaX + this.options.directionLockThreshold) {
deltaX = 0;
}
deltaX = this.hasHorizontalScroll ? deltaX : 0;
deltaY = this.hasVerticalScroll ? deltaY : 0;
var newX = this.x + deltaX;
var newY = this.y + deltaY;
// Slow down if outside of the boundaries
if (newX > 0 || newX < this.maxScrollX) {
newX = this.options.bounce ? this.x + deltaX / 3 : newX > 0 ? 0 : this.maxScrollX;
}
if (newY > 0 || newY < this.maxScrollY) {
newY = this.options.bounce ? this.y + deltaY / 3 : newY > 0 ? 0 : this.maxScrollY;
}
if (!this.requestAnimationFrame) {
this._updateTranslate();
}
this.direction = detail.deltaX > 0 ? 'right' : 'left';
this.moved = true;
this.x = newX;
this.y = newY;
$.trigger(this.scroller, 'scroll', this);
},
_flick: function(e) {
// if (!this.moved || this.needReset) {
// return;
// }
if (!this.moved) {
return;
}
e.stopPropagation();
var detail = e.detail;
this._clearRequestAnimationFrame();
if (e.type === 'dragend' && detail.flick) { //dragend
return;
}
var newX = Math.round(this.x);
var newY = Math.round(this.y);
this.isInTransition = false;
// reset if we are outside of the boundaries
if (this.resetPosition(this.options.bounceTime)) {
return;
}
this.scrollTo(newX, newY); // ensures that the last position is rounded
if (e.type === 'dragend') { //dragend
$.trigger(this.scroller, 'scrollend', this);
return;
}
var time = 0;
var easing = '';
// start momentum animation if needed
if (this.options.momentum && detail.flickTime < 300) {
momentumX = this.hasHorizontalScroll ? this._momentum(this.x, detail.flickDistanceX, detail.flickTime, this.maxScrollX, this.options.bounce ? this.wrapperWidth : 0, this.options.deceleration) : {
destination: newX,
duration: 0
};
momentumY = this.hasVerticalScroll ? this._momentum(this.y, detail.flickDistanceY, detail.flickTime, this.maxScrollY, this.options.bounce ? this.wrapperHeight : 0, this.options.deceleration) : {
destination: newY,
duration: 0
};
newX = momentumX.destination;
newY = momentumY.destination;
time = Math.max(momentumX.duration, momentumY.duration);
this.isInTransition = true;
}
if (newX != this.x || newY != this.y) {
if (newX > 0 || newX < this.maxScrollX || newY > 0 || newY < this.maxScrollY) {
easing = ease.quadratic;
}
this.scrollTo(newX, newY, time, easing);
return;
}
$.trigger(this.scroller, 'scrollend', this);
// e.stopPropagation();
},
_end: function(e) {
this.needReset = false;
if ((!this.moved && this.needReset) || e.type === $.EVENT_CANCEL) {
this.resetPosition();
}
},
_transitionEnd: function(e) {
if (e.target != this.scroller || !this.isInTransition) {
return;
}
this._transitionTime();
if (!this.resetPosition(this.options.bounceTime)) {
this.isInTransition = false;
$.trigger(this.scroller, 'scrollend', this);
}
},
_scrollend: function(e) {
if ((this.y === 0 && this.maxScrollY === 0) || (Math.abs(this.y) > 0 && this.y <= this.maxScrollY)) {
$.trigger(this.scroller, 'scrollbottom', this);
}
},
_resize: function() {
var that = this;
clearTimeout(that.resizeTimeout);
that.resizeTimeout = setTimeout(function() {
that.refresh();
}, that.options.resizePolling);
},
_transitionTime: function(time) {
time = time || 0;
this.scrollerStyle['webkitTransitionDuration'] = time + 'ms';
if (this.parallaxElement && this.options.scrollY) { //目前仅支持竖向视差效果
this.parallaxStyle['webkitTransitionDuration'] = time + 'ms';
}
if (this.options.fixedBadAndorid && !time && $.os.isBadAndroid) {
this.scrollerStyle['webkitTransitionDuration'] = '0.001s';
if (this.parallaxElement && this.options.scrollY) { //目前仅支持竖向视差效果
this.parallaxStyle['webkitTransitionDuration'] = '0.001s';
}
}
if (this.indicators) {
for (var i = this.indicators.length; i--;) {
this.indicators[i].transitionTime(time);
}
}
if (time) { //自定义timer保证webkitTransitionEnd始终触发
this.transitionTimer && this.transitionTimer.cancel();
this.transitionTimer = $.later(function() {
$.trigger(this.scroller, 'webkitTransitionEnd');
}, time + 100, this);
}
},
_transitionTimingFunction: function(easing) {
this.scrollerStyle['webkitTransitionTimingFunction'] = easing;
if (this.parallaxElement && this.options.scrollY) { //目前仅支持竖向视差效果
this.parallaxStyle['webkitTransitionDuration'] = easing;
}
if (this.indicators) {
for (var i = this.indicators.length; i--;) {
this.indicators[i].transitionTimingFunction(easing);
}
}
},
_translate: function(x, y) {
this.x = x;
this.y = y;
},
_clearRequestAnimationFrame: function() {
if (this.requestAnimationFrame) {
cancelAnimationFrame(this.requestAnimationFrame);
this.requestAnimationFrame = null;
}
},
_updateTranslate: function() {
var self = this;
if (self.x !== self.lastX || self.y !== self.lastY) {
self.setTranslate(self.x, self.y);
}
self.requestAnimationFrame = requestAnimationFrame(function() {
self._updateTranslate();
});
},
_createScrollBar: function(clazz) {
var scrollbar = document.createElement('div');
var indicator = document.createElement('div');
scrollbar.className = CLASS_SCROLLBAR + ' ' + clazz;
indicator.className = CLASS_INDICATOR;
scrollbar.appendChild(indicator);
if (clazz === CLASS_SCROLLBAR_VERTICAL) {
this.scrollbarY = scrollbar;
this.scrollbarIndicatorY = indicator;
} else if (clazz === CLASS_SCROLLBAR_HORIZONTAL) {
this.scrollbarX = scrollbar;
this.scrollbarIndicatorX = indicator;
}
this.wrapper.appendChild(scrollbar);
return scrollbar;
},
_preventDefaultException: function(el, exceptions) {
for (var i in exceptions) {
if (exceptions[i].test(el[i])) {
return true;
}
}
return false;
},
_reLayout: function() {
if (!this.hasHorizontalScroll) {
this.maxScrollX = 0;
this.scrollerWidth = this.wrapperWidth;
}
if (!this.hasVerticalScroll) {
this.maxScrollY = 0;
this.scrollerHeight = this.wrapperHeight;
}
this.indicators.map(function(indicator) {
indicator.refresh();
});
//以防slider类嵌套使用
if (this.options.snap && typeof this.options.snap === 'string') {
var items = this.scroller.querySelectorAll(this.options.snap);
this.itemLength = 0;
this.snaps = [];
for (var i = 0, len = items.length; i < len; i++) {
var item = items[i];
if (item.parentNode === this.scroller) {
this.itemLength++;
this.snaps.push(item);
}
}
this._initSnap(); //需要每次都_initSnap么。其实init的时候执行一次后续resize的时候执行一次就行了吧.先这么做吧,如果影响性能,再调整
}
},
_momentum: function(current, distance, time, lowerMargin, wrapperSize, deceleration) {
var speed = parseFloat(Math.abs(distance) / time),
destination,
duration;
deceleration = deceleration === undefined ? 0.0006 : deceleration;
destination = current + (speed * speed) / (2 * deceleration) * (distance < 0 ? -1 : 1);
duration = speed / deceleration;
if (destination < lowerMargin) {
destination = wrapperSize ? lowerMargin - (wrapperSize / 2.5 * (speed / 8)) : lowerMargin;
distance = Math.abs(destination - current);
duration = distance / speed;
} else if (destination > 0) {
destination = wrapperSize ? wrapperSize / 2.5 * (speed / 8) : 0;
distance = Math.abs(current) + destination;
duration = distance / speed;
}
return {
destination: Math.round(destination),
duration: duration
};
},
_getTranslateStr: function(x, y) {
if (this.options.hardwareAccelerated) {
return 'translate3d(' + x + 'px,' + y + 'px,0px) ' + this.translateZ;
}
return 'translate(' + x + 'px,' + y + 'px) ';
},
//API
setStopped: function(stopped) {
this.stopped = !!stopped;
},
setTranslate: function(x, y) {
this.x = x;
this.y = y;
this.scrollerStyle['webkitTransform'] = this._getTranslateStr(x, y);
if (this.parallaxElement && this.options.scrollY) { //目前仅支持竖向视差效果
var parallaxY = y * this.options.parallaxRatio;
var scale = 1 + parallaxY / ((this.parallaxHeight - parallaxY) / 2);
if (scale > 1) {
this.parallaxImgStyle['opacity'] = 1 - parallaxY / 100 * this.options.parallaxRatio;
this.parallaxStyle['webkitTransform'] = this._getTranslateStr(0, -parallaxY) + ' scale(' + scale + ',' + scale + ')';
} else {
this.parallaxImgStyle['opacity'] = 1;
this.parallaxStyle['webkitTransform'] = this._getTranslateStr(0, -1) + ' scale(1,1)';
}
}
if (this.indicators) {
for (var i = this.indicators.length; i--;) {
this.indicators[i].updatePosition();
}
}
this.lastX = this.x;
this.lastY = this.y;
$.trigger(this.scroller, 'scroll', this);
},
reLayout: function() {
this.wrapper.offsetHeight;
var paddingLeft = parseFloat($.getStyles(this.wrapper, 'padding-left')) || 0;
var paddingRight = parseFloat($.getStyles(this.wrapper, 'padding-right')) || 0;
var paddingTop = parseFloat($.getStyles(this.wrapper, 'padding-top')) || 0;
var paddingBottom = parseFloat($.getStyles(this.wrapper, 'padding-bottom')) || 0;
var clientWidth = this.wrapper.clientWidth;
var clientHeight = this.wrapper.clientHeight;
this.scrollerWidth = this.scroller.offsetWidth;
this.scrollerHeight = this.scroller.offsetHeight;
this.wrapperWidth = clientWidth - paddingLeft - paddingRight;
this.wrapperHeight = clientHeight - paddingTop - paddingBottom;
this.maxScrollX = Math.min(this.wrapperWidth - this.scrollerWidth, 0);
this.maxScrollY = Math.min(this.wrapperHeight - this.scrollerHeight, 0);
this.hasHorizontalScroll = this.options.scrollX && this.maxScrollX < 0;
this.hasVerticalScroll = this.options.scrollY && this.maxScrollY < 0;
this._reLayout();
},
resetPosition: function(time) {
var x = this.x,
y = this.y;
time = time || 0;
if (!this.hasHorizontalScroll || this.x > 0) {
x = 0;
} else if (this.x < this.maxScrollX) {
x = this.maxScrollX;
}
if (!this.hasVerticalScroll || this.y > 0) {
y = 0;
} else if (this.y < this.maxScrollY) {
y = this.maxScrollY;
}
if (x == this.x && y == this.y) {
return false;
}
this.scrollTo(x, y, time, this.options.scrollEasing);
return true;
},
_reInit: function() {
var groups = this.wrapper.querySelectorAll('.' + CLASS_SCROLL);
for (var i = 0, len = groups.length; i < len; i++) {
if (groups[i].parentNode === this.wrapper) {
this.scroller = groups[i];
break;
}
}
this.scrollerStyle = this.scroller && this.scroller.style;
},
refresh: function() {
this._reInit();
this.reLayout();
$.trigger(this.scroller, 'refresh', this);
this.resetPosition();
},
scrollTo: function(x, y, time, easing) {
var easing = easing || ease.circular;
// this.isInTransition = time > 0 && (this.lastX != x || this.lastY != y);
//暂不严格判断x,y否则会导致部分版本上不正常触发轮播
this.isInTransition = time > 0;
if (this.isInTransition) {
this._clearRequestAnimationFrame();
this._transitionTimingFunction(easing.style);
this._transitionTime(time);
this.setTranslate(x, y);
} else {
this.setTranslate(x, y);
}
},
scrollToBottom: function(time, easing) {
time = time || this.options.scrollTime;
this.scrollTo(0, this.maxScrollY, time, easing);
},
gotoPage: function(index) {
this._gotoPage(index);
},
destroy: function() {
this._initEvent(true); //detach
delete $.data[this.wrapper.getAttribute('data-scroll')];
this.wrapper.setAttribute('data-scroll', '');
}
});
//Indicator
var Indicator = function(scroller, options) {
this.wrapper = typeof options.el == 'string' ? document.querySelector(options.el) : options.el;
this.wrapperStyle = this.wrapper.style;
this.indicator = this.wrapper.children[0];
this.indicatorStyle = this.indicator.style;
this.scroller = scroller;
this.options = $.extend({
listenX: true,
listenY: true,
fade: false,
speedRatioX: 0,
speedRatioY: 0
}, options);
this.sizeRatioX = 1;
this.sizeRatioY = 1;
this.maxPosX = 0;
this.maxPosY = 0;
if (this.options.fade) {
this.wrapperStyle['webkitTransform'] = this.scroller.translateZ;
this.wrapperStyle['webkitTransitionDuration'] = this.options.fixedBadAndorid && $.os.isBadAndroid ? '0.001s' : '0ms';
this.wrapperStyle.opacity = '0';
}
}
Indicator.prototype = {
handleEvent: function(e) {
},
transitionTime: function(time) {
time = time || 0;
this.indicatorStyle['webkitTransitionDuration'] = time + 'ms';
if (this.scroller.options.fixedBadAndorid && !time && $.os.isBadAndroid) {
this.indicatorStyle['webkitTransitionDuration'] = '0.001s';
}
},
transitionTimingFunction: function(easing) {
this.indicatorStyle['webkitTransitionTimingFunction'] = easing;
},
refresh: function() {
this.transitionTime();
if (this.options.listenX && !this.options.listenY) {
this.indicatorStyle.display = this.scroller.hasHorizontalScroll ? 'block' : 'none';
} else if (this.options.listenY && !this.options.listenX) {
this.indicatorStyle.display = this.scroller.hasVerticalScroll ? 'block' : 'none';
} else {
this.indicatorStyle.display = this.scroller.hasHorizontalScroll || this.scroller.hasVerticalScroll ? 'block' : 'none';
}
this.wrapper.offsetHeight; // force refresh
if (this.options.listenX) {
this.wrapperWidth = this.wrapper.clientWidth;
this.indicatorWidth = Math.max(Math.round(this.wrapperWidth * this.wrapperWidth / (this.scroller.scrollerWidth || this.wrapperWidth || 1)), 8);
this.indicatorStyle.width = this.indicatorWidth + 'px';
this.maxPosX = this.wrapperWidth - this.indicatorWidth;
this.minBoundaryX = 0;
this.maxBoundaryX = this.maxPosX;
this.sizeRatioX = this.options.speedRatioX || (this.scroller.maxScrollX && (this.maxPosX / this.scroller.maxScrollX));
}
if (this.options.listenY) {
this.wrapperHeight = this.wrapper.clientHeight;
this.indicatorHeight = Math.max(Math.round(this.wrapperHeight * this.wrapperHeight / (this.scroller.scrollerHeight || this.wrapperHeight || 1)), 8);
this.indicatorStyle.height = this.indicatorHeight + 'px';
this.maxPosY = this.wrapperHeight - this.indicatorHeight;
this.minBoundaryY = 0;
this.maxBoundaryY = this.maxPosY;
this.sizeRatioY = this.options.speedRatioY || (this.scroller.maxScrollY && (this.maxPosY / this.scroller.maxScrollY));
}
this.updatePosition();
},
updatePosition: function() {
var x = this.options.listenX && Math.round(this.sizeRatioX * this.scroller.x) || 0,
y = this.options.listenY && Math.round(this.sizeRatioY * this.scroller.y) || 0;
if (x < this.minBoundaryX) {
this.width = Math.max(this.indicatorWidth + x, 8);
this.indicatorStyle.width = this.width + 'px';
x = this.minBoundaryX;
} else if (x > this.maxBoundaryX) {
this.width = Math.max(this.indicatorWidth - (x - this.maxPosX), 8);
this.indicatorStyle.width = this.width + 'px';
x = this.maxPosX + this.indicatorWidth - this.width;
} else if (this.width != this.indicatorWidth) {
this.width = this.indicatorWidth;
this.indicatorStyle.width = this.width + 'px';
}
if (y < this.minBoundaryY) {
this.height = Math.max(this.indicatorHeight + y * 3, 8);
this.indicatorStyle.height = this.height + 'px';
y = this.minBoundaryY;
} else if (y > this.maxBoundaryY) {
this.height = Math.max(this.indicatorHeight - (y - this.maxPosY) * 3, 8);
this.indicatorStyle.height = this.height + 'px';
y = this.maxPosY + this.indicatorHeight - this.height;
} else if (this.height != this.indicatorHeight) {
this.height = this.indicatorHeight;
this.indicatorStyle.height = this.height + 'px';
}
this.x = x;
this.y = y;
this.indicatorStyle['webkitTransform'] = this.scroller._getTranslateStr(x, y);
},
fade: function(val, hold) {
if (hold && !this.visible) {
return;
}
clearTimeout(this.fadeTimeout);
this.fadeTimeout = null;
var time = val ? 250 : 500,
delay = val ? 0 : 300;
val = val ? '1' : '0';
this.wrapperStyle['webkitTransitionDuration'] = time + 'ms';
this.fadeTimeout = setTimeout((function(val) {
this.wrapperStyle.opacity = val;
this.visible = +val;
}).bind(this, val), delay);
}
};
$.Scroll = Scroll;
$.fn.scroll = function(options) {
var scrollApis = [];
this.each(function() {
var scrollApi = null;
var self = this;
var id = self.getAttribute('data-scroll');
if (!id) {
id = ++$.uuid;
var _options = $.extend({}, options);
if (self.classList.contains($.className('segmented-control'))) {
_options = $.extend(_options, {
scrollY: false,
scrollX: true,
indicators: false,
snap: $.classSelector('.control-item')
});
}
$.data[id] = scrollApi = new Scroll(self, _options);
self.setAttribute('data-scroll', id);
} else {
scrollApi = $.data[id];
}
scrollApis.push(scrollApi);
});
return scrollApis.length === 1 ? scrollApis[0] : scrollApis;
};
})(mui, window, document);