-
Notifications
You must be signed in to change notification settings - Fork 13
/
Copy pathfragmention.js
98 lines (80 loc) · 3.07 KB
/
fragmention.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
// detect native/existing fragmention support
if (!('fragmention' in window.location)) (function () {
// populate fragmention
location.fragmention = location.fragmention || '';
// return first element in scope containing case-sensitive text
function getElementsByText(scope, text) {
// iterate descendants of scope
for (var all = scope.childNodes, index = 0, element, list = []; (element = all[index]); ++index) {
// conditionally return element containing visible, whitespace-insensitive, case-sensitive text (a match)
if (element.nodeType === 1 && (element.innerText || element.textContent || '').replace(/\s+/g, ' ').indexOf(text) !== -1) {
list = list.concat(getElementsByText(element, text));
}
}
// return scope (no match)
return list.length ? list : scope;
}
function getAnchorableElementByName(fragment) {
var elements = document.getElementsByName(fragment), index = -1;
while (elements[++index] && !/^A(REA)?$/.test(elements[index].nodeName)) {}
return elements[index];
}
// on dom ready or hash change
function onHashChange() {
// do nothing if the dom is not ready
if (!/e/.test(document.readyState)) return;
// set location fragmention as uri-decoded text (from href, as hash may be decoded)
var
id = location.href.match(/#((?:#|%23)?)(.+)/) || [0,'',''],
node = document.getElementById(id[1]+id[2]) || getAnchorableElementByName(id[1]+id[2]),
match = decodeURIComponent(id[2].replace(/\+/g, ' ')).split(' ');
location.fragmention = match[0];
location.fragmentionIndex = parseFloat(match[1]) || 0;
// conditionally remove stashed element fragmention attribute
if (element) {
element.removeAttribute('fragmention');
// DEPRECATED: trigger style in IE8
if (element.runtimeStyle) {
element.runtimeStyle.windows = element.runtimeStyle.windows;
}
}
// if fragmention exists
if (!node && location.fragmention) {
var
// get all elements containing text (or document)
elements = getElementsByText(document, location.fragmention),
// get total number of elements
length = elements.length,
// get index of element
modulus = length && location.fragmentionIndex % length,
index = length && modulus >= 0 ? modulus : length + modulus;
// get element
element = length && elements[index];
// if element found
if (element) {
// scroll to element
element.scrollIntoView();
// set fragmention attribute
element.setAttribute('fragmention', '');
// DEPRECATED: trigger style in IE8
if (element.runtimeStyle) {
element.runtimeStyle.windows = element.runtimeStyle.windows;
}
}
// otherwise clear stashed element
else {
element = null;
}
}
}
var
// DEPRECATED: configure listeners
defaultListener = 'addEventListener',
addEventListener = defaultListener in window ? [defaultListener, ''] : ['attachEvent', 'on'],
// set stashed element
element;
// add listeners
window[addEventListener[0]](addEventListener[1] + 'hashchange', onHashChange);
document[addEventListener[0]](addEventListener[1] + 'readystatechange', onHashChange);
onHashChange();
})();