B站笔记快捷键←↓→

在B站笔记中:打开笔记快捷键:Ctrl+Shift+↑,关闭笔记快捷键:Ctrl+Shift+↓,笔记窗口切换到视频窗口:Ctrl+Shift+←,视频窗口切换到笔记窗口:Ctrl+Shift+→,截图+时间戳快捷键:Ctrl+↓,时间戳+截图快捷键:Ctrl+↑,时间戳快捷键:Ctrl+←,截图快捷键:Ctrl+→,下载视频笔记Ctrl+Shift+S

当前为 2023-01-16 提交的版本,查看 最新版本

// ==UserScript==
// @name         B站笔记快捷键←↓→
// @namespace    https://space.bilibili.com/1208812226
// @version      2.8.4
// @description  在B站笔记中:打开笔记快捷键:Ctrl+Shift+↑,关闭笔记快捷键:Ctrl+Shift+↓,笔记窗口切换到视频窗口:Ctrl+Shift+←,视频窗口切换到笔记窗口:Ctrl+Shift+→,截图+时间戳快捷键:Ctrl+↓,时间戳+截图快捷键:Ctrl+↑,时间戳快捷键:Ctrl+←,截图快捷键:Ctrl+→,下载视频笔记Ctrl+Shift+S
// @author       大王鹅鹅鹅
// @match        http*://www.bilibili.com/video/*
// @match        https://www.bilibili.com/medialist/*
// @icon         https://static.hdslb.com/images/favicon.ico
// @require      https://cdn.bootcdn.net/ajax/libs/jszip/3.10.1/jszip.js
// @grant        none
// @license      AGPL License
// ==/UserScript==

(function () {
    "use strict";
    // JS监听键盘快捷键事件
    document.addEventListener("keydown", function (event) {
        if (event.shiftKey && event.keyCode == 38) {//打开笔记——快捷键:Ctrl+Shift+↑
            if (event.ctrlKey) {
                document.querySelector(".note-btn__blue").click();
                setTimeout( function(){document.querySelector(".operation-desc").click();},0);
                var count = 2;
                var timeId = setInterval(function () {
                    count--;
                    if (count <= 0) {
                        clearInterval(timeId);
                    }
                    initFocusDown();

                }, 600);

                //固定笔记
                var myresize = document.querySelector(".resizable-component");
                myresize.style.position = 'fixed';
                myresize.style.setProperty('position', 'fixed', 'important');

            }
        }

        if (event.shiftKey && event.keyCode == 40) {//关闭笔记——快捷键:Ctrl+Shift+↓
            if (event.ctrlKey) {
                document
                    .querySelector(".close-note")
                    .click();
            }
        }

        if (event.shiftKey && event.keyCode == 37) {//回到视频——快捷键:Ctrl+Shift+←
            if (event.ctrlKey) {
                var pNodel = document.querySelector(
                    "div.editor-innter.ql-container.ql-snow > div.ql-editor"
                );
                pNodel.blur();

                let videos = document.getElementsByTagName('video');
                let video = videos[0];
                if(toggle()){
                    video.pause();
                }else{
                    video.play();
                }

            }
        }
        if (event.shiftKey && event.keyCode == 39) {//回到笔记——快捷键:Ctrl+Shift+→
            if (event.ctrlKey) {
                initFocusDown();
            }
        }

        if (event.ctrlKey && event.keyCode == 37) {//时间戳——快捷键:Ctrl+←
            if (!event.shiftKey) {
                document.querySelector("i.bili-note-iconfont.iconicon_flag_L").click();
                setTimeout(function () {
                    document
                        .querySelector(
                        "div.dialog-btn.tag-dialog__btn--confirm:nth-child(2)"
                    )
                        .click();
                }, 0);
            }
        }


        if (event.ctrlKey && event.keyCode == 38) {//时间戳+截图——快捷键:Ctrl+↑
            if (!event.shiftKey) {
                imageAfterTime();

            }
        }

        if (event.ctrlKey && event.keyCode == 39) {//截图——快捷键:Ctrl+→
            if (!event.shiftKey) {
                document.querySelector("i.bili-note-iconfont.iconcapture-app").click();
                setTimeout(function () {
                    var parentNode = document.querySelector(
                        "div.editor-innter.ql-container.ql-snow > div.ql-editor"
                    );
                    var scrollHeight = parentNode.scrollHeight;
                    parentNode.scrollTo(0, scrollHeight);
                    parentNode.focus();
                }, 500);

            }
        }

        if (event.ctrlKey && event.keyCode == 40) {//截图+时间戳——快捷键:Ctrl+↓
            if (!event.shiftKey) {
                timeAfterImage();
            }
        }

        if (event.shiftKey && event.keyCode == 83) {//下载笔记为markdown格式——快捷键:Ctrl+Shift+S
            if (event.ctrlKey) {
                openNote();
            }
        }

        if (event.altKey && event.keyCode == 83) {//下载笔记为markdown格式——快捷键:Ctrl+Alt+S
            if (event.ctrlKey) {
                openNote();
            }
        }



        if (event.shiftKey && event.keyCode == 90) {//弹幕一键存入笔记——快捷键:Ctrl+Shift+Z
            if (event.ctrlKey) {
                getDanMu();
            }
        }

        if (event.shiftKey && event.keyCode == 86) {//截图一键存入笔记——快捷键:Ctrl+Shift+V
            if (event.ctrlKey) {
                //下载所有弹幕
                //getDanMuDownload();

                //下载所有截图
                // getPicDownload();



            }
        }

         if (event.shiftKey && event.keyCode == 51) {//截图一键存入笔记——快捷键:Ctrl+Shift+3
            if (event.ctrlKey) {
                var playbackrate = document.querySelector(".bpx-player-ctrl-playbackrate-menu");
                playbackrate.children[2].click();

            }
        }




        if (event.altKey && event.keyCode == 67) {//获取视频当前精确播放位置链接到剪贴板——快捷键:Alt+C
            let videos = document.getElementsByTagName('video');
            let video = videos[0];
            let pUrl = window.location.href;
            let match = pUrl.match(/(https\:\/\/www.bilibili.com\/video\/)(BV|av)[a-zA-Z0-9]+(?=[\/\?])?/g);
            let r= new RegExp(`[?&]p=([^&]*)`);
            let regex = new RegExp(`[?&]p=([^&]*)`);
            if(regex.test(pUrl)){

                let results = r.exec(pUrl);
                let p = results ? results[1] : null;
                pUrl = match ? match[0] +"?p="+p+"&t="+video.currentTime: pUrl;
            }else{
                pUrl = match ? match[0] +"?t="+video.currentTime: pUrl;

            }

            navigator.clipboard.writeText(pUrl).then(function() {
                console.log('Copied text to clipboard');
            }, function(err) {
                console.error('Failed to copy text: ', err);
            });
        }


    });

    var mystate = false;
    function toggle() {
        mystate = !mystate;
        return mystate;
    }

    function initFocusDown() {
        var el = document.querySelector(
            "div.editor-innter.ql-container.ql-snow > div.ql-editor"
          );
        if(el!=null){
            var scrollHeight = el.scrollHeight;
            el.scrollTo(0, scrollHeight);
            el.focus();
            if (
                typeof window.getSelection != "undefined" &&
                typeof document.createRange != "undefined"
            ) {
                var range = document.createRange();
                range.selectNodeContents(el);
                range.collapse(false);
                var sel = window.getSelection();
                sel.removeAllRanges();
                sel.addRange(range);
            } else if (typeof document.body.createTextRange != "undefined") {
                var textRange = document.body.createTextRange();
                textRange.moveToElementText(el);
                textRange.collapse(false);
                textRange.select();
            }
        }
    }

    function openNote() {
      const editor = document.querySelector(".ql-editor")
        ? document.querySelector(".ql-editor")
        : null;
        if(!editor){alert("请先打开视频笔记,快捷键:Ctrl+Shift+↑");return;}
      const isBlank = editor && !editor.classList.contains("ql-blank");

      if (isBlank) {
        const noteContent = getNoteContent(editor);
        downloadMD(noteContent);
      } else {
        alert("笔记为空,无法下载");
      }
    }

    function getNoteContent(editor) {
      if (
        editor.nodeType === 1 &&
        editor.childNodes &&
        editor.childNodes.length > 0
      ) {
        return getNodes2Array(editor.childNodes);
      }
      return "";
    }

    function getNodes2Array(nodes) {
      const contents = [];
      for (let i = 0; i < nodes.length; i++) {
        const node = nodes[i];
        if (node.nodeType == 1) {
          if (node.nodeName == "P") {
              if (node.querySelector('div') || node.querySelector('span') || node.querySelector('strong') || node.querySelector('u') || node.querySelector('s')) {
                   contents.push(node2string(node));
              }else{
                   contents.push(node.innerHTML.replace(/src="(?!https:)(.*?)"/g, 'src="https:$1"').replace(/<br\s*\/?>/gi, "\n"));
              }
            
          } else if (node.nodeName == "OL" || node.nodeName == "UL") {
            contents.push(getLiContent(node));
          } else if (node.nodeName == "DIV") {
            contents.push(divImgContent(node));
          } else {
            contents.push(node.textContent);
          }
        }
      }
      return contents;
    }

    function getLiContent(node) {
      let allLiContent = "";
      let nodeName = node.nodeName;
      let liNodes = node.childNodes;
      let indentLevels = {};

      if (liNodes && liNodes.length > 0) {
        for (let li of liNodes) {
          let liclass = li.getAttribute("class");
          let indentLevel = 0;
          if (liclass) {
            let match = liclass.match(/ql-indent-(\d+)/);
            if (match) {
              indentLevel = parseInt(match[1]);
            }
          }

          indentLevels[indentLevel] = (indentLevels[indentLevel] || 0) + 1;

          let indent = "";
          for (let i = 0; i < indentLevel; i++) {
            indent += "\t";
          }

          allLiContent +=
            indent +
            (nodeName == "OL" ? indentLevels[indentLevel] + ". " : "- ") +
            node2string(li) +
            "\n";
        }
      }

      return allLiContent;
    }

    function divImgContent(node) {
      if (!node.classList || !node.classList.contains("ql-image-preview")) {
        return;
      }

      let img = node.querySelector("img");
      let src = img ? "https:" + img.getAttribute("src") : "";
      if (!src.includes("http")) {
        return;
      }

      let content = src ? src : "";
      if (!content.match(/api.bilibili.com\/x\/note\/image/g)) {
        content = "![](" + content + ")";
      }
      return content;
    }

    function node2string(node) {
      let nodes = node.childNodes;
      if (!nodes || nodes.length <= 0) return "";
      let line = "";

      for (let i = 0; i < nodes.length; i++) {
        let child = nodes[i];
        let tag = child.nodeName;
        let myContents = "";

        if (
          (child.classList && child.classList.contains("ql-tag-blot")) ||
          tag === "DIV"
        ) {
          //时间戳
          let timeText = child.querySelector(".time-tag-item__text");
          let desc = child.querySelector(".time-tag-item__desc");
          let p = child.getAttribute("data-index");
          let second = child.getAttribute("data-seconds");
          let pUrl = window.location.href;
          let match = pUrl.match(
            /(https\:\/\/www.bilibili.com\/video\/)(BV|av)[a-zA-Z0-9]+(?=[\/\?])?/g
          );
          pUrl = match ? match[0] : pUrl;

          let title =
            "🚩" +
            timeText.firstChild.textContent +
            (desc ? " " + desc.textContent.replace(/~/g, '') : "");
          let time = `${pUrl}?p=${p}&t=${second}`;

          myContents += `[${title}](${time})`;
        } else {
          if (!child.children || child.children.length == 0) {
              myContents += child.textContent;
          } else {
            myContents += node2string(child);
          }

          //针对文字ql-bg
          let className = "";
          if (child.classList) {
            let myClassList = child.classList;
            for (let a = 0; a < myClassList.length; a++) {
              if (myClassList[a].match(/ql-bg/)) {
                className += "background-color:" + myClassList[a].slice(-7) + ";";
              }
              if (myClassList[a].match(/ql-color/)) {
                className += "color:" + myClassList[a].slice(-7) + ";";
              }
              if (myClassList[a].match(/ql-size/)) {
                className +=
                  "font-size:" +
                  myClassList[a].slice(8, myClassList[a].length) +
                  ";";
              }
            }
          }

         if (child.children && child.querySelector('.ql-tag-blot')==null) {

          let font1 = className != "" ? "<font style='" + className + "'>" : "";
          let font2 = className != "" ? "</font>" : "";
          myContents = font1 + myContents + font2;


          if (tag == "U" || tag == "S" || tag == "STRONG") {
            myContents =
              "<" +
              tag.toLowerCase() +
              ">" +
              myContents +
              "</" +
              tag.toLowerCase() +
              ">";
          } else if (tag == "BR") {
              myContents += "\n";
          } else {
              myContents += "";
          }
         }
        }
        line += myContents;
      }
      return line;
    }
      function getNowTime(){
          var now = new Date();
          var year = now.getFullYear();
          var month = now.getMonth() + 1;
          var date = now.getDate();
          var hours = now.getHours();
          var minutes = now.getMinutes();
          var seconds = now.getSeconds();
          hours = hours < 10 ? '0' + hours : hours;
          minutes = minutes < 10 ? '0' + minutes : minutes;
          seconds = seconds < 10 ? '0' + seconds : seconds;

          var dateString = year + "-" + month + "-" + date + " " + hours + ":" + minutes + ":" + seconds;
          return dateString;
      }

   function getInfo(){
          let allContent ="";
          try{
              const username = document.querySelector(".username").firstChild.textContent.trim();
              const pudate_text = document.querySelector(".pudate-text").firstChild.textContent.trim();
              const views = document.querySelector("span.view.item").getAttribute("title");
              let pUrl = window.location.href;
              let match = pUrl.match(/(https\:\/\/www.bilibili.com\/video\/)(BV|av)[a-zA-Z0-9]+(?=[\/\?])?/g);
              pUrl = match ? match[0] : pUrl;
              allContent = "---\ntags: bilibili_note\nurl: "+pUrl+"\nup: "+username+"\nup_date: "+pudate_text+"\ndown_time: "+getNowTime()+"\nviews: "+views+"\n---\n\n";
          }catch(e){
              let pUrl = window.location.href;
              let match = pUrl.match(/(https\:\/\/www.bilibili.com\/video\/)(BV|av)[a-zA-Z0-9]+(?=[\/\?])?/g);
              pUrl = match ? match[0] : pUrl;
              allContent = "---\ntags: bilibili_note\nurl: "+pUrl+"\ndown_time: "+getNowTime()+"\n---\n\n";
          }
      return allContent;
      }

    async function getVideoInfo(src) {
        return await fetch(src, {
            method: 'get',
            credentials:  src.includes("bilibili.com") ? "include" : "omit",
        }).then(function(res) {
            return res.text();
        }).then(function(data) {
            return data;
        });

    }

    function downloadMD(notes) {
      let zip = new JSZip();
      let allContent =getInfo();
      let isOnlyMD=false;
      let bid = window.location.href.match(
        /(?<=bilibili.com\/video\/)(BV|av)[a-zA-Z0-9]+(?=[\/\?])?/g
      );

      for (var i = 0; i < notes.length; i++) {
        if (notes[i] != "") {
          let images_id = notes[i].match(
            /(?<=api.bilibili.com\/x\/note\/image\?image_id=)\d{6}/g
          );
          if (images_id) {
            allContent +=
              "![](assets/" + bid[0] + "_" + images_id[0] + ".jpg)" + "\n";

           let fileName = notes[i].slice(-6);
           let img_bolb= downloadImages(notes[i]);
           zip.folder("assets").file(bid[0] + "_" + fileName + ".jpg", img_bolb);
           isOnlyMD=true;
          } else {
            allContent += notes[i] + "\n";
          }
        } else {
          allContent += "\n";
        }
      }

      let video_title = document.querySelector(".video-title");
      let v_title = video_title ? video_title.textContent : new Date().getTime();
      if(isOnlyMD){
          zip.file("bilibili_" + v_title + ".md", allContent);
          zip.generateAsync({
              type: 'blob',
              compression: "DEFLATE",
              compressionOptions: {
                  level: 9
              }
          }).then(function(c) {
              let filename = v_title+'.zip';
              let a = document.createElement('a');
              a.download = filename;
              a.style.display = 'none';
              a.href = URL.createObjectURL(c);
              document.body.appendChild(a);
              a.click();
              URL.revokeObjectURL(c);
              document.body.removeChild(a);
          });
      }else{
          let blob = new Blob([allContent]);
          let a = document.createElement('a');
          a.download = "bilibili_" + v_title + ".md";
          a.href = URL.createObjectURL(blob);
          document.body.appendChild(a);
          a.click();
          URL.revokeObjectURL(blob);
          document.body.removeChild(a);

      }
    }

    async function downloadImages(src) {
      let res = await fetch(src, {
        method: "get",
        credentials: src.includes("bilibili.com") ? "include" : "omit",
      });
      if (!res.ok) return;
      let blob = await res.blob();
      return blob;
      
    }


    function imageAfterTime() {//时间戳+截图 ↑
        setTimeout( function(){
            document.querySelector('i.bili-note-iconfont.iconicon_flag_L').click();
            setTimeout( function(){
                document.querySelector('div.dialog-btn.tag-dialog__btn--confirm:nth-child(2)').click();
                setTimeout( function(){
                    document.querySelector('i.bili-note-iconfont.iconcapture-app').click();
                    setTimeout( function(){
                        var qlEditor=document.querySelector('.ql-editor');
                        var lastp = qlEditor.lastElementChild;
                        console.log(lastp.previousSibling)
                        if(lastp!=null && lastp.previousSibling.previousSibling !=null && lastp.previousSibling.previousSibling.firstElementChild.nodeName=="BR"){
                            lastp.previousSibling.previousSibling.remove();
                        }
                        var scrollHeight = qlEditor.scrollHeight;
                        qlEditor.scrollTo(0, scrollHeight);
                        qlEditor.lastElementChild.focus();
                    },110);
                },0);
            },0);
        },0);
    }


    function timeAfterImage(){//截图+时间 ↓
        setTimeout( function(){
            document.querySelector('i.bili-note-iconfont.iconcapture-app').click();
            setTimeout( function(){
                getTimestamp();
                setTimeout( function(){
                    var qlEditor=document.querySelector('.ql-editor');
                    var lastp = qlEditor.lastElementChild;
                    console.log(lastp.previousSibling)
                    if(lastp!=null && lastp.previousSibling.previousSibling !=null && lastp.previousSibling.previousSibling.firstElementChild.nodeName=="BR"){
                        lastp.previousSibling.previousSibling.remove();
                    }
                    var scrollHeight = qlEditor.scrollHeight;
                    qlEditor.scrollTo(0, scrollHeight);
                    qlEditor.lastElementChild.focus();
                },110);
            },110);
        },110);

    }



       function toEnd(el) {
           var scrollHeight = el.scrollHeight;
           el.scrollTo(0, scrollHeight);
           el.lastElementChild.focus();

           if (typeof window.getSelection != "undefined" && typeof document.createRange != "undefined") {
               var range = document.createRange();
               range.selectNodeContents(el);
               range.collapse(false);
               var sel = window.getSelection();
               sel.removeAllRanges();
               sel.addRange(range);
           } else if (typeof document.body.createTextRange != "undefined") {
               var textRange = document.body.createTextRange();
               textRange.moveToElementText(el);
               textRange.collapse(false);
               textRange.select();
           }
       }

    function getDanMu() {
        let state = window.__INITIAL_STATE__;
        let p = state.p || 1;
        let cid = state.videoData.pages[p - 1].cid;
        let src = "https://api.bilibili.com/x/v1/dm/list.so?oid=" + cid;

        return fetch(src, {
            method: "get",
            dataType: "text/xml",
            credentials: 'include',
        }).then(function (response) {

            return response.text();
        }).then(function (result) {

            const editor = document.querySelector(".ql-editor")? document.querySelector(".ql-editor"): null;
            if(editor){
                parseDanMu(result,state,editor);
                const timestamps = document.querySelectorAll('.ql-tag-blot');
                timestamps.forEach(function(timestamp) {
                    timestamp.addEventListener('click', function(e) {
                        let dataSeconds=timestamp.getAttribute("data-seconds");
                        let videos = document.getElementsByTagName('video');
                        let video = videos[0];

                        video.currentTime = dataSeconds;
                        video.play();
                        setTimeout(function () {
                             document.querySelector("i.bili-note-iconfont.iconcapture-app").click();
                        }, 200);


                    });

                   editor.scrollTo(0,0);
                   editor.focus();
                });

            }else{
                alert("请先打开视频笔记,快捷键:Ctrl+Shift+↑");
            }

        });
    }

      function getDanMuDownload() {
        let state = window.__INITIAL_STATE__;
        let p = state.p || 1;
        let cid = state.videoData.pages[p - 1].cid;
        let src = "https://api.bilibili.com/x/v1/dm/list.so?oid=" + cid;

        return fetch(src, {
            method: "get",
            dataType: "text/xml",
            credentials: 'include',
        }).then(function (response) {

            return response.text();
        }).then(function (xmlString) {

            let myContents =getInfo();
            let p = state.p || 1;
            let cid = state.videoData.pages[p - 1].cid;
            let title = state.videoData.title;
            let pUrl = window.location.href;
            let match = pUrl.match(/(https\:\/\/www.bilibili.com\/video\/)(BV|av)[a-zA-Z0-9]+(?=[\/\?])?/g);
            pUrl = match ? match[0] : pUrl;

            let parser = new DOMParser();
            let xmlDoc = parser.parseFromString(xmlString, "text/xml");
            let danMuElements=sortDNode(xmlDoc)
            for (let i = 0; i < danMuElements.length; i++) {
                let element = danMuElements[i];
                let pnode = element.getAttribute("p");
                let text = element.textContent;
                let pArr=pnode.split(",");
                let second= parseInt(pArr[0]);
                let title ="🚩" +secondsToMinutes(second) +text.replace(/~/g, '').replace(/\*/g, '');
                let time = `${pUrl}?p=${p}&t=${second}`;
                myContents += `[${title}](${time})`;
                myContents += "\n";
            }

            let video_title = document.querySelector(".video-title");
            let v_title = video_title ? video_title.textContent : new Date().getTime();
            let blob = new Blob([myContents]);
            let a = document.createElement('a');
            a.download = "bilibili_" + v_title + ".md";
            a.href = URL.createObjectURL(blob);
            document.body.appendChild(a);
            a.click();
            URL.revokeObjectURL(blob);
            document.body.removeChild(a);

        });
      }

    function getPicDownload(){
        let videos = document.getElementsByTagName('video');
        let video = videos[0];
        let duration = video.duration;

        video.currentTime = duration;
        video.play();

        const blobArr = [];

        video.onseeked = function() {


            let canvas = document.createElement('canvas');
            canvas.setAttribute('width',video.videoWidth);
            canvas.setAttribute('height',video.videoHeight);
            let ctx = canvas.getContext('2d');
            ctx.drawImage(video, 0, 0, canvas.width, canvas.height);
            let dataURL = canvas.toDataURL('image/jpeg');
            let dataBlob=dataURItoBlob(dataURL);

            blobArr.push(dataBlob);
            downloadPicMD(blobArr);
        };

    }

    async function captureVideoFrames(video, duration) {
        const blobArr = [];

        for (let i = 0; i < duration; i++) {
            (function(i) {
                video.currentTime = i;
                video.pause();

                const canvas = document.createElement('canvas');
                canvas.setAttribute('width', video.videoWidth);
                canvas.setAttribute('height', video.videoHeight);

                const ctx = canvas.getContext('2d');
                ctx.drawImage(video, 0, 0, canvas.width, canvas.height);
                const dataURL = canvas.toDataURL('image/jpeg');
                const dataBlob = dataURItoBlob(dataURL);

                blobArr.push(dataBlob);
                canvas.remove();
            })(i);
        }

        return blobArr;
    }



    function downloadPicMD(blobArr) {
        let zip = new JSZip();
        let allContent =getInfo();
        let bid = window.location.href.match(
            /(?<=bilibili.com\/video\/)(BV|av)[a-zA-Z0-9]+(?=[\/\?])?/g
        );

        for (let i = 0; i < blobArr.length; i++) {
            if (blobArr[i] != "") {
                allContent += "![](assets/" + bid[0] +i + ".jpg)\n";
                zip.folder("assets").file(bid[0] +i + ".jpg", blobArr[i]);
            } else {
                allContent += "\n";
            }
        }
        let video_title = document.querySelector(".video-title");
        let v_title = video_title ? video_title.textContent : new Date().getTime();
        zip.file("bilibili_" + v_title + ".md", allContent);
        zip.generateAsync({
            type: 'blob',
            compression: "DEFLATE",
            compressionOptions: {
                level: 9
            }
        }).then(function(c) {
            let filename = v_title+'.zip';
            let a = document.createElement('a');
            a.download = filename;
            a.style.display = 'none';
            a.href = URL.createObjectURL(c);
            document.body.appendChild(a);
            a.click();
            URL.revokeObjectURL(c);
            document.body.removeChild(a);
        });
    }

    function dataURItoBlob(dataURI) {
        var byteString = atob(dataURI.split(',')[1]);
        var mimeString = dataURI.split(',')[0].split(':')[1].split(';')[0]
        var ab = new ArrayBuffer(byteString.length);
        var ia = new Uint8Array(ab);
        for (var i = 0; i < byteString.length; i++) {
            ia[i] = byteString.charCodeAt(i);
        }
        var blob = new Blob([ab], {type: mimeString});
        return blob;
    }


    function sortDNode(xmlDoc){
        const dNodes = xmlDoc.querySelectorAll('d');
        const sortedDNodes = [...dNodes].sort((a, b) => {
            const p1 = a.getAttribute('p');
            const p2 = b.getAttribute('p');
            let p1Arr=p1.split(",");
            let p2Arr=p2.split(",");
            let p1Int=parseInt(p1Arr[0]);
            let p2Int=parseInt(p2Arr[0]);
            if (p1Int < p2Int) {
                return -1;
            } else if (p1Int > p2Int) {
                return 1;
            } else {
                return 0;
            }
        });
        return sortedDNodes;
    }



    function parseDanMu(xmlString,state,editor) {
        let p = state.p || 1;
        let cid = state.videoData.pages[p - 1].cid;
        let title = state.videoData.title;


        let parser = new DOMParser();
        let xmlDoc = parser.parseFromString(xmlString, "text/xml");
        let danMuElements=sortDNode(xmlDoc)
       //let danMuElements = sortedXmlDoc.getElementsByTagName("d");
        for (let i = 0; i < danMuElements.length; i++) {
            let pnew = document.createElement('p');
            let element = danMuElements[i];
            let pnode = element.getAttribute("p");
            let text = element.textContent;
            let pArr=pnode.split(",");
            pnew.append(BHtmlParser(parseInt(pArr[0]),text,cid,p,title,pArr[7].slice(-13)));
            editor.append(pnew);
        }
        var scrollHeight = editor.scrollHeight;
        editor.scrollTo(0, scrollHeight);
    }


    function BHtmlParser(time,text,cid,p,title,datakey){
        let div1 = document.createElement('div');
        div1.setAttribute("class", "ql-tag-blot");
        div1.setAttribute("data-oid_type", "0");
        div1.setAttribute("data-cid", cid);
        div1.setAttribute("data-epid", 0);
        div1.setAttribute("data-status", "0");
        div1.setAttribute("data-index", p);
        div1.setAttribute("data-seconds", time);
        div1.setAttribute("data-cid-count", "1");
        div1.setAttribute("data-key", datakey);
        div1.setAttribute("data-title", title);
        div1.setAttribute("data-desc", text);
        // let span2 = document.createElement('span');
        //  span2.setAttribute("contenteditable", "false");
        let div2 = document.createElement('div');
        div2.setAttribute("class", "time-tag-item");
        div2.setAttribute("contenteditable", "false");

        let i3 = document.createElement('i');
        i3.setAttribute("class", "bili-note-iconfont iconicon_flag_s");

        let span4 = document.createElement('span');
        span4.setAttribute("class", "time-tag-item__text");

        let desc5 = document.createElement('desc');
        desc5.setAttribute("class", "time-tag-item__desc");
        desc5.setAttribute("title", text);
        desc5.append(text);

        div1.append(div2);
        div2.append(i3);
        div2.append(span4);
        span4.append(secondsToMinutes(time));
        span4.append(desc5);
        return div1;
    }

    function secondsToMinutes(seconds) {
        var minutes = Math.floor(seconds / 60);
        var formattedMinutes = ("0" + minutes).slice(-2);
        var formattedSeconds = ("0" + (seconds % 60)).slice(-2);
        return formattedMinutes + ":" + formattedSeconds;
    }

    const state = window.__INITIAL_STATE__;

    function getTimestamp(){
        const editor = document.querySelector(".ql-editor")? document.querySelector(".ql-editor"): null;
        let p = state.p || 1;
        let cid = state.videoData.pages[p - 1].cid;
        let title = state.videoData.title;

        let videos = document.getElementsByTagName('video');
        let video = videos[0];
        let currentTime=video.currentTime;

        let pnew = document.createElement('p');
        var t = (new Date).getTime();

        var timeDom= BHtmlParser(parseInt(currentTime),'',cid,p,title,t);
       
        timeDom.addEventListener('click', function() {
            let videos1 = document.getElementsByTagName('video');
            let video1 = videos1[0];
            video1.currentTime = currentTime;
            video1.play();
        });

        editor.append(timeDom);
        let brnew = document.createElement('br');
        editor.append(brnew);

        toEnd(editor);
        editor.addEventListener('keydown', event => {
            if (event.keyCode === 37) {
                event.preventDefault();
            }
        });

        video.play();
    }

  })();


         window.onload=function(){
                  let newDiv = document.createElement('div');
                  newDiv.style.position="absolute";
                  newDiv.style.zIndex=99999999999999999999;
                  newDiv.style.cursor="pointer";
                  newDiv.setAttribute("id", "my_icon")
                  newDiv.style.display="none";
                  const svg = document.createElementNS('http://www.w3.org/2000/svg', 'svg');
                  svg.setAttribute('version', '1.1');
                  svg.setAttribute('id', 'mu');
                  svg.setAttribute('xmlns', 'http://www.w3.org/2000/svg');
                  svg.setAttribute('xmlns:xlink', 'http://www.w3.org/1999/xlink');
                  svg.setAttribute('x', '0px');
                  svg.setAttribute('y', '0px');
                  svg.setAttribute('width', '40px');
                  svg.setAttribute('height', '40px');
                  svg.setAttribute('viewBox', '0 0 40 40');
                  svg.setAttribute('enable-background', 'new 0 0 40 40');
                  svg.setAttribute('xml:space', 'preserve');
                  svg.innerHTML = "<image id='image0' width='40' height='40' x='0' y='0' href='' />";
                  newDiv.appendChild(svg);
                  document.body.after(newDiv);

             function getSelect(){
                 let select = null, range = null;
                 let selectedText = "",selectedHtml="";

                 if(window.getSelection){
                     select = window.getSelection();
                     selectedText = select.toString();
                     if (select.rangeCount > 0) {
                         range = select.getRangeAt(0);
                         var docFragment = range.cloneContents();
                         var tempDiv = document.createElement("div");
                         tempDiv.appendChild(docFragment);
                         selectedHtml = tempDiv.innerHTML;
                     }

                 } else if(document.selection){
                     select = document.selection;
                     range = select.createRange();
                     selectedText = range.text;
                     selectedHtml = range.htmlText;
                 }

                 return selectedHtml;
             };

             var currentText=null;
             var interval = "";
             var oBaby =document.body;
             var oIcon = document.getElementById('my_icon');

             oBaby.onmouseup = function (event) {
                 var ev = event || window.event;
                 var left = ev.pageX;
                 var top = ev.pageY;

                 currentText=getSelect();
                 if (currentText.length > 1) {
                     setTimeout(function () {
                         oIcon.style.display = 'block';
                         oIcon.style.left = left + 'px';
                         oIcon.style.top = top-20 + 'px';
                     }, 100);
                 } else {
                     oIcon.style.display = 'none';
                 }
             };
             oBaby.onclick = function (event) {
                 var ev = event || window.event;
                 ev.cancelBubble = true;
             };
             document.onclick = function () {
                 setTimeout(function () {
                     oIcon.style.display = 'none';
                     clearInterval(interval);
                 }, 500);
             };

             oIcon.onclick = function () {
                 let angle = 0;
                 interval = setInterval(() => {
                     angle += 60;
                     this.style.transformOrigin = '20px 20px';
                     this.style.transform = `rotate(${angle}deg)`;

                 }, 80);
                 const editor = document.querySelector(".ql-editor")? document.querySelector(".ql-editor"): null;
                 if(editor){
                     editor.append(getAllComment(currentText));
                     var scrollHeight = editor.scrollHeight;
                     editor.scrollTo(0, scrollHeight);

                 }else{
                     alert("请先打开视频笔记,快捷键:Ctrl+Shift+↑");
                 }
             };

             //comment
             function getComment(htmlString){
                 var tempStr="";
                 let newspan = document.createElement('span');
                 var tStr=parserString2HTML(htmlString);

                 var leftEntry = tStr.querySelector('.left-entry');
                 if(leftEntry!=null){
                     return newspan;
                 }

                 htmlString = htmlString.replace(/<(?!img|br)[^>]*>/g, '');
                 htmlString = htmlString.replace(/<img[^>]*src="data:image[^"]*"[^>]*>/g, '');

                 htmlString = htmlString.replace(/<img[^>]+src="([^"]+)"[^>]+>/gi, function(match, p1) {
                     if (p1.endsWith('.svg')) {
                         return '';
                     }
                     return match;
                 });




                 var newHtml = htmlString.replace(/<img[^>]*>/g, function(match) {
                     var src = match.match(/src="([^"]*)"/)[1];
                     return "<img>"+src+"<img>";
                 });

                 tempStr=newHtml.split(/<img[^>]*>/g)
                 for(var i=0;i<tempStr.length;i++){
                     if(/png/.test(tempStr[i])){
                         let newImg = document.createElement('img');
                         newImg.setAttribute('width', '24px');
                         newImg.setAttribute('height', '24px');
                         newImg.setAttribute('src', tempStr[i]);
                         newspan.append(newImg);
                         //newspan.append("\n")
                     }else{
                         newspan.append(tempStr[i].replace(/<br>/g,"\n"))

                     }
                 }
                 return newspan;
             }

       
             function getAllComment(htmlString){

                 let newspan = document.createElement('span');
                 var tempStr=parserString2HTML(htmlString);
                 var subReplyItem = tempStr.querySelectorAll(".sub-reply-item");
                 var subUserInfo = tempStr.querySelector('.sub-user-info');
                 var replyItem = tempStr.querySelectorAll(".reply-item");
                 var userInfo = tempStr.querySelector('.user-info');
                 if(replyItem!=null && replyItem.length>0){

                     for(let i=0;i<replyItem.length;i++){
                         let rootReplyContainer = replyItem[i].querySelector('.root-reply-container');
                         if(rootReplyContainer!=null){
                         let biliAvatar = rootReplyContainer.querySelector('.root-reply-avatar .avatar .bili-avatar');

                         //主评论头像
                         if(biliAvatar!=null){
                             let imgSrc = biliAvatar.firstElementChild.getAttribute('src');
                             let newImg = document.createElement('img');
                             newImg.setAttribute('width', '24px');
                             newImg.setAttribute('height', '24px');

                             newImg.style.setProperty('border', 'none');
                             newImg.style.setProperty('border-radius', '50%');
                             newImg.style.setProperty('overflow', 'hidden');


                             newImg.setAttribute('src', imgSrc);
                             newspan.append(newImg);
                         }

                         //主评论userid和用户名
                         let userName = rootReplyContainer.querySelector('.content-warp .user-info .user-name');
                         if(userName!=null){
                             let dataUserId=userName.getAttribute('data-user-id');
                             let userNameReal=userName.textContent;
                             let newA = document.createElement('a');
                             newA.setAttribute('href', 'https://space.bilibili.com/'+dataUserId);
                             newA.textContent=userNameReal+":";
                             newspan.append(newA);
                         }

                         }

                         let noteText = replyItem[i].querySelector('.note-text-container .note-text');
                         if(noteText!=null){
                             newspan.append(getComment(noteText.firstElementChild.innerHTML))
                         }else{
                             let replyContentContainer = replyItem[i].querySelector('.root-reply .reply-content');
                             if(replyContentContainer!=null){
                                 newspan.append(getComment(replyContentContainer.innerHTML))
                             }
                         }
                         newspan.append("\n\n")

                         let sReplyItem = replyItem[i].querySelectorAll('.sub-reply-item');
                         for(let j=0;j<sReplyItem.length;j++){
                             newspan.append(avatarUserNameContent(sReplyItem[j],sReplyItem[j]));
                             newspan.append("\n\n")
                         }


                     }

                 }else if(userInfo!=null){
                       //主评论userid和用户名
                         let userName = userInfo.querySelector('.user-info .user-name');
                         if(userName!=null){
                             let dataUserId=userName.getAttribute('data-user-id');
                             let userNameReal=userName.textContent;
                             let newA = document.createElement('a');
                             newA.setAttribute('href', 'https://space.bilibili.com/'+dataUserId);
                             newA.textContent=userNameReal+":";
                             newspan.append(newA);
                         }

                         let noteText = tempStr.querySelector('.note-text-container .note-text');
                         if(noteText!=null){
                             newspan.append(getComment(noteText.firstElementChild.innerHTML))
                         }else{

                             let replyContentContainer = tempStr.querySelector('.reply-content-container .reply-content');
                             if(replyContentContainer!=null){
                                 newspan.append(getComment(replyContentContainer.innerHTML))
                             }
                         }
                         newspan.append("\n\n")

                         let sReplyItem = tempStr.querySelectorAll('.sub-reply-item');

                         for(let j=0;j<sReplyItem.length;j++){
                             newspan.append(avatarUserNameContent(sReplyItem[j],sReplyItem[j]));
                             newspan.append("\n\n")
                         }


                 }else if(subReplyItem!=null && subReplyItem.length>0){
                     let replyContentContainer = tempStr.querySelector('.root-reply .reply-content-container .reply-content');
                     if(replyContentContainer!=null){
                         newspan.append(getComment(replyContentContainer.innerHTML))
                          newspan.append("\n\n")
                     }

                  for(let i=0;i<subReplyItem.length;i++){
                     newspan.append(avatarUserNameContent(subReplyItem[i],subReplyItem[i]));
                     newspan.append("\n\n")
                  }

                 }else if(subUserInfo!=null){
                     newspan.append(avatarUserNameContent(subUserInfo,tempStr));

                 }else{
                  let isSubUserInfo = tempStr.querySelector('.sub-reply-avatar .avatar .bili-avatar');
                   if(isSubUserInfo==null){
                         newspan.append(getComment(htmlString))
                     }
                 }

                 return newspan;
             }



             function parserString2HTML(htmlString){
                 const parser = new DOMParser();
                 const htmlDoc = parser.parseFromString(htmlString, "text/html");
                 return htmlDoc;

             }

             function avatarUserNameContent(subUserInfo,tempStr){
                 let nSpan = document.createElement('span');
                 let subBiliAvatar = subUserInfo.querySelector('.sub-reply-avatar .avatar .bili-avatar');
                     //sub评论头像
                     if(subBiliAvatar!=null){
                         let subImgSrc = subBiliAvatar.firstElementChild.getAttribute('src');
                         let newImg = document.createElement('img');
                         newImg.setAttribute('width', '24px');
                         newImg.setAttribute('height', '24px');

                         newImg.style.setProperty('border', 'none');
                         newImg.style.setProperty('border-radius', '50%');
                         newImg.style.setProperty('overflow', 'hidden');


                         newImg.setAttribute('src', subImgSrc);
                         nSpan.append(newImg);
                     }

                     let subUserName = subUserInfo.querySelector('.sub-user-name');
                     if(subUserName!=null){
                         let dataUserId=subUserName.getAttribute('data-user-id');
                         let UserName=subUserName.textContent;
                          let newA = document.createElement('a');
                         newA.setAttribute('href', 'https://space.bilibili.com/'+dataUserId);
                         newA.textContent=UserName+":";
                         nSpan.append(newA);
                     }

                     let replyContentContainer = tempStr.querySelector('.reply-content-container .reply-content');
                     if(replyContentContainer!=null){
                         nSpan.append(getComment(replyContentContainer.innerHTML))
                     }
                 return nSpan;
             }
         };