-
Notifications
You must be signed in to change notification settings - Fork 39
/
scrollable.js
188 lines (176 loc) · 6.32 KB
/
scrollable.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
;
(function (angular, $, undefined) {
'use strict';
if ($.fn['nanoScroller'] === undefined) {
throw new Error("nanoScrollerJS is not defined in jQuery");
}
//jQuery must be used, cause angular method 'find' has different behavior
if (angular.element !== $) {
throw new Error("Angular must use jQuery not jqLite");
}
var AS_ELEMENT = 1, AS_ATTRIBUTE = 0;
/**
* Wrapper of nanoScrollerJS
* @name sun.scrollable
*/
var module = angular.module('sun.scrollable', ['ng']);
/**
* Configuration for the directive
* @name scrollableConfig
* @param scrollableConfig.template Template of the scroller
* @param scrollableConfig.bottomMargin Available number of pixels from the bottom,
* in which it is considered that scroller is in bottom
*/
module.constant("scrollableConfig", {
template : '<div class="{nanoClass}"><div class="{contentClass}" ng-transclude></div></div>',
bottomMargin: 40
});
/**
* Default configuration of the nanoScroller
* @name nanoScrollerDefaults
*/
module.constant("nanoScrollerDefaults", {
nanoClass : 'nano',
contentClass: 'nano-content'
});
module.directive("scrollable", createScrollableDirective(AS_ELEMENT));
module.directive("scrollable", createScrollableDirective(AS_ATTRIBUTE));
/**
* Create directive 'scrollable' specified by variable `type`.
* If type equals to `AS_ELEMENT` directive will replace element, otherwise template will be appended.
* @param type Type of the directive: `AS_ELEMENT` or `AS_ATTRIBUTE`
* @returns {function} Directive creation function
*/
function createScrollableDirective(type) {
var directive = function ($timeout, scrollableConfig, nanoScrollerDefaults) {
return {
transclude: true,
replace : type === AS_ELEMENT,
restrict : type === AS_ELEMENT ? 'E' : 'A',
priority : 1000,
template : format(scrollableConfig.template, nanoScrollerDefaults),
link : function (scope, element, attr) {
var oldHeight,
contentClass = nanoScrollerDefaults.contentClass,
nanoClass = nanoScrollerDefaults.nanoClass,
contentElement = element.find('.' + contentClass)[0],
parentElement = contentElement.parentElement,
$nanoElement = element.hasClass(nanoClass) ? element : element.find('.' + nanoClass),
options = angular.extend({}, nanoScrollerDefaults, convertStringToValue(attr), scope.$eval(attr['scrollable']));
function listener(newHeight, oldHeight) {
// If this is first run, create nanoScroller
if (newHeight === oldHeight) {
// First run must be async
scope.$evalAsync(function () {
$nanoElement.nanoScroller(options);
$nanoElement.nanoScroller();
});
}
//If scroller was on the bottom, scroll to bottom
else if (newHeight !== oldHeight && contentElement.scrollTop &&
(oldHeight - contentElement.scrollTop - parentElement.clientHeight) < scrollableConfig.bottomMargin) {
scope.$evalAsync(function () {
// To make right calculation scroller must be reseted
// See https://github.com/maxaon/angular-nanoscroller/issues/4
$nanoElement.nanoScroller();
$nanoElement.nanoScroller({scroll: 'bottom'});
});
}
// Otherwise just update the pane
else {
$nanoElement.nanoScroller();
}
}
function collectionListener() {
var newHeight = contentElement.scrollHeight;
if (oldHeight === undefined) {
oldHeight = newHeight;
}
listener(newHeight, oldHeight);
}
if (attr['static']) {
// Call scroller after transclusion
listener();
}
else if (typeof attr['watch'] === 'string' || attr['watchCollection']) {
angular.forEach(splitter(attr['watch']), function (name) {
scope.$watch(name, collectionListener);
});
angular.forEach(splitter(attr['watchCollection']), function (name) {
scope.$watchCollection(name, collectionListener);
});
}
// If no watchers are supplied fall back to content element height check
else {
// http://jsperf.com/angular-watch-collection-vs-element-scroll-height
// Call nanoScroller, when height of content is changed
scope.$watch(function () {
return contentElement.scrollHeight;
},
listener);
}
scope.$on("$destroy", function () {
$nanoElement.nanoScroller({ destroy: true });
$nanoElement = contentElement = parentElement = null;
});
}
};
};
directive.$inject = ['$timeout', 'scrollableConfig', 'nanoScrollerDefaults'];
return directive;
}
/**
* Convert element attributes from stings to objects.
* Also filter attribute (starting from '$' and class)
* @param attr Attribute array
* @returns {{}} Filtered object of converted attributes
*/
function convertStringToValue(attr) {
var result = {};
angular.forEach(attr, function (value, key) {
if (key.indexOf("$") === 0) {
return;
}
if (key == 'iosNativeScrolling') {
key = 'iOSNativeScrolling';
}
switch (key) {
case "true":
value = true;
break;
case "false":
value = false;
break;
case "null":
value = null;
break;
case "class":
return;
}
result[key] = value;
});
return result;
}
/**
* Split text in attribute by `,` or `;`
* @param val Attribute value
* @returns {Array} Array of name
*/
function splitter(val) {
if (!val) {
return [];
}
return val.replace(",", ";").split(";");
}
/**
* Format `str` in python way (only dictionary)
* @param str Input string
* @param params Replacing variables
* @returns {String}
*/
function format(str, params) {
return str.replace(new RegExp("{.*?}", "g"), function (variable) {
return params[variable.slice(1, -1)] || "";
});
}
}(angular, jQuery));