短视频去水印(支持抖音网页版)

一款短视频去水印下载工具,支持抖音网页版视频无水印下载

// ==UserScript==
// @name         短视频去水印(支持抖音网页版)
// @version      1.2.2
// @license      MIT
// @description  一款短视频去水印下载工具,支持抖音网页版视频无水印下载
// @icon         
// @author       我爱工具箱
// @match      *://*.douyin.com/*
// @require      https://lib.baomitu.com/jquery/3.6.0/jquery.js
// @require      https://lib.baomitu.com/sweetalert/2.1.2/sweetalert.min.js
// @require      https://lib.baomitu.com/clipboard.js/2.0.6/clipboard.min.js
// @grant        GM_addStyle
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_deleteValue
// @grant        GM_listValues
// @grant        GM_openInTab
// @grant        GM_notification
// @grant        GM_xmlhttpRequest
// @grant        GM_download
// @grant       unsafeWindow
// @namespace https://greasyfork.org/users/894875
// ==/UserScript==

(function() {
    let showTipErrorSwal = function(err) {
        showSwal(err, { icon: 'error' });
    }

    const divTips = document.createElement('div');
    divTips.id = "divTips";

    let showSwal = function(content, option) {
        divTips.innerHTML = content;
        option.content = divTips;
        if (!option.hasOwnProperty('button')) {
            option.button = '我知道了'
        }
        swal(option);
    }

    let headers = {
        "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/126.0.0.0 Safari/537.36",
        "Origin": "https://www.douyin.com",
        "Referer": "https://www.douyin.com"
    }

    let isDouyinPage = function() {
        let url = location.href;
        if (url.indexOf(".douyin.com/") != -1) {
            return true;
        } else {
            return false;
        }
    };

    let getVideoUrl = function() {
        let RENDER_DATA = document.getElementById("RENDER_DATA");
        if (RENDER_DATA == null) {
            return "";
        }
        let RENDER_DATA_STR = RENDER_DATA.textContent;
        renderData = decodeURIComponent(RENDER_DATA_STR);
        let jsonObject = JSON.parse(renderData);
        let data = null;
        for (let key in jsonObject) {
            if ("_location" == key || "app" == key || "11" == key) {
                continue;
            }
            data = key;
        }
        let dataJo = jsonObject[data];
        let detail = dataJo["videoDetail"];
        let video = detail["video"];
        let playAddr = video["playAddr"];
        let video1 = playAddr[0];
        let src = video1["src"];
        let videoUrl = "https:" + src;
        videoUrl = videoUrl.substring(0, videoUrl.length - 1);
        return videoUrl;
    }

    var isDownVideo = false;

    let successCall = function(videoUrl, videoName) {
        if (videoUrl == null) {
            videoUrl = getVideoUrl();
        }
        if (!isDownVideo) {
            GM_download({ url: videoUrl, name: videoName + '.mp4', headers: headers});
            isDownVideo = true;
        }
    }

    let initButtonEvent = function(btn) {
        if (isDouyinPage()) {
            var isClickDownVideo = true;
            let videoType = Number.parseInt(btn.getAttribute("type"));
            if(videoType == 1) {
                videoDownloadNum = 0;
                let videoIndex = btn.getAttribute("videoIndex");
                let videoSlide = document.getElementById("slidelist");
                let videoClassArr = videoSlide.firstChild.firstChild;
                //推荐~朋友
                let videoArr = videoClassArr.getElementsByClassName("page-recommend-container");
                if(videoArr.length == 0) {
                    //首页
                    videoArr = videoClassArr.getElementsByClassName("dySwiperSlide");
                }
                let videoItemDiv = videoArr[videoIndex];
    
                //视频名
                let accountNameSpanNode = videoItemDiv.getElementsByClassName("title")[0].childNodes[0];
                let videoName = accountNameSpanNode.firstElementChild.lastElementChild.textContent;
                if(videoName == "") {
                  videoName = "无标题视频";
                }
    
                //视频链接
                let sliderVideoDiv = videoItemDiv.getElementsByClassName("slider-video");
                let xgVideoContainerDiv = sliderVideoDiv[0].getElementsByClassName("xg-video-container");
                let videoNode = xgVideoContainerDiv[0].getElementsByTagName("video");
                let videoSrc = videoNode[0].getAttribute("src");
                if(videoSrc != null) {
                    // showTipErrorSwal("暂不支持该视频下载,正在加紧适配中...");
                    const originFetch = fetch;
                    unsafeWindow.fetch = (...arg) => {
                        // console.log('fetch arg', ...arg);
                        if (arg[0].indexOf('douyin') > -1 && isClickDownVideo) {
                            let videoUrl =  decodeURI('https:' + arg[0]);
                            GM_xmlhttpRequest(headRequest(videoUrl, videoName, successCall));
                            if(successCall) {
                                isClickDownVideo = false;
                            }
                        }
                    }

                }else{
                  let sourceNodes = videoNode[0].getElementsByTagName("source");
                  for (let i = 0; i < sourceNodes.length; i++) {
                      try {
                          let videoUrl = decodeURI('https:' + sourceNodes[i].getAttribute("src"));
                          GM_xmlhttpRequest(headRequest(videoUrl, videoName, successCall));
                          if(successCall) {
                            isDownVideo = false;
                            break;
                          }
                      } catch (error) {
                          console.error(error);
                      }
                  }
                }
            }else if(videoType == 2) {
                let videoClassArr = document.getElementsByClassName("playerControlHeight");
                let videoInfoArr = videoClassArr[0].getElementsByClassName("leftContainer");
                let videoNameArr = videoInfoArr[0].childNodes[2].firstElementChild.getElementsByTagName("h1");
                let videoName = videoNameArr[0].textContent;
                if(videoName == "") {
                  videoName = "无标题视频";
                }
                let videoArr = videoClassArr[0].getElementsByClassName("xg-video-container");
                let videoNode = videoArr[0].getElementsByTagName("video");
                let sourceNodes = videoNode[0].getElementsByTagName("source");
                  for (let i = 0; i < sourceNodes.length; i++) {
                      try {
                          let videoUrl = decodeURI('https:' + sourceNodes[i].getAttribute("src"));
                          GM_xmlhttpRequest(headRequest(videoUrl, videoName, successCall));
                          if(successCall) {
                            isDownVideo = false;
                            break;
                          }
                      } catch (error) {
                          console.error(error);
                      }
                  }
            }else if(videoType == 3) {
                let videoDiv= document.getElementById("slideMode");
                //视频名
                let accountNameSpanNode = videoDiv.getElementsByClassName("title")[0].childNodes[0];
                let videoName = accountNameSpanNode.firstElementChild.lastElementChild.textContent;
                if(videoName == "") {
                  videoName = "无标题视频";
                }
                let xgVideoContainerDiv = videoDiv.getElementsByClassName('xg-video-container');
                let videoNode = xgVideoContainerDiv[0].getElementsByTagName("video");
                let videoSrc = videoNode[0].getAttribute("src");
                if(videoSrc != null) {
                    // showTipErrorSwal("暂不支持该视频下载,正在加紧适配中...");
                    const originFetch = fetch;
                    unsafeWindow.fetch = (...arg) => {
                        // console.log('fetch arg', ...arg);
                        if (arg[0].indexOf('douyin') > -1 && isClickDownVideo) {
                            let videoUrl =  decodeURI('https:' + arg[0]);
                            GM_xmlhttpRequest(headRequest(videoUrl, videoName, successCall));
                            if(successCall) {
                                isClickDownVideo = false;
                            }
                        }
                    }
                }else{
                  let sourceNodes = videoNode[0].getElementsByTagName("source");
                  for (let i = 0; i < sourceNodes.length; i++) {
                      try {
                          let videoUrl = decodeURI('https:' + sourceNodes[i].getAttribute("src"));
                          GM_xmlhttpRequest(headRequest(videoUrl, videoName, successCall));
                          if(successCall) {
                            isDownVideo = false;
                            break;
                          }
                      } catch (error) {
                          console.error(error);
                      }
                  }
                }
            }
        }
    };

    let headRequest = function(url, videoName, call) {
        let userAgent = navigator.userAgent;
        let method = "HEAD";
        if(userAgent.indexOf("Firefox") != -1) {
            method = "GET";
        }
        return {
            method: method,
            timeout: 300000, // 30秒超时
            headers: headers,
            url: url,
            onload: function(res) {
                if (res.status === 200 || res.status === 401) {
                    call(url, videoName);
                }else if(res.status === 302) {
                    let Location = res.headers.getAttribute("Location");
                    call(Location, videoName);
                }else if(res.status === 403) {
                    showTipErrorSwal("浏览器不支持该视频");
                } else {
                    console.error(res);
                }
            },
            ontimeout: (res) => {
                console.error(res);
            },
            onerror: (res) => {
                console.error(res);
            }
        }
    };

    let start = function() {
        if (!isDouyinPage()) {
            // console.log('非抖音页面,1秒后将重新查找!');
            return;
        }

        if(document.body.innerHTML == "") {
            return;
        }
        let isIndexVideo = getIndexVideo();
        if(!isIndexVideo) {
            if(!getIndexDetailVideo()) {
                getDetailVideo();
            }
        }
    }

    /**
     * 首页/推荐/我的
     * @returns 
     */
    function getIndexVideo() {
        let videoSlide = document.getElementById("slidelist");
        if(videoSlide == null) {
            // console.log('未查找到视频列表div!');
            return false;
        }
        let videoClassArr = videoSlide.firstChild.firstChild;
        if(videoClassArr.length == 0) {
            // console.log('未查找到视频div!');
            return false;
        }

        //推荐~朋友
        let videoArr = videoClassArr.getElementsByClassName("page-recommend-container");
        if(videoArr.length == 0) {
            //首页
            videoArr = videoClassArr.getElementsByClassName("dySwiperSlide");
        }
        if(videoArr.length == 0) {
            // console.log('未查找到视频div列表');
            return false;
        }

        //偏移量
        let shiftingDiv = videoSlide.getElementsByClassName("fullscreen_capture_feedback");
        let height = shiftingDiv[0].style.height;
        let shifting = 0;
        if(height != "100%") {
          shifting = height.match(/calc\(\d+%\s*-\s*(\d+)px\)/i)[1];
        }
        //每个视频的高度
        let firstVideoDiv = videoArr[0];
        let firstVideoHeight = firstVideoDiv.style.height;
        firstVideoHeight = firstVideoHeight.replace("px", "");
        let videoTransForm= videoClassArr.style.transform || '';
        //视频下标
        var videoIndex = 0;
        let videoTransFormarr = videoTransForm.match(/translate3d\(\d+px,\s*(-+\d+)px,\s*(\d+)px\)/i);
        if(videoTransFormarr != null) {
          let videoTransFormY = videoTransFormarr[1];
          videoTransFormY = -videoTransFormY;
          videoIndex = videoTransFormY/(parseInt(firstVideoHeight) + parseInt(shifting));
        }

        let btnBox = videoArr[videoIndex].getElementsByClassName('positionBox');
        if(btnBox.length == 0) {
            // console.log('可能为直播页面或登录弹出页面');
            return false;
        }
        if(btnBox[0].childNodes.length > 1) {
            var btnShare = btnBox[0].childNodes[1];
        }else{
            var btnShare = btnBox[0].childNodes[0];
        }
        if(btnShare == undefined) {
            // console.log('可能还未初始化分享按钮');
            return false;
        }
    
        let btnDownload = {
            class: 'btnClickDownload',
            title: '点击下载视频',
            html: function() {
                return `<img height="26" width="26" src=""/>`;
            }
        }
        //如果是已有下载按钮,则不添加
        let buttonArr = btnShare.getElementsByClassName(btnDownload.class);
        if (buttonArr.length != 0) {
            return false;
        }

        let btn = document.createElement('a');
        btn.setAttribute("type", 1);
        btn.setAttribute("class", btnDownload.class);
        btn.setAttribute("videoIndex", videoIndex);
        btn.title = btnDownload.title;
        btn.innerHTML = btnDownload.html();
        btn.addEventListener('click', function(e) {
            initButtonEvent(btn);
            e.preventDefault();
        });

        btnShare.appendChild(btn);
        return true;
    }

    /**
     * 首页/推荐/我的-详情
     * @returns 
     */
    function getIndexDetailVideo() {
        let videoDiv= document.getElementById("slideMode");
        if(videoDiv == null) {
            // console.log('未查找到视频div!');
            return false;
        }
        let positionBoxDiv = videoDiv.getElementsByClassName("positionBox");
        let btnShare = positionBoxDiv[0].childNodes[1];
        if(btnShare == undefined) {
            // console.log('可能还未初始化分享按钮');
            return false;
        }
    
        let btnDownload = {
            class: 'btnClickDownload',
            title: '点击下载视频',
            html: function() {
                return `<img height="26" width="26" src=""/>`;
            }
        }
        //如果是已有下载按钮,则不添加
        let buttonArr = btnShare.getElementsByClassName(btnDownload.class);
        if (buttonArr.length != 0) {
            return false;
        }

        let btn = document.createElement('a');
        btn.setAttribute("type", 3);
        btn.setAttribute("class", btnDownload.class);
        btn.title = btnDownload.title;
        btn.innerHTML = btnDownload.html();
        btn.addEventListener('click', function(e) {
            initButtonEvent(btn);
            e.preventDefault();
        });

        btnShare.appendChild(btn);
        return true;
    }

    /**
     * 进入详情页详情
     * @returns 
     */
    function getDetailVideo() {
        let videoClassArr = document.getElementsByClassName("playerControlHeight");
        if(videoClassArr.length == 0) {
            // console.log('未查找到视频总div!');
            return false;
        }
        let videoDiv = videoClassArr[0];

        //详情
        let  videoArr = videoDiv.getElementsByClassName("xg-video-container");
        if(videoArr.length == 0) {
            // console.log('未查找到视频div');
            return false;
        }

        let btnGrid = videoDiv.getElementsByClassName('xg-right-grid');

        let btnDownload = {
            class: 'btnClickDownload',
            title: '点击下载视频',
            html: function() {
                return `<img height="22" width="22" src=""/>`;
            }
        }
        //如果是已有下载按钮,则不添加
        let buttonArr = btnGrid[0].getElementsByClassName(btnDownload.class);
        if (buttonArr.length != 0) {
            return false;
        }

        let btn = document.createElement('a');
        btn.setAttribute("type", 2);
        btn.setAttribute("class", btnDownload.class);
        btn.title = btnDownload.title;
        btn.innerHTML = btnDownload.html();
        btn.addEventListener('click', function(e) {
            initButtonEvent(btn);
            e.preventDefault();
        });

        btnGrid[0].appendChild(btn);
        return true;
    }

    setInterval(function() {
        start();
    }, 1000)
})();