Call for Nano

自动发送Nanoなの☆エボリューション应援歌词

Dovrai installare un'estensione come Tampermonkey, Greasemonkey o Violentmonkey per installare questo script.

You will need to install an extension such as Tampermonkey to install this script.

Dovrai installare un'estensione come Tampermonkey o Violentmonkey per installare questo script.

Dovrai installare un'estensione come Tampermonkey o Userscripts per installare questo script.

Dovrai installare un'estensione come ad esempio Tampermonkey per installare questo script.

Dovrai installare un gestore di script utente per installare questo script.

(Ho già un gestore di script utente, lasciamelo installare!)

Dovrai installare un'estensione come ad esempio Stylus per installare questo stile.

Dovrai installare un'estensione come ad esempio Stylus per installare questo stile.

Dovrai installare un'estensione come ad esempio Stylus per installare questo stile.

Dovrai installare un'estensione per la gestione degli stili utente per installare questo stile.

Dovrai installare un'estensione per la gestione degli stili utente per installare questo stile.

Dovrai installare un'estensione per la gestione degli stili utente per installare questo stile.

(Ho già un gestore di stile utente, lasciamelo installare!)

// ==UserScript==
// @name         Call for Nano
// @namespace    http://tampermonkey.net/
// @version      0.1.2
// @description  自动发送Nanoなの☆エボリューション应援歌词
// @author       ADDD
// @include      /https?:\/\/live\.bilibili\.com\/?\??.*/
// @include      /https?:\/\/live\.bilibili\.com\/\d+\??.*/
// @include      /https?:\/\/live\.bilibili\.com\/(blanc\/)?\d+\??.*/
// @require      https://code.jquery.com/jquery-3.6.0.min.js
// @require      https://cdn.bootcss.com/jqueryui/1.12.1/jquery-ui.min.js
// @require      https://cdn.staticfile.org/axios/0.27.2/axios.min.js
// @grant        none
// @license      MIT
// @icon         https://i0.hdslb.com/bfs/garb/d926ea632254c7dff67f7cbf59a0a9eaaf74bb1b.png
// ==/UserScript==

(function() {
    // 歌曲来源 https://shiinanoha.com/archives/10936 - 菜の花字幕组
    const AUDIO_SRC = "https://shiinanoha.com/wp-content/uploads/2022/07/Nano%E3%81%AA%E3%81%AE%E2%98%86%E3%82%A8%E3%83%9C%E3%83%AA%E3%83%A5%E3%83%BC%E3%82%B7%E3%83%A7%E3%83%B3.mp3";

    // 打Call图片
    const IMAGE_SRC = "";

    // Nano用户Id
    const NANO_UID = 623441612;

    // Nano直播间Id
    const NANO_ROOM_ID = 22347054;

    // 弹幕间隔时间
    const INTERVAL = 1000;

    // 原曲歌词
    const ORIGIN_LYRICS = [
        { time:"00:00.00", content:""},
        { time:"00:02.32", content:"Nanoなの☆はい!"},
        { time:"00:14.54", content:"ぐるぐる迷路 地図片手に"},
        { time:"00:20.67", content:"指差し確認して 準備OK"},
        { time:"00:26.25", content:"朝まで解いた宿題持って"},
        { time:"00:30.76", content:"今日もいってきます(いってきまーす)"},
        { time:"00:36.69", content:"天気予報土砂降り雨でも"},
        { time:"00:42.31", content:"傘なんていらない!いっせーのでJump!"},
        { time:"00:47.52", content:"背伸びしても届かないなら"},
        { time:"00:53.47", content:"お空目指して 羽ばたこう"},
        { time:"00:59.02", content:"謎々だらけ この地球を"},
        { time:"01:04.61", content:"ぐるっと一回り"},
        { time:"01:10.15", content:"小っちゃくても大きいハートで"},
        { time:"01:12.70", content:"あなたの一番になりたいから"},
        { time:"01:15.57", content:"私のプログレス ずっと見ていて"},
        { time:"01:18.96", content:"やっぱり君なの Nanoなの☆yeah!"},
        { time:"01:32.64", content:"大きめブラシふんわりチーク"},
        { time:"01:38.83", content:"赤いマニキュア フリルのスカート"},
        { time:"01:44.36", content:"鏡の私とウインク練習"},
        { time:"01:48.92", content:"上手にできるかな?"},
        { time:"01:54.84", content:"あれもこれも 過ぎてく時間に"},
        { time:"02:00.47", content:"待って待って追いかけてjump!"},
        { time:"02:05.64", content:"あなたの瞳に小さくても"},
        { time:"02:11.59", content:"私可愛く映っていますか?"},
        { time:"02:17.16", content:"昨日の涙拭ったら"},
        { time:"02:22.75", content:"明日を迎えに行くの"},
        { time:"02:28.32", content:"小っちゃくても大きいハートで"},
        { time:"02:30.92", content:"あなたの一番でいられるなら"},
        { time:"02:33.75", content:"楽しいことは10億倍 やっぱり君なの"},
        { time:"03:01.46", content:"こんなに小さな声でも"},
        { time:"03:07.43", content:"見つけてくれてありがとう"},
        { time:"03:12.94", content:"これからもずっとずっと"},
        { time:"03:18.54", content:"そばにいてくれますか?"},
        { time:"03:26.54", content:"背伸びしても届かないなら"},
        { time:"03:32.52", content:"お空目指して 羽ばたこう"},
        { time:"03:38.11", content:"謎々だらけ この地球を"},
        { time:"03:43.71", content:"ぐるっと一回り"},
        { time:"03:49.27", content:"小っちゃくても大きいハートで"},
        { time:"03:51.94", content:"あなたの一番になりたいから"},
        { time:"03:54.66", content:"私のプログレス ずっと見ていて"},
        { time:"03:58.04", content:"やっぱり君なの Nanoなの☆yeah!"},
        { time:"04:12.14", content:"Nanoなの☆"},
    ];

    // 打call歌词
    const CHEER_LYRICS = [
        { mode: [], time: "00:00.00", content: "" },
        { mode: [1], time: "00:03.18", content: "嗨!" },
        { mode: [2], time: "00:03.58", content: "wu oi!wu oi!wu oi!wu oi!" }, // ウーオイ!ウーオイ!ウーオイ!ウーオイ!  // 过长
        { mode: [3], time: "00:06.36", content: "a~👏👏sha-ikuzo!" }, // あーーよっしゃいくぞー!
        { mode: [1, 2], time: "00:09.49", content: "tiger fire cyber fiber" }, // タイガー!ファイヤー!サイバー!ファイバー! // 过长
        { mode: [3], time: "00:12.25", content: "diver viber jia-jia-!" }, // ダイバー!バイバー!ジャージャー! // 过长
        { mode: [1, 2], time: "00:19.27", content: "nanoha-!" },
        { mode: [3], time: "00:23.13", content: "cho-zetu kawaii nanoha-!" }, // 问就是 「超絶可愛い なのはー!」 有b站屏蔽词 // 过长
        { mode: [1, 3], time: "00:30.41", content: "nanoha-!" },
        { mode: [2], time: "00:34.26", content: "cho-zetu kawaii nanoha-!" }, // 问就是 「超絶可愛い なのはー!」 有b站屏蔽词 // 过长
        { mode: [1, 3], time: "00:37.05", content: "na-noha! na-noha!" },
        { mode: [2], time: "00:39.85", content: "na-noha! na-noha!" }, // 频率过快
        { mode: [2, 3], time: "00:45.76", content: "yeah tiger faibo wiper" }, // タイガー!ファイボー!ワイパー! // 过长
        { mode: [1], time: "00:49.48", content: "Ah-fufu-!" },
        { mode: [2], time: "00:52.43", content: "👏👏 fuwafuwa!" },
        { mode: [1], time: "00:55.58", content: "嗨 se-no!" },
        { mode: [3], time: "00:56.57", content: "嗨~嗨!嗨嗨嗨嗨" },
        { mode: [2], time: "01:00.64", content: "Ah-fufu-!" },
        { mode: [1, 3], time: "01:03.46", content: "👏👏 fuwafuwa!" },
        { mode: [1, 2, 3], time: "01:10.56", content: "喔~嗨!喔~嗨!喔~嗨!喔~嗨!" },
        { mode: [1], time: "01:21.33", content: "嗨!" },
        { mode: [2], time: "01:21.75", content: "wu oi!wu oi!wu oi!wu oi!" }, // ウーオイ!ウーオイ!ウーオイ!ウーオイ! // 过长
        { mode: [3], time: "01:24.49", content: "a~👏👏sha-ikuzo!" },
        { mode: [2, 3], time: "01:27.64", content: "tora hi jinzou seni" }, // 问就是 「(se)ni 」是b站屏蔽词 // 虎(とら)、火(ひ)、人造(じんぞう)、繊维(せんい)
        { mode: [1], time: "01:30.42", content: "ama shindou kasse-n!" }, // 海女(あま)、振动(しんどう)、化繊飞除去(かせんとびじょきょ)
        { mode: [2, 3], time: "01:37.39", content: "nanoha-!" }, // なのはー!
        { mode: [1], time: "01:41.25", content: "cho-zetu kawaii nanoha-!" }, // 问就是 「超絶可愛い なのはー!」 有b站屏蔽词 // 过长
        { mode: [1, 3], time: "01:48.58", content: "nanoha-!" }, // なのはー!
        { mode: [2], time: "01:52.41", content: "cho-zetu kawaii nanoha-!" }, // 问就是 「超絶可愛い なのはー!」 有b站屏蔽词 // 过长
        { mode: [1, 2], time: "02:03.92", content: "yeah tiger faibo wiper" }, // 过长
        { mode: [3], time: "02:07.61", content: "Ah-fufu-!" },
        { mode: [1, 2], time: "02:10.56", content: "👏👏 fuwafuwa!" },
        { mode: [3], time: "02:13.71", content: "嗨 se-no!" },
        { mode: [1, 2], time: "02:14.75", content: "嗨~嗨!嗨嗨嗨嗨" },
        { mode: [3], time: "02:18.76", content: "Ah-fufu-!" },
        { mode: [1, 2], time: "02:21.74", content: "👏👏 fuwafuwa!" },
        { mode: [1, 2, 3], time: "02:28.71", content: "喔~嗨!喔~嗨!喔~嗨!喔~嗨!" },
        { mode: [2, 3], time: "02:39.87", content: "iitai kotoga arundayo" }, // 言いたいことがあるんだよ! // 过长
        { mode: [1], time: "02:42.60", content: "yappari nanohawa kawaiiyo" }, // やっぱりなのははかわいいよ! // 过长
        { mode: [3], time: "02:45.46", content: "suki suki daisuki yappa suki" }, // すきすき大好き!やっぱ好き! // 过长
        { mode: [1], time: "02:48.22", content: "yatto mituketa ohimesama" }, // 问就是 「やっと見つけたお姫様!」 有b站屏蔽词 // 过长
        { mode: [2], time: "02:50.96", content: "orega umarete kitariyuu" }, // 俺が生まれてきた理由! // 过长
        { mode: [3], time: "02:53.80", content: "sorewa nanohani deautame" }, // 问就是 「それはなのはに出会うため!」 有b站屏蔽词 // 过长
        { mode: [1, 2], time: "02:56.56", content: "oreto Isshoni jinsei ayumou" }, // 问就是 jin(se)i 是b站屏蔽词 俺と一緒に人生歩もう // 过长
        { mode: [3], time: "02:59.32", content: "sekaide itiban aishiteru-!" }, // 世界で一番愛してる! // 过长
        { mode: [2, 3], time: "03:24.53", content: "👏👏👏👏" },
        { mode: [1], time: "03:27.31", content: "👏~👏~👏~👏~" },
        { mode: [3], time: "03:34.57", content: "嗨 se-no!" },
        { mode: [1], time: "03:35.63", content: "嗨~嗨!嗨嗨嗨嗨" },
        { mode: [2], time: "03:39.69", content: "Ah-fufu-!" },
        { mode: [1, 3], time: "03:42.72", content: "👏👏 fuwafuwa!" },
        { mode: [1, 3], time: "03:49.64", content: "喔~嗨!喔~嗨!喔~嗨!喔~嗨!" },
        { mode: [2], time: "03:50.64", content: "👏👏👏👏*4" },
        { mode: [3], time: "04:00.42", content: "嗨!" },
        { mode: [1], time: "04:00.80", content: "wu oi!wu oi!wu oi!wu oi!" }, // ウーオイ!ウーオイ!ウーオイ!ウーオイ! // 过长
        { mode: [2], time: "04:03.60", content: "wu oi!wu oi!wu oi!wu oi!" }, // ウーオイ!ウーオイ!ウーオイ!ウーオイ! // 过长
        { mode: [1, 3], time: "04:06.36", content: "wu oi!wu oi!wu oi!wu oi!" }, // ウーオイ!ウーオイ!ウーオイ!ウーオイ! // 过长
        { mode: [1, 2, 3], time: "04:13.42", content: "foo-!" },
    ];

    // 布局设置
    const setup = () => {
        $(".player-section").append(
            `<style>
      #call-container {
        position: absolute;
        left: 10%;
        top: 10%;
        color: white;
        font-size: 1.2rem;
        font-family: "微软雅黑";
      }

      #call-img-container {
        position: absolute;
      }

      #action-container {
        position: absolute;
        width: 440px;
        background-color: #333;
        margin: auto;
        opacity: 0.9;
      }

      #lyric-content {
        width: 440px;
        height: 480px;
        overflow: hidden;
        position: relative;
        opacity: 0.9;
      }

      #action-bar {
        display: flex;
        flex-direction: row;
        margin: 14px;
        align-items: center;
      }

      #button-group {
        display: flex;
        height: 54px;
        flex-direction: column;
        align-items: flex-start;
        justify-content: space-around;
      }

      #button-group button {
        display: flex;
        color: black;
        font-size: 0.8rem;
      }

      #audio-container {
        display: flex;
      }

      #audio-container audio {
        display: flex;
        height: 30px;
      }

      #call-img {
        width: 50px;
        height: 50px;
        border-radius: 10px;
      }

      #lyric-content ul {
        width: 100%;
        position: absolute;
        top: 0;
        left: 0;
        list-style: none;
      }

      .original {
        height: 30px;
        line-height: 30px;
        text-align: left;
        padding-left: 30px;
      }

      .original.active {
        color: #2ecc71;
        font-weight: bold;
        font-size: 20px;
      }

      .cheerful {
        height: 30px;
        line-height: 30px;
        text-align: right;
        padding-right: 30px;
      }

      .cheerful.active {
        color: #f35858;
        font-weight: bold;
        font-size: 20px;
      }
      </style>

      <div id="call-container">
        <div id="call-img-container">
          <img id="call-img" />
        </div>
        <div id="action-container">
          <div id="lyric-content"></div>
          <div id="action-bar">
            <div id="button-group">
              <button id="mode">模式1</button>
              <button id="call">点我打Call</button>
            </div>
            <div id="audio-container">
              <audio controls></audio>
            </div>
          </div>
        </div>
      </div>`);
    };

    // 初始化插件
    const initCheer = () => {
        const $ul = $("<ul></ul>");
        const parsedOriginLyrics = [];
        const parsedCheerLyrics = [];
        let isCalling = false;
        let mode = 0;
        const audio = $("#audio-container audio")[0];

        // 初始化音频
        const initAudio = (audioSrc) => {
            audio.src = audioSrc;
            audio.muted = true;

            let lastCheerLineNo = 0;
            let timer = null;
            // 当快进或者倒退时 找到该时点所属行
            const getLineNo = (currentTime, lyrics) => {
                const length = lyrics.length - 1;
                for (let i = 0; i < length; ++i) {
                    if (
                        currentTime >= parseFloat(lyrics[i].time) &&
                        currentTime < parseFloat(lyrics[i + 1].time)
                    ) {
                        return i;
                    }
                }
                return length;
            };
            // 节流
            const throttle = (func) => {
                timer = setTimeout(() => {
                    func;
                    timer = null;
                }, INTERVAL);
            };

            // 歌曲播放时渲染
            audio.addEventListener("timeupdate", () => {
                const MIN_SCROLL_LINE = 6; // 第6行起开始滚动歌词
                const LINE_HEIGHT = -30; // 每次滚动的距离

                if ($("li").eq(0).hasClass("active")) {
                    $("ul").css("top", "0");
                }
                // 获取原曲该时点播放行
                const originLineNo = getLineNo(audio.currentTime, parsedOriginLyrics);
                // 获取打call时该时点播放行
                const cheerLineNo = getLineNo(audio.currentTime, parsedCheerLyrics);
                // 输出模式判断
                // if (isCalling && timer === null && cheerLineNo !== lastCheerLineNo && parsedCheerLyrics[cheerLineNo].mode.includes(mode + 1)) {
                if (isCalling && timer === null && cheerLineNo !== lastCheerLineNo) {
                    // 播放其他行歌词
                    lastCheerLineNo = cheerLineNo;
                    // 发送弹幕
                    throttle(sendMessage(parsedCheerLyrics[cheerLineNo].content));
                }
                // 歌词高亮
                $("li.original")
                    .eq(originLineNo)
                    .addClass("active")
                    .siblings(".original")
                    .removeClass("active");
                $("li.cheerful")
                    .eq(cheerLineNo)
                    .addClass("active")
                    .siblings(".cheerful")
                    .removeClass("active");
                // 滚动播放
                if (originLineNo > MIN_SCROLL_LINE || cheerLineNo > MIN_SCROLL_LINE) {
                    $ul
                        .stop(true, true)
                        .animate({ top: (originLineNo + cheerLineNo - MIN_SCROLL_LINE) * LINE_HEIGHT });
                }
            });
        };

        // 初始化歌词内容
        const initLyricContent = (originalLyrics, cheerLyrics) => {
            const lyricContent = $("#lyric-content");

            // 时间处理
            const parseTimeFromLyric = (lyric) => {
                const splittedTime = lyric.time.split(":");
                const minute = splittedTime[0];
                const second = splittedTime[1];
                return (parseInt(minute) * 60 + parseFloat(second)).toFixed(4) - 0;
            };

            // 文本处理
            const parseContentFromLyric = (lyric) => {
                return lyric.content;
            };

            originalLyrics.forEach((lyric) => {
                parsedOriginLyrics.push({
                    time: parseTimeFromLyric(lyric),
                    content: parseContentFromLyric(lyric),
                });
            });

            cheerLyrics.forEach((lyric) => {
                parsedCheerLyrics.push({
                    mode: lyric.mode,
                    time: parseTimeFromLyric(lyric),
                    content: parseContentFromLyric(lyric),
                });
            });

            const originLength = parsedOriginLyrics.length;
            const cheerLength = parsedCheerLyrics.length;
            let i = 0, j = 0;
            while (i < originLength && j < cheerLength) {
                const $li = $("<li></li>");
                // 根据时间设置歌词
                if (parsedOriginLyrics[i].time <= parsedCheerLyrics[j].time) {
                    // 设置原曲歌词
                    $li.text(parsedOriginLyrics[i++].content).addClass("original");
                } else {
                    // 设置打call歌词
                    $li.text(parsedCheerLyrics[j++].content).addClass("cheerful");
                }
                $ul.append($li);
            }
            // 追加结尾处歌词
            while (i < originLength) {
                const $li = $("<li></li>");
                $li.text(parsedOriginLyrics[i++].content).addClass("original");
                $ul.append($li);
            }
            while (j < cheerLength) {
                const $li = $("<li></li>");
                $li.text(parsedCheerLyrics[j++].content).addClass("cheerful");
                $ul.append($li);
            }

            lyricContent.append($ul);
        };

        // 初始化操作入口
        const initEntrance = () => {
            const actor = $("#action-container");
            const callImg = $("#call-img");
            const modeButton = $("#mode");
            const callButton = $("#call");

            const imageSrc = IMAGE_SRC;
            callImg.attr("src", imageSrc);
            callImg.draggable();

            callImg.click(() => {
                actor.toggle(200);
            });
            callImg.hover(() => {
                callImg.css("cursor", "pointer");
            });

            actor.hide();
            actor.draggable();

            initAudio(AUDIO_SRC);
            initLyricContent(ORIGIN_LYRICS, CHEER_LYRICS);

            callButton.click(() => {
                isCalling = 1 - isCalling;
                callButton.text(isCalling ? "发送中..." : "弹幕打Call");
            });

            modeButton.click(() => {
                mode = (mode + 1) % 3;
                modeButton.text(`模式${mode + 1}`);
            });
        };

        initEntrance();
    };

    // 客户端请求
    const apiClient = axios.create({
        baseURL: "https://api.live.bilibili.com",
        withCredentials: true,
    });

    // 获取勋章数据
    let medalInfos = [];
    try {
        setTimeout(async () => {
            const res = await apiClient
            .get("/xlive/web-ucenter/user/MedalWall", {
                params: {
                    target_id: window.__NEPTUNE_IS_MY_WAIFU__.userLabInfo.data.uid
                }
            });
            medalInfos = res.data.data.list;
        }, 1000);
    } catch (e) {
        console.warn("查看是否加入粉丝团时出错", e);
    }

    // 是否加入粉丝团
    const filteredMedalInfo = medalInfos.filter((item) => {
        return NANO_UID === item.medal_info.target_id;
    });
    const isNanoFan = filteredMedalInfo.length > 0;

    // 获取房间id
    const getRoomId = () => {
        if (window.__NEPTUNE_IS_MY_WAIFU__) {
            return window.__NEPTUNE_IS_MY_WAIFU__.roomInfoRes.data.room_info.room_id;
        } else {
            const url = document.URL;
            const re = /\/\d+/.exec(url);
            return re[0].substr(1);
        }
    };

    const pattern = /(room|official)(_\d+){1,2}/;
    const data = new FormData();
    const roomId = getRoomId();
    // 获取CsrfToken
    const jct = document.cookie.match(/\bbili_jct=(.+?)(?:;|$)/)[1];

    data.set("bubble", "0");
    data.set("color", "16777215");
    data.set("mode", "1");
    data.set("fontsize", "25");
    data.set("rnd", parseInt(Date.now() / 1000));
    data.set("roomid", getRoomId());
    data.set("csrf", jct);
    data.set("csrf_token", jct);

    // 发送弹幕
    const sendMessage = (message) => {
        if (data.has("dm_type")) {
            data.delete("dm_type");
        }
        data.set("msg", message);
        if (message.includes("👏")) {
            if (roomId === NANO_ROOM_ID && isNanoFan) {
                // 如果在nano直播间且已加入粉丝团则发送表情包
                data.set("dm_type", "1");
                data.set("msg", "room_22347054_1816")
            } else {
                // 否则替换掉
                data.set("msg", message.replaceAll("👏", ""));
            }
        }
        apiClient
            .post("/msg/send", data)
            .then((res) => {
            if (res.data.code === 0) {
                switch (res.data.msg) {
                    case "":
                        console.log("发送成功 - " + message);
                        break;
                    case "f":
                        console.warn("发送失败 - 包含B站屏蔽词: " + message);
                        break;
                    case "k":
                        console.warn("发送失败 - 包含直播间屏蔽词: " + message);
                        break;
                    case "same restriction":
                        console.warn("发送失败 该弹幕已被限制 请选择其它弹幕");
                        break;
                    case "max limit exceeded":
                        console.warn("发送失败 弹幕池达到上限");
                        break;
                    default:
                        console.warn("发送失败 - " + res.data.message);
                        console.warn(res)
                        console.warn(res.data)
                }
            } else {
                console.warn("发送失败 - " + res.data.message);
            }
        })
            .catch(() => {
            console.warn("发送失败 - " + message);
        });
    };

    setTimeout(() => {
        setup();
        initCheer();
    }, 2000);
})();