您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
b站直播间内使用悬浮列表快捷输入弹幕,保存一条弹幕历史,弹幕内容添加拼音发送
// ==UserScript== // @name B直播快捷弹幕 // @namespace http://tampermonkey.net/ // @version 1.3 // @description b站直播间内使用悬浮列表快捷输入弹幕,保存一条弹幕历史,弹幕内容添加拼音发送 // @author RecursiveMaple // @match https://live.bilibili.com/* // @icon  // @require https://cdn.staticfile.org/jquery/3.6.3/jquery.min.js // @require https://cdn.staticfile.org/jquery-cookie/1.4.1/jquery.cookie.min.js // @require https://unpkg.com/[email protected]/dist/index.js // @license GNU General Public License v3.0 or later // @grant GM_setValue // @grant GM_getValue // @grant GM_addStyle // @grant unsafeWindow // ==/UserScript== var htmlText = ` <div class="blds-main-window"> <li class="blds-shotcuts-list"></li> <hr> <li class="blds-util-list"> <ul> <span>🕑</span> <input type="button" id="blds-input-history" value=""> </ul> <ul> <span>➕</span> <input type="text" id="blds-input-add" maxlength="20" placeholder="添加快捷弹幕(限长20)"> <button id="blds-btn-add">+</button> </ul> <ul> <span>py</span> <input type="text" id="blds-input-py" maxlength="20" placeholder="词语添加拼音(限长20)例:'雫露露1'"> <button id="blds-btn-py">↵</button> </ul> </li> </div> `; var cssText = ` .blds-main-window { all: initial; z-index: 100; display: none; position: absolute; bottom: 100%; right: 0%; width: 100%; border: 2px solid black; background-color: lightgray; font-size: 14px; counter-reset: shortcut-list; } .blds-main-window li { display: block; width: 100%; border: none; width: 100%; } .blds-main-window hr { border: none; border-top: 3px double black; margin: 1px; } .blds-main-window ul { display: flex; border-bottom: 1px solid black; padding: 2px; } .blds-main-window span { width: 23px; height: 23px; text-align: center; font-weight: bold; border-right: 1px dashed black; } .blds-shotcuts-list { max-height: 25em; overflow-y: scroll; overscroll-behavior: contain; scrollbar-width: thin; } .blds-shotcuts-list span::before { counter-increment: shortcut-list; content: counter(shortcut-list) "."; } .blds-main-window input { flex: auto; margin-left: 3px; text-align: left; font-size: 10px; } .blds-main-window button { margin: 0px 3px; width: 23px; height: 23px; padding: 0px; } `; /*---------- 快捷弹幕数组相关操作 ----------*/ // templateVarIndex由css自动填写 // templateString不能动态更新,这里用函数填写模板 function getListItem(value) { return ` <ul> <span></span> <input type="button" value=${value}> <button class="blds-btn-del">−</button> </ul> `; } var shortcutList = []; function initList() { $(".blds-shotcuts-list").empty(); for (var i in shortcutList) { // js的for in是以字符串形式的数字作索引!?你妈的为什么。。。 $(".blds-shotcuts-list").append(getListItem(shortcutList[i])); } } function addListItem(value) { shortcutList.push(value); saveList(); $(".blds-shotcuts-list").append(getListItem(value)); } function delListItem($ul) { // 接受jquery对象,找到下标删除 var index = $ul.index(); shortcutList.splice(index, 1); saveList(); $ul.remove(); } function loadList() { shortcutList = GM_getValue("shortcutList", []); } function saveList() { GM_setValue("shortcutList", shortcutList); } /*---------- vvvvvvvvvv ----------*/ /*---------- 拼音转换相关操作 ----------*/ // 引入拼音转换模块 var { pinyin } = pinyinPro; // 正则格式:中文词语后加数字串(可选) var regex = /(^[\u4E00-\u9FA5]+)([0-9]*$)/; function addPinyin(str) { // 读入中文词语,判字符串形式 var resStr = ""; if (!regex.test(str)) return resStr; var regObj = str.match(regex); var words = regObj[1]; var numbers = regObj[2]; var wordsList = words.split(''); var numbersList = numbers.split('').map(Number); // var resList = pinyin(words, { type: 'array' });//这种形式会被吞弹幕 var resList = pinyin(words, { toneType: 'num', type: 'array' }); var mode = numbers == "" ? "FULL" : "PARTIAL"; for (var i = 0; i < wordsList.length; i++) { resStr += wordsList[i]; if (mode == "FULL" || numbersList.includes(i + 1)) { resStr += resList[i]; } } return resStr; } function setPinyinHint(hintStr) { if (hintStr == "") { $("#blds-input-add").attr("placeholder", "添加快捷弹幕(限长20)"); } else { $("#blds-input-add").attr("placeholder", hintStr); } } /*---------- vvvvvvvvvv ----------*/ /*---------- 脚本发送弹幕 ----------*/ var postUrl = "https://api.live.bilibili.com/msg/send"; var data = { fontsize: "25", csrf: "", csrf_token: "", roomid: "", mode: "1",//默认滚动, color: "16777215",//默认白色 bubble: "0",// TODO 弹幕背景气泡,不知道怎么获取,0是无气泡,5是舰长气泡? //以上为初始化时一次性填入,以下为每次发送弹幕时填入 rnd: "", msg: "", } var danmakuPositionMap = { "滚动": "1", "底部": "4", "顶部": "5" }; function rgbToDecimal(rgbText) { // 输入例:"rgb(88, 193, 222)" // 输出例:"5816798" var colorRegex = /([0-9]+)/g; var regObj = rgbText.match(colorRegex); var hexStr = regObj.map(function (decimalStr) { return parseInt(decimalStr).toString(16); }).join(""); var decimalStr = parseInt(hexStr, 16).toString(); return decimalStr; } function initDataBlock() { // 要保证在网页资源加载后执行! data.csrf = $.cookie('bili_jct'); data.csrf_token = data.csrf; var roomUrl = $(location).attr('href'); var roomIdRegex = /.com\/([0-9]+)/; data.roomid = roomUrl.match(roomIdRegex)[1]; $("#control-panel-ctnr-box").one("DOMNodeInserted", function (event) { var t = event.target; if (!$(t).hasClass("danmakuPreference")) return; // console.log("进入DOMNodeInserted处理");//TODO DEBUG $(t).hide(); // 等待加载动画播放完,目标元素动态插入 var elemSearchCount = 0; var timer = setInterval(function () { elemSearchCount++; if ($(t).find(".dot-wrapper.active").length > 0) { clearInterval(timer); var modeStr = $(t).find(".danmaku-position-item.active").attr("title"); data.mode = danmakuPositionMap[modeStr]; var colorStr = $(t).find(".dot-wrapper.active span").css("background-color"); data.color = rgbToDecimal(colorStr); console.log("Danmaku preference loaded. Mode=", data.mode, ",Color=", data.color);//INFO $(this).unbind("DOMNodeInserted"); $("#control-panel-ctnr-box span[title='弹幕设置']").click(); } else if (elemSearchCount >= 30) { clearInterval(timer); console.log("Loading danmaku preference element failed. [in function initDataBlock()]");//INFO $(this).unbind("DOMNodeInserted"); $("#control-panel-ctnr-box span[title='弹幕设置']").click(); } }, 100) }); $("#control-panel-ctnr-box span[title='弹幕设置']").click(); } function sendDanmaku(msg) { data.rnd = parseInt(Date.now() / 1000); data.msg = msg; // console.log(data);//DEBUG // 不用ajax发送表单,可能会被吞弹幕,改用formdata var formData = new FormData(); formData.append("bubble", data.bubble); formData.append("msg", data.msg); formData.append("color", data.color); formData.append("mode", data.mode); formData.append("fontsize", data.fontsize); formData.append("rnd", data.rnd); formData.append("roomid", data.roomid); formData.append("csrf", data.csrf); formData.append("csrf_token", data.csrf_token); // $.post()不支持加cookie $.ajax({ type: 'POST', url: postUrl, // data: data, data: formData, xhrFields: { withCredentials: true // 请求加入cookie }, contentType: false, processData: false, success: function (data) { // console.log(data);//DEBUG if (data.msg == "") { // 发送成功,保存为历史 $("#blds-input-history").val(msg); } }, }); } /*---------- vvvvvvvvvv ----------*/ /*---------- 注入窗口、初始化设置、添加按钮逻辑、添加鼠标悬浮事件等 ----------*/ var selWhereInsertHTMLBefore = "textarea.chat-input"; var selWhereGetWinWidth = ".chat-input-ctnr"; var selWhereDetectMouseHover = ".chat-input-ctnr div:last-child"; var selWhereSendDanmaku = "#control-panel-ctnr-box .bl-button"; function main() { // 弃用$(".chat-input-ctnr div:last-child").css({"position":"relative","display":"inline-block"}); // 插入html $(selWhereInsertHTMLBefore).before(htmlText); if ($(".blds-main-window").length == 0) { console.log("Inserting HTML failed. [in function main()]"); return; } // 插入css,调整窗口大小 GM_addStyle(cssText); var blds_main_window_width = $(selWhereGetWinWidth).width(); $(".blds-main-window").css("width", blds_main_window_width); // 初始化装入list项 loadList(); initList(); // 抓取弹幕设置 initDataBlock(); //按钮:删除快捷弹幕 $(".blds-shotcuts-list").on("click", "button", function (event) { //给未来元素自动绑定事件用on,先绑定到子元素数量变化的父元素 var t = event.target; delListItem($(t).parent()); }); // 按钮:添加快捷弹幕 $("#blds-btn-add").click(function () { var value = $("#blds-input-add").val(); if (value.trim() != "") addListItem(value); $("#blds-input-add").val("");// 清空input }); // 按钮:转拼音 $("#blds-btn-py").click(function () { var value = $("#blds-input-py").val(); var rev = addPinyin(value); if (rev != "") { sendDanmaku(rev); $("#blds-input-py").val("");// 清空input setPinyinHint("");// 清空预览 } }); // 检测输入内容改变,显示拼音预览 $("#blds-input-py").on("input", function (event) { var value = $("#blds-input-py").val(); var rev = addPinyin(value); setPinyinHint(rev); }); // 发送弹幕按钮,要求未来元素自动绑定 $(".blds-main-window").on("click", "input[type='button']", function (event) { var t = event.target; var msg = $(t).val(); sendDanmaku(msg); }); // 检测输入框按下回车键 $(".blds-main-window input[type='text']").keypress(function (event) { if (event.which == '13') { // console.log("按下回车");//DEBUG // next元素是按钮 $(this).next().click(); } }); // 监听B站发送弹幕按钮,将内容转存到历史记录 // 由于只能在原click事件之后绑定,value会被原事件处理函数清空 // 也要考虑用回车发弹幕的情况,决定用input事件监视输入框 // B站发送弹幕后清空输入框不会触发input var tempValue = ""; $(selWhereInsertHTMLBefore).on("input", function (event) { tempValue = event.target.value; }); $(selWhereSendDanmaku).click(function () { if (tempValue.trim() != "") { $("#blds-input-history").val(tempValue); // console.log("内容转存到历史记录:",tempValue);//TODO DEBUG } tempValue = ""; }); // B站弹幕输入框响应不了keypress,focus等事件,你妈的为什么 $(selWhereInsertHTMLBefore).keydown(function (event) { if (event.which == '13') { // console.log("按下回车");//DEBUG if (tempValue.trim() != "") { $("#blds-input-history").val(tempValue); // console.log("内容转存到历史记录:",tempValue);//TODO DEBUG } tempValue = ""; } }); // 控制显示和隐藏窗口 $(selWhereDetectMouseHover).hover( function () { // 每次进入后重置输入框 if ($(".blds-main-window input[type='text']:focus").length == 0) { $(".blds-main-window input[type='text']").val(""); } $(".blds-main-window").css("display", "block"); }, function () { // timer处理hover被输入法界面遮挡的情况 var timer = setInterval(function () { if ($(selWhereDetectMouseHover + ":hover").length > 0) { clearInterval(timer); } else if ($(".blds-main-window input[type='text']:focus").length == 0) { $(".blds-main-window").css("display", "none"); clearInterval(timer); } }, 100) } ); } /*---------- vvvvvvvvvv ----------*/ (function () { 'use strict'; // Your code here... //最外层$("#chat-control-panel-vm .chat-input-ctnr") //上层$(".chat-input-ctnr div:last-child") //本层$("textarea.chat-input") var maxRetry = 20; var elemSearchCount = 0; var timer = setInterval(function () { elemSearchCount++; if ($(selWhereInsertHTMLBefore).length > 0) { console.log("Ready after", elemSearchCount, "tries"); clearInterval(timer); main(); } else if (elemSearchCount >= maxRetry) { console.log("Searching failed after", elemSearchCount, "tries"); clearInterval(timer); } }, 500) })();