Greasy Fork is available in English.
Оценка силы существ в расчёте на лидерство/стоимость
// ==UserScript== // @name HWM_UnitsAssessment // @namespace Небылица // @version 1.1 // @description Оценка силы существ в расчёте на лидерство/стоимость // @author Небылица // @include /^https{0,1}:\/\/((www|qrator)\.heroeswm\.ru|178\.248\.235\.15)\/(army_info|leader_army)\.php/ // @grant GM_setValue // @grant GM_getValue // ==/UserScript== (function() { "use strict"; // Настройки – вес стата const statWeight = 0.033; // // Вспомогательные функции Number.prototype.roundTo = function(digits){ // Oкругление числа до digits знаков после запятой return Math.round(this*(10**digits))/(10**digits); } function insertAfter(newNode, referenceNode){ // Вставка newNode после referenceNode referenceNode.parentNode.insertBefore(newNode, referenceNode.nextSibling); } function assess(){ // Расчёт и вывод показателей для текущего существа var params = document.querySelectorAll(".scroll_content_half > div"), paramsNumbers = (LGArmyPage) ? [0, 2, 4, 6, 10, 11] : [0, 2, 4, 6, 7, 9], // номера нужных граф на странице ГЛ и на странице существа attack = parseInt(params[paramsNumbers[0]].innerText), defence = parseInt(params[paramsNumbers[1]].innerText), damageArr = params[paramsNumbers[2]].innerText.split("-"), damageMin = parseInt(damageArr[0]), damageMax = parseInt(damageArr[1]), damageBonus = (!LGArmyPage && GM_getValue("BFEnabled")) ? 1 : 0, // учёт урона с ББ на странице отряда при наличии соотв. настройки damageAvg = (damageMin+damageMax)/2 + damageBonus, HPBonus = (!LGArmyPage && GM_getValue("vitalityEnabled")) ? 2 : 0, // учёт ХП со стойкостью на странице отряда при наличии соотв. настройки HP = parseInt(params[paramsNumbers[3]].innerText) + HPBonus, ini = parseInt(params[paramsNumbers[4]].innerText), leadershipText = params[paramsNumbers[5]].innerText.replace(",", "") || params[paramsNumbers[5]].firstChild.value, // лидерство без и с полем ввода своей цены leadership = parseInt(leadershipText) || "?", // 1000 – нормировочная константа для получения удобного для восприятия порядка величин damage = 1000*(damageAvg*(ini/10)*(1+statWeight*attack)), defenceWeight = (!LGArmyPage && GM_getValue("calculateWithoutDef")) ? 0 : statWeight, // расчёт без учёта дефа на странице отряда при наличии соотв. настройки vitality = 1000*(HP*(1+defenceWeight*defence)), damagePer = (damage/leadership).roundTo(2) || "?", vitalityPer = (vitality/leadership).roundTo(2) || "?", strength = Math.sqrt(damagePer*vitalityPer).roundTo(2) || "?", ECE = (100*(damage/vitality)).roundTo(2); damagePerInnerDiv.innerText = damagePer; vitalityPerInnerDiv.innerText = vitalityPer; strengthInnerDiv.innerText = strength; ECEInnerDiv.innerText = ECE; } function setupReassessment(){ // Привязка перерасчёта параметров к смене юнита на странице ГЛ var unitNameDiv = document.getElementById("unit_name"), observer = new MutationObserver(function(mutations){ mutations.forEach(function(mutation){ assess(); }); }), config = {characterData: false, attributes: false, childList: true, subtree: false}; observer.observe(unitNameDiv, config); } function setLeadershipTitle(){ // Замена "Лидерства" на "Стоимость" в зависимости от того, выбран ли ручной ввод leadershipTitle = (customPriceCheckbox.checked) ? "Стоимость" : "Лидерство"; leadershipTitleNode.nodeValue = leadershipTitle; } function setupLeadershipNumberElement(){ // Отрисовывает элемент со стоимостью стека и вешает событие по изменению кастомной цены в зависимости от сохранённой настройки if (GM_getValue("customPriceEnabled")){ leadershipNumber.innerHTML = "<input type='text' value='" + oldLeadership + "'>"; var leadershipNumberInput = leadershipNumber.firstChild; leadershipNumberInput.setAttribute("id", "unitsAssessmentLeadershipNumberInput"); leadershipNumberInput.style = "width: 50px;" + "height: 19px;" + "background-image: linear-gradient(rgb(166, 166, 166), rgb(163, 163, 163));" + "border: 0px;" + "text-align: right;" + "font-size: 15.6px;" + "font-weight: bold;" + "font-family: verdana, geneva, arial cyr;" + "color: #000000;"; // вешаем событие по изменению кастомной цены leadershipNumberInput.oninput = function(){ if (leadershipNumberInput.value.match(/^\d+$/)){ // если лидерство – число, то считаем показатели assess(); } }; } else{ leadershipNumber.innerHTML = oldLeadership; } } function imitateCheck(checkbox){ // Имитация нажатия на чекбокс checkbox.checked = !checkbox.checked; checkbox.dispatchEvent(changeEvent); } // const LGArmyPage = location.href.match("/leader_army"), emptyImageSrc = ""; var damagePerDiv = document.createElement("div"), vitalityPerDiv = document.createElement("div"), strengthDiv = document.createElement("div"), ECEDiv = document.createElement("div"), armyInfo = document.getElementsByClassName("info_text_content")[0]; damagePerDiv.setAttribute("class", "scroll_content_half"); vitalityPerDiv.setAttribute("class", "scroll_content_half"); strengthDiv.setAttribute("class", "scroll_content_half"); ECEDiv.setAttribute("class", "scroll_content_half"); var rightColPaddingLeft = (LGArmyPage) ? "0.45em" : "0.3em"; vitalityPerDiv.style.paddingLeft = rightColPaddingLeft; ECEDiv.style.paddingLeft = rightColPaddingLeft; damagePerDiv.innerHTML = "<img src='" + emptyImageSrc + "' width='48' height='48'>Дамажность<div id='unitsAssessmentDamagePerInnerDiv'></div>"; vitalityPerDiv.innerHTML = "<img src='" + emptyImageSrc + "' width='48' height='48'>Живучесть<div id='unitsAssessmentVitalityPerInnerDiv'></div>"; strengthDiv.innerHTML = "<img src='" + emptyImageSrc + "' width='48' height='48'>Сила<div id='unitsAssessmentStrengthInnerDiv'></div>"; ECEDiv.innerHTML = "<img src='" + emptyImageSrc + "' width='48' height='48'>КПД<div id='unitsAssessmentECEInnerDiv'></div>"; damagePerDiv.title = "Чем Дамажность больше, тем больше ДПС (дамага на время) приходится на единицу лидерства. " + "Учитываются урон, статы атаки и инициатива (как модификатор при расчёте ДПС, без учёта возможного эффекта от права первого удара). " + "Значение ориентировочное, абилки НЕ учитываются. Рассчитано с весом стата = " + statWeight + ". Совокупность Дамажности и Живучести составляет Силу. Отношение этих двух показателей – КПД."; vitalityPerDiv.title = "Чем Живучесть больше, тем больше мяса приходится на единицу лидерства. Учитываются здоровье и статы защиты. " + "Значение ориентировочное, абилки НЕ учитываются. Рассчитано с весом стата = " + statWeight + ". Совокупность Дамажности и Живучести составляет Силу. Отношение этих двух показателей – КПД."; strengthDiv.title = "Чем Сила больше, тем больше ДПС и мяса приходится на единицу лидерства (и тем лучше). " + "Учитываются урон, здоровье, статы атаки и защиты, инициатива (как модификатор при расчёте ДПС, без учёта возможного эффекта от права первого удара). " + "Значение ориентировочное, абилки НЕ учитываются. Рассчитано с весом стата = " + statWeight + ". Увеличение/уменьшение лидерства в n раз = синхронное уменьшение/увеличение и урона, и ХП в n раз = уменьшение/увеличение Силы в n раз. " + "При досчитывании показателя отдельно по 1 составляющей (для учёта абилок в каждом конкретном сценарии их реализации) " + "надо брать корень из модификатора разницы (урон вдвое выше – Сила в 1.41 раза выше)."; ECEDiv.title = "Сколько процентов составляет урон от мяса (актуально не только для ГЛ, лидерство не влияет). " + "Чем он больше, тем больше ДПС приходится на единицу ХП (юнит более дамажный и тонкий); чем меньше – тем более мясной. " + "Могут быть оптимальны различные показатели в зависимости от стиля игры. " + "В общем случае желательна близость всех боевых юнитов в наборе по КПД (за исключением коробочной связки). " + "Рассчитано с весом стата = " + statWeight + "."; armyInfo.appendChild(damagePerDiv); armyInfo.appendChild(vitalityPerDiv); armyInfo.appendChild(strengthDiv); armyInfo.appendChild(ECEDiv); var damagePerInnerDiv = document.getElementById("unitsAssessmentDamagePerInnerDiv"), vitalityPerInnerDiv = document.getElementById("unitsAssessmentVitalityPerInnerDiv"), strengthInnerDiv = document.getElementById("unitsAssessmentStrengthInnerDiv"), ECEInnerDiv = document.getElementById("unitsAssessmentECEInnerDiv"); // активируем расчёт в зависимости от страницы if (LGArmyPage){ setupReassessment(); } else{ assess(); // рисуем и вставляем чекбоксы с настройками var frameDiv = document.getElementsByClassName("army_info")[0], showArmyDiv = document.getElementById("show_army"), settingsDiv = document.createElement("div"); if (frameDiv.clientHeight < 410){frameDiv.style.height = "410px";} // растягиваем поле с параметрами при недостаточной высоте settingsDiv.setAttribute("id", "unitsAssessmentSettingsDiv"); settingsDiv.innerHTML = "<style>" + "#unitsAssessmentSettingsDiv{" + "width: 300px;" + "height: 40px;" + "margin-left: 656px;" + "font-size: 10pt;" + "font-weight: bold;" + "color: #272727;" + "}" + "#unitsAssessmentSettingsCustomPriceSpan, #unitsAssessmentSettingsCalculateWithoutDefSpan," + "#unitsAssessmentSettingsBFSpan, #unitsAssessmentSettingsVitalitySpan{" + "display: inline-block;" + "width: 48%;" + "margin: 2px 0px 0px 4px;" + "}" + "#unitsAssessmentSettingsCustomPriceTitle, #unitsAssessmentSettingsCalculateWithoutDefTitle," + "#unitsAssessmentSettingsBFTitle, #unitsAssessmentSettingsVitalityTitle{" + "display: inline-block;" + "width: 87%;" + "text-align: left;" + "}" + "#unitsAssessmentSettingsCustomPriceCheckboxSpan, #unitsAssessmentSettingsCalculateWithoutDefCheckboxSpan," + "#unitsAssessmentSettingsBFCheckboxSpan, #unitsAssessmentSettingsVitalityCheckboxSpan{" + "display: inline-block;" + "width: 13%;" + "text-align: right;" + "}" + "</style>" + "<span id='unitsAssessmentSettingsCustomPriceSpan'><span id='unitsAssessmentSettingsCustomPriceTitle'>Ручной ввод</span><span id='unitsAssessmentSettingsCustomPriceCheckboxSpan'><input type='checkbox' id='unitsAssessmentSettingsCustomPriceCheckbox'></span></span>" + "<span id='unitsAssessmentSettingsCalculateWithoutDefSpan'><span id='unitsAssessmentSettingsCalculateWithoutDefTitle'>Без учёта дефа</span><span id='unitsAssessmentSettingsCalculateWithoutDefCheckboxSpan'><input type='checkbox' id='unitsAssessmentSettingsCalculateWithoutDefCheckbox'></span></span><br>" + "<span id='unitsAssessmentSettingsBFSpan'><span id='unitsAssessmentSettingsBFTitle'>Боевое безумие</span><span id='unitsAssessmentSettingsBFCheckboxSpan'><input type='checkbox' id='unitsAssessmentSettingsBFCheckbox'></span></span>" + "<span id='unitsAssessmentSettingsVitalitySpan'><span id='unitsAssessmentSettingsVitalityTitle'>Стойкость</span><span id='unitsAssessmentSettingsVitalityCheckboxSpan'><input type='checkbox' id='unitsAssessmentSettingsVitalityCheckbox'></span></span>"; insertAfter(settingsDiv, showArmyDiv); // задаём переменные для созданных элементов var customPriceSpan = document.getElementById("unitsAssessmentSettingsCustomPriceSpan"), calculateWithoutDefSpan = document.getElementById("unitsAssessmentSettingsCalculateWithoutDefSpan"), BFSpan = document.getElementById("unitsAssessmentSettingsBFSpan"), vitalitySpan = document.getElementById("unitsAssessmentSettingsVitalitySpan"), customPriceTitle = document.getElementById("unitsAssessmentSettingsCustomPriceTitle"), calculateWithoutDefTitle = document.getElementById("unitsAssessmentSettingsCalculateWithoutDefTitle"), BFTitle = document.getElementById("unitsAssessmentSettingsBFTitle"), vitalityTitle = document.getElementById("unitsAssessmentSettingsVitalityTitle"), customPriceCheckbox = document.getElementById("unitsAssessmentSettingsCustomPriceCheckbox"), calculateWithoutDefCheckbox = document.getElementById("unitsAssessmentSettingsCalculateWithoutDefCheckbox"), BFCheckbox = document.getElementById("unitsAssessmentSettingsBFCheckbox"), vitalityCheckbox = document.getElementById("unitsAssessmentSettingsVitalityCheckbox"), RBCorner = document.querySelector(".corner_rb"); // и для уголка customPriceSpan.title = "Ручной ввод стоимости (для сравнения юнитов по текущей цене в ивенте, а не по лидерству в рамках ГЛ)."; calculateWithoutDefSpan.title = "Расчёт показателей с нулевым весом дефа (против хаоситов)."; BFSpan.title = "Расчёт показателей с учётом взятия Боевого безумия."; vitalitySpan.title = "Расчёт показателей с учётом взятия Стойкости."; RBCorner.title = vitalitySpan.title; // ставим галочки при наличии сохранённых положительных значений customPriceCheckbox.checked = GM_getValue("customPriceEnabled"); calculateWithoutDefCheckbox.checked = GM_getValue("calculateWithoutDef"); BFCheckbox.checked = GM_getValue("BFEnabled"); vitalityCheckbox.checked = GM_getValue("vitalityEnabled"); // сохраняем реальное лидерство var leadershipDiv = armyInfo.children[9], leadershipTitleNode = leadershipDiv.firstChild.nextSibling.nextSibling, leadershipTitle, leadershipNumber = leadershipDiv.children[1], oldLeadership = leadershipNumber.innerText.replace(",", ""); // выставляем подпись лидерству, отрисовываем поле ввода цены в соответствии с сохранёнными настройками setLeadershipTitle(); setupLeadershipNumberElement(); // вешаем события по изменению чекбоксов customPriceCheckbox.onchange = function(){ GM_setValue("customPriceEnabled", customPriceCheckbox.checked); setLeadershipTitle(); setupLeadershipNumberElement(); if (!customPriceCheckbox.checked){ // при снятии чекбокса пересчитываем показатели (уже по старому лидерству) assess(); } }; calculateWithoutDefCheckbox.onchange = function(){ GM_setValue("calculateWithoutDef", calculateWithoutDefCheckbox.checked); assess(); }; BFCheckbox.onchange = function(){ GM_setValue("BFEnabled", BFCheckbox.checked); assess(); }; vitalityCheckbox.onchange = function(){ GM_setValue("vitalityEnabled", vitalityCheckbox.checked); assess(); }; // имитируем нажатия на чекбоксы по клику на заголовках (для стойкости – и на НП уголке) var changeEvent = new Event("change"); customPriceTitle.onclick = function(){imitateCheck(customPriceCheckbox);}; calculateWithoutDefTitle.onclick = function(){imitateCheck(calculateWithoutDefCheckbox);}; BFTitle.onclick = function(){imitateCheck(BFCheckbox);}; vitalityTitle.onclick = function(){imitateCheck(vitalityCheckbox);}; RBCorner.onclick = function(){imitateCheck(vitalityCheckbox);}; } })();