MZ Tactics Manager

Userscript to manage tactics in ManagerZone

Install this script?
Author's suggested script

You may also like ylOppTactsPreview (Modified).

Install this script
// ==UserScript==
// @name         MZ Tactics Manager
// @namespace    douglaskampl
// @version      12.1.1
// @description  Userscript to manage tactics in ManagerZone
// @author       Douglas Vieira
// @match        https://www.managerzone.com/?p=tactics
// @match        https://www.managerzone.com/?p=national_teams&sub=tactics&type=*
// @icon         https://yt3.googleusercontent.com/ytc/AIdro_mDHaJkwjCgyINFM7cdUV2dWPPnL9Q58vUsrhOmRqkatg=s160-c-k-c0x00ffffff-no-rj
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_deleteValue
// @grant        GM_addStyle
// @require      https://cdnjs.cloudflare.com/ajax/libs/jsSHA/3.3.1/sha256.js
// @require      https://cdnjs.cloudflare.com/ajax/libs/i18next/23.7.16/i18next.min.js
// @license      MIT
// ==/UserScript==

(function () {
    'use strict';

    // ==============================
    // STYLES
    // ==============================
    GM_addStyle(`@import url("https://fonts.googleapis.com/css2?family=Space+Grotesk:wght@300;400;500;600&display=swap");@import url("https://fonts.googleapis.com/css2?family=Pacifico&display=swap");:root{--bg-color:#2a2d40;--text-color:#dcdde1;--highlight-color:#a37acc;--accent-color:#6c70d1;--shadow-color-dark:rgba(0,0,0,0.3);--shadow-color-light:rgba(255,255,255,0.05);--border-radius:12px;--shadow-base:3px 3px 6px var(--shadow-color-dark),-3px -3px 6px var(--shadow-color-light);--shadow-inset:inset 2px 2px 4px var(--shadow-color-dark),inset -2px -2px 4px var(--shadow-color-light);--shadow-concave:4px 4px 8px var(--shadow-color-dark),-4px -4px 8px var(--shadow-color-light),inset 1px 1px 2px var(--shadow-color-light),inset -1px -1px 2px var(--shadow-color-dark);--short-passing-color:#54a0ff;--wing-play-color:#5dd39e;--other-style-color:#ffcb77;--uncategorized-color:#8395a7;}#mz_tactics_panel{font-family:"Space Grotesk",-apple-system,sans-serif;background-color:var(--bg-color);border-radius:var(--border-radius);padding:20px;margin:12px;box-shadow:var(--shadow-base);border:1px solid rgba(255,255,255,0.05);transition:max-height 0.4s ease-out, padding 0.4s ease-out, margin 0.4s ease-out, opacity 0.3s ease-out;max-height:1000px;opacity:1;color:var(--text-color);overflow:visible;}#mz_tactics_panel.collapsed{max-height:0 !important; padding-top: 0 !important; padding-bottom: 0 !important; margin-top: 0 !important; margin-bottom: 0 !important; opacity:0 !important; border:none !important; overflow: hidden !important;}.mz-group{background-color:rgba(0,0,0,0.1);border-radius:var(--border-radius);padding:16px;margin:10px 0;box-shadow:none;border:1px solid rgba(255,255,255,0.05);position:relative;}.mz-group-main-title{display:flex;justify-content:space-between;align-items:center;color:var(--text-color);font-size:18px;font-weight:500;margin:-4px 0 12px 0;padding-bottom:8px;border-bottom:1px solid rgba(255,255,255,0.1);}.mz-main-title{color:var(--text-color);font-family:"Space Grotesk",sans-serif;font-size:20px;font-weight:500;margin:0;padding:0;text-align:center;letter-spacing:0.2px;}.mz-version-text{color:var(--highlight-color);font-family:'Pacifico',cursive;font-size:1.1em;font-weight:400;margin-left:8px;transform:rotate(-5deg);}.mz-divider{width:50px;height:2px;background:var(--text-color);margin:10px auto 0;opacity:0.2;}#toggle_panel_btn{background:transparent;border:none;color:var(--text-color);cursor:pointer;padding:8px;width:32px;height:32px;border-radius:50%;margin-left:auto;font-size:18px;transition:all 0.3s ease;display:inline-flex;align-items:center;justify-content:center;}#toggle_panel_btn:hover{background:rgba(255,255,255,0.1);}#collapsed_icon{position:fixed;top:20px;right:20px;background:var(--bg-color);border-radius:50%;width:48px;height:48px;display:flex;align-items:center;justify-content:center;cursor:pointer;opacity:0;transition:all 0.3s ease;transform:scale(0);box-shadow:var(--shadow-base);z-index:1000;color:var(--text-color);font-size:16px; font-weight: bold; border: 1px solid rgba(255,255,255,0.1);}#collapsed_icon.visible{opacity:1;transform:scale(1);}#collapsed_icon:hover{transform:scale(1.05);box-shadow:0 0 15px rgba(163, 122, 204, 0.5);}#mz_tactics_panel .mzbtn{display:inline-flex;align-items:center;justify-content:center;padding:8px 14px;margin:4px;font-family:"Space Grotesk",sans-serif;font-size:13px;font-weight:500;color:var(--text-color);background:rgba(255,255,255,0.05);border:1px solid rgba(255,255,255,0.1);border-radius:8px;cursor:pointer;transition:all 0.2s ease;min-height:36px;box-shadow: 0 1px 2px rgba(0,0,0,0.1);}#mz_tactics_panel .mzbtn:hover{background:rgba(255,255,255,0.1);border-color:rgba(255,255,255,0.2);transform:translateY(-1px);box-shadow: 0 3px 6px rgba(0,0,0,0.15);}#mz_tactics_panel .mzbtn:active{background:rgba(0,0,0,0.1);transform:translateY(0);box-shadow: inset 0 1px 2px rgba(0,0,0,0.2);}#mz_tactics_panel select{font-family:"Space Grotesk",sans-serif;font-size:14px;color:var(--text-color);padding:8px 14px;border:1px solid rgba(255,255,255,0.1);border-radius:8px;background-color:rgba(255,255,255,0.05);cursor:pointer;margin:0 4px;transition:all 0.2s ease;-webkit-appearance:none;-moz-appearance:none;appearance:none;background-image:url("data:image/svg+xml;utf8,<svg fill='%23dcdde1' height='24' viewBox='0 0 24 24' width='24' xmlns='http://www.w3.org/2000/svg'><path d='M7 10l5 5 5-5z'/></svg>");background-repeat:no-repeat;background-position:right 10px top 50%;padding-right:30px;height:36px;box-sizing:border-box;}#mz_tactics_panel select:hover{background:rgba(255,255,255,0.1);border-color:rgba(255,255,255,0.2);}#mz_tactics_panel select:focus{outline:none;border-color:var(--accent-color);box-shadow:0 0 0 2px rgba(108, 112, 209, 0.3);}#mz_tactics_panel select option{background-color:var(--bg-color);color:var(--text-color);padding:5px 10px;}.tactics-selector-section{margin-bottom:12px;}.tactics-selector-label{display:none;}#language_flag{height:12px;width:16px;margin:6px;border:none;border-radius:4px;}#info_modal,#useful_links_modal{background:var(--bg-color);padding:24px;border-radius:var(--border-radius);color:var(--text-color);width:90%;max-width:500px;box-shadow:var(--shadow-base); border: 1px solid rgba(255,255,255,0.1);}#info_modal a,#useful_links_modal a{color:var(--accent-color);text-decoration:none;transition:color 0.3s ease;}#info_modal a:hover,#useful_links_modal a:hover{color:var(--highlight-color); text-decoration: underline;}#info_modal ul,#useful_links_modal ul{list-style:none;padding:0;}#info_modal ul li,#useful_links_modal ul li{margin:12px 0;padding:8px 12px;border-radius:8px;background:rgba(255,255,255,0.05);border:1px solid rgba(255,255,255,0.08);transition:all 0.3s ease;}#info_modal ul li:hover,#useful_links_modal ul li:hover{background:rgba(255,255,255,0.1);}#mz-modal-overlay{position:fixed;top:0;left:0;right:0;bottom:0;background-color:rgba(42, 45, 64, 0.8);backdrop-filter:blur(4px);display:flex;align-items:center;justify-content:center;z-index:10000;opacity:0;transition:opacity 0.3s ease;}#mz-modal-container{background:var(--bg-color);border-radius:var(--border-radius);padding:24px;box-shadow:0 10px 25px rgba(0,0,0,0.3);border:1px solid rgba(255,255,255,0.1);max-width:500px;width:90%;transform:scale(0.9);transition:transform 0.3s ease;color:var(--text-color);font-family:"Space Grotesk",-apple-system,sans-serif;}#mz-modal-overlay.active{opacity:1;}#mz-modal-overlay.active #mz-modal-container{transform:scale(1);}#mz-modal-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:20px;border-bottom:1px solid rgba(255,255,255,0.1);padding-bottom:12px;}#mz-modal-title{font-size:20px;font-weight:500;margin:0;}#mz-modal-close{background:transparent;border:none;color:var(--text-color);font-size:22px;cursor:pointer;transition:all 0.3s ease;padding:0;width:36px;height:36px;display:flex;align-items:center;justify-content:center;border-radius:50%;}#mz-modal-close:hover{background:rgba(255,255,255,0.1);color:var(--highlight-color);}#mz-modal-content{margin-bottom:24px;white-space:pre-line;line-height:1.5;}#mz-modal-input{width:calc(100% - 32px);background:rgba(255,255,255,0.05);border:1px solid rgba(255,255,255,0.1);color:var(--text-color);padding:14px 16px;border-radius:8px;font-family:"Space Grotesk",sans-serif;font-size:15px;margin-bottom:20px;transition:all 0.3s ease;box-sizing:border-box;}#mz-modal-input:focus{outline:none;border-color:var(--accent-color);box-shadow:0 0 0 2px rgba(108, 112, 209, 0.3);background:rgba(255,255,255,0.08);}#mz-modal-buttons{display:flex;justify-content:flex-start;gap:12px;}.mz-modal-btn{display:inline-flex;align-items:center;justify-content:center;padding:10px 18px;font-family:"Space Grotesk",sans-serif;font-size:15px;font-weight:500;color:var(--text-color);background:rgba(255,255,255,0.05);border:1px solid rgba(255,255,255,0.1);border-radius:8px;cursor:pointer;transition:all 0.2s ease;min-width:90px;box-shadow: 0 1px 2px rgba(0,0,0,0.1);}.mz-modal-btn:hover{background:rgba(255,255,255,0.1);border-color:rgba(255,255,255,0.2);transform:translateY(-1px);box-shadow: 0 3px 6px rgba(0,0,0,0.15);}.mz-modal-btn:active{background:rgba(0,0,0,0.1);transform:translateY(0);box-shadow: inset 0 1px 2px rgba(0,0,0,0.2);}.mz-modal-btn.primary{background:var(--accent-color);color:white;font-weight:600; border: none;}.mz-modal-btn.primary:hover{background:var(--highlight-color);}.mz-modal-btn.cancel{background:transparent;color:#aaa;border:1px solid rgba(255,255,255,0.1);}.mz-modal-btn.cancel:hover{background:rgba(255,255,255,0.05);color:var(--text-color); border-color:rgba(255,255,255,0.2);}.mz-modal-icon{display:inline-flex;align-items:center;justify-content:center;width:36px;height:36px;border-radius:50%;margin-right:14px;background:rgba(0,0,0,0.1);}.mz-modal-icon.success{color:#5dd39e; background: rgba(93, 211, 158, 0.1);}.mz-modal-icon.error{color:#ff6b6b; background: rgba(255, 107, 107, 0.1);}.mz-modal-icon.info{color:#54a0ff; background: rgba(84, 160, 255, 0.1);}.mz-modal-title-with-icon{display:flex;align-items:center;}.tactics-selector-container{position:relative;width:100%; display: flex; align-items: center; gap: 8px;}.tactics-dropdown-container{display:flex;flex-wrap:nowrap;gap:8px;margin-top:0; flex-grow: 1; align-items: center;}.tactics-search-box{width:160px !important;padding:8px 12px;margin-bottom:0 !important;border:1px solid rgba(255,255,255,0.1);border-radius:8px;background-color:rgba(255,255,255,0.05);color:var(--text-color);font-family:"Space Grotesk",sans-serif;font-size:14px;box-sizing:border-box;height:36px;transition:all 0.2s ease;position:relative; flex-shrink: 0;}.tactics-search-box:focus{outline:none;border-color:var(--accent-color);box-shadow:0 0 0 2px rgba(108, 112, 209, 0.3);background:rgba(255,255,255,0.08);}.tactics-search-box.filtering{border-bottom:2px solid var(--highlight-color);animation:pulse-border 1.5s infinite;}@keyframes pulse-border{0%{border-color:var(--highlight-color);}50%{border-color:transparent;}100%{border-color:var(--highlight-color);}}.tactics-filter-tabs{display:flex;margin:0;padding-bottom:0; overflow-x: auto; flex-shrink: 1; min-width: 100px; max-width: 300px; align-items: center; height: 36px; scrollbar-width: thin; scrollbar-color: rgba(255,255,255,0.2) transparent;}.tactics-filter-tab{position:relative; display:inline-flex; align-items: center; padding:4px 10px;margin-right:6px;border:1px solid transparent;border-radius:12px; background-color:transparent;color:var(--text-color);opacity:0.7;font-family:"Space Grotesk",sans-serif;font-size:11px; cursor:pointer;white-space:nowrap;transition:all 0.2s ease; flex-shrink: 0; height: 24px; line-height: 16px;}.tactics-filter-tab:hover{background-color:rgba(255,255,255,0.08);opacity:1;}.tactics-filter-tab.active{background-color: var(--category-color, rgba(255,255,255,0.15)); border-color:transparent;font-weight:500;opacity:1; color: #fff; text-shadow: 0 0 3px rgba(0,0,0,0.5);}.remove-category-btn{margin-left:5px; font-weight: bold; font-size: 10px; color:rgba(255,255,255,0.5); background:rgba(0,0,0,0.2); border-radius: 50%; width: 14px; height: 14px; display: inline-flex; justify-content: center; align-items: center; line-height: 14px; cursor: pointer; transition: all 0.2s ease;}.tactics-filter-tab:hover .remove-category-btn{color:rgba(255,255,255,0.8); background:rgba(0,0,0,0.4);}.remove-category-btn:hover{color:#fff; background:rgba(255,50,50,0.7); transform: scale(1.1);}.tactics-dropdown-wrapper{flex:1;min-width:180px;position:relative; flex-grow: 1; flex-shrink: 1;}.tactics-style-indicator{display:inline-block;width:8px;height:8px;border-radius:50%;margin-right:6px;}.tactics-style-indicator.short_passing{background-color:var(--short-passing-color);}.tactics-style-indicator.wing_play{background-color:var(--wing-play-color);}.tactics-style-indicator.other{background-color:var(--other-style-color);}.tactics-style-indicator.uncategorized{background-color:var(--uncategorized-color);}.tactics-category-header{color:rgba(220, 221, 225, 0.7);font-size:12px;font-weight:600;padding:4px 10px;background:rgba(0,0,0,0.2);margin-top:4px;border-radius:4px;}#category-selector{width:100%;margin-top:10px;padding:10px 12px;border:1px solid rgba(255,255,255,0.1);border-radius:8px;background-color:rgba(255,255,255,0.05);color:var(--text-color);font-family:"Space Grotesk",sans-serif;font-size:14px;box-sizing:border-box;}#category-selector option{padding:8px; background-color: var(--bg-color);}.category-selection-container{margin-top:15px;margin-bottom:5px;}.category-selection-label{display:block;margin-bottom:5px;font-size:14px;color:var(--text-color);opacity:0.9;}.mz-language-container{display:flex;align-items:center;gap:10px;}.mz-language-label{font-size:14px;font-weight:500;}.mz-language-dropdown{flex:1;}.new-category-input-container{margin-top:10px;display:none;}.new-category-input-container.visible{display:block;}#new-category-input{width:100%;padding:10px 12px;border:1px solid rgba(255,255,255,0.1);border-radius:8px;background-color:rgba(255,255,255,0.05);color:var(--text-color);font-family:"Space Grotesk",sans-serif;font-size:14px;box-sizing:border-box;}#new-category-input:focus{outline:none;border-color:var(--accent-color);box-shadow:0 0 0 2px rgba(108, 112, 209, 0.3);background:rgba(255,255,255,0.08);}#tactics_selector{height:36px;box-sizing:border-box;max-height:300px;overflow-y:auto; width: 100%;}#tactics_selector option{animation:fadeIn 0.3s ease;background-color:var(--bg-color);padding:8px 12px;margin:2px 0; color: var(--text-color);}#tactics_selector optgroup{background-color:#580000;border-left:3px solid var(--accent-color);font-weight:600;padding:8px 10px;margin-top:5px;border-radius:6px;color:var(--text-color);}@keyframes fadeIn{from{opacity:0;transform:translateY(-5px);}to{opacity:1;transform:translateY(0);}}@keyframes shake{0%,100%{transform:translateX(0);}25%{transform:translateX(-2px);}50%{transform:translateX(0);}75%{transform:translateX(2px);}}.tactics-dropdown-wrapper.filtering:after{content:'';position:absolute;width:10px;height:10px;border-radius:50%;background-color:var(--highlight-color);right:40px;top:13px;animation:pulse 1.5s infinite;}@keyframes pulse{0%{transform:scale(0.8);opacity:0.5;}50%{transform:scale(1.2);opacity:1;}100%{transform:scale(0.8);opacity:0.5;}}.action-buttons-section{display: flex; flex-wrap: wrap; margin-top: 10px; justify-content: flex-start; gap: 4px;}.action-dropdown-menu{position:absolute;background-color:var(--bg-color);border-radius:8px;box-shadow:0 5px 15px rgba(0,0,0,0.3);padding:8px;z-index:100;display:none;min-width:150px; border: 1px solid rgba(255,255,255,0.1);}.action-dropdown-menu button{display:block;width:100%;margin:4px 0;text-align:left; background: transparent; border: none; box-shadow: none;}.action-dropdown-menu button:hover{background:rgba(255,255,255,0.1); color: var(--highlight-color); transform: none; box-shadow: none;}.footer-actions{display:flex;align-items:center;gap:10px;}#combined_info_modal_content > div { margin-bottom: 20px; } #combined_info_modal_content h3 { font-size: 18px; margin-bottom: 10px; border-bottom: 1px solid rgba(255,255,255,0.1); padding-bottom: 5px; color: var(--highlight-color); }#manage_action_dropdown_menu{max-height:200px;overflow-y:auto;overflow-x:hidden;}`);

    // ==============================
    // CONSTANTS & CONFIG
    // ==============================
    const OUTFIELD_PLAYERS_SELECTOR = '.fieldpos.fieldpos-ok.ui-draggable:not(.substitute):not(.goalkeeper):not(.substitute.goalkeeper), .fieldpos.fieldpos-collision.ui-draggable:not(.substitute):not(.goalkeeper):not(.substitute.goalkeeper)';
    const GOALKEEPER_SELECTOR = '.fieldpos.fieldpos-ok.goalkeeper.ui-draggable';
    const FORMATION_TEXT_SELECTOR = '#formation_text';
    const TACTIC_SLOT_SELECTOR = '.ui-state-default.ui-corner-top.ui-tabs-selected.ui-state-active.invalid';
    const MIN_OUTFIELD_PLAYERS = 10;
    const MAX_TACTIC_NAME_LENGTH = 50;
    const 中国地区 = ['CN', 'HK', 'MO', 'TW'];
    const CDN_URLS = {
        default: {
            tactics: 'https://u18mz.vercel.app/mz/userscript/tactics/json/defaultTactics.json',
            lang: 'https://u18mz.vercel.app/mz/userscript/tactics/json/lang/'
        },
        china: {
            tactics: 'https://pub-02de1c06eac643f992bb26daeae5c7a0.r2.dev/json/defaultTactics.json',
            lang: 'https://pub-02de1c06eac643f992bb26daeae5c7a0.r2.dev/json/lang/'
        }
    };
    const BASE_FLAG_URL = 'https://flagcdn.com/w320/';
    const LANGUAGES = [
        { code: 'en', name: 'English', flag: BASE_FLAG_URL + 'gb.png' }, { code: 'pt', name: 'Português', flag: BASE_FLAG_URL + 'br.png' }, { code: 'zh', name: '中文', flag: BASE_FLAG_URL + 'cn.png' }, { code: 'sv', name: 'Svenska', flag: BASE_FLAG_URL + 'se.png' }, { code: 'no', name: 'Norsk', flag: BASE_FLAG_URL + 'no.png' }, { code: 'da', name: 'Dansk', flag: BASE_FLAG_URL + 'dk.png' }, { code: 'es', name: 'Español', flag: BASE_FLAG_URL + 'ar.png' }, { code: 'pl', name: 'Polski', flag: BASE_FLAG_URL + 'pl.png' }, { code: 'nl', name: 'Nederlands', flag: BASE_FLAG_URL + 'nl.png' }, { code: 'id', name: 'Bahasa Indonesia', flag: BASE_FLAG_URL + 'id.png' }, { code: 'de', name: 'Deutsch', flag: BASE_FLAG_URL + 'de.png' }, { code: 'it', name: 'Italiano', flag: BASE_FLAG_URL + 'it.png' }, { code: 'fr', name: 'Français', flag: BASE_FLAG_URL + 'fr.png' }, { code: 'ro', name: 'Română', flag: BASE_FLAG_URL + 'ro.png' }, { code: 'tr', name: 'Türkçe', flag: BASE_FLAG_URL + 'tr.png' }, { code: 'ko', name: '한국어', flag: BASE_FLAG_URL + 'kr.png' }, { code: 'ru', name: 'Русский', flag: BASE_FLAG_URL + 'ru.png' }, { code: 'ar', name: 'العربية', flag: BASE_FLAG_URL + 'sa.png' }, { code: 'jp', name: '日本語', flag: BASE_FLAG_URL + 'jp.png' }
    ];
    const SCRIPT_VERSION = '12.1.1';
    const DISPLAY_VERSION = '12';
    const VERSION_KEY = 'mz_tactics_version';
    const COLLAPSED_KEY = 'mz_tactics_collapsed';
    const CATEGORIES_STORAGE_KEY = 'mz_tactics_categories';
    const TACTICS_STORAGE_KEY = 'ls_tactics';
    const DEFAULT_CATEGORIES = {
        'short_passing': { id: 'short_passing', name: 'Short Passing', color: '#54a0ff' },
        'wing_play': { id: 'wing_play', name: 'Wing Play', color: '#5dd39e' }
    };
    const NEW_CATEGORY_ID = 'new_category';
    const OTHER_CATEGORY_ID = 'other';
    const USERSCRIPT_STRINGS = {
        addButton: 'Add', addCurrentTactic: 'Add Current', addWithXmlButton: 'Add via XML', manageButton: 'Manage', deleteButton: 'Delete', renameButton: 'Edit', updateButton: 'Save Positions', clearButton: 'Clear All', resetButton: 'Default', importButton: 'Import', exportButton: 'Export', infoButton: '❔', usefulLinksButton: 'Links', aboutButton: 'About', tacticNamePrompt: 'Please enter a name for the tactic', addAlert: 'Tactic {} added successfully.', deleteAlert: 'Tactic {} deleted successfully.', renameAlert: 'Tactic {} successfully edited.', updateAlert: 'Tactic {} updated successfully.', clearAlert: 'Tactics cleared successfully.', resetAlert: 'Default tactics loaded.', importAlert: 'Tactics imported successfully.', exportAlert: 'Tactics copied to clipboard.', deleteConfirmation: 'Do you really want to delete {}?', updateConfirmation: 'Do you really want to update {}?', clearConfirmation: 'Do you really want to clear all saved tactics?', resetConfirmation: 'Reset to default tactics? This will remove all your custom tactics.', invalidTacticError: 'Invalid tactic. Ensure 11 players are on the pitch.', noTacticNameProvidedError: 'No tactic name provided.', alreadyExistingTacticNameError: 'Tactic name already exists.', tacticNameMaxLengthError: 'Tactic name is too long (max 50 chars).', noTacticSelectedError: 'No tactic selected.', duplicateTacticError: 'This formation already exists.', noChangesMadeError: 'No changes detected in player positions.', invalidImportError: 'Invalid import data. Please provide valid JSON.', modalContentInfoText: 'Manage your tactics efficiently.', modalContentFeedbackText: 'For feedback or suggestions, contact <b>douglaskampl</b> via GB/Chat.', usefulContent: 'Community Resources:', tacticsDropdownMenuLabel: 'Select a tactic:', languageDropdownMenuLabel: 'Language:', errorTitle: 'Error', doneTitle: 'Success', confirmationTitle: 'Confirmation', deleteTacticConfirmButton: 'Delete', cancelConfirmButton: 'Cancel', updateConfirmButton: 'Update', clearTacticsConfirmButton: 'Clear All', resetTacticsConfirmButton: 'Reset', addConfirmButton: 'Add', xmlValidationError: 'Invalid XML format.', xmlParsingError: 'Error parsing XML.', xmlPlaceholder: 'Paste XML here', tacticNamePlaceholder: 'Tactic name', managerTitle: 'MZ Tactics Manager', tacticActionsTitle: 'Actions', otherActionsTitle: 'Other', searchPlaceholder: 'Search...', allTacticsFilter: 'All', selectTacticButton: 'Select', openTacticsSelector: 'Browse Tactics', noTacticsFound: 'No tactics found', welcomeMessage: `Welcome to MZ Tactics Manager v${SCRIPT_VERSION}!\n\nThis version includes:\n• New UI for better space usage.\n\nEnjoy managing your tactics!`, welcomeGotIt: 'Got it!', removeCategoryConfirmation: 'Remove category "{}"? All tactics in this category will be moved to "Other".', removeCategoryAlert: 'Category "{}" removed successfully.', removeCategoryButton: 'Remove'
    };
    const ELEMENT_STRING_KEYS = {
        delete_tactic_button: 'deleteButton', rename_tactic_button: 'renameButton', update_tactic_button: 'updateButton', tactics_dropdown_menu_label: 'tacticsDropdownMenuLabel', language_dropdown_menu_label: 'languageDropdownMenuLabel', info_modal_info_text: 'modalContentInfoText', info_modal_feedback_text: 'modalContentFeedbackText',
    };
    const DEFAULT_MODAL_STRINGS = {
        ok: 'OK', cancel: 'Cancel', error: 'Error', close: '×'
    };

    // ==============================
    // GLOBAL VARIABLES
    // ==============================
    const region = isLikelyFromChina() ? 'china' : 'default';
    const defaultTacticsDataUrl = CDN_URLS[region].tactics;
    const langDataBaseUrl = CDN_URLS[region].lang;
    let dropdownMenuTactics = [];
    let activeLanguage;
    let currentFilter = 'all';
    let searchTerm = '';
    let categories = {};
    let activeDropdownMenu = null;

    // ==============================
    // MODAL & ALERT FUNCTIONS
    // ==============================
    function createModalIcon(type) {
        if (!type) return null;
        const icon = document.createElement('div');
        icon.classList.add('mz-modal-icon');
        if (type === 'success') {
            icon.classList.add('success');
            icon.innerHTML = '✓';
        } else if (type === 'error') {
            icon.classList.add('error');
            icon.innerHTML = '✗';
        }
        return icon;
    }

    function validateModalInput(input, validator, errorContainerId) {
        if (!validator) return null;
        const validationError = validator(input.value);
        const errorContainer = document.getElementById(errorContainerId) || document.createElement('div');
        errorContainer.id = errorContainerId;
        errorContainer.style.color = '#ff6b6b';
        errorContainer.style.marginTop = '-10px';
        errorContainer.style.marginBottom = '10px';
        errorContainer.style.fontSize = '13px';
        const existingError = document.getElementById(errorContainerId);
        if (existingError) existingError.remove();
        if (!validationError) {
            return null;
        }
        errorContainer.textContent = validationError;
        if (!existingError) {
            input.parentNode.insertBefore(errorContainer, input.nextSibling);
        }
        return validationError;
    }

    function closeModal(overlay, callback) {
        overlay.classList.remove('active');
        setTimeout(() => {
            if (overlay.parentNode === document.body) {
                document.body.removeChild(overlay);
            }
            if (callback) callback();
        }, 300);
    }

    function handleAlertConfirm(options, input, categorySelector, newCategoryInput, overlay, resolve) {
        if (options.input === 'text' && options.inputValidator) {
            const hasError = validateModalInput(input, options.inputValidator, 'mz-modal-error');
            if (hasError) return;
        }
        let categoryValue = null;
        let newCategoryName = null;
        if (categorySelector) {
            categoryValue = categorySelector.value;
            if (categoryValue === NEW_CATEGORY_ID && newCategoryInput) {
                newCategoryName = newCategoryInput.value.trim();
                const categoryErrorContainer = document.getElementById('new-category-error');
                if (categoryErrorContainer) categoryErrorContainer.remove();
                if (!newCategoryName) {
                    const errorText = document.createElement('div');
                    errorText.style.color = '#ff6b6b';
                    errorText.style.marginTop = '5px';
                    errorText.style.fontSize = '13px';
                    errorText.textContent = 'Category name cannot be empty';
                    errorText.id = 'new-category-error';
                    newCategoryInput.parentNode.appendChild(errorText);
                    return;
                }
                const existingCategory = Object.values(categories).find(
                    cat => cat.name.toLowerCase() === newCategoryName.toLowerCase()
                );
                if (existingCategory) {
                    const errorText = document.createElement('div');
                    errorText.style.color = '#ff6b6b';
                    errorText.style.marginTop = '5px';
                    errorText.style.fontSize = '13px';
                    errorText.textContent = 'This category already exists';
                    errorText.id = 'new-category-error';
                    newCategoryInput.parentNode.appendChild(errorText);
                    return;
                }
            }
        }
        closeModal(overlay, () => {
            if (options.input === 'text') {
                const result = { value: input ? input.value : null, isConfirmed: true };
                if (categorySelector) {
                    if (categoryValue === NEW_CATEGORY_ID && newCategoryName) {
                        const newCategoryId = generateCategoryId(newCategoryName);
                        const newCategory = {
                            id: newCategoryId,
                            name: newCategoryName,
                            color: generateCategoryColor(newCategoryName)
                        };
                        result.category = newCategory;
                        addCategory(newCategory);
                    } else {
                        result.category = categories[categoryValue] || { id: OTHER_CATEGORY_ID, name: 'Other', color: '#ffcb77' };
                    }
                }
                resolve(result);
            } else {
                resolve({ isConfirmed: true });
            }
        });
    }

    function handleAlertCancel(overlay, resolve) {
        closeModal(overlay, () => {
            resolve({ isConfirmed: false, value: null });
        });
    }

    function setUpKeyboardHandler(handleConfirm, handleCancel, input) {
        return function (e) {
            if (e.key === 'Escape') {
                handleCancel();
            } else if (e.key === 'Enter' && !(input && document.activeElement !== input)) {
                if (input && input.tagName === 'TEXTAREA') {} else {
                     handleConfirm();
                }
            }
        };
    }

    function showAlert(options) {
        return new Promise((resolve) => {
            const overlay = document.createElement('div');
            overlay.id = 'mz-modal-overlay';
            const container = document.createElement('div');
            container.id = 'mz-modal-container';
            const header = document.createElement('div');
            header.id = 'mz-modal-header';
            const titleContainer = document.createElement('div');
            titleContainer.classList.add('mz-modal-title-with-icon');
            const icon = createModalIcon(options.type);
            if (icon) titleContainer.appendChild(icon);
            const title = document.createElement('h2');
            title.id = 'mz-modal-title';
            title.textContent = options.title || '';
            titleContainer.appendChild(title);
            header.appendChild(titleContainer);
            const closeBtn = document.createElement('button');
            closeBtn.id = 'mz-modal-close';
            closeBtn.innerHTML = DEFAULT_MODAL_STRINGS.close;
            header.appendChild(closeBtn);
            const content = document.createElement('div');
            content.id = 'mz-modal-content';
            if (options.htmlContent) {
                 content.appendChild(options.htmlContent);
            } else {
                 content.textContent = options.text || '';
            }
            let input;
            let categorySelector;
            let newCategoryInput;
            if (options.input === 'text') {
                input = document.createElement('input');
                input.id = 'mz-modal-input';
                input.type = 'text';
                input.value = options.inputValue || '';
                input.placeholder = options.placeholder || '';
            }
            if (options.showCategorySelector) {
                const categoryContainer = document.createElement('div');
                categoryContainer.className = 'category-selection-container';
                const categoryLabel = document.createElement('label');
                categoryLabel.className = 'category-selection-label';
                categoryLabel.textContent = 'Category:';
                categoryContainer.appendChild(categoryLabel);
                categorySelector = document.createElement('select');
                categorySelector.id = 'category-selector';

                const usedCategoryIds = new Set(dropdownMenuTactics.map(t => t.style).filter(Boolean));
                if (options.currentCategory) {
                    usedCategoryIds.add(options.currentCategory);
                }

                const availableCategories = Object.values(categories).filter(cat =>
                     cat.id === 'short_passing' ||
                     cat.id === 'wing_play' ||
                     cat.id === OTHER_CATEGORY_ID ||
                     usedCategoryIds.has(cat.id)
                );

                availableCategories.sort((a, b) => {
                    if (a.id === OTHER_CATEGORY_ID) return 1;
                    if (b.id === OTHER_CATEGORY_ID) return -1;
                    return a.name.localeCompare(b.name);
                });

                availableCategories.forEach(category => {
                    if (category.id !== OTHER_CATEGORY_ID) {
                        const option = document.createElement('option');
                        option.value = category.id;
                        option.textContent = category.name;
                        categorySelector.appendChild(option);
                    }
                });

                const otherOption = document.createElement('option');
                otherOption.value = OTHER_CATEGORY_ID;
                otherOption.textContent = getCategoryName(OTHER_CATEGORY_ID);
                categorySelector.appendChild(otherOption);

                const addNewOption = document.createElement('option');
                addNewOption.value = NEW_CATEGORY_ID;
                addNewOption.textContent = '+ New category';
                categorySelector.appendChild(addNewOption);

                if (options.currentCategory && categories[options.currentCategory]) {
                     categorySelector.value = options.currentCategory;
                } else {
                     categorySelector.value = OTHER_CATEGORY_ID;
                }

                categorySelector.addEventListener('change', function() {
                    const newCategoryContainer = document.querySelector('.new-category-input-container');
                    if (this.value === NEW_CATEGORY_ID) {
                        newCategoryContainer.classList.add('visible');
                        newCategoryInput.focus();
                    } else {
                        newCategoryContainer.classList.remove('visible');
                        const categoryErrorContainer = document.getElementById('new-category-error');
                        if (categoryErrorContainer) categoryErrorContainer.remove();
                    }
                });
                categoryContainer.appendChild(categorySelector);
                const newCategoryContainer = document.createElement('div');
                newCategoryContainer.className = 'new-category-input-container';
                newCategoryInput = document.createElement('input');
                newCategoryInput.id = 'new-category-input';
                newCategoryInput.type = 'text';
                newCategoryInput.placeholder = 'New category name';
                newCategoryContainer.appendChild(newCategoryInput);
                categoryContainer.appendChild(newCategoryContainer);
                if (content.textContent || content.hasChildNodes()) {
                     categoryContainer.style.marginTop = '15px';
                }
                content.appendChild(categoryContainer);
            }
            const buttons = document.createElement('div');
            buttons.id = 'mz-modal-buttons';
            const confirmHandler = () => {
                handleAlertConfirm(options, input, categorySelector, newCategoryInput, overlay, resolve);
            };
            const cancelHandler = () => handleAlertCancel(overlay, resolve);
            const confirmBtn = document.createElement('button');
            confirmBtn.classList.add('mz-modal-btn', 'primary');
            confirmBtn.textContent = options.confirmButtonText || DEFAULT_MODAL_STRINGS.ok;
            confirmBtn.addEventListener('click', confirmHandler);
            buttons.appendChild(confirmBtn);
            if (options.showCancelButton) {
                const cancelBtn = document.createElement('button');
                cancelBtn.classList.add('mz-modal-btn', 'cancel');
                cancelBtn.textContent = options.cancelButtonText || DEFAULT_MODAL_STRINGS.cancel;
                cancelBtn.addEventListener('click', cancelHandler);
                buttons.appendChild(cancelBtn);
            }
            closeBtn.addEventListener('click', cancelHandler);
            const keydownHandler = setUpKeyboardHandler(confirmHandler, cancelHandler, input);
            document.addEventListener('keydown', keydownHandler);
            container.appendChild(header);
            container.appendChild(content);
            if (input) container.appendChild(input);
            container.appendChild(buttons);
            overlay.appendChild(container);
            document.body.appendChild(overlay);
            setTimeout(() => {
                overlay.classList.add('active');
                if (input) input.focus();
                if (categorySelector && categorySelector.value === NEW_CATEGORY_ID) {
                    newCategoryInput.focus();
                }
            }, 10);
            overlay.addEventListener('transitionend', () => {
                if (!overlay.classList.contains('active')) {
                    document.removeEventListener('keydown', keydownHandler);
                }
            });
        });
    }

    function showSuccessMessage(title, text) {
        return showAlert({
            title: title || USERSCRIPT_STRINGS.doneTitle,
            text: text,
            type: 'success'
        });
    }

    function showErrorMessage(title, text) {
        return showAlert({
            title: title || USERSCRIPT_STRINGS.errorTitle,
            text: text,
            type: 'error'
        });
    }

    function showWelcomeMessage() {
        return showAlert({
            title: 'Welcome!',
            text: USERSCRIPT_STRINGS.welcomeMessage,
            confirmButtonText: USERSCRIPT_STRINGS.welcomeGotIt
        });
    }

    // ==============================
    // UTILITY FUNCTIONS
    // ==============================
    function isFootball() {
        const element = document.querySelector('div#tactics_box.soccer.clearfix');
        return !!element;
    }

    function sha256Hash(str) {
        const shaObj = new jsSHA('SHA-256', 'TEXT');
        shaObj.update(str);
        return shaObj.getHash('HEX');
    }

    function insertAfterElement(something, element) {
        element.parentNode.insertBefore(something, element.nextSibling);
    }

    function appendChildren(parent, children) {
        children.forEach((ch) => {
             if (ch) parent.appendChild(ch);
        });
    }

    function isLikelyFromChina() {
        const lang = navigator.language || navigator.userLanguage || ''; const ua = navigator.userAgent.toLowerCase(); const region = navigator.language?.split('-')[1]?.toUpperCase() || '';
        return lang.startsWith('zh-') || ua.includes('micromessenger') || ua.includes('qq') || ua.includes('ucbrowser') || 中国地区.includes(region);
    }

    // ==============================
    // STORAGE & DATA HANDLING
    // ==============================
    async function fetchTacticsFromGMStorage() {
        const storedTactics = GM_getValue(TACTICS_STORAGE_KEY);
        if (storedTactics) {
            return storedTactics;
        } else {
            const jsonTactics = await fetchTacticsFromJson();
            storeTacticsInGMStorage(jsonTactics);
            return jsonTactics;
        }
    }

    async function fetchTacticsFromJson() {
        try {
            const response = await fetch(defaultTacticsDataUrl);
            if (!response.ok) {
                throw new Error();
            }
            return await response.json();
        } catch (_e) {
            const fallbackURL = (defaultTacticsDataUrl === CDN_URLS.default.tactics) ? CDN_URLS.china.tactics : CDN_URLS.default.tactics;
            const fallbackResponse = await fetch(fallbackURL);
            return await fallbackResponse.json();
        }
    }

    function storeTacticsInGMStorage(data) {
        GM_setValue(TACTICS_STORAGE_KEY, data);
    }

    async function validateDuplicateTactic(id) {
        const tacticsData = (await GM_getValue(TACTICS_STORAGE_KEY)) || { tactics: [] };
        return tacticsData.tactics.some((tactic) => tactic.id === id);
    }

    async function saveTacticToStorage(tactic) {
        const tacticsData = (await GM_getValue(TACTICS_STORAGE_KEY)) || { tactics: [] };
        tacticsData.tactics.push(tactic);
        await GM_setValue(TACTICS_STORAGE_KEY, tacticsData);
    }

    async function validateDuplicateTacticWithUpdatedCoord(newId, selectedTac, tacticsData) {
        if (newId === selectedTac.id) {
            return 'unchanged';
        } else if (tacticsData.tactics.some((tac) => tac.id === newId)) {
            return 'duplicate';
        } else {
            return 'unique';
        }
    }

    // ==============================
    // TACTIC/FORMATION LOGIC
    // ==============================
    function generateUniqueId(coordinates) {
        coordinates.sort((a, b) => { if (a[1] !== b[1]) { return a[1] - b[1]; } else { return a[0] - b[0]; } });
        const coordString = coordinates.map(coord => `${coord[0]},${coord[1]}`).join(';');
        return sha256Hash(coordString);
    }

    function handleTacticsSelection(tacticName) {
        if (!tacticName) return;
        const outfieldPlayers = Array.from(document.querySelectorAll(OUTFIELD_PLAYERS_SELECTOR));
        const selectedTactic = dropdownMenuTactics.find((tacticData) => tacticData.name === tacticName);
        if (selectedTactic) {
            if (outfieldPlayers.length < MIN_OUTFIELD_PLAYERS) {
                const hiddenTriggerButton = document.getElementById('hidden_trigger_button');
                if(hiddenTriggerButton) hiddenTriggerButton.click();
                setTimeout(() => rearrangePlayers(selectedTactic.coordinates), 1);
            } else {
                rearrangePlayers(selectedTactic.coordinates);
            }
        }
    }

    function rearrangePlayers(coordinates) {
        const outfieldPlayers = Array.from(document.querySelectorAll(OUTFIELD_PLAYERS_SELECTOR));
        findBestPositions(outfieldPlayers, coordinates);
        for (let i = 0; i < outfieldPlayers.length; ++i) {
            if (coordinates[i]) {
                outfieldPlayers[i].style.left = coordinates[i][0] + 'px';
                outfieldPlayers[i].style.top = coordinates[i][1] + 'px';
                removeCollision(outfieldPlayers[i]);
            }
        }
        removeTacticSlotInvalidStatus();
        updateFormationText(getFormation(coordinates));
    }

    function findBestPositions(players, coordinates) {
        players.sort((a, b) => parseInt(a.style.top) - parseInt(b.style.top));
        coordinates.sort((a, b) => a[1] - b[1]);
    }

    function removeCollision(player) {
        if (player.classList.contains('fieldpos-collision')) {
            player.classList.remove('fieldpos-collision');
            player.classList.add('fieldpos-ok');
        }
    }

    function removeTacticSlotInvalidStatus() {
        const slot = document.querySelector(TACTIC_SLOT_SELECTOR);
        if (slot) {
            slot.classList.remove('invalid');
        }
    }

    function updateFormationText(formation) {
        const formationTextElement = document.querySelector(FORMATION_TEXT_SELECTOR);
        if (formationTextElement) {
            const defs = formationTextElement.querySelector('.defs');
            const mids = formationTextElement.querySelector('.mids');
            const atts = formationTextElement.querySelector('.atts');
            if (defs) defs.textContent = formation.defenders;
            if (mids) mids.textContent = formation.midfielders;
            if (atts) atts.textContent = formation.strikers;
        }
    }

    function getFormation(coordinates) {
        let strikers = 0;
        let midfielders = 0;
        let defenders = 0;
        for (const coo of coordinates) {
            const y = coo[1];
            if (y < 103) {
                strikers++;
            } else if (y <= 204) {
                midfielders++;
            } else {
                defenders++;
            }
        }
        return { strikers, midfielders, defenders };
    }

    function validateTacticPlayerCount(outfieldPlayers) {
        const isGoalkeeper = document.querySelector(GOALKEEPER_SELECTOR);
        outfieldPlayers = outfieldPlayers.filter((player) => !player.classList.contains('fieldpos-collision'));
        if (outfieldPlayers.length < MIN_OUTFIELD_PLAYERS || !isGoalkeeper) {
            showErrorMessage(USERSCRIPT_STRINGS.errorTitle, USERSCRIPT_STRINGS.invalidTacticError);
            return false;
        }
        return true;
    }

    // ==============================
    // CATEGORY MANAGEMENT
    // ==============================
    function generateCategoryId(categoryName) {
        return sha256Hash(categoryName.toLowerCase()).substring(0, 10);
    }

    function generateCategoryColor(categoryName) {
        const hash = sha256Hash(categoryName);
        const hue = parseInt(hash.substring(0, 6), 16) % 360;
        const saturation = 50 + (parseInt(hash.substring(6, 8), 16) % 30);
        const lightness = 55 + (parseInt(hash.substring(8, 10), 16) % 15);
        return `hsl(${hue}, ${saturation}%, ${lightness}%)`;
    }

    function addCategory(category) {
        categories[category.id] = category;
        saveCategories();
    }

    function saveCategories() {
        GM_setValue(CATEGORIES_STORAGE_KEY, categories);
    }

    function loadCategories() {
        const storedCategories = GM_getValue(CATEGORIES_STORAGE_KEY);
        if (storedCategories && typeof storedCategories === 'object') {
            categories = storedCategories;
            if (!categories['short_passing']) categories['short_passing'] = DEFAULT_CATEGORIES['short_passing'];
            if (!categories['wing_play']) categories['wing_play'] = DEFAULT_CATEGORIES['wing_play'];
        } else {
            categories = { ...DEFAULT_CATEGORIES };
            saveCategories();
        }
        if (!categories[OTHER_CATEGORY_ID]) {
            categories[OTHER_CATEGORY_ID] = { id: OTHER_CATEGORY_ID, name: 'Other', color: '#ffcb77' };
        }
    }

    function loadCategoryColor(categoryId) {
        if (categories[categoryId]) {
            return categories[categoryId].color;
        } else if (categoryId === 'short_passing') {
            return DEFAULT_CATEGORIES.short_passing.color;
        } else if (categoryId === 'wing_play') {
            return DEFAULT_CATEGORIES.wing_play.color;
        } else if (categoryId === 'other' || !categoryId) {
            return '#ffcb77';
        } else {
            return '#8395a7';
        }
    }

    function getCategoryName(categoryId) {
        if (categories[categoryId]) {
            return categories[categoryId].name;
        } else if (categoryId === 'short_passing') {
            return 'Short Passing';
        } else if (categoryId === 'wing_play') {
            return 'Wing Play';
        } else if (categoryId === OTHER_CATEGORY_ID || !categoryId) {
            return 'Other';
        } else {
            return categoryId || 'Uncategorized';
        }
    }

    async function removeCategory(categoryId) {
        if (!categoryId || categoryId === 'all' || categoryId === OTHER_CATEGORY_ID) {
            console.error("Attempted to remove non-removable category:", categoryId);
            return;
        }

        const categoryName = getCategoryName(categoryId);
        const confirmResult = await showAlert({
            title: USERSCRIPT_STRINGS.confirmationTitle,
            text: USERSCRIPT_STRINGS.removeCategoryConfirmation.replace('{}', categoryName),
            showCancelButton: true,
            confirmButtonText: USERSCRIPT_STRINGS.removeCategoryButton,
            cancelButtonText: USERSCRIPT_STRINGS.cancelConfirmButton,
            type: 'error'
        });

        if (!confirmResult.isConfirmed) {
            return;
        }

        const tacticsData = await GM_getValue(TACTICS_STORAGE_KEY, { tactics: [] });
        let tacticsUpdated = false;

        tacticsData.tactics = tacticsData.tactics.map(tactic => {
            if (tactic.style === categoryId) {
                tactic.style = OTHER_CATEGORY_ID;
                tacticsUpdated = true;
            }
            return tactic;
        });

        dropdownMenuTactics = dropdownMenuTactics.map(tactic => {
            if (tactic.style === categoryId) {
                tactic.style = OTHER_CATEGORY_ID;
            }
            return tactic;
        });

        if (tacticsUpdated) {
            await GM_setValue(TACTICS_STORAGE_KEY, tacticsData);
        }

        if (categoryId !== 'short_passing' && categoryId !== 'wing_play') {
            delete categories[categoryId];
            saveCategories();
        }

        if (currentFilter === categoryId) {
            currentFilter = 'all';
        }

        updateTacticsDropdown();
        updateFilterTabs();
        await showSuccessMessage(USERSCRIPT_STRINGS.doneTitle, USERSCRIPT_STRINGS.removeCategoryAlert.replace('{}', categoryName));
    }

    // ==============================
    // TACTIC ACTIONS (ADD, DELETE, RENAME, UPDATE, etc.)
    // ==============================
    async function addNewTactic() {
        const outfieldPlayers = Array.from(document.querySelectorAll(OUTFIELD_PLAYERS_SELECTOR));
        const tacticCoordinates = outfieldPlayers.map((player) => [parseInt(player.style.left), parseInt(player.style.top)]);
        if (!validateTacticPlayerCount(outfieldPlayers)) {
            return;
        }
        const tacticId = generateUniqueId(tacticCoordinates);
        const isDuplicate = await validateDuplicateTactic(tacticId);
        if (isDuplicate) {
            await showErrorMessage(USERSCRIPT_STRINGS.errorTitle, USERSCRIPT_STRINGS.duplicateTacticError);
            return;
        }
        const result = await showAlert({
            title: USERSCRIPT_STRINGS.tacticNamePrompt, input: 'text', inputValue: '', placeholder: USERSCRIPT_STRINGS.tacticNamePlaceholder,
            inputValidator: (value) => {
                if (!value) return USERSCRIPT_STRINGS.noTacticNameProvidedError;
                if (value.length > MAX_TACTIC_NAME_LENGTH) return USERSCRIPT_STRINGS.tacticNameMaxLengthError;
                if (dropdownMenuTactics.some((t) => t.name === value)) return USERSCRIPT_STRINGS.alreadyExistingTacticNameError;
                return null;
            },
            showCategorySelector: true, showCancelButton: true, confirmButtonText: USERSCRIPT_STRINGS.addConfirmButton, cancelButtonText: USERSCRIPT_STRINGS.cancelConfirmButton
        });
        if (!result.isConfirmed || !result.value) {
            return;
        }
        const tacticName = result.value;
        const tacticCategory = result.category.id;
        const tactic = { name: tacticName, coordinates: tacticCoordinates, id: tacticId, style: tacticCategory };
        await saveTacticToStorage(tactic);
        dropdownMenuTactics.push(tactic);
        dropdownMenuTactics.sort((a, b) => a.name.localeCompare(b.name));
        updateTacticsDropdown();
        updateFilterTabs();
        const tacticsSelector = document.getElementById('tactics_selector');
        tacticsSelector.value = tactic.name;
        handleTacticsSelection(tactic.name);
        await showSuccessMessage(USERSCRIPT_STRINGS.doneTitle, USERSCRIPT_STRINGS.addAlert.replace('{}', tactic.name));
    }

    async function addNewTacticWithXml() {
        const xmlResult = await showAlert({
            title: USERSCRIPT_STRINGS.xmlPlaceholder, input: 'text', showCancelButton: true, confirmButtonText: USERSCRIPT_STRINGS.addConfirmButton, cancelButtonText: USERSCRIPT_STRINGS.cancelConfirmButton
        });
        if (!xmlResult.isConfirmed || !xmlResult.value) {
            return;
        }
        const xml = xmlResult.value;
        const nameResult = await showAlert({
            title: USERSCRIPT_STRINGS.tacticNamePrompt, input: 'text', placeholder: USERSCRIPT_STRINGS.tacticNamePlaceholder,
            inputValidator: (value) => {
                if (!value) return USERSCRIPT_STRINGS.noTacticNameProvidedError;
                if (value.length > MAX_TACTIC_NAME_LENGTH) return USERSCRIPT_STRINGS.tacticNameMaxLengthError;
                if (dropdownMenuTactics.some((t) => t.name === value)) return USERSCRIPT_STRINGS.alreadyExistingTacticNameError;
                return null;
            },
            showCategorySelector: true, showCancelButton: true, confirmButtonText: USERSCRIPT_STRINGS.addConfirmButton, cancelButtonText: USERSCRIPT_STRINGS.cancelConfirmButton
        });
        if (!nameResult.isConfirmed || !nameResult.value) {
            return;
        }
        const name = nameResult.value;
        const category = nameResult.category.id;
        try {
            const newTactic = await convertXmlToTacticJson(xml, name);
            newTactic.style = category;
            const tacticId = generateUniqueId(newTactic.coordinates);
            const isDuplicate = await validateDuplicateTactic(tacticId);
            if (isDuplicate) {
                await showErrorMessage(USERSCRIPT_STRINGS.errorTitle, USERSCRIPT_STRINGS.duplicateTacticError);
                return;
            }
            newTactic.id = tacticId;
            await saveTacticToStorage(newTactic);
            dropdownMenuTactics.push(newTactic);
            dropdownMenuTactics.sort((a, b) => a.name.localeCompare(b.name));
            updateTacticsDropdown();
            updateFilterTabs();
            const tacticsSelector = document.getElementById('tactics_selector');
            tacticsSelector.value = newTactic.name;
            handleTacticsSelection(newTactic.name);
            await showSuccessMessage(USERSCRIPT_STRINGS.doneTitle, USERSCRIPT_STRINGS.addAlert.replace('{}', newTactic.name));
        } catch (e) {
            console.error('XMLError:', e);
            await showErrorMessage(USERSCRIPT_STRINGS.errorTitle, USERSCRIPT_STRINGS.xmlParsingError + (e.message ? `: ${e.message}` : ''));
        }
    }

    async function deleteTactic() {
        const tacticsSelector = document.getElementById('tactics_selector');
        const selectedTactic = dropdownMenuTactics.find((tactic) => tactic.name === tacticsSelector.value);
        if (!selectedTactic) {
            await showErrorMessage(USERSCRIPT_STRINGS.errorTitle, USERSCRIPT_STRINGS.noTacticSelectedError);
            return;
        }
        const confirmResult = await showAlert({
            title: USERSCRIPT_STRINGS.confirmationTitle, text: USERSCRIPT_STRINGS.deleteConfirmation.replace('{}', selectedTactic.name), showCancelButton: true, confirmButtonText: USERSCRIPT_STRINGS.deleteTacticConfirmButton, cancelButtonText: USERSCRIPT_STRINGS.cancelConfirmButton
        });
        if (!confirmResult.isConfirmed) {
            return;
        }
        const deletedCategoryId = selectedTactic.style;
        const tacticsData = (await GM_getValue(TACTICS_STORAGE_KEY)) || { tactics: [] };
        tacticsData.tactics = tacticsData.tactics.filter((tactic) => tactic.id !== selectedTactic.id);
        await GM_setValue(TACTICS_STORAGE_KEY, tacticsData);
        dropdownMenuTactics = dropdownMenuTactics.filter((tactic) => tactic.id !== selectedTactic.id);
        const categoryStillHasTactics = dropdownMenuTactics.some(tactic => tactic.style === deletedCategoryId);
        if (!categoryStillHasTactics && deletedCategoryId !== 'short_passing' && deletedCategoryId !== 'wing_play' && deletedCategoryId !== OTHER_CATEGORY_ID) {
             delete categories[deletedCategoryId];
             saveCategories();
             if (currentFilter === deletedCategoryId) {
                  currentFilter = 'all';
             }
        }

        updateTacticsDropdown();
        updateFilterTabs();
        await showSuccessMessage(USERSCRIPT_STRINGS.doneTitle, USERSCRIPT_STRINGS.deleteAlert.replace('{}', selectedTactic.name));
    }

    async function renameTactic() {
        const tacticsSelector = document.getElementById('tactics_selector');
        const selectedTactic = dropdownMenuTactics.find((tactic) => tactic.name === tacticsSelector.value);
        if (!selectedTactic) {
            await showErrorMessage(USERSCRIPT_STRINGS.errorTitle, USERSCRIPT_STRINGS.noTacticSelectedError);
            return;
        }
        const oldName = selectedTactic.name;
        const oldCategory = selectedTactic.style;
        const result = await showAlert({
            title: 'Edit Tactic', input: 'text', inputValue: oldName, placeholder: USERSCRIPT_STRINGS.tacticNamePlaceholder,
            inputValidator: (value) => {
                if (!value) return USERSCRIPT_STRINGS.noTacticNameProvidedError;
                if (value.length > MAX_TACTIC_NAME_LENGTH) return USERSCRIPT_STRINGS.tacticNameMaxLengthError;
                if (value !== oldName && dropdownMenuTactics.some((t) => t.name === value)) return USERSCRIPT_STRINGS.alreadyExistingTacticNameError;
                return null;
            },
            showCategorySelector: true, currentCategory: selectedTactic.style, showCancelButton: true, confirmButtonText: USERSCRIPT_STRINGS.updateConfirmButton, cancelButtonText: USERSCRIPT_STRINGS.cancelConfirmButton
        });
        if (!result.isConfirmed || (!result.value && !result.category) ) {
            return;
        }
        const newName = result.value || oldName;
        const newCategory = result.category?.id || oldCategory;
        if (newName === oldName && newCategory === oldCategory) {
            return;
        }
        const categoryChanged = oldCategory !== newCategory;
        const tacticsData = (await GM_getValue(TACTICS_STORAGE_KEY)) || { tactics: [] };
        tacticsData.tactics = tacticsData.tactics.map((tactic) => {
            if (tactic.id === selectedTactic.id) {
                tactic.name = newName;
                tactic.style = newCategory;
            }
            return tactic;
        });
        await GM_setValue(TACTICS_STORAGE_KEY, tacticsData);

        dropdownMenuTactics = dropdownMenuTactics.map((tactic) => {
            if (tactic.id === selectedTactic.id) {
                tactic.name = newName;
                tactic.style = newCategory;
            }
            return tactic;
        });

        if (categoryChanged) {
            const oldCategoryStillHasTactics = dropdownMenuTactics.some(tactic => tactic.style === oldCategory);
            if (!oldCategoryStillHasTactics && oldCategory !== 'short_passing' && oldCategory !== 'wing_play' && oldCategory !== OTHER_CATEGORY_ID) {
                 delete categories[oldCategory];
                 saveCategories();
                 if (currentFilter === oldCategory) {
                     currentFilter = 'all';
                 }
            }
        }

        dropdownMenuTactics.sort((a, b) => a.name.localeCompare(b.name));
        updateTacticsDropdown();
        updateFilterTabs();
        tacticsSelector.value = newName;
        await showSuccessMessage(USERSCRIPT_STRINGS.doneTitle, USERSCRIPT_STRINGS.renameAlert.replace('{}', newName));
    }

    async function updateTactic() {
        const outfieldPlayers = Array.from(document.querySelectorAll(OUTFIELD_PLAYERS_SELECTOR));
        const tacticsSelector = document.getElementById('tactics_selector');
        const selectedTactic = dropdownMenuTactics.find((tactic) => tactic.name === tacticsSelector.value);
        if (!selectedTactic) {
            await showErrorMessage(USERSCRIPT_STRINGS.errorTitle, USERSCRIPT_STRINGS.noTacticSelectedError);
            return;
        }
        if (!validateTacticPlayerCount(outfieldPlayers)) {
            return;
        }
        const updatedCoordinates = outfieldPlayers.map((player) => [parseInt(player.style.left), parseInt(player.style.top)]);
        const newId = generateUniqueId(updatedCoordinates);
        const tacticsData = (await GM_getValue(TACTICS_STORAGE_KEY)) || { tactics: [] };
        const validationOutcome = await validateDuplicateTacticWithUpdatedCoord(newId, selectedTactic, tacticsData);
        if (validationOutcome === 'unchanged') {
            await showErrorMessage(USERSCRIPT_STRINGS.errorTitle, USERSCRIPT_STRINGS.noChangesMadeError);
            return;
        } else if (validationOutcome === 'duplicate') {
            await showErrorMessage(USERSCRIPT_STRINGS.errorTitle, USERSCRIPT_STRINGS.duplicateTacticError);
            return;
        }
        const result = await showAlert({
            title: USERSCRIPT_STRINGS.confirmationTitle, text: USERSCRIPT_STRINGS.updateConfirmation.replace('{}', selectedTactic.name), showCancelButton: true, confirmButtonText: USERSCRIPT_STRINGS.updateConfirmButton, cancelButtonText: USERSCRIPT_STRINGS.cancelConfirmButton
        });
        if (!result.isConfirmed) {
            return;
        }
        for (const tactic of tacticsData.tactics) {
            if (tactic.id === selectedTactic.id) {
                tactic.coordinates = updatedCoordinates;
                tactic.id = newId;
            }
        }
        const memoryTactic = dropdownMenuTactics.find(t => t.id === selectedTactic.id);
        if (memoryTactic) {
             memoryTactic.coordinates = updatedCoordinates;
             memoryTactic.id = newId;
        }
        await GM_setValue(TACTICS_STORAGE_KEY, tacticsData);
        await showSuccessMessage(USERSCRIPT_STRINGS.doneTitle, USERSCRIPT_STRINGS.updateAlert.replace('{}', selectedTactic.name));
    }

    async function clearTactics() {
        const confirmResult = await showAlert({
            title: USERSCRIPT_STRINGS.confirmationTitle, text: USERSCRIPT_STRINGS.clearConfirmation, showCancelButton: true, confirmButtonText: USERSCRIPT_STRINGS.clearTacticsConfirmButton, cancelButtonText: USERSCRIPT_STRINGS.cancelConfirmButton, type: 'error'
        });
        if (!confirmResult.isConfirmed) {
            return;
        }
        await GM_deleteValue(TACTICS_STORAGE_KEY);
        dropdownMenuTactics = [];
        currentFilter = 'all';
        categories = { ...DEFAULT_CATEGORIES };
        if (!categories[OTHER_CATEGORY_ID]) {
            categories[OTHER_CATEGORY_ID] = { id: OTHER_CATEGORY_ID, name: 'Other', color: '#ffcb77' };
        }
        saveCategories();
        updateTacticsDropdown();
        updateFilterTabs();
        await showSuccessMessage(USERSCRIPT_STRINGS.doneTitle, USERSCRIPT_STRINGS.clearAlert);
    }

    async function resetTactics() {
        const confirmResult = await showAlert({
            title: USERSCRIPT_STRINGS.confirmationTitle, text: USERSCRIPT_STRINGS.resetConfirmation, showCancelButton: true, confirmButtonText: USERSCRIPT_STRINGS.resetTacticsConfirmButton, cancelButtonText: USERSCRIPT_STRINGS.cancelConfirmButton, type: 'error'
        });
        if (!confirmResult.isConfirmed) {
            return;
        }
        await GM_deleteValue(TACTICS_STORAGE_KEY);
        currentFilter = 'all';
        categories = { ...DEFAULT_CATEGORIES };
        if (!categories[OTHER_CATEGORY_ID]) {
            categories[OTHER_CATEGORY_ID] = { id: OTHER_CATEGORY_ID, name: 'Other', color: '#ffcb77' };
        }
        saveCategories();
        try {
            const response = await fetch(defaultTacticsDataUrl);
            if (!response.ok) {
                throw new Error();
            }
            const data = await response.json();
            const defaultTactics = data.tactics || [];
            defaultTactics.forEach(tactic => {
                if (!tactic.hasOwnProperty('style')) {
                    tactic.style = OTHER_CATEGORY_ID;
                }
            });
            await GM_setValue(TACTICS_STORAGE_KEY, { tactics: defaultTactics });
            dropdownMenuTactics = defaultTactics;
            dropdownMenuTactics.sort((a, b) => a.name.localeCompare(b.name));
        } catch (error) {
            console.log();
            try {
                 const fallbackURL = (defaultTacticsDataUrl === CDN_URLS.default.tactics) ? CDN_URLS.china.tactics : CDN_URLS.default.tactics;
                 const fallbackResponse = await fetch(fallbackURL);
                 if (!fallbackResponse.ok) {
                     throw new Error();
                 }
                 const fallbackData = await fallbackResponse.json();
                 const defaultTactics = fallbackData.tactics || [];
                 defaultTactics.forEach(tactic => {
                     if (!tactic.hasOwnProperty('style')) {
                         tactic.style = OTHER_CATEGORY_ID;
                     }
                 });
                 await GM_setValue(TACTICS_STORAGE_KEY, { tactics: defaultTactics });
                 dropdownMenuTactics = defaultTactics;
                 dropdownMenuTactics.sort((a, b) => a.name.localeCompare(b.name));
            } catch (e) {
                 console.error(e);
                 dropdownMenuTactics = [];
                 await GM_setValue(TACTICS_STORAGE_KEY, { tactics: [] });
                 await showErrorMessage('Error', 'Could not load default tactics.');
                 return;
            }
        }
        updateTacticsDropdown();
        updateFilterTabs();
        await showSuccessMessage(USERSCRIPT_STRINGS.doneTitle, USERSCRIPT_STRINGS.resetAlert);
    }

    async function importTactics() {
        try {
            const result = await showAlert({
                title: 'Import Tactics', input: 'text', inputValue: '', placeholder: 'Paste Tactics JSON here', showCancelButton: true, confirmButtonText: USERSCRIPT_STRINGS.importButton, cancelButtonText: USERSCRIPT_STRINGS.cancelConfirmButton
            });
            if (!result.isConfirmed || !result.value) {
                return;
            }
            let importedData;
            try {
                importedData = JSON.parse(result.value);
            } catch (e) {
                await showErrorMessage(USERSCRIPT_STRINGS.errorTitle, USERSCRIPT_STRINGS.invalidImportError);
                return;
            }
            if (!importedData || !Array.isArray(importedData.tactics)) {
                await showErrorMessage(USERSCRIPT_STRINGS.errorTitle, USERSCRIPT_STRINGS.invalidImportError);
                return;
            }
            const importedTactics = importedData.tactics;
            importedTactics.forEach(tactic => {
                if (!tactic.hasOwnProperty('style')) {
                    tactic.style = OTHER_CATEGORY_ID;
                }
                if (!tactic.name || !tactic.id || !Array.isArray(tactic.coordinates)) {
                     throw new Error('Invalid tactic structure in imported data.');
                }
                if (tactic.style && !categories[tactic.style] && tactic.style !== OTHER_CATEGORY_ID && tactic.style !== 'short_passing' && tactic.style !== 'wing_play') {
                    addCategory({ id: tactic.style, name: tactic.style, color: generateCategoryColor(tactic.style) });
                }
            });
            let existingTacticsData = await GM_getValue(TACTICS_STORAGE_KEY, { tactics: [] });
            let existingTactics = existingTacticsData.tactics || [];
            const mergedTactics = [...existingTactics];
            let addedCount = 0;
            for (const importedTactic of importedTactics) {
                if (!existingTactics.some((tactic) => tactic.id === importedTactic.id)) {
                    mergedTactics.push(importedTactic);
                    addedCount++;
                }
            }
            await GM_setValue(TACTICS_STORAGE_KEY, { tactics: mergedTactics });
            mergedTactics.sort((a, b) => a.name.localeCompare(b.name));
            dropdownMenuTactics = mergedTactics;
            updateTacticsDropdown();
            updateFilterTabs();
            await showSuccessMessage(USERSCRIPT_STRINGS.doneTitle, USERSCRIPT_STRINGS.importAlert + (addedCount > 0 ? ` (${addedCount} new tactics added)`: ''));
        } catch (error) {
            console.error('ImportError:', error);
            await showErrorMessage(USERSCRIPT_STRINGS.errorTitle, USERSCRIPT_STRINGS.invalidImportError + (error.message ? `: ${error.message}`: ''));
        }
    }

    async function exportTactics() {
        try {
            const tactics = GM_getValue(TACTICS_STORAGE_KEY, { tactics: [] });
            const tacticsJson = JSON.stringify(tactics, null, 2);
            if (navigator.clipboard?.writeText) {
                try {
                    await navigator.clipboard.writeText(tacticsJson);
                    await showSuccessMessage(USERSCRIPT_STRINGS.doneTitle, USERSCRIPT_STRINGS.exportAlert);
                    return;
                } catch (clipboardError) {
                    console.warn('Clipboard write failed, falling back to modal.', clipboardError);
                }
            }
            const textarea = document.createElement('textarea');
            textarea.value = tacticsJson;
            textarea.style.width = '100%';
            textarea.style.minHeight = '150px';
            textarea.style.marginTop = '10px';
            textarea.style.backgroundColor = 'rgba(0,0,0,0.2)';
            textarea.style.color = 'var(--text-color)';
            textarea.style.border = '1px solid rgba(255,255,255,0.1)';
            textarea.style.borderRadius = '4px';
            textarea.readOnly = true;
            const container = document.createElement('div');
            container.appendChild(document.createTextNode('Copy the JSON data below:'));
            container.appendChild(textarea);
            await showAlert({
                title: 'Export Tactics', htmlContent: container, confirmButtonText: 'Done'
            });
            textarea.select();
            textarea.setSelectionRange(0, 99999);
        } catch (error) {
            console.error('Export error:', error);
            await showErrorMessage(USERSCRIPT_STRINGS.errorTitle, 'Failed to export tactics.');
        }
    }

    async function convertXmlToTacticJson(xmlString, tacticName) {
        const parser = new DOMParser();
        const xmlDoc = parser.parseFromString(xmlString, 'text/xml');
        const parserError = xmlDoc.getElementsByTagName('parsererror');
        if (parserError.length > 0) {
            console.error('XMLError:', parserError[0].textContent);
            throw new Error(USERSCRIPT_STRINGS.xmlValidationError);
        }
        const posElements = Array.from(xmlDoc.getElementsByTagName('Pos'));
        const normalPosElements = posElements.filter(el => el.getAttribute('pos') === 'normal');
        if (normalPosElements.length !== MIN_OUTFIELD_PLAYERS) {
             throw new Error(`XML must contain exactly ${MIN_OUTFIELD_PLAYERS} outfield players ('Pos' elements with pos="normal"). Found ${normalPosElements.length}.`);
         }
        const coordinates = normalPosElements.map(el => {
            const x = parseInt(el.getAttribute('x'));
            const y = parseInt(el.getAttribute('y'));
            if (isNaN(x) || isNaN(y)) {
                 throw new Error('Invalid coordinates found in XML.');
             }
            const htmlLeft = x - 7;
            const htmlTop = y - 9;
            return [htmlLeft, htmlTop];
        });
        return {
            name: tacticName, coordinates: coordinates
        };
    }

    // ==============================
    // UI CREATION & UPDATES
    // ==============================
    function createTacticsSelector() {
        const container = document.createElement('div');
        container.className = 'tactics-selector-section';
        const dropdownContainer = document.createElement('div');
        dropdownContainer.className = 'tactics-dropdown-container';
        const dropdownWrapper = document.createElement('div');
        dropdownWrapper.className = 'tactics-dropdown-wrapper';
        const dropdown = document.createElement('select');
        dropdown.id = 'tactics_selector';
        dropdown.addEventListener('change', function() {
            handleTacticsSelection(this.value);
        });
        dropdownWrapper.appendChild(dropdown);
        dropdownContainer.appendChild(dropdownWrapper);
        const searchBox = document.createElement('input');
        searchBox.type = 'text';
        searchBox.className = 'tactics-search-box';
        searchBox.placeholder = USERSCRIPT_STRINGS.searchPlaceholder;
        searchBox.addEventListener('input', (e) => {
            searchTerm = e.target.value.toLowerCase();
            updateTacticsDropdown();
        });
        dropdownContainer.appendChild(searchBox);
        const filterTabs = document.createElement('div');
        filterTabs.className = 'tactics-filter-tabs';
        filterTabs.id = 'tactics-filter-tabs';
        dropdownContainer.appendChild(filterTabs);
        container.appendChild(dropdownContainer);
        return container;
    }

    function createFilterTab(categoryId, label, isActive = false, isRemovable = false) {
        const tab = document.createElement('button');
        tab.className = 'tactics-filter-tab';
        if (isActive) tab.classList.add('active');

        const labelSpan = document.createElement('span');
        labelSpan.textContent = label;
        tab.appendChild(labelSpan);

        tab.dataset.filter = categoryId;
        const categoryColor = loadCategoryColor(categoryId);
        tab.style.setProperty('--category-color', categoryColor);

        tab.addEventListener('click', (e) => {
            if (e.target.classList.contains('remove-category-btn')) {
                return;
            }
            document.querySelectorAll('.tactics-filter-tab').forEach(t => t.classList.remove('active'));
            tab.classList.add('active');
            currentFilter = categoryId;
            updateTacticsDropdown();
        });

        if (isRemovable) {
            const removeBtn = document.createElement('span');
            removeBtn.className = 'remove-category-btn';
            removeBtn.textContent = 'x';
            removeBtn.title = `Remove category "${label}"`;
            removeBtn.addEventListener('click', (e) => {
                e.stopPropagation();
                removeCategory(categoryId).catch(console.error);
            });
            tab.appendChild(removeBtn);
            tab.style.paddingRight = '5px';
        }

        return tab;
    }


    function updateFilterTabs() {
        const filterTabsContainer = document.getElementById('tactics-filter-tabs');
        if (!filterTabsContainer) return;
        filterTabsContainer.innerHTML = '';

        const usedCategories = new Set(dropdownMenuTactics.map(tactic => tactic.style || OTHER_CATEGORY_ID));

        let categoriesToShow = [];
        categoriesToShow.push({ id: 'all', name: USERSCRIPT_STRINGS.allTacticsFilter, removable: false });

        if (usedCategories.has(OTHER_CATEGORY_ID) || currentFilter === OTHER_CATEGORY_ID) {
            categoriesToShow.push({ id: OTHER_CATEGORY_ID, name: getCategoryName(OTHER_CATEGORY_ID), removable: false });
        }

        const otherCategoryIds = Object.keys(categories)
            .filter(id => id !== 'all' && id !== OTHER_CATEGORY_ID && usedCategories.has(id))
            .map(id => ({
                id: id,
                name: getCategoryName(id),
                removable: (id !== 'short_passing' && id !== 'wing_play')
            }))
            .sort((a, b) => a.name.localeCompare(b.name));

        categoriesToShow = categoriesToShow.concat(otherCategoryIds);

        categoriesToShow.forEach(catInfo => {
            const isActive = currentFilter === catInfo.id;
            const tab = createFilterTab(catInfo.id, catInfo.name, isActive, catInfo.removable);
             if (catInfo.id === 'all' && isActive) {
                 tab.style.setProperty('--category-color', 'rgba(255,255,255,0.15)');
             }
            filterTabsContainer.appendChild(tab);
        });

        if (!categoriesToShow.some(cat => cat.id === currentFilter)) {
            currentFilter = 'all';
            const allTab = filterTabsContainer.querySelector('.tactics-filter-tab[data-filter="all"]');
            if (allTab) allTab.classList.add('active');
        }
    }


    function updateTacticsDropdown() {
        const dropdown = document.getElementById('tactics_selector');
        const dropdownWrapper = document.querySelector('.tactics-dropdown-wrapper');
        const searchBox = document.querySelector('.tactics-search-box');
        if (!dropdown) return;
        const previouslySelectedValue = dropdown.value;
        dropdown.innerHTML = '';
        if (searchTerm.length > 0) {
            dropdownWrapper?.classList.add('filtering');
            searchBox?.classList.add('filtering');
        } else {
            dropdownWrapper?.classList.remove('filtering');
            searchBox?.classList.remove('filtering');
        }
        const placeholderOption = document.createElement('option');
        placeholderOption.value = '';
        placeholderOption.textContent = 'Select a Tactic';
        placeholderOption.disabled = true;
        placeholderOption.selected = true;
        dropdown.appendChild(placeholderOption);
        const filteredTactics = dropdownMenuTactics.filter(tactic => {
             const nameMatch = searchTerm === '' || tactic.name.toLowerCase().includes(searchTerm);
             const categoryMatch = currentFilter === 'all' || (currentFilter === OTHER_CATEGORY_ID && (tactic.style === OTHER_CATEGORY_ID || !tactic.style)) || tactic.style === currentFilter;
             return nameMatch && categoryMatch;
        });
        const groupedTactics = {};
        Object.keys(categories).forEach(id => {
             groupedTactics[id] = [];
        });
        if (!groupedTactics[OTHER_CATEGORY_ID]) groupedTactics[OTHER_CATEGORY_ID] = [];
        filteredTactics.forEach(tactic => {
            const categoryId = tactic.style || OTHER_CATEGORY_ID;
            if (!groupedTactics[categoryId]) {
                 groupedTactics[OTHER_CATEGORY_ID].push(tactic);
            } else {
                 groupedTactics[categoryId].push(tactic);
            }
        });
        const categoryOrder = Object.keys(groupedTactics)
          .filter(id => groupedTactics[id].length > 0)
          .sort((a, b) => {
              if (a === currentFilter) return -1;
              if (b === currentFilter) return 1;
              if (a === OTHER_CATEGORY_ID) return 1;
              if (b === OTHER_CATEGORY_ID) return -1;
              return (getCategoryName(a) || '').localeCompare(getCategoryName(b) || '');
          });
        categoryOrder.forEach(categoryId => {
             if (groupedTactics[categoryId].length > 0) {
                 addTacticOptionsGroup(dropdown, groupedTactics[categoryId], getCategoryName(categoryId));
             }
        });
        if (filteredTactics.length === 0 && dropdownMenuTactics.length > 0) {
            const noTacticsOption = document.createElement('option');
            noTacticsOption.disabled = true;
            noTacticsOption.textContent = USERSCRIPT_STRINGS.noTacticsFound;
            dropdown.appendChild(noTacticsOption);
            placeholderOption.selected = false;
        } else if (filteredTactics.length === 0 && dropdownMenuTactics.length === 0) {
             placeholderOption.textContent = 'No tactics saved';
        }
        let foundPrevious = false;
        for (let i = 0; i < dropdown.options.length; i++) {
            if (dropdown.options[i].value === previouslySelectedValue) {
                dropdown.selectedIndex = i;
                foundPrevious = true;
                break;
            }
        }
        if (!foundPrevious) {
             if (filteredTactics.length === 1 && categoryOrder.length === 1 && groupedTactics[categoryOrder[0]].length === 1) {
                for (let i = 0; i < dropdown.options.length; i++) {
                    if (!dropdown.options[i].disabled) {
                        dropdown.selectedIndex = i;
                        break;
                    }
                }
             } else {
                 dropdown.selectedIndex = 0;
             }
        }
        dropdown.disabled = dropdownMenuTactics.length === 0;
    }

    function addTacticOptionsGroup(dropdown, tactics, groupLabel) {
        if (tactics.length === 0) return;
        const groupHeader = document.createElement('optgroup');
        groupHeader.label = groupLabel;
        dropdown.appendChild(groupHeader);
        tactics.sort((a, b) => a.name.localeCompare(b.name));
        tactics.forEach(tactic => {
            const option = document.createElement('option');
            option.value = tactic.name;
            option.dataset.style = tactic.style || OTHER_CATEGORY_ID;
            option.textContent = tactic.name;
            dropdown.appendChild(option);
        });
    }

    function createButton(id, text, clickHandler) {
        const button = document.createElement('button');
        setUpButton(button, id, text);
        if (clickHandler) {
            button.addEventListener('click', function (e) {
                e.stopPropagation();
                hideActiveDropdownMenu();
                clickHandler().catch((error) => {
                    console.error('Button click handler failed:', error);
                    showErrorMessage('Action Failed', `${error}`);
                 });
            });
        }
        return button;
    }

    function createActionDropdownButton(id, text, menuItems) {
        const wrapper = document.createElement('div');
        wrapper.style.position = 'relative';
        wrapper.style.display = 'inline-block';
        const button = createButton(id, text + ' ▼');
        const menu = document.createElement('div');
        menu.className = 'action-dropdown-menu';
        menu.id = id + '_menu';
        menuItems.forEach(item => {
             if (item && item.id && item.text && item.handler) {
                const menuItem = createButton(item.id, item.text, item.handler);
                menu.appendChild(menuItem);
             }
        });
        wrapper.appendChild(button);
        wrapper.appendChild(menu);
        button.addEventListener('click', (e) => {
            e.stopPropagation();
            toggleDropdownMenu(menu);
        });
        return wrapper;
    }

    function toggleDropdownMenu(menu) {
        if (activeDropdownMenu && activeDropdownMenu !== menu) {
            activeDropdownMenu.style.display = 'none';
        }
        if (menu.style.display === 'block') {
            menu.style.display = 'none';
            activeDropdownMenu = null;
        } else {
            menu.style.display = 'block';
            activeDropdownMenu = menu;
            menu.style.top = menu.previousElementSibling.offsetHeight + 4 + 'px';
            menu.style.left = '0';
        }
    }

    function hideActiveDropdownMenu() {
        if (activeDropdownMenu) {
            activeDropdownMenu.style.display = 'none';
            activeDropdownMenu = null;
        }
    }

    function createRenameTacticButton() {
        return createButton('rename_tactic_button', USERSCRIPT_STRINGS.renameButton, renameTactic);
    }

    function createUpdateTacticButton() {
        return createButton('update_tactic_button', USERSCRIPT_STRINGS.updateButton, updateTactic);
    }

    function createDeleteTacticButton() {
        return createButton('delete_tactic_button', USERSCRIPT_STRINGS.deleteButton, deleteTactic);
    }

    function createAddActionButton() {
        const menuItems = [
            { id: 'add_current_tactic_item', text: USERSCRIPT_STRINGS.addCurrentTactic, handler: addNewTactic },
            { id: 'add_xml_tactic_item', text: USERSCRIPT_STRINGS.addWithXmlButton, handler: addNewTacticWithXml }
        ];
        return createActionDropdownButton('add_action_dropdown', USERSCRIPT_STRINGS.addButton, menuItems);
    }

    function createManageActionButton() {
        const menuItems = [
            { id: 'import_tactics_item', text: USERSCRIPT_STRINGS.importButton, handler: importTactics },
            { id: 'export_tactics_item', text: USERSCRIPT_STRINGS.exportButton, handler: exportTactics },
            { id: 'reset_tactics_item', text: USERSCRIPT_STRINGS.resetButton, handler: resetTactics },
            { id: 'clear_tactics_item', text: USERSCRIPT_STRINGS.clearButton, handler: clearTactics }
        ];
        return createActionDropdownButton('manage_action_dropdown', USERSCRIPT_STRINGS.manageButton, menuItems);
    }

    async function checkVersion() {
        const storedVersion = GM_getValue(VERSION_KEY, null);
        if (!storedVersion || storedVersion !== SCRIPT_VERSION) {
            await showWelcomeMessage();
            GM_setValue(VERSION_KEY, SCRIPT_VERSION);
        }
    }

    function playRandomAudio(audios) {
        if (audios.length === 0) { return; }
        const randomIdx = Math.floor(Math.random() * audios.length);
        const activeAudio = audios.splice(randomIdx, 1)[0];
        playAudio(activeAudio, audios);
        return activeAudio;
    }

    function playAudio(currAudio, audios) {
        currAudio.play().catch(e => console.error('Cannot play 猫 シ Corp.', e));
        currAudio.onended = function () { playRandomAudio(audios); };
    }

    function pauseAudio(audio) {
        if (audio) { audio.pause(); audio.currentTime = 0; }
    }

    function updateAudioIcon(button, isPlaying) {
        button.textContent = isPlaying ? '⏸️' : '🔊';
    }

    function createAudioButton() {
        const button = document.createElement('button');
        setUpButton(button, 'audio_button', '🔊');
        button.style.background = 'transparent'; button.style.border = 'none'; button.style.boxShadow = 'none';
        const audioUrls = [ 'https://ia801901.us.archive.org/31/items/corp.-palm-mall-01-palm-mall/%E7%8C%AB%20%E3%82%B7%20Corp.%20-%20Palm%20Mall%20-%2003%20Special%20Discount.mp3', 'https://ia801901.us.archive.org/31/items/corp.-palm-mall-01-palm-mall/%E7%8C%AB%20%E3%82%B7%20Corp.%20-%20Palm%20Mall%20-%2004%20First%20Floor.mp3', 'https://ia801901.us.archive.org/31/items/corp.-palm-mall-01-palm-mall/%E7%8C%AB%20%E3%82%B7%20Corp.%20-%20Palm%20Mall%20-%2006%20Second%20Floor.mp3', 'https://ia801901.us.archive.org/7/items/palm-mall-mars-remastered/%E7%8C%AB%20%E3%82%B7%20Corp.%20%26%20SEPHORA%E8%84%B3%E3%83%90%E3%82%A4%E3%83%96%E3%82%B9%20-%20Palm%20Mall%20Mars%20%28remastered%29%20-%2006%20Second%20floor-%20%ED%99%98%EB%8C%80%20%26%20%EC%9D%8C%EC%95%85.mp3', 'https://ia801901.us.archive.org/7/items/palm-mall-mars-remastered/%E7%8C%AB%20%E3%82%B7%20Corp.%20-%20Palm%20Mall%20Mars%20%28remastered%29%20-%2001%20%E3%82%B9%E3%82%AD%E3%83%9D%E3%83%BC%E3%83%AB%E7%A9%BA%E6%B8%AF%20Plaza.mp3', 'https://ia801901.us.archive.org/7/items/palm-mall-mars-remastered/%E7%8C%AB%20%E3%82%B7%20Corp.%20-%20Palm%20Mall%20Mars%20%28remastered%29%20-%2009%20Sembikiya%20Restaurant.mp3', 'https://ia804504.us.archive.org/20/items/5-wn9896/%E7%8C%AB%20%E3%82%B7%20Corp.%20-%20%E3%82%B7%E3%83%A7%E3%83%83%E3%83%97%20%40%20%E3%83%98%E3%83%AB%E3%82%B7%E3%83%B3%E3%82%AD%20-%2001%20FORUM%20%E6%B6%88%E8%B2%BB%E8%80%85-kuluttaja-.mp3', 'https://ia904504.us.archive.org/20/items/5-wn9896/%E7%8C%AB%20%E3%82%B7%20Corp.%20-%20%E3%82%B7%E3%83%A7%E3%83%83%E3%83%97%20%40%20%E3%83%98%E3%83%AB%E3%82%B7%E3%83%B3%E3%82%AD%20-%2002%20Pelican%20Self%20Storage%20-Tilaa%20Kaikelle-.mp3', 'https://ia904504.us.archive.org/20/items/5-wn9896/%E7%8C%AB%20%E3%82%B7%20Corp.%20-%20%E3%82%B7%E3%83%A7%E3%83%83%E3%83%97%20%40%20%E3%83%98%E3%83%AB%E3%82%B7%E3%83%B3%E3%82%AD%20-%2003%20%E8%B2%B7%E3%81%86%40JUMBO%20-Kauppakeskus-.mp3', 'https://ia904504.us.archive.org/20/items/5-wn9896/%E7%8C%AB%20%E3%82%B7%20Corp.%20-%20%E3%82%B7%E3%83%A7%E3%83%83%E3%83%97%20%40%20%E3%83%98%E3%83%AB%E3%82%B7%E3%83%B3%E3%82%AD%20-%2005%20Hesburger%20%E6%98%A0%E7%94%BB%E9%A4%A8%20-hampurilainen-.mp3', 'https://ia804504.us.archive.org/20/items/5-wn9896/%E7%8C%AB%20%E3%82%B7%20Corp.%20-%20%E3%82%B7%E3%83%A7%E3%83%83%E3%83%97%20%40%20%E3%83%98%E3%83%AB%E3%82%B7%E3%83%B3%E3%82%AD%20-%2006%20%E9%83%BD%E5%B8%82%E3%83%95%E3%82%A9%E3%83%BC%E3%83%A9%E3%83%A0%20Consumer%20-kahvi-.mp3' ];
        let audios = []; let isPlaying = false; let currentAudio = null; let audioInitialized = false;
        const initializeAudio = () => { if (!audioInitialized) { audios = audioUrls.map(url => { const audio = new Audio(); audio.src = url; audio.preload = 'metadata'; return audio; }); audioInitialized = true; } };
        button.addEventListener('click', function () { initializeAudio(); if (!isPlaying) { const playableAudios = audios.map(a => a.cloneNode()); currentAudio = playRandomAudio(playableAudios); isPlaying = true; } else { pauseAudio(currentAudio); isPlaying = false; } updateAudioIcon(button, isPlaying); });
        return button;
    }

    function createMainContainer() {
        const container = document.createElement('div');
        container.id = 'mz_tactics_panel'; container.classList.add('mz-panel');
        const tacticGroup = document.createElement('div');
        tacticGroup.classList.add('mz-group');
        const mainTitle = document.createElement('h2');
        mainTitle.classList.add('mz-group-main-title');
        const titleText = document.createElement('span');
        titleText.textContent = 'MZ Tactics Manager'; mainTitle.appendChild(titleText);
        const vText = document.createElement('span');
        vText.textContent = 'v' + DISPLAY_VERSION; vText.classList.add('mz-version-text'); mainTitle.appendChild(vText);
        const tacticsSelectorSection = createTacticsSelector();
        const buttonsSection = document.createElement('div');
        buttonsSection.className = 'action-buttons-section';
        const addActionBtn = createAddActionButton();
        const renameTacticBtn = createRenameTacticButton();
        const updateTacticBtn = createUpdateTacticButton();
        const deleteTacticBtn = createDeleteTacticButton();
        const manageActionBtn = createManageActionButton();
        appendChildren(buttonsSection, [ addActionBtn, renameTacticBtn, updateTacticBtn, deleteTacticBtn, manageActionBtn ]);
        appendChildren(tacticGroup, [ mainTitle, tacticsSelectorSection, buttonsSection, createHiddenTriggerButton() ]);
        const otherGroup = document.createElement('div');
        otherGroup.classList.add('mz-group'); otherGroup.style.padding = '10px 16px';
        const otherContainer = document.createElement('div');
        otherContainer.style.display = 'flex'; otherContainer.style.justifyContent = 'space-between'; otherContainer.style.alignItems = 'center'; otherContainer.style.width = '100%';
        const otherLeftGroup = document.createElement('div');
        otherLeftGroup.className = 'footer-actions';
        const infoBtn = createCombinedInfoButton();
        const audioBtn = createAudioButton();
        appendChildren(otherLeftGroup, [infoBtn, audioBtn]);
        const otherRightGroup = document.createElement('div');
        otherRightGroup.className = 'mz-language-container';
        const languageLabel = document.createElement('div');
        languageLabel.id = 'language_dropdown_menu_label'; languageLabel.className = 'mz-language-label'; languageLabel.textContent = USERSCRIPT_STRINGS.languageDropdownMenuLabel;
        const languageDropdownWrapper = document.createElement('div');
        languageDropdownWrapper.className = 'mz-language-dropdown';
        const languageDropdownMenu = createLanguageDropdownMenu();
        languageDropdownWrapper.appendChild(languageDropdownMenu);
        const flagImage = createFlagImage();
        appendChildren(otherRightGroup, [languageLabel, languageDropdownWrapper, flagImage]);
        appendChildren(otherContainer, [otherLeftGroup, otherRightGroup]);
        appendChildren(otherGroup, [otherContainer]);
        appendChildren(container, [tacticGroup, otherGroup]);
        return container;
    }

    function createHiddenTriggerButton() {
        const button = document.createElement('button');
        button.id = 'hidden_trigger_button'; button.textContent = '';
        button.style.position = 'absolute'; button.style.opacity = '0'; button.style.pointerEvents = 'none';
        button.style.width = '0'; button.style.height = '0'; button.style.padding = '0'; button.style.margin = '0'; button.style.border = '0';
        button.addEventListener('click', function () {
            const tacticsPresetInfo = { elem: document.getElementById('tactics_preset'), resetValue: '5-3-2' };
            if (tacticsPresetInfo.elem) { tacticsPresetInfo.elem.value = tacticsPresetInfo.resetValue; tacticsPresetInfo.elem.dispatchEvent(new Event('change')); }
        });
        return button;
    }

    function setUpButton(button, id, textContent) {
        button.id = id; button.classList.add('mzbtn'); button.textContent = textContent;
    }

    function createLanguageDropdownMenu() {
        const dropdown = document.createElement('select');
        dropdown.id = 'language_dropdown_menu';
        for (const lang of LANGUAGES) {
            const option = document.createElement('option');
            option.value = lang.code; option.textContent = lang.name;
            if (lang.code === activeLanguage) { option.selected = true; }
            dropdown.appendChild(option);
        }
        dropdown.addEventListener('change', function () { changeLanguage(this.value).catch((_) => { }); });
        return dropdown;
    }

    function createFlagImage() {
        const img = document.createElement('img');
        img.id = 'language_flag';
        const activeLang = LANGUAGES.find((lang) => lang.code === activeLanguage);
        if (activeLang) { img.src = activeLang.flag; }
        return img;
    }

    function getActiveLanguage() {
        let language = GM_getValue('language');
        if (!language) {
            let browserLanguage = navigator.language || 'en';
            browserLanguage = browserLanguage.split('-')[0];
            const languageExists = LANGUAGES.some((lang) => lang.code === browserLanguage);
            language = languageExists ? browserLanguage : 'en';
        }
        return language;
    }

    function updateTranslation() {
        for (const key in USERSCRIPT_STRINGS) { USERSCRIPT_STRINGS[key] = i18next.t(key, { defaultValue: USERSCRIPT_STRINGS[key] }); }
        const renameBtn = document.getElementById('rename_tactic_button'); if(renameBtn) renameBtn.textContent = 'Edit';
        const updateBtn = document.getElementById('update_tactic_button'); if(updateBtn) updateBtn.textContent = 'Update Coords';
        const deleteBtn = document.getElementById('delete_tactic_button'); if(deleteBtn) deleteBtn.textContent = USERSCRIPT_STRINGS.deleteButton;
        const addActionBtn = document.getElementById('add_action_dropdown'); if (addActionBtn) addActionBtn.textContent = 'Add' + ' ▼';
        const manageActionBtn = document.getElementById('manage_action_dropdown'); if (manageActionBtn) manageActionBtn.textContent = 'Other' + ' ▼';
        const addCurrentItem = document.getElementById('add_current_tactic_item'); if (addCurrentItem) addCurrentItem.textContent = USERSCRIPT_STRINGS.addCurrentTactic;
        const addXmlItem = document.getElementById('add_xml_tactic_item'); if (addXmlItem) addXmlItem.textContent = USERSCRIPT_STRINGS.addWithXmlButton;
        const importItem = document.getElementById('import_tactics_item'); if (importItem) importItem.textContent = USERSCRIPT_STRINGS.importButton;
        const exportItem = document.getElementById('export_tactics_item'); if (exportItem) exportItem.textContent = USERSCRIPT_STRINGS.exportButton;
        const resetItem = document.getElementById('reset_tactics_item'); if (resetItem) resetItem.textContent = USERSCRIPT_STRINGS.resetButton;
        const clearItem = document.getElementById('clear_tactics_item'); if (clearItem) clearItem.textContent = USERSCRIPT_STRINGS.clearButton;
        const infoBtn = document.getElementById('info_button'); if (infoBtn) infoBtn.textContent = USERSCRIPT_STRINGS.infoButton;
        const langLabel = document.getElementById('language_dropdown_menu_label'); if(langLabel) langLabel.textContent = USERSCRIPT_STRINGS.languageDropdownMenuLabel;
        const allFilterTab = document.querySelector('.tactics-filter-tab[data-filter="all"]'); if (allFilterTab) { const span = allFilterTab.querySelector('span:not(.remove-category-btn)'); if(span) span.textContent = i18next.t('allTacticsFilter', { defaultValue: 'All'}); }
        document.querySelectorAll('.tactics-filter-tab').forEach(tab => {
            const filterId = tab.dataset.filter;
            if (filterId && filterId !== 'all') {
                 const span = tab.querySelector('span:not(.remove-category-btn)');
                 if (span) {
                     span.textContent = getCategoryName(filterId);
                     const removeBtn = tab.querySelector('.remove-category-btn');
                     if (removeBtn) {
                         removeBtn.title = `Remove category "${span.textContent}"`;
                     }
                 }
            }
        });
        const searchBox = document.querySelector('.tactics-search-box'); if (searchBox) searchBox.placeholder = i18next.t('searchPlaceholder', { defaultValue: 'Search...'});
        updateTacticsDropdown();
        updateFilterTabs();
    }


    async function changeLanguage(languageCode) {
        try {
            const translationDataUrl = langDataBaseUrl + languageCode + '.json';
            let translations;
            try { const response = await fetch(translationDataUrl); if (!response.ok) { throw new Error('Primary language URL failed'); } translations = await response.json(); }
            catch (error) {
                console.log(`Primary language URL (${languageCode}) failed, trying fallback URL`);
                const fallbackBaseUrl = (langDataBaseUrl === CDN_URLS.default.lang) ? CDN_URLS.china.lang : CDN_URLS.default.lang;
                const fallbackUrl = fallbackBaseUrl + languageCode + '.json';
                try { const fallbackResponse = await fetch(fallbackUrl); if (!fallbackResponse.ok) throw new Error('Fallback language URL failed'); translations = await fallbackResponse.json(); }
                catch (fallbackError) { console.error(`Failed to load language ${languageCode} from primary and fallback sources. Using defaults.`); translations = {}; }
            }
            await i18next.changeLanguage(languageCode);
            i18next.addResourceBundle(languageCode, 'translation', translations, true, true);
            GM_setValue('language', languageCode); activeLanguage = languageCode; updateTranslation();
            const language = LANGUAGES.find((lang) => lang.code === languageCode);
            if (language) { const flagImage = document.getElementById('language_flag'); if (flagImage) flagImage.src = language.flag; }
        } catch (e) { console.error('Failed to change language:', e); }
    }

    function createCombinedInfoModalContent() {
        const wrapper = document.createElement('div'); wrapper.id = 'combined_info_modal_content';
        const aboutSection = document.createElement('div');
        const aboutTitle = document.createElement('h3'); aboutTitle.textContent = i18next.t('aboutButton');
        const aboutInfoText = document.createElement('p'); aboutInfoText.id = 'info_modal_info_text'; aboutInfoText.innerHTML = i18next.t('modalContentInfoText');
        const aboutFeedbackText = document.createElement('p'); aboutFeedbackText.id = 'info_modal_feedback_text'; aboutFeedbackText.innerHTML = i18next.t('modalContentFeedbackText');
        appendChildren(aboutSection, [aboutTitle, aboutInfoText, aboutFeedbackText]);
        const linksSection = document.createElement('div');
        const linksTitle = document.createElement('h3'); linksTitle.textContent = i18next.t('usefulLinksButton');
        const linksContent = createUsefulContent();
        const resources = new Map([ ['gewlaht - BoooM', 'https://www.managerzone.com/?p=forum&sub=topic&topic_id=11415137&forum_id=49&sport=soccer'], ['taktikskola by honken91', 'https://www.managerzone.com/?p=forum&sub=topic&topic_id=12653892&forum_id=4&sport=soccer'], ['peto - mix de dibujos', 'https://www.managerzone.com/?p=forum&sub=topic&topic_id=12196312&forum_id=255&sport=soccer'], ['The Zone Chile', 'https://www.managerzone.com/thezone/paper.php?paper_id=18036&page=9&sport=soccer'], ['Tactics guide by lukasz87o/filipek4', 'https://www.managerzone.com/?p=forum&sub=topic&topic_id=12766444&forum_id=12&sport=soccer&share_sport=soccer'], ['MZExtension/van.mz.playerAdvanced by vanjoge', 'https://greasyfork.org/pt-BR/scripts/373382-van-mz-playeradvanced'], ['Mazyar Userscript', 'https://greasyfork.org/pt-BR/scripts/476290-mazyar'], ['Stats Xente Userscript', 'https://greasyfork.org/pt-BR/scripts/491442-stats-xente-script'], ['More userscripts', 'https://greasyfork.org/pt-BR/users/1088808-douglasdotv'] ]);
        const usefulLinksList = createLinksList(resources);
        appendChildren(linksSection, [linksTitle, linksContent, usefulLinksList]);
        appendChildren(wrapper, [aboutSection, linksSection]);
        return wrapper;
    }

    function createUsefulContent() {
        const usefulContent = document.createElement('p'); usefulContent.id = 'useful_content'; usefulContent.textContent = i18next.t('usefulContent'); return usefulContent;
    }

    function createLinksList(hrefs) {
        const list = document.createElement('ul');
        hrefs.forEach((href, title) => {
            const listItem = document.createElement('li'); const link = document.createElement('a');
            link.href = href; link.target = '_blank'; link.rel = 'noopener noreferrer'; link.textContent = title;
            listItem.appendChild(link); list.appendChild(listItem);
        });
        return list;
    }

    function createCombinedInfoButton() {
        const button = document.createElement('button'); setUpButton(button, 'info_button', USERSCRIPT_STRINGS.infoButton);
        button.style.background = 'transparent'; button.style.border = 'none'; button.style.boxShadow = 'none';
        button.addEventListener('click', function (event) {
            event.stopPropagation(); showAlert({ title: 'Info & Links', htmlContent: createCombinedInfoModalContent(), confirmButtonText: DEFAULT_MODAL_STRINGS.ok });
        });
        return button;
    }

    function createToggleButton() {
        const button = document.createElement('button'); button.id = 'toggle_panel_btn'; button.innerHTML = '✕'; button.title = 'Hide panel'; return button;
    }

    function createCollapsedIcon() {
        const icon = document.createElement('div'); icon.id = 'collapsed_icon'; icon.innerHTML = 'TM'; icon.title = 'Show MZ Tactics Manager'; document.body.appendChild(icon); return icon;
    }

    // ==============================
    // INITIALIZATION & PANEL SETUP
    // ==============================
    function initializeLanguage() {
        return new Promise((resolve, reject) => {
            activeLanguage = getActiveLanguage();
            i18next.init({ lng: activeLanguage, fallbackLng: 'en', resources: { [activeLanguage]: { translation: {} } }, interpolation: { escapeValue: false } })
            .then(async () => {
                try {
                    let json = {};
                    try { const url = langDataBaseUrl + activeLanguage + '.json'; const res = await fetch(url); if (!res.ok) { throw new Error('Primary language URL failed during initialization'); } json = await res.json(); }
                    catch (error) {
                        console.log(`Primary language URL (${activeLanguage}) failed during initialization, trying fallback URL or using defaults.`);
                        const fallbackBaseURL = (langDataBaseUrl === CDN_URLS.default.lang) ? CDN_URLS.china.lang : CDN_URLS.default.lang;
                        const fallbackUrl = fallbackBaseURL + activeLanguage + '.json';
                         try { const fallbackRes = await fetch(fallbackUrl); if (!fallbackRes.ok) throw new Error('Fallback language URL failed too'); json = await fallbackRes.json(); }
                         catch (fallbackError) { console.error(`Failed to load language ${activeLanguage} from primary and fallback sources. Using defaults.`); }
                    }
                    i18next.addResourceBundle(activeLanguage, 'translation', json, true, true);
                    loadCategories(); await checkVersion(); resolve();
                } catch (error) { reject(error); }
            }).catch(reject);
        });
    }

    function setUpTacticsInterface(mainContainer) {
        const mainTitle = mainContainer.querySelector('.mz-group-main-title');
        const toggleBtn = createToggleButton();
        const collapsedIcon = createCollapsedIcon();
        mainTitle.appendChild(toggleBtn);

        let isCollapsed = GM_getValue(COLLAPSED_KEY, false);

        const applyCollapsedState = (instant = false) => {
             if (isCollapsed) {
                 if (instant) {
                     mainContainer.style.transition = 'none';
                     mainContainer.classList.add('collapsed');
                     collapsedIcon.classList.add('visible');
                     toggleBtn.innerHTML = '☰';
                     toggleBtn.title = 'Show panel';
                     void mainContainer.offsetHeight;
                     mainContainer.style.transition = '';
                 } else {
                     mainContainer.classList.add('collapsed');
                     collapsedIcon.classList.add('visible');
                     toggleBtn.innerHTML = '☰';
                     toggleBtn.title = 'Show panel';
                 }
             } else {
                 mainContainer.classList.remove('collapsed');
                 collapsedIcon.classList.remove('visible');
                 toggleBtn.innerHTML = '✕';
                 toggleBtn.title = 'Hide panel';
             }
        };

        applyCollapsedState(true);

        function togglePanel() {
            isCollapsed = !isCollapsed;
            GM_setValue(COLLAPSED_KEY, isCollapsed);
            applyCollapsedState();
        }
        toggleBtn.addEventListener('click', (e) => { e.stopPropagation(); togglePanel(); });
        collapsedIcon.addEventListener('click', () => { togglePanel(); });
    }

    async function loadTacticsData() {
        try {
            const data = await fetchTacticsFromGMStorage();
            dropdownMenuTactics = data.tactics || [];
            dropdownMenuTactics.forEach(tactic => { if (!tactic.hasOwnProperty('style')) { tactic.style = OTHER_CATEGORY_ID; } });
            dropdownMenuTactics.sort((a, b) => a.name.localeCompare(b.name));
            updateTacticsDropdown();
            updateFilterTabs();
        } catch (error) {
            console.error('ErrorLoadingTactics:', error);
            showErrorMessage('Load Error', 'Failed to load saved tactics.');
        }
    }

    function initialize() {
        const tacticsBox = document.getElementById('tactics_box');
        if (!tacticsBox) return;
        initializeLanguage()
            .then(() => {
                const mainContainer = createMainContainer();
                setUpTacticsInterface(mainContainer);
                if (isFootball()) { insertAfterElement(mainContainer, tacticsBox); }
                updateTranslation();
                return loadTacticsData();
            })
            .catch(error => {
                console.error('InitializationError:', error);
                const errorDiv = document.createElement('div');
                errorDiv.textContent = 'Error initializing MZ Tactics Manager.'; errorDiv.style.color = 'red'; errorDiv.style.padding = '10px'; errorDiv.style.border = '1px solid red'; errorDiv.style.margin = '10px';
                insertAfterElement(errorDiv, tacticsBox);
            });
    }

    // ==============================
    // EVENT LISTENERS & STARTUP
    // ==============================
    window.addEventListener('load', initialize);
    document.addEventListener('click', (e) => {
        if (activeDropdownMenu) {
            const triggerButton = activeDropdownMenu.previousElementSibling;
            if (!activeDropdownMenu.contains(e.target) && triggerButton !== e.target && !triggerButton.contains(e.target)) {
                hideActiveDropdownMenu();
            }
        }
    });
})();