<aside> 🔴 НЕ ВСТАВЛЯЙТЕ скрипты (включая эти) в браузер, если не понимаете как они работают. Это не безопасно. Попросите проверить скрипт того, кому доверяете. Ну или хотя бы прогоните через ChatGPT.
</aside>
Если по какой-то неведомой причине вы попали сюда случайно, то вот ссылка на игру: https://t.me/hamsteR_kombat_bot/start
Автор: https://t.me/i_urKing
ТГ канал: https://t.me/js_is_easy
YouTube: @IlyaYurkin
Аналитика показывает топ 5 лучших позиций для покупки в порядке убывания полезности в формате
Для начала нужно открыть игру в браузере (см. оригинальный пост)
Открываем инструменты разработчика (F12)
Выбираем чат с ботом и запускаем игру
Ищем во вкладке Sources следующий файл top → clicker → hamsterkombat.io → js → telegram-web-app.js
Вставляем в самом верху следующий скрипт (добавил подсказки в комментариях)
// Константа с путем к изображению монеты
const COIN_IMAGE_PATH = "";
// Массив с заголовками таблицы, где каждый объект описывает колонку таблицы
const HEADERS = [
{ name: "name", title: "Название", width: "32%" }, // Колонка "Название" с шириной 32%
{ name: "section", title: "Группа", width: "20%" }, // Колонка "Группа" с шириной 20%
{ name: "price", title: "Цена", width: "18%" }, // Колонка "Цена" с шириной 18%
{ name: "profit", title: "Прибыль/Цена", width: undefined }, // Колонка "Прибыль/Цена" без заданной ширины
];
// Переменная для хранения данных таблицы
let TABLE_DATA = [];
// Функция для сокращения числа с использованием суффиксов
function shortenNumber(number, useSuffix = true) {
// Если число меньше 1000, не используем суффиксы
useSuffix = number < 1000 ? false : useSuffix;
// Определяем количество знаков после запятой
const fractionDigits = useSuffix ? 2 : 0;
// Определяем массив суффиксов
const suffixes = ["", "K", "M", "B", "T", "Qua", "Qui", "S"];
// Берем абсолютное значение числа
const absValue = Math.abs(number);
// Определяем индекс суффикса
const suffixIndex = absValue < 1 ? 0 : Math.floor(Math.log10(absValue) / 3);
// Берем суффикс или "😈" по умолчанию
const suffix = suffixes[suffixIndex] ?? "😈";
// Делим число на соответствующую степень тысячи
const value = number / Math.pow(10, suffixIndex * 3);
// Округляем число до нужного количества знаков
const roundFactor = Math.pow(10, fractionDigits);
// Форматируем число
const formattedNumber = new Intl.NumberFormat({
maximumFractionDigits: fractionDigits,
minimumFractionDigits: 0,
}).format(
useSuffix
? Math.floor(value * roundFactor) / roundFactor
: Math.floor(number)
);
// Возвращаем форматированное число с суффиксом или без него
return useSuffix ? `${formattedNumber}${suffix}` : formattedNumber;
}
// Функция для создания таблицы из данных
function createTable(data) {
console.log(TABLE_DATA); // Логируем данные таблицы
const table = document.createElement("table"); // Создаем элемент таблицы
table.style.width = "100%";
table.style.borderCollapse = "collapse";
table.style.marginBottom = "16px";
table.style.background = "#272a2f";
table.style.borderRadius = "10px";
table.style.padding = "6px";
table.style.tableLayout = "fixed";
const thead = document.createElement("thead"); // Создаем заголовок таблицы
const headerRow = document.createElement("tr"); // Создаем строку заголовка
// Проходимся по каждому заголовку и создаем ячейку заголовка
HEADERS.forEach((header) => {
const th = document.createElement("th");
th.textContent = header.title;
th.style.padding = "8px";
th.style.textAlign = "left";
th.style.fontSize = "14px";
th.style.color = "#fff";
th.style.width = header.width;
headerRow.appendChild(th);
});
thead.appendChild(headerRow); // Добавляем строку заголовка в заголовок таблицы
table.appendChild(thead); // Добавляем заголовок таблицы в таблицу
const tbody = document.createElement("tbody"); // Создаем тело таблицы
// Проходимся по каждому элементу данных и создаем строку таблицы
data.forEach((item) => {
const row = document.createElement("tr");
// Проходимся по каждому заголовку и создаем ячейку строки таблицы
HEADERS.forEach(({ name }) => {
const td = document.createElement("td");
td.style.padding = "8px";
td.style.textAlign = "left";
td.style.fontSize = "12px";
td.style.color = "#fff";
td.style.borderBottom = "1px solid #1c1f24";
td.style.whiteSpace = "nowrap";
td.style.overflow = "hidden";
td.style.textOverflow = "ellipsis";
// Если колонка "price" или "profit", добавляем изображение монеты
if (name === "price" || name === "profit") {
const div = document.createElement("div");
div.style.display = "flex";
div.style.alignItems = "center";
const img = document.createElement("img");
img.style.width = "14px";
img.style.height = "14px";
img.style.marginRight = "8px";
img.src = COIN_IMAGE_PATH;
div.appendChild(img);
div.appendChild(
document.createTextNode(
name === "price" ? shortenNumber(item[name]) : item[name].toFixed(5)
)
);
td.appendChild(div);
} else {
td.textContent = item[name];
}
row.appendChild(td); // Добавляем ячейку в строку таблицы
});
tbody.appendChild(row); // Добавляем строку в тело таблицы
});
table.appendChild(tbody); // Добавляем тело таблицы в таблицу
return table; // Возвращаем таблицу
}
// Функция для добавления таблицы рекомендаций
const addRecommendationTable = () => {
const oldTableContainer = document.querySelector("#recommendation_table"); // Ищем существующую таблицу рекомендаций
const dailyComboProgress = document.querySelector(".tabs-inner"); // Ищем элемент, перед которым будем вставлять таблицу
// Если старая таблица существует, удаляем ее
if (oldTableContainer) {
oldTableContainer.remove();
} else {
const header = document.createElement("div"); // Создаем заголовок секции
header.classList.add("section-title");
const headerTitle = document.createElement("span");
headerTitle.innerText = "Рекомендации покупок";
header.appendChild(headerTitle);
dailyComboProgress.parentNode.insertBefore(header, dailyComboProgress); // Вставляем заголовок перед элементом "tabs-inner"
}
const tableContainer = document.createElement("div"); // Создаем контейнер для таблицы
tableContainer.setAttribute("id", "recommendation_table");
tableContainer.appendChild(createTable(TABLE_DATA)); // Добавляем таблицу в контейнер
dailyComboProgress.parentNode.insertBefore(
tableContainer,
dailyComboProgress
); // Вставляем контейнер с таблицей перед элементом "tabs-inner"
};
// Добавляем обработчик события для загрузки DOM
document.addEventListener("DOMContentLoaded", () => {
const observer = new MutationObserver((mutationsList) => {
// Обрабатываем мутации в DOM
for (const mutation of mutationsList) {
if (mutation.type === "childList" && mutation.addedNodes.length > 0) {
mutation.addedNodes.forEach((node) => {
// Проверяем, был ли добавлен элемент с классом "tabs-inner"
const hasTabsInnerAdded =
node.nodeType === Node.ELEMENT_NODE &&
(node.classList.contains("tabs-inner") ||
node.querySelector(".tabs-inner"));
if (hasTabsInnerAdded) {
addRecommendationTable(); // Добавляем таблицу рекомендаций
}
});
}
}
});
// Начинаем наблюдение за всем телом документа
observer.observe(document.body, { childList: true, subtree: true });
});
// Сохраняем оригинальную функцию fetch
let originalFetch = window.fetch;
// Переопределяем функцию fetch
window.fetch = function () {
// Сохраняем аргументы, переданные в fetch
const args = arguments;
// Вызываем оригинальную функцию fetch и перехватываем промис с ответом
return originalFetch.apply(this, args).then(function (response) {
// Клонируем ответ, так как response можно прочитать только один раз
const clone = response.clone();
// Читаем текст клонированного ответа
clone.json().then(function (json) {
if (
args[0].includes("upgrades-for-buy") ||
args[0].includes("clicker/buy-upgrade")
) {
const upgrades = json.upgradesForBuy;
// Фильтруем доступные и не истекшие апгрейды без кулдауна
const filteredUpgrades = upgrades.filter(
({ isExpired, isAvailable, cooldownSeconds }) =>
!isExpired &&
isAvailable &&
(!cooldownSeconds || cooldownSeconds === 0)
);
// Преобразуем данные апгрейдов в формат для таблицы
const mappedUpgrades = filteredUpgrades.map(
({ section, name, price, profitPerHourDelta }) => ({
name: name,
section: section,
profit: Number((profitPerHourDelta / price).toFixed(5)),
price: price,
})
);
// Сортируем апгрейды по прибыльности
const sortedUpgrades = mappedUpgrades.sort(
(a, b) => b.profit - a.profit
);
// Сохраняем топ-5 апгрейдов в TABLE_DATA
TABLE_DATA = sortedUpgrades.filter((_, index) => index < 5);
// Добавляем таблицу рекомендаций после небольшого таймаута
setTimeout(() => {
addRecommendationTable();
}, 0);
}
});
// Возвращаем оригинальный ответ для дальнейшей обработки
return response;
});
};
Сохраняем изменения ctrl + s
Переоткрываем игру. Готово
Если что-то пошло не так или хотите вернуться к оригинальному скрипту просто удалите сохраненные скрипты с компа (вы выбирали папку, в которой будут хранится файлы когда разрешали хомяку открываться на компе)
Или просто нажмите сюда