');
this.$wrap = this.$container.parent();
this.$wrap.height(this.$element.height());
if (position !== 'static') {
this.$wrap.css('position', position);
} else {
this.$wrap.css('position', 'relative');
}
}
if (this.options.contentSelector) {
this.$content = this.$container.find(this.options.contentSelector);
} else {
this.$content = this.$container.wrap('
');
this.$container = this.$content.parent();
}
switch (this.options.direction) {
case 'vertical': {
this.vertical = true;
break;
}
case 'horizontal': {
this.horizontal = true;
break;
}
case 'both': {
this.horizontal = true;
this.vertical = true;
break;
}
case 'auto': {
let overflowX = this.$element.css('overflow-x'),
overflowY = this.$element.css('overflow-y');
if (overflowX === 'scroll' || overflowX === 'auto') {
this.horizontal = true;
}
if (overflowY === 'scroll' || overflowY === 'auto') {
this.vertical = true;
}
break;
}
default: {
break;
}
}
if (!this.vertical && !this.horizontal) {
return;
}
this.$wrap.addClass(this.classes.wrap);
this.$container.addClass(this.classes.container);
this.$content.addClass(this.classes.content);
if (this.options.skin) {
this.$wrap.addClass(this.classes.skin);
}
this.$wrap.addClass(this.options.enabledClass);
if (this.vertical) {
this.$wrap.addClass(`${this.classes.wrap}-vertical`);
this.initLayout('vertical');
this.createBar('vertical');
}
if (this.horizontal) {
this.$wrap.addClass(`${this.classes.wrap}-horizontal`);
this.initLayout('horizontal');
this.createBar('horizontal');
}
this.bindEvents();
this.trigger('ready');
}
bindEvents() {
if (this.options.responsive) {
$(window).on(this.eventNameWithId('orientationchange'), () => {
this.update();
});
$(window).on(this.eventNameWithId('resize'), this.throttle(() => {
this.update();
}, this.options.throttle));
}
if (!this.horizontal && !this.vertical) {
return;
}
let that = this;
this.$wrap.on(this.eventName('mouseenter'), () => {
that.$wrap.addClass(this.options.hoveringClass);
that.enter('hovering');
that.trigger('hover');
});
this.$wrap.on(this.eventName('mouseleave'), () => {
that.$wrap.removeClass(this.options.hoveringClass);
if (!that.is('hovering')) {
return;
}
that.leave('hovering');
that.trigger('hovered');
});
if (this.options.showOnHover) {
if (this.options.showOnBarHover) {
this.$bar.on('asScrollbar::hover', () => {
if(that.horizontal){
that.showBar('horizontal');
}
if(that.vertical){
that.showBar('vertical');
}
}).on('asScrollbar::hovered', () => {
if(that.horizontal){
that.hideBar('horizontal');
}
if(that.vertical){
that.hideBar('vertical');
}
});
} else {
this.$element.on(`${NAMESPACE$1}::hover`, $.proxy(this.showBar, this));
this.$element.on(`${NAMESPACE$1}::hovered`, $.proxy(this.hideBar, this));
}
}
this.$container.on(this.eventName('scroll'), () => {
if (that.horizontal) {
let oldLeft = that.offsetLeft;
that.offsetLeft = that.getOffset('horizontal');
if (oldLeft !== that.offsetLeft) {
that.trigger('scroll', that.getPercentOffset('horizontal'), 'horizontal');
if (that.offsetLeft === 0) {
that.trigger('scrolltop', 'horizontal');
}
if (that.offsetLeft === that.getScrollLength('horizontal')) {
that.trigger('scrollend', 'horizontal');
}
}
}
if (that.vertical) {
let oldTop = that.offsetTop;
that.offsetTop = that.getOffset('vertical');
if (oldTop !== that.offsetTop) {
that.trigger('scroll', that.getPercentOffset('vertical'), 'vertical');
if (that.offsetTop === 0) {
that.trigger('scrolltop', 'vertical');
}
if (that.offsetTop === that.getScrollLength('vertical')) {
that.trigger('scrollend', 'vertical');
}
}
}
});
this.$element.on(`${NAMESPACE$1}::scroll`, (e, api, value, direction) => {
if (!that.is('scrolling')) {
that.enter('scrolling');
that.$wrap.addClass(that.options.scrollingClass);
}
let bar = api.getBarApi(direction);
bar.moveTo(conventToPercentage(value), false, true);
clearTimeout(that._timeoutId);
that._timeoutId = setTimeout(() => {
that.$wrap.removeClass(that.options.scrollingClass);
that.leave('scrolling');
}, 200);
});
this.$bar.on('asScrollbar::change', (e, api, value) => {
if(typeof e.target.direction === 'string') {
that.scrollTo(e.target.direction, conventToPercentage(value), false, true);
}
});
this.$bar.on('asScrollbar::drag', () => {
that.$wrap.addClass(that.options.draggingClass);
}).on('asScrollbar::dragged', () => {
that.$wrap.removeClass(that.options.draggingClass);
});
}
unbindEvents() {
this.$wrap.off(this.eventName());
this.$element.off(`${NAMESPACE$1}::scroll`).off(`${NAMESPACE$1}::hover`).off(`${NAMESPACE$1}::hovered`);
this.$container.off(this.eventName());
$(window).off(this.eventNameWithId());
}
initLayout(direction) {
if (direction === 'vertical') {
this.$container.css('height', this.$wrap.height());
}
let attributes = this.attributes[direction],
container = this.$container[0];
// this.$container.css(attributes.overflow, 'scroll');
let parentLength = container.parentNode[attributes.crossClientLength],
scrollbarWidth = this.getBrowserScrollbarWidth(direction);
this.$content.css(attributes.crossLength, `${parentLength}px`);
this.$container.css(attributes.crossLength, `${scrollbarWidth + parentLength}px`);
if (scrollbarWidth === 0 && isFFLionScrollbar) {
this.$container.css(attributes.ffPadding, 16);
}
}
createBar(direction) {
let options = $.extend(this.options.scrollbar, {
namespace: this.classes.bar,
direction: direction,
useCssTransitions: false,
keyboard: false
});
let $bar = $('
');
$bar.asScrollbar(options);
if (this.options.showOnHover) {
$bar.addClass(this.classes.barHide);
}
$bar.appendTo(this.$wrap);
this[`$${direction}`] = $bar;
if (this.$bar === null) {
this.$bar = $bar;
} else {
this.$bar = this.$bar.add($bar);
}
this.updateBarHandle(direction);
}
trigger(eventType, ...params) {
const data = [this].concat(params);
// event
this.$element.trigger(`${NAMESPACE$1}::${eventType}`, data);
// callback
eventType = eventType.replace(/\b\w+\b/g, (word) => {
return word.substring(0, 1).toUpperCase() + word.substring(1);
});
const onFunction = `on${eventType}`;
if (typeof this.options[onFunction] === 'function') {
this.options[onFunction].apply(this, params);
}
}
/**
* Checks whether the carousel is in a specific state or not.
*/
is(state) {
return this._states[state] && this._states[state] > 0;
}
/**
* Enters a state.
*/
enter(state) {
if (this._states[state] === undefined) {
this._states[state] = 0;
}
// this._states[state]++;
this._states[state] = 1;
}
/**
* Leaves a state.
*/
leave(state) {
// this._states[state]--;
this._states[state] = -1;
}
eventName(events) {
if (typeof events !== 'string' || events === '') {
return `.${this.options.namespace}`;
}
events = events.split(' ');
let length = events.length;
for (let i = 0; i < length; i++) {
events[i] = `${events[i]}.${this.options.namespace}`;
}
return events.join(' ');
}
eventNameWithId(events) {
if (typeof events !== 'string' || events === '') {
return `.${this.options.namespace}-${this.instanceId}`;
}
events = events.split(' ');
let length = events.length;
for (let i = 0; i < length; i++) {
events[i] = `${events[i]}.${this.options.namespace}-${this.instanceId}`;
}
return events.join(' ');
}
/**
* _throttle
* @description Borrowed from Underscore.js
*/
throttle(func, wait) {
const _now = Date.now || function() {
return new Date().getTime();
};
let timeout;
let context;
let args;
let result;
let previous = 0;
let later = function() {
previous = _now();
timeout = null;
result = func.apply(context, args);
if (!timeout) {
context = args = null;
}
};
return (...params) => {
/*eslint consistent-this: "off"*/
let now = _now();
let remaining = wait - (now - previous);
context = this;
args = params;
if (remaining <= 0 || remaining > wait) {
if (timeout) {
clearTimeout(timeout);
timeout = null;
}
previous = now;
result = func.apply(context, args);
if (!timeout) {
context = args = null;
}
} else if (!timeout) {
timeout = setTimeout(later, remaining);
}
return result;
};
}
getBrowserScrollbarWidth(direction) {
let attributes = this.attributes[direction],
outer, outerStyle;
if (attributes.scrollbarWidth) {
return attributes.scrollbarWidth;
}
outer = document.createElement('div');
outerStyle = outer.style;
outerStyle.position = 'absolute';
outerStyle.width = '100px';
outerStyle.height = '100px';
outerStyle.overflow = 'scroll';
outerStyle.top = '-9999px';
document.body.appendChild(outer);
attributes.scrollbarWidth = outer[attributes.offset] - outer[attributes.clientLength];
document.body.removeChild(outer);
return attributes.scrollbarWidth;
}
getOffset(direction) {
let attributes = this.attributes[direction],
container = this.$container[0];
return (container[attributes.pageOffset] || container[attributes.scroll]);
}
getPercentOffset(direction) {
return this.getOffset(direction) / this.getScrollLength(direction);
}
getContainerLength(direction) {
return this.$container[0][this.attributes[direction].clientLength];
}
getScrollLength(direction) {
let scrollLength = this.$content[0][this.attributes[direction].scrollLength];
return scrollLength - this.getContainerLength(direction);
}
scrollTo(direction, value, trigger, sync) {
let type = typeof value;
if (type === 'string') {
if (isPercentage(value)) {
value = convertPercentageToFloat(value) * this.getScrollLength(direction);
}
value = parseFloat(value);
type = 'number';
}
if (type !== 'number') {
return;
}
this.move(direction, value, trigger, sync);
}
scrollBy(direction, value, trigger, sync) {
let type = typeof value;
if (type === 'string') {
if (isPercentage(value)) {
value = convertPercentageToFloat(value) * this.getScrollLength(direction);
}
value = parseFloat(value);
type = 'number';
}
if (type !== 'number') {
return;
}
this.move(direction, this.getOffset(direction) + value, trigger, sync);
}
move(direction, value, trigger, sync) {
if (this[direction] !== true || typeof value !== 'number') {
return;
}
this.enter('moving');
if (value < 0) {
value = 0;
} else if (value > this.getScrollLength(direction)) {
value = this.getScrollLength(direction);
}
let attributes = this.attributes[direction];
var that = this;
let callback = () => {
that.leave('moving');
};
if (sync) {
this.$container[0][attributes.scroll] = value;
if (trigger !== false) {
this.trigger('change', value / this.getScrollLength(direction), direction);
}
callback();
} else {
this.enter('animating');
let startTime = getTime();
let start = this.getOffset(direction);
let end = value;
let run = (time) => {
let percent = (time - startTime) / that.options.duration;
if (percent > 1) {
percent = 1;
}
percent = that.easing.fn(percent);
let current = parseFloat(start + percent * (end - start), 10);
that.$container[0][attributes.scroll] = current;
if (trigger !== false) {
that.trigger('change', value / that.getScrollLength(direction), direction);
}
if (percent === 1) {
window.cancelAnimationFrame(that._frameId);
that._frameId = null;
that.leave('animating');
callback();
} else {
that._frameId = window.requestAnimationFrame(run);
}
};
this._frameId = window.requestAnimationFrame(run);
}
}
scrollXto(value, trigger, sync) {
return this.scrollTo('horizontal', value, trigger, sync);
}
scrollYto(value, trigger, sync) {
return this.scrollTo('vertical', value, trigger, sync);
}
scrollXby(value, trigger, sync) {
return this.scrollBy('horizontal', value, trigger, sync);
}
scrollYby(value, trigger, sync) {
return this.scrollBy('vertical', value, trigger, sync);
}
getBar(direction) {
if (direction && this[`$${direction}`]) {
return this[`$${direction}`];
}
return this.$bar;
}
getBarApi(direction) {
return this.getBar(direction).data('asScrollbar');
}
getBarX() {
return this.getBar('horizontal');
}
getBarY() {
return this.getBar('vertical');
}
showBar(direction) {
this.getBar(direction).removeClass(this.classes.barHide);
}
hideBar(direction) {
this.getBar(direction).addClass(this.classes.barHide);
}
updateBarHandle(direction) {
let api = this.getBarApi(direction);
if (!api) {
return;
}
let containerLength = this.getContainerLength(direction),
scrollLength = this.getScrollLength(direction);
if (scrollLength > 0) {
if (api.is('disabled')) {
api.enable();
}
api.setHandleLength(api.getBarLength() * containerLength / (scrollLength + containerLength), true);
} else {
api.disable();
}
}
disable() {
if (!this.is('disabled')) {
this.enter('disabled');
this.$wrap.addClass(this.options.disabledClass).removeClass(this.options.enabledClass);
this.unbindEvents();
this.unStyle();
}
this.trigger('disable');
}
enable() {
if (this.is('disabled')) {
this.leave('disabled');
this.$wrap.addClass(this.options.enabledClass).removeClass(this.options.disabledClass);
this.bindEvents();
this.update();
}
this.trigger('enable');
}
update() {
if (this.is('disabled')) {
return;
}
if(this.$element.is(':visible')) {
if (this.vertical) {
this.initLayout('vertical');
this.updateBarHandle('vertical');
}
if (this.horizontal) {
this.initLayout('horizontal');
this.updateBarHandle('horizontal');
}
}
}
unStyle() {
if (this.horizontal) {
this.$container.css({
height: '',
'padding-bottom': ''
});
this.$content.css({
height: ''
});
}
if (this.vertical) {
this.$container.css({
width: '',
height: '',
'padding-right': ''
});
this.$content.css({
width: ''
});
}
if (!this.options.containerSelector) {
this.$wrap.css({
height: ''
});
}
}
destroy() {
this.$wrap.removeClass(`${this.classes.wrap}-vertical`)
.removeClass(`${this.classes.wrap}-horizontal`)
.removeClass(this.classes.wrap)
.removeClass(this.options.enabledClass)
.removeClass(this.classes.disabledClass);
this.unStyle();
if (this.$bar) {
this.$bar.remove();
}
this.unbindEvents();
if (this.options.containerSelector) {
this.$container.removeClass(this.classes.container);
} else {
this.$container.unwrap();
}
if (!this.options.contentSelector) {
this.$content.unwrap();
}
this.$content.removeClass(this.classes.content);
this.$element.data(NAMESPACE$1, null);
this.trigger('destroy');
}
static setDefaults(options) {
$.extend(DEFAULTS, $.isPlainObject(options) && options);
}
}
var info = {
version:'0.4.10'
};
const NAMESPACE = 'asScrollable';
const OtherAsScrollable = $.fn.asScrollable;
const jQueryAsScrollable = function(options, ...args) {
if (typeof options === 'string') {
let method = options;
if (/^_/.test(method)) {
return false;
} else if ((/^(get)/.test(method))) {
let instance = this.first().data(NAMESPACE);
if (instance && typeof instance[method] === 'function') {
return instance[method](...args);
}
} else {
return this.each(function() {
let instance = $.data(this, NAMESPACE);
if (instance && typeof instance[method] === 'function') {
instance[method](...args);
}
});
}
}
return this.each(function() {
if (!$(this).data(NAMESPACE)) {
$(this).data(NAMESPACE, new AsScrollable(this, options));
}
});
};
$.fn.asScrollable = jQueryAsScrollable;
$.asScrollable = $.extend({
setDefaults: AsScrollable.setDefaults,
noConflict: function() {
$.fn.asScrollable = OtherAsScrollable;
return jQueryAsScrollable;
}
}, info);