// ==UserScript==
// @name BetterLZT-lite By LolzNews & Openresty
// @namespace betterlzt
// @version 2.0.0
// @author Openresty \w LolzNews
// @license Apache 2.0
// @description Legendary extension for Lolzteam from LolzNews
// @icon https://lolz.live/styles/brand/download/avatars/three_avatar.svg
// @match https://zelenka.guru/*
// @match https://lzt.market/*
// @match https://lolz.market/*
// @match https://lolz.live/*
// @connect *
// @connect hasan.ovh
// @connect hasan.su
// @grant GM.getValue
// @grant GM.setValue
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_xmlhttpRequest
// @grant unsafeWindow
// @run-at document-body
// ==/UserScript==
//
// ЛЮБЫЕ МОДИФИКАЦИИ КОДА, КОТОРЫЕ НАРУШАЮТ ПРАВИЛА ФОРУМА, ПРИВОДЯТ К СБОЯМ И НАРУШАЮТ РАБОТУ
// ФОРУМА/РАСШИРЕНИЯ ПРИВЕДУТ К БЛОКИРОВКАМ (ФОРУМНОЙ УЧЕТНОЙ ЗАПИСИ/УЧЕТНОЙ ЗАПИСИ BETTERLZT LIVE)
//
(async function () {
'use strict';
var _GM_getValue = /* @__PURE__ */ (() => typeof GM_getValue != "undefined" ? GM_getValue : void 0)();
var _GM_setValue = /* @__PURE__ */ (() => typeof GM_setValue != "undefined" ? GM_setValue : void 0)();
var _GM_xmlhttpRequest = /* @__PURE__ */ (() => typeof GM_xmlhttpRequest != "undefined" ? GM_xmlhttpRequest : void 0)();
var _unsafeWindow = /* @__PURE__ */ (() => typeof unsafeWindow != "undefined" ? unsafeWindow : void 0)();
function conVar(convar, isGlobal = false) {
try {
if (isGlobal) return globalConfig[convar];
return userConfig[convar];
} catch (e) {
return false;
}
}
function set_conVar(convar, value, isGlobal = false) {
if (isGlobal) {
globalConfig[convar] = value;
return saveConfig(true);
}
userConfig[convar] = value;
saveConfig();
}
function saveConfig(isGlobal = false) {
if (isGlobal)
return _GM_setValue("globalConfig", JSON.stringify(globalConfig));
_GM_setValue("userConfig", JSON.stringify(userConfig));
}
function makeWatermark(text, link = "#") {
if (!text) return false;
const waterm = document.createElement("h1");
waterm.style = "position:fixed;bottom:5px;right:5px;opacity:0.5;z-index:99;color:white;font-size: 25px;";
waterm.innerHTML = text;
waterm.href = link;
return document.body.append(waterm);
}
function isJson(str) {
try {
JSON.parse(str);
} catch (e) {
console.error("[BetterLZT Tools] JSON parse failed: " + e);
return false;
}
return true;
}
function changeUserStatus(status) {
if (!status) return;
let statusArea = document.querySelector(".userBlurb.current_text ");
return statusArea.innerHTML = status;
}
function changeUserAvatar(mode = "def") {
if (conVar("lolznews")) mode = "lolznews";
try {
let imageUrl;
switch (mode) {
case "lolznews":
imageUrl = "https://hasan.ovh/better/lolznews.png";
break;
default:
imageUrl = "";
break;
}
let imageArea = document.querySelector(".avatarScaler img");
imageArea.src = imageUrl;
return true;
} catch {
return false;
}
}
function inProfile() {
return !!document.querySelector(".avatarScaler");
}
function xfAlert(text) {
return XenForo.alert(text, 1, 1e4);
}
const css = ` <style>
.main-text {
font-size: 13px;
font-style: normal;
font-weight: 600;
line-height: normal;
display: inline;
}
.btns-l {
margin-bottom: 10px;
margin-right: 10px;
border-radius: 6px;
display: inline-block;
padding: 7px 15px;
background: #363636;
justify-content: center;
align-items: center;
gap: 12px;
font-size: 13px;
font-style: normal;
font-weight: 600;
}
details {
width: 100%;
background: #272727;
border: solid 3px #363636;
box-shadow: 0 0.1rem 1rem -0.5rem rgba(0, 0, 0, .4);
border-radius: 8px;
overflow: hidden;
margin-top: -25px;
margin-bottom: 35px;
}
summary {
padding: 15px;
display: block;
background: #363636;
position: relative;
cursor: pointer;
color: #D6D6D6;
font-family: Open Sans;
font-size: 14px;
font-style: normal;
font-weight: 600;
line-height: normal;
}
summary span {
color: #949494;
font-size: 13px;
}
details span {
color: #949494;
font-size: 13px;
}
summary:after {
font-family: "Font Awesome 5 Pro";
color: rgb(148,148,148);
content: '>';
position: absolute;
left: 97%;
top: 50%;
transform: translate(-50%, -50%) rotate(90deg);
transform-origin: 0.2rem 50%;
transition: 0.25s transform ease;
}
details[open] > summary:after {
transform: translate(-50%, -50%) rotate(270deg);
}
details[open] > div {
padding: 10px 20px;
}
details .leftButton {
margin-right: 10px;
}
details button {
width: 45px;
height: 45px;
padding: 5px;
justify-content: center;
align-items: center;
color: rgb(34,142,93);
border-radius: 6px;
background: #363636;
border: none;
font-size: 25px;
margin-bottom: 10px;
margin-right: 10px;
}
details button.active {
border: 1.6px solid #07C682;
background: linear-gradient(180deg, rgba(7, 198, 130, 0.12) 0%, rgba(7, 198, 130, 0.00) 100%), #363636;
}
details input.input{
width: 77%;
padding: 6px;
border-radius: 6px;
height: 20px;
background: #303030;
color: white;
border: 1px solid rgb(54, 54, 54);
}
details input[type=checkbox] {
width: auto;
}
details input[type=checkbox]:after {
border-radius: 4px;
}
.reportBtn {
font-weight: bold; padding: 3px 10px; background: #218e5d; border-radius: 50px; margin-right: 5px; cursor: pointer; color: #fff; border: 0;
}
</style>`;
function init() {
if (conVar("customBackground")) {
document.querySelector("body").style = `background-size: cover;
background-position: center;
background-attachment: fixed;
background-repeat: no-repeat;
background-image: linear-gradient(rgba(54, 54, 54, 0.85), rgba(54, 54, 54, 0.85)), url('${conVar(
"customBackground"
)}')`;
}
}
function saveBackground() {
let bgUrl = document.querySelector("#custombg").value;
xfAlert("Сохранено");
if (bgUrl.length < 1) return set_conVar("customBackground", false);
return set_conVar("customBackground", bgUrl);
}
const render = `
let handling = 0;
let step = 0;
window.addEventListener('message', function(event) {
if (event.data == '20100') {
handling = 0;
step = 0
iframe = document.querySelectorAll('#areaext');
iframe.forEach(function (e){
e.contentWindow.postMessage("10100", "*");
e.contentWindow.postMessage(betterVersion, "*");
e.contentWindow.postMessage(XenForo.visitor.user_id, "*");
})
}
if (event.data == '20200') {
handling = 20200
step = 0
}
if (event.data == '20300') {
handling = 20300
step = 0
set_conVar('userScripts', 0)
xfAlert('Успешно!')
}
else if (handling == 20200 && event.data != '20200') {
if (step == 0) {
window.betterAPI.loadScript(event.data, 'Ext'+ event.data, function() {
window.betterAPI.runScript(event.data, 0);
});
handling = 20200;
step = 1;
xfAlert('Успешно!')
}
if (step == 1) {
xfAlert('Попытка установить скрипт (LiveID#' +event.data + ')')
handling = 0;
step = 0;
}
}
});
function steam(arg) {
e.contentWindow.postMessage(arg, "*");
}
`;
const renderFrame = `<iframe src='https://hasan.su/bettermarket/1_2.php' width='100%' frameborder='0' height='800px' id='areaext'> </iframe> `;
function exuiOnPage() {
let htmlall = `<br>
<details>
<summary>Основные<br><span>Реклама, секретный вопрос</span></summary>
<div><br>
<div class='btns-l' id='rai'> ИИ Ассистент (Розыгрыши)</div>
<div class='btns-l' id='adblock'> Блокировщик рекламы</div>
<div class="btns-l" id="trustFactor">Фактор доверия</div>
<div class='btns-l' id='hideLikes'> Скрывать счетчик лайков в профиле</div> ${globalConfig["unlock_dev"] ? "<div class='btns-l' id='visitorDetector'> Обнаруживать [visitor] в сообщениях</div>" : ""}
<div class='btns-l' id='reportBtns'> Показывать кнопки для быстрой подачи жалоб</div>
<div class='btns-l' id='nickCopy'> Показывать кнопки для копирования ника</div>
<hr style="border: solid 1px #363636;">
<p class="main-text" onclick="window.location.href = 'https://telegra.ph/Security-note-for-BetterLZT-feature-06-27'">Автоматический ввод секретной фразы: (Важная заметка! кликабельно)<br></p>
<br>
<input id="secretph" class="input" placeholder="Введите вашу секретную фразу"> <a class="button leftButton primary" onclick="saveSecret()" id="secretphBtn">Сохранить</a>
</div>
</details>
<br>
<details>
<summary>Внешний вид<br><span>Оформи форум под себя</span></summary>
<div><br>
<p>Не работает в LTS</p>
<p class="main-text">Кастомный фон для всего форума (ссылка на картинку): <br></p>
<br>
<input id="custombg" class="input" placeholder="URL"> <a class="button leftButton primary" onclick="saveBackground()">Сохранить</a>
</div>
</details>
<br>
<details>
<summary>Кнопки быстрой подачи жалоб<br><span>Редактирование текста на кнопках</span></summary>
<div><br>
<p class="main-text">Нажмите на кнопку и введите текст. Учтите, что этот текст будет отправлен в жалобе. Пустой текст скрывает кнопку<br></p>
<br>
<input type="text" id="reportBtn-1" class="reportBtn" value="${userConfig["reportBtn-1"]}"> <input type="text" id="reportBtn-2" class="reportBtn" value="${userConfig["reportBtn-2"]}"> <input type="text" id="reportBtn-3" class="reportBtn" value="${userConfig["reportBtn-3"]}">
</div>
</details>
<br>
<details>
<summary>Маркетплейс расширений<br><span>Запускайте лушие скрипты из раздела "Дополнения" без установки в Tampermonkey<br>Все расширения проверены и безопасны</span></summary>
<div><br>
${globalConfig["unlock_dev"] ? "DEV: <input type='text' id='loadcustom' placeholder='Введите ссылку'><a class='btns-l' id='saveCustom'>Сохранить</a>" : ""}
${renderFrame}
</div>
</details>
`;
let script = document.createElement("script");
script.appendChild(document.createTextNode(render));
document.head.appendChild(script);
let html_prem = `
<div style="background: rgb(54, 54, 54);
margin: 5px 10px;
padding: 0 15px 15px 15px; border-radius: 0px; ${localCache.serverNotice ? "" : "display: none"}"><br>
${localCache.serverNotice ?? ""}
</div><br>
${globalConfig["unlock_dev"] ? '<div style="background: rgb(54, 54, 54);margin: 5px 15px;padding: 0 15px 15px 15px; border-radius: 10px;"><br>BetterLZT Dev mode</div>' : ""}<br>
${htmlall}<br>
<div style="display: flex;
width: 598px;
justify-content: space-between;
align-items: flex-start;">
Version ${betterVersion}<br>
Marketplace 1.2
</div>
<a class="button leftButton primary" target="_blank" href="https://t.me/lolz_news">LolzNews - Подпишитесь)</a>
<br><br>
<a class="button leftButton primary" target="_blank" href="https://lolz.live/payment/balance-transfer?user_id=2626330&hold=0&comment=%D0%94%D0%BE%D0%B1%D1%80%D0%BE%D0%B2%D0%BE%D0%BB%D1%8C%D0%BD%D0%BE%D0%B5%20%D0%BF%D0%BE%D0%B6%D0%B5%D1%80%D1%82%D0%B2%D0%BE%D0%B2%D0%B0%D0%BD%D0%B8%D0%B5">Разработчику на доширак</a>
<a class="button leftButton" href="https://t.me/trixydev">ТГк Разработчика</a> <a class="button leftButton" onclick="resetConfig()">Сброс настроек</a>
${css}
`;
document.querySelector(".titleBar").innerHTML = "<h1>LN BetterLZT - Настройки</h1>";
document.querySelector(".errorOverlay").innerHTML = html_prem;
return true;
}
const adlist_w = [
"zelenka.guru/threads/3649746",
"http://proxysoxy.com",
"zelenka.guru/threads/2770783",
"https://t.me/talkthenews",
"https://zelenka.guru/threads/5862277/",
"zelenka.guru/threads/5802663/",
"@UniServBot",
"zelenka.guru/threads/5886612",
"https://zelenka.guru/threads/5830418/",
"zelenka.guru/angeldrainer/",
"zelenka.guru/threads/5883557",
"zelenka.guru/threads/5720998",
"https://zelenka.guru/threads/5488501",
"https://zelenka.guru/threads/4871985/",
"zelenka.guru/threads/3649746",
"zelenka.guru/threads/5402454",
"zelenka.guru/threads/2630352",
"https://t.me/poseidon_project",
"https://zelenka.guru/threads/4826265/",
"zelenka.guru/threads/4939541",
"zelenka.guru/threads/4073607",
"zelenka.guru/threads/5071761/",
"https://zelenka.guru/threads/3695705/",
"zelenka.guru/members/4177803",
"@verif_ads",
"verifteam",
"SmmPanelUS.com",
"lteboost.ru"
];
const adlist_white = ["t.me/lolz_news", "https://t.me/lolz_news"];
await( _GM_getValue("adBlock"));
function run$4() {
if (!conVar("adblock")) return false;
if (document.querySelector(".avatarScaler")) return profile$1();
}
function containsLink(str) {
const urlRegex = /https?:\/\/[^\s|<>"]+(?!(?:(?:[^<]*<){2}))/g;
const noHtmlStr = str.replace(/<[^>]*>/g, "");
const matches = noHtmlStr.match(urlRegex);
return !!matches;
}
function profile$1() {
let userStatus = document.querySelector(".userBlurb.current_text");
try {
if (containsLink(userStatus.innerHTML) && !adlist_white.some((o) => userStatus.innerHTML.toLowerCase().includes(o))) {
changeUserStatus("Реклама скрыта");
changeUserAvatar();
return true;
}
if (adlist_w.some((o) => userStatus.innerHTML.toLowerCase().includes(o)) && !adlist_white.some((o) => userStatus.innerHTML.toLowerCase().includes(o))) {
changeUserStatus("Реклама скрыта");
changeUserAvatar();
return true;
}
} catch (e) {
if (globalConfig["unlock_dev"])
console.log("метод обнаружения рекламы не сработал: " + e);
return false;
}
return userStatus.innerHTML === "Реклама скрыта";
}
const ui_primary = "#228e5e";
const ui_error = "#8e2222";
async function show() {
let theme2 = await conVar("dashboard", true);
switch (await theme2) {
case "exui":
exuiOnPage();
break;
default:
exuiOnPage();
break;
}
buttonsRender();
const buttons = document.querySelectorAll(".btns-l");
const report_buttons = document.querySelectorAll(".reportBtn");
buttons.forEach((button) => {
button.addEventListener("click", handleClick);
});
report_buttons.forEach((button) => {
button.addEventListener("change", handleClick);
});
}
function handleClick(e) {
e = e.target;
let state;
switch (e.id) {
case "adblock":
state = conVar("adblock");
set_conVar("adblock", !state);
e.style.background = state ? "" : ui_primary;
if (!state) run$4();
break;
case "hideLikes":
state = conVar("hideLikes");
set_conVar("hideLikes", !state);
e.style.background = state ? "" : ui_primary;
break;
case "trustFactor":
state = conVar("trustFactor");
set_conVar("trustFactor", !state);
e.style.background = state ? "" : ui_primary;
break;
case "visitorDetector":
state = conVar("visitorDetector");
set_conVar("visitorDetector", !state);
e.style.background = state ? "" : ui_primary;
break;
case "reportBtns":
state = conVar("reportBtns");
set_conVar("reportBtns", !state);
e.style.background = state ? "" : ui_primary;
break;
case "saveCustom":
window.betterAPI.loadScript(document.querySelector("#loadcustom").value, "Test", function() {
window.betterAPI.runScript("Test", 0);
});
break;
// Report buttons. TODO: refactor
case "reportBtn-1":
set_conVar("reportBtn-1", e.value);
xfAlert("Сохранено");
break;
case "reportBtn-2":
set_conVar("reportBtn-2", e.value);
xfAlert("Сохранено");
break;
case "reportBtn-3":
set_conVar("reportBtn-3", e.value);
xfAlert("Сохранено");
break;
case "nickCopy":
state = conVar("nickCopy");
set_conVar("nickCopy", !state);
e.style.background = state ? "" : ui_primary;
break;
case "hide_ftapanel":
state = conVar("hide_ftapanel");
set_conVar("hide_ftapanel", !state);
e.style.background = state ? "" : ui_primary;
break;
case "rai":
state = conVar("rai");
set_conVar("rai", !state);
e.style.background = state ? "" : ui_primary;
break;
default:
e.style.background = ui_error;
xfAlert(
"Данная функция временно недоступна. Обновите страницу"
);
break;
}
}
function settingsButtons() {
let accountMenu = document.querySelector(
"#AccountMenu > ul > li:nth-child(12) > a"
);
let settingsMenuItem = document.createElement("li");
settingsMenuItem.innerHTML = '<a href="betterlzt_settings">BetterLZT</a>';
accountMenu.parentNode.insertBefore(
settingsMenuItem,
accountMenu.nextSibling
);
}
function buttonsRender() {
let buttons = document.querySelectorAll(".btns-l");
buttons.forEach((e) => {
if (conVar(e.id)) e.style.background = ui_primary;
});
}
function saveSecret() {
let phrase = document.querySelector("#secretph").value;
if (!phrase || conVar("secureTest", false) !== true) {
set_conVar("secretPhrase", false);
set_conVar("safeMode", true, true);
return alert("Обнаружено нарушение безопасности. Секретная фраза удалена из кеша BetterLZT\nФункция автозаполнения секретной фразы отключена");
}
if (conVar("safeMode", true) || conVar("secureTest", false) !== true) {
return alert("Включен безопасный режим\nФункция автозаполнения секретной фразы недоступна\nПереустановите расширение");
}
document.querySelector("#secretphBtn").innerHTML = "Сохранено!";
return set_conVar("secretPhrase", phrase);
}
function pasteSecret() {
let secretArea = document.querySelector("input[name=secret_answer]");
if (secretArea && conVar("secretPhrase")) {
secretArea.value = conVar("secretPhrase");
return true;
}
return false;
}
function calculate(user) {
try {
if (localCache.TrustFactorService.minExtVersion > betterVersion || !localCache.TrustFactorService.status) {
xfAlert("В работе службы TrustFactor возникла ошибка: Службы LIVE недоступны");
return "unavailable";
}
user.registrationDate = parseDate(user.registrationDate);
const weights = {
test: localCache.TrustFactorService.Weight,
likes: localCache.TrustFactorService.Weights.likes ?? 0.25,
// Вес симпатий: 25%
registrationDays: localCache.TrustFactorService.Weights.registrationDays ?? 0.2,
// Вес времени с момента регистрации: 20%
deposit: localCache.TrustFactorService.Weights.deposit ?? 0.35,
// Вес страхового депозита: 35%
trophies: localCache.TrustFactorService.Weights.trophies ?? 0.05,
// Вес трофеев: 5%
messages: localCache.TrustFactorService.Weights.Messages ?? 0.05,
// Вес количества сообщений: 5%
giveaways: localCache.TrustFactorService.Weights.giveaways ?? 0.1
// Вес числа розыгрышей: 10%
};
const { likes, registrationDate, deposit, trophies, messages, giveaways } = user;
const maxLikes = localCache.TrustFactorService.Values.likes ?? 4e3;
const maxDaysRegistered = localCache.TrustFactorService.Values.registrationDays ?? 365 * 2;
const minDeposit = localCache.TrustFactorService.Values.minDeposit ?? 1e4;
const maxDeposit = localCache.TrustFactorService.Values.maxDeposit ?? 1e5;
const maxTrophies = localCache.TrustFactorService.Values.Trophies ?? 15;
const maxMessages = localCache.TrustFactorService.Values.Messages ?? 1e3;
const maxGiveaways = localCache.TrustFactorService.Values.Giveaways ?? 20;
const today = /* @__PURE__ */ new Date();
const registrationDays = Math.floor((today - new Date(registrationDate)) / (1e3 * 60 * 60 * 24));
const normalizedLikes = Math.min(likes / maxLikes, 1);
const normalizedRegistration = Math.min(registrationDays / maxDaysRegistered, 1);
const normalizedDeposit = Math.min((deposit - minDeposit) / (maxDeposit - minDeposit), 1);
const normalizedTrophies = Math.min(trophies / maxTrophies, 1);
const normalizedMessages = Math.min(messages / maxMessages, 1);
const normalizedGiveaways = Math.min(giveaways / maxGiveaways, 1);
const trustLevel = (normalizedLikes * weights.likes + normalizedRegistration * weights.registrationDays + normalizedDeposit * weights.deposit + normalizedTrophies * weights.trophies + normalizedMessages * weights.messages + normalizedGiveaways * weights.giveaways) * 100;
return Math.min(Math.max(trustLevel, 0), 100);
} catch (e) {
xfAlert("В работе службы TrustFactor возникла проблема. Работа фактора доверия остановлена");
if (conVar("unlock_dev"), 1) console.error("[BetterLZT] TrustService.js failed: " + e);
}
}
function parseDate(dateString) {
const monthMap = {
"янв": "01",
"фев": "02",
"мар": "03",
"апр": "04",
"май": "05",
"июн": "06",
"июл": "07",
"авг": "08",
"сен": "09",
"окт": "10",
"ноя": "11",
"дек": "12",
"Jan": "01",
"Feb": "02",
"Mar": "03",
"Apr": "04",
"May": "05",
"Jun": "06",
"Jul": "07",
"Aug": "08",
"Sep": "09",
"Oct": "10",
"Nov": "11",
"Dec": "12"
};
const regexRu = /(\d{1,2})\s([а-яёА-ЯЁ]+)\s(\d{4})/;
const regexEn = /([A-Za-z]+)\s(\d{1,2}),\s(\d{4})/;
let day, month, year;
if (regexRu.test(dateString)) {
const [, dayStr, monthStr, yearStr] = dateString.match(regexRu);
day = dayStr.padStart(2, "0");
month = monthMap[monthStr.toLowerCase().slice(0, 3)];
year = yearStr;
} else if (regexEn.test(dateString)) {
const [, monthStr, dayStr, yearStr] = dateString.match(regexEn);
day = dayStr.padStart(2, "0");
month = monthMap[monthStr.slice(0, 3)];
year = yearStr;
} else {
throw new Error("Unknown date format");
}
return `${year}-${month}-${day}`;
}
function profile() {
try {
let user = {
likes: parseInt(document.querySelector(".page_counter .count").innerHTML.replace(" ", "")),
registrationDate: document.querySelector(".labeled .DateTime").innerHTML,
deposit: parseInt(document.querySelector("h3.amount").innerHTML.replaceAll(" ", "").replace("₽", "")),
trophies: parseInt(document.querySelectorAll(".page_counter .count")[3].innerHTML.replace(" ", "")),
messages: parseInt(document.querySelectorAll(".page_counter .count")[2].innerHTML.replace(" ", "")),
giveaways: parseInt(document.querySelectorAll(".page_counter .count")[4].innerHTML.replace(" ", ""))
};
let result = calculate(user).toFixed(0);
let blzt_trust = document.querySelector(".insuranceDeposit");
let blzt_trust_color = result > 35 ? "green" : "red";
let blzt_trust_render = `
<br>
<div class="section insuranceDeposit">
<div class="secondaryContent">
<h3>
<a href="https://lolz.live/threads/5821466/" class="username" style="max-width: 200px; word-wrap: break-word;">
Уровень доверия ß
</a>
</h3>
<h3 style="margin-bottom: 0px; font-size: 18px !important; color: ${blzt_trust_color}" class="amount" title="${result}">
${result} / 100
</h3>
</div>
</div>`;
let blzt_trust_block = document.createElement("div");
blzt_trust_block.innerHTML = blzt_trust_render;
blzt_trust.append(blzt_trust_block);
} catch (e) {
xfAlert("В работе фактора доверия произошла ошибка.");
if (conVar("unlock_dev", 1)) console.error("[BetterLZT] Trust.js: " + e);
}
}
function run$3() {
try {
userID();
if (conVar("nickCopy")) nickCopy$1();
if (conVar("hideLikes")) hideLikes();
if (conVar("trustFactor")) profile();
if (isOwn()) {
let likes = document.querySelector(".page_counter .count").innerHTML;
set_conVar("likes", likes);
}
} catch (e) {
xfAlert(
"В работе расширения произошла ошибка (Модуль inProfile остановлен)"
);
if (conVar("unlock_dev", 1)) {
console.error(`[BetterLZT] inProfile.js failed: ${e}`);
}
}
return true;
}
function isOwn() {
return document.querySelector(".button.block").href.includes("account/personal-details");
}
function userID() {
const id = /market\/user\/(\d+)\/items/.exec(document.querySelector(
'.userContentLinks .button[href^="market/"]'
).href)[1];
let idhtml = document.createElement("div");
idhtml.innerHTML = `
<div class="label fl_l">ID пользователя: </div><div class="labeled">${id}<span data-phr="ID скопирован в буфер обмена" onclick="Clipboard.copy(${id}, this)" class="copyButton Tooltip" title="" data-cachedtitle="Скопировать ID" tabindex="0"><i class="far fa-clone" aria-hidden="true"></i>
</span></div>`;
document.querySelector(".pairsJustified").prepend(idhtml);
return true;
}
function nickCopy$1() {
const user_nick = document.querySelector("h1.username span").innerHTML.replace(/ <i.*?>.*?<\/i>/ig, "");
let nickhtml = document.createElement("span");
nickhtml.id = "nick_copy";
nickhtml.innerHTML = `<span data-phr="Ник скопирован в буфер обмена" onclick="Clipboard.copy('${user_nick}', this)" class="copyButton Tooltip" title="" data-cachedtitle="Скопировать ник" tabindex="0"><i class="far fa-clone" aria-hidden="true"></i></span>`;
document.querySelector("h1.username span").append(nickhtml);
return true;
}
function hideLikes() {
return document.querySelectorAll(".page_counter")[1].remove();
}
function run$2() {
window.betterAPI = {
scripts: {},
settingsFields: [],
ApiVer: 1.2,
// Динамичный запуск
loadScript: function(url, className, callback) {
let script = document.createElement("script");
const expression = /[-a-zA-Z0-9@:%._\+~#=]{1,256}\.[a-zA-Z0-9()]{1,6}\b([-a-zA-Z0-9()@:%_\+.~#?&//=]*)/gi;
const regex = new RegExp(expression);
url = url.toString();
if (!url.match(regex)) {
url = `https://hasan.ovh/bettermarket/storage/${url}.js`;
}
script.src = url;
script.onload = function() {
console.log(className + " Загружен.");
unsafeWindow.betterAPI.saveToCache(className, url);
if (typeof callback === "function") {
callback();
}
unsafeWindow.betterAPI.runScript(className, 0);
};
script.onerror = function() {
console.error("Error loading script: " + url);
};
document.head.appendChild(script);
},
// инстанс и запуск
runScript: function(className, ...args) {
if (typeof unsafeWindow[className] === "function") {
this.scripts[className] = new unsafeWindow[className](...args);
if (typeof this.scripts[className].run === "function") {
this.scripts[className].run();
} else {
console.warn(className + " Не содержит метода загрузки");
xfAlert(`Внимание! Скрипт ${className} не отвечает требованиям (1). Рекомендуется удаление`);
}
} else {
xfAlert("Скрипт успешно установлен! Перезапустите страницу");
}
if (!unsafeWindow[className]) {
return xfAlert("Перезапустите страницу");
}
return true;
},
// скриптинстанс
getScriptInstance: function(className) {
return this.scripts[className] || null;
},
// сохранение в кеш
saveToCache: function(className, url) {
let cache = this.getCache();
cache[className] = url;
set_conVar("userScripts", JSON.stringify(cache));
},
// получение кеша
getCache: function() {
let cache = conVar("userScripts");
return cache ? JSON.parse(cache) : {};
},
// запуск всех скриптов
loadCachedScripts: async function() {
let cache = this.getCache();
for (let className in cache) {
this.loadScript(cache[className], className);
}
},
getData: async function(variable) {
return conVar(variable);
},
inProfile: function() {
if (document.querySelector(".avatarScaler")) return true;
},
getLiveID: function() {
return 0;
},
isAd: function() {
return run$4();
},
changeUserStatus: function(text) {
return changeUserStatus(text);
},
changeUserAvatar: function(mode) {
return changeUserAvatar(mode);
},
// Регистрация полей настройки для скриптов
addSettingsField: function(className, label, id, type2, defaultValue, onChange) {
if (!this.settingsFields[className]) {
this.settingsFields[className] = [];
}
this.settingsFields[className].push({
label,
id,
type: type2,
defaultValue,
onChange
});
},
showSettingsMenu: async function(activeScript) {
if (!activeScript || !this.settingsFields[activeScript]) {
xfAlert("Настройки для этого скрипта не найдены.");
return;
}
let settingsHTML = "";
let settingsValues = {};
for (let field of this.settingsFields[activeScript]) {
settingsValues[field.id] = await GM.getValue(field.id, field.defaultValue);
settingsHTML += `<label for="${field.id}">${field.label}:</label><br>`;
if (field.type === "text") {
settingsHTML += `<input type="text" id="${field.id}" value="${settingsValues[field.id]}"><br><br>`;
} else if (field.type === "checkbox") {
const checked = settingsValues[field.id] ? "checked" : "";
settingsHTML += `<input type="checkbox" id="${field.id}" ${checked}><br><br>`;
}
}
settingsHTML += `<button id="saveSettings">Сохранить</button>`;
XenForo.alert(settingsHTML);
document.getElementById("saveSettings").onclick = () => {
for (let field of this.settingsFields[activeScript]) {
const newValue = field.type === "checkbox" ? document.getElementById(field.id).checked : document.getElementById(field.id).value;
GM.setValue(field.id, newValue);
if (typeof field.onChange === "function") {
field.onChange(newValue);
}
}
xfAlert("Настройки сохранены!");
};
},
showSettingsMenuAll: async function() {
let settingsHTML = "";
let settingsValues = {};
for (let field of this.settingsFields) {
settingsValues[field.id] = await GM.getValue(field.id, field.defaultValue);
settingsHTML += `<label for="${field.id}">${field.label}:</label><br>`;
if (field.type === "text") {
settingsHTML += `<input type="text" id="${field.id}" value="${settingsValues[field.id]}"><br><br>`;
} else if (field.type === "checkbox") {
const checked = settingsValues[field.id] ? "checked" : "";
settingsHTML += `<input type="checkbox" id="${field.id}" ${checked}><br><br>`;
}
}
settingsHTML += `<button id="saveSettings">Сохранить</button>`;
XenForo.alert(settingsHTML);
document.getElementById("saveSettings").onclick = () => {
for (let field of this.settingsFields) {
const newValue = field.type === "checkbox" ? document.getElementById(field.id).checked : document.getElementById(field.id).value;
GM.setValue(field.id, newValue);
if (typeof field.onChange === "function") {
field.onChange(newValue);
}
}
xfAlert("Настройки сохранены!");
};
},
updateInterface: function(buttonText, statusText) {
this.changeUserStatus(statusText);
}
};
}
function request(url) {
return new Promise((resolve, reject) => _GM_xmlhttpRequest({
method: "GET",
url,
mode: "no-cors",
onload: (response) => resolve(response.responseText),
onerror: (error) => resolve(error)
}));
}
function isAvailable() {
if (!conVar("rai")) {
return false;
}
if (conVar("likes") < 500) {
return false;
} else {
return true;
}
}
async function contestRender() {
const renderParent = document.querySelector(".forumModerators");
const renderTextArea = document.createElement("span");
const renderTextArea2 = document.createElement("span");
const counter = document.querySelector(".counterText").innerHTML;
const max = counter.substring("/")[1];
const current = counter.substring("/")[0];
renderTextArea.className = "rai";
if (document.querySelector(".rai")) {
return;
}
if (!isAvailable()) {
renderTextArea.style.color = `red`;
renderTextArea.innerHTML = `К сожалению, на данный момент функции BetterLZT RAI недоступны для вашего профиля<br>`;
return renderParent.prepend(renderTextArea);
} else {
const resultRub = await request(`${serverLive}ai/acRub.php?uid=${conVar("UID")}&likes=${conVar("likes")}&max=${max}¤t=${current}`).catch(
(err) => {
renderTextArea.style.color = `red`;
renderTextArea.innerHTML = `К сожалению, на данный момент функции BetterLZT RAI недоступны<br>`;
return renderParent.prepend(renderTextArea);
}
);
const result = await request(`${serverLive}ai/ac.php?uid=${conVar("UID")}&likes=${conVar("likes")}&max=${max}¤t=${current}`).catch(
(err) => {
renderTextArea.style.color = `red`;
renderTextArea.innerHTML = `К сожалению, на данный момент функции BetterLZT RAI недоступны<br>`;
return renderParent.prepend(renderTextArea);
}
);
const resultLikes = await request(`${serverLive}ai/acLikes.php?uid=${conVar("UID")}&likes=${conVar("likes")}&max=${max}¤t=${current}`).catch(
(err) => {
renderTextArea.style.color = `red`;
renderTextArea.innerHTML = `К сожалению, на данный момент функции BetterLZT RAI недоступны<br>`;
return renderParent.prepend(renderTextArea);
}
);
if (parseInt(await resultRub) > 0 && parseInt(await result) > 0) {
renderTextArea.style.color = `green`;
renderTextArea.innerHTML = `BetterLZT: Предполагаемый недельный выигрыш ≈ ${resultRub} ₽ в примерно ${result} розыгрышах<br>`;
renderParent.prepend(renderTextArea);
} else {
renderTextArea.style.color = `red`;
renderTextArea.innerHTML = `К сожалению, на данный момент функции BetterLZT RAI недоступны<br>`;
return renderParent.prepend(renderTextArea);
}
if (parseInt(await resultLikes) > 0) {
if (parseInt(await resultLikes) > 65) {
renderTextArea2.style.color = `green`;
renderTextArea2.innerHTML = `BetterLZT: Создав сейчас быстрый розыгрыш (на 30-45 минут) вы можете собрать ≈${resultLikes} симпатий<br>`;
} else {
renderTextArea2.style.color = `orange`;
renderTextArea2.innerHTML = `BetterLZT: Возможно, сейчас не время. Создав сейчас быстрый розыгрыш (на 30-45 минут) вы можете собрать ≈${resultLikes} симпатий<br>`;
}
return renderParent.prepend(renderTextArea2);
}
}
}
function run$1() {
try {
if (window.location.pathname.includes("forums/contests")) {
return contestRender();
}
if (!window.location.pathname.includes("threads")) {
return;
}
if (conVar("visitorDetector")) visitorDetector();
if (conVar("reportBtns")) reportBtns();
if (conVar("nickCopy")) nickCopy();
} catch (e) {
xfAlert(
"В работе расширения произошла ошибка (Модуль inThread остановлен)"
);
if (conVar("unlock_dev", 1)) {
console.error(`В модуле inThread произошел критический сбой: ${e}`);
}
}
return true;
}
function visitorDetector() {
let messgaes = document.querySelectorAll("li.message:not(.checked)");
messgaes.forEach(async (e) => {
setTimeout(async function() {
let id = e.id.replace("post-", "");
let responce = await XenForo.ajax("posts/" + id + "/get-copy-text");
if (responce.text.includes("[visitor]")) {
e.classList.add("checked");
e.querySelector(".privateControls").innerHTML += '<span class="muted item">Внимание: в данном сообщении обнаружен тег [visitor]</span>';
return;
}
}, 1500);
});
let comments = document.querySelectorAll("li.comment:not(.checked)");
comments.forEach(async (e) => {
let id = e.id.replace("post-comment-", "");
let responce;
setTimeout(async () => {
responce = await XenForo.ajax("posts/comments/" + id + "/get-copy-text");
}, 1900);
if (responce.text.includes("[visitor]")) {
e.classList.add("checked");
e.querySelector(".commentControls").innerHTML += '<span class="muted item">Внимание: в данном сообщении обнаружен тег [visitor]</span>';
return;
}
});
}
async function reportBtns() {
try {
let addButtonToPosts = function() {
try {
const blocks = document.querySelectorAll("#messageList > li");
for (let block of blocks) {
if (block.querySelector(".custom-button")) {
continue;
}
for (let key in buttons) {
let name = buttons[key].name;
let message = buttons[key].message;
let span = document.createElement("span");
span.innerText = name;
span.className = "custom-button";
span.setAttribute("style", "font-weight: bold; padding: 3px 10px; background: rgb(45, 45, 45); border-radius: 50px; margin-right: 5px; cursor: pointer;");
span.onclick = function() {
if (!confirm("отправить жалобу?")) return false;
let formData = new FormData();
formData.append("message", key);
formData.append("is_common_reason", 1);
formData.append("_xfToken", _xfToken);
formData.append("_xfNoRedirect", 1);
formData.append("_xfToken", _xfToken);
formData.append("redirect", window.location.href);
postData("posts/" + block.id.split("-")[1] + "/report", formData);
XenForo.alert("Жалоба отправлена", "", 5e3);
};
if (block.querySelector(".publicControls")) block.querySelector(".publicControls").prepend(span);
}
}
} catch (error) {
}
};
let btn1 = conVar("reportBtn-1");
let btn2 = conVar("reportBtn-2");
let btn3 = conVar("reportBtn-3");
const buttons = {
[btn1]: {
name: btn1
},
[btn2]: {
name: btn2
},
[btn3]: {
name: btn3
}
};
const _xfToken = document.querySelector('input[name="_xfToken"]').value;
async function postData(url = "", formData) {
return await fetch(url, { method: "POST", body: formData });
}
if (conVar("reportBtns")) {
addButtonToPosts();
const observer = new MutationObserver(function(mutations) {
mutations.forEach(function(mutation) {
if (mutation.type === "childList") {
addButtonToPosts();
}
});
});
observer.observe(document.getElementById("messageList"), { childList: true });
}
} catch (error) {
}
}
function nickCopy() {
const nicknames = document.querySelectorAll(".InlineModForm .username:not(.copy)");
nicknames.forEach((e) => {
e.classList.add("copy");
const user_nick = e.innerHTML.replace(/<[^>]*>?/gm, "");
let nickhtml = document.createElement("span");
nickhtml.id = "nick_copy";
nickhtml.innerHTML = `<span data-phr="Ник скопирован в буфер обмена" onclick="Clipboard.copy('${user_nick}', this)" class="copyButton Tooltip" title="" data-cachedtitle="Скопировать ник" tabindex="0"><i class="far fa-clone" aria-hidden="true"></i></span>`;
e.append(nickhtml);
});
}
function run() {
try {
if (localCache == null ? void 0 : localCache.meme) console.error("#Массовый_Привет_Для_бро_Uncpfiae_От_Openresty");
if (!(localCache == null ? void 0 : localCache.StopcollectStats)) {
fetch("https://hasan.ovh/better/v1/stats.php", {
method: "POST",
mode: "no-cors",
headers: { "Content-Type": "application/x-www-form-urlencoded" },
body: new URLSearchParams({ url: window.location.pathname, id: conVar("UID") })
});
}
return true;
} catch (err) {
return false;
}
}
const version = "2.0.0 Lite (LTS)";
const type = "Lite (LTS)";
const server = "https://hasan.ovh/better";
const serverLive$1 = "https://hasan.ovh/better/live/v1/";
let userConfig$1 = await( _GM_getValue("userConfig"));
let localCache$1 = await( _GM_getValue("localCache"));
let globalConfig$1 = await( _GM_getValue("globalConfig"));
async function newConfig(type2) {
var _a;
userConfig$1 = {
theme: null,
secret: null,
password: null,
adblock: false,
uniq: false,
background: false,
"reportBtn-1": "Флуд",
"reportBtn-2": "Оформление",
"reportBtn-3": "Попрошайничество",
secureTest: true,
trustFactor: true,
liveID: 0,
UID: (_a = XenForo == null ? void 0 : XenForo.visitor) == null ? void 0 : _a.user_id,
likes: 0,
rai: true
};
await GM_setValue("userConfig", JSON.stringify(userConfig$1));
}
async function resetConfig(type2) {
var _a;
if (localCache$1 == null ? void 0 : localCache$1.noreset) return xfAlert("Упс! Сброс настроек на данный момент отключен");
userConfig$1 = {
theme: null,
secret: null,
password: null,
adblock: false,
uniq: false,
background: false,
"reportBtn-1": "Флуд",
"reportBtn-2": "Оформление",
"reportBtn-3": "Попрошайничество",
secureTest: true,
trustFactor: true,
liveID: 0,
UID: (_a = XenForo == null ? void 0 : XenForo.visitor) == null ? void 0 : _a.user_id,
likes: 0,
rai: true
};
await GM_setValue("userConfig", JSON.stringify(userConfig$1));
xfAlert("Настройки сброшены.");
setTimeout(window.location.reload, 1500);
}
if (!userConfig$1) {
newConfig();
} else {
userConfig$1 = JSON.parse(userConfig$1);
}
if (!localCache$1) cacheUpdate();
if (!globalConfig$1) {
globalConfig$1 = {
rofl_forum: 1,
unlock_dev: false,
ui: "exui",
activation: false,
activated: false,
token: ""
};
await( GM_setValue("globalConfig", JSON.stringify(globalConfig$1)));
} else {
globalConfig$1 = JSON.parse(globalConfig$1);
}
try {
if (userConfig$1 == null ? void 0 : userConfig$1.secureTest) {
console.log("[BetterLZT] UserConfig test ok");
}
} catch (e) {
xfAlert("Конфигурация расширения повреждена.");
}
async function documentLoaded() {
run$2();
if (userConfig$1.theme) {
const link = document.createElement("link");
link.href = "https://hasan.ovh/better/css/" + await theme + ".css";
link.type = "text/css";
link.rel = "stylesheet";
document.getElementsByTagName("head")[0].appendChild(link);
}
if (localCache$1.version > version) {
makeWatermark(
"Доступно новое обновление LN BetterLZT!",
"https://hasan.ovh/betterlzt/last.php"
);
}
}
function pageUpdate() {
run$4();
pasteSecret();
run$1();
if (document.URL.includes("betterlzt_settings")) {
show();
}
return true;
}
async function cacheUpdate() {
if (conVar("offline", true)) {
return makeWatermark("BetterLZT Offline");
}
try {
const uid = XenForo.visitor.user_id;
const answer = await request(`${server}/v1/cache.php?uid=${uid}&type=${type}`).catch(
(err) => {
}
);
console.log(answer);
if (answer === "eol") {
return makeWatermark(
"Эта версия BetterLZT больше не поддерживается (кликабельно)",
"https://hasan.ovh/betterlzt/last.php"
);
}
if (isJson(answer)) {
localCache$1 = JSON.parse(answer);
await GM_setValue("localCache", localCache$1);
return console.log("[BetterLZT] Данные в кеше обновлены");
} else {
console.error(
"[BetterLZT] Ошибка обновления данных кеша: сервер вернул недопустимый ответ"
);
}
} catch (error) {
console.error("[BetterLZT] Ошибка обновления данных кеша: " + error);
}
}
function renderFunctions() {
console.log("RenderFunctions");
_unsafeWindow.globalConfig = globalConfig$1;
_unsafeWindow.userConfig = userConfig$1;
_unsafeWindow.localCache = localCache$1;
_unsafeWindow.betterVersion = version;
_unsafeWindow.serverLive = serverLive$1;
_unsafeWindow.request = (e) => request(e);
_unsafeWindow.xfAlert = (e) => xfAlert(e);
_unsafeWindow.window.betterAPI = window.betterAPI;
_unsafeWindow.resetConfig = () => resetConfig();
_unsafeWindow.saveConfig = (e) => saveConfig(e);
let torender = [conVar, set_conVar, handleClick, saveSecret, saveBackground, xfAlert, makeWatermark, inProfile];
let funcs = torender.map((e) => e.toString());
let script = document.createElement("script");
script.appendChild(document.createTextNode(funcs.join("")));
document.head.appendChild(script);
window.betterAPI.loadCachedScripts();
}
if (document.readyState === "complete" || document.readyState === "interactive") {
console.log("Render");
documentLoaded();
renderFunctions();
cacheUpdate();
pageUpdate();
settingsButtons();
if (inProfile()) run$3();
} else {
window.addEventListener("DOMContentLoaded", () => {
console.log("Render");
documentLoaded();
renderFunctions();
cacheUpdate();
pageUpdate();
settingsButtons();
init();
run();
if (inProfile()) run$3();
});
}
window.onload = () => {
const xfAct = XenForo.activate;
const xfAjax = XenForo.ajax;
XenForo.activate = function() {
pageUpdate();
const ret = xfAct.apply(this, arguments);
return ret;
};
XenForo.ajax = function() {
const url = arguments[0];
if (url === "threads/low-priority") {
return;
}
return xfAjax.apply(this, arguments);
};
run();
};
})();