// ==UserScript== // @name b站直播聊天室弹幕发送增强 // @namespace http://tampermonkey.net/ // @version 0.3.3 // @description 原理是分开发送。接管了发送框,会提示屏蔽词 // @author Pronax // @include /https:\/\/live\.bilibili\.com\/(blanc\/)?\d+/ // @icon http://bilibili.com/favicon.ico // @grant GM_addStyle // @run-at document-end // @require https://lf26-cdn-tos.bytecdntp.com/cdn/expire-1-M/jquery/1.12.4/jquery.min.js // @require https://greasyfork.org/scripts/439903-blive-room-info-api/code/blive_room_info_api.js?version=1037039 // @downloadURL none // ==/UserScript== // 待办: // 分段指针 // 接管全屏输入栏 // 自动判断屏蔽词 23237777 ; (async function () { 'use strict'; if (!document.cookie.match(/bili_jct=(\w*); /)) { return; } let jct = document.cookie.match(/bili_jct=(\w*); /)[1]; let roomId = await ROOM_INFO_API.getRid(); let toastCount = 0; let isProcessing = false; let formData = new FormData(); let timeout = undefined; formData.set("bubble", 0); formData.set("color", 16777215); formData.set("mode", 1); formData.set("fontsize", 25); formData.set("roomid", roomId); const LIMIT = await ROOM_INFO_API.getDanmuLength(roomId); const riverCrabs = { "不能": "f", "上船就送": "f", "上舰就送": "f", "上舰送": "fire", "上船送": "fire", "神仙水": "f", "小赤佬": "f", "速器": "fire", "商丘": "fire", "谨慎": "fire", "慎判": "fire", "代练": "f", "违规直播": "f", "低俗": "f", "系统": "f", "渣女": "f", "肥": "fire", "墙了": "f", "变质": "f", "小熊": "f", "疫情": "f", "感染": "f", "分钟": "f", "爽死": "f", "黑历史": "f", "超度": "f", "渣男": "f", "和谐": "f", "河蟹": "f", "敏感": "f", "你妈": "f", "代孕": "f", "硬了": "f", "抖音": "f", "保卫": "f", "被gan": "f", "寄吧": "f", "郭楠": "f", "里番": "f", "小幸运": "f", "试看": "f", "加QQ": "f", "警察": "f", "营养": "f", "资料": "f", "家宝": "f", "饿死": "f", "不认字": "f", "横幅": "f", "hentai": "f", "诱惑": "f", "垃圾": "f", "福报": "f", "拉屎": "f", "顶不住": "f", "一口气": "f", "苏联": "f", "哪个平": "f", "老鼠台": "f", "顶得住": "f", "gay": "f", "黑幕": "f", "蜀黍我啊": "f", "梯子": "f", "美国": "f", "米国": "f", "未成年": "f", "爪巴": "f", "包子": "fire", "党": "fire", "89": "fire", "戏精": "fire", "八九": "fire", "八十九": "fire", "你画我猜": "fire", "叔叔我啊": "fire", "爬": "fire" }; let wordTree = {}; initTree(); // 组件CSS GM_addStyle("#medal-selector{height:20px;}.medal-section{display:inline-block;position:relative;top:2px;left:5px;}.dialog-ctnr>.arrow{display:none}.chat-input-ctnr>div:first-of-type{width:100%}.chat-input-ctnr .input-limit-hint{z-index:10;bottom:0!important;right:53px!important}#chat-control-panel-vm{height:102px}.chat-history-panel{height:calc(100% - 128px - 152px)!important}#liveDanmuSendBtn{height:100%;min-width:50px;padding-top:5px;border-radius:0 3px 3px 0}.link-toast.error{left:40px;right:40px;white-space:normal;margin:auto;text-align:center;box-shadow:0 .2em .1em .1em rgb(255 100 100 / 20%)}#liveDanmuInputArea{background-color:transparent;position:relative;z-index:1;padding:4px 8px;line-height:18px;word-break:break-all;overflow:auto;scrollbar-width:thin}#liveDanmuInputArea::-webkit-scrollbar,.chat-input-cover::-webkit-scrollbar{width:6px}#liveDanmuInputArea::-webkit-scrollbar-thumb,.chat-input-cover::-webkit-scrollbar-thumb{background-color:#aaa}.control-panel-icon-row>.icon-right-part{margin-right:6px}.chat-input-ctnr{margin-top:0 !important}.control-panel-icon-row .medal-section .medal-item-margin{margin:2px}"); // combo-card适配CSS GM_addStyle(".chat-history-panel{padding-bottom:0!important}#combo-card{background:linear-gradient(to right,rgb(255 198 217) 20%,rgba(255,102,153,0) 100%)!important;bottom:108px!important;z-index:20;}"); // 新版小表情CSS // GM_addStyle(".danmaku-preference-ctnr .img-pane .emotion-wrap.emoji-wrap .emoticon-item>img{pointer-events:none;}"); GM_addStyle(".emotion-recent-wrap{display:none !important;}"); // 输入框周围组件默认css GM_addStyle(".medal-section .action-item.medal.get-medal,.medal-section .action-item.medal.wear-medal{width:49px !important;height:20px !important;background-image:url()}.medal-section .action-item.medal{background-size:cover;border:0}.medal-section .action-item{display:inline-block;margin:0 2px;font-size:12px;color:#fff;line-height:14px;text-align:center;border-radius:2px;cursor:pointer;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}"); // 输入框边框 GM_addStyle(".chat-input-ctnr{transition:border-color .2s linear;}.chat-input-ctnr:focus-within:hover{border-color: var(--brand_blue);}"); // 输入框css GM_addStyle("div.chat-input.border-box{position:absolute;padding:4px 8px;line-height:18px;word-break:break-all;overflow:auto;padding-right:58px;scrollbar-width:thin;color:transparent}div.chat-input .f-word{background-color:var(--Ly4)}div.chat-input .fire-word{background-color:#ff4500}"); let itv = setInterval(() => { let text = $("textarea.chat-input"); if (text.length) { clearInterval(itv); $(".control-panel-icon-row").prepend($(".medal-section")); // 背景框 let displayDiv = document.createElement("div"); displayDiv.className = "chat-input border-box chat-input-cover"; for (let key in text[0].dataset) { displayDiv.dataset[key] = ""; } text.before(displayDiv); // 输入框 let tempText = text.clone().attr("id", "liveDanmuInputArea"); text.after(tempText).remove(); // 发送按钮 let sendBtn = $(".bottom-actions>.right-action").removeClass("p-absolute"); $(sendBtn).find("button").attr("id", "liveDanmuSendBtn"); $(".chat-input-ctnr").append(sendBtn.clone()); sendBtn.remove(); // 适配新版小表情 let emojiBtn = document.querySelector(".emoticons-panel"); emojiBtn.addEventListener("click", () => { let deadline = Date.now() + 1000; (function init() { // todo 最近使用表情支持 let emojiPanel = document.querySelector(".emoji-wrap"); if (emojiPanel) { emojiPanel.onclick = e => { let target = e.target; if (e.target.tagName == "IMG") { target = e.target.parentNode; } let inputArea = document.querySelector("#liveDanmuInputArea"); inputArea.value = inputArea.value + target.title; inputArea.oninput(); } } else if (Date.now() < deadline) { requestIdleCallback(init, { timeout: 1000 }); } })(); }); $("#liveDanmuSendBtn").click(async () => { let msg = $("#liveDanmuInputArea").val(); if ((!msg) || isProcessing) { if (isProcessing) { toast("有弹幕正在发送中", 1500, "info") } return; } isProcessing = true; let page = 1; let segment = LIMIT; if (msg.length > segment) { // 自动平均每条弹幕的长度 while (msg.length / segment % 1 < 0.7 && msg.length / segment % 1 != 0) { segment--; } page = Math.ceil(msg.length / segment); console.log(`长度:${msg.length} 间隔:${segment} 分页:${page}`); } let count = 0; do { let str = msg.substr(0, segment); let result = await sendMsg(str, count++ ? 500 + Math.random() * 1000 >> 1 : 0); msg = msg.substr(segment); $("#liveDanmuInputArea").val(msg); document.querySelector("#liveDanmuInputArea").oninput(); } while (msg.length > 0); isProcessing = false; }); document.querySelector('#liveDanmuInputArea').oninput = function () { clearTimeout(timeout); timeout = setTimeout(() => { document.querySelector(".chat-input-cover").innerHTML = filter(this.value); }, 200); displayDiv.scrollTop = this.scrollTop; let length = this.value.length; $(".input-limit-hint").text(length); if (length > LIMIT) { $(".input-limit-hint").css("color", "#23ade5"); } else { $(".input-limit-hint").css("color", ""); } }; document.querySelector('#liveDanmuInputArea').onkeydown = function (e) { if ((!e.shiftKey) && e.keyCode == 13) { e.returnValue = false; $("#liveDanmuSendBtn").click(); } }; document.querySelector('#liveDanmuInputArea').onscroll = function () { // 同步滚动 displayDiv.scrollTop = this.scrollTop; } } }, 100); function filter(str) { let result = testStr(str); // console.log(result); for (let word of result) { str = str.replaceAll(word.w, `${word.w}`); } // 替换换行 str = str.replaceAll("\n", "
"); return str; } async function sendMsg(msg, timer = 500) { return new Promise((resolve, reject) => { setTimeout(() => { jct = document.cookie.match(/bili_jct=(\w*); /)[1]; formData.set("csrf", jct); formData.set("csrf_token", jct); formData.set("msg", msg); formData.set("rnd", Math.floor(new Date() / 1000)); fetch("//api.live.bilibili.com/msg/send", { credentials: 'include', method: 'POST', body: formData }) .then(response => response.json()) .then(result => { if (result.code != 0 || result.msg != "") { switch (result.msg) { case "f": result.msg = "弹幕含有敏感词"; break; case "fire": result.msg = "弹幕含有系统屏蔽词"; break; case "k": result.msg = "内容含有房间屏蔽词"; break; default: result.msg = result.message; } if (result.code == -111) { jct = document.cookie.match(/bili_jct=(\w*); /)[1]; formData.set("csrf", jct); formData.set("csrf_token", jct); } toast(result.msg); isProcessing = false; reject(result); } else { resolve(true); } }) .catch(err => { console.log("发送弹幕出错:", err); toast(err.msg || err.message); isProcessing = false; reject(err); }); }, timer); }); } function testStr(str) { let result = []; for (let index = 0; index < str.length; index++) { let r = check(wordTree, str, index); if (r.i >= 0) { result.push({ s: index, e: r.i, w: r.w }); index = r.i - 1; // 减一用于抵消++ } } return result; function check(obj, str, index, word = "") { let letter = str[index]; if (str.length > index && obj[letter]) { word += letter; if (obj[letter].end) { return { i: index + 1, w: word }; } return check(obj[letter], str, index + 1, word); } else { return { i: -1, w: word }; } } } function initTree() { for (const item of Object.keys(riverCrabs)) { init(wordTree, item, 0); } function init(obj, str, index) { if (!obj[str[index]]) { obj[str[index]] = {}; } if (str.length - 1 == index) { obj[str[index]].end = true; } else { Reflect.deleteProperty(obj[str[index]], "end"); obj[str[index]] = init(obj[str[index]], str, index + 1); } return obj; } } function toast(msg, time = 2000, type = "error") { let id = Math.random() * 1000 >> 1; $(".chat-control-panel").append(``); setTimeout(() => { toastCount--; $(`.link-toast-${id}`).remove(); }, time); } })();