链接预览

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

// ==UserScript==
// @name         链接预览
// @name:zh-cn   链接预览
// @name:en      Link Previewer
// @namespace    https://greasyfork.org/zh-CN/users/1073-hzhbest
// @version      2.5
// @description  鼠标指向链接标识图标预览链接网页
// @description:zh-cn  鼠标指向链接标识图标预览链接网页
// @description:en Hovering to preview a link
// @author       hzhbest
// @match        http*://*/*
// @exclude      https://mega.nz/file/*
// @exclude      https://*.github.com/*
// @exclude      https://addons.mozilla.org/*
// @exclude      https://*.cloudflare.com/*
// @grant        GM_registerMenuCommand
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_openInTab
// @license      MIT
// @run-at       document-end
// ==/UserScript==

// 2.5: 可预览的视频按视频比例调整预览窗大小;域名相关的预览窗大小记录;优化预览窗位置函数,预览窗边缘距新位置不足20像素则不移动;最小宽高
// TODO: 长按(隐藏图标)模式;

(function () {
    'use strict';

    // --------自定义区-------- //
    const scriptName = "链接预览";
    const recordPrevHist = false;           // 将预览链接记入历史:T/F
    const minHgap = 30, minVgap = 30;       // 预览窗距窗口边缘水平、垂直最小距离(minHgap同时为距初始位置最小水平距离):像素
    const minW = 50, minH = 40;             // 预览窗最小宽高:像素
    var winWidth = 700, winHeight = 550;    // 预览窗默认宽、高:像素
    const scale = 0.9;                      // 预览窗内页面放大率:(0~1]
    const animationTime = 0.5;              // 动画时长:秒
    const iconSize = 20;                    // 图标放大后大小:像素
    const iconTrans = 0.7;                  // 图标放大后不透明度:(0~1)
    const pre = "░░░░░░ ";                  // 预览窗拖放手柄外观:字符串
    var delaySec = 1;                       // 触发预览等候延时:秒
    const more_delaySec = 1;                // 对小链接增加等候延时:秒
    const showLeftSizer = false;            // 显示左侧调整大小手柄:T/F
    const closeOnScoll = false;             // 在预览窗外滚动滚轮关闭预览窗:T/F
    var default_pinned = 0;                 // 初始钉住状态:{0:否,1:是}
    var default_enabled = 1;                // 初始启用状态:{0:否,1:是}
    const toggleHotkey = "C-M-v";           // 切换启用状态快捷键(Ctrl:“C-”,Alt/⌥:“M-”,字母键大写则代表含Shift键)
    var onClick = 0;                        // 是否点击预览默认值:{0:否,1:是}
    const specialLnkArr = [                 // 因链接所在元素含position样式属性而需排除的特征数组
        {
            name: "Google主链接",
            scrurl: /google\.com\/search\?/,
            linkslc: "#rso .g a:has(h3)"
        }
    ];
    const urlSubstArr = [                    // 网址替换规则数组,{命名,链接网址正则,关键编号正则,替换网址模板,出错特征元素}
        {
            name: "B站",
            scrurl: /bilibili\.com\//,
            coreID: /(av|BV)[^\?&\/]+/,
            desturl: "https://player.bilibili.com/player.html?bvid=--coreID--&high_quality=1&autoplay=1"
            // desturl: "https://www.bilibili.com/blackboard/html5mobileplayer.html?bvid=--coreID--&autoplay=1" //2024-04-28发现不会自动播放
        },
        {
            name: "油管",
            scrurl: /youtube\.com\//,
            coreID: /(?<=(watch\?v=|\/shorts\/|\/live\/))[^&\/]+/,
            desturl: "https://www.youtube.com/embed/--coreID--",
            errorElem: "div.ytp-error"
        },
        {
            name: "油管短",
            scrurl: /youtu\.be\//,
            coreID: /(?<=(\.be\/))[^&\/]+/,
            desturl: "https://www.youtube.com/embed/--coreID--",
            errorElem: "div.ytp-error"
        },
        {
            name: "西瓜",
            scrurl: /ixigua\.com\//,
            coreID: /(?<=\/)\d+\?/,
            desturl: "https://www.ixigua.com/iframe/--coreID--autoplay=1"
        },
        {
            name: "PH",
            scrurl: /pornhub\.com\//,
            coreID: /(?<=viewkey=).+/,
            desturl: "https://www.pornhub.com/embed/--coreID--"
        }
    ];
    // --------自定义区结束--------//

    const preID = "__link__prev_win"        // 预览窗id
    const uSA_len = urlSubstArr.length;
    const ixonimg = "";
    const inonimg = "";
    const loadimg = "";
    const _txt = {
        swc: [["hover", "click"], ["悬停", "点击"]],
        ntc: ["Switched to --swc-- preview mode", "已切换到--swc--预览模式"],
        gmc: ["Toggle preview mode", "切换预览模式"],
        swp: [["Pinless", "Pinned"], ["非钉住", "钉住"]],
        ntp: ["Switched to --swp-- mode by default", "已切换到默认--swp--模式"],
        gmp: ["Toggle default pinned", "切换默认钉住模式"],
        enpin: ["Pin Preview Win", "钉住预览窗"],
        unpin: ["Unpin Preview Win", "解钉预览窗"],
        optab: ["Open this page in a new tab", "在新标签页打开当中页面"],
        swt: [["disable", "enable"], ["禁用", "启用"]],
        ntt: ["Switched to --swt-- status", "已切换到--swt--状态"],
        gmt: ["Toggle preview status", "切换预览状态"],
        gmd: ["Set / Unset Preview Win Size for this DOMAIN", "设置/删除当前域名下预览窗大小"],
        msd: ["Click \"SET\" button to set Preview Win Size (--wh--) for [--dm--].", "点击“设置”按钮为[--dm--]设置预览窗大小(--wh--)。"],
        mdd: ["\n OR \nClick \"UNSET\" button to remove Size setting for [--dm--].", "\n 或 \n点击“删除”按钮删除[--dm--]的预览窗大小设置。"],
        bsd: [["SET", "SET √"], ["设置", "设置 √"]],
        bud: [["UNSET", "UNSET √"], ["删除", "删除 √"]],
        bcd: ["CANCEL", "取消"]
    }
    //language detection
    const _L = (navigator.language.indexOf('zh-') == -1) ? 0 : 1;
    const is_onClick = "Link_preview_on_click";
    const is_default_pin = "Link_preview_default_pinned";
    const is_default_enable = "Link_preview_default_enabled";
    onClick = GM_getValue(is_onClick, onClick);
    default_pinned = GM_getValue(is_default_pin, default_pinned);
    default_enabled = GM_getValue(is_default_enable, default_enabled);
    document.body.classList.toggle("__link_pre_clk", (onClick == 1));     // 全局点击样式标志
    document.body.classList.toggle("__link_pre_disable", (default_enabled == 0));     // 全局禁用样式标志
    delaySec = (onClick) ? delaySec / 2 : delaySec;

    var perDomainWinSize, default_pDWS, dialogDWS, btnsDWS, matchedDWS = -1;
    const pDWS = "Link_preview_domain_winsize";
    default_pDWS = [];
    var spDWS = GM_getValue(pDWS);
    if (!!spDWS && spDWS.slice(0, 1) === "[") {
        perDomainWinSize = JSON.parse(spDWS);
    } else {							//否则将默认设置写入储存
        GM_setValue(pDWS, JSON.stringify(default_pDWS));
    }
    var site = location.host;
    for (let i = 0; i < perDomainWinSize.length; i++) {     // 若域名匹配,则以记录的宽高为默认宽高
        if (perDomainWinSize[i].domain == site) {
            winHeight = perDomainWinSize[i].h;
            winWidth = perDomainWinSize[i].w;
            matchedDWS = i;
        }
    }

    const domainRegex = /(?<=:\/\/)[^\/]+/;
    const id = "__link__prev";
    const css = `
        /* 链接样式 */
        a.__link__preved {
            outline: 3px solid #3e3ed3;
        }
        a.___prevlink.__pr {
            position: relative;
        }
        /* 图标样式 */
        a.___prevlink>img.___previcon {
            width: 16px !important; height: 16px !important;
            opacity: 0%; transition: opacity 0.5s ease-out, background 0s;
            position: absolute; pointer-events: auto !important;
            display: inline-block !important; margin: 0 !important;
            max-width: initial !important; max-height: initial !important;
        }
        a.___prevlink>img.___previcon:hover {
            background: #3e3ed3 !important; filter: invert(1) hue-rotate(150deg);
            transition: background ${delaySec}s, filter ${delaySec}s !important;
        }
        body.__link_pre_clk a.___prevlink>img.___previcon:hover {
            background: #c652e6 !important; outline: 2px solid #76268d !important; filter: invert(1) hue-rotate(150deg);
            transition: background 0.1s, filter 0.1s !important;
        }
        body.__link_pre_disable a.___prevlink>img.___previcon {
            display: none !important;
        }
        a.___prevlink>img.___previcon.__moredelay:hover, body.__link_pre_clk a.___prevlink>img.___previcon.__moredelay:hover {
            transition: background ${delaySec + more_delaySec}s, filter ${delaySec + more_delaySec}s, outline 0s ease ${delaySec + more_delaySec}s !important;
            outline: 3px solid #be18ec !important;
        }
        *:hover>a.___prevlink>img.___previcon {
            opacity: 30%;
        }
        a.___prevlink:hover {
            outline: 2px dotted #939393;
        }
        a.___prevlink:hover>img.___previcon {
            opacity: ${iconTrans}; background: white; border: 1px solid #8d8d8db1;
            transition: opacity 0.5s linear; z-index: 10008;
            transform: scale(${iconSize / 16});
        }
        /* 预览窗样式 */
        div#${preID} {
            opacity: 0; position: absolute; z-index: 1000001; box-shadow: 1px 1px 7px 1px #717171;
            padding: 0; margin: 0; background-color: #ffffff; transition: ${animationTime}s ease;
        }
        div#${preID}.__close {
            transition: 0s !important; display: none;
        }
        div#${preID}.__loading {
            background: url(${loadimg}) no-repeat center center; background-color: #e7e7e7;
            box-shadow: 1px 1px 7px 1px #d8db7c !important;
        }
        div#${preID}.__visible {
            opacity: 1; transition: 0.2s !important;
        }
        div#${preID}.visible .__link__prev_ifr {
            border: none; transform: scale(${scale});
            max-width: unset !important; width: calc(100% / ${scale});
            max-height: unset !important; height: calc(100% / ${scale});
        }
        div#${preID}.__onmove, div#${preID}.__onsize, div#${preID}.__onpin{
            transition: 0s !important;
        }
        div#${preID}.__onmove {
            /* opacity: 0.7; */
        }
        div#${preID}.__pinned {
            box-shadow: 1px 1px 7px 1px #a36835; position: fixed;
        }
        /* 关闭按钮样式 */
        @keyframes rotating_button {
            from {
                transform: rotate(0deg);
            }
            to {
                transform: rotate(360deg);
            }
        }
        div#${preID}>.__link__clos_btn {
            position: absolute; top: -10px; right: -10px; z-index: 10010;
            line-height: 24px; width: 24px; height: 24px;
            color: white; font-family: Consolas; font-size: 16px; text-align: center;
            background: #df2020; border: 1px solid black; border-radius: 50%;
            box-shadow: 1px 1px 5px #333; cursor: default; padding: 0 !important; margin: 0;
        }
        div#${preID}>.__link__clos_btn.__loading {
            animation: 3s linear 0s infinite rotating_button; background: #d76565;
        }
        div#${preID}>.__link__clos_btn.__loading:hover {
            animation: none; background: #df2020;
        }
        div#${preID}>.__link__clos_btn:hover {
            outline: 3px solid #be3737a2;
        }
        div#${preID}>.__link__clos_btn.__close {
            display: none;
        }
        /* 拖动手柄样式 */
        div#${preID}>div.__link__prev_mv {
            position: absolute; top: -5px; left: 0px; z-index: 10010;
            height: 6px; width: 30px; border: 1px solid #242424; background: #3e3e3e; cursor: move;
            border-radius: 3px; overflow: hidden; max-width: 95%;
            font-size: 14px !important; color: #d7d7d7; line-height: 19px;
            display: grid; grid-template-columns: auto auto auto 1fr;
        }
        div#${preID}>div.__link__prev_mv>* {
            display: block;
        }
        div#${preID}:hover>div.__link__prev_mv {
            height: 20px; top: -20px; width: auto;
        }
        div#${preID}>div.__link__prev_mv:hover {
            color: #e0a52e;
        }
        div#${preID}.__onmove>div.__link__prev_mv {
            background: #6f6122; height: 30px; top: -20px; width: fit-content;
        }
        div#${preID}.__loading>div.__link__prev_mv {
            background: #aa9537;
        }
        div#${preID} .__link__prev_ttl {
            font-size: 14px !important; font-family: Arial; white-space: nowrap; margin: 0 3px; overflow: hidden;
            text-overflow: ellipsis; width: 100%;
        }
        div#${preID}>.__link__prev_rs {
            height: 4px; width: 4px; border: 1px solid #616161; color: #9a9276; font-size: 13px;
            position: absolute; right: 0; bottom: 0; z-index: 10011; background: #3e3e3e67;
            transition: 0.3s ease-out; font-size: 10px; line-height: 10px; overflow: hidden;
            cursor: nwse-resize;
        }
        div#${preID}>.__link__prev_rs.__rs_l {
            left: 0; right: unset !important; cursor: nesw-resize !important;
        }
        div#${preID}:hover>.__link__prev_rs {
            background: #3e3e3e; height: 12px; width: 12px;
        }
        div#${preID}>.__link__prev_rs:hover {
            border-color: #2c96e2af; height: 18px; width: 18px;
        }
        div#${preID}.__onsize>.__link__prev_rs.__onsize {
            height: 40px; width: 60px; transition: 0s; color: #e0a52e
        }
        /* 顶栏按钮样式 */
        div#${preID} .__link__prev_btn {
            height: 18px; width: 18px !important; box-sizing: content-box; border-width: 0 1px; border-style: solid;
            vertical-align: bottom; padding: 0 6px;text-align: center; cursor: pointer;
        }
        div#${preID} .__link__prev_btn:hover {
            border-color: #f1db27; color: #f1db27; background: #ffffff36;
        }
        div#${preID} .__link__prev_btn:active {
            background: #ffffff82;
        }
        div#${preID}.__pinned .__link__prev_btn.__pinbtn {
            background: #f5bc4f82; font-weight: 800;
        }
    `;
    addCSS(css, id);                                            // 添加的style节点添加id,用于判断是否需重添加
    if (window.self == window.top) {
        window.addEventListener('load', function () {
            GM_registerMenuCommand(_txt.gmc[_L], changeMode);
            GM_registerMenuCommand(_txt.gmp[_L], changeDfpin);
            GM_registerMenuCommand(_txt.gmt[_L], changeStatus);
            GM_registerMenuCommand(_txt.gmd[_L], setDWSdlg);
        }, false);
    }
    document.addEventListener('keydown', keyhandler, false);
    var specSiteInd = -1;
    for (let i = 0; i < specialLnkArr.length; i++) {
        if (specialLnkArr[i].scrurl.test(location.href)) {
            specSiteInd = i;
            break;
        }
    }
    var previewwin, pre_ir, pinbtn, closebtn, movehand, titlebox, ondrag, resizehand, resizehand2, openbtn;
    var rsdrag, rsldrag, winpos, pinned, opened;    // winpos{x,y,w,h}:预览窗(若显示)的位置和大小
    var timer, linkprev, prevedUrl, mousepos, vscale, moretimer;
    var isiniframe = false, isinprevwin = false;
    pinned = (default_pinned == 1);
    opened = false;                     // 预览窗开启状态
    winpos = { h: winHeight, w: winWidth };

    if (window.self != window.top) {    // 只在iframe中使用的事件处理
        isiniframe = true;
        window.addEventListener('message', (e) => {  // 接收来自预览窗(外部)的信息,若有,则标识为预览窗内并回发消息
            // console.log(e.data);
            if (e.data.from == '__link__prev_top') {
                isinprevwin = true;
                var vaspect;
                setTimeout(() => {
                    if (!!e.data.errorElem && !!document.querySelector(e.data.errorElem)) { // 若找到errorElem,则重载原网址;应对yt“只能在网页播放”的视频
                        window.self.location.href = e.data.oriUrl;
                        return;
                    }
                    var videoElem = document.querySelector('video');
                    if (!!videoElem) {
                        videoElem.addEventListener('loadeddata', () => {
                            if (!!videoElem.videoHeight) {       // 若存在视频元素且有高度(不管是否替换过网址)
                                vaspect = videoElem.videoWidth / videoElem.videoHeight;
                                window.top.postMessage({
                                    from: '__link__prev_invideo',
                                    vaspect: vaspect
                                }, '*');
                            }
                        });
                    }
                }, 600);
                window.top.postMessage({        // 向top窗口发送消息,报告当前页面url
                    from: '__link__prev_inwin',
                    url: window.self.location.href
                }, '*');
                makePrevWin();                  // 在预览窗内嵌套预览窗
            }
        }, true);
    } else {
        makePrevWin();                  // 在顶层窗口建预览窗
        document.addEventListener('keydown', (evt) => {
            if (evt.key == "Escape" && isPrevVisual()) {    // ESC键按下时若预览窗可见,则关闭预览窗
                hidePrevWin();
            }
        });
        window.addEventListener('message', function (e) {   // 处理iframe发来的消息
            if (e.data.from == '__link__prev_inwin') {
                if (e.data.url && isPrevVisual()) {         // 向预览窗传输iframe中的实际网址
                    openbtn.dataset.url = e.data.url;
                    linkprev.classList.add("__link__preved");   // 预览成功再显示链接被预览的状态
                }
            } else if (e.data.from == '__link__prev_iframe') {
                if (!!e.data.prevhref) {
                    callPrevWin(e.data.prevod, e.data.prevhref, e.data.prevcont);
                }
            } else if (e.data.from == '__link__prev_invideo') { // 如果预览窗中有视频,按视频比例调整窗口
                if (!!e.data.vaspect) {
                    var vaspect = e.data.vaspect;
                    var maxh = window.innerHeight * 0.8;
                    var maxw = window.innerWidth * 0.6;
                    var wh = maxw / vaspect;
                    wh = (maxh > wh) ? wh : maxh;
                    var ww = wh * vaspect;
                    var origin = JSON.parse(previewwin.dataset.origin);
                    // 根据新预览窗大小调整位置(以原窗口初始起点)
                    if (!isPrevVisPinned()) winpos = setWinPos({ ...origin, w: ww, h: wh });
                }
            }
        }, true);
    }

    setTimeout(function () {                                        // 延时启动
        document.addEventListener('mouseover', (evt) => {           // 检查鼠标下的节点,查找链接添加图标
            const t = evt.target;
            // console.log('overed>: ', t);
            if (t.tagName == "A" && isPageLink(t)) {
                addIconTo([t]);
            } else {
                var links = getLinksInThreeLayer(t);
                // console.log('links(1): ', links);
                if (links.length == 0) {
                    links = getLinkInParent(t);
                }
                // console.log('links(2): ', links);
                if (links.length > 0) {
                    addIconTo(links);
                }
            }
            mousepos = { x: evt.clientX, y: evt.clientY };
        });

        // 设置监视器,若标题变化则重新加载
        var watch = document.querySelector('title');
        new (window.MutationObserver || window.WebKitMutationObserver)(function (mutations) {
            // console.log('标题变了', document.title);
            if (!document.querySelector("#" + id)) {
                addCSS(css, id);
            }
            if (!document.querySelector("#" + preID)) {
                makePrevWin();
            }
            site = location.host;
        }).observe(watch, { childList: true, subtree: true, characterData: true });
    }, 1000);

    function addIconTo(links) {     // 往链接上添加图标;输入:链接的数组
        // console.log('going2addIcon::links: ', links);
        var lnkl = links.length;
        var exl = 0;
        // console.log(lnkl);
        for (let i = 0; i < lnkl; i++) {
            const lnk = links[i];
            if (!lnk.classList.contains("___prevlink")) {    // 包括被抹掉了a的样式标志
                lnk.classList.toggle("___prevlink", true);
            } else if (!!lnk.querySelector('img.___previcon')) {    // 跳过已有图标的链接
                continue;
            }

            var islinkpositionable = true;
            if (specSiteInd > -1) {
                if ([...document.querySelectorAll(specialLnkArr[specSiteInd].linkslc)].includes(lnk)) { // 若为特殊网站特殊链接则不添加定位样式
                    islinkpositionable = false;
                }
            }
            if (islinkpositionable && window.getComputedStyle(lnk).position == "static") {    // 若链接本身无position设置则添加定位样式
                lnk.classList.add("__pr");
            }

            var lnklcposit = getLastChildSize(lnk);         // 获取链接末个子节点的占位
            var lnkposit = getTrueSize(lnk);                // 无末个子节点则获取链接占位
            var lcl, lcw;
            if (!lnklcposit) {
                lcl = lnkposit.l;
                lcw = lnkposit.w;
            } else {
                lcl = lnklcposit.l;
                lcw = lnklcposit.w;
            }
            var img;
            img = creaElemIn('img', lnk);
            // img.dataset.for = url;
            img.classList.add("___previcon");
            var adl = lcl - lnkposit.l + lcw + 4; // 末子节点右边界右4像素;如果被父节点遮挡或自身隐藏
            var adb = -0.3;
            if (lnkposit.r < (4 + iconSize) || window.getComputedStyle(lnk).overflow !== "visible") {
                adl = Math.min(adl, lnkposit.w - 5 - iconSize); // 则避开遮挡
                adb = 0.05;
            }
            img.style.left = adl + "px";
            img.style.bottom = adb + "lh";
            if (domainRegex.exec(lnk.href)[0] !== site) {   // 站内还是站外链接
                img.src = ixonimg;
                exl += 1;
            } else {
                img.src = inonimg;
            }
            if (lnkposit.w <= (iconSize * 10) && lnkposit.h <= (iconSize * 4) && adl < lnkposit.w) {    // 链接不宽不高且图标位于链接内的话
                img.classList.add("__moredelay");
            }
            img.addEventListener('mouseover', (evt) => {    // 鼠标悬停图标上的动作
                var etarget = evt.target;
                if (onClick == 1) {                   // 点击预览模式时,对小链接增加延时,延时后点击才预览
                    if (etarget.classList.contains("__moredelay")) {
                        moretimer = setTimeout(() => {
                            etarget.classList.add("__moredelayfin");
                        }, more_delaySec * 1000);
                    }
                    return;
                }
                var od = delaySec;
                if (evt.target.classList.contains("__moredelay")) {
                    od += more_delaySec;
                }
                od *= 1000;
                previewEvent(evt, od);
            });
            img.addEventListener('click', (evt) => {    // 鼠标点击图标的动作
                if (onClick == 0 || (evt.target.classList.contains("__moredelay") && !evt.target.classList.contains("__moredelayfin"))) {
                    return;
                }
                clearTimeout(timer);
                evt.preventDefault();
                evt.stopPropagation();
                previewEvent(evt, 0);
            });
            img.addEventListener('mouseleave', (evt) => {      // 鼠标离开图标则清除计时
                if (evt.target.classList.contains("__moredelayfin")) {
                    evt.target.classList.remove("__moredelayfin");
                }
                clearTimeout(timer);
                clearTimeout(moretimer);
            });
            // lnk.addEventListener('mouseover', (evt) => {
            //     console.log("lcl:", getLastChildSize(evt.target));
            //     console.log("ls", getTrueSize(evt.target));
            // });
        }
        //console.log("external link count:" + exl);

        window.addEventListener('mousedown', (e) => {       // 点击页面的动作
            var t = e.target;
            if (isPrevVisual() && !pinned && !previewwin.contains(t) && !t.classList.contains("___previcon")) {
                hidePrevWin(e);   // 若预览窗开启且非钉住状态且点击位置不在预览窗内或预览图标上则隐藏预览窗
            }
        }, true);
        if (closeOnScoll) {
            window.addEventListener('scroll', () => {       // 滚动页面的动作
                if (isPrevVisual() && !pinned) {
                    hidePrevWin();   // 若预览窗开启且非钉住状态且滚动位置不在预览窗内(window天然隔离iframe)则隐藏预览窗
                }
            }, true);
        }
    }

    function previewEvent(evt, od) {                        // 触发预览
        if (prevedUrl == evt.target.parentNode.href && isPrevVisual()) {  // 已在预览的链接相同则无视
            return;
        }
        let nowprev = document.querySelector('.__link__preved');
        if (!!nowprev) {                            // 若有链接在预览中,去除其预览中状态
            nowprev.classList.remove("__link__preved");
        }
        linkprev = evt.target.parentNode;           // 指定预览目标链接
        if (isiniframe && !isinprevwin) {           // 在iframe中且非预览窗中,则发送消息
            window.top.postMessage({                // 向top窗口发送消息,请求调用预览窗
                from: '__link__prev_iframe',
                prevhref: linkprev.href,
                prevcont: linkprev.textContent,
                prevod: od
            }, '*');
        } else {                                    // 否则直接调预览窗
            callPrevWin(od);
        }
    }

    function callPrevWin(od, prevhref, prevcont) {  // 召唤预览窗
        var origin = {
            x: mousepos.x,
            y: mousepos.y
        };
        setTimeout(() => {
            if (previewwin.classList.contains("__close")) { // 若预览窗关闭状态则移动到鼠标下
                setWinPos(origin, true);
            }
        }, od);
        timer = setTimeout(previewLink, od, prevhref, prevcont, origin); // od秒后开启预览窗
    }

    function makePrevWin() {        // 生成预览窗
        previewwin = creaElemIn('div', document.body);
        previewwin.id = preID;
        previewwin.style.width = '300px';  // 显示前预览窗大小,显示时形成放大效果
        previewwin.style.height = '200px';
        previewwin.classList.add("__close");    // 初始状态关闭

        movehand = creaElemIn('div', previewwin);   // 移动手柄
        movehand.classList.add("__link__prev_mv");

        resizehand = creaElemIn('div', previewwin); // 调整大小手柄
        resizehand.className = "__link__prev_rs";

        if (showLeftSizer) {
            resizehand2 = creaElemIn('div', previewwin); // 调整大小手柄
            resizehand2.className = "__link__prev_rs";
            resizehand2.classList.add("__rs_l");
        }

        var movehead = creaElemIn('div', movehand); // 移动手柄头
        movehead.textContent = pre;

        pinbtn = creaElemIn('div', movehand);     // 钉住按钮
        pinbtn.classList.add("__link__prev_btn");
        pinbtn.classList.add("__pinbtn");
        pinbtn.textContent = "T";
        pinbtn.title = (pinned) ? _txt.unpin[_L] : _txt.enpin[_L];
        pinbtn.addEventListener('click', togglePin);

        openbtn = creaElemIn('div', movehand);     // 新标签页打开按钮
        openbtn.classList.add("__link__prev_btn");
        openbtn.classList.add("__opnbtn");
        openbtn.textContent = "↗";
        openbtn.title = _txt.optab[_L];
        openbtn.addEventListener('click', (evt) => {
            if (!!evt.target.dataset.url) {
                openInTab(evt.target.dataset.url);
            }
        });

        titlebox = creaElemIn('div', movehand);     // 标题栏
        titlebox.className = "__link__prev_ttl";

        pre_ir = creaElemIn('iframe', previewwin);  // 预览容器iframe
        pre_ir.className = "__link__prev_ifr";
        pre_ir.style.border = 0;
        pre_ir.style.maxHeight = "unset";
        pre_ir.style.maxWidth = "unset";

        closebtn = creaElemIn('div', previewwin);   // 关闭按钮
        closebtn.classList.add("__link__clos_btn");
        closebtn.textContent = "✖";
        closebtn.addEventListener('click', hidePrevWin);

        ondrag = endrag(previewwin, movehand);      // 绑定移动预览窗对象
        ondrag.hook('__drag_begin', () => {         // 绑定移动开始、结束状态样式
            winpos = getWinPos();
            previewwin.classList.add("__onmove");
            ondrag.position._x = winpos.x;          // position 绑定预览窗左上坐标
            ondrag.position._y = winpos.y;
        });
        ondrag.hook('__dragging', () => {           // 绑定移动响应功能
            if (ondrag.isDragging) {                // 是否是拖动中状态
                winpos = setWinPos({ x: ondrag.position._x, y: ondrag.position._y }, true);    // 随着拖动自由移动位置
            }
        });
        ondrag.hook('__drag_end', () => {
            previewwin.classList.remove("__onmove");
        });

        rsdrag = endrag(previewwin, resizehand);    // 绑定调整预览窗大小对象
        rsdrag.hook('__drag_begin', () => {         // 绑定调整大小开始、结束状态样式、显示
            winpos = getWinPos();
            previewwin.classList.add("__onsize");
            resizehand.classList.add("__onsize");
            rsdrag.position._x = winpos.w;          // position 绑定预览窗宽高
            rsdrag.position._y = winpos.h;
        });
        rsdrag.hook('__dragging', () => {           // 绑定调整大小响应功能
            if (rsdrag.isDragging) {                // 是否是拖动中状态
                winpos.w = rsdrag.position._x;
                winpos.h = rsdrag.position._y;
                winpos = setWinPos(winpos, true);
                if (previewwin.classList.contains("__onsize")) {    // 调整中显示当前大小(该状态需加入已开始判断)
                    resizehand.textContent = `
                        W: ${winpos.w} px\nH: ${winpos.h} px
                    `;
                }
            }
        });
        rsdrag.hook('__drag_end', () => {
            previewwin.classList.remove("__onsize");
            resizehand.classList.remove("__onsize");
            resizehand.textContent = "";
        });

        if (showLeftSizer) {
            rsldrag = endrag(previewwin, resizehand2);  // 绑定左侧调整预览窗大小对象
            rsldrag.hook('__drag_begin', () => {        // 绑定调整大小开始、结束状态样式、显示
                winpos = getWinPos();
                previewwin.classList.add("__onsize");
                resizehand2.classList.add("__onsize");
                rsldrag.position._x = winpos.w * -1;    // 左边手柄拖动,鼠标横坐标增则宽度减
                rsldrag.position._y = winpos.h;
            });
            rsldrag.hook('__dragging', () => {          // 绑定调整大小、移动响应功能
                if (rsldrag.isDragging) {               // 是否是拖动中状态
                    var pos = {
                        x: winpos.x - ((rsldrag.position._x * -1) - winpos.w),  // 左边手绑拖动,宽度增则x坐标减
                        y: winpos.y,
                        w: rsldrag.position._x * -1,
                        h: rsldrag.position._y
                    }
                    winpos = setWinPos(pos, true);  // 同步调整位置和大小
                    if (previewwin.classList.contains("__onsize")) {    // 调整中显示当前大小(该状态需加入已开始判断)
                        resizehand2.textContent = `
                            W: ${winpos.w} px\nH: ${winpos.h} px
                        `;
                    }
                }
            });
            rsldrag.hook('__drag_end', () => {
                previewwin.classList.remove("__onsize");
                resizehand2.classList.remove("__onsize");
                resizehand2.textContent = "";
            });
        }
    }

    function previewLink(prevhref, prevcont, origin) {             // 在预览窗中加载链接
        // console.log('linkprev: ', linkprev);
        if (!prevhref) {
            prevhref = linkprev.href;
            prevcont = linkprev.textContent;
        }
        prevedUrl = prevhref;
        // var url = linkprev.getAttribute('url') || linkprev.href;
        var url = prevedUrl;
        // console.log('url: ', url);
        vscale = scale;
        var errorelem;
        if (uSA_len > 0) {                        // 链接替换
            for (let i = 0; i < uSA_len; i++) {
                const ur = urlSubstArr[i];
                if (ur.scrurl.test(url)) {      // 若网址匹配
                    const urid = ur.coreID.exec(url);   // 提取coreID
                    if (!!urid) {               // 若coreID存在则继续
                        url = ur.desturl.replace("--coreID--", urid[0]);    // 网址替换
                        pre_ir.allow = "autoplay";
                        pre_ir.sandbox = "allow-top-navigation allow-same-origin allow-forms allow-scripts";
                        vscale = 1;
                        errorelem = ur?.errorElem;
                        break;
                    }
                }
            }
        }
        var lh = location.href;
        pre_ir.src = url;                           // 加载页面,调整缩放
        var regSameOri = /[^\/:]+\:\/\/[^a-z0-9\.\:]\//;
        if (recordPrevHist && (regSameOri.exec(lh) == regSameOri.exec(url))) {
            history.pushState({ foo: "callback" }, null, url);     // 推送预览url到历史中
            history.replaceState({ foo: "callback" }, null, lh);     // 用当前页面地址替换历史state
        }
        titlebox.textContent = pre + prevcont;                // 以链接文本为标题
        titlebox.title = prevcont + "\n" + prevhref;
        pre_ir.style.transform = "scale(" + vscale + ")";
        pre_ir.style.height = 100 / vscale + "%";
        pre_ir.style.width = 100 / vscale + "%";
        previewwin.classList.toggle("__close", false);      // 预览窗去除关闭、添加加载中状态
        previewwin.classList.toggle("__loading", true);
        previewwin.classList.toggle('__pinned', pinned);
        previewwin.dataset.origin = JSON.stringify(origin); // 保存预览窗初始位置
        setTimeout(() => {                                  // 延时0.1秒(等待预览窗到位)显示、移动、调整大小
            previewwin.classList.toggle("__visible", true);
            if (!isPrevVisPinned()) {                       // 若已是显示中的钉住预览窗,则不调整位置
                winpos = setWinPos({ ...origin, w: winWidth, h: winHeight });
            }
            closebtn.classList.remove("__close");
            closebtn.classList.add("__loading");
            opened = true;                                  // 预览窗开启状态开
        }, 100);
        pre_ir.onload = () => {                     // iframe加载完成后去除加载中状态
            closebtn.classList.remove("__loading");
            previewwin.classList.remove("__loading");
            pre_ir.contentWindow.postMessage({      // 向其中的iframe发送加载完成信息,请求检查errorElem
                from: '__link__prev_top',
                errorElem: errorelem,
                oriUrl: prevedUrl
            }, '*');
        };
    }

    function getWinPos() {
        var p = previewwin.getBoundingClientRect();       // 预览窗屏幕位置大小
        return { x: p.left, y: p.top, w: p.width, h: p.height };
    }

    function calWinPos(pos) {                           // 计算基于 pos 避让屏幕边框的预览窗屏幕位置
        var x, y, w, h, p, x1, x2, y1, lok, rok;
        p = winpos;
        w = (pos?.w ?? p?.w) || winWidth;               // 若 pos 中无宽高信息,则获取现有或预设预览窗的宽高
        h = (pos?.h ?? p?.h) || winHeight;
        x1 = pos.x + minHgap / 2;                       // 在初始位置右的x
        x2 = pos.x - w - minHgap / 2;                   // 在初始位置左的x
        rok = x1 < (window.innerWidth - w - minHgap)    // 位置右的右边空间充足
        lok = x2 > minHgap;                             // 位置左的左边空间充足
        if (rok && lok) {                               // 若两边空间充足
            if (opened) {                               // 若预览窗开启中,横坐标取移动距离短的x
                x = (Math.abs(x1 - p.x) <= Math.abs(x2 - p.x)) ? x1 : x2;
            } else {                                    // 否则倾向右边
                x = x1;
            }
        } else if (!rok && !lok) {                      // 若两边空间都不足
            if (opened) {                               // 若预览窗开启中,横坐标以当前位置避让两边
                x = Math.max(minHgap, Math.min(window.innerWidth - w - minHgap, p.x));
            } else {
                x = Math.min(window.innerWidth - w - minHgap, x1);    // 否则靠右取
            }
        } else {                                        // 若一边空间充足
            x = (rok) ? x1 : x2;                        // 哪边空间充足取哪边的x
        }
        y1 = pos.y - 0.382 * h;                       // 理想y位置
        y = Math.max(Math.min(y1, window.innerHeight - minVgap - h), minVgap);  // 避让上下边缘

        if (opened) {
            if (Math.abs(x - p.x) <= 20) {              // 若相比现有位置差异不大则不移动
                x = p.x;
            }
            if (Math.abs(y - p.y) <= 20) {
                y = p.y;
            }
        }
        return { x: x, y: y, w: w, h: h };
    }

    function setWinPos(pos, isfree) {
        // 设置预览窗相对页面位置,和大小
        // pos({x,y,w,h}:左、顶、宽、高)数组,相对viewport位置
        // isfree(T/F)是否不避让屏幕边框
        if (!isfree) {                              // “非自由移动”,如预览窗跟随预览链接位置
            pos = calWinPos(pos);                   // 否则计算避让位置
        }
        var f = (pinned) ? 0 : 1;
        previewwin.style.left = pos.x + f * window.scrollX + 'px';
        previewwin.style.top = pos.y + f * window.scrollY + 'px';
        var w, h;
        w = pos?.w ?? 0;                            // 没传入大小则不更改大小
        h = pos?.h ?? 0;
        if (w >= minW && h >= minH) {               // 限制修改大小时的最小大小
            setPrevWinSize(w, h);
        }
        return { x: pos.x, y: pos.y, w: w || winpos.w, h: h || winpos.h };
    }

    function setPrevWinSize(w, h) {                 // 只设置预览窗大小
        previewwin.style.width = w + 'px';
        previewwin.style.height = h + 'px';
        setIfrWinSize(w, h);
    }

    function setIfrWinSize(w, h) {         // 根据放大率设置iframe大小
        var f = (vscale == 1) ? 0 : (1 - vscale) / 2 * -1 / vscale;
        pre_ir.width = w / vscale;
        pre_ir.height = h / vscale;
        pre_ir.style.marginLeft = (w * f) + 'px';
        pre_ir.style.marginTop = (h * f) + 'px';
    }

    function togglePin(ispinned) {                  // 切换钉住状态
        if (typeof (ispinned) == "boolean") {
            pinned = ispinned;
        } else {
            pinned = !pinned;
        }
        setWinPos(getWinPos(), true);                   // 按预览窗实际屏幕位置重新定位
        previewwin.classList.toggle('__onpin', true);       // 钉住“切换中”状态,以避免取消钉住后过快恢复动画
        previewwin.classList.toggle('__pinned', pinned);    // 已钉住状态
        setTimeout(() => {
            previewwin.classList.toggle('__onpin', false);  // 钉住“切换中”状态结束
        }, 100);
    }

    function openInTab(url) {
        GM_openInTab(url, { active: true });
        hidePrevWin();
    }

    function hidePrevWin() {
        pinned = (default_pinned == 1);
        pre_ir.src = '';
        var origin = JSON.parse(previewwin.dataset.origin); // 读取预览窗初始位置
        setWinPos({ ...origin, w: 100, h: 100 }, true);   // 预览窗向初始位置缩小
        previewwin.classList.remove('__visible');           // 预览窗隐藏
        linkprev.classList.remove('__link__preved');        // 去除链接高亮中状态
        closebtn.classList.add("__close");
        setTimeout(() => {                                  // 动画效果完成后添加关闭状态
            previewwin.classList.add("__close");
            opened = false;                                 // 预览窗开启状态关
        }, animationTime * 1000 + 10);
    }

    function isPrevVisual() {               // 预览窗是否可见
        return previewwin && previewwin.classList.contains('__visible');
    }

    function isPrevVisPinned() {
        return isPrevVisual() && pinned;
    }

    function getLinksInThreeLayer(elem) {   // 对当前元素及其下两层的链接进行提取
        return getLinksInChild(elem, 3);
    }

    function getLinksInChild(elem, lcnt) {  // 递归提取元素中的链接
        var lnks = [];
        if (elem.childNodes.length == 0) {
            return lnks;
        } else {
            elem.childNodes.forEach((cnode) => {
                if ((cnode.tagName == "A") && isVisible(cnode)) {
                    if (isPageLink(cnode)) {
                        lnks.push(cnode);
                    } else {
                        if (lcnt - 1 > 0) {
                            lnks.concat(getLinksInChild(cnode, lcnt - 1));
                        }
                    }
                }
            });
        }
        return lnks;
    }

    function getLinkInParent(elem) {
        if (isPageLink(elem)) {
            return [elem];
        } else {
            let pelem = elem.parentNode;
            if (!elem || pelem.tagName == 'BODY') {
                return [];
            } else {
                return getLinkInParent(pelem);
            }
        }
    }

    function isPageLink(node) {             // 若A节点是指向实网址(非js非锚点)返回true
        if (!node.href || (!!node.target && !/_(self|blank|top)/.test(node.target))) {  // 无网址或有框架指向的链接(点击不刷新主体页面)不处理
            return false;
        }
        const h = node.href, l = location.href; // 仅处理非js非本页锚点
        if (h.indexOf('javascript:') == 0 || h.replace(l.split('#')[0], "").indexOf("#") == 0) {
            return false;
        } else {
            return true;
        }
    }

    function isVisible(node) {              // 若节点非隐藏状态返回true
        var p = node.getBoundingClientRect();
        if ((p.width == 0 && p.height == 0) || (p.top == 0 && p.bottom == 0)) {
            return false;
        }
        return true;
    }

    function changeMode() {                 // 切换预览触发模式
        onClick = (onClick == 0) ? 1 : 0;
        const str = _txt.ntc[_L].replace('--swc--', _txt.swc[_L][onClick]);
        popupNotice(str);
        document.body.classList.toggle("__link_pre_clk", (onClick == 1));
        GM_setValue(is_onClick, onClick);
    }

    function changeDfpin() {                // 切换默认顶住状态
        default_pinned = (default_pinned == 0) ? 1 : 0;
        const str = _txt.ntp[_L].replace('--swp--', _txt.swp[_L][default_pinned]);
        popupNotice(str);
        togglePin(default_pinned == 1);
        GM_setValue(is_default_pin, default_pinned);
    }

    function changeStatus() {               // 切换启用状态
        default_enabled = (default_enabled == 0) ? 1 : 0;
        const str = _txt.ntt[_L].replace('--swt--', _txt.swt[_L][default_enabled]);
        popupNotice(str);
        document.body.classList.toggle("__link_pre_disable", (default_enabled == 0));
        GM_setValue(is_default_enable, default_enabled);
    }

    function setDWSdlg() {                  // 设置/删除当前域名下预览窗大小
        let dg, maintxt, wxh;
        wxh = winpos.w + " x " + winpos.h;
        if (matchedDWS > -1) {              // 若已有匹配,则多显示“删除”按钮
            maintxt = (_txt.msd[_L] + _txt.mdd[_L]).replaceAll("--dm--", site).replaceAll("--wh--", wxh);
            dg = popupDialog(maintxt, [_txt.bsd[_L][0], _txt.bud[_L][0], _txt.bcd[_L]]);
        } else {                            // 若无匹配,只显示“设置”按钮和“取消”按钮
            maintxt = _txt.msd[_L].replaceAll("--dm--", site).replaceAll("--wh--", wxh);
            dg = popupDialog(maintxt, [_txt.bsd[_L][0], _txt.bcd[_L]]);
        }
        dialogDWS = dg.dialog;
        btnsDWS = dg.buttons;
        btnsDWS[0].addEventListener("click", () => {    // “设置”按钮,保存设置
            setDWS(true);
            btnsDWS[0].textContent = _txt.bsd[_L][1];
            setTimeout(() => {
                removeNode(dialogDWS);
            }, 2000);
        });
        if (matchedDWS > -1) {
            btnsDWS[1].addEventListener("click", () => {    // “删除”按钮,删除设置
                setDWS(false);
                btnsDWS[1].textContent = _txt.bud[_L][1];
                setTimeout(() => {
                    removeNode(dialogDWS);
                }, 2000);
            });
        }
        btnsDWS[btnsDWS.length - 1].addEventListener("click", () => {
            removeNode(dialogDWS);
        });
    }

    function setDWS(isSet) {                // 设置/删除预览窗大小命令
        if (isSet) {
            saveDomainWinSize(site, winpos.w, winpos.h);    // 添加设置
        } else {
            perDomainWinSize = perDomainWinSize.filter((item, ind) => {  // 从数组中删除设置
                if (ind !== matchedDWS) {
                    return true;
                } else {
                    return false;
                }
            });
            matchedDWS = -1;
        }
        GM_setValue(pDWS, JSON.stringify(perDomainWinSize));    // 写入储存
    }

    function saveDomainWinSize(domain, w, h) {
        if (matchedDWS == -1) {
            perDomainWinSize.push({ domain: domain, w: w, h: h });
            matchedDWS = perDomainWinSize.length - 1;
        } else {
            perDomainWinSize[matchedDWS].w = w;
            perDomainWinSize[matchedDWS].h = h;
        }
    }

    function keyhandler(evt) {
        var fullkey = get_key(evt);
        if (fullkey == toggleHotkey) {
            changeStatus();
        }
    }

    // 按键evt.which转换为键名
    function get_key(evt) {
        const keyCodeStr = {			//key press 事件返回的which代码对应按键键名对应表对象
            8: 'BAC',
            9: 'TAB',
            10: 'RET',
            13: 'RET',
            27: 'ESC',
            33: 'PageUp',
            34: 'PageDown',
            35: 'End',
            36: 'Home',
            37: 'Left',
            38: 'Up',
            39: 'Right',
            40: 'Down',
            45: 'Insert',
            46: 'Delete',
            112: 'F1',
            113: 'F2',
            114: 'F3',
            115: 'F4',
            116: 'F5',
            117: 'F6',
            118: 'F7',
            119: 'F8',
            120: 'F9',
            121: 'F10',
            122: 'F11',
            123: 'F12'
        };
        const whichStr = {
            32: 'SPC'
        };
        var key = String.fromCharCode(evt.which),
            ctrl = evt.ctrlKey ? 'C-' : '',
            meta = (evt.metaKey || evt.altKey) ? 'M-' : '';
        if (!evt.shiftKey) {
            key = key.toLowerCase();
        }
        if (evt.ctrlKey && evt.which >= 186 && evt.which < 192) {
            key = String.fromCharCode(evt.which - 144);
        }
        if (evt.key && evt.key !== 'Enter' && !/^U\+/.test(evt.key)) {
            key = evt.key;
        } else if (evt.which !== evt.keyCode) {
            key = keyCodeStr[evt.keyCode] || whichStr[evt.which] || key;
        } else if (evt.which <= 32) {
            key = keyCodeStr[evt.keyCode] || whichStr[evt.which];
        }
        return ctrl + meta + key;
    }

    function popupNotice(str) {             // 在页面内显示弹出通知
        const vpos = 0.7, hpos = 0.5;
        let notice = creaElemIn('div', document.body);
        notice.textContent = scriptName + ":" + str;
        notice.style = `
            transform: translate(-50%,-50%); position: fixed; left: ${hpos * 100 + '%'}; top: ${vpos * 100 + '%'};
            border: none; outline: 3px solid white; box-shadow: 0 4px 7px 4px #3f3f3f;
            background: black; color: white; font-family: Arial; font-size: 14pt;
            transition: opacity 0.5s; opacity: 0; padding: 20px; border-radius: 15px;
            z-index: 2000000;
        `;
        setTimeout(() => {
            notice.style.opacity = 0.8;
        }, 500);
        setTimeout(() => {
            notice.style.opacity = 0;
        }, 5500);
        setTimeout(() => {
            removeNode(notice);
        }, 6000);
    }

    function popupDialog(str, btns) {             // 在页面内显示对话框
        const vpos = 0.5, hpos = 0.5;
        let dialog = creaElemIn('div', document.body);
        dialog.id = "popupDialog";
        let txtdiv = creaElemIn('div', dialog);
        let btndiv = creaElemIn('div', dialog);
        txtdiv.textContent = scriptName + ":" + str;
        let buttons = [];
        for (let i = 0; i < btns.length; i++) {
            const btn = btns[i];
            buttons[i] = creaElemIn('div', btndiv);
            buttons[i].textContent = btns[i];
            buttons[i].style = `
                border: 2px solid white; border-radius: 7px; height: 40px; width: 120px; margin: 10px 20px;
                background: rgb(51, 51, 51); text-align: center; line-height: 36px; cursor: pointer;
            `;
        }

        dialog.style = `
            transform: translate(-50%,-50%); position: fixed; left: ${hpos * 100 + '%'}; top: ${vpos * 100 + '%'};
            border: none; outline: 3px solid white; box-shadow: 0 4px 7px 4px #3f3f3f;
            background: #646464; color: white; font-family: Arial; font-size: 14pt;
            transition: opacity 0.5s; opacity: 1; padding: 20px; border-radius: 15px;
            z-index: 2000000;
        `;
        txtdiv.style = `
            text-align = center; width: 100%;
        `;
        btndiv.style = `
            display: flex; width: 100%; padding: 15px;
        `;
        return { dialog, buttons };
    }

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

    function removeNode(node) {
        if (!!node.parentNode) {
            node.parentNode.removeChild(node);
        }
    }

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

    function getLastChildSize(elem) {       // 返回节点的最后一个子节点的实占位对象
        var cs = elem.childNodes;
        if (cs.length == 0) {
            return false;
        }
        var lc = cs[cs.length - 1];
        if (!lc) return false;
        if (lc.nodeType == 3 && lc.length > 0) {
            var crect = elem.getClientRects();
            var rp = elem.getBoundingClientRect().right;
            if (crect.length > 1) {             // 若文本折行导致多个 client 的 rect 存在,则取最后一个有大小的实占
                for (let i = crect.length - 1; i >= 0; i--) {
                    const p = crect[i];
                    if (p.width > 0 && p.height > 0) {
                        return {
                            w: p.width,
                            h: p.height,
                            t: p.top,
                            l: p.left,
                            r: rp - p.right
                        };
                    }
                }
            } else {
                return getTextNodeSize(lc, rp);     // 不折行则直接取文本实占
            }
        } else {
            return getTrueSize(lc);             // 非文本或空文本则取整个实占
        }
    }

    // 输入元素,返回实占位对象{元素可见的宽、高、顶、左、一层父元素右侧余量}
    function getTrueSize(elem, posiz) {
        // console.log(elem);
        if (!posiz) {
            var p = elem.getBoundingClientRect();
            posiz = {
                w: p.width,
                h: p.height,
                t: p.top,
                l: p.left,
                r: 0,
            };
        }
        var pp = elem.parentNode.getBoundingClientRect();
        var pr = posiz.l + posiz.w, pb = posiz.t + posiz.h;
        if (posiz.r == 0) posiz.r = pp.right - pr;                  // 父元素对当前元素的右侧余量
        var isvi = {
            l: posiz.l < pp.right && posiz.l >= pp.left,            // 子左在父左右之间
            r: pr > pp.left && pr <= pp.right,                      // 子右在父左右之间
            t: posiz.t < pp.bottom && posiz.t >= pp.top,            // 子顶在父顶底之间
            b: pb > pp.top && pb <= pp.bottom                       // 子底在父顶底之间
        };
        if (isvi.l && isvi.r && isvi.t && isvi.b) {                 // 子全在父之内,则返回子占位
            return posiz;
        } else {
            var ppl = (isvi.l) ? posiz.l : pp.left;                 // 确定可见四边(在父之内按子,否则按父)
            var ppt = (isvi.t) ? posiz.t : pp.top;
            var ppr = (isvi.r) ? posiz.l + posiz.w : pp.right;
            var ppb = (isvi.b) ? posiz.t + posiz.h : pp.bottom;
            return getTrueSize(elem.parentNode, {
                w: ppr - ppl,
                h: ppb - ppt,
                t: ppt,
                l: ppl,
                r: posiz.r
            });
        }
    }

    // 输入文本节点,返回实占位对象{文本节点的宽、高、顶、左、一层父元素右侧余量}
    function getTextNodeSize(textNode, parentright) {
        if (textNode.nodeType !== 3) {
            return false;
        }
        if (document.createRange) {
            var range = document.createRange();
            range.selectNodeContents(textNode);
            if (range.getBoundingClientRect) {
                var rect = range.getBoundingClientRect();
                if (rect) {
                    var pos = {
                        w: rect.width,
                        h: rect.height,
                        t: rect.top,
                        l: rect.left,
                        r: parentright - rect.right
                    };
                    return getTrueSize(textNode, pos);
                }
            }
        }
        return false;
    }

    // 将target绑定到对象obj上,使拖动handle时,实现绑定的响应功能
    // 输入:目标元素target,拖动手柄handle
    // 用法:obj=endrag(target,handle);
    // obj.hook('__drag_begin',fn); 其中 fn 为拖动开始时需要响应的功能
    // obj.hook('__dragging',fn); 其中 fn 为拖动过程中需要响应的功能
    // obj.hook('__drag_end',fn); 其中 fn 为拖动结束时需要响应的功能
    // obj.isDragging:用于判断是否为拖动中状态
    // obj.position{x,y,_x,_y}:
    //   (x, y)为鼠标viewport位置,(_x, _y)为随(x, y)增减变化的值,
    //    可在拖动开始时输入,可在拖动过程中和拖动结束时输出,输入输出同步为乘负系数则增减反向
    function endrag(target, handle) {
        var isDragging;
        endrag = function (target, handle) {
            return new endrag.proto(target, handle);
        }
        endrag.proto = function (target, handle) {
            var self = this;
            this.target = target;
            this.style = target.style;
            this.handle = handle;
            this.isDragging = false; // 实例属性
            this.position = { x: 0, y: 0, _x: 0, _y: 0 }; // 实例属性,初始化为0
            this.drag_begin = function (e) { self.__drag_begin(e); };
            this.dragging = function (e) { self.__dragging(e); };
            this.drag_end = function (e) { self.__drag_end(e); };
            this.handle.addEventListener('mousedown', this.drag_begin, false); //only drag on handler
            document.addEventListener('mousemove', this.dragging, false);
            document.addEventListener('mouseup', this.drag_end, false);
        };
        endrag.proto.prototype = {
            __drag_begin: function (e) {
                if (e.button == 0) {
                    this.isDragging = isDragging = true;
                    this.position.x = e.clientX;
                    this.position.y = e.clientY;
                    e.preventDefault();
                }
            },
            __dragging: function (e) {
                if (!this.isDragging) return;               // 这句无法移到this.dragging的绑定上,否则一直不会进入hook 到 __dragging方法
                var x = Math.floor(e.clientX), y = Math.floor(e.clientY);
                var x_border = window.innerWidth - 30, y_border = window.innerHeight - 30;
                x = Math.max(0, Math.min(x, x_border));     // 防止超出屏幕
                y = Math.max(0, Math.min(y, y_border));
                this.position._x += (x - this.position.x);
                this.position._y += (y - this.position.y);
                this.position.x = x;
                this.position.y = y;
            },
            __drag_end: function (e) {
                if (e.button === 0 && this.isDragging) {
                    this.isDragging = false;
                }
            },
            hook: function (method, func) {
                if (typeof this[method] === 'function') {
                    var orimethod = this[method];
                    this[method] = function () {
                        if (func.apply(this, arguments) === false) {
                            return;
                        }
                        orimethod.apply(this, arguments);
                    };
                }
            }
        };
        return endrag(target, handle);
    }



})();