您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Mortal牌谱解析增强脚本 (雀魂麻将/天凤/麻雀一番街)
当前为
// ==UserScript== // @name Mortal 显示恶手率 // @description Mortal牌谱解析增强脚本 (雀魂麻将/天凤/麻雀一番街) // @version 2.2.4 // @homepage https://www.bilibili.com/read/cv26608482/ // @namespace https://viayoo.com/ // @author Miku39 // @icon  // @resource js_layui https://cdn.staticfile.org/layui/2.8.17/layui.min.js // @resource js_cryptojs https://cdn.staticfile.org/crypto-js/4.1.1/crypto-js.min.js // @resource js_immutable https://cdn.staticfile.org/immutable/4.3.4/immutable.min.js // @resource css_layui https://cdn.staticfile.org/layui/2.8.17/css/layui.min.css // @resource css_fontAwesome https://cdn.staticfile.org/font-awesome/6.4.2/css/all.min.css // @match *://mjai.ekyu.moe/* // @grant GM_info // @grant unsafeWindow // @grant GM_getResourceText // @grant GM_getResourceURL // @grant GM_setClipboard // @run-at document-start // @license BSD 3-Clause License // ==/UserScript== ((g_window)=>{ 'use strict'; // ---------------------------- 全局变量: App变量 ---------------------------- const badMoveUpperLimit = 5; //恶手率 const badMoveUpperLimitCustom = 10; //恶手率 let badChooseNum = 0; let badChooseNumCustom = 0; const pageLanguage = document.documentElement.lang; var i18nText; const localization = { zh_cn: { badMove: "恶手", badMoveRatio: "恶手率", matchRatio: "AI 一致率", metaData: "元数据", }, ja: { badMove: "Bad move", badMoveRatio: "bad moves/total", matchRatio: "AI一致率", metaData: "メタデータ", }, ko: { badMove: "Bad move", badMoveRatio: "bad moves/total", matchRatio: "matches/total", metaData: "메타데이터", }, en: { badMove: "Bad move", badMoveRatio: "bad moves/total", matchRatio: "matches/total", metaData: "Metadata", } }; if (pageLanguage == "zh-CN") { i18nText = localization.zh_cn; } else if (pageLanguage == "ja") { i18nText = localization.ja; } else if (pageLanguage == "ko") { i18nText = localization.ko; } else { i18nText = localization.en; } i18nText.modelTag = "model tag"; // i18nText.seatTypeA0 = "东起"; i18nText.seatTypeA1 = "南起"; i18nText.seatTypeA2 = "西起"; i18nText.seatTypeA3 = "北起"; // i18nText.seatTypeB0 = "东家"; i18nText.seatTypeB1 = "南家"; i18nText.seatTypeB2 = "西家"; i18nText.seatTypeB3 = "北家"; // i18nText.seatTypeC0 = "自家"; i18nText.seatTypeC1 = "下家"; i18nText.seatTypeC2 = "对家"; i18nText.seatTypeC3 = "上家"; // i18nText.Ron = "荣和"; i18nText.Tsumo = '自摸'; i18nText.Ryuukyoku = '流局'; i18nText.RyuukyokuTsumo = "流局满贯"; i18nText.RyuukyokuType1 = "九种九牌"; //四风连打、四杠散了、四家立直 //三家和了 // i18nText.badMoveError = "(恶手率统计只支持最新版本Mortal,当前版本生成结果不可靠)" // i18nText.badMoveUp = " (严重鸡打 权重<=5%)"; i18nText.badMoveDown = " (普通错误 权重5~10%)"; i18nText.badMoveNull = " "; i18nText.badMoveSymbol = "%"; i18nText.badMoveDiffer = "差值: "; // i18nText.badMoveDiffer1 = "微差(0~5): "; i18nText.badMoveDiffer2 = "小幅差距(5~10): "; i18nText.badMoveDiffer3 = "低等差距(10~20): "; i18nText.badMoveDiffer4 = "中等差距(20~40): "; i18nText.badMoveDiffer5 = "高等差距(40~60): "; i18nText.badMoveDiffer6 = "大幅度差距(60~80): "; i18nText.badMoveDiffer7 = "压倒性差距(80~100): "; // i18nText.badMoveSum = " (总计)"; const tenhouText = { Ron: '和了', //荣和 Tsumo: '和了', //自摸 Ryuukyoku: '流局', //荒牌流局 RyuukyokuTsumo: "流し満貫", //流局满贯 RyuukyokuType1: "九種九牌", //九种九牌 //四风连打、四杠散了、四家立直 //三家和了 }; const matchRule = { isInit: false, isRon3: true, //是否允许三家荣和 }; let strArray2 = []; let strArray3 = []; // ---------------------------- 全局变量: 模板 ---------------------------- const CSSCompatibilityFlag = ".ui"; const CSSCompatibilityFlagNoPoint = CSSCompatibilityFlag.substring(1); const commonCSS = '\ .position_re { position: relative; }\ .position_ab { position: absolute; }\ .position_fi { position: fixed; }\ \ .font_weight_900 { font-weight: 900; } \ .font_weight_700 { font-weight: 700; } \ .font_weight_400 { font-weight: 400; } \ \ .w-auto { width: auto; }\ .w-100 { width: 100%; }\ .w-80 { width: 80%; }\ .w-75 { width: 75%; }\ .w-60 { width: 60%; }\ .w-50 { width: 50%; }\ .w-40 { width: 40%; }\ .w-25 { width: 25%; }\ .w-20 { width: 20%; }\ .w-10 { width: 10%; }\ .w-5 { width: 5%; }\ \ .min-w-auto { min-width: auto; }\ .min-w-100 { min-width: 100%; }\ .min-w-80 { min-width: 80%; }\ .min-w-75 { min-width: 75%; }\ .min-w-60 { min-width: 60%; }\ .min-w-50 { min-width: 50%; }\ .min-w-40 { min-width: 40%; }\ .min-w-25 { min-width: 25%; }\ .min-w-20 { min-width: 20%; }\ .min-w-10 { min-width: 10%; }\ .min-w-5 { min-width: 5%; }\ \ .max-w-auto { max-width: auto; }\ .max-w-100 { max-width: 100%; }\ .max-w-80 { max-width: 80%; }\ .max-w-75 { max-width: 75%; }\ .max-w-60 { max-width: 60%; }\ .max-w-50 { max-width: 50%; }\ .max-w-40 { max-width: 40%; }\ .max-w-25 { max-width: 25%; }\ .max-w-20 { max-width: 20%; }\ .max-w-10 { max-width: 10%; }\ .max-w-5 { max-width: 5%; }\ '; const pageExtendBaseCSS = '\ details.collapseEntryL1 { border: 2px solid #f00; } \ details.collapseEntryL2 { border: 2px solid #6600FF; } \ .badChoose { font-size: 20px; } \ .level1 { color: #f00; font-weight: 900; } \ .level2 { color: #6600FF; font-weight: 700; } \ .color1 { color: #f00; } \ .color2 { color: #6600FF; } \ .color3 { color: #FF0066; } \ .color4 { color: #990000; } \ .l-130px { left: 130px; } \ .l-170px { left: 170px; } \ .color5 { color: #CC0000; } \ .color6 { color: #333; } \ '; const MainAppBaseCSS = '\ .commonDIV { padding: 5px; } \ #settingUI { width: 800px; height: 600px; border: 1px solid red; position: fixed; top: 100px; left: 100px; z-index: 10; } \ #catalogUI { border: 1px solid blue; position: fixed; z-index: 10; } \ '; const origSettingUITemplate = '\ <div class="commonDIV" id="origSettingUI">\ <fieldset id="panel">\ <legend>设置</legend>\ <span style="font-weight:700">布局:</span>\ <label><input checked="" name="layout" onclick="toggleLayout()" type="radio" value="vertical">垂直</label>\ <label><input name="layout" onclick="toggleLayout()" type="radio" value="horizontal">水平</label>\ <br><span style="font-weight:700">展开:</span><label><input name="expand" onclick="toggleExpand()" type="radio" value="all">全部</label>\ <label><input checked="" name="expand" onclick="toggleExpand()" type="radio" value="diff-only">仅差异项</label>\ <label><input name="expand" onclick="toggleExpand()" type="radio" value="none">无</label>\ <br><a class="no-visit" download="" href="">💾保存本页面</a>\ </fieldset>\ </div>\ '; const settingUIBase = '\ <div class="ui commonDIV" id="settingUI">\ </div>\ '; const settingUITemplate = '\ '; const catalogUIBase = '\ <div class="ui commonDIV" id="catalogUI">\ </div>\ '; const catalogUITemplate = '\ <div class="ui layui-btn-container" id="catalogUIBuf">\ <button type="button" class="ui layui-btn" id="kyoku_prev">\ <i class="ui layui-icon layui-icon-prev"></i>\ </button>\ <button type="button" class="ui layui-btn" id="kyoku_next">\ <i class="ui layui-icon layui-icon-next"></i>\ </button>\ <button type="button" class="ui layui-btn layui-btn-disabled" id="diff_prev">\ <i class="ui layui-icon layui-icon-left"></i>\ </button>\ <button type="button" class="ui layui-btn layui-btn-disabled" id="diff_next">\ <i class="ui layui-icon layui-icon-right"></i>\ </button>\ </div>\ <ul class="ui layui-nav layui-nav-tree layui-bg-gray" id="selectorGroups" lay-filter="selector-filter-nav" style="width: 100%; overflow: auto;">\ <!-- 对局选择器 -->\ <li class="ui layui-nav-item layui-nav-itemed">\ <a class="ui" href="javascript:;">对局选择器</a>\ <dl class="ui layui-nav-child" id="selector1">\ {{# layui.each(d.selector1, function(index, item){ }}\ <dd class="ui"><a class="ui" href="{{= item.href }}" name="{{= item.href }}">{{= item.name }}</a></dd>\ {{# }); }}\ </dl>\ </li>\ <!-- 不一致选择器 -->\ <li class="ui layui-nav-item">\ <a class="ui" href="javascript:;">不一致选择器 (等待完善)</a>\ <dl class="ui layui-nav-child" id="selector2">\ {{# layui.each(d.selector2, function(index, item){ }}\ <dd class="ui"><a class="ui" href="{{= item.href }}" name="{{= item.href }}">{{= item.name }}</a></dd>\ {{# }); }}\ </dl>\ </li>\ <!-- 恶手选择器 -->\ <li class="ui layui-nav-item">\ <a class="ui" href="javascript:;">恶手选择器 (等待完善)</a>\ <dl class="ui layui-nav-child" id="selector3">\ {{# layui.each(d.selector3, function(index, item){ }}\ <dd class="ui"><a class="ui" href="{{= item.href }}" name="{{= item.href }}">{{= item.name }}</a></dd>\ {{# }); }}\ </dl>\ </li>\ <li class="ui layui-nav-item"><a class="ui" href="javascript:;">打开设置 (等待完善)</a></li>\ </ul>\ '; // ---------------------------- 其他 ---------------------------- //交叉观察器 var ObserverEleSetList = []; const io = new IntersectionObserver((ioes) => { ioes.forEach(ioe => { const ele = ioe.target const intersectionRatio = ioe.intersectionRatio if (intersectionRatio > 0 && intersectionRatio <= 1) { //0~1范围内的值为显示状态, 等于0的值为隐藏 const section = ele.parentElement; const titleEle = section.children[0]; const titleKyokuEle = titleEle.getElementsByTagName("a"); //只有1个元素 //Console.orig.log("可见: " + titleKyokuEle[0].innerText, titleKyokuEle[0].id); let targetEleArray = document.getElementsByName(titleKyokuEle[0].id); setNewActivateEle(targetEleArray); } ele.onload = () => {} ele.onerror = () => {} }); }); function setNewActivateEle(targetEleArray) { if(targetEleArray.length > 0) { //清除旧数据 for (let j = 0; j < ObserverEleSetList.length; j++) { const element = ObserverEleSetList[j]; element.parentElement.classList.remove("layui-this"); } ObserverEleSetList.length = 0; //清空数组 //处理新数据 let tagretChildEle = targetEleArray[0]; tagretChildEle.parentElement.classList.add("layui-this"); ObserverEleSetList.push(tagretChildEle); } } function isInViewPortByObserver(ele) { io.observe(ele); //添加观察对象 } // ---------------------------- 基本框架 ---------------------------- const FILETYPE = { JS: "js", CSS: "css", } const JSLOADTYPE = { ASYNC: "async", DEFER: "defer", } class Utils { static { Utils.addNewScript("js_cryptojs", GM_getResourceText("js_cryptojs")); Utils.loadjscssFile(GM_getResourceURL("js_immutable"), FILETYPE.JS, JSLOADTYPE.ASYNC); } static loadjscssFile(filePath, fileType, LOADTYPE) { let ele = undefined; if (fileType == FILETYPE.JS) { ele = document.createElement('script'); ele.setAttribute("src", filePath); if(LOADTYPE == JSLOADTYPE.ASYNC){ ele.setAttribute(JSLOADTYPE.ASYNC, JSLOADTYPE.ASYNC); }else if(LOADTYPE == JSLOADTYPE.DEFER){ ele.setAttribute(JSLOADTYPE.DEFER, JSLOADTYPE.DEFER); } } else if (fileType == FILETYPE.CSS) { ele = document.createElement("link"); ele.setAttribute("rel", "stylesheet"); ele.setAttribute("href", filePath); }else{ Console.orig.log(`不支持的文件类型: ${fileType}`); return; } if (ele != undefined) { document.getElementsByTagName("head")[0].appendChild(ele); } } static addNewStyle(id, newStyle) { var styleElement = document.getElementById(id); if (!styleElement) { styleElement = document.createElement('style'); styleElement.id = id; document.getElementsByTagName('head')[0].appendChild(styleElement); } styleElement.appendChild(document.createTextNode(newStyle)); } static addNewScript(id, newScript) { var scriptElement = document.getElementById(id); if (!scriptElement) { scriptElement = document.createElement('script'); scriptElement.id = id; document.getElementsByTagName('head')[0].appendChild(scriptElement); } scriptElement.appendChild(document.createTextNode(newScript)); } static addElementsByHTMLTemplateText(htmlTemplate, parentElement) { let ele = document.createElement("div"); if(parentElement == undefined) document.body.appendChild(ele); else parentElement.appendChild(ele); ele.outerHTML = htmlTemplate; } } class CustomUtils extends Utils { static{ // 拓展系统库 Math.constructor.roundEx = (nValue, n) => { //保留n位小数 return Math.round(nValue*Math.pow(10,n))/Math.pow(10,n); } } static handleCSSCompatibility(idPrefix, cssText) { //处理css兼容性, 防止外部css库修改原始css样式 let retCssText = cssText; let splitStrArray = cssText.split(/\{[\s\S]+?\}/g); //提取每一段的css选择器 let startIndex = 0, endIndex = 0; splitStrArray.forEach((item,index)=> { let cssSelectArray = item.match(/[A-Za-z0-9\:\-\_\@\.\#\+\>\<\*\?\;\!\[\]\=\"\'\`\/\\\(\)\{\}\~\^\$\|\%\&]+/g); //提取单个css选择器 //排除了, //Console.orig.log("cssSelectArray: " + cssSelectArray); if(cssSelectArray != null) { for (let i = 0; i < cssSelectArray.length; i++) { const cssSelectStr = cssSelectArray[i]; if(cssSelectStr.startsWith('@')) //需要排除的内容 break; if(cssSelectStr == "from" || cssSelectStr == "to") //需要排除的内容 break; if(cssSelectStr.indexOf("%") != -1) //需要排除的内容 break; if(cssSelectStr.indexOf(":") != -1) //需要排除的内容 (伪类元素选择器) break; //Console.orig.log("cssSelectArray: " + cssSelectArray); startIndex = retCssText.indexOf(cssSelectStr, startIndex); endIndex = startIndex + cssSelectStr.length; retCssText = retCssText.substring(0, startIndex) + cssSelectStr + idPrefix + retCssText.substring(endIndex); startIndex += idPrefix.length + 1 + cssSelectStr.length; //重新设置startIndex,防止重复查找 } startIndex = retCssText.indexOf("}", startIndex) + 1; //重新设置startIndex,防止从错误的位置开始查找 //Console.orig.log("\n" + retCssText); } }); //重定向字体文件 retCssText = retCssText.replaceAll("url(../", "url(https://cdn.staticfile.org/layui/2.8.17/"); //Console.orig.log("\n" + retCssText); return retCssText; } static handleJSCompatibility(idPrefix, jsText) { //处理js兼容性, 解决动态添加的节点与css兼容性的问题 jsText = jsText.replaceAll('class="', `class="${CSSCompatibilityFlagNoPoint} `); //全部匹配 //Console.orig.log("\n" + jsText); return jsText; } } class Debug { static #bDebug = false; //调试模式 static #bPublicApi = true; static { let ApiList = [ "GM_setValue", "GM_getValue", "GM_addStyle", "GM_deleteValue", "GM_listValues", "GM_addValueChangeListener", "GM_removeValueChangeListener", "GM_log", "GM_getResourceText", "GM_getResourceURL", "GM_registerMenuCommand", "GM_unregisterMenuCommand", "GM_openInTab", "GM_xmlhttpRequest", "GM_download", "GM_getTab", "GM_saveTab", "GM_getTabs", "GM_notification", "GM_setClipboard", "GM_info", ]; if(Debug.#bDebug && Debug.#bPublicApi){ let execute = ""; for (const [index, item] of ApiList.entries()) { execute += `if(typeof ${item} != 'undefined') g_window.${item} = ${item};\n`; } eval(execute); } } static set setDebug(debug){ Debug.#bDebug = debug; } static get getDebug(){ return Debug.#bDebug; } } class Console { static orig = window.console; //保存原始对象 static clear(){ clear(); } static log(...args){ Debug.getDebug && Console.orig.log('%c[log]', 'background: #ffa500; padding: 1px; color: #fff;', args); } static warn(...args){ Debug.getDebug && Console.orig.log('%c[warn]', 'background: #ffa500; padding: 1px; color: #fff;', args); } static error(...args){ Debug.getDebug && Console.orig.log('%c[error]', 'background: red; padding: 1px; color: #fff;', args); } static info(...args){ Debug.getDebug && Console.orig.log('%c[info]', 'background: #ffa500; padding: 1px; color: #fff;', args); } static table(...args){ Debug.getDebug && Console.orig.table(args); } } g_window.Console = Console; class Performance { constructor(name){ if(name != undefined) this._name = name; else this._name = ''; this.nTimeStart = 0; this.nTimeEnd = 0; } setStartTime() { this.nTimeStart = performance.now(); } setEndTime() { this.nTimeEnd = performance.now(); } getEndTime() { this.nTimeEnd = performance.now(); let executionTime = this.nTimeEnd - this.nTimeStart; Console.orig.log(`${this._name}代码执行时间: ${executionTime.toFixed(3)} 毫秒`); // return executionTime; } } const CallType = { Positive: "Positive", Reverse: "Reverse", } const HookType = { Single: "Single", Multi: "Multi", } class ProxyGenerator { //代理生成器: 生成代理方法 generateSingle(fcName_, origFcAddr_, newFcAddr_, newFcAddrLastCall_) { //单hook function _proxy_s(...args){ // debugger let prototype = _proxy_s.prototype; if(prototype.hasOwnProperty("_fcData")) { let fcData = _proxy_s.prototype._fcData; let origArgs = [...args]; //复制数组 args.unshift(origFcAddr_); //添加到数组开头 let lastCallArgs = [...args]; //复制数组 if(fcData.newFcAddr.length == 2) args.splice(1, 0, undefined);; //将undefined插入到数组第二个元素 let interceptor = fcData.newFcAddr.apply(this, args); //调用newFcAddr if(interceptor == true || interceptor == undefined){ let ret = fcData.origFcAddr.apply(this, origArgs); //调用原函数 //this是调用方传递的,也一并传递 if(fcData.newFcAddrLastCall && fcData.newFcAddrLastCall.length == 2) lastCallArgs.splice(1, 0, ret);; //将ret插入到数组第二个元素 fcData.newFcAddrLastCall.apply(this, lastCallArgs); //调用newFcAddrLastCall return ret; }else{ return; //否则就拦截 } } } _proxy_s.prototype._fcData = { fcName: fcName_ || "", origFcAddr: origFcAddr_ || null, newFcAddr: newFcAddr_ || null, newFcAddrLastCall: newFcAddrLastCall_ || null, }; return _proxy_s; } generateMulti(key_) { //多层hook-正序调用 function _proxy_m(...args){ // debugger let prototype = _proxy_m.prototype; if(prototype.hasOwnProperty("_hk")) { let hk = _proxy_m.prototype._hk; let key = hk.key; let hook = hk.obj; let hookDataArray = hook.fcHookMap.get(key); let ret; let origArgs = [...args]; //复制数组 for (const [index, item] of hookDataArray.entries()) { //遍历多层hook args.unshift(item.origFcAddr); //添加到数组开头 let lastCallArgs = [...args]; //复制数组 if(item.newFcAddr.length == 2) args.splice(1, 0, undefined);; //将undefined插入到数组第二个元素 let interceptor = item.newFcAddr.apply(this, args); //调用newFcAddr if(interceptor == true || interceptor == undefined){ ret = item.origFcAddr.apply(this, origArgs); //调用原函数 //this是调用方传递的,也一并传递 if(item.newFcAddrLastCall && item.newFcAddrLastCall.length == 2) lastCallArgs.splice(1, 0, ret);; //将ret插入到数组第二个元素 item.newFcAddrLastCall.apply(this, lastCallArgs); //调用newFcAddrLastCall if(index == hookDataArray.length -1){ //在多层hook结尾, 返回返回值 return ret; } }else{ return; //否则就拦截 } } } } _proxy_m.prototype._hk = { key: key_ || "", obj: this, }; return _proxy_m; } generateMultiReverse(key_) { //多层hook-逆序调用 function _proxy_mr(...args){ // debugger let prototype = _proxy_mr.prototype; if(prototype.hasOwnProperty("_hk")) { let hk = _proxy_mr.prototype._hk; let key = hk.key; let hook = hk.obj; let hookDataArray = hook.fcHookMap.get(key); let ret; let origArgs = [...args]; //复制数组 for (let i = hookDataArray.length -1; i >= 0; i--) { //遍历多层hook const item = hookDataArray[i]; args.unshift(item.origFcAddr); //添加到数组开头 let lastCallArgs = [...args]; //复制数组 if(item.newFcAddr.length == 2) args.splice(1, 0, undefined);; //将undefined插入到数组第二个元素 let interceptor = item.newFcAddr.apply(this, args); //调用newFcAddr if(interceptor == true || interceptor == undefined){ ret = item.origFcAddr.apply(this, origArgs); //调用原函数 //this是调用方传递的,也一并传递 if(item.newFcAddrLastCall && item.newFcAddrLastCall.length == 2) lastCallArgs.splice(1, 0, ret);; //将ret插入到数组第二个元素 item.newFcAddrLastCall.apply(this, lastCallArgs); //调用newFcAddrLastCall if(i == 0){ //在多层hook开头, 返回返回值 return ret; } }else{ return; //否则就拦截 } } } } _proxy_mr.prototype._hk = { key: key_ || "", obj: this, }; return _proxy_mr; } } class Hook { #proxyGenerator = new ProxyGenerator(); #callType = CallType.Positive; #proxyInfo = [ CryptoJS.SHA1(this.#proxyGenerator.generateSingle().toString().replace(/\s+/g, "")).toString(), CryptoJS.SHA1(this.#proxyGenerator.generateMulti().toString().replace(/\s+/g, "")).toString(), CryptoJS.SHA1(this.#proxyGenerator.generateMultiReverse().toString().replace(/\s+/g, "")).toString(), ]; constructor(proxyGenerator){ if(proxyGenerator != undefined) this.setProxyGenerator = proxyGenerator; this.fcHookMap = new Map(); //this.objHookMap = new Map(); } set setProxyGenerator(proxyGenerator) { this.#proxyGenerator = proxyGenerator; } setCallType(callType) { this.#callType = callType; for (const [key, item] of this.fcHookMap) { //遍历map if(item.length > 1){ //排除单hook,因为单hook没有顺序区别 if(callType == CallType.Positive){ eval(`${key}=this.#proxyGenerator.generateMulti('${key}')`); //替换 return true; }else if(callType == CallType.Reverse){ eval(`${key}=this.#proxyGenerator.generateMultiReverse('${key}')`); //替换 return true; }else{ return false; } } } } get getCallType(){ return this.#callType; } //设置hook时,设置的是代理函数,然后在代理函数里调用newFcAddr //流程: 其他代码调用 fcAddr =>(实际调用) 代理函数 ->(里面先拿到原函数) 再调用newFcAddr ->(结束后根据返回值如果为true则调用原函数,并返回返回值,false就拦截) //如果newFcAddr没有返回值 (说明不使用拦截功能),就正常调用原函数,返回返回值 //代理函数是动态生成的,并且代理函数里不能调用被hook的函数,不然会无限递归调用 setSingleHook(fcAddrStr, newFcAddr, newFcAddrLastCall){ // debugger if(Object.prototype.toString.apply(fcAddrStr) != '[object String]'){ Console.orig.log("setSingleHook()失败, fcAddrStr参数错误, 应该是字符串! 实际是: " + fcAddrStr); return false; } let fcAddr = eval(fcAddrStr); let type = Object.prototype.toString.apply(fcAddr); let name = fcAddr.name; let key = fcAddrStr; if(type == '[object Function]'){ if(!this.fcHookMap.has(key)){ Console.orig.log("新fcHook: " + key); this.fcHookMap.set(key, [{ fcName: name, origFcAddr: fcAddr, newFcAddr: newFcAddr, newFcAddrLastCall: newFcAddrLastCall, }]); //保存原始 eval(`${fcAddrStr}=this.#proxyGenerator.generateSingle('${name}', fcAddr, newFcAddr, newFcAddrLastCall)`); //替换 return true; }else{ //先取消hook,再重新单hook let hookDataArray = this.fcHookMap.get(key); this.unHookByNameOfSpecifyOrAll(fcAddrStr, hookDataArray[0].newFcAddr); return this.setSingleHook(fcAddrStr, newFcAddr, newFcAddrLastCall); } }else{ Console.orig.log("setSingleHook()失败, 不支持的type: " + type); return false; } } setHook(fcAddrStr, newFcAddr, newFcAddrLastCall){ // debugger if(Object.prototype.toString.apply(fcAddrStr) != '[object String]'){ Console.orig.log("setHook()失败, fcAddrStr参数错误, 应该是字符串! 实际是: " + fcAddrStr); return false; } let fcAddr = eval(fcAddrStr); let type = Object.prototype.toString.apply(fcAddr); let name = fcAddr.name; let key = fcAddrStr; if(type == '[object Function]'){ if(!this.fcHookMap.has(key)){ Console.orig.log("新fcHook: " + key); this.fcHookMap.set(key, [{ fcName: name, origFcAddr: fcAddr, newFcAddr: newFcAddr, newFcAddrLastCall: newFcAddrLastCall, }]); //保存原始 eval(`${fcAddrStr}=this.#proxyGenerator.generateSingle('${name}', fcAddr, newFcAddr, newFcAddrLastCall)`); //替换 return true; }else{ //多层Hook Console.orig.log("多层fcHook: " + key); let hookDataArray = this.fcHookMap.get(key); hookDataArray.push({ fcName: name, origFcAddr: hookDataArray[0].origFcAddr, newFcAddr: newFcAddr, newFcAddrLastCall: newFcAddrLastCall, }); //保存原始 this.fcHookMap.set(key, hookDataArray); if(this.#callType == CallType.Positive){ eval(`${fcAddrStr}=this.#proxyGenerator.generateMulti('${key}')`); //替换 return true; }else if(this.#callType == CallType.Reverse){ eval(`${fcAddrStr}=this.#proxyGenerator.generateMultiReverse('${key}')`); //替换 return true; } } }else{ Console.orig.log("setHook()失败, 不支持的type: " + type); return false; } } isHook(fcAddrStr){ if(Object.prototype.toString.apply(fcAddrStr) != '[object String]'){ Console.orig.log("isHook()失败, fcAddrStr参数错误, 应该是字符串! 实际是: " + fcAddrStr); return false; } if(this.fcHookMap.has(fcAddrStr)){ return true; }else{ let fcAddr = eval(fcAddrStr); if(fcAddr == undefined){ return false; } let hash = CryptoJS.SHA1(fcAddr.toString().replace(/\s+/g, "")).toString(); for (const [index, item] of this.#proxyInfo.entries()) { if(item == hash) { return true; } } return false; } } getHookType(fcAddrStr){ if(Object.prototype.toString.apply(fcAddrStr) != '[object String]'){ Console.orig.log("getHookType()失败, fcAddrStr参数错误, 应该是字符串! 实际是: " + fcAddrStr); return false; } const hookDataArray = this.fcHookMap.get(fcAddrStr); if(hookDataArray != undefined) { return hookDataArray.length == 1 ? HookType.Single : HookType.Multi; }else{ return null; } } #unHookCommon_unload(fcAddrStr, hookData, hookDataArray, i){ if(hookDataArray.length == 1){ //单hook 或 只有1个元素 eval(`${fcAddrStr}=hookData.origFcAddr`); //还原 this.fcHookMap.delete(fcAddrStr); return true; }else{ //多层 hookDataArray.splice(i, 1); //删除当前记录 this.fcHookMap.set(fcAddrStr, hookDataArray); return true; } } #unHookCommon_mode(fcAddrStr, newFcAddr, hookData, hookDataArray, i){ if(newFcAddr != undefined && hookData.newFcAddr == newFcAddr){ //unHookByName //取消指定key某一层hook return this.#unHookCommon_unload(fcAddrStr, hookData, hookDataArray, i); }else if(newFcAddr == undefined){ //unAllHookByName //取消指定key下的所有hook return this.#unHookCommon_unload(fcAddrStr, hookData, hookDataArray, i); } } //取消hook, 主要针对多层hook //unHook(key, 新函数地址) //unHook(key) //如果指定newFcAddr, 则取消指定key某一层hook //如果只指定fcAddrStr, 则取消指定key下的所有hook unHookByNameOfSpecifyOrAll(fcAddrStr, newFcAddr){ // debugger if(Object.prototype.toString.apply(fcAddrStr) != '[object String]'){ Console.orig.log("unHookByNameOfSpecifyOrAll()失败, fcAddrStr参数错误, 应该是字符串! 实际是: " + fcAddrStr); return false; } if(this.isHook(fcAddrStr)){ let fcAddr = eval(fcAddrStr); let type = Object.prototype.toString.apply(fcAddr); let key = fcAddrStr; if(type == '[object Function]'){ if(this.fcHookMap.has(key)){ let hookDataArray = this.fcHookMap.get(key); for (let i = hookDataArray.length -1; i >= 0; i--) { //逆序遍历 const hookData = hookDataArray[i]; let ret = this.#unHookCommon_mode(fcAddrStr, newFcAddr, hookData, hookDataArray, i); if(newFcAddr != undefined && ret){ //单unHook, 并且执行成功, 则直接返回 return true; }else if(newFcAddr == undefined && i == 0){ //否则是 取消指定key下的所有hook, 就判断是否循坏完成, 循坏完成后返回 return true; } } Console.orig.log("unHook()失败, 找到hook, 但是实参newFcAddr有误: " + newFcAddr.name); return false; }else{ Console.orig.log("unHook()失败, 找到hook, 但是实参key有误: " + key); return false; } }else{ Console.orig.log("unHook()失败, 不支持的type: " + type); return false; } }else{ Console.orig.log("unHook()失败, 找不到指定的hook!"); return false; } } unAllHook(){ //取消所有 for (const [key, hookDataArray] of this.fcHookMap) { //遍历map for (let i = hookDataArray.length -1; i >= 0; i--) { //逆序遍历 const hookData = hookDataArray[i]; this.#unHookCommon_mode(key, undefined, hookData, hookDataArray, i); } } return true; } } class DynamicProxy { static apply(obj, fcName, ...args) { //执行目标对象的方法(对象, 方法名, 方法参数) let prototype = Object.getPrototypeOf(obj); if(prototype.hasOwnProperty(fcName)) { return eval("prototype." + fcName).apply(obj, args); } } static autoWired(mainObj, targetClass, CallBack) { //依赖注入 for (const [name, value] of Object.entries(mainObj)) { //遍历对象获取指定类型的属性, 属性的类型通过参数传入 if(value instanceof targetClass){ return CallBack(name, value); //CallBack(属性名, 属性值) } } return null; } } class CodeTemplate { static hook = new Hook(); static autoWiredProxy(mainObj, executeFcName){ //依赖注入代理 DynamicProxy.autoWired(mainObj, Performance, (name, value)=>{ //使用这个属性调用apply实现具体的功能 DynamicProxy.apply(value, executeFcName); //apply(属性值(对象), 要执行的方法) return true; }); } static autoWired_Performance(fcAddrStr, mainObj){ const fcStartStub = (origFcAddr, ...args)=>{ //mainObj.perfor.setStartTime(); //直接调用 //DynamicProxy.apply(mainObj.perfor, "setStartTime"); //间接调用 CodeTemplate.autoWiredProxy(mainObj, "setStartTime"); //依赖注入 }; const fcEndStub = (origFcAddr, ...args)=>{ //mainObj.perfor.getEndTime(); //直接调用 //DynamicProxy.apply(mainObj.perfor, "getEndTime",); //间接调用 CodeTemplate.autoWiredProxy(mainObj, "getEndTime"); //依赖注入 }; CodeTemplate.hook.setHook(fcAddrStr, fcStartStub, fcEndStub); } } const URLTYPE = { LINK: 1, //链接 FUZZYMATCH: 1 << 1, //模糊匹配 RegExp: 1 << 2, //正则表达式 } class URL { static test(testURL, urlTYPE, urlRule) { if(urlTYPE == URLTYPE.LINK){ return urlRule.includes(testURL); //子串 }else if(urlTYPE == URLTYPE.FUZZYMATCH){ let targetURL = urlRule; //转义可能的特殊字符 //\和/这种字符必须首先执行 //*和?这种通配符放在后面单独处理 let specharsArray = ['\\', '/', '[', ']', '(', ')', '{', '}', '^', '$', '-', '.', '+', '|', ',', ':', '=', '!', '<', '%']; specharsArray.forEach((item,index)=> { targetURL = targetURL.replaceAll(item, '\\' + item); }); //替换通配符 //?必须比*先执行 targetURL = targetURL.replaceAll("?", "[\\s\\S]?"); targetURL = targetURL.replaceAll("*", "([\\s\\S]+)?"); //测试URL let regExp = new RegExp(targetURL); return regExp.test(testURL); }else if(urlTYPE == URLTYPE.RegExp){ return urlRule.test(testURL); } } } class CallBack { static reportInit() { //加载库 Utils.addNewScript("js_layui", CustomUtils.handleJSCompatibility(CSSCompatibilityFlag, GM_getResourceText("js_layui"))); Utils.addNewStyle("css_layui", CustomUtils.handleCSSCompatibility(CSSCompatibilityFlag, GM_getResourceText("css_layui"))); Utils.loadjscssFile(GM_getResourceURL("css_fontAwesome"), FILETYPE.CSS); //加载自定义内容 Utils.addNewStyle("commonCSS", commonCSS); Utils.addNewStyle("pageExtend", pageExtendBaseCSS); Utils.addNewStyle("MainAppBaseCSS", MainAppBaseCSS); //addElementsByHTMLTemplateText(settingUITemplate); Utils.addElementsByHTMLTemplateText(catalogUIBase); //执行初始化 PageExtend.init(); MainApp.init(); return true; } static reportPageCallBack() { PageExtend.run(); MainApp.run(); return true; } static mainPageInit() { return true; } static mainPageCallBack() { document.getElementsByName("show-rating").forEach((ele)=>{ele.checked = true}); //默认勾选 显示Rating const map = new Map(); let childEle = document.getElementById("mortal-model-tag").children; for (let i = 0; i < childEle.length; i++) { const ele = childEle[i]; map.set(ele.value, ele.innerText); //将数据保存到map } const jsonStr = JSON.stringify(Object.fromEntries(map)); localStorage.setItem("Mortal_New", childEle[0].value); //牌谱解析页面,默认使用最新的Mortal localStorage.setItem("Mortal_Type", jsonStr); return true; } } class App { static perfor = new Performance("dom"); static URLRuleSet = [{ name: "牌谱解析页面", url: /^https?:\/\/mjai.ekyu.moe\/report\/[A-Za-z0-9-_]+.html/, urlType: URLTYPE.RegExp, exclude: [ ], init: CallBack.reportInit, entry: CallBack.reportPageCallBack, },{ name: "主页", url: /^https?:\/\/mjai.ekyu.moe\/?([A-Za-z0-9-_]+.html)?/, urlType: URLTYPE.RegExp, exclude: [ //排除主动访问非页面URL的内容,比如图片等 /^https?:\/\/mjai.ekyu.moe\/?([A-Za-z0-9-_]+)?\/?[A-Za-z0-9-_]+.(?!.*(html|htm|jsp|php|asp))/, ], init: CallBack.mainPageInit, entry: CallBack.mainPageCallBack, }, ]; //URL规则集 constructor(){ this.init(); } init() { document.addEventListener('readystatechange', event => { if (event.target.readyState === 'interactive') { App.perfor.setStartTime(); //Console.orig.log('interactive'); this.yun(); //initLoader(); } else if (event.target.readyState === 'complete') { App.perfor.getEndTime(); //Console.orig.log('complete'); //initApp(); } }); return true; } yun(){ let isExclude = false; for (const [index, item] of App.URLRuleSet.entries()) { let currentURL = window.location.origin + window.location.pathname; if(URL.test(currentURL, item.urlType, item.url)){ //测试URL是否匹配 if(item.exclude.length > 0){ for (const [i, excludeItem] of item.exclude.entries()) { if(URL.test(currentURL, item.urlType, excludeItem)){ //测试URL是否被排除 isExclude = true; break; } } } if(!isExclude){ if(item.init.apply()) item.entry.apply(); } return; } } } } // ---------------------------- 功能定义 ---------------------------- class MortalBase { constructor(){ } static init(){ return true; } static run(){ } static stop(){ } static clean(){ } } const OUTSTYLE = { A: 1, B: 1 << 1, C: 1 << 2, } class PageExtend extends MortalBase { static perfor = new Performance("PageExtend"); constructor(){ //super(); } static init(){ if(Debug.getDebug){ CodeTemplate.autoWired_Performance("PageExtend.run", PageExtend); } return true; } static run(){ //显示恶手 PageExtend.showBadChoose(); //修改 元数据 选项卡 PageExtend.alterMetaData(); //起始信息详细化 PageExtend.showStartInfo(); //列出选择权重 PageExtend.showChooseWeight(); } static stop(){ } static clean(){ } //显示恶手 static showBadChoose() { const orderLossEleArray = document.getElementsByClassName("order-loss"); for (let i = 0; i < orderLossEleArray.length; i++) { const orderLoss = orderLossEleArray[i]; const nChooseIndex = parseInt(orderLoss.innerText.match(/[\d]+/)[0]); const nChooseSum = parseInt(orderLoss.nextSibling.textContent.match(/[\d]+/)[0]); const turnInfo = orderLoss.parentElement; const summary = turnInfo.parentElement; const collapseEntry = summary.parentElement; const table = collapseEntry.lastChild.firstChild; const tbody = table.lastChild; const nChooseTR = tbody.childNodes[nChooseIndex -1]; const nChooseWeightTD = nChooseTR.lastChild; const chosenWeight = parseFloat(nChooseWeightTD.innerHTML.replace(/<.*?>/g, "")); //过滤html标签, 只保留文字内容 const kyokuTitleEle = collapseEntry.parentElement.parentElement.firstChild.getElementsByTagName("a")[0]; if (chosenWeight <= parseFloat(badMoveUpperLimit)) { //严重恶手 const badChooseNode = document.createElement("span"); badChooseNode.classList.add("badChoose"); badChooseNode.classList.add("level1"); badChooseNode.innerHTML = ` \u00A0\u00A0\u00A0${i18nText.badMove}${i18nText.badMoveUp}`; turnInfo.appendChild(badChooseNode); collapseEntry.classList.add("collapseEntryL1"); badChooseNum++; //收集数据3-恶手 strArray3.push({ name: kyokuTitleEle.textContent + " " + summary.textContent, href: kyokuTitleEle.href.match(/#[\S\s]+$/)[0] }); }else if (chosenWeight <= parseFloat(badMoveUpperLimitCustom)) { //普通恶手 const badChooseNode = document.createElement("span"); badChooseNode.classList.add("badChoose"); badChooseNode.classList.add("level2"); badChooseNode.innerHTML = ` \u00A0\u00A0\u00A0${i18nText.badMove}${i18nText.badMoveDown}`; turnInfo.appendChild(badChooseNode); collapseEntry.classList.add("collapseEntryL2"); badChooseNumCustom++; //收集数据3-恶手 strArray3.push({ name: kyokuTitleEle.textContent + " " + summary.textContent, href: kyokuTitleEle.href.match(/#[\S\s]+$/)[0] }); } //收集数据2-不一致 strArray2.push({ name: kyokuTitleEle.textContent + " " + summary.textContent, href: kyokuTitleEle.href.match(/#[\S\s]+$/)[0] }); } //for } //修改 元数据 选项卡 static alterMetaData() { // 新增 显示 Mortal 版本 const jsonStr = localStorage.getItem("Mortal_Type"); const mortal_New = localStorage.getItem("Mortal_New"); var mortalMap = null; if(jsonStr != null) { let obj = Object.entries(JSON.parse(jsonStr)); mortalMap = new Map(obj); } // 修改 元数据 选项卡 let metaData = null; const detailsElements = document.getElementsByTagName("details"); for (let i = 0; i < detailsElements.length; i++) { const details = detailsElements[i]; const summary = details.firstChild; if (summary.firstChild.textContent == i18nText.metaData) { metaData = details; metaData.toggleAttribute("open", true); //打开 元数据 选项卡 break; } } const metaDataDL = metaData.lastChild; let matchRatioDD = null; let version = null; for (let i = 0; i < metaDataDL.childNodes.length; i++) { const metaDataChild = metaDataDL.childNodes[i]; if(metaDataChild.nodeName == "DT" && metaDataChild.textContent == i18nText.modelTag) { let ele = metaDataDL.childNodes[i + 1]; //判断当前是否是最新版本的mortal if(mortal_New != null) { if(mortal_New != ele.innerText) { let aiEle = metaDataDL.childNodes[i - 1]; aiEle.classList.add("color1"); aiEle.innerText = aiEle.innerText + ` \u00A0\u00A0\u00A0${i18nText.badMoveError}`; } } //处理当前版本 if(mortalMap != null) { let mortalValue = mortalMap.get(ele.innerText); if(mortalValue != undefined) { ele.innerText = mortalValue; } } } if (metaDataChild.nodeName == "DT" && metaDataChild.textContent == i18nText.matchRatio) { matchRatioDD = metaDataDL.childNodes[i + 1]; version = metaDataDL.childNodes[i + 2]; // metaDataDL.childNodes[i-2].classList.add("color2"); //rating metaDataDL.childNodes[i-1].classList.add("color2"); metaDataDL.childNodes[i].classList.add("color2"); //AI 一致率 metaDataDL.childNodes[i+1].classList.add("color2"); break; } } const matchRatioText = matchRatioDD.textContent; const chooseNumStr = matchRatioText.substring(matchRatioText.indexOf("/") + 1); const chooseNum = parseInt(chooseNumStr); const badChooseRatioDT = document.createElement("dt"); badChooseRatioDT.classList.add("color3"); badChooseRatioDT.innerHTML = i18nText.badMoveRatio + i18nText.badMoveNull + badMoveUpperLimit + i18nText.badMoveSymbol; const badChooseRatioDD = document.createElement("dd"); badChooseRatioDD.classList.add("color3"); badChooseRatioDD.innerHTML = `${badChooseNum}/${chooseNum} = ${(100 * badChooseNum / chooseNum).toFixed(3)}%`; metaDataDL.insertBefore(badChooseRatioDD, version); metaDataDL.insertBefore(badChooseRatioDT, badChooseRatioDD); /* 新增 计算总恶手数 */ badChooseNumCustom += badChooseNum; //计算总恶手数 const badChooseRatioDT2 = document.createElement("dt"); badChooseRatioDT2.classList.add("color3"); badChooseRatioDT2.innerText = i18nText.badMoveRatio + i18nText.badMoveNull + badMoveUpperLimitCustom + i18nText.badMoveSymbol; const badChooseRatioDD2 = document.createElement("dd"); badChooseRatioDD2.classList.add("color3"); badChooseRatioDD2.innerHTML = `${badChooseNumCustom}/${chooseNum} = ${(100 * badChooseNumCustom / chooseNum).toFixed(3)}%`; metaDataDL.insertBefore(badChooseRatioDD2, version); metaDataDL.insertBefore(badChooseRatioDT2, badChooseRatioDD2); } //起始信息详细化 static showStartInfo() { /* 起始信息详细化 */ function parmeHandle(eastScoreChange, southScoreChange, westScoreChange, northScoreChange) { let scoreArray = [{sc: eastScoreChange, i: 0}, {sc: southScoreChange, i: 1}, {sc: westScoreChange, i: 2}, {sc: northScoreChange, i: 3}]; let newScoreArray = scoreArray.filter((obj) => { return obj.sc != 0; }); let scoreAddArray = newScoreArray.filter((obj) => { //荣和的玩家 return obj.sc > 0; }); let scoreSubArray = newScoreArray.filter((obj) => { //放铳的玩家 return obj.sc < 0; }); scoreAddArray.sort((a,b)=>{return b.sc-a.sc}); return {scoreArray: scoreArray, newScoreArray: newScoreArray, scoreAddArray: scoreAddArray, scoreSubArray: scoreSubArray}; } function handleRon(kyoku, startPlayerIndex, eastScoreChange, southScoreChange, westScoreChange, northScoreChange) { //处理荣和 let obj = parmeHandle(eastScoreChange, southScoreChange, westScoreChange, northScoreChange); let scoreAddArray = obj.scoreAddArray; let scoreSubArray = obj.scoreSubArray; let selfPlayerIndex = getPlayerIndexByPlayerSeatName(getPlayerSeatNameByPlayerIndex(startPlayerIndex, kyoku, OUTSTYLE.B), OUTSTYLE.B); // let str = ""; for (let i = 0; i < scoreAddArray.length; i++) { const scoreAdd = scoreAddArray[i]; let scAddPlayerSeatName = getPlayerSeatNameByPlayerIndex(scoreAdd.i, kyoku, OUTSTYLE.B); //荣和的玩家 let scAddPlayerViewName = getSelfViewPlayerNameByTargetPlayerIndex(selfPlayerIndex, getPlayerIndexByPlayerSeatName(scAddPlayerSeatName, OUTSTYLE.B)); let scSubPlayerSeatName = getPlayerSeatNameByPlayerIndex(scoreSubArray[0].i, kyoku, OUTSTYLE.B); //放铳的玩家 let scSubPlayerViewName = getSelfViewPlayerNameByTargetPlayerIndex(selfPlayerIndex, getPlayerIndexByPlayerSeatName(scSubPlayerSeatName, OUTSTYLE.B)); str += scAddPlayerSeatName + ` (${scAddPlayerViewName}) ` + `<span class="color4">${i18nText.Ron}</span>` + i18nText.badMoveNull + scSubPlayerSeatName + ` (${scSubPlayerViewName}) ` + "+" + scoreAdd.sc + i18nText.badMoveNull + scoreSubArray[0].sc; } return str; } function handleTsumo(kyoku, startPlayerIndex, eastScoreChange, southScoreChange, westScoreChange, northScoreChange) { //处理自摸 } function handleRyuukyoku(kyoku, startPlayerIndex, eastScoreChange, southScoreChange, westScoreChange, northScoreChange) { //处理流局 } function getTextByOutStyle(outStyle, index) { if(outStyle == OUTSTYLE.A) return eval("i18nText.seatTypeA" + index); //seatTypeA0 //东起 else if(outStyle == OUTSTYLE.B) return eval("i18nText.seatTypeB" + index); //seatTypeB0 //东家 else if(outStyle == OUTSTYLE.C) return eval("i18nText.seatTypeC" + index); //seatTypeC0 //自家 else{ console.warn("outStyle是无效的!"); } } function getSelfViewPlayerNameByTargetPlayerIndex(selfIndex, targetPlayerIndex) { const viewMap = new Map(); let viewArray = [ { value: -3, name: i18nText.seatTypeC1 }, //下家 { value: -2, name: i18nText.seatTypeC2 }, //对家 { value: -1, name: i18nText.seatTypeC3 }, //上家 { value: 0, name: i18nText.seatTypeC0 }, //自家 { value: 1, name: i18nText.seatTypeC1 }, //下家 { value: 2, name: i18nText.seatTypeC2 }, //对家 { value: 3, name: i18nText.seatTypeC3 }, //上家 ]; viewArray.forEach((item,index)=> { viewMap.set(item.value, item.name); }); const getValue = (offset) => { if(offset > 0) return offset -4; else return offset +4; } let offset = targetPlayerIndex - selfIndex; let result; do { result = viewMap.get(offset); } while (offset = getValue(offset), result == undefined); return result; } function getPlayerIndexByPlayerSeatName(playerSeatName, outStyle) { var seatArray; if(outStyle == OUTSTYLE.A) seatArray = [i18nText.seatTypeA0, i18nText.seatTypeA1, i18nText.seatTypeA2, i18nText.seatTypeA3]; //东起, 南起, 西起, 北起 else if(outStyle == OUTSTYLE.B) seatArray = [i18nText.seatTypeB0, i18nText.seatTypeB1, i18nText.seatTypeB2, i18nText.seatTypeB3]; //东家, 南家, 西家, 北家 else{ console.warn("outStyle是无效的!"); seatArray = [i18nText.seatTypeA0, i18nText.seatTypeA1, i18nText.seatTypeA2, i18nText.seatTypeA3]; //东起, 南起, 西起, 北起 } const seatMap = new Map(); seatArray.forEach((item,index)=> { seatMap.set(item, index); }); return seatMap.get(playerSeatName); } function getPlayerSeatNameByPlayerIndex(playerIndex, kyoku, outStyle) { var seatArray; if(outStyle == OUTSTYLE.A) seatArray = [i18nText.seatTypeA0, i18nText.seatTypeA3, i18nText.seatTypeA2, i18nText.seatTypeA1]; //东起, 北起, 西起, 南起 else if(outStyle == OUTSTYLE.B) seatArray = [i18nText.seatTypeB0, i18nText.seatTypeB3, i18nText.seatTypeB2, i18nText.seatTypeB1]; //东家, 北家, 西家, 南家 else{ console.warn("outStyle是无效的!"); seatArray = [i18nText.seatTypeA0, i18nText.seatTypeA3, i18nText.seatTypeA2, i18nText.seatTypeA1]; //东起, 北起, 西起, 南起 } const seatMap = new Map(); seatArray.forEach((item,index)=> { seatMap.set(item, index); }); const getValueByEachArray = (array, startIndex, eachCount) => { let length = array.length; let targetIndex = startIndex; for (let i = eachCount; i > 0; i--) { if(++targetIndex >= length) targetIndex = 0; } return array[targetIndex]; } const getStart = (playerIndex, kyoku, outStyle) => { switch (playerIndex + kyoku) { case 0: return getTextByOutStyle(outStyle, 0); //东起 case 1: return getTextByOutStyle(outStyle, 1); //南起 case 2: return getTextByOutStyle(outStyle, 2); //西起 case 3: return getTextByOutStyle(outStyle, 3); //北起 default: return getStart(playerIndex, kyoku -4, outStyle); } } const get = (playerIndex, kyoku, outStyle) => { let startIndex = seatMap.get(getStart(playerIndex, 0, outStyle)); return getValueByEachArray(seatArray, startIndex, kyoku); } if(kyoku == 0) return getStart(playerIndex, kyoku, outStyle); else return get(playerIndex, kyoku, outStyle); } function getTextLineNum(ele) { let styles = getComputedStyle(ele, null); let lineHeight = parseFloat(styles.lineHeight); let height = parseFloat(styles.height); let offsetHeight = parseFloat(ele.offsetHeight); let lineNum = offsetHeight / lineHeight; return Math.round(lineNum); } const summaryEle = document.getElementsByClassName("kyoku-toc")[0]; summaryEle.classList.add("position_re"); for (let j = 0; j < summaryEle.children.length; j++) { const summary = summaryEle.children[j]; summary.classList.add("min-w-20"); } const kyokuEle = summaryEle.getElementsByTagName("a"); const endInfoEle = summaryEle.getElementsByClassName("end-status"); // const section = document.getElementsByTagName("section"); for (let i = 0, length = section.length; i != length; ++i) { const titleEle = section[i].children[0]; const titleKyokuEle = titleEle.getElementsByTagName("a"); //只有1个元素 const titleEndInfoEle = titleEle.getElementsByClassName("end-status"); //只有1个元素 const tenhouData = section[i].getElementsByTagName("iframe")[0].src; const playerIndexStr = tenhouData.match(/tw=[0-3]/)[0]; const startPlayerIndex = parseInt(playerIndexStr.substring(playerIndexStr.length -1)); //起始玩家索引 const json = JSON.parse(decodeURI(tenhouData.substring(tenhouData.indexOf("{")))); //天凤对局数据 const kyoku = json.log[0][0][0]; //局数 const count = json.log[0][0][1]; //本场数 const currScore = json.log[0][1]; //当前点数 const scoreChange = json.log[0][json.log[0].length -1]; //点数变动 const endMode = scoreChange[0]; //解析规则 if(matchRule.isInit == false) { const disp = json.rule.disp; if(disp.indexOf("間") != -1) { //雀魂 matchRule.isRon3 = true; //Console.orig.log("雀魂牌谱"); }else if(disp.indexOf("Player") != -1) { //一番街 matchRule.isRon3 = false; //Console.orig.log("一番街牌谱"); }else{ //默认为天凤 matchRule.isRon3 = false; //Console.orig.log("默认为天凤牌谱(包括自定义牌谱)"); } matchRule.isInit = true; } //四家当前分数 const eastScore = currScore[0]; //东 const southScore = currScore[1]; //南 const westScore = currScore[2]; //西 const northScore = currScore[3]; //北 //四家分数变化(直接) //送棒的-1000没有显示 let eastScoreChange = []; let southScoreChange = []; let westScoreChange = []; let northScoreChange = []; let ronCount = (scoreChange.length -1) / 2; if(scoreChange.length > 1) { //比如九种九牌, 是没有分数变化的数据的 //是否有多家荣和 for (let j = 0; j < ronCount; j++) { //处理可能的多家荣和 eastScoreChange.push(scoreChange[1+ j*2][0]); //东 southScoreChange.push(scoreChange[1+ j*2][1]); //南 westScoreChange.push(scoreChange[1+ j*2][2]); //西 northScoreChange.push(scoreChange[1+ j*2][3]); //北 } } //判断模式 if(endMode == tenhouText.Ron) { //自摸、荣和 let str = ""; for (let j = 0; j < ronCount; j++) { //处理可能的多家荣和 if(j>0) //处理多家 str += ", "; let obj = parmeHandle(eastScoreChange[j], southScoreChange[j], westScoreChange[j], northScoreChange[j]); if(eastScoreChange[j] == 0 || southScoreChange[j] == 0 || westScoreChange[j] == 0 || northScoreChange[j] == 0) { //如果有任何一家分数变动为0, 则为荣和 str += handleRon(kyoku, startPlayerIndex, eastScoreChange[j], southScoreChange[j], westScoreChange[j], northScoreChange[j]); }if(obj.scoreAddArray.length == 3) { //判断是否是三家荣和 if(matchRule.isRon3) //如果启用了三家和了的规则 //? 可能是没有必要的判断? 等待使用三种游戏牌谱分别进行查证 str += handleRon(kyoku, startPlayerIndex, eastScoreChange[j], southScoreChange[j], westScoreChange[j], northScoreChange[j]); else{ //流局 //? 可能是没有必要的判断? 等待使用三种游戏牌谱分别进行查证 // let str = handleRyuukyoku(kyoku, startPlayerIndex, eastScoreChange[j], southScoreChange[j], westScoreChange[j], northScoreChange[j]); // Console.orig.log(str); } }else{ //否则都是自摸 // str += handleTsumo(kyoku, startPlayerIndex, eastScoreChange[j], southScoreChange[j], westScoreChange[j], northScoreChange[j]); } }//for if(str.length > 0) { const span = document.createElement("span"); span.innerHTML = ` \u00A0\u00A0\u00A0` + str; endInfoEle[i].parentElement.appendChild(span); let lineNum = getTextLineNum(span) if(lineNum > 1){ //如果新添加的文字有多行, 则进行对齐 for (let j = 1; j < lineNum; j++) { summaryEle.children[0].insertBefore(document.createElement("br"), kyokuEle[i].parentElement.nextElementSibling); } } } }else if(endMode == tenhouText.Ryuukyoku){ //荒牌流局 //流局, 如果有分数改变则处理 // let str = handleRyuukyoku(kyoku, startPlayerIndex, eastScoreChange[j], southScoreChange[j], westScoreChange[j], northScoreChange[j]); // Console.orig.log(str); }else if(endMode == tenhouText.RyuukyokuTsumo) { //流局满贯 (等同于自摸8000) // let str = handleRyuukyoku(kyoku, startPlayerIndex, eastScoreChange[j], southScoreChange[j], westScoreChange[j], northScoreChange[j]); // Console.orig.log(str); }else if(endMode == tenhouText.RyuukyokuType1) { //九种九牌 //流局, 如果有分数改变则处理 // let str = handleRyuukyoku(kyoku, startPlayerIndex, eastScoreChange[j], southScoreChange[j], westScoreChange[j], northScoreChange[j]); // Console.orig.log(str); }else{ //四风连打、四杠散了、四家立直 //三家和了 //流局, 如果有分数改变则处理 // let str = handleRyuukyoku(kyoku, startPlayerIndex, eastScoreChange[j], southScoreChange[j], westScoreChange[j], northScoreChange[j]); // Console.orig.log(str); } const span = document.createElement("span"); let str = getPlayerSeatNameByPlayerIndex(startPlayerIndex, kyoku, OUTSTYLE.A); span.innerText = ` \u00A0\u00A0\u00A0` + str; span.classList.add("position_ab"); span.classList.add("l-130px"); if(str == i18nText.seatTypeA0) //东起 span.classList.add("color5"); else span.classList.add("color6"); kyokuEle[i].parentElement.appendChild(span); titleKyokuEle[0].innerText += `\u00A0\u00A0` + str; // 将页面滚动导致的元素可见性改变与选择器绑定 //分别获取父元素的第一个子元素和最后一个子元素 //第一个子元素作为向下滚动时的监听对象, 最后一个子元素作为向上滚动时的监听对象 //这样无需去监听滚动事件,提高性能 titleKyokuEle[0].id = titleKyokuEle[0].href.match(/#[\S\s]+$/)[0]; //第一个子元素 isInViewPortByObserver(titleEle); //最后一个子元素 let lastEle = section[i].children[section[i].children.length -1]; isInViewPortByObserver(lastEle); }//for } //列出选择权重 static showChooseWeight() { /* 列出选择权重 */ const map = new Map(); //使用map保证重置循坏后的唯一性 const boxObjStr = '{"left":0,"top":0}'; const entry = document.getElementsByClassName("collapse entry"); for (let i = 0, length = entry.length; i != length; ++i) { entry[i].classList.add("position_re"); const roleEle = entry[i].getElementsByClassName("role"); let selfPai = roleEle[0].parentElement; let mortalPai = roleEle[1].nextElementSibling; if(mortalPai.tagName.toLocaleLowerCase() == 'details') { mortalPai = roleEle[1].nextSibling; } if (Object.prototype.toString.call(selfPai.childNodes[selfPai.childNodes.length -2]) == '[object SVGSVGElement]') { if(selfPai.childNodes[selfPai.childNodes.length -2].tagName.toLocaleLowerCase() == 'svg') { //如果有多张牌图片,就使用最后一张牌图片 selfPai = selfPai.childNodes[selfPai.childNodes.length -2]; } } if(mortalPai.nextElementSibling.tagName.toLocaleLowerCase() == 'svg') { //如果有多张牌图片,就使用最后一张牌图片 mortalPai = mortalPai.nextElementSibling; } const dataEle = entry[i].getElementsByTagName("tbody")[0].childNodes; var selfPaiData = 0; var mortalPaiData = 0; var selfBoxObj = JSON.parse(boxObjStr), mortalBoxObj = JSON.parse(boxObjStr); map.clear(); //清除map let j = 0, size = dataEle.length; while (j != size) { let selfPaiStr = null; let mortalPaiStr = null; let isResetLoop = false; //是否重置循坏 if(selfPai != null){ if(Object.prototype.toString.call(selfPai) == '[object Text]') { selfPaiStr = selfPai.data; }else{ let obj = selfPai.getElementsByClassName("face"); if(obj[0] != null){ selfPaiStr = obj[0].href.baseVal; }else{ //选择跳过的情况 selfPaiStr = selfPai.childNodes[selfPai.childNodes.length -1].data; } } } if(mortalPai != null){ if(Object.prototype.toString.call(mortalPai) == '[object Text]') { mortalPaiStr = mortalPai.data; }else{ let obj = mortalPai.getElementsByClassName("face"); if(obj[0] != null){ mortalPaiStr = obj[0].href.baseVal; }else{ //选择跳过的情况 mortalPaiStr = mortalPai.childNodes[mortalPai.childNodes.length -1].data; } } } let data = dataEle[j].childNodes[2].innerHTML.replace(/<.*?>/g, ""); //过滤html标签, 只保留文字内容 let obj1 = dataEle[j].childNodes[0].getElementsByClassName("face"); let dataPaiStr; if(obj1[obj1.length - 1] != null) { // 如果有多张牌, 则选择最后一张牌作为对比牌 (主要用于吃的情况、碰杠这些牌都是一样的) dataPaiStr = obj1[obj1.length - 1].href.baseVal; }else{ dataPaiStr = dataEle[j].childNodes[0].innerHTML; if(map.has(j) == false) { map.set(j, true); //保存当前j的值,防止重复开始循坏 j = 0; //如果 有 选择跳过的情况, 则重新开始循坏, 以找到正确的数据 isResetLoop = true; } } if(selfPaiStr == dataPaiStr) { //如果目标操作是自己的操作 selfPaiData = data; const span = document.createElement("span"); span.innerText = ` \u00A0\u00A0\u00A0` + data; span.classList.add("position_ab"); if(selfBoxObj.top == 0) { if(!isNaN(selfPai.offsetTop)) { selfBoxObj.top = (selfPai.offsetTop + selfPai.offsetHeight / 2 - 10); }else if(!isNaN(selfPai.parentElement.offsetTop)) { selfBoxObj.top = (selfPai.parentElement.offsetTop + selfPai.parentElement.offsetHeight / 2 - 10); }else{ console.error("[dom struct inconsistency] source:", "selfPai"); console.info("[debug]", `i: ${i}`); } span.style.top = selfBoxObj.top + 2 + "px"; } span.classList.add("l-170px"); entry[i].insertBefore(span, entry[i].childNodes[3].nextSibling); selfPai = null; //置null, 防止继续计算 }else if(mortalPaiStr == dataPaiStr) { //如果目标操作是Mortal的操作 mortalPaiData = data; const span = document.createElement("span"); span.innerText = ` \u00A0\u00A0\u00A0` + data; span.classList.add("position_ab"); if(mortalBoxObj.top == 0) { if(!isNaN(mortalPai.offsetTop)) { mortalBoxObj.top = (mortalPai.offsetTop + mortalPai.offsetHeight / 2 - 10); }else if(!isNaN(mortalPai.previousElementSibling.offsetTop)) { mortalBoxObj.top = (mortalPai.previousElementSibling.offsetTop + mortalPai.previousElementSibling.offsetHeight / 2 - 10); }else if(!isNaN(mortalPai.previousElementSibling.previousElementSibling.offsetTop)) { mortalBoxObj.top = (mortalPai.previousElementSibling.previousElementSibling.offsetTop + mortalPai.previousElementSibling.previousElementSibling.offsetHeight / 2 - 10); }else{ console.error("[dom struct inconsistency] source:", "mortalPai"); console.info("[debug]", `i: ${i}`); } span.style.top = mortalBoxObj.top + 1 + "px"; } span.classList.add("l-170px"); if(Object.prototype.toString.call(mortalPai.nextSibling) == '[object Text]') { //如果有多张牌图片,就使用最后一张牌图片后面的文字的位置 mortalPai = mortalPai.nextSibling; } entry[i].insertBefore(span, mortalPai.nextSibling); mortalPai = null; //置null, 防止继续计算 } if(selfPaiStr == mortalPaiStr) { //如果自己选择打出的牌与mortal选择打出的牌相同 if(map.has(j) == false) { map.set(j, true); //保存当前j的值,防止重复开始循坏 j = 0; //如果 有 选择跳过的情况, 则重新开始循坏, 以找到正确的数据 isResetLoop = true; } } if(selfPai == null && mortalPai == null) { //是否处理完毕 break; //跳出循坏 } if(isResetLoop == false){ //不重置循坏时, index++ ++j; } }//for /* 计算自己的选择与mortal选择的差值 */ const defaultHandleFunc = (newNode, index, colorStr) => { newNode.style.color = colorStr; //设置为目标颜色 newNode.innerHTML = ` \u00A0\u00A0\u00A0` + eval("i18nText.badMoveDiffer" + index) + differData; } const differData = Math.constructor.roundEx(Math.abs(mortalPaiData - selfPaiData), 5); //保留5位小数 if (differData != 0) { //忽略自己和mortal打出的牌一样的结果 const turnInfo = entry[i].children[0]; const newNode = document.createElement("span"); newNode.classList.add("font_weight_400"); if (differData < 5) { //微差 defaultHandleFunc(newNode, 1, "#000"); //黑色 }else if (differData < 10) { //小幅差距 defaultHandleFunc(newNode, 2, "#996633"); //褐色 }else if (differData < 20) { //低等差距 defaultHandleFunc(newNode, 3, "#009966"); //淡绿 }else if (differData < 40) { //中等差距 defaultHandleFunc(newNode, 4, "#3399FF"); //淡蓝 }else if (differData < 60) { //高等差距 defaultHandleFunc(newNode, 5, "#3333CC"); //深蓝 }else if (differData < 80) { //大幅度差距 defaultHandleFunc(newNode, 6, "#CC0099"); //淡红 }else{ //压倒性差距 defaultHandleFunc(newNode, 7, "#f00"); //红色 } turnInfo.appendChild(newNode); } }//for } } class MainApp extends MortalBase { static perfor = new Performance("MainApp"); constructor(){ //super(); } static init(){ if(Debug.getDebug){ CodeTemplate.autoWired_Performance("MainApp.run", MainApp); } return true; } static run(){ MainApp.createCatalogUI(); } static stop(){ } static clean(){ } // static createCatalogUI() { const summaryEle = document.getElementsByClassName("kyoku-toc")[0]; //数据处理 let catalogUI = document.getElementById("catalogUI"); layui.use(function(){ var element = layui.element; var laytpl = layui.laytpl; var $ = layui.$; //收集数据1-对局 let strArray1 = []; let array = summaryEle.children[0].children; let length = summaryEle.children[0].children.length; for (let j = 0; j < length; j++) { const ele = array[j]; strArray1.push({ name: ele.children[0].innerText + " " + ele.children[1].innerText.trim(), href: ele.children[0].href.match(/#[\S\s]+$/)[0] }); } let data = { //模板数据 selector1: strArray1, //对局选择器 selector2: strArray2, //不一致选择器 selector3: strArray3, //恶手选择器 }; //使用模板进行解析 var compile = laytpl(catalogUITemplate); // 模板解析 compile.render(data, (htmlStr)=>{ $('#catalogUI').html(htmlStr); }); // 模板渲染 // 渲染导航组件 element.render('nav', 'selector-filter-nav'); /* 新增右置对局选择器和切换器(固定/可滑动呼出) */ let catalogUIBuf = document.getElementById("catalogUIBuf"); let selectorGroups = document.getElementById("selectorGroups"); let eleSection = document.getElementsByTagName("section")[0]; selectorGroups.style.height = (document.documentElement.clientHeight - 100 - catalogUIBuf.offsetHeight) + "px"; //滚动条 catalogUI.style.left = (eleSection.offsetLeft + eleSection.offsetWidth + 60) + "px"; //x位置 let vTop = (eleSection.offsetTop - catalogUI.offsetHeight); catalogUI.style.top = 60 + "px"; //y位置 document.getElementById("kyoku_prev").addEventListener("click", function(e){ let nextEle = document.getElementsByClassName("ui layui-this")[0].previousElementSibling; if(nextEle != null){ nextEle = nextEle.children[0]; setNewActivateEle([nextEle]); nextEle.click(); } }, false); document.getElementById("kyoku_next").addEventListener("click", function(e){ let nextEle = document.getElementsByClassName("ui layui-this")[0].nextElementSibling; if(nextEle != null){ nextEle = nextEle.children[0]; setNewActivateEle([nextEle]); nextEle.click(); } }, false); document.getElementById("diff_prev").addEventListener("click", function(e){ }, false); document.getElementById("diff_next").addEventListener("click", function(e){ }, false); }); } // static createSettingUI() { } } const app = new App(); })(unsafeWindow);
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址