🎉 欢迎访问GreasyFork.Org 镜像站!本镜像站由公众号【爱吃馍】搭建,用于分享脚本。联系邮箱📮

Greasy fork 爱吃馍镜像

Lazyloadfier Userscript

9/8/2023, 1:42:06 AM

Dovrai installare un'estensione come Tampermonkey, Greasemonkey o Violentmonkey per installare questo script.

You will need to install an extension such as Tampermonkey to install this script.

Dovrai installare un'estensione come Tampermonkey o Violentmonkey per installare questo script.

Dovrai installare un'estensione come Tampermonkey o Userscripts per installare questo script.

Dovrai installare un'estensione come ad esempio Tampermonkey per installare questo script.

Dovrai installare un gestore di script utente per installare questo script.

(Ho già un gestore di script utente, lasciamelo installare!)

🚀 安装遇到问题?关注公众号获取帮助

公众号二维码

扫码关注【爱吃馍】

回复【脚本】获取最新教程和防失联地址

Dovrai installare un'estensione come ad esempio Stylus per installare questo stile.

Dovrai installare un'estensione come ad esempio Stylus per installare questo stile.

Dovrai installare un'estensione come ad esempio Stylus per installare questo stile.

Dovrai installare un'estensione per la gestione degli stili utente per installare questo stile.

Dovrai installare un'estensione per la gestione degli stili utente per installare questo stile.

Dovrai installare un'estensione per la gestione degli stili utente per installare questo stile.

(Ho già un gestore di stile utente, lasciamelo installare!)

🚀 安装遇到问题?关注公众号获取帮助

公众号二维码

扫码关注【爱吃馍】

回复【脚本】获取最新教程和防失联地址

// ==UserScript==
// @name        Lazyloadfier Userscript
// @namespace   Lazyloadfier Userscript - Violentmonkey Scripts
// @match       *://*/*
// @grant       none
// @version     1.0
// @run-at      document-start
// @author      -
// @description 9/8/2023, 1:42:06 AM
// ==/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) &&
        nodeIsHidden(node) &&
        node[LOADING_ATTRIBUTE_NAME] != LAZY_LOADING_ATTRIBUTE_VALUE;
}

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 intersectionObserverOptions = {
        rootMargin: '100% 0px 100% 0px' 
    };

    const intersectionObserver = new IntersectionObserver(
        (entries, observer) => intersectionObserverCallback(entries, node, observer, { src, srcset, poster }), 
        intersectionObserverOptions
    );
    
    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();
            }
        }
    }
}