Greasy Fork

Vdownload

音视频下载器

目前为 2022-08-10 提交的版本。查看 最新版本

// ==UserScript==
// @name         Vdownload
// @name:zh-CN   下载器(个人使用)
// @namespace    http://tampermonkey.net/
// @version      1.0
// @description  音视频下载器
// @author       浣熊
// @match        *://*/*
// @exclude      *://*.runoob.com/*
// @icon         
// @grant        none
// @run-at       document-start
// @require      https://cdn.bootcdn.net/ajax/libs/jquery/2.2.2/jquery.min.js
// @license      https://github.com/1024Vip/Vdownload/blob/main/LICENSE
// ==/UserScript==

(function () {
    window.autoDownload = 0;
    window.isComplete = 0;
    window.audio = [];
    window.video = [];
    window.downloadAll = 0;
    window.quickPlay = 1.0;
    var isDelTabPage = 0; //1关闭;
    var downloadName = document.title;
    var isVideo = 0;//1存在视频
    var loadTrue = false;
    function Insertstyle(css, ref) {
        var insertAt, head, style;
        void 0 === ref && (ref = {}), insertAt = ref.insertAt, css && "undefined" != typeof document && (head = document.head || document.getElementsByTagName("head")[0],
                                                                                                         (style = document.createElement("style")).type = "text/css", "top" === insertAt && head.firstChild ? head.insertBefore(style, head.firstChild) : head.appendChild(style),
                                                                                                         style.styleSheet ? style.styleSheet.cssText = css : style.appendChild(document.createTextNode(css)));
    }
    function XF() {
        let parentID = 'VdownLoad-nav';
        //鼠标移动
        var scrollMove = function () {
            window.onscroll = window.onload = function () {
                var scrollsidebar = document.getElementById(parentID);
                var scrolltop = document.body.scrollTop || document.documentElement.scrollTop;
                startMove(
                    parseInt(
                        (document.documentElement.clientHeight -
                         scrollsidebar.offsetHeight) / 2 + scrolltop
                    )
                );
            };
        };
        //获取元素
        var dv = document.getElementById(parentID);
        var x = 0;
        var y = 0;
        var l = 0;
        var t = 0;
        var isDown = false;
        //鼠标按下事件
        dv.onmousedown = function (e) {
            //获取x坐标和y坐标
            x = e.clientX;
            y = e.clientY;

            //获取左部和顶部的偏移量
            l = dv.offsetLeft;
            t = dv.offsetTop;
            //开关打开
            isDown = true;
            //设置样式
            dv.style.cursor = 'move';
        }
        //鼠标移动
        dv.onmousemove = function (e) {
            if (isDown == false) {
                return;
            }
            //获取x和y
            var nx = e.clientX;
            var ny = e.clientY;
            //计算移动后的左偏移量和顶部的偏移量
            var nl = nx - (x - l);
            var nt = ny - (y - t);

            dv.style.left = nl + 'px';
            dv.style.top = nt + 'px';
        }
        //鼠标抬起事件
        dv.onmouseup = function () {
            //开关关闭
            isDown = false;
            dv.style.cursor = 'default';
        }
    };
    const SetplaybackRate = (e) => {
        var pageX = e.pageX,
            pageY = e.pageY;
        let ID = 'div_play';
        if ($('#' + ID).length > 0) {
            $('#' + ID).remove();
        }
        let SD = 1;
        if (document.querySelector('video') !== null)
            SD = document.querySelector('video').playbackRate;
        $('body').prepend(`
            <div id="${ID}" class="div_play" style="display:none">
                <input type="number" min="0" max="16" value="${SD}" id="numPlay">
                <input type="button" value="取消" onclick="$('#${ID}').fadeOut(()=>$(this).parent().remove())">
                <input type="button" value="确定" id="playSDSure"">
            </div>
        `)
        $(document).on('click', '#playSDSure', () => { playSDSure('numPlay') })
        // onclick="playSDSure('numPlay')
        let py = document.getElementById(ID);
        $(py).fadeIn();
        py.style.left = pageX + 'px';
        py.style.top = pageY + 'px';
    }
    const playSDSure = (id) => {
        let val = parseFloat($('#' + id).val());
        val = val === 0 ? 1 : val;
        if (document.querySelector('video') !== null)
            document.querySelector('video').playbackRate = val;
        console.log(val);
        $('#' + id).parent().fadeOut(() => { $(this).remove() });
    }
    const Vdown = (isDel) => {
        isDelTabPage = isDel;
        window.autoDownload = 1;
    }

    const XFrun = () => {
        var css_asd1 = `
        .div_play {
            background-color: #98cad9;
            opacity: .8;
            padding: 10px;
            border-radius: 10px;
            position: fixed;
            z-index: 999999;
        }
        html .hxaside-nav {
            -ms-text-size-adjust: 100%;
            -webkit-text-size-adjust: 100%;
            -webkit-font-smoothing: antialiased;
            font-size: 62.5%
        }

        body .hxaside-nav {
            font-family: "Helvetica Neue", Helvetica, "Microsoft YaHei", Arial, sans-serif;
            margin: 0;
            font-size: 1.6rem;
            color: #4e546b
        }

        .hxaside-nav {
            position: fixed;
            bottom: 0;
            left: -47px;
            width: 260px;
            height: 260px;
            -webkit-filter: url(#goo);
            filter: url(#goo);
            -ms-user-select: none;
            -moz-user-select: none;
            -webkit-user-select: none;
            user-select: none;
            opacity: .75;
            z-index: 999998
        }

        .hxaside-nav.no-filter {
            -webkit-filter: none;
            filter: none
        }

        .hxaside-nav .hxaside-menu {
            position: absolute;
            width: 70px;
            height: 70px;
            -webkit-border-radius: 50%;
            border-radius: 50%;
            background: #65c5da;
            left: -95px;
            top: 0;
            right: 0;
            bottom: 0;
            margin: auto;
            text-align: center;
            line-height: 70px;
            color: #fff;
            font-size: 20px;
            z-index: 1;
            cursor: move
        }

        .hxaside-nav .hxmenu-item {
            position: absolute;
            width: 60px;
            height: 60px;
            background-color: #e6b451;
            left: -95px;
            top: 0;
            right: 0;
            bottom: 0;
            margin: auto;
            line-height: 60px;
            text-align: center;
            -webkit-border-radius: 50%;
            border-radius: 50%;
            text-decoration: none;
            color: #fff;
            -webkit-transition: background .5s, -webkit-transform .6s;
            transition: background .5s, -webkit-transform .6s;
            -moz-transition: transform .6s, background .5s, -moz-transform .6s;
            transition: transform .6s, background .5s;
            transition: transform .6s, background .5s, -webkit-transform .6s, -moz-transform .6s;
            font-size: 14px;
            -webkit-box-sizing: border-box;
            -moz-box-sizing: border-box;
            box-sizing: border-box
        }

        .hxaside-nav .hxmenu-item:hover {
            background: #a9c734
        }

        .hxaside-nav .hxmenu-line {
            line-height: 20px;
            padding-top: 10px
        }

        .hxaside-nav:hover {
            opacity: 1
        }

        .hxaside-nav:hover .hxaside-menu {
            -webkit-animation: jello 1s;
            -moz-animation: jello 1s;
            animation: jello 1s
        }

        .hxaside-nav:hover .hxmenu-first {
            -webkit-transform: translate3d(0, -135%, 0);
            -moz-transform: translate3d(0, -135%, 0);
            transform: translate3d(0, -135%, 0)
        }

        .hxaside-nav:hover .hxmenu-second {
            -webkit-transform: translate3d(120%, -70%, 0);
            -moz-transform: translate3d(120%, -70%, 0);
            transform: translate3d(120%, -70%, 0)
        }

        .hxaside-nav:hover .hxmenu-third {
            -webkit-transform: translate3d(120%, 70%, 0);
            -moz-transform: translate3d(120%, 70%, 0);
            transform: translate3d(120%, 70%, 0)
        }

        .hxaside-nav:hover .hxmenu-fourth {
            -webkit-transform: translate3d(0, 135%, 0);
            -moz-transform: translate3d(0, 135%, 0);
            transform: translate3d(0, 135%, 0)
        }

        @-webkit-keyframes jello {

            from,
            11.1%,
            to {
                -webkit-transform: none;
                transform: none
            }

            22.2% {
                -webkit-transform: skewX(-12.5deg) skewY(-12.5deg);
                transform: skewX(-12.5deg) skewY(-12.5deg)
            }

            33.3% {
                -webkit-transform: skewX(6.25deg) skewY(6.25deg);
                transform: skewX(6.25deg) skewY(6.25deg)
            }

            44.4% {
                -webkit-transform: skewX(-3.125deg) skewY(-3.125deg);
                transform: skewX(-3.125deg) skewY(-3.125deg)
            }

            55.5% {
                -webkit-transform: skewX(1.5625deg) skewY(1.5625deg);
                transform: skewX(1.5625deg) skewY(1.5625deg)
            }

            66.6% {
                -webkit-transform: skewX(-.78125deg) skewY(-.78125deg);
                transform: skewX(-.78125deg) skewY(-.78125deg)
            }

            77.7% {
                -webkit-transform: skewX(0.390625deg) skewY(0.390625deg);
                transform: skewX(0.390625deg) skewY(0.390625deg)
            }

            88.8% {
                -webkit-transform: skewX(-.1953125deg) skewY(-.1953125deg);
                transform: skewX(-.1953125deg) skewY(-.1953125deg)
            }
        }

        @-moz-keyframes jello {

            from,
            11.1%,
            to {
                -moz-transform: none;
                transform: none
            }

            22.2% {
                -moz-transform: skewX(-12.5deg) skewY(-12.5deg);
                transform: skewX(-12.5deg) skewY(-12.5deg)
            }

            33.3% {
                -moz-transform: skewX(6.25deg) skewY(6.25deg);
                transform: skewX(6.25deg) skewY(6.25deg)
            }

            44.4% {
                -moz-transform: skewX(-3.125deg) skewY(-3.125deg);
                transform: skewX(-3.125deg) skewY(-3.125deg)
            }

            55.5% {
                -moz-transform: skewX(1.5625deg) skewY(1.5625deg);
                transform: skewX(1.5625deg) skewY(1.5625deg)
            }

            66.6% {
                -moz-transform: skewX(-.78125deg) skewY(-.78125deg);
                transform: skewX(-.78125deg) skewY(-.78125deg)
            }

            77.7% {
                -moz-transform: skewX(0.390625deg) skewY(0.390625deg);
                transform: skewX(0.390625deg) skewY(0.390625deg)
            }

            88.8% {
                -moz-transform: skewX(-.1953125deg) skewY(-.1953125deg);
                transform: skewX(-.1953125deg) skewY(-.1953125deg)
            }
        }

        @keyframes jello {

            from,
            11.1%,
            to {
                -webkit-transform: none;
                -moz-transform: none;
                transform: none
            }

            22.2% {
                -webkit-transform: skewX(-12.5deg) skewY(-12.5deg);
                -moz-transform: skewX(-12.5deg) skewY(-12.5deg);
                transform: skewX(-12.5deg) skewY(-12.5deg)
            }

            33.3% {
                -webkit-transform: skewX(6.25deg) skewY(6.25deg);
                -moz-transform: skewX(6.25deg) skewY(6.25deg);
                transform: skewX(6.25deg) skewY(6.25deg)
            }

            44.4% {
                -webkit-transform: skewX(-3.125deg) skewY(-3.125deg);
                -moz-transform: skewX(-3.125deg) skewY(-3.125deg);
                transform: skewX(-3.125deg) skewY(-3.125deg)
            }

            55.5% {
                -webkit-transform: skewX(1.5625deg) skewY(1.5625deg);
                -moz-transform: skewX(1.5625deg) skewY(1.5625deg);
                transform: skewX(1.5625deg) skewY(1.5625deg)
            }

            66.6% {
                -webkit-transform: skewX(-.78125deg) skewY(-.78125deg);
                -moz-transform: skewX(-.78125deg) skewY(-.78125deg);
                transform: skewX(-.78125deg) skewY(-.78125deg)
            }

            77.7% {
                -webkit-transform: skewX(0.390625deg) skewY(0.390625deg);
                -moz-transform: skewX(0.390625deg) skewY(0.390625deg);
                transform: skewX(0.390625deg) skewY(0.390625deg)
            }

            88.8% {
                -webkit-transform: skewX(-.1953125deg) skewY(-.1953125deg);
                -moz-transform: skewX(-.1953125deg) skewY(-.1953125deg);
                transform: skewX(-.1953125deg) skewY(-.1953125deg)
            }
        }

        .animated {
            -webkit-animation-duration: 1s;
            -moz-animation-duration: 1s;
            animation-duration: 1s;
            -webkit-animation-fill-mode: both;
            -moz-animation-fill-mode: both;
            animation-fill-mode: both
        }

        @-webkit-keyframes divhxUp {

            from,
            60%,
            75%,
            90%,
            to {
                -webkit-animation-timing-function: cubic-bezier(0.215, .61, .355, 1);
                animation-timing-function: cubic-bezier(0.215, .61, .355, 1)
            }

            from {
                opacity: 0;
                -webkit-transform: translate3d(0, 800px, 0);
                transform: translate3d(0, 800px, 0)
            }

            60% {
                opacity: 1;
                -webkit-transform: translate3d(0, -20px, 0);
                transform: translate3d(0, -20px, 0)
            }

            75% {
                -webkit-transform: translate3d(0, 10px, 0);
                transform: translate3d(0, 10px, 0)
            }

            90% {
                -webkit-transform: translate3d(0, -10px, 0);
                transform: translate3d(0, -10px, 0)
            }

            to {
                -webkit-transform: translate3d(0, 0, 0);
                transform: translate3d(0, 0, 0)
            }
        }

        @-moz-keyframes divhxUp {

            from,
            60%,
            75%,
            90%,
            to {
                -moz-animation-timing-function: cubic-bezier(0.215, .61, .355, 1);
                animation-timing-function: cubic-bezier(0.215, .61, .355, 1)
            }

            from {
                opacity: 0;
                -moz-transform: translate3d(0, 800px, 0);
                transform: translate3d(0, 800px, 0)
            }

            60% {
                opacity: 1;
                -moz-transform: translate3d(0, -20px, 0);
                transform: translate3d(0, -20px, 0)
            }

            75% {
                -moz-transform: translate3d(0, 10px, 0);
                transform: translate3d(0, 10px, 0)
            }

            90% {
                -moz-transform: translate3d(0, -5px, 0);
                transform: translate3d(0, -5px, 0)
            }

            to {
                -moz-transform: translate3d(0, 0, 0);
                transform: translate3d(0, 0, 0)
            }
        }

        @keyframes divhxUp {

            from,
            60%,
            75%,
            90%,
            to {
                -webkit-animation-timing-function: cubic-bezier(0.215, .61, .355, 1);
                -moz-animation-timing-function: cubic-bezier(0.215, .61, .355, 1);
                animation-timing-function: cubic-bezier(0.215, .61, .355, 1)
            }

            from {
                opacity: 0;
                -webkit-transform: translate3d(0, 800px, 0);
                -moz-transform: translate3d(0, 800px, 0);
                transform: translate3d(0, 800px, 0)
            }

            60% {
                opacity: 1;
                -webkit-transform: translate3d(0, -20px, 0);
                -moz-transform: translate3d(0, -20px, 0);
                transform: translate3d(0, -20px, 0)
            }

            75% {
                -webkit-transform: translate3d(0, 10px, 0);
                -moz-transform: translate3d(0, 10px, 0);
                transform: translate3d(0, 10px, 0)
            }

            90% {
                -webkit-transform: translate3d(0, -5px, 0);
                -moz-transform: translate3d(0, -5px, 0);
                transform: translate3d(0, -5px, 0)
            }

            to {
                -webkit-transform: translate3d(0, 0, 0);
                -moz-transform: translate3d(0, 0, 0);
                transform: translate3d(0, 0, 0)
            }
        }

        .divhxUp {
            -webkit-animation-name: divhxUp;
            -moz-animation-name: divhxUp;
            animation-name: divhxUp;
            -webkit-animation-delay: 1s;
            -moz-animation-delay: 1s;
            animation-delay: 1s
        }
        .MsgQB{
            position: fixed;
            border-radius: 10px;
            padding:10px;
            opacity:.8;
            left:-1px;
            bottom: 80%;
            // animation:pf 5s;
            z-index: 999999;
        }
        @keyframes pf{
            0%   {bottom:0;}
            10%  {bottom:80%;}
            80%  {bottom:80%;opacity: .8;}
            88%  {opacity: .5;}
            100%  {opacity: 0;bottom:100%;}
        }
        @keyframes pfhide{
            0%   {}
            20%  {opacity: .8;}
            40%  {opacity: .6;}
            60%  {opacity: .4;}
            80%  {opacity: .2;}
            100%  {opacity: 0;bottom:100%;}
        }`;
            Insertstyle(css_asd1);
            if (loadTrue) {
                return;
            }
            if (isVideo !== 1)
                return;
            Msg('开始缓存啦');
            if (location.href.indexOf('51moot') > 0) {
                window.onblur = null;
                downloadName = $('.vedio-play-nav ul li').last().get(0).innerText
                //document.querySelector('video').playbackRate = 16.0;
            }
            $('body').prepend(`
            <div class="hxaside-nav divhxUp animated no-filter" id="VdownLoad-nav">
                <label for="" class="hxaside-menu" title="按住拖动">Videos</label>
                <a href="javascript:void(0)" title="本次关闭" id="XFclose"
                    class="hxmenu-item hxmenu-line hxmenu-first">本次<br>关闭</a>
                <a href="javascript:void(0)" id="vdown0" title="缓存进度条完成时,自动下载视频"
                    class="hxmenu-item hxmenu-line hxmenu-second">下载<br>视频</a>
                <a href="javascript:void(0)" id="vdown1" title="缓存进度条完成时,自动下载视频,并自动关闭当前页"
                    class="hxmenu-item hxmenu-line hxmenu-third">下载后<br>关闭</a>
                <a href="javascript:void(0)" title="设置播放速度" id="SetplaybackRate"
                    class="hxmenu-item hxmenu-line hxmenu-fourth">设置<br>速度</a>
            </div>
            `)
            $(document).on('click', '#XFclose', () => { let $nav = $('#VdownLoad-nav'); $nav.fadeOut(() => { loadTrue = false; $nav.remove() }) })
            $(document).on('click', '#SetplaybackRate', () => { SetplaybackRate(event) })
            $(document).on('click', '#vdown0', () => { Vdown(0) })
            $(document).on('click', '#vdown1', () => { Vdown(1) })
            XF();
        }

        XFrun();
    //----------------------------分割线----------------------------------------------



    const _endOfStream = window.MediaSource.prototype.endOfStream
    window.MediaSource.prototype.endOfStream = function () {
        window.isComplete = 1;
        return _endOfStream.apply(this, arguments)
    }
    window.MediaSource.prototype.endOfStream.toString = function () {
        //  console.log('endOfStream hook is detecting!');
        return _endOfStream.toString();
    }

    const _addSourceBuffer = window.MediaSource.prototype.addSourceBuffer
    window.MediaSource.prototype.addSourceBuffer = function (mime) {
        // console.log("MediaSource.addSourceBuffer ", mime)
        if (mime.toString().indexOf('audio') !== -1) {
            window.audio = [];
            //    console.log('audio array cleared.');
        } else if (mime.toString().indexOf('video') !== -1) {
            window.video = [];
            //    console.log('video array cleared.');
        }
        let sourceBuffer = _addSourceBuffer.call(this, mime)
        const _append = sourceBuffer.appendBuffer
        sourceBuffer.appendBuffer = function (buffer) {
            //    console.log(mime, buffer);
            if (mime.toString().indexOf('audio') !== -1) {
                window.audio.push(buffer);
            } else if (mime.toString().indexOf('video') !== -1) {
                window.video.push(buffer)
            }
            _append.call(this, buffer)
        }

        sourceBuffer.appendBuffer.toString = function () {
            //    console.log('appendSourceBuffer hook is detecting!');
            return _append.toString();
        }
        isVideo = 1;
        downloadName = document.title;
        XFrun();
        loadTrue = true;
        return sourceBuffer
    }

    window.MediaSource.prototype.addSourceBuffer.toString = function () {
        // console.log('addSourceBuffer hook is detecting!');
        return _addSourceBuffer.toString();
    }

    function download() {
        if (location.href.indexOf('51moot') > 0) {
            downloadName = $('.vedio-play-nav ul li').last().get(0).innerText
        }
        let a = document.createElement('a');
        a.href = window.URL.createObjectURL(new Blob(window.audio));
        a.download = 'audio_' + downloadName + '.mp4';
        a.click();
        a.href = window.URL.createObjectURL(new Blob(window.video));
        a.download = 'video_' + downloadName + '.mp4';
        a.click();
        Msg(downloadName+'  ,下载成功啦');
        setTimeout(()=>{Msg(downloadName+'  ,下载成功啦')},1000)
        window.downloadAll = 0;
        window.isComplete = 0;
        window.autoDownload = 0;
        if (isDelTabPage === 1) {
            closePage();
        }
    }
    function closePage() {
        if (navigator.userAgent.indexOf("Firefox") != -1 || navigator.userAgent.indexOf("Chrome") != -1) {
            window.location.href = "about:blank";
            window.close();
        } else {
            window.opener = null;
            window.open("", "_self");
            window.close();
        }
    }
    // setInterval(() => {
    //     if (window.downloadAll === 1) {
    //         download();
    //     }
    // }, 2000);

    let autoDownInterval = setInterval(() => {
        if (window.isComplete === 1 && window.autoDownload === 1) {
            download();
        }
    }, 2000);

    (function (that) {
        let removeSandboxInterval = setInterval(() => {
            if (that.document.querySelectorAll('iframe')[0] !== undefined) {
                that.document.querySelectorAll('iframe').forEach((v, i, a) => {
                    let ifr = v;
                    ifr.removeAttribute('sandbox');
                    const parentElem = that.document.querySelectorAll('iframe')[i].parentElement;
                    a[i].remove();
                    parentElem.appendChild(ifr);
                });
                clearInterval(removeSandboxInterval);
            }
        }, 1000);
    })(window);
    })();

    const ColorReverse = (OldColorValue) => {
            OldColorValue = "0x" + OldColorValue.replace(/#/g, "");
            var str = "000000" + (0xFFFFFF - OldColorValue).toString(16);
            return '#' + str.substring(str.length - 6, str.length);
        }
        const guid = () => {
            return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) {
                var r = Math.random() * 16 | 0, v = c == 'x' ? r : (r & 0x3 | 0x8);
                return v.toString(16);
            });
        }
        const Msg = (msgtxt) => {
            let id = guid();
            var vbo = "80%";
            $('.MsgQB').each((i, e) => {
                let bo = $(e).css("bottom").replace('px', '');
                bo = parseInt(bo) - 50;
                vbo = bo.toString() + 'px';
            })
            $('body').append('<div id="' + id + '" class="MsgQB" style="display:none;bottom:' + vbo + '">' + msgtxt + '  ヾ(•ω•`)o</div>');
            let randomColor = Math.floor(Math.random() * 16777215).toString(16);
            let $el = $('#'+id);
            $el.fadeIn().animate({"left":"10px"});
            $el.css('background', ColorReverse(randomColor)).css('color', ColorReverse(ColorReverse(randomColor))).css('box-shadow',' 0 0 5px 2px '+ColorReverse(ColorReverse(randomColor)));
            $el.get(0).addEventListener('animationend', () => {
                $el.remove();
            })
            setTimeout(() => { $el.css('animation', 'pfhide 1s') }, 4.5 * 1000)
            $el.get(0).addEventListener('animationstart', () => {
                $('.MsgQB').each((i, e) => {
                    let bo = $(e).css("bottom").replace('px', '');
                    bo = parseInt(bo) + 50;
                    $(e).animate({ "bottom": bo.toString() + 'px' })
                })
            })
        }