-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathLazyloadfier Userscript.user.js
148 lines (133 loc) · 5.51 KB
/
Lazyloadfier Userscript.user.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
// ==UserScript==
// @name Lazyloadfier Userscript
// @namespace Lazyloadfier Userscript - Violentmonkey Scripts
// @match *://*/*
// @grant none
// @version 1.0
// @run-at document-start
// @author -
// ==/UserScript==
/* global document, addEventListener, MutationObserver, IntersectionObserver, innerHeight, innerWidth, setTimeout, clearTimeout */
const SRC_ATTRIBUTE_NAME = "src";
const SRCSET_ATTRIBUTE_NAME = "srcset";
const POSTER_ATTRIBUTE_NAME = "poster";
const LOADING_ATTRIBUTE_NAME = "loading";
const LAZY_LOADING_ATTRIBUTE_VALUE = "lazy";
const IMG_TAG_NAME = "IMG";
const VIDEO_TAG_NAME = "VIDEO";
const AUDIO_TAG_NAME = "AUDIO";
const SOURCE_TAG_NAME = "SOURCE";
const IFRAME_TAG_NAME = "IFRAME";
const FRAME_TAG_NAME = "FRAME";
const EMBED_TAG_NAME = "EMBED";
const TAG_NAMES_WITH_SRC_ATTRIBUTE = new Set([IMG_TAG_NAME, VIDEO_TAG_NAME, AUDIO_TAG_NAME, SOURCE_TAG_NAME, IFRAME_TAG_NAME, FRAME_TAG_NAME, EMBED_TAG_NAME]);
const TAG_NAMES_WITH_SRCSET_ATTRIBUTE = new Set([IMG_TAG_NAME, SOURCE_TAG_NAME]);
const TAG_NAMES_WITH_POSTER_ATTRIBUTE = new Set([VIDEO_TAG_NAME]);
const OBSERVED_TAGS_SELECTOR = Array.from(TAG_NAMES_WITH_SRC_ATTRIBUTE).join(",");
const UNSENT_READY_STATE = 0;
const DOM_CONTENT_LOADED_EVENT = "DOMContentLoaded";
const EMPTY_DEFAULT_DATA_URI = "data:,";
const EMPTY_IMAGE_DATA_URI = "";
const EMPTY_TEXT_DATA_URI = "data:text/plain,";
const EMPTY_DATA_URI = new Map([
[VIDEO_TAG_NAME, EMPTY_DEFAULT_DATA_URI],
[AUDIO_TAG_NAME, EMPTY_DEFAULT_DATA_URI],
[SOURCE_TAG_NAME, EMPTY_IMAGE_DATA_URI],
[IMG_TAG_NAME, EMPTY_IMAGE_DATA_URI],
[IFRAME_TAG_NAME, EMPTY_TEXT_DATA_URI],
[FRAME_TAG_NAME, EMPTY_TEXT_DATA_URI],
[EMBED_TAG_NAME, EMPTY_TEXT_DATA_URI]
]);
const MUTATION_OBSERVER_OPTIONS = { childList: true, subtree: true };
const MINIMUM_INTERSECTION_RATIO = 0;
const MUTATION_OBSERVER_TIMEOUT = 2500;
observeDocumentMutations();
function observeDocumentMutations() {
const disconnectObserverTimeout = {};
const mutationObserver = new MutationObserver(mutationsList => mutationObserverCallback(mutationsList, callDeferDisconnectObserver));
mutationObserver.observe(document, MUTATION_OBSERVER_OPTIONS);
addEventListener(DOM_CONTENT_LOADED_EVENT, callDeferDisconnectObserver);
function callDeferDisconnectObserver() {
deferDisconnectObserver(mutationObserver, disconnectObserverTimeout);
}
}
function deferDisconnectObserver(mutationObserver, disconnectObserverTimeout) {
if (disconnectObserverTimeout.id) {
clearTimeout(disconnectObserverTimeout.id);
}
disconnectObserverTimeout.id = setTimeout(() => mutationObserver.disconnect(), MUTATION_OBSERVER_TIMEOUT);
}
function mutationObserverCallback(mutationsList, onProgressCallback) {
const observedNodes = getObservedNodes(mutationsList);
if (observedNodes.length) {
observedNodes.forEach(observeNodeIntersection);
onProgressCallback(observedNodes);
}
}
function getObservedNodes(mutationsList) {
const observedNodes = [];
mutationsList.forEach(mutationRecord => {
const newNodes = new Set(mutationRecord.addedNodes);
newNodes.forEach(node => {
if (node.querySelectorAll) {
node.querySelectorAll(OBSERVED_TAGS_SELECTOR).forEach(node => newNodes.add(node));
}
});
observedNodes.splice(0, 0, ...Array.from(newNodes).filter(matchObservedNode));
});
return observedNodes;
}
function matchObservedNode(node) {
return TAG_NAMES_WITH_SRC_ATTRIBUTE.has(node.tagName);
}
function nodeIsHidden(node) {
const boundingClientRect = node.getBoundingClientRect();
return boundingClientRect.bottom < 0 ||
boundingClientRect.top > innerHeight ||
boundingClientRect.left < 0 ||
boundingClientRect.right > innerWidth;
}
function observeNodeIntersection(node) {
const src = resetSource(node, SRC_ATTRIBUTE_NAME);
const srcset = resetSource(node, SRCSET_ATTRIBUTE_NAME);
const poster = resetSource(node, POSTER_ATTRIBUTE_NAME);
const intersectionObserver = new IntersectionObserver((entries, observer) => intersectionObserverCallback(entries, node, observer, { src, srcset, poster }));
intersectionObserver.observe(node.tagName == SOURCE_TAG_NAME ? node.parentElement : node);
}
function intersectionObserverCallback(entries, node, observer, values) {
const entry = entries[0];
if (entry) {
if (entry.intersectionRatio > MINIMUM_INTERSECTION_RATIO) {
replaceSource(node, values);
observer.disconnect();
}
}
}
function replaceSource(node, values) {
setSource(node, SRC_ATTRIBUTE_NAME, values.src);
if (TAG_NAMES_WITH_SRCSET_ATTRIBUTE.has(node.tagName)) {
setSource(node, SRCSET_ATTRIBUTE_NAME, values.srcset);
}
if (TAG_NAMES_WITH_POSTER_ATTRIBUTE.has(node.tagName)) {
setSource(node, POSTER_ATTRIBUTE_NAME, values.poster);
}
}
function resetSource(node, attributeName) {
const originalValue = node[attributeName];
if (originalValue) {
node[attributeName] = EMPTY_DATA_URI.get(node.tagName);
return originalValue;
}
}
function setSource(node, attributeName, value) {
if (node[attributeName] == EMPTY_DATA_URI.get(node.tagName)) {
if (value) {
node[attributeName] = value;
} else {
node.removeAttribute(attributeName);
if (node.readyState === UNSENT_READY_STATE) {
node.load();
}
}
}
}