TravianHelper(Travian Legends++)

Villages, Troops, Resources Overview, Quick links. Based on Travian Legends++.

// ==UserScript==
// @name        TravianHelper(Travian Legends++)
// @namespace   https://greasyfork.org/
// @author      bingkx & SkillsBoY
// @version     0.1.0
// @license     GPLv3
// @description Villages, Troops, Resources Overview, Quick links. Based on Travian Legends++.
// @include     *://*.travian.*
// @include     *://*/*.travian.*
// @exclude     *://*.travian*.*/hilfe.php*
// @exclude     *://*.travian*.*/log*.php*
// @exclude     *://*.travian*.*/index.php*
// @exclude     *://*.travian*.*/anleitung.php*
// @exclude     *://*.travian*.*/impressum.php*
// @exclude     *://*.travian*.*/anmelden.php*
// @exclude     *://*.travian*.*/gutscheine.php*
// @exclude     *://*.travian*.*/spielregeln.php*
// @exclude     *://*.travian*.*/links.php*
// @exclude     *://*.travian*.*/geschichte.php*
// @exclude     *://*.travian*.*/tutorial.php*
// @exclude     *://*.travian*.*/manual.php*
// @exclude     *://*.travian*.*/ajax.php*
// @exclude     *://*.travian*.*/ad/*
// @exclude     *://*.travian*.*/chat/*
// @exclude     *://forum.travian*.*
// @exclude     *://board.travian*.*
// @exclude     *://shop.travian*.*
// @exclude     *://*.travian*.*/activate.php*
// @exclude     *://*.travian*.*/support.php*
// @exclude     *://help.travian*.*
// @exclude     *://*.answers.travian*.*
// @exclude     *.css
// @exclude     *.js
// @grant       GM_addStyle
// @grant       GM_info
// ==/UserScript==

const myVersion = GM_info.script.version;


// inject css
{
    GM_addStyle(`
        .us-draggable {
            position: absolute;
            z-index: 9;
            background-color: #f1f1f1;
            border: 1px solid #d3d3d3;
            text-align: center !important;
            white-space: nowrap;
        }

        .us-draggable .us-draggable--header {
            padding: 10px;
            cursor: move;
            z-index: 10;
            background-color: #2196f3;
            color: #fff;
        }

        .us-draggable li {
            text-align: left;
        }

        .us-draggable ul {
            display: flex;
            flex-direction: column;
            justify-content: center;
            margin: 0 5px;
            padding: 0 0 0 16px;
        }

        .us-draggable td > a {
            display: block;
            padding: 14px;
        }

        .us-draggable td {
            padding: unset;
        }

        .us-draggable li.done {
            background-color: green;
        }

        .us-barBox {
            width: 50px;
            height: 7px;
            background-color: #52372a;
            margin: 0 2px;
        }

        .us-resources {
            display: flex;
            flex-direction: row;
            margin: 5px;
        }

        .us-bar {
            background-color: #699e32;
            height: 100%;
        }

        .us-resourceValue {
            font-size: 11px;
            font-weight: 700;
            text-align: end;
            margin-right: 2px;
        }

        .us--text-alert {
            color: #0022af;
        }

        .us-map-open {
            position: fixed !important;
            inset: 0 !important;
            width: unset !important;
            height: unset !important;
        }

        .us-alertFill--yellow {
            fill: yellow;    
        }
    
        .us--display-block {
            display: block !important;
        }

        .us-settingsIconSVG > svg:hover {
            fill: whitesmoke;
        }
    `);
}

// travian utilities
{
    /**
     * Makes the building icons on the right over the villages available.
     * (same functionality as travian plus *and more)
     */
    function makeBuildingIconsAvailable() {
        // marketplace
        {
            const marketIcon = document.querySelector("#sidebarBoxActiveVillage > div.header > div").childNodes[1];
            const marketIconClone = marketIcon.cloneNode(true);
            marketIconClone.classList.remove("gold");
            marketIconClone.classList.add("green");
            marketIconClone.removeAttribute("onClick");
            marketIconClone.title = "Marketplace||";
            marketIconClone.classList.contains("disabled") ? marketIconClone.href = "#" : marketIconClone.href = "/build.php?gid=17";
            marketIcon.replaceWith(marketIconClone);
        }

        // barracks
        {
            const barracksIcon = document.querySelector("#sidebarBoxActiveVillage > div.header > div").childNodes[5];
            const barracksIconClone = barracksIcon.cloneNode(true);
            barracksIconClone.classList.remove("gold");
            barracksIconClone.classList.add("green");
            barracksIconClone.removeAttribute("onClick");
            barracksIconClone.title = "Barracks||";
            barracksIconClone.classList.contains("disabled") ? barracksIconClone.href = "#" : barracksIconClone.href = "/build.php?gid=19";
            barracksIcon.replaceWith(barracksIconClone);
        }

        // stable
        {
            const stableIcon = document.querySelector("#sidebarBoxActiveVillage > div.header > div").childNodes[9];
            const stableIconClone = stableIcon.cloneNode(true);
            stableIconClone.classList.remove("gold");
            stableIconClone.classList.add("green");
            stableIconClone.removeAttribute("onClick");
            stableIconClone.title = "Stable||";
            stableIconClone.classList.contains("disabled") ? stableIconClone.href = "#" : stableIconClone.href = "/build.php?gid=20";
            stableIcon.replaceWith(stableIconClone);
        }

        // workshop
        {
            const workshopIcon = document.querySelector("#sidebarBoxActiveVillage > div.header > div").childNodes[13];
            const workshopIconClone = workshopIcon.cloneNode(true);
            workshopIconClone.classList.remove("gold");
            workshopIconClone.classList.add("green");
            workshopIconClone.removeAttribute("onClick");
            workshopIconClone.title = "Workshop||";
            workshopIconClone.classList.contains("disabled") ? workshopIconClone.href = "#" : workshopIconClone.href = "/build.php?gid=21";
            workshopIcon.replaceWith(workshopIconClone);
        }

        const villageList = document.querySelector("#sidebarBoxVillagelist > div.header > div");
        const villageStatistics = document.querySelector("#sidebarBoxVillagelist > div.header > div").childNodes[1];

        // rally point
        {
            const villageStatisticsClone = villageStatistics.cloneNode(true);
            const villageStatisticsCloneSVG = villageStatisticsClone.querySelector("svg");
            const img = document.createElement("img");
            img.src = "";
            img.style = "position: absolute; z-index: 3; height: 37px;";
            villageStatisticsCloneSVG.replaceWith(img);
            villageStatisticsClone.classList.remove("gold");
            villageStatisticsClone.classList.add("green");
            villageStatisticsClone.removeAttribute("onClick");
            villageStatisticsClone.removeAttribute("id");
            villageStatisticsClone.title = "Rally Point||";
            villageStatisticsClone.href = "/build.php?id=39&gid=16";
            villageList.insertBefore(villageStatisticsClone, villageStatistics);
        }

        // academy
        {
            const villageStatisticsClone = villageStatistics.cloneNode(true);
            const villageStatisticsCloneSVG = villageStatisticsClone.querySelector("svg");
            const img = document.createElement("img");
            img.src = "";
            img.style = "position: absolute; z-index: 3; height: 37px;";
            villageStatisticsCloneSVG.replaceWith(img);
            villageStatisticsClone.classList.remove("gold");
            villageStatisticsClone.classList.add("green");
            villageStatisticsClone.removeAttribute("onClick");
            villageStatisticsClone.removeAttribute("id");
            villageStatisticsClone.title = "Academy||";
            villageStatisticsClone.href = "/build.php?gid=22";
            villageList.insertBefore(villageStatisticsClone, villageStatistics);
        }

        // smithy
        {
            const villageStatisticsClone = villageStatistics.cloneNode(true);
            const villageStatisticsCloneSVG = villageStatisticsClone.querySelector("svg");
            const img = document.createElement("img");
            img.src = "";
            img.style = "position: absolute; z-index: 3; height: 37px;";
            villageStatisticsCloneSVG.replaceWith(img);
            villageStatisticsClone.classList.remove("gold");
            villageStatisticsClone.classList.add("green");
            villageStatisticsClone.removeAttribute("onClick");
            villageStatisticsClone.removeAttribute("id");
            villageStatisticsClone.title = "Smithy||";
            villageStatisticsClone.href = "/build.php?gid=13";
            villageList.insertBefore(villageStatisticsClone, villageStatistics);
        }

        // town hall
        {
            const villageStatisticsClone = villageStatistics.cloneNode(true);
            const villageStatisticsCloneSVG = villageStatisticsClone.querySelector("svg");
            const img = document.createElement("img");
            img.src = "";
            img.style = "position: absolute; z-index: 3; height: 37px;";
            villageStatisticsCloneSVG.replaceWith(img);
            villageStatisticsClone.classList.remove("gold");
            villageStatisticsClone.classList.add("green");
            villageStatisticsClone.removeAttribute("onClick");
            villageStatisticsClone.removeAttribute("id");
            villageStatisticsClone.title = "Town Hall||";
            villageStatisticsClone.href = "/build.php?gid=24";
            villageList.insertBefore(villageStatisticsClone, villageStatistics);
        }

    }

    /**
     * On coordinates click, this will fill in the village destination fields.
     * (same functionality as travian plus)
     */
    function coordsClickFillVillageDestinationFields() {
        document.querySelector("#sidebarBoxVillagelist > div.content > div.villageList").classList.add("shortcutsEnabled");

        const notActiveVillageList = document.querySelectorAll("#sidebarBoxVillagelist > div.content > div.villageList > div.dropContainer > div.listEntry:not(.active)");

        notActiveVillageList.forEach((village) => {
            village.querySelector(".coordinatesGrid").onclick = function () {
                const villageNameInput = document.querySelector("#enterVillageName");
                const xCoordInput = document.querySelector("#xCoordInput");
                const yCoordInput = document.querySelector("#yCoordInput");

                const name = village.querySelector(".name").textContent;
                let coordinateX = village.querySelector(".coordinateX").textContent;
                let coordinateY = village.querySelector(".coordinateY").textContent;

                coordinateX = parseInt(coordinateX.match(/[−\d]+/g).join("").replace("−", "-"));
                coordinateY = parseInt(coordinateY.match(/[−\d]+/g).join("").replace("−", "-"));

                villageNameInput.value = name;
                xCoordInput.value = coordinateX;
                yCoordInput.value = coordinateY;
            };
        });
    }

    /**
     * Parse active village building list.
     */
    function getBuildingList() {
        const buildingList = document.querySelector("#contentOuterContainer > div div.buildingList");
        if (!buildingList) return [];

        // for each list item parse name, lvl, build duration and timer (.name)(.lvl)(.buildDuration)(.timer)
        const buildingListLI = buildingList.querySelectorAll("li");

        const parsedBuildingList = [...buildingListLI].map(buildingItem => {
            const name = buildingItem.querySelector(".name").childNodes[0].textContent.trim();
            const lvl = buildingItem.querySelector(".lvl").textContent;
            const timestamp = Math.floor(Date.now() / 1000);
            const buildDuration = parseInt(buildingItem.querySelector(".buildDuration > .timer").getAttribute("value"));
            const timestampEnd = timestamp + buildDuration;

            return {
                name,
                lvl,
                buildDuration,
                timestampEnd
            };
        });

        return parsedBuildingList;
    }

    /**
     * Parse active village id and name.
     */
    function getActiveVillage() {
        const activeVillage = document.querySelector("#sidebarBoxVillagelist > div.content > div.villageList .listEntry.active");
        const id = activeVillage.dataset.did;
        const name = activeVillage.querySelector(".name").textContent;

        return {
            id,
            name
        };
    }

    /**
     * Parse active village storage capacity.
     */
    function getStorage() {
        return {
            warehouse: parseInt(document.querySelector("#stockBar > div.warehouse > div > div").textContent.replace(/[^\x00-\x7F]|,/g, "")),
            granary: parseInt(document.querySelector("#stockBar > div.granary > div > div").textContent.replace(/[^\x00-\x7F]|,/g, ""))
        };
    }

    /**
     * Parse active village available resources.
     */
    function getResources() {
        return {
            lumber: resources.storage.l1,
            clay: resources.storage.l2,
            iron: resources.storage.l3,
            crop: resources.storage.l4
        };
    }

    /**
     * Parse active village hourly resource production.
     */
    function getHourlyProduction() {
        // const lumberTitle = document.querySelector("#stockBar > div.warehouse > a:nth-child(2)").title;
        // const clayTitle = document.querySelector("#stockBar > div.warehouse > a:nth-child(3)").title;
        // const ironTitle = document.querySelector("#stockBar > div.warehouse > a:nth-child(4)").title;

        // const cropTitle = document.querySelector("#stockBar > div.granary > a:nth-child(3)").title;
        // const cropTemp = cropTitle.substring(cropTitle.indexOf(":") + 2, cropTitle.indexOf("<"));
        // const spanTemp = document.createElement("span");
        // spanTemp.innerHTML = cropTemp;

        return {
            lumber: resources.production.l1,
            clay: resources.production.l2,
            iron: resources.production.l3,
            crop: resources.production.l4
        };
    }

    /**
     * Parse village list.
     */
    function getVillageList() {
        const villageList = document.querySelectorAll("#sidebarBoxVillagelist > div.content > div.villageList > div.dropContainer");

        return [...villageList].map(village => {
            const index = village.dataset.sortindex; // starts at 1
            const id = village.querySelector(".listEntry").dataset.did;
            const href = village.querySelector(".listEntry a").getAttribute("href");
            const name = village.querySelector(".listEntry .name").textContent;

            return {
                index,
                id,
                href,
                name
            };
        });
    }

    /**
     * Parse barracks training list.
     */
    function getBarracksTrainingList() {
        const training = document.querySelectorAll("#build > table > tbody > tr:not(.next)");
        if (!training) return [];

        return [...training].map(unit => {
            // const unitIcon = unit.querySelector(".desc>.unit").outerHTML;
            const name = unit.querySelector(".desc").childNodes[2].textContent.replace(/\s+/g, " ").trim();
            const fin = unit.querySelector(".fin>span").textContent;
            const timestamp = Math.floor(Date.now() / 1000);
            const dur = parseInt(unit.querySelector(".dur>.timer").getAttribute("value"));
            const timestampEnd = timestamp + dur;

            return {
                // unitIcon,
                name,
                timestampEnd,
                fin
            };
        });
    }

    /**
     * Parse stable training list. 
     */
    function getStableTrainingList() {
        const training = document.querySelectorAll("#build > table > tbody > tr:not(.next)");
        if (!training) return [];

        return [...training].map(unit => {
            // const unitIcon = unit.querySelector(".desc>.unit").outerHTML;
            const name = unit.querySelector(".desc").childNodes[2].textContent.replace(/\s+/g, " ").trim();
            const fin = unit.querySelector(".fin>span").textContent;
            const timestamp = Math.floor(Date.now() / 1000);
            const dur = parseInt(unit.querySelector(".dur>.timer").getAttribute("value"));
            const timestampEnd = timestamp + dur;

            return {
                // unitIcon,
                name,
                timestampEnd,
                fin
            };
        });
    }

    /**
     * // TODO not tested
     * Parse workshop training list. 
     */
    function getWorkshopTrainingList() {
        const training = document.querySelectorAll("#build > table > tbody > tr:not(.next)");
        if (!training) return [];

        return [...training].map(unit => {
            // const unitIcon = unit.querySelector(".desc>.unit").outerHTML;
            const name = unit.querySelector(".desc").childNodes[2].textContent.replace(/\s+/g, " ").trim();
            const fin = unit.querySelector(".fin>span").textContent;
            const timestamp = Math.floor(Date.now() / 1000);
            const dur = parseInt(unit.querySelector(".dur>.timer").getAttribute("value"));
            const timestampEnd = timestamp + dur;

            return {
                // unitIcon,
                name,
                timestampEnd,
                fin
            };
        });
    }

    /**
     * Parse player name
     */
    function getPlayerName() {
        return document.querySelector("#sidebarBoxActiveVillage > div.content > div.playerName").textContent;
    }

    /**
     * Get building list from localStorage.
     */
    function getLocalStorage_buildingList(playerName, villageList) {
        let localStorage_buildingList = localStorage[`us_${playerName}_buildingList`];

        if (localStorage_buildingList === undefined) {
            const temp = {};

            for (const village of villageList) {
                temp[village.id] = [];
            }

            localStorage_buildingList = temp;
            localStorage[`us_${playerName}_buildingList`] = JSON.stringify(temp);
        } else {
            localStorage_buildingList = JSON.parse(localStorage_buildingList);

            villageList.forEach(village => {
                if (localStorage_buildingList[village.id] === undefined) localStorage_buildingList[village.id] = [];
            });

            localStorage[`us_${playerName}_buildingList`] = JSON.stringify(localStorage_buildingList);
        }

        return localStorage_buildingList;
    }

    /**
     * Set building list to localStorage.
     */
    function setLocalStorage_buildingList(playerName, activeVillageID, localStorage_buildingList, buildingList) {
        localStorage_buildingList[activeVillageID] = buildingList;
        localStorage[`us_${playerName}_buildingList`] = JSON.stringify(localStorage_buildingList);
    }

    /**
     * Get barracks training list from localStorage.
     */
    function getLocalStorage_barracksTrainingList(playerName, villageList) {
        let localStorage_barracksTrainingList = localStorage[`us_${playerName}_barracksTrainingList`];

        if (localStorage_barracksTrainingList === undefined) {
            const temp = {};

            for (const village of villageList) {
                temp[village.id] = [];
            }

            localStorage_barracksTrainingList = temp;
            localStorage[`us_${playerName}_barracksTrainingList`] = JSON.stringify(temp);
        } else {
            localStorage_barracksTrainingList = JSON.parse(localStorage_barracksTrainingList);

            villageList.forEach(village => {
                if (localStorage_barracksTrainingList[village.id] === undefined) localStorage_barracksTrainingList[village.id] = [];
            });

            localStorage[`us_${playerName}_barracksTrainingList`] = JSON.stringify(localStorage_barracksTrainingList);
        }

        return localStorage_barracksTrainingList;
    }

    /**
     * Set barracks training list to localStorage.
     */
    function setLocalStorage_barracksTrainingList(playerName, activeVillageID, localStorage_barracksTrainingList, barracksTrainingList) {
        localStorage_barracksTrainingList[activeVillageID] = barracksTrainingList;
        localStorage[`us_${playerName}_barracksTrainingList`] = JSON.stringify(localStorage_barracksTrainingList);
    }

    /**
     * Get stable training list from localStorage.
     */
    function getLocalStorage_stableTrainingList(playerName, villageList) {
        let localStorage_stableTrainingList = localStorage[`us_${playerName}_stableTrainingList`];

        if (localStorage_stableTrainingList === undefined) {
            const temp = {};

            for (const village of villageList) {
                temp[village.id] = [];
            }

            localStorage_stableTrainingList = temp;
            localStorage[`us_${playerName}_stableTrainingList`] = JSON.stringify(temp);
        } else {
            localStorage_stableTrainingList = JSON.parse(localStorage_stableTrainingList);

            villageList.forEach(village => {
                if (localStorage_stableTrainingList[village.id] === undefined) localStorage_stableTrainingList[village.id] = [];
            });

            localStorage[`us_${playerName}_stableTrainingList`] = JSON.stringify(localStorage_stableTrainingList);
        }

        return localStorage_stableTrainingList;
    }

    /**
     * Set stable training list to localStorage.
     */
    function setLocalStorage_stableTrainingList(playerName, activeVillageID, localStorage_stableTrainingList, stableTrainingList) {
        localStorage_stableTrainingList[activeVillageID] = stableTrainingList;
        localStorage[`us_${playerName}_stableTrainingList`] = JSON.stringify(localStorage_stableTrainingList);
    }

    /**
     * // TODO not tested
     * Get workshop training list from localStorage.
     */
    function getLocalStorage_workshopTrainingList(playerName, villageList) {
        let localStorage_workshopTrainingList = localStorage[`us_${playerName}_workshopTrainingList`];

        if (localStorage_workshopTrainingList === undefined) {
            const temp = {};

            for (const village of villageList) {
                temp[village.id] = [];
            }

            localStorage_workshopTrainingList = temp;
            localStorage[`us_${playerName}_workshopTrainingList`] = JSON.stringify(temp);
        } else {
            localStorage_workshopTrainingList = JSON.parse(localStorage_workshopTrainingList);

            villageList.forEach(village => {
                if (localStorage_workshopTrainingList[village.id] === undefined) localStorage_workshopTrainingList[village.id] = [];
            });

            localStorage[`us_${playerName}_workshopTrainingList`] = JSON.stringify(localStorage_workshopTrainingList);
        }

        return localStorage_workshopTrainingList;
    }

    /**
     * // TODO not tested
     * Set workshop training list to localStorage.
     */
    function setLocalStorage_workshopTrainingList(playerName, activeVillageID, localStorage_workshopTrainingList, workshopTrainingList) {
        localStorage_workshopTrainingList[activeVillageID] = workshopTrainingList;
        localStorage[`us_${playerName}_workshopTrainingList`] = JSON.stringify(localStorage_workshopTrainingList);
    }

    /**
     * Get resources from localStorage.
     */
    function getLocalStorage_resources(playerName, villageList) {
        const localstorageProp = `us_${playerName}_resources`;
        let localstorage = localStorage[localstorageProp];

        if (localstorage === undefined) {
            const temp = {};

            for (const village of villageList) {
                temp[village.id] = [];
            }

            localstorage = temp;
            localStorage[localstorageProp] = JSON.stringify(temp);
        } else {
            localstorage = JSON.parse(localstorage);

            villageList.forEach(village => {
                if (localstorage[village.id] === undefined) localstorage[village.id] = [];
            });

            localStorage[localstorageProp] = JSON.stringify(localstorage);
        }

        return localstorage;
    }

    /**
     * Set resources to localStorage.
     */
    function setLocalStorage_resources(playerName, activeVillageID, localstorage, data) {
        const localstorageProp = `us_${playerName}_resources`;
        localstorage[activeVillageID] = data;
        localStorage[localstorageProp] = JSON.stringify(localstorage);
    }

    /**
     * Create a draggable element that contains the Village Overview.
     */
    function createVillageOverviewHTML() {
        const playerName = getPlayerName();
        const villageList = getVillageList();
        const activeVillage = getActiveVillage();

        // get draggable element coordinates
        let localStorage_villageOverview_coords = localStorage[`us_${playerName}_villageOverview_coords`];
        if (localStorage_villageOverview_coords === undefined) {
            localStorage_villageOverview_coords = [0, 0]; // top left
            localStorage[`us_${playerName}_villageOverview_coords`] = JSON.stringify(localStorage_villageOverview_coords);
        } else {
            localStorage_villageOverview_coords = JSON.parse(localStorage_villageOverview_coords);
        }

        const localStorage_buildingList = getLocalStorage_buildingList(playerName, villageList);
        const localStorage_barracksTrainingList = getLocalStorage_barracksTrainingList(playerName, villageList);
        const localStorage_stableTrainingList = getLocalStorage_stableTrainingList(playerName, villageList);
        const localStorage_workshopTrainingList = getLocalStorage_workshopTrainingList(playerName, villageList);

        const barracksPNG = "";
        const stablePNG = "";
        const workshopPNG = "";

        const tableRowsHTML = villageList.map(village => {
            const buildingListHTML = localStorage_buildingList[village.id].map(building => {
                const timestamp = Math.floor(Date.now() / 1000);
                const timerValue = building.timestampEnd - timestamp;
                const isDone = timerValue < 0;

                const timerHTML = isDone ? `<span style="width:50px; display:inline-block;">DONE</span>` : `<span style="width:50px; display:inline-block;" class="timer" counting="down" value="${timerValue}"></span>`;

                return `<li ${isDone ? 'class="done"' : ""}>
                            <span>${building.name}</span>
                            ${timerHTML}
                        </li>`;
            }).join("");

            let barracksImageHTML = "";
            if (localStorage_barracksTrainingList[village.id].length > 0) {
                const img = document.createElement("img");
                img.src = barracksPNG;
                img.width = "45";
                img.title = localStorage_barracksTrainingList[village.id].map(troop => {
                    const timestamp = parseInt(Math.floor(Date.now() / 1000));
                    const timerValue = troop.timestampEnd - timestamp;
                    const isDone = timerValue < 0;

                    if (isDone) return "";
                    return `${troop.name} | ${new Date(timerValue * 1000).toISOString().slice(11, 19)} | ${troop.fin}<br>`;
                }).join("");

                if (img.title !== "") {
                    img.title = img.title + "||";
                    barracksImageHTML = img.outerHTML;
                }
            }

            let stableImageHTML = "";
            if (localStorage_stableTrainingList[village.id].length > 0) {
                const img = document.createElement("img");
                img.src = stablePNG;
                img.width = "45";
                img.title = localStorage_stableTrainingList[village.id].map(troop => {
                    const timestamp = parseInt(Math.floor(Date.now() / 1000));
                    const timerValue = troop.timestampEnd - timestamp;
                    const isDone = timerValue < 0;

                    if (isDone) return "";
                    return `${troop.name} | ${new Date(timerValue * 1000).toISOString().slice(11, 19)} | ${troop.fin}<br>`;
                }).join("");

                if (img.title !== "") {
                    img.title = img.title + "||";
                    stableImageHTML = img.outerHTML;
                }
            }

            let workshopImageHTML = "";
            if (localStorage_workshopTrainingList[village.id].length > 0) {
                const img = document.createElement("img");
                img.src = workshopPNG;
                img.width = "45";
                img.title = localStorage_workshopTrainingList[village.id].map(troop => {
                    const timestamp = parseInt(Math.floor(Date.now() / 1000));
                    const timerValue = troop.timestampEnd - timestamp;
                    const isDone = timerValue < 0;

                    if (isDone) return "";
                    return `${troop.name} | ${new Date(timerValue * 1000).toISOString().slice(11, 19)} | ${troop.fin}<br>`;
                }).join("");

                if (img.title !== "") {
                    img.title = img.title + "||";
                    workshopImageHTML = img.outerHTML;
                }
            }

            const troopsHTML = barracksImageHTML + stableImageHTML + workshopImageHTML;

            return `<tr>
                        <td>
                            <a ${activeVillage.id === village.id ? "style=color:black;" : ""} href=${village.href}>${village.name}</a>
                        </td>
                        <td>
                            <ul>
                                ${buildingListHTML}
                            </ul>
                        </td>
                        <td>
                            ${troopsHTML}
                        </td>
                    </tr>`;
        }).join("");

        const villageOverviewHTML = `<div id="us-overview" class="us-draggable" style="top: ${localStorage_villageOverview_coords[0]}px; left: ${localStorage_villageOverview_coords[1]}px;">
                                        <div class="us-overview__header us-draggable--header" style="display: flex; align-items: center;"><div style="flex-grow: 1;">Click here to move</div><div class="us-settingsIconSVG" style="margin-left: 5px; cursor: pointer;"></div></div>
                                        <table>
                                            <thead>
                                                <tr>
                                                    <th>Villages</th>
                                                    <th>Building</th>
                                                    <th>Troops</th>
                                                </tr>
                                            </thead>
                                            <tbody>${tableRowsHTML}</tbody>
                                        </table>
                                    </div>`;

        document.body.insertAdjacentHTML("beforeend", villageOverviewHTML);
        dragElement(document.getElementById("us-overview"), `us_${playerName}_villageOverview_coords`);

        // create settings
        const us_settings = createSettingsHTML();

        const us_settings_icon = document.querySelector("#us-overview .us-settingsIconSVG");
        us_settings_icon.id = "us-settingsIcon";
        us_settings_icon.title = "Userscript Settings ||";
        us_settings_icon.onmousedown = function (e) {
            e.stopPropagation();
        };
        us_settings_icon.onmouseup = function (e) {
            us_settings.classList.toggle("us--display-block");
        };

        const svg = document.createElementNS("http://www.w3.org/2000/svg", "svg");
        svg.setAttributeNS(null, "fill", "#000000");
        svg.setAttributeNS(null, "viewBox", "0 0 48 48");
        svg.setAttributeNS(null, "width", "20");
        svg.setAttributeNS(null, "heigth", "20");
        const path = document.createElementNS("http://www.w3.org/2000/svg", "path");
        path.setAttributeNS(null, "d", `M39.139,26.282C38.426,25.759,38,24.919,38,24.034s0.426-1.725,1.138-2.247l3.294-2.415	c0.525-0.386,0.742-1.065,0.537-1.684c-0.848-2.548-2.189-4.872-3.987-6.909c-0.433-0.488-1.131-0.642-1.728-0.38l-3.709,1.631	c-0.808,0.356-1.749,0.305-2.516-0.138c-0.766-0.442-1.28-1.23-1.377-2.109l-0.446-4.072c-0.071-0.648-0.553-1.176-1.191-1.307	c-2.597-0.531-5.326-0.54-7.969-0.01c-0.642,0.129-1.125,0.657-1.196,1.308l-0.442,4.046c-0.097,0.88-0.611,1.668-1.379,2.11	c-0.766,0.442-1.704,0.495-2.515,0.138l-3.729-1.64c-0.592-0.262-1.292-0.11-1.725,0.377c-1.804,2.029-3.151,4.35-4.008,6.896	c-0.208,0.618,0.008,1.301,0.535,1.688l3.273,2.4C9.574,22.241,10,23.081,10,23.966s-0.426,1.725-1.138,2.247l-3.294,2.415	c-0.525,0.386-0.742,1.065-0.537,1.684c0.848,2.548,2.189,4.872,3.987,6.909c0.433,0.489,1.133,0.644,1.728,0.38l3.709-1.631	c0.808-0.356,1.748-0.305,2.516,0.138c0.766,0.442,1.28,1.23,1.377,2.109l0.446,4.072c0.071,0.648,0.553,1.176,1.191,1.307	C21.299,43.864,22.649,44,24,44c1.318,0,2.648-0.133,3.953-0.395c0.642-0.129,1.125-0.657,1.196-1.308l0.443-4.046	c0.097-0.88,0.611-1.668,1.379-2.11c0.766-0.441,1.705-0.493,2.515-0.138l3.729,1.64c0.594,0.263,1.292,0.111,1.725-0.377	c1.804-2.029,3.151-4.35,4.008-6.896c0.208-0.618-0.008-1.301-0.535-1.688L39.139,26.282z M24,31c-3.866,0-7-3.134-7-7s3.134-7,7-7	s7,3.134,7,7S27.866,31,24,31z`);

        svg.append(path);
        us_settings_icon.append(svg);
        document.body.append(us_settings);
    }

    /**
     * Create a draggable element that contains Village Resources.
     */
    function createResourcesHTML() {
        /**
         * Update resourcesHTML values.
         */
        function updateResources(localStorage_resources, villageList) {
            villageList.forEach(village => {
                const id = village.id;

                const resources = localStorage_resources[id][0];

                if (resources === undefined) {
                    // document.querySelector(`#us-${id}-l1 .us-bar`).style.width = `${lumberPercent}%`;
                    // document.querySelector(`#us-${id}-l2 .us-bar`).style.width = `${clayPercent}%`;
                    // document.querySelector(`#us-${id}-l3 .us-bar`).style.width = `${ironPercent}%`;
                    // document.querySelector(`#us-${id}-l4 .us-bar`).style.width = `${cropPercent}%`;

                    document.querySelector(`#us-${id}-l1 .us-resourceValue`).textContent = "??";
                    document.querySelector(`#us-${id}-l2 .us-resourceValue`).textContent = "??";
                    document.querySelector(`#us-${id}-l3 .us-resourceValue`).textContent = "??";
                    document.querySelector(`#us-${id}-l4 .us-resourceValue`).textContent = "??";

                    return;
                }

                const lumber = resources.resources.lumber;
                const clay = resources.resources.clay;
                const iron = resources.resources.iron;
                const crop = resources.resources.crop;
                const lumberH = resources.hourlyProduction.lumber;
                const clayH = resources.hourlyProduction.clay;
                const ironH = resources.hourlyProduction.iron;
                const cropH = resources.hourlyProduction.crop;
                const warehouse = resources.storage.warehouse;
                const granary = resources.storage.granary;
                const timestamp = resources.timestamp;

                const currentTimestamp = parseInt(Math.floor(Date.now() / 1000));
                const diffTimestamp = currentTimestamp - timestamp;
                const lumberCalc = (lumberH * diffTimestamp) / 3600;
                const clayCalc = (clayH * diffTimestamp) / 3600;
                const ironCalc = (ironH * diffTimestamp) / 3600;
                const cropCalc = (cropH * diffTimestamp) / 3600;

                const lumberValue = lumber + Math.floor(lumberCalc);
                const clayValue = clay + Math.floor(clayCalc);
                const ironValue = iron + Math.floor(ironCalc);
                const cropValue = crop + Math.floor(cropCalc);

                const lumberPercent = Math.floor((lumberValue * 100) / warehouse);
                const clayPercent = Math.floor((clayValue * 100) / warehouse);
                const ironPercent = Math.floor((ironValue * 100) / warehouse);
                const cropPercent = Math.floor((cropValue * 100) / granary);

                document.querySelector(`#us-${id}-l1 .us-bar`).style.width = `${lumberPercent <= 100 ? lumberPercent : 100}%`;
                document.querySelector(`#us-${id}-l2 .us-bar`).style.width = `${clayPercent <= 100 ? clayPercent : 100}%`;
                document.querySelector(`#us-${id}-l3 .us-bar`).style.width = `${ironPercent <= 100 ? ironPercent : 100}%`;
                document.querySelector(`#us-${id}-l4 .us-bar`).style.width = `${cropPercent <= 100 ? cropPercent : 100}%`;

                document.querySelector(`#us-${id}-l1 .us-resourceValue`).textContent = `${lumberValue <= warehouse ? lumberValue : warehouse}`;
                document.querySelector(`#us-${id}-l2 .us-resourceValue`).textContent = `${clayValue <= warehouse ? clayValue : warehouse}`;
                document.querySelector(`#us-${id}-l3 .us-resourceValue`).textContent = `${ironValue <= warehouse ? ironValue : warehouse}`;
                document.querySelector(`#us-${id}-l4 .us-resourceValue`).textContent = `${cropValue <= granary ? cropValue : granary}`;

                if (cropH < 0) {
                    document.querySelector(`#us-${id}-l4 .us-resourceValue`).classList.add("us--text-alert");
                }
            });
        }

        const playerName = getPlayerName();
        const villageList = getVillageList();
        const activeVillage = getActiveVillage();
        const localStorageProp = `us_${playerName}_resources_coords`;

        // get draggable element coordinates
        let localStorage_coords = localStorage[localStorageProp];
        if (localStorage_coords === undefined) {
            localStorage_coords = [250, 0]; // top left
            localStorage[localStorageProp] = JSON.stringify(localStorage_coords);
        } else {
            localStorage_coords = JSON.parse(localStorage_coords);
        }
        const top = localStorage_coords[0];
        const left = localStorage_coords[1];

        const localStorage_buildingList = getLocalStorage_buildingList(playerName, villageList);
        const localStorage_resources = getLocalStorage_resources(playerName, villageList);




        // html
        const tableRowsHTML = villageList.map(village => {
            const resources = localStorage_resources[village.id][0];
            let warehouse;
            let granary;

            if (resources === undefined) {
                warehouse = "??";
                granary = "??";
            } else {
                warehouse = resources.storage.warehouse;
                granary = resources.storage.granary;
            }

            return `<tr>
                        <td>
                            <a ${activeVillage.id === village.id ? "style=color:black;" : ""} href=${village.href}>${village.name}</a>
                        </td>
                        <td>
                            <div class="us-resources">
                                <div id="us-${village.id}-l1" class="us-resource" title="${warehouse}||">
                                    <div class="us-resourceValue"></div>
                                    <div class="us-barBox">
                                        <div class="us-bar"></div>
                                    </div>
                                </div>
                                <div id="us-${village.id}-l2" class="us-resource" title="${warehouse}||">
                                    <div class="us-resourceValue"></div>
                                    <div class="us-barBox">
                                        <div class="us-bar"></div>
                                    </div>
                                </div>
                                <div id="us-${village.id}-l3" class="us-resource" title="${warehouse}||">
                                    <div class="us-resourceValue"></div>
                                    <div class="us-barBox">
                                        <div class="us-bar"></div>
                                    </div>
                                </div>
                                <div id="us-${village.id}-l4" class="us-resource" title="${granary}||">
                                    <div class="us-resourceValue"></div>
                                    <div class="us-barBox">
                                        <div class="us-bar"></div>
                                    </div>
                                </div>
                            </div>
                        </td>
                    </tr>`;
        }).join("");

        const html = `<div id="us-resources" class="us-draggable" style="top:${top}px; left:${left}px;">
                            <div class="us-resources__header us-draggable--header">Click here to move</div>
                            <table>
                                <thead>
                                    <tr>
                                        <th>Villages</th>
                                        <th>Resources (not 100% accurate)</th>
                                    </tr>
                                </thead>
                                <tbody>${tableRowsHTML}</tbody>
                            </table>
                        </div>`;

        document.body.insertAdjacentHTML("beforeend", html);
        dragElement(document.getElementById("us-resources"), localStorageProp);
        updateResources(localStorage_resources, villageList);

        setInterval(() => {
            updateResources(localStorage_resources, villageList);
        }, 1000);
    }

    /**
     * Create a icon on the map toolbar that can create a semi larger map (a bit like the travian plus larger map feature).
     */
    function createSemiLargerMapHTML() {
        const div = document.createElement("div");
        div.classList.add("iconButton", "viewFullGold");
        div.title = "Semi Larger map (no plus needed) ||";
        div.open = false;

        const mapToolbar = document.querySelector("#toolbar > div.ml > div > div > div.contents");
        const iconCropFinder = document.querySelector("#iconCropfinder");


        div.onclick = function () {
            const map1 = document.querySelector("#mapContainer > div:nth-child(1)");
            const map2 = document.querySelector("#mapContainer > div:nth-child(1) > div:nth-child(2)");
            const elements = [
                document.querySelector("#topBar"),
                document.querySelector("#topBarHeroWrapper"),
                document.querySelector("#header"),
                document.querySelector("#servertime"),
                document.querySelector("#sidebarBeforeContent"),
                document.querySelector("#sidebarAfterContent"),
                document.querySelector("#mapContainer > div.ruler.x"),
                document.querySelector("#mapContainer > div.ruler.y"),
                document.querySelector("#us-overview"),
                document.querySelector("#us-resources")

            ];

            if (this.open) {
                elements.forEach(el => {
                    el.style.visibility = "";
                });

                map1.style.overflow = "hidden";
                map2.classList.remove("us-map-open");

                this.open = false;
                return;
            }

            elements.forEach(el => {
                el.style.visibility = "hidden";
            });

            map1.style.overflow = "unset";
            map2.classList.add("us-map-open");

            this.open = true;
        };

        mapToolbar.insertBefore(div, iconCropFinder);
    }

    /**
     * Create settings container.
     */
    function createSettingsHTML() {
        const us_settings = document.createElement("div");
        us_settings.id = "us-settings";
        us_settings.style = "position: absolute; top: 10px; left: 50%; transform: translate(-50%, 0); z-index: 999; background-color: #D2BDA1;";
        us_settings.style.minWidth = "600px";
        us_settings.style.display = "none";

        const html = `<div style="background-color: #FFFFFF; margin: 10px; padding: 10px;">
            <div>
                <div style="background: #E0EBDF; border-radius: 7px; padding: 5px;"><strong>Settings</strong></div>
                <div>
                    <ul>
                        <li style="margin-bottom: 10px;"><strong><a onclick="this.preventDefault;localStorage.clear();location.reload();">Clear Data</a></strong></li>
                    </ul>
                </div>
            </div> 
            <div>
                <div style="background: #E0EBDF; border-radius: 7px; padding: 5px;"><strong>Features</strong></div>
                <div>
                    <ul>
                        <li style="margin-bottom: 10px;">
                            <a href="https://greasyfork.org/zh-CN/scripts/463332-travianhelper-travian-legends" target="_blank">Features</a>
                        </li>
                    </ul>
                </div>
            </div>
            <div>
                <div style="background: #E0EBDF; border-radius: 7px; padding: 5px;"><strong>Script Info</strong></div>
                <div>
                    <ul>
                        <li style="margin-bottom: 10px;"><strong>Name:</strong> TravianHelper(Travian Legends++)</li>
                        <li style="margin-bottom: 10px;"><strong>Version:</strong> ${myVersion}</li>
                        <li style="margin-bottom: 10px;"><strong>Author:</strong> bingkx & SkillsBoY</li>
                        <li style="margin-bottom: 10px;"><strong>Want new features?</strong> <a href="https://greasyfork.org/zh-CN/scripts/463332-travianhelper-travian-legends/feedback" target="_blank">Feedback</a></li>
                    </ul>
                </div>
            </div>  
            <div style="text-align: center; cursor: pointer;" onclick="document.querySelector('#us-settings').classList.toggle('us--display-block');"><strong>CLOSE</strong></div>         
        </div>`;

        us_settings.insertAdjacentHTML("beforeend", html);

        return us_settings;
    }



}

// utilities
{
    /**
     * Make an element draggable
     */
    function dragElement(elmnt, localStorageProp) {
        var pos1 = 0, pos2 = 0, pos3 = 0, pos4 = 0;

        elmnt.querySelector(`.${elmnt.id}__header`).onmousedown = dragMouseDown;

        function dragMouseDown(e) {
            e = e || window.event;
            e.preventDefault();
            // get the mouse cursor position at startup:
            pos3 = e.clientX;
            pos4 = e.clientY;
            document.onmouseup = closeDragElement;
            // call a function whenever the cursor moves:
            document.onmousemove = elementDrag;
        }

        function elementDrag(e) {
            e = e || window.event;
            e.preventDefault();
            // calculate the new cursor position:
            pos1 = pos3 - e.clientX;
            pos2 = pos4 - e.clientY;
            pos3 = e.clientX;
            pos4 = e.clientY;
            // set the element's new position:
            elmnt.style.top = (elmnt.offsetTop - pos2) + "px";
            elmnt.style.left = (elmnt.offsetLeft - pos1) + "px";
        }

        function closeDragElement() {
            // stop moving when mouse button is released:
            document.onmouseup = null;
            document.onmousemove = null;

            if (elmnt.offsetLeft + elmnt.offsetWidth <= 50) {
                elmnt.style.left = 50 - elmnt.offsetWidth + "px";
            }

            if (elmnt.offsetTop < 0) {
                elmnt.style.top = "0";
            }

            if (localStorageProp !== undefined) {
                localStorage[localStorageProp] = JSON.stringify([elmnt.offsetTop, elmnt.offsetLeft]);
            }
        }
    }
}

// main code
{
    const playerName = getPlayerName();
    const villageList = getVillageList();
    const activeVillage = getActiveVillage();
    const resources = getResources();
    const storage = getStorage();
    const hourlyProduction = getHourlyProduction();
    const timestamp = parseInt(Math.floor(Date.now() / 1000));

    const resourceData = [{
        resources,
        storage,
        hourlyProduction,
        timestamp
    }];

    const localStorage_buildingList = getLocalStorage_buildingList(playerName, villageList);
    const localStorage_barracksTrainingList = getLocalStorage_barracksTrainingList(playerName, villageList);
    const localStorage_stableTrainingList = getLocalStorage_stableTrainingList(playerName, villageList);
    const localStorage_workshopTrainingList = getLocalStorage_workshopTrainingList(playerName, villageList);
    const localStorage_resources = getLocalStorage_resources(playerName, villageList);

    // if current page is dorf1.php or dorf2.php
    if (/dorf1.php|dorf2.php/.test(window.location.pathname)) {
        const buildingList = getBuildingList();
        setLocalStorage_buildingList(playerName, activeVillage.id, localStorage_buildingList, buildingList);
    }

    // if current page is dorf1.php
    if (/dorf1.php/.test(window.location.pathname)) {

    }

    // if current building is gid=17 (marketplace) or gid=16 (rally point)
    if (/gid=17|gid=16/.test(window.location.search)) {
        coordsClickFillVillageDestinationFields();
    }

    // if current building is gid=19 (barracks)
    if (/gid=19/.test(window.location.search)) {
        const barracksTrainingList = getBarracksTrainingList();
        setLocalStorage_barracksTrainingList(playerName, activeVillage.id, localStorage_barracksTrainingList, barracksTrainingList);
    }

    // if current building is gid=20 (stable)
    if (/gid=20/.test(window.location.search)) {
        const stableTrainingList = getStableTrainingList();
        setLocalStorage_stableTrainingList(playerName, activeVillage.id, localStorage_stableTrainingList, stableTrainingList);
    }

    // if current building is gid=21 (workshop)
    if (/gid=21/.test(window.location.search)) {
        const workshopTrainingList = getWorkshopTrainingList();
        setLocalStorage_workshopTrainingList(playerName, activeVillage.id, localStorage_workshopTrainingList, workshopTrainingList);
    }

    // if current page is on the map (/karte.php)
    if (/karte.php/.test(window.location.pathname)) {
        createSemiLargerMapHTML();
    }


    setLocalStorage_resources(getPlayerName(), getActiveVillage().id, localStorage_resources, resourceData);

    makeBuildingIconsAvailable();
    createVillageOverviewHTML();
    createResourcesHTML();
}