Greasy Fork is available in English.

下载用户B站收藏夹中的音频

通过脚本下载收藏夹里面的歌曲,实现免费听歌

// ==UserScript==
// @name         下载用户B站收藏夹中的音频
// @namespace    crystal
// @version      1.0.0
// @description  通过脚本下载收藏夹里面的歌曲,实现免费听歌
// @author       奎里斯托
// @match        https://space.bilibili.com/*
// @grant        none
// @license      MIT
// ==/UserScript==
(function () {
  // 变量前面加gl(global),表示当前变量是全局的
  let GL_popup = null
  let GL_favoriteList = []
  let GL_lateOnList = []
  // 获取收藏夹信息
  function getFavoriteList() {
    return new Promise((resolve, reject) => {
      try {
        ajax({
          url: 'https://api.bilibili.com/x/v3/fav/folder/list4navigate',
        }).then((res) => {
          const result = res
          const [myFavorite, lateOn] = result.data
          GL_favoriteList = myFavorite.mediaListResponse.list
          GL_lateOnList = lateOn.mediaListResponse.list
          resolve(GL_favoriteList)
        })
      } catch (error) {
        reject(error)
      }
    })
  }
  let GL_tableData = []
  let GL_PageCount = 0
  // 根据收藏夹信息获取单个收藏夹的详细信息
  function getFavoriteDetail(searchParams) {
    return new Promise((resolve, reject) => {
      try {
        const defaultParam = {
          media_id: '',
          pn: 1,
          ps: 10,
          keyword: '',
          order: 'mtime',
          type: 0,
          tid: 0,
          platform: 'web',
        }
        const params = {
          ...defaultParam,
          ...searchParams
        }
        ajax({
          url: 'https://api.bilibili.com/x/v3/fav/resource/list',
          params: params
        }).then((res) => {
          const result = res
          const medias = result.data.medias
          GL_PageCount = result.data.info.media_count
          GL_tableData = medias
          GL_pagePsPn.sumPn = Math.ceil(GL_PageCount / GL_pagePsPn.ps)
          initPage(GL_pagePsPn.sumPn, GL_pagePsPn.pn)
          resolve(medias)
        })
      } catch (error) {
        reject(error)
      }
    })
  }
  // 获取单个视频的详细信息
  function getVideoDetail(searchParams) {
    return new Promise((resolve, reject) => {
      const defaultParam = {
        bvid: '',
        cid: '',
        fnval: 4048
      }
      try {
        ajax({
          url: 'https://api.bilibili.com/x/player/playurl',
          params: {
            ...defaultParam,
            ...searchParams
          }
        }).then((res) => {
          const result = res
          const url = result.data.dash.audio[0].base_url
          resolve(url)
        })
      } catch (error) {
        reject(error)
      }
    })
  }
  // 下载音频
  function downloadAudio(url, title) {
    return new Promise((resolve, reject) => {
      try {
        ajax({
          url: url,
          responseType: 'blob',
          credential: false
        }).then((res) => {
          const reader = new FileReader();
          reader.readAsDataURL(res);
          reader.onload = function (e) {
            const a = document.createElement('a');
            a.download = `${title}.mp3`
            a.href = e.target.result;
            document.documentElement.appendChild(a);
            a.click();
            a.remove();
            audioState = false;
            resolve(true)
          }
        })
      } catch (error) {
        reject(error)
      }
    })

  }
  // 基于promise和XMLHttpRequest封装ajax对象
  function ajax(options) {
    return new Promise((resolve, reject) => {
      // 存储的是默认值
      const defaults = {
        type: 'get',
        url: '',
        data: {},
        async: true,
        header: {
          'Content-Type': 'application/json'
        },
        responseType: 'json',
        credential: true,
        params: {},
      };
      // 使用options对象中的属性覆盖defaults对象中的属性
      Object.assign(defaults, options);
      // 先判断是否要拼接参数
      const params = defaults.params
      const paramsKey = Object.keys(params)
      if (paramsKey.length > 0) {
        const str = paramsKey.reduce((pre, cur) => {
          pre = pre + `&${cur}=${params[cur]}`
          return pre
        }, '?')
        defaults.url += str
      }
      const xhr = new XMLHttpRequest();
      // 设置该值就可以自动获取到登录后的权限
      if (defaults.credential) {
        xhr.withCredentials = true;
      }
      xhr.responseType = defaults.responseType
      xhr.open(defaults.type, defaults.url, defaults.async);
      if (defaults.type === 'post') {
        let contentType = defaults.header['Content-Type']
        xhr.setRequestHeader('Content-Type', contentType);
        if (contentType === 'application/json') {
          xhr.send(JSON.stringify(defaults.data))
        }
      } else {
        xhr.send()
      }
      xhr.onreadystatechange = function () {
        if (xhr.readyState !== 4) return
        const response = xhr.response;
        if (xhr.status == 200) {
          resolve(response)
        } else {
          reject(response)
        }
      }
    })
  }
  // 增加操作界面
  function addPanel() {
    const domPanel = `
        <div class="panel">
          <div id="menu" class="mediate hideMenu"></div>
          <div class="menu_list mediate">
            <div data-index="1"></div>
            <div data-index="2"></div>
            <div data-index="3"></div>
          </div>
        </div>
    `
    const bodyDOm = document.body
    const panelDiv = document.createElement('div')
    panelDiv.innerHTML = domPanel
    bodyDOm.appendChild(panelDiv)
    // 这里必须提前添加好,否则会出现第一次点击时菜单不正常出现的问题
    const menuListDom = document.querySelector('.menu_list')
    const menuListClass = menuListDom.classList
    menuListClass.add('hideMenu')
    const childrenDom = menuListDom.children
    Array.prototype.forEach.call(childrenDom, (ele, idx) => {
      setStyle(ele, { transform: 'translate(0, 0)', opacity: 0 })
    })
  }
  // 展开/隐藏菜单
  function switchMenu() {
    const menuListDom = document.querySelector('.menu_list')
    const menuListClass = menuListDom.classList
    const result = Array.prototype.includes.call(menuListClass, 'hideMenu')
    if (result) {
      // 执行展示
      menuListClass.remove('hideMenu')
      menuListClass.add('showMenu')
      const childrenDom = menuListDom.children
      const moveKeys = ['translate(0, -70px)', 'translate(0, -125px)', 'translate(0, -180px)']
      Array.prototype.forEach.call(childrenDom, (ele, idx) => {
        setStyle(ele, { transform: moveKeys[idx], opacity: 1 })
      })
    } else {
      menuListClass.remove('showMenu')
      menuListClass.add('hideMenu')
      const childrenDom = menuListDom.children
      Array.prototype.forEach.call(childrenDom, (ele) => {
        setStyle(ele, { transform: 'translate(0, 0)', opacity: 0 })
      })
    }
  }
  // 初始化
  function init() {
    addPanel()
    GL_popup = new Popup()
    const menuDom = document.getElementById('menu')
    menuDom.addEventListener('click', switchMenu);

    const menuListDom = document.querySelector('.menu_list')
    menuListDom.addEventListener('click', function (e) {
      const index = e.target.dataset.index
      if (index === '3') {
        showFavoritePage()
      } else {
        alert('更多功能,尽情期待!!!!!!!!!!!!!!')
      }
    })
    addComplexCss()
  }
  // 为了防止样式被重复添加,这里对于复杂的样式提前
  function addComplexCss() {
    //分页器的样式
    const paginationCss = `
      .pagination {
        display: flex;
        height: 35px;
        margin: 0 auto;
        position: absolute;
        right: 0;
        margin-bottom: 4px
      }

      .pagination span a {
        box-sizing: border-box;
        text-decoration: none;
        color: black;
      }

      .pagination span {
        text-align: center;
        width: 40px;
        height: 35px;
        line-height: 35px;
        margin: 0px 2px;
      }

      .pagination span i {
        font-size: 10px;
        font-weight: 100;
      }

      .pagination span img {
        object-fit: cover;
        height: 100%;
      }

      .pagination .pageStyle a {
        display: block;
        text-align: center;
        width: 40px;
        height: 35px;
        line-height: 35px;
        /* background-color: bisque; */
        border: 1px solid #ccc;
        border-radius: 5px;
      }

      .pagination .pageStyle a:hover {
        border: 1px solid rgb(27, 129, 121);
      }

      .pagination .active {
        background-color: rgba(251, 114, 153);
        color:#fff;
      }
    `
    // 表格样式
    const tableCss = `
    .favorite_detail table{
      width:100%;
      border: 1px solid #000;
      border: 1px solid;
    }
    .favorite_detail table thead tr{
      background-color: rgba(251, 114, 153);
      color:#fff;
    }
    .favorite_detail table thead tr th {
      border-bottom: 0 !important;
    }
    .favorite_detail table tbody tr:nth-child(2n){
      background-color: rgba(251, 114, 153,0.5);
    }
    .favorite_detail table tbody tr:nth-child(2n+1){
      background-color: #fff;
    }
    .favorite_detail table tr, th, td {
      text-align: center;
    }
    .favorite_detail table tr td{
      padding: 1px 1px;
      line-height: 1rem;
    }
    .favorite_detail table tr td input{
      width:95%;
      line-height: 1rem;
    }
    `
    // 操作面板的样式
    const panelCss = `
    .panel {
      position: fixed;
      top: 70%;
      left: 30px;
      width: 70px;
      height: 70px;
      border-radius: 50%;
      box-sizing: border-box;
    }

    .mediate {
      position: absolute;
      top: 50%;
      left: 50%;
      transform: translate(-50%, -50%);
    }

    #menu {
      width: 50px;
      height: 50px;
      border-radius: 10%;
      background-size: cover;
      background-repeat: no-repeat;
      background-color: rgba(251, 114, 153);
      background-image: url();
      position: relative;
      z-index: 10;
    }
    .menu_list {
      width: 50px;
      height: 50px;
      z-index: 9;
      display: flex;
      justify-content: center;
    }

    .menu_list div {
      width: 42px;
      height: 42px;
      border-radius: 10%;
      position: absolute;
      background-color: rgba(251, 114, 153);
      background-size: cover;
      background-repeat: no-repeat;
      opacity: 0;
      transition: all 0.4s ease-in-out;
    }

    .menu_list div:nth-child(1) {
      transform: translate(0, 0);
      background-image: url();
    }

    .menu_list div:nth-child(2) {
      transform: translate(0, 0);
      background-image: url();
    }

    .menu_list div:nth-child(3) {
      transform: translate(0, 0);
      background-image: url();
    }
    `
    const strCss = paginationCss + tableCss + panelCss
    addStyle(strCss)
  }
  // 展示收藏夹弹窗
  function showFavoritePage() {
    const cssContent = `
      .favorite_header{
        min-height: 100px;
      }
      .favorite_btn{
        display: flex;
        justify-content: flex-start;
        align-items: center;
      }
      .favorite_content{
        display: flex;
        justify-content: center;
        width: 100%;
        max-height:75vh;
        margin-top:4px;
      }
      .favorite_list{
        flex: 1 1 22%;
        overflow-y:auto;
        min-height:400px;
      }
      .favorite_list div{
        color: white;
        background-color: rgba(251, 114, 153);
        border: 1px solid rgb(219, 66, 110);
        margin: 5px 0;
        padding: 2px 4px;
      }
      .favorite_list div:hover{
        background-color: #fff;
        color: rgba(251, 114, 153);
      }
      .favorite_detail{
        flex: 1 1 78%;
        padding: 5px;
        overflow-y:auto;
        position: relative;
      }
      .table_box{
        max-height: 50vh;
        overflow-y: auto;
      }
      .btn {
        background-color: rgba(251, 114, 153);
        border: 2px solid rgb(219, 66, 110);
        border-radius: 6px; 
        color: white;
        padding: 2px 8px;
        text-align: center;
        display: inline-block;
        font-size: 16px;
        margin: 4px 2px;
        transition-duration: 0.4s;
        cursor: pointer;
        text-decoration: none;
        text-transform: uppercase;
        user-select: none;
      }
      /* 悬停样式 */
      .btn:hover {
        background-color: #fff;
        color: rgba(251, 114, 153);
      }
    `
    const domContent = `
        <div class="container">
        <div class="favorite_header">
          <div class="favorite_hint">
            <p>1.请先选择你要查询的收藏夹,并点击查询。之后才能点击下载,下载当前表格展示的歌曲</p>
            <p>2.默认情况下,文件名的格式为“视频名称.mp3”,你可以填入在文件名称一列输入当前自定义的文件名</p>
            <p>3.默认情况下,点击下载将下载当页全部的格式。倘若在表格进行了勾选,那么只会下载勾选的歌曲</p>
            <p>4.为了保证下载的稳定性,请下载完当前页的歌曲,在点击下一页进行下载</p>
          </div>
          <div class="favorite_btn">
            <div class="btn" data-type="search">查询</div>
            <div class="btn" data-type="download">下载</div>
          </div>
        </div>
        <div class="favorite_content">
          <div class="favorite_list"></div>
          <div class="favorite_detail">
            <div class="table_box"></div>
            <div class="pagination"></div>
          </div>
        </div>
      </div>
    `
    addStyle(cssContent)
    GL_popup.dialog({
      width: '70vw',
      title: '收藏夹信息',
      content: domContent,
      onReady: function () {
        // 监听单选框的改变
        const favoriteListDom = document.querySelector('.favorite_list')
        favoriteListDom.addEventListener('click', favoriteListChange)
        getFavoriteList().then((value) => {
          createFavoriteList(value)
        })
        //操作按钮绑定事件
        const favoriteBtnDom = document.querySelector('.favorite_btn')
        favoriteBtnDom.addEventListener('click', favoriteBtnChange)
        // 创造表格
        createTable()
        // 监听全选框
        const selectAllDom = document.getElementById('table_selectAll')
        selectAllDom.addEventListener('click', selectAllOrCancel)
        // 监听单个选择框
        const tbody = document.getElementById('table_body')
        tbody.addEventListener('click', selectSingle)
      }
    })
  }
  function createFavoriteList(list) {
    const favoriteListDom = document.querySelector('.favorite_list')
    let template = ''
    list.forEach((item) => {
      const tpl = `
        <div>
          <input type="radio" id="${item.id}" name="favoriteList" value="${item.id}" che>
          <label for="${item.id}">${item.title}</label>
        </div>
      `
      template += tpl
    })
    favoriteListDom.innerHTML = template
  }
  // 初始化表格
  function createTable() {
    const tableBoxDom = document.querySelector('.table_box')
    const table = document.createElement('table')
    const thead = document.createElement('thead')
    const tbody = document.createElement('tbody')
    tbody.id = 'table_body'
    const headContent = `
      <tr>
          <th>
              <input type="checkbox" name="check_sum" id="table_selectAll" />
          </th>
          <th>序号</th>
          <th>视频名称</th>
          <th>up主名称</th>
          <th>文件名称</th>
      </tr>
    `
    thead.innerHTML = headContent
    table.appendChild(thead)
    table.appendChild(tbody)
    tableBoxDom.appendChild(table)
  }
  // 重绘表格
  function resetTable(list) {
    const tbody = document.getElementById('table_body')
    let bodyContent = ''
    list.forEach((ele, idx) => {
      const tpl = `
        <tr>
          <td>
              <input type="checkbox" name="check_item" data-index="${idx}"/>
          </td>
          <td>${(GL_pagePsPn.pn - 1) * GL_pagePsPn.ps + idx + 1}</td>
          <td>${ele.title}</td>
          <td>${ele.upper.name}</td>
          <td>
            <input type="text" name="fileName" />
          </td>
        </tr>
      `
      bodyContent += tpl
    })
    tbody.innerHTML = bodyContent
  }
  let GL_favoriteId = null
  let GL_selectFavoriteId = null
  // 分页记录
  let GL_pagePsPn = {
    ps: 10,
    pn: 1,
    sumPn: 0,
  }
  function initPage(sumPN, pn) {
    const pageEle = document.querySelector(".pagination");
    new PageClass(pageEle, sumPN, pn, function (page) {
      GL_pagePsPn.pn = page
      const params = {
        pn: GL_pagePsPn.pn,
        media_id: GL_favoriteId,
      }
      getFavoriteDetail(params).then((value) => {
        resetTable(value)
      })
    });
  }
  // 监听收藏夹的修改
  function favoriteListChange(e) {
    // 过滤掉其他元素带来的干扰
    if (e.target.localName !== 'input') return
    GL_selectFavoriteId = e.target.value
  }
  // 监听查询/下载的按钮事件
  function favoriteBtnChange(e) {
    const type = e.target.dataset.type
    if (type === 'search') {
      search()
    } else if (type === 'download') {
      beforeDownload()
    }
  }
  function search() {
    if (!GL_selectFavoriteId) {
      alert('请先选择收藏夹')
      return
    }
    // 判断收藏夹是否发生了改变
    if (GL_favoriteId !== GL_selectFavoriteId) {
      GL_pagePsPn.pn = 1
    }
    GL_favoriteId = GL_selectFavoriteId
    const params = {
      media_id: GL_favoriteId,
      pn: GL_pagePsPn.pn,
      ps: GL_pagePsPn.ps,
    }
    getFavoriteDetail(params).then((value) => {
      resetTable(value)
    })
  }
  let GL_downloadTableData = []
  function beforeDownload() {
    if (GL_tableData.length <= 0) {
      alert('请先查询出收藏夹信息')
      return
    }
    // 先设置一次自定义文件名
    setInfo()
    const selectAllDom = document.getElementById('table_selectAll')
    const checkList = document.getElementsByName('check_item')
    const result = arrayMethod('every', checkList, (ele) => ele.checked)
    // 表示全部下载
    if (selectAllDom.checked || result) {
      GL_downloadTableData = GL_tableData
    } else {
      GL_downloadTableData = GL_tableData.filter((ele, idx) => {
        if (checkList[idx].checked) {
          return ele
        }
      })
    }
    download()

  }
  async function download(index = 0) {
    const value = GL_downloadTableData[index]
    if (!value) return
    const { bvid, title, ugc: { first_cid: cid }, fileName } = value
    const url = await getVideoDetail({ bvid, cid })
    let name = fileName ? fileName : title
    downloadAudio(url, name).then((res) => {
      if (res) {
        // 通过递归调用,把歌曲全部下载完
        index++
        download(index)
      }
    })

  }
  // 设置自定义的文件名字
  function setInfo() {
    const fileNameDomList = document.getElementsByName('fileName')
    const fileNameList = arrayMethod('map', fileNameDomList, (ele) => ele.value)
    GL_tableData.forEach((ele, idx) => {
      if (fileNameList[idx]) {
        ele.fileName = fileNameList[idx]
      }
    })
  }
  // 全选和反选
  function selectAllOrCancel() {
    const selectAllDom = document.getElementById('table_selectAll')
    const checkList = document.getElementsByName('check_item')
    arrayMethod('forEach', checkList, (ele) => {
      ele.checked = selectAllDom.checked
    })
  }
  // 单选
  function selectSingle(e) {
    const selectAllDom = document.getElementById('table_selectAll')
    const checkList = document.getElementsByName('check_item')
    const result = arrayMethod('every', checkList, (ele) => { return ele.checked })
    selectAllDom.checked = result
  }
  // 让类数组使用数组的方法
  function arrayMethod(type, list, fn) {
    return Array.prototype[type].call(list, fn)
  }
  // 增加样式
  function addStyle(css) {
    const myStyle = document.createElement('style');
    myStyle.textContent = css
    // 插入到head或者html标签中
    const doc = document.head || document.documentElement;
    doc.appendChild(myStyle);
  }
  // 增加样式方法2
  function setStyle(ele, styleObj) {
    for (let attr in styleObj) {
      ele.style[attr] = styleObj[attr]
    }
  }
  // 全局弹窗
  class Popup {
    _mask = null
    _dialog = null
    _dialogHeader = null
    _dialogContent = null
    _dialogTitle = null
    _dialogclose = null
    constructor() {
      // 遮罩
      this._mask = document.createElement('div')
      this.setStyle(this._mask, {
        "width": '100%',
        "height": '100%',
        "backgroundColor": 'rgba(0, 0, 0, .6)',
        "position": 'fixed',
        "left": "0px",
        "top": "0px",
        "bottom": "0px",
        "right": "0px",
        "z-index": "99999"
      })
      // 弹窗
      this._dialog = document.createElement('div')
      this.setStyle(this._dialog, {
        "overflow": 'hidden',
        "backgroundColor": '#fff',
        "boxShadow": '0 0 2px #999',
        "position": 'absolute',
        "left": '50%',
        "top": '50%',
        "transform": 'translate(-50%,-50%)',
        "borderRadius": '3px'
      })
      this._mask.appendChild(this._dialog)
      // 弹窗头部
      this._dialogHeader = document.createElement('div')
      this.setStyle(this._dialogHeader, {
        "width": '100%',
        "height": '40px',
        "lineHeight": '40px',
        "boxSizing": 'border-box',
        "background-color": 'rgba(251, 114, 153)',
        "color": '#FFF',
        "text-align": 'center',
        "font-weight": "700",
        "font-size": "16px"
      })
      this._dialog.appendChild(this._dialogHeader)
      // 弹窗内容
      this._dialogContent = document.createElement('div')
      this.setStyle(this._dialogContent, {
        "max-height": '70vh',
        "overflow-y": 'auto',
        "min-width": '400px',
        "width": '50vw',
      })
      this._dialog.appendChild(this._dialogContent)
      // 标题
      this._dialogTitle = document.createElement('span')
      this._dialogTitle.innerText = '全局弹窗'
      this.setStyle(this._dialogTitle, {
        "textDecoration": 'none',
        "color": '#666',
        "position": 'absolute',
        "left": '10px',
        "top": '0px',
        "fontSize": '25px',
        "color": '#FFF',
        "display": "inline-block"
      })
      this._dialogHeader.appendChild(this._dialogTitle)
      // 关闭按钮
      this._dialogclose = document.createElement('span')
      this._dialogclose.innerText = 'X'
      this._dialogclose.onclick = () => this.close(clearGlobal)
      // 设置样式
      this.setStyle(this._dialogclose, {
        "textDecoration": 'none',
        "color": '#666',
        "position": 'absolute',
        "right": '10px',
        "top": '0px',
        "fontSize": '25px',
        "color": '#FFF',
        "display": "inline-block",
        "cursor": "pointer"
      })
      this._dialogHeader.appendChild(this._dialogclose)

    }
    // 初始化dialog的内容区
    initContent(content, title = "全局弹窗", width = '') {
      // 置空,前后干扰
      this._dialogContent.innerHTML = ''
      this._dialogTitle.innerText = ''
      this._dialogTitle.innerText = title
      this._dialogContent.innerHTML = content
      width && this.setStyle(this._dialogContent, {
        "width": width
      })
    }
    dialog(params) {
      const { width, content, title, onReady } = params
      this.initContent(content, title, width)
      document.body.appendChild(this._mask)
      // 在弹窗创造完成后,执行这个方法
      if (onReady && typeof onReady === 'function') {
        setTimeout(() => {
          onReady(this)
        })
      }
    }
    close(fn) {
      document.body.removeChild(this._mask)
      if (typeof fn === 'function') {
        // 这里可以在执行额外的函数,来处理一些东西
        fn()
      }
    }
    setStyle(ele, styleObj) {
      for (let attr in styleObj) {
        ele.style[attr] = styleObj[attr]
      }
    }
  }
  // 分页器
  class PageClass {                               //定义一个分页器类      
    constructor(ele, pageNum, page, cb) {   //需要传入4个参数,第一个容器元素,第二个页面总数,第三个当前页面数,第四个为回调函数
      this.ele = ele;            //定义属性
      this.pageNum = pageNum;
      this.page = page;
      this.cb = cb;
      this.renderPage();     //执行渲染的方法
      this.operate();         //给实例化对象绑定各种操作的方法
    }
    renderPage() {                   //在类的原型上定义一个渲染的方法
      let str = '';
      if (this.pageNum > 5) {     //判断当前分页的页面总数是否超过5页
        if (this.page <= 4) {         //如果页面总数大于5  ,再判断当前页是否小于或者等于第四页
          for (let i = 0; i < 5; i++) {        //如果当前页是小于等于4以内的页数 ,遍历1到5
            if ((i + 1) == this.page) {       //判断当前页是否等于 当前   索引值(索引值从0开始) +  1
              str += `<span class="pageStyle" myPage="${i + 1}"><a href="javascript:;" class="active">${i + 1}</a></span>`;//等于的话说明渲染的是当前页,给当前页一个active的类名进行渲染
            } else {  //如果渲染的不是当前页
              str += `<span class="pageStyle" myPage="${i + 1}"><a href="javascript:;">${i + 1}</a></span>`;//普通渲染就行了
            }
          }
          str += `<span><i>•••</i></span><span class="pageStyle" myPage="${this.pageNum}"><a href="javascript:;">${this.pageNum}</a></span>`;//最后渲染一个最末尾页
        } else if (this.page > 4 && this.page < this.pageNum - 3) {   //判断当前页是否大于第四页,且小于最大页数减去 3 
          str += `<span class="pageStyle" myPage="1"><a href="javascript:;">1</a></span><span><i>•••</i></span>`; //渲染一个首页
          for (let i = this.page - 3; i < this.page + 2; i++) {
            if ((i + 1) == this.page) { //判断当前正在渲染的是否为当前页
              str += `<span class="pageStyle" myPage="${i + 1}"><a href="javascript:;" class="active">${i + 1}</a></span>`;//如果是当前页,给一个active类名进行渲染
            } else {
              str += `<span class="pageStyle" myPage="${i + 1}"><a href="javascript:;">${i + 1}</a></span>`;//如果不是,普通渲染就行
            }
          }
          str += `<span><i>•••</i></span><span class="pageStyle" myPage="${this.pageNum}"><a href="javascript:;">${this.pageNum}</a></span>`;//渲染一个尾页
        } else if (this.page >= this.pageNum - 3) {//如果当前页数大于或者等于最大页数 - 3 
          str += `<span class="pageStyle" myPage="1"><a href="javascript:;">1</a></span><span><i>•••</i></span>`; //渲染一个首页
          for (let i = this.pageNum - 5; i < this.pageNum; i++) {//从倒数第5页开始渲染
            if ((i + 1) == this.page) {   //如果渲染的是当前页
              str += `<span class="pageStyle" myPage="${i + 1}"><a href="javascript:;" class="active">${i + 1}</a></span>`;//增加一个active类名,进行渲染
            } else {
              str += `<span class="pageStyle" myPage="${i + 1}"><a href="javascript:;">${i + 1}</a></span>`;//否则普通渲染
            }
          }
        }
      } else {        //如果最大页数小于等于5
        for (let i = 0; i < this.pageNum; i++) {     //直接渲染到当前最大页数
          if ((i + 1) == this.page) {           //判断渲染的是否为当前页数
            str += `<span class="pageStyle" myPage="${i + 1}"><a href="javascript:;" class="active">${i + 1}</a></span>`;//如果是增加一个active类名,进行渲染
          } else {
            str += `<span class="pageStyle" myPage="${i + 1}"><a href="javascript:;">${i + 1}</a></span>`;//否则普通渲染就行了
          }
        }
      }
      //将所有内容渲染到容器盒子里
      //第一页
      //上一页
      //中间页面
      //下一页
      //最后一页
      this.ele.innerHTML = `
          <span class="first"><img src=""></span>
          <span class="prev"><img src=""></span>
          ${str}
          <span class="next"><img src=""></span>
          <span class="last"><img src=""></span>
      `
    }
    operate() {                  //在类原型上定义一个operate的方法,给这个类绑定点击事件
      const _that = this;        //申明一个常量接收this
      let firstEle = this.ele.querySelector(".first");    //获取第一页的元素
      let lastEle = this.ele.querySelector(".last");      //获取最后一页的元素
      let nextEle = this.ele.querySelector(".next");      //获取下一页的元素myPage.active
      let prevEle = this.ele.querySelector(".prev");      //获取上一页的元素
      let pageStyleEles = this.ele.querySelectorAll(".pageStyle");    //获取所有的中间页面的元素
      firstEle.onclick = function () {     //当点击第一页时
        _that.page = 1;                //让page属性重新赋值为1
        _that.cb(_that.page);           //并将page属性通过回调函数cb传递出去
      }
      lastEle.onclick = function () {      //当点击最后一页时
        _that.page = _that.pageNum;    //让page属性重新赋值为页面最大值
        _that.cb(_that.page);          //并将page属性通过回调函数cb传递出去
      }
      nextEle.onclick = function () {      //当点击下一页时
        if (_that.page < _that.pageNum) {   //先判断当前页是否小于最大页
          _that.page = _that.page + 1;   //如果没有超过,给page属性自增1
          _that.cb(_that.page);           //并将page属性通过回调函数cb传递出去
        }
      }
      prevEle.onclick = function () {      //当点击上一页时
        if (_that.page > 1) {           //先判断当前页是否大于第一页
          _that.page = _that.page - 1;   //如果是大于第一页的话,page属性自减1
          _that.cb(_that.page);           //并将page属性通过回调函数cb传递出去
        }
      }
      pageStyleEles.forEach(function (pageStyleEle) {       //遍历获取到中间页面的所有元素
        pageStyleEle.onclick = function () {                //当点击其中一页时
          _that.page = parseInt(this.getAttribute("myPage")); //获取这个元素自定义属性,myPage,属性值是当前页 ,并将当前页赋值给page属性
          _that.cb(_that.page);                           //将page属性通过回调函数cb传递出去
        }
      });
    }
  }
  // 全局变量置空
  function clearGlobal() {
    GL_favoriteId = null
    GL_selectFavoriteId = null
    GL_downloadTableData = []
    GL_favoriteList = []
    GL_lateOnList = []
    GL_PageCount = 0
    GL_pagePsPn = {
      ps: 10,
      pn: 1,
      sumPn: 0,
    }
    GL_tableData = []
  }
  setTimeout(() => {
    init()
  }, 500)
}())