bilibili自动发送直播弹幕

bilibili自动发送直播弹幕, 刷屏和互动专用, 右上角插件菜单中启动

目前為 2024-08-06 提交的版本,檢視 最新版本

// ==UserScript==
// @name          bilibili自动发送直播弹幕
// @namespace     https://greasyfork.org/zh-CN/users/1196880-ling2ling4
// @version       1.1.2
// @author        Ling2Ling4
// @description   bilibili自动发送直播弹幕, 刷屏和互动专用, 右上角插件菜单中启动
// @license       AGPL-3.0-or-later
// @icon 
// @match         *://live.bilibili.com/*
// @run-at        document-end
// @grant         GM_registerMenuCommand
// @grant         GM_notification
// @noframes
// @compatible    chrome
// @compatible    edge
// @compatible    firefox
// ==/UserScript==

(() => {
  "use strict";
  function getListAndWeight(infoItems, weightArr = null) {
    const newWeightArr = [],
      newList = [];
    let flag = !1;
    for (let i = 0; i < infoItems.length; i++) {
      const item = infoItems[i];
      let num = weightArr && weightArr.length > 0 ? weightArr[i] : 1;
      if (
        item &&
        (item.includes("--") || item.includes("=>") || item.includes("=tmp>"))
      ) {
        let arr;
        (arr = item.split("=tmp>")),
          1 === arr.length &&
            ((arr = item.split("=>")),
            1 === arr.length && (arr = item.split("--")));
        let curNum = +arr[1];
        if ((!curNum && 0 !== curNum) || "" === arr[1]) {
          newWeightArr.push(num), newList.push(item);
          continue;
        }
        (flag = !0), newWeightArr.push(curNum), newList.push(arr[0]);
      } else newWeightArr.push(num), newList.push(item);
    }
    return flag
      ? { tagList: newList, weightArr: newWeightArr }
      : { tagList: newList, weightArr: new Array(newList.length).fill(1) };
  }
  function getRandomWeight(weightArr) {
    const num = (function getRandom(min, max) {
      return Math.floor(Math.random() * (max - min + 1) + min);
    })(
      1,
      weightArr.reduce((a, b) => parseInt(a) + parseInt(b), 0)
    );
    let curSum = 0;
    for (let i = 0; i < weightArr.length; i++) {
      if (((curSum += parseInt(weightArr[i])), num <= curSum)) return i;
    }
    return 0;
  }
  const info = {
    data: {
      text: "",
      time: 5,
      stopTime: 30,
      baseText: "",
      baseTime: 5,
      baseStopTime: 30,
      curIndex: 0,
      startTime: 0,
    },
    keys: {
      text: "auto-send-dm_text",
      time: "auto-send-dm_time",
      stopTime: "auto-send-dm_stopTime",
    },
    txt: {
      textMin: "请设置自动发送的直播弹幕的内容\n默认: base\n当前: current",
      text: "请设置自动发送的直播弹幕的内容\n当前: current\n————也可书写多条弹幕\n    1.若每条弹幕用 ;; 分隔(两个分号,中英文字符都可以), 则每次将随机发送其中的一条弹幕\n    2.若每条弹幕用 == 分隔, 则依次发送每条弹幕\n    可在每条弹幕的后面通过 --N 的格式添加权重, 权重越高则越容易发送该条弹幕或重复次数越多(默认权重为1), 权重之和可以超过100.\n————如:\n    666;;赞--9;;好耶--90  表示发送666的概率为1%, 赞的概率9%, 好耶的概率90%\n    666==赞--10  表示发送1次666后发送10次赞",
      time: "请设置弹幕发送的间隔时间 (单位秒)\n默认: base\n当前: current",
      stopTime:
        "请设置运行时长(分钟), 0表示运行后不自动停止\n默认: base\n当前: current",
    },
    classList: {
      textarea: "#control-panel-ctnr-box .chat-input-ctnr textarea",
      btn: "#control-panel-ctnr-box .bottom-actions button",
    },
    doms: {},
    timerId: null,
  };
  function getDoms() {
    (info.doms.textarea = document.querySelector(info.classList.textarea)),
      (info.doms.btn = document.querySelector(info.classList.btn)),
      console.log(info.doms.textarea, info.doms.btn);
  }
  function getSendText(str) {
    const newStr = str.replaceAll(";", ";");
    let arr = [],
      text = str;
    if (newStr.includes(";;"))
      (arr = newStr.split(";;")),
        (text = (function getRandomWeightItem(infoItems) {
          const result = getListAndWeight(infoItems);
          return result.tagList[getRandomWeight(result.weightArr)];
        })(arr));
    else {
      arr = newStr.split("==");
      const obj = (function getOrderItem(infoItems, index = 0) {
        const result = getListAndWeight(infoItems),
          oldArr = result.tagList,
          sizeArr = result.weightArr,
          len = sizeArr.reduce((a, b) => parseInt(a) + parseInt(b), 0),
          curP = index + 1;
        let curRange = 0;
        for (let i = 0; i < oldArr.length; i++)
          if (((curRange += parseInt(sizeArr[i])), curP <= curRange))
            return { value: oldArr[i], length: len };
        return { value: oldArr[0], length: len };
      })(arr, info.data.curIndex);
      (info.data.curIndex = (info.data.curIndex + 1) % obj.length),
        (text = obj.value);
    }
    return text;
  }
  function start() {
    stop();
    let errorTimer = null;
    const sendDm = () => {
      if (
        !(
          (info.doms.textarea && info.doms.btn) ||
          (getDoms(), info.doms.textarea && info.doms.btn)
        )
      )
        return (
          info.doms.textarea || console.log("文本框的dom获取失败"),
          info.doms.btn || console.log("发送按钮的dom获取失败"),
          info.timerId && clearInterval(info.timerId),
          errorTimer && clearTimeout(errorTimer),
          (errorTimer = setTimeout(() => {
            getDoms(),
              info.doms.textarea && info.doms.btn
                ? (info.timerId = setInterval(sendDm, 1e3 * info.data.time))
                : (info.timerId && clearInterval(info.timerId),
                  GM_notification({
                    title: "脚本启动失败",
                    text: "脚本启动失败, 请前往 https://greasyfork.org/zh-CN/users/1196880-ling2ling4 进行反馈",
                    timeout: 1e4,
                  }));
          }, 5e3)),
          -1
        );
      const sendText = getSendText(info.data.text);
      if (
        ((info.doms.textarea.value = sendText),
        (function emitEvent(ele, eventType) {
          try {
            if (ele.dispatchEvent) {
              var evt = new Event(eventType, { bubbles: !1, cancelable: !1 });
              ele.dispatchEvent(evt);
            } else ele.fireEvent && ele.fireEvent("on" + eventType);
          } catch (e) {}
        })(info.doms.textarea, "input"),
        info.doms.btn.click(),
        console.log("发送弹幕:", sendText),
        0 !== info.data.stopTime)
      ) {
        if (
          (new Date().getTime() - info.data.startTime) / 1e3 / 60 >
          info.data.stopTime
        )
          return (
            clearInterval(info.timerId),
            GM_notification({
              text: `结束运行, 已运行${info.data.stopTime}分钟`,
              timeout: 4500,
            }),
            console.log(`结束运行, 已运行${info.data.stopTime}分钟`),
            -1
          );
      }
    };
    GM_notification({ text: "开始发送弹幕", timeout: 3500 }),
      (info.data.startTime = new Date().getTime());
    -1 !== sendDm() &&
      (info.timerId = setInterval(sendDm, 1e3 * info.data.time));
  }
  function stop(isLog = !1) {
    isLog && GM_notification({ text: "已停止发送弹幕", timeout: 3500 }),
      info.timerId && clearInterval(info.timerId),
      (info.data.curIndex = 0);
  }
  function setTextItem({
    txt,
    base,
    key,
    isChangeTxt = !0,
    verification = null,
  } = {}) {
    let val = localStorage.getItem(key);
    null == val &&
      ((val = base),
      "string" != typeof base && (base = JSON.stringify(base)),
      localStorage.setItem(key, base)),
      isChangeTxt && (txt = txt.replace("base", base).replace("current", val));
    let newVal = prompt(txt, val);
    return (
      null !== newVal &&
      ("function" != typeof verification ||
        ((newVal = verification(newVal, val)), null !== newVal)) &&
      newVal !== val &&
      ("string" != typeof newVal && (newVal = JSON.stringify(newVal)),
      localStorage.setItem(key, newVal),
      !0)
    );
  }
  !(function main() {
    !(function getData() {
      (info.data.text =
        localStorage.getItem(info.keys.text) || info.data.baseText),
        (info.data.time =
          +localStorage.getItem(info.keys.time) || info.data.baseTime),
        (info.data.stopTime = +localStorage.getItem(info.keys.stopTime)),
        0 === info.data.stopTime ||
          info.data.stopTime ||
          (info.data.stopTime = info.data.baseStopTime);
    })(),
      getDoms(),
      (function setMenu() {
        GM_registerMenuCommand("开始自动发送弹幕", () => {
          info.data.text
            ? start()
            : (setTextItem({
                txt: info.txt.textMin,
                base: info.data.baseText,
                key: info.keys.text,
                verification: (newVal, val) =>
                  newVal
                    ? ((info.data.text = newVal), newVal)
                    : ((info.data.text = val), val),
              }),
              info.data.text ? start() : stop());
        }),
          GM_registerMenuCommand("停止", () => {
            stop(!0);
          }),
          GM_registerMenuCommand("设置", () => {
            setTextItem({
              txt: info.txt.text,
              base: info.data.baseText,
              key: info.keys.text,
              verification: (newVal, val) =>
                newVal
                  ? ((info.data.text = newVal), newVal)
                  : ((info.data.text = val), val),
            }),
              setTextItem({
                txt: info.txt.time,
                base: info.data.baseTime,
                key: info.keys.time,
                verification: (newVal, val) =>
                  +newVal
                    ? ((info.data.time = +newVal), newVal)
                    : ((info.data.time = +val), val),
              }),
              setTextItem({
                txt: info.txt.stopTime,
                base: info.data.baseStopTime,
                key: info.keys.stopTime,
                verification: (newVal, val) =>
                  0 == +newVal
                    ? ((info.data.stopTime = +newVal), newVal)
                    : +newVal < 0 || !newVal
                    ? ((info.data.stopTime = +val), val)
                    : ((info.data.stopTime = +newVal), newVal),
              });
          });
      })();
  })();
})();