您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
自动发送Nanoなの☆エボリューション应援歌词
// ==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); })();