Greasy Fork

Greasy Fork is available in English.

链接预览

鼠标指向链接标识图标预览链接网页

当前为 2023-12-26 提交的版本,查看 最新版本

// ==UserScript==
// @name         链接预览
// @name:en      Link Previewer
// @namespace    http://greasyfork.icu/zh-CN/users/1073-hzhbest
// @version      0.2
// @description  鼠标指向链接标识图标预览链接网页
// @description:en Hovering to preview a link
// @author       hzhbest
// @match        *://*/*
// @grant        none
// @license      MIT
// ==/UserScript==

(function() {
    'use strict';

    const ixonimg = "";
    const inonimg = "";
    const loadimg = "";
    var domainregex = /(?<=:\/\/)[^\/]+/;
    var previewwin, pre_ir, timer, linkprev, closebtn;
    const css = `
        a.___external>img.___external,a.___internal>img.___internal {
            width: 16px !important; height: 16px !important; position: absolute;
            opacity: 0%; transition: opacity 0.5s ease-out;
        }
        *:hover>a.___external>img.___external, *:hover>a.___internal>img.___internal {
            opacity: 30%;
        }
        a.___external:hover>img.___external,a.___internal:hover>img.___internal {
            opacity: 60%; background: white; z-index: 11000;
            /*transition: opacity 0.5s linear;*/
        }
        a.___external>img.___external:hover,a.___internal>img.___internal:hover {
            background: #3e3ed3 !important; transition: background 1s !important;
        }
        img.___external,img.___internal {
            transition: background 0s;
            display: inline-block !important; margin: 0 !important; transform: translate(-16px, -6px);
        }
        a.__link__preved {
            outline: 3px solid #3e3ed3;
        }
        div#__link__prev_win {
            display: none; position: absolute; z-index: 10009; box-shadow: 1px 1px 7px 1px #717171;
            padding: 0; margin: 0; background-color: white;
        }
        div#__link__prev_win.__loading {
            background: url(${loadimg}) no-repeat center center; background-color: #e7e7e7;
        } 
        div#__link__prev_win.__link__visible {
            display: block; transform: scale(0.9); 
        }
        div#__link__prev_win.visible .__link__prev_ifr {
            border: none;
        }
        @keyframes rotating_button {
            from {
                transform: rotate(0deg);
            }
            to {
                transform: rotate(360deg);
            }
        }
        div#__link__prev_win>.__link__prev_btn {
            position: absolute; top: -10px; right: -10px; width: 24px; height 24px; z-index: 10010;
            color: white; font-family: Consolas; font-size: 11pt; text-align: center;
            background: #df2020; border: 1px solid black; border-radius: 12px; 
            box-shadow: 1px 1px 5px #333; cursor: default;
        }
        div#__link__prev_win>.__link__prev_btn.__loading {
            animation: 3s linear 0s infinite rotating_button;
        }
        div#__link__prev_win>.__link__prev_btn.__loading:hover {
            animation: none;
        }
        div#__link__prev_win>.__link__prev_btn:hover {
            outline: 3px solid #be3737a2;
        }
    `;
    addCSS(css);
    var site = location.host;
    document.addEventListener('mouseover', (evt) => {
        if (evt.target.tagName == "A") {
            go([evt.target]);
        } else {
            var links = getLinksInThreeLayer(evt.target);
            go(links);
        }
    });

    function go(lnks) {
        var lnkl = lnks.length;
        var exl = 0;
        console.log(lnkl);
        for (let i = 0; i < lnkl; i++) {
            const lnk = lnks[i];
            if (lnk.classList.contains('___external') || lnk.classList.contains('___internal')) {
                continue;
            }
            var img;            
            if (domainregex.exec(lnk.href)[0] !== site) {
                lnk.classList.add('___external');
                img = creaElemIn('img', lnk);
                img.className = '___external';
                img.src = ixonimg;
                exl += 1;
            } else {
                lnk.classList.add('___internal');
                img = creaElemIn('img', lnk);
                img.className = '___internal';
                img.src = inonimg;
            }
            img.addEventListener('mouseover', (evt) => {
                //clearTimeout(timer);
                timer = setTimeout(previewlink, 1000, evt);
            });
            img.addEventListener('mouseleave', () => {
                clearTimeout(timer);
            }); 
        }
        //console.log("external link count:" + exl);

        window.addEventListener('mousedown', function (e) {
            var target = e.target;
            // 隐藏预览
            if (isprevvisual() && !previewwin.contains(target)) {
                hidepreview();
            }
        }, true);
    }

    function previewlink(evt) {
        if (!previewwin) {
            previewwin = creaElemIn('div', document.body);
            previewwin.id = "__link__prev_win";
            pre_ir = creaElemIn('iframe', previewwin);
            pre_ir.className = "__link__prev_ifr";
            pre_ir.style.border = 0;
            closebtn = creaElemIn('div', previewwin);
            closebtn.classList.add("__link__prev_btn");
            closebtn.innerHTML = "X";
            closebtn.addEventListener('click', hidepreview);
        }
        if (!!linkprev) {
            linkprev.classList.remove('__link__preved');
        }
        linkprev = evt.target.parentNode;
        linkprev.classList.add('__link__preved');
        pre_ir.src = linkprev.href;
        closebtn.classList.add("__loading");
        previewwin.classList.add("__loading");
        var minHgap = 20, minVgap = 10;
        var winWidth = 780, winHeight = 600;
        previewwin.style.left = (Math.max(minHgap, Math.min(window.innerWidth - winWidth - minHgap, evt.x))) + window.scrollX + 'px';
        previewwin.style.top = (Math.max(minVgap, Math.min(window.innerHeight - winHeight - minVgap, evt.y))) + window.scrollY + 'px';
        previewwin.style.width = winWidth + 'px';  // 小窗大小
        previewwin.style.height = winHeight + 'px';
        pre_ir.width = winWidth;
        pre_ir.height = winHeight;
        setTimeout(() => {
            previewwin.classList.add('__link__visible');
        }, 100);
        pre_ir.onload = () => {
            closebtn.classList.remove("__loading");
            previewwin.classList.remove("__loading");
        };
    }

    function hidepreview() {
        previewwin.classList.remove('__link__visible');
        linkprev.classList.remove('__link__preved');
        pre_ir.src = '';
    }

    function isprevvisual() {
        return previewwin && previewwin.classList.contains('__link__visible');
    }

    function getLinksInThreeLayer(elem) {
        var links = [];
        elem.childNodes.forEach(cnode => {
            if (ispagelink(cnode)) {
                links.push(cnode);
            } else {
                cnode.childNodes.forEach(ccnode => {
                    if (ispagelink(ccnode)) {
                        links.push(ccnode);
                    } else {
                        ccnode.childNodes.forEach(cccnode => {
                            if (ispagelink(cccnode)) {
                                links.push(cccnode);
                            }
                        });
                    }
                });
            }
        });
        return links;
    }

    function ispagelink(node) {
        if (node.tagName !== "A" || !node.href) {
            return false;
        } else if (node.href.indexOf('javascript:') == 0 || node.href.replace(location.href, "").indexOf("#") == 0) {
            return false;
        } else {
            return true;
        }
    }

    function creaElemIn(tagname, destin) {	//在 destin 内末尾创建元素 tagname
		let theElem = destin.appendChild(document.createElement(tagname));
		return theElem;
    }

    function addCSS(css, cssid) {
        let stylenode = creaElemIn('style',document.getElementsByTagName('head')[0]);
        stylenode.textContent = css;
        stylenode.type = 'text/css';
        stylenode.id = cssid || '';
    }
})();