HWM_UnitsAssessment

Оценка силы существ в расчёте на лидерство/стоимость

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==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);};
    }
})();