forked from filamentgroup/Overthrow
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathoverthrow.js
392 lines (335 loc) · 13.8 KB
/
overthrow.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
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
/*! Overthrow v.0.1.0. An overflow:auto polyfill for responsive design. (c) 2012: Scott Jehl, Filament Group, Inc. http://filamentgroup.github.com/Overthrow/license.txt */
(function (w, undefined) {
var doc = w.document,
docElem = doc.documentElement,
enabledClassName = "overthrow-enabled",
scrollIndicatorClassName = "overthrow",
// Touch events are used in the polyfill, and thus are a prerequisite
canBeFilledWithPoly = "ontouchmove" in doc,
// The following attempts to determine whether the browser has native overflow support
// so we can enable it but not polyfill
overflowProbablyAlreadyWorks =
// Features-first. iOS5 overflow scrolling property check - no UA needed here. thanks Apple :)
"WebkitOverflowScrolling" in docElem.style ||
// Touch events aren't supported and screen width is greater than X
// ...basically, this is a loose "desktop browser" check.
// It may wrongly opt-in very large tablets with no touch support.
( !canBeFilledWithPoly && w.screen.width > 1200 ) ||
// Hang on to your hats.
// Whitelist some popular, overflow-supporting mobile browsers for now and the future
// These browsers are known to get overlow support right, but give us no way of detecting it.
(function () {
var ua = w.navigator.userAgent,
// Webkit crosses platforms, and the browsers on our list run at least version 534
webkit = ua.match(/AppleWebKit\/([0-9]+)/),
wkversion = webkit && webkit[1],
wkLte534 = webkit && wkversion >= 534;
return (
/* Android 3+ with webkit gte 534
~: Mozilla/5.0 (Linux; U; Android 3.0; en-us; Xoom Build/HRI39) AppleWebKit/534.13 (KHTML, like Gecko) Version/4.0 Safari/534.13 */
ua.match(/Android ([0-9]+)/) && RegExp.$1 >= 3 && wkLte534 ||
/* Windows Phone
~: Mozilla/5.0 (compatible; MSIE 10.0; Windows Phone 8.0; Trident/6.0; IEMobile/10.0; ARM; Touch; NOKIA; Lumia 920) */
ua.match(/IEMobile\/(\d\d).\d+/) && RegExp.$1 >= 10 ||
/* Blackberry 7+ with webkit gte 534
~: Mozilla/5.0 (BlackBerry; U; BlackBerry 9900; en-US) AppleWebKit/534.11+ (KHTML, like Gecko) Version/7.0.0 Mobile Safari/534.11+ */
ua.match(/ Version\/([0-9]+)/) && RegExp.$1 >= 0 && w.blackberry && wkLte534 ||
/* Blackberry Playbook with webkit gte 534
~: Mozilla/5.0 (PlayBook; U; RIM Tablet OS 1.0.0; en-US) AppleWebKit/534.8+ (KHTML, like Gecko) Version/0.0.1 Safari/534.8+ */
ua.indexOf(/PlayBook/) > -1 && RegExp.$1 >= 0 && wkLte534 ||
/* Firefox Mobile (Fennec) 4 and up
~: Mozilla/5.0 (Mobile; rv:15.0) Gecko/15.0 Firefox/15.0 */
ua.match(/Firefox\/([0-9]+)/) && RegExp.$1 >= 4 ||
/* WebOS 3 and up (TouchPad too)
~: Mozilla/5.0 (hp-tablet; Linux; hpwOS/3.0.0; U; en-US) AppleWebKit/534.6 (KHTML, like Gecko) wOSBrowser/233.48 Safari/534.6 TouchPad/1.0 */
ua.match(/wOSBrowser\/([0-9]+)/) && RegExp.$1 >= 233 && wkLte534 ||
/* Nokia Browser N8
~: Mozilla/5.0 (Symbian/3; Series60/5.2 NokiaN8-00/012.002; Profile/MIDP-2.1 Configuration/CLDC-1.1 ) AppleWebKit/533.4 (KHTML, like Gecko) NokiaBrowser/7.3.0 Mobile Safari/533.4 3gpp-gba
~: Note: the N9 doesn't have native overflow with one-finger touch. wtf */
ua.match(/NokiaBrowser\/([0-9\.]+)/) && parseFloat(RegExp.$1) === 7.3 && webkit && wkversion >= 533
);
})(),
// Easing can use any of Robert Penner's equations (http://www.robertpenner.com/easing_terms_of_use.html). By default, overthrow includes ease-out-cubic
// arguments: t = current iteration, b = initial value, c = end value, d = total iterations
// use w.overthrow.easing to provide a custom function externally, or pass an easing function as a callback to the toss method
defaultEasing = function (t, b, c, d) {
return c * ((t = t / d - 1) * t * t + 1) + b;
},
enabled = false,
// Keeper of intervals
timeKeeper,
//can be used to configure custom names for the css classes
configure = function (config) {
enabledClassName = config.enabledClassName || enabledClassName;
scrollIndicatorClassName = config.scrollIndicatorClassName || scrollIndicatorClassName;
},
/* toss scrolls and element with easing
// elem is the element to scroll
// options hash:
* left is the desired horizontal scroll. Default is "+0". For relative distances, pass a string with "+" or "-" in front.
* top is the desired vertical scroll. Default is "+0". For relative distances, pass a string with "+" or "-" in front.
* duration is the number of milliseconds the throw will take. Default is 100.
* easing is an optional custom easing function. Default is w.overthrow.easing. Must follow the easing function signature
*/
toss = function (elem, options) {
var i = 0,
sLeft = elem.scrollLeft,
sTop = elem.scrollTop,
// Toss defaults
o = {
top: "+0",
left: "+0",
duration: 100,
easing: w.overthrow.easing
},
endLeft, endTop;
// Mixin based on predefined defaults
if (options) {
for (var j in o) {
if (options[ j ] !== undefined) {
o[ j ] = options[ j ];
}
}
}
// Convert relative values to ints
// First the left val
if (typeof o.left === "string") {
o.left = parseFloat(o.left);
endLeft = o.left + sLeft;
}
else {
endLeft = o.left;
o.left = o.left - sLeft;
}
// Then the top val
if (typeof o.top === "string") {
o.top = parseFloat(o.top);
endTop = o.top + sTop;
}
else {
endTop = o.top;
o.top = o.top - sTop;
}
timeKeeper = setInterval(function () {
if (i++ < o.duration) {
elem.scrollLeft = o.easing(i, sLeft, o.left, o.duration);
elem.scrollTop = o.easing(i, sTop, o.top, o.duration);
}
else {
if (endLeft !== elem.scrollLeft) {
elem.scrollLeft = endLeft;
}
if (endTop !== elem.scrollTop) {
elem.scrollTop = endTop;
}
intercept();
}
}, 1);
// Return the values, post-mixin, with end values specified
return { top: endTop, left: endLeft, duration: o.duration, easing: o.easing };
},
// find closest overthrow (elem or a parent)
closest = function (target, ascend) {
if (!target) {
return undefined;
}
return !ascend &&
target.className &&
target.className.indexOf(scrollIndicatorClassName) > -1 &&
target.className.indexOf(enabledClassName) === -1 &&
target || closest(target.parentNode);
},
// Intercept any throw in progress
intercept = function () {
clearInterval(timeKeeper);
},
// Enable and potentially polyfill overflow
enable = function () {
// If it's on,
if (enabled) {
return;
}
// It's on.
enabled = true;
// If overflowProbablyAlreadyWorks or at least the element canBeFilledWithPoly, add a class to cue CSS that assumes overflow scrolling will work (setting height on elements and such)
if (overflowProbablyAlreadyWorks || canBeFilledWithPoly) {
docElem.className += " " + enabledClassName;
}
// Destroy everything later. If you want to.
w.overthrow.forget = function () {
// Strip the class name from docElem
docElem.className = docElem.className.replace(enabledClassName, "");
// Remove touch binding (check for method support since this part isn't qualified by touch support like the rest)
if (doc.removeEventListener) {
doc.removeEventListener("touchstart", start, false);
}
// reset easing to default
w.overthrow.easing = defaultEasing;
// Let 'em know
enabled = false;
};
// If overflowProbablyAlreadyWorks or it doesn't look like the browser canBeFilledWithPoly, our job is done here. Exit viewport left.
if (overflowProbablyAlreadyWorks || !canBeFilledWithPoly) {
return;
}
// Fill 'er up!
// From here down, all logic is associated with touch scroll handling
// elem references the overthrow element in use
var elem,
// The last several Y values are kept here
lastTops = [],
// The last several X values are kept here
lastLefts = [],
// lastDown will be true if the last scroll direction was down, false if it was up
lastDown,
// lastRight will be true if the last scroll direction was right, false if it was left
lastRight,
// For a new gesture, or change in direction, reset the values from last scroll
resetVertTracking = function () {
lastTops = [];
lastDown = null;
},
resetHorTracking = function () {
lastLefts = [];
lastRight = null;
},
// After releasing touchend, throw the overthrow element, depending on momentum
finishScroll = function () {
// Come up with a distance and duration based on how
// Multipliers are tweaked to a comfortable balance across platforms
var top = ( lastTops[ 0 ] - lastTops[ lastTops.length - 1 ] ) * 8,
left = ( lastLefts[ 0 ] - lastLefts[ lastLefts.length - 1 ] ) * 8,
duration = Math.max(Math.abs(left), Math.abs(top)) / 8;
// Make top and left relative-style strings (positive vals need "+" prefix)
top = ( top > 0 ? "+" : "" ) + top;
left = ( left > 0 ? "+" : "" ) + left;
// Make sure there's a significant amount of throw involved, otherwise, just stay still
if (!isNaN(duration) && duration > 0 && ( Math.abs(left) > 80 || Math.abs(top) > 80 )) {
toss(elem, { left: left, top: top, duration: duration });
}
},
// On webkit, touch events hardly trickle through textareas and inputs
// Disabling CSS pointer events makes sure they do, but it also makes the controls innaccessible
// Toggling pointer events at the right moments seems to do the trick
// Thanks Thomas Bachem http://stackoverflow.com/a/5798681 for the following
inputs,
setPointers = function (val) {
inputs = elem.querySelectorAll("textarea, input");
for (var i = 0, il = inputs.length; i < il; i++) {
inputs[ i ].style.pointerEvents = val;
}
},
// For nested overthrows, changeScrollTarget restarts a touch event cycle on a parent or child overthrow
changeScrollTarget = function (startEvent, ascend, e) {
if (doc.createEvent) {
var newTarget = ( !ascend || ascend === undefined ) && elem.parentNode || elem.touchchild || elem,
tEnd;
if (newTarget !== elem) {
tEnd = doc.createEvent("HTMLEvents");
// borrowing last touchmove touches so tEnd event behaves more or less like a real touch event
tEnd.touches = e.touches;
tEnd.initEvent("touchend", true, true);
elem.dispatchEvent(tEnd);
newTarget.touchchild = elem;
elem = newTarget;
newTarget.dispatchEvent(startEvent);
}
}
},
// Touchstart handler
// On touchstart, touchmove and touchend are freshly bound, and all three share a bunch of vars set by touchstart
// Touchend unbinds them again, until next time
start = function (e) {
// Stop any throw in progress
intercept();
// Reset the distance and direction tracking
resetVertTracking();
resetHorTracking();
elem = closest(e.target);
if (!elem || elem === docElem || e.touches.length > 1) {
return;
}
setPointers("none");
var touchStartE = e,
scrollT = elem.scrollTop,
scrollL = elem.scrollLeft,
height = elem.offsetHeight,
width = elem.offsetWidth,
startY = e.touches[ 0 ].pageY,
startX = e.touches[ 0 ].pageX,
scrollHeight = elem.scrollHeight,
scrollWidth = elem.scrollWidth,
// Touchmove handler
move = function (e) {
var ty = scrollT + startY - e.touches[ 0 ].pageY,
tx = scrollL + startX - e.touches[ 0 ].pageX,
down = ty >= ( lastTops.length ? lastTops[ 0 ] : 0 ),
right = tx >= ( lastLefts.length ? lastLefts[ 0 ] : 0 );
// If down and lastDown are inequal, the y scroll has changed direction. Reset tracking.
if (lastDown && down !== lastDown) {
resetVertTracking();
}
// If right and lastRight are inequal, the x scroll has changed direction. Reset tracking.
if (lastRight && right !== lastRight) {
resetHorTracking();
}
// remember the last direction in which we were headed
lastDown = down;
lastRight = right;
// set the container's scroll
elem.scrollTop = ty;
elem.scrollLeft = tx;
lastTops.unshift(ty);
lastLefts.unshift(tx);
if (lastTops.length > 3) {
lastTops.pop();
}
if (lastLefts.length > 3) {
lastLefts.pop();
}
// If there's room to scroll the current container, prevent the default window scroll
if (( ty > 0 && ty < scrollHeight - height ) || ( tx > 0 && tx < scrollWidth - width )) {
e.preventDefault();
}
// Has no parent element to scroll, maintain the target to allow inversion of scroll direction
else if (!closest(elem, true)) {
e.preventDefault();
}
// This bubbling is dumb. Needs a rethink.
else {
changeScrollTarget(touchStartE, undefined, e);
}
},
// Touchend handler
end = function (e) {
// Apply momentum based easing for a graceful finish
finishScroll();
// Bring the pointers back
setPointers("auto");
setTimeout(function () {
setPointers("none");
}, 450);
elem.removeEventListener("touchmove", move, false);
elem.removeEventListener("touchend", end, false);
};
elem.addEventListener("touchmove", move, false);
elem.addEventListener("touchend", end, false);
};
// Bind to touch, handle move and end within
doc.addEventListener("touchstart", start, false);
};
// Expose overthrow API
w.overthrow = {
configure: configure,
set: enable,
forget: function () {
},
easing: defaultEasing,
toss: toss,
intercept: intercept,
closest: closest,
support: overflowProbablyAlreadyWorks ? "native" : canBeFilledWithPoly && "polyfilled" || "none"
};
// Auto-init
enable();
})(this);