NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==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} <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); })();