// ==UserScript== // @name PT站点魔力计算器 // @namespace https://github.com/neoblackxt/PTMyBonusCalc // @version 2.0.9 // @description 在使用NexusPHP架构的PT站点显示每个种子的A值和每GB的A值。 // @author neoblackxt, LaneLau // @require https://cdn.jsdelivr.net/npm/jquery@3/dist/jquery.min.js // @require https://cdn.jsdelivr.net/npm/echarts@5/dist/echarts.min.js // @match *://*.beitai.pt/torrents* // @match *://*.pttime.org/torrents* // @match *://*.ptsbao.club/torrents* // @match *://*.pthome.net/torrents* // @match *://kp.m-team.cc/* // @match *://zp.m-team.io/* // @match *://*.hddolby.com/torrents* // @match *://*.leaguehd.com/torrents* // @match *://*.hdhome.org/torrents* // @match *://*.hdsky.me/torrents* // @match *://*.ourbits.club/torrents* // @match *://*.u2.dmhy.org/torrents* // @match *://*.hdzone.me/torrents* // @match *://*.hdatmos.club/torrents* // @match *://*.pt.soulvoice.club/torrents* // @match *://*.pt.soulvoice.club/live* // @match *://*.discfan.net/torrents* // @match *://*.hdtime.org/torrents* // @match *://*.nicept.net/torrents* // @match *://*.pterclub.com/torrents* // @match *://*.hdarea.co/torrents* // @match *://*.hdfans.org/torrents* // @match *://pt.btschool.club/torrents* // @match *://*.1ptba.com/torrents* // @match *://www.oshen.win/torrents* // @match *://*.rousi.zip/torrents* // @match *://*.kufei.org/torrents* // @match *://*.tjupt.org/torrents* // @match *://*.tjupt.org/bonus* // @match *://*/mybonus* // @license GPL License // @grant GM_setValue // @grant GM_getValue // @grant window.onurlchange // @downloadURL https://update.greasyfork.cloud/scripts/439369/PT%E7%AB%99%E7%82%B9%E9%AD%94%E5%8A%9B%E8%AE%A1%E7%AE%97%E5%99%A8.user.js // @updateURL https://update.greasyfork.cloud/scripts/439369/PT%E7%AB%99%E7%82%B9%E9%AD%94%E5%8A%9B%E8%AE%A1%E7%AE%97%E5%99%A8.meta.js // ==/UserScript== function run() { var $ = jQuery; let argsReady = true; let T0 = GM_getValue(host + ".T0"); let N0 = GM_getValue(host + ".N0"); let B0 = GM_getValue(host + ".B0"); let L = GM_getValue(host + ".L"); if (!(T0 && N0 && B0 && L)) { argsReady = false if (!isMybonusPage) { alert("未找到魔力值参数,请打开魔力值系统说明获取(/mybonus)"); } } if (isMybonusPage) { T0 = parseInt($("li:has(b:contains('T0'))")[1].innerText.split(" = ")[1]); N0 = parseInt($("li:has(b:contains('N0'))")[1].innerText.split(" = ")[1]); B0 = parseInt($("li:has(b:contains('B0'))")[1].innerText.split(" = ")[1]); L = parseInt($("li:has(b:contains('L'))")[1].innerText.split(" = ")[1]); GM_setValue(host + ".T0", T0); GM_setValue(host + ".N0", N0); GM_setValue(host + ".B0", B0); GM_setValue(host + ".L", L); if (!argsReady) { if (T0 && N0 && B0 && L) { alert("魔力值参数已更新") } else { alert("魔力值参数获取失败") } } function calcB(A) { return B0 * (2 / Math.PI) * Math.atan(A / L) } function calcAbyB(B) { //从B值反推A值 return Math.tan(B / B0 / (2 / Math.PI)) * L } let A = isMTeam ? 0 : parseFloat($("div:contains(' (A = ')")[0].innerText.split(" = ")[1]); let B = isMTeam ? parseFloat($("td:contains('基本獎勵')+td+td")[0].innerText) : calcB(A); // 剔除M-Team的基本奖励中做种数奖励 if (isMTeam) { let matches = $("h5:contains('做種每小時將得到如下的魔力值')").next().children().first().text() .match(/(\d+(\.\d+)?)個魔力值.*最多計(\d+)個/); let seedingBonusPerSeed = parseFloat(matches[1]); let seedingBonusLimit = parseInt(matches[3]); let currentSeedingNode = $("span:contains('當前活動')").parent().clone(); currentSeedingNode.find('img').replaceWith(function () { return "img"; }); let currentSeeding = parseInt(currentSeedingNode.text().match(/(\d+)/)[1]); B = B - (currentSeeding > seedingBonusLimit ? seedingBonusPerSeed * seedingBonusLimit : seedingBonusPerSeed * currentSeeding); } // 对于M-Team,B>B0是因为网页获取的基本奖励包括了做种数的奖励,上面代码已经进行排除。 // 其他站暂不明确是否有该问题,下面一行的代码暂时保留 B = B >= B0 ? B0 * 0.98 : B if (isMTeam) { A = calcAbyB(B); } let spot = [A, B] let data = [] for (let i = 0; i < (1.1 * A > 25 * L ? 1.1 * A : 25 * L); i = i + L / 4) { data.push([i, calcB(i)]) } let insertPos = isMTeam ? $("ul+table") : $("table+h1") insertPos.before('
') var myChart = echarts.init(document.getElementById('main')); // 指定图表的配置项和数据 var option = { title: { text: 'B - A 图', top: 'bottom', left: 'center' }, tooltip: { trigger: 'axis', axisPointer: { type: 'cross' }, backgroundColor: 'rgba(255, 255, 255, 0.8)', position: function (pos, params, el, elRect, size) { var obj = {top: 10}; obj[['left', 'right'][+(pos[0] < size.viewSize[0] / 2)]] = 30; return obj; }, extraCssText: 'width: 170px' }, xAxis: { name: 'A', }, yAxis: { name: 'B' }, axisPointer: { label: { backgroundColor: '#777' } }, series: [ { type: 'line', data: data, symbol: 'none' }, { type: 'line', data: [spot], symbolSize: 6 } ] }; // 使用刚指定的配置项和数据显示图表。 myChart.setOption(option); } function calcA(T, S, N) { var c1 = 1 - Math.pow(10, -(T / T0)); // 当断种时,显示续种后的实际值,因为当前状态值无意义 N = N ? N : 1; // 当前状态值,加入做种后实际值会小于当前值 // TODO: 改为双行显示为当前值和实际值 var c2 = 1 + Math.pow(2, .5) * Math.pow(10, -(N - 1) / (N0 - 1)); return c1 * S * c2; } function makeA($this, i_T, i_S, i_N) { var time = $this.children('td:eq(' + i_T + ')').find("span").attr("title"); // 适配m-team的发生时间 if (time == undefined || time == "") { time = $this.children('td:eq(' + i_T + ')').find("span").text(); } // 适配tjupt的发生时间 if (time == undefined || time == "") { time = $this.children('td:eq(' + i_T + ')').html().replace("