Raw Source
RU_Next / MT_Player_Stats_Widget

// ==UserScript==
// @name        MT_Player_Stats_Widget
// @namespace 	http://tampermonkey.net/
// @version 	2.5
// @description Скрипт добавляет виджет для отображения статистики игрока на форуме игры "Мир танков"
// @author 		Qwen2.5-Max, Nikolay (Next) Bespalov, and many thanks to @luxero for the information provided ;)
// @match 		http://forum.tanki.su/index.php?/topic/*
// @grant 		GM_xmlhttpRequest
// @grant 		GM_setValue
// @grant 		GM_getValue
// @grant 		GM_listValues
// @grant 		GM_deleteValue
// @connect 	tanki.su
// @connect 	lesta.ru
// @license 	MIT
// @updateURL 	https://gist.github.com/next-ru/f22f90a0ef505340d2374b553436045a/raw/player_widget.user.js
// @downloadURL https://gist.github.com/next-ru/f22f90a0ef505340d2374b553436045a/raw/player_widget.user.js
// @icon64 		
// ==/UserScript==

(function() {
    'use strict';

    // Предварительная загрузка шрифта
    const fontLink = document.createElement('link');
    fontLink.rel = 'stylesheet';
    fontLink.href = 'https://next-ru.github.io/player_widget/MTSans-RegularCondensed.ttf';
    document.head.appendChild(fontLink);

    // Предварительная загрузка фонового изображения
    const bgImage = new Image();
    bgImage.src = 'https://next-ru.github.io/player_widget/bg-content.png';

    // Время жизни кэша (5 минут)
    const CACHE_TTL = 5 * 60 * 1000;

    // Функция для очистки устаревшего кэша
    function clearOldCache() {
        const now = Date.now();
        const keys = GM_listValues().filter(key => key.startsWith('playerCache_'));
        keys.forEach(key => {
            const cacheData = GM_getValue(key);
            if(now - cacheData.timestamp > CACHE_TTL) {
                GM_deleteValue(key);
            }
        });
    }

    // Вызов функции очистки кэша при запуске скрипта
    clearOldCache();

    // Добавление пользовательского шрифта и стилей
    const style = document.createElement('style');
    style.innerHTML = ` @font-face {
            font-family: 'MTSans RegularCondensed';
            src: url(${fontLink.href}) format('truetype');
            font-weight: 400;
            font-style: normal;
    }

    .regular-condensed {
            font-family: 'MTSans RegularCondensed', sans-serif;
            font-weight: 400;
            font-style: normal;
    }

    .widget-container {
            display: flex;
            flex-direction: column;
            gap: 16px;
    }

    .widget-top,
    .widget-bottom {
            display: flex;
            gap: 8px;
            align-items: flex-start;
    }

    .widget-block.player {
            width: 250px;
    }

    .widget-block.clan {
            min-width: 250px;
            max-width: 300px;
            flex-grow: 1;
    }

    .value {
            white-space: nowrap;
            overflow: hidden;
            text-overflow: ellipsis;
    }

    .clan-emblem {
            display: flex;
            justify-content: flex-end;
    }

    .clan-emblem img {
            width: 64px;
            height: 64px;
    }

    `;
    document.head.appendChild(style);

    // Создание элемента для виджета
    let widget = document.createElement('div');
    widget.id = 'player-widget'; // Уникальный ID для виджета
    widget.style.position = 'absolute'; // Позиционирование будет обновляться динамически
    widget.style.backgroundImage = `url(${bgImage.src})`; // Устанавливаем фоновое изображение
    widget.style.backgroundColor = '#000'; // Задаем запасной цвет фона (черный)
    widget.style.color = '#fff'; // Задаем цвет текста (белый)
    widget.style.padding = '8px'; // Добавляем внутренние отступы для лучшей читаемости
    widget.style.borderRadius = '8px'; // Добавляем скругление углов
    widget.style.fontSize = '16px'; // Устанавливаем размер шрифта для текста
    widget.style.boxShadow = '0 4px 8px rgba(0, 0, 0, 0.2)'; // Добавляем тень для объемного эффекта
    widget.style.whiteSpace = 'nowrap'; // Запрещаем перенос текста на новую строку
    widget.style.display = 'none'; // Скрываем виджет при создании
    widget.classList.add('regular-condensed'); // Добавляем CSS-класс для дополнительных стилей
    document.body.appendChild(widget);

    // Функция для обновления позиции виджета
    function updateWidgetPosition(widget, target) {
        const offsetX = 0; // Отступ по горизонтали (в пикселях)
        const offsetY = 15; // Отступ по вертикали (в пикселях)

        const targetRect = target.getBoundingClientRect(); // Получаем координаты целевого элемента
        const widgetRect = widget.getBoundingClientRect(); // Получаем размеры виджета

        let leftPosition = targetRect.right + window.scrollX + offsetX; // Виджет справа от элемента с отступом
        let topPosition;

        // Определяем, где находится целевой элемент: в верхней или нижней части экрана
        const isTargetInUpperHalf = targetRect.top < (window.innerHeight / 2);

        if (isTargetInUpperHalf) {
            // Если целевой элемент в верхней половине экрана, виджет рисуем снизу
            topPosition = targetRect.bottom + window.scrollY + offsetY;
        } else {
            // Если целевой элемент в нижней половине экрана, виджет рисуем сверху
            topPosition = targetRect.top - widgetRect.height + window.scrollY - offsetY;
        }

        // Проверяем, выходит ли виджет за пределы экрана
        if (topPosition < 0) {
            // Если виджет выходит за верхний край, перемещаем его вниз
            topPosition = targetRect.bottom + window.scrollY + offsetY;
        } else if (topPosition + widgetRect.height > window.innerHeight + window.scrollY) {
            // Если виджет выходит за нижний край, перемещаем его вверх
            topPosition = targetRect.top - widgetRect.height + window.scrollY - offsetY;
        }

        // Применяем стили
        Object.assign(widget.style, {
            top: `${topPosition}px`,
            left: `${leftPosition}px`
    });
    }

    // Функция для форматирования времени из Unix-формата
    function formatUnixTime(timestamp) {
        const date = new Date(timestamp * 1000);
        return date.toLocaleString();
    }

    // Функция для обновления содержимого виджета
    function updateWidgetContent(nickname, spaId, regTimestamp, lastBattleAt, summaryData, statisticsData, clanId, clanInfo, playerClanInfo) {
        // Верхняя секция: информация об игроке
        const playerTopInfo = `
        <div class="value">Никнейм: ${nickname}&nbsp;<a href="https://tanki.su/ru/community/accounts/${spaId}-${nickname}/" target="_blank"><img src="" alt="Профиль игрока" style="vertical-align: middle;"></a></div>
        <div class="value">Дата регистрации: ${regTimestamp ? formatUnixTime(regTimestamp) : 'Неизвестно'}</div>
        <div class="value">Последний бой: ${lastBattleAt ? formatUnixTime(lastBattleAt) : 'Неизвестно'}</div>
        `;

        // Верхняя секция: эмблема клана (если есть)
        const clanTopInfo = clanInfo ? `
        <img src="https://lesta.ru${clanInfo.large_emblem_url}?nocache=${Date.now()}" alt="Эмблема клана">
        ` : '';

        // Нижняя секция: статистика игрока
        const playerBottomInfo = `
        <div class="value">Личный рейтинг: ${summaryData?.global_rating ?? 'Неизвестно'}</div>
        <div class="value">Бои: ${statisticsData?.battles_count ?? 'Неизвестно'}</div>
        <div class="value">Победы: ${statisticsData?.wins_count_percent != null ? `${statisticsData.wins_count_percent}%` : 'Неизвестно'}</div>
        <div class="value">Средний урон: ${statisticsData?.damage_dealt_avg ?? 'Неизвестно'}</div>
        <div class="value">Попадания: ${summaryData?.hits_ratio != null ? `${summaryData.hits_ratio}%` : 'Неизвестно'}</div>
        <div class="value">Средний опыт за бой: ${statisticsData?.xp_amount_avg ?? 'Неизвестно'}</div>
        <div class="value">Максимум уничтожено за бой: ${statisticsData?.frags_max ?? 'Неизвестно'}</div>
        <div class="value">Максимальный опыт за бой: ${statisticsData?.xp_max ?? 'Неизвестно'}</div>
        <div class="value">Максимальный урон за бой: ${statisticsData?.damage_max ?? 'Неизвестно'}</div>
        <div class="value">Знаки классности «Мастер»: ${summaryData?.mastery ? `${summaryData.mastery.mastery_count}/${summaryData.mastery.vehicles_count}` : 'Неизвестно'}</div>
        `;

        // Нижняя секция: статистика клана (если есть)
        const clanBottomInfo = clanInfo ? `
        <div class="value">Клан-тег: [${clanInfo.tag ?? 'Неизвестно'}]</div>
        <div class="value">Название: ${clanInfo.name ?? 'Неизвестно'}</div>
        <div class="value">Должность: ${playerClanInfo?.localized_name ?? 'Неизвестно'}</div>
        <div class="value">Дней в клане: ${playerClanInfo?.days_in_clan ?? 'Неизвестно'}</div>
        <div class="value">Рейтинг клана: ${clanInfo.rating ?? '-'}</div>
        <div class="value">Позиция клана в общем рейтинге: ${clanInfo.rating_position ?? '-'}</div>
        <div class="value">Активные игроки: ${clanInfo.members_count ?? 'Неизвестно'}</div>
        <div class="value">Среднее количество боёв: ${clanInfo.average_battles_count ?? 'Неизвестно'}</div>
        <div class="value">Средний опыт за бой: ${clanInfo.average_xp_per_battle ?? 'Неизвестно'}</div>
        <div class="value">Средний урон за бой: ${clanInfo.average_damage_per_battle ?? 'Неизвестно'}</div>
        <div class="value">Средний процент побед: ${clanInfo?.average_win_rate != null ? `${clanInfo.average_win_rate}%` : 'Неизвестно'}</div>
        ` : '';

        // Вставляем всё в виджет
        widget.innerHTML = `
        <div class="widget-container">
                <!-- Верхняя часть -->
                <div class="widget-top">
                        <div class="widget-block player">${playerTopInfo}</div>
                        ${clanTopInfo ? `<div class="widget-block clan clan-emblem">${clanTopInfo}</div>` : ''}
                </div>
                <!-- Нижняя часть -->
                <div class="widget-bottom">
                        <div class="widget-block player">${playerBottomInfo}</div>
                        ${clanBottomInfo ? `<div class="widget-block clan">${clanBottomInfo}</div>` : ''}
                </div>
        </div>
        `;
    }

    // Функция для получения данных из кэша
    function getCache(key) {
        const cacheData = GM_getValue(`playerCache_${key}`);
        if(cacheData && Date.now() - cacheData.timestamp < CACHE_TTL) {
            return cacheData.data;
        }
        return null;
    }

    // Функция для сохранения данных в кэш
    function setCache(key, value) {
        GM_setValue(`playerCache_${key}`, {
            timestamp: Date.now(),
            data: value
        });
    }

    // Функция для отображения небольшого текстового сообщения в виджете
    function showWidgetMessage(text) {
        widget.innerHTML = text; // Устанавливаем текст сообщения
        widget.style.display = 'block'; // Делаем виджет видимым
        widget.style.minWidth = '200px'; // Минимальная ширина
        widget.style.minHeight = '50px'; // Минимальная высота
        widget.style.display = 'flex'; // Используем flexbox для центрирования
        widget.style.justifyContent = 'center'; // Центрируем по горизонтали
        widget.style.alignItems = 'center'; // Центрируем по вертикали
    }

    // Функция для получения данных об игроке
    async function fetchData(spaId, nickname) {
        const cacheKey = `${spaId}_${nickname}`;
        const cachedData = getCache(cacheKey);
        if(cachedData) {
            updateWidgetContent(
                nickname,
                spaId,
                cachedData.regTimestamp,
                cachedData.lastBattleAt,
                cachedData.summaryData,
                cachedData.statisticsData,
                cachedData.clanId,
                cachedData.clanInfo,
                cachedData.playerClanInfo
            );
            return;
        }

        let regTimestamp = null;
        let lastBattleAt = null;
        let clanId = null;
        let summaryData = null;
        let statisticsData = null;
        let clanInfo = null;
        let playerClanInfo = null;

        // Параллельные запросы: profileResponse, summaryData и statisticsData
        const [profileResponse, summaryPromise, statisticsPromise] = await Promise.all([
            // Запрос данных профиля игрока
            (async () => {
                try {
                    const response = await new Promise((resolve, reject) => {
                        GM_xmlhttpRequest({
                            method: "GET",
                            url: `https://tanki.su/ru/community/accounts/${spaId}-${nickname}/`,
                            headers: {
                                "accept": "text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7",
                                "accept-language": "ru-RU,ru;q=0.9",
                                "priority": "u=0, i",
                                "sec-ch-ua": "\"Not(A:Brand\";v=\"99\", \"Google Chrome\";v=\"133\", \"Chromium\";v=\"133\"",
                                "sec-ch-ua-mobile": "?0",
                                "sec-ch-ua-platform": "\"Windows\"",
                                "sec-fetch-dest": "document",
                                "sec-fetch-mode": "navigate",
                                "sec-fetch-site": "same-origin",
                                "sec-fetch-user": "?1",
                                "upgrade-insecure-requests": "1"
                            },
                            referrer: "https://tanki.su/ru/community/accounts/",
                            referrerPolicy: "unsafe-url",
                            onload: resolve,
                            onerror: reject
                        });
                    });

                    const parser = new DOMParser();
                    const doc = parser.parseFromString(response.responseText, "text/html");

                    const scripts = doc.querySelectorAll('script');
                    for(const script of scripts) {
                        const scriptContent = script.textContent;
                        if(scriptContent.includes("USER_DATA")) {
                            const regTimestampMatch = scriptContent.match(/"reg_timestamp":\s*(\d+)/);
                            const lastBattleAtMatch = scriptContent.match(/"last_battle_at":\s*(\d+)/);
                            const clanIdMatch = scriptContent.match(/"id":\s*(\d+)/);

                            if(regTimestampMatch) regTimestamp = parseInt(regTimestampMatch[1], 10);
                            if(lastBattleAtMatch) lastBattleAt = parseInt(lastBattleAtMatch[1], 10);
                            if(clanIdMatch) clanId = parseInt(clanIdMatch[1], 10);
                            break;
                        }
                    }
                } catch {}
            })(),

            // Запрос данных о суммарной статистике игрока
            (async () => {
                try {
                    const response = await new Promise((resolve, reject) => {
                        GM_xmlhttpRequest({
                            method: "GET",
                            url: `https://tanki.su/wotup/profile/summary/?spa_id=${spaId}&battle_type=random`,
                            headers: {
                                "accept": "*/*",
                                "accept-language": "ru-RU,ru;q=0.9",
                                "priority": "u=1, i",
                                "sec-ch-ua": "\"Not(A:Brand\";v=\"99\", \"Google Chrome\";v=\"133\", \"Chromium\";v=\"133\"",
                                "sec-ch-ua-mobile": "?0",
                                "sec-ch-ua-platform": "\"Windows\"",
                                "sec-fetch-dest": "empty",
                                "sec-fetch-mode": "cors",
                                "sec-fetch-site": "same-origin",
                                "x-requested-with": "XMLHttpRequest"
                            },
                            referrer: `https://tanki.su/ru/community/accounts/${spaId}-${nickname}/`,
                            referrerPolicy: "unsafe-url",
                            onload: resolve,
                            onerror: reject
                        });
                    });

                    const rawSummaryData = JSON.parse(response.responseText)?.data ?? null;
                    if(rawSummaryData) {
                        summaryData = {
                            global_rating: rawSummaryData.global_rating,
                            hits_ratio: rawSummaryData.hits_ratio,
                            mastery: rawSummaryData.mastery
                        };
                    }
                } catch {}
            })(),

            // Запрос данных о статистике игрока
            (async () => {
                try {
                    const response = await new Promise((resolve, reject) => {
                        GM_xmlhttpRequest({
                            method: "GET",
                            url: `https://tanki.su/wotup/profile/statistics/?spa_id=${spaId}&battle_type=random`,
                            headers: {
                                "accept": "*/*",
                                "accept-language": "ru-RU,ru;q=0.9",
                                "priority": "u=1, i",
                                "sec-ch-ua": "\"Not(A:Brand\";v=\"99\", \"Google Chrome\";v=\"133\", \"Chromium\";v=\"133\"",
                                "sec-ch-ua-mobile": "?0",
                                "sec-ch-ua-platform": "\"Windows\"",
                                "sec-fetch-dest": "empty",
                                "sec-fetch-mode": "cors",
                                "sec-fetch-site": "same-origin",
                                "x-requested-with": "XMLHttpRequest"
                            },
                            referrer: `https://tanki.su/ru/community/accounts/${spaId}-${nickname}/`,
                            referrerPolicy: "unsafe-url",
                            onload: resolve,
                            onerror: reject
                        });
                    });

                    const rawStatisticsData = JSON.parse(response.responseText)?.data ?? null;
                    if(rawStatisticsData) {
                        statisticsData = {
                            damage_max: rawStatisticsData.damage_max,
                            wins_count_percent: rawStatisticsData.wins_count_percent,
                            battles_count: rawStatisticsData.battles_count,
                            damage_dealt_avg: rawStatisticsData.damage_dealt_avg,
                            xp_amount_avg: rawStatisticsData.xp_amount_avg,
                            frags_max: rawStatisticsData.frags_max,
                            xp_max: rawStatisticsData.xp_max
                        };
                    }
                } catch {}
            })()
        ]);

        // Если игрок в клане, запрашиваем информацию о клане
        if(clanId) {
            const [clanPromise, playersPromise] = await Promise.all([
                // Запрос информации о клане
                (async () => {
                    try {
                        const response = await new Promise((resolve, reject) => {
                            GM_xmlhttpRequest({
                                method: "GET",
                                url: `https://lesta.ru/clans/wot/${clanId}/api/claninfo/`,
                                headers: {
                                    "accept": "application/json, text/javascript, */*; q=0.01",
                                    "accept-language": "ru-RU,ru;q=0.9",
                                    "priority": "u=1, i",
                                    "sec-ch-ua": "\"Not(A:Brand\";v=\"99\", \"Google Chrome\";v=\"133\", \"Chromium\";v=\"133\"",
                                    "sec-ch-ua-mobile": "?0",
                                    "sec-ch-ua-platform": "\"Windows\"",
                                    "sec-fetch-dest": "empty",
                                    "sec-fetch-mode": "cors",
                                    "sec-fetch-site": "same-origin",
                                    "x-requested-with": "XMLHttpRequest"
                                },
                                referrer: `https://lesta.ru/clans/wot/${clanId}/`,
                                referrerPolicy: "strict-origin-when-cross-origin",
                                onload: resolve,
                                onerror: reject
                            });
                        });

                        const clanData = JSON.parse(response.responseText)?.clanview ?? null;
                        if(clanData) {
                            clanInfo = {
                                tag: clanData.clan.tag,
                                name: clanData.clan.name,
                                rating: clanData.rating.rating,
                                average_battles_count: clanData.rating.average_battles_count,
                                rating_position: clanData.rating.rating_position,
                                average_xp_per_battle: clanData.rating.average_xp_per_battle,
                                average_damage_per_battle: clanData.rating.average_damage_per_battle,
                                average_win_rate: clanData.rating.average_win_rate,
                                members_count: clanData.clan.members_count,
                                large_emblem_url: clanData.clan.large_emblem_url
                            };
                        }
                    } catch {}
                })(),

                // Запрос информации о игроке в клане
                (async () => {
                    try {
                        const response = await new Promise((resolve, reject) => {
                            GM_xmlhttpRequest({
                                method: "GET",
                                url: `https://lesta.ru/clans/wot/${clanId}/api/players/?offset=0&limit=25&o=-role&timeframe=all&battle_type=default`,
                                headers: {
                                    "accept": "application/json, text/javascript, */*; q=0.01",
                                    "accept-language": "ru-RU,ru;q=0.9",
                                    "priority": "u=1, i",
                                    "sec-ch-ua": "\"Not(A:Brand\";v=\"99\", \"Google Chrome\";v=\"133\", \"Chromium\";v=\"133\"",
                                    "sec-ch-ua-mobile": "?0",
                                    "sec-ch-ua-platform": "\"Windows\"",
                                    "sec-fetch-dest": "empty",
                                    "sec-fetch-mode": "cors",
                                    "sec-fetch-site": "same-origin",
                                    "x-requested-with": "XMLHttpRequest"
                                },
                                referrer: `https://lesta.ru/clans/wot/${clanId}/players/`,
                                referrerPolicy: "strict-origin-when-cross-origin",
                                onload: resolve,
                                onerror: reject
                            });
                        });

                        const playersData = JSON.parse(response.responseText)?.items ?? null;
                        const player = playersData.find(p => p.id === parseInt(spaId, 10));
                        if(player) {
                            playerClanInfo = {
                                localized_name: player.role.localized_name,
                                days_in_clan: player.days_in_clan
                            };
                        }
                    } catch {}
                })()
            ]);
        }

        // Сохранение данных в кэш
        const playerData = {
            regTimestamp,
            lastBattleAt,
            summaryData,
            statisticsData,
            clanId,
            clanInfo,
            playerClanInfo
        };

        setCache(cacheKey, playerData);

        // Обновление содержимого виджета
        updateWidgetContent(
            nickname,
            spaId,
            playerData.regTimestamp,
            playerData.lastBattleAt,
            playerData.summaryData,
            playerData.statisticsData,
            playerData.clanId,
            playerData.clanInfo,
            playerData.playerClanInfo
        );
    }

    let widgetTimeout = null; // Переменная для хранения таймера
    const WIDGET_TIMEOUT_DELAY = 200; // Задержка перед исчезновением виджета (в миллисекундах)

    // Обработчик наведения мыши на элемент или виджет
    function handleHover(event) {
        const target = event.target;

        // Проверяем, находится ли курсор на элементе
        if (
            target.tagName === 'SPAN' &&
            target.parentElement &&
            target.parentElement.classList.contains('ccw-word')
        ) {
            const parentSpan = target.parentElement;
            const nickname = parentSpan.getAttribute('data-value');
            const authorInfo = parentSpan.closest('.author_info');
            if (authorInfo) {
                const parentAnchor = authorInfo.querySelector('a[hovercard-spaid]');
                if (parentAnchor) {
                    const spaId = parentAnchor.getAttribute('hovercard-spaid');

                    // Отменяем предыдущий таймер, если он был запущен
                    if (widgetTimeout) {
                        clearTimeout(widgetTimeout);
                        widgetTimeout = null;
                    }

                    // Показываем индикатор загрузки
                    showWidgetMessage('Загрузка...');
                    updateWidgetPosition(widget, target);

                    // Загружаем данные и обновляем позицию виджета
                    fetchData(spaId, nickname).then(() => {
                        updateWidgetPosition(widget, target);
                    });
                } else {
                    // Добавляем обработку случая, когда элемент hovercard-spaid не найден
                    showWidgetMessage('SPA ID не найден. Аккаунт удалён.');
                    updateWidgetPosition(widget, target);
                }
            }
        }

        // Если курсор находится на самом виджете, отменяем таймер
        if (target === widget || widget.contains(target)) {
            if (widgetTimeout) {
                clearTimeout(widgetTimeout);
                widgetTimeout = null;
            }
        }
    }

    // Обработчик ухода курсора с элемента или виджета
    function handleMouseOut(event) {
        const target = event.target;

        // Если курсор ушел с элемента или с виджета
        if (
            (target.tagName === 'SPAN' &&
             target.parentElement.classList.contains('ccw-word')) ||
            target === widget ||
            widget.contains(target)
        ) {
            // Запускаем таймер для скрытия виджета
            widgetTimeout = setTimeout(() => {
                widget.innerHTML = ''; // Очищаем содержимое виджета
                widget.style.display = 'none'; // Скрываем виджет
                widgetTimeout = null; // Сбрасываем таймер
            }, WIDGET_TIMEOUT_DELAY);
        }
    }

    document.addEventListener('mouseover', handleHover);
    document.addEventListener('mouseout', handleMouseOut);
})();