BilibiliCover

B站显示视频封面

À partir de 2018-02-10. Voir la dernière version.

// ==UserScript==
// @name         BilibiliCover
// @version      2.0.1
// @description  B站显示视频封面
// @author       AnnAngela
// @namespace    https://greasyfork.org/users/129402
// @mainpage     https://greasyfork.org/zh-CN/scripts/33411-bilibilicover
// @supportURL   https://greasyfork.org/zh-CN/scripts/33411-bilibilicover/feedback
// @license      GNU General Public License v3.0 or later
// @compatible   chrome
// @compatible   firefox
// @compatible   opera
// @compatible   safari
// @match        *://www.bilibili.com/video/av*
// @match        *://www.bilibili.com/watchlater*
// @match        *://www.bilibili.com/bangumi/play/*
// @match        *://live.bilibili.com/*
// @run-at       document-start
// @grant        unsafeWindow
// @grant        GM_xmlhttpRequest
// @noframes
// @icon         
// @icon64       
// ==/UserScript==
unsafeWindow.addEventListener("load", function() {
    var LF = String.fromCharCode(10),
        XX = String.fromCharCode(47);
    var helper = {
        coverImage: function coverImage(url) {
            /* 本函数来自 https://greasyfork.org/zh-CN/scripts/30714-获取哔哩哔哩视频的封面图片-get-bilibili-cover-image/code?version=202372 特此感谢*/
            var coverImageBigUrl = url;
            // 去除url中的裁剪标识
            if (url.indexOf("@") > -1) {
                //处理以@做裁剪标识的url
                coverImageBigUrl = url.split("@")[0];
            }
            if (url.indexOf("jpg_") > -1) {
                //处理以_做裁剪标识的url
                coverImageBigUrl = url.split("jpg_")[0] + "jpg";
            }
            if (url.indexOf("png_") > -1) {
                //处理以_做裁剪标识的url
                coverImageBigUrl = url.split("png_")[0] + "png";
            }
            if (url.indexOf("/320_200/") > -1) {
                //有时裁剪标识是在后缀名之前的 目前主要发现的是“番剧”板块的列表里有,但尚不清楚其他地方的情况
                coverImageBigUrl = url.replace("/320_200", "");
            }
            if (coverImageBigUrl.substring(0, 2) === XX + XX) coverImageBigUrl = "https:" + coverImageBigUrl;
            else if (coverImageBigUrl.substring(0, 5) === "http:") coverImageBigUrl = coverImageBigUrl.replace("http:", "https:");
            return coverImageBigUrl;
        },
        window: undefined,
        openWin: function(win, src) {
            if (this.window) this.setImg(src);
            else {
                var self = this,
                    doc = win.document;
                var w = win.innerWidth || doc.docElement.clientWidth || doc.body.clientWidth,
                    h = win.innerHeight || doc.docElement.clientHeight || doc.body.clientHeight;
                self.window = window.open("about:blank", "bilibiliCover", "location=1,scrollbars=1,channelmode=1,width=" + w * .8 + ",height=" + h * .95 + ",left=" + w * .1 + ",top=" + h * .1);
                setTimeout(function() {
                	self.window.document.title = 'BilibiliCover - 封面获取窗口';
                    self.window.document.body.innerHTML = '<div style="text-align: start;">视频封面地址:</div><textarea readonly="readonly" style="width: 100%; height: 1.1em; font-size: 24px; box-sizing: content-box; overflow: hidden; background-color: white; color: initial;"></textarea><hr><img src="' + src + '" style="max-width: 100%; height: auto;"><p id="realsize" style="text-align: center;"></p>';
                    self.window.document.body.innerHTML += '<style> a { cursor: pointer; background-position: center right;background-repeat: no-repeat;background-image: -webkit-linear-gradient(transparent, transparent), url(data:image/svg+xml,%3C%3Fxml%20version%3D%221.0%22%20encoding%3D%22UTF-8%22%3F%3E%3Csvg%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20width%3D%2210%22%20height%3D%2210%22%3E%3Cg%20transform%3D%22translate%28-826.429%20-698.791%29%22%3E%3Crect%20width%3D%225.982%22%20height%3D%225.982%22%20x%3D%22826.929%22%20y%3D%22702.309%22%20fill%3D%22%23fff%22%20stroke%3D%22%2306c%22%2F%3E%3Cg%3E%3Cpath%20d%3D%22M831.194%20698.791h5.234v5.391l-1.571%201.545-1.31-1.31-2.725%202.725-2.689-2.689%202.808-2.808-1.311-1.311z%22%20fill%3D%22%2306f%22%2F%3E%3Cpath%20d%3D%22M835.424%20699.795l.022%204.885-1.817-1.817-2.881%202.881-1.228-1.228%202.881-2.881-1.851-1.851z%22%20fill%3D%22%23fff%22%2F%3E%3C%2Fg%3E%3C%2Fg%3E%3C%2Fsvg%3E); background-image: linear-gradient(transparent, transparent), url(data:image/svg+xml,%3C%3Fxml%20version%3D%221.0%22%20encoding%3D%22UTF-8%22%3F%3E%3Csvg%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20width%3D%2210%22%20height%3D%2210%22%3E%3Cg%20transform%3D%22translate%28-826.429%20-698.791%29%22%3E%3Crect%20width%3D%225.982%22%20height%3D%225.982%22%20x%3D%22826.929%22%20y%3D%22702.309%22%20fill%3D%22%23fff%22%20stroke%3D%22%2306c%22%2F%3E%3Cg%3E%3Cpath%20d%3D%22M831.194%20698.791h5.234v5.391l-1.571%201.545-1.31-1.31-2.725%202.725-2.689-2.689%202.808-2.808-1.311-1.311z%22%20fill%3D%22%2306f%22%2F%3E%3Cpath%20d%3D%22M835.424%20699.795l.022%204.885-1.817-1.817-2.881%202.881-1.228-1.228%202.881-2.881-1.851-1.851z%22%20fill%3D%22%23fff%22%2F%3E%3C%2Fg%3E%3C%2Fg%3E%3C%2Fsvg%3E); padding-right: 13px; } </style>';
                    self.window.document.body.innerHTML += '<p style="display: flex; flex-wrap: nowrap; justify-content: space-around;"><a target="_blank" data-href="https://image.baidu.com/pcdutu?queryImageUrl=%s">Baidu识图搜索</a><a target="_blank" data-href="https://www.google.com/searchbyimage?image_url=%s&encoded_image=&image_content=&filename=&hl=zh-CN">Google识图搜索</a></p>';
                    self.window.document.body.style.textAlign = "center";
                    self.setImg(src)
                    var t = self.window.document.querySelector("textarea");
                    t.addEventListener("mouseup", function(e) {
                        if (e.which !== 1) return;
                        var selection = self.window.getSelection();
                        if (selection.toString() !== "") return;
                        this.focus();
                        this.select();
                    });
                    Array.from(self.window.document.querySelectorAll("a")).forEach(function(ele) {
                        ele.addEventListener("click", function() {
                            window.open(ele.dataset.href.replace("%s", encodeURIComponent(self.window.document.querySelector("textarea").value)), "_blank").focus();
                        });
                    });
                    self.window.addEventListener("beforeunload", function() {
                        self.window = undefined;
                    });
                    self.window.focus();
                }, 0);
            };
        },
        setImg: function setImg(src) {
            if (!this.window) return;
            var img = this.window.document.querySelector("img");
            this.window.document.querySelector("img").src = src;
            this.window.document.querySelector("textarea").value = src;
            this.setNaturalSize(img, this.window.document.querySelector("#realsize"));
            this.window.focus();
        },
        setNaturalSize: function setNaturalSize(img, node) {
            var self = this;
            if (img.naturalWidth > 0 && img.naturalHeight > 0) node.innerText = "(" + img.naturalWidth + "×" + img.naturalHeight + ")";
            else setTimeout(function() {
                self.setNaturalSize(img, node);
            }, 100);
        },
        closeWin: function focusWin() {
            if (this.window) this.window.close();
        }
    }
    var body = unsafeWindow.document.body,
        html = unsafeWindow.document.documentElement;
    var innerWidth = unsafeWindow.innerWidth;
    var scrollbarWidth = 0;
    switch ('scroll') {
        case getComputedStyle(body).overflowY:
            scrollbarWidth = innerWidth - body.clientWidth;
            break;
        case getComputedStyle(html).overflowY:
            scrollbarWidth = innerWidth - html.clientWidth;
            break;
        default:
            var backup = body.style.overflowY;
            body.style.overflowY = 'scroll';
            scrollbarWidth = unsafeWindow.innerWidth - body.clientWidth;
            body.style.overflowY = backup;
    }
    if (unsafeWindow.location.host.includes('www.bilibili.com'))(function loop() {
        var doc = unsafeWindow.document;
        var plw = doc.querySelector(".player-wrapper, .video-box-module > .bili-wrapper");
        if (!plw) return setTimeout(loop, 1e3);
        var sbw = scrollbarWidth;

        var img;
        img = doc.createElement("img");
        img.id = "cover_img";
        setInterval(function() {
            try {
                doc.querySelector("#cover_img").src = helper.coverImage(doc.querySelector(".cover_image, .bilibili-player-auxiliary-area .bilibili-player-watchlater-item[data-state-play=true] .bilibili-player-watchlater-cover-cell img, .bangumi-info-wrapper .info-cover img").src);
            } catch (_) {
                console.info("bilibiliCover", "no img");
            }
        }, 1e3);
        plw.appendChild(img);
        img.style.display = "block";
        img.style.position = "absolute";
        img.style.cursor = "pointer";

        function calc() {
            var flag = true;
            var bsn = Array.from(plw.children).filter(function(ele, i) {
                return plw.children[i - 1] && ele.nodeName !== "IMG" && (plw.children[i - 1].id === "bofqi" || /scontent|player-content|bangumi-player/i.test(plw.children[i - 1].classList.value));
            })[0];
            if (!bsn) {
                bsn = plw.querySelector(".scontent, .player-content, .bangumi-player");
                flag = false;
            }
            var ofs = getComputedStyle(bsn);
            var ofl = parseInt(ofs.width) + parseInt(ofs.marginLeft) + parseInt(ofs.paddingLeft) + parseInt(ofs.borderLeftWidth),
                ofb = flag ? parseInt(ofs.height) + parseInt(ofs.marginTop) + parseInt(ofs.paddingTop) + parseInt(ofs.borderTopWidth) : parseInt(getComputedStyle(doc.querySelector("#bofqi")).marginBottom),
                wdt = ofs.width;
            var w = window.innerWidth || doc.docElement.clientWidth || doc.body.clientWidth;
            img.style.left = "calc(" + ofl + "px + 1em)";
            img.style.bottom = "calc(" + ofb + "px + 1em)";
            img.style.width = "calc(" + w + "px / 2 - " + wdt + " / 2 - 2em - " + sbw / 2 + "px)";
        }
        calc();
        img.title = "此处是本视频封面大图!" + LF + LF + "右键菜单可复制图片大图地址," + LF + "左键单击可在新窗口查看大图!";
        window.addEventListener("resize", function() {
            calc();
        });
        img.addEventListener("click", function() {
            var src = this.src;
            helper.openWin(window, src);
        });
    })();
    else if (unsafeWindow.location.host.includes('live.bilibili.com'))(function loop() {
        var container = unsafeWindow.document.querySelector('.seeds-wrap');
        if (!container) return setTimeout(loop, 100);
        var link = unsafeWindow.document.createElement('div');
        link.style.display = 'inline-block';
        link.style.marginRight = '1em';
        link.innerHTML = '<a href="javascript:void(0);" style="color: #23ade5;">查看封面</a>';
        container.insertBefore(link, container.firstChild);
        var error = function() {
            link.innerHTML = '<span style="color: red">封面获取失败=。=</span>';
        };
        link.querySelector('a').addEventListener('click', function() {
            var roomid = unsafeWindow.location.pathname.match(/^\/(\d+)/)[1];
            if (roomid) {
                GM_xmlhttpRequest({
                    url: 'https://search.bilibili.com/api/search?search_type=live&keyword=' + roomid + '&from_source=livetop_search',
                    method: 'GET',
                    onerror: error,
                    onload: function(res) {
                        try {
                            response = JSON.parse(res.responseText);
                        } catch (e) {
                            response = false;
                        }
                        if (!response) error();
                        var data = response.result.live_room;
                        var cover;
                        data.forEach(function(info) {
                            if (info.roomid = roomid && info.user_cover) cover = helper.coverImage(info.user_cover);
                        });
                        if (cover) {
                            helper.openWin(unsafeWindow, cover);
                        }
                    }
                })
            }
        });
    })()
    window.addEventListener("beforeunload", function() {
        helper.closeWin();
    });
});