Carrizo / BR | Tex Script for Forum

// ==UserScript==
// @name         BR | Tex Script for Forum
// @namespace    https://forum.blackrussia.online
// @version      1.0.4.16
// @description  try to take over the world!
// @author       Carrizo
// @match        https://forum.blackrussia.online/index.php?threads/*
// @include      https://forum.blackrussia.online/index.php?threads/
// @grant        none
// @license 	 MIT
// @collaborator Carrizo
// @icon https://icons.iconarchive.com/icons/aha-soft/iron-man/48/Ironman-Mask-3-Old-icon.png
// @copyright 2021, Carrizo (https://openuserjs.org/users/Carrizo)
// @updateURL https://openuserjs.org/meta/Carrizo/BR_Tex_Script_for_Forum.meta.js
// ==/UserScript==

(function () {
  'use strict';
const UNACCEPT_PREFIX = 4; // Prefix that will be set when thread closes
const ACCEPT_PREFIX = 8; // Prefix that will be set when thread accepted
const PIN_PREFIX = 2; // Prefix that will be set when thread pins
const COMMAND_PREFIX = 10; // Prefix that will be set when thread send to project team
const WATCHED_PREFIX = 9; 
const CLOSE_PREFIX = 7; 
const buttons = [
	{
	  title: 'Приветствие',
	  content:
		'[FONT=Courier New][CENTER]{{ greeting }}, уважаемый {{ user.mention }}![/CENTER]<br>' + '[CENTER]  [/CENTER][/FONT]',
	},
	{
	  title: 'Правила раздела',
	  content:
		'[FONT=Courier New][CENTER]{{ greeting }}, уважаемый {{ user.mention }}[/CENTER]<br>' +
		"[CENTER]Пожалуйста, убедительная просьба, ознакомится с назначением данного раздела в котором Вы создали тему, так как ваш запрос никоим образом не относится к технической проблеме.[/CENTER]" +
		'[CENTER]Отказано, закрыто.[/CENTER][/FONT]',
	},
	{
	  title: 'Восстановления',
	  content:
		'[FONT=Courier New][CENTER]{{ greeting }}, уважаемый {{ user.mention }}[/CENTER]<br>' +
		"[CENTER]Пожалуйста, убедительная просьба, ознакомится с правилами восстановлений https://clck.ru/NeHEQ. Вы создали тему, которая никоим образом не относится к технической проблеме. Восстановлено – не будет.[/CENTER]<br>" +
		'[CENTER]Отказано, закрыто.[/CENTER][/FONT]',
	},
	{
	  title: 'Краш/вылет',
	  content:
		'[FONT=Courier New][CENTER]{{ greeting }}, уважаемый {{ user.mention }}[/CENTER]<br>' +
		"[CENTER]В том случае, если Вы вылетели из игры во время игрового процесса (произошел краш), в обязательно порядке необходимо обратиться в сообщество ВКонтакте – https://vk.com/brclient. [/CENTER]<br>" +
		"[CENTER][CODE]1. Ваш никнейм, сервер [Lev_Kalashnikov, Black]:<br>2. В каком моменте произошел краш (совершали какие-то действия? где-то ездили? и прочее, описать подробно):<br>3. Скриншот места на карте, где произошел краш:<br>4. Модель телефона:[/CODE]<br><br>" +
		'[CENTER]Решено, заполните данную форму в личных сообщениях сообщества.[/CENTER][/FONT]',
	},
	{
	  title: 'Дублирование темы',
	  content:
		'[FONT=Courier New][CENTER]{{ greeting }}, уважаемый {{ user.mention }}[/CENTER]<br>' +
		"[CENTER]Дублирование темы. Напоминаем, при 3 дублированиях – форумный аккаунт будет заблокирован.<br>" +
		'[CENTER]Отказано, закрыто.[/CENTER][/FONT]',
	},
	{
	  title: 'Форма темы',
	  content:
		'[FONT=Courier New][CENTER]{{ greeting }}, уважаемый {{ user.mention }}[/CENTER]<br>' +
		"[CENTER]Пожалуйста, заполните форму, создав новую тему: <br>[CODE]01. Ваш игровой никнейм:<br>02. Сервер, на котором Вы играете:<br>03. Суть Вашего возникшей проблемы (описать максимально подробно и раскрыто): <br>04. Любые скриншоты, которые могут помочь в решении проблемы (если таковые имеются):<br>05. Дата и время произошедшей технической проблемы (постарайтесь указать максимально точно):[/CODE]<br><br>" +
		'[CENTER]Отказано, закрыто.[/CENTER][/FONT]',
	},
	{
	  title: 'Известно о проблеме',
	  content:
		'[FONT=Courier New][CENTER]{{ greeting }}, уважаемый {{ user.mention }}[/CENTER]<br>' +
		"[CENTER]Разработчикам проекта уже известно о данной проблеме. Данная недоработка на стадии исправления.<br><br>" +
		'[CENTER]Закрыто.[/CENTER][/FONT]',
	},
	{
	  title: 'Команде проекта',
	  content:
		'[FONT=Courier New][CENTER]{{ greeting }}, уважаемый {{ user.mention }}[/CENTER]<br>' +
		"[CENTER]Ваша теме перенесена, и находится на рассмотрении. Пожалуйста, ожидайте выноса вердикта разработчиков.<br><br>" +
		'[CENTER]Создавать новые темы с данной проблемой — не нужно.[/CENTER][/FONT]',
	},
	{
	  title: 'Жалобы на адм',
	  content:
		'[FONT=Courier New][CENTER]{{ greeting }}, уважаемый {{ user.mention }}[/CENTER]<br>' +
		"[CENTER]Обратитесь в раздел «Жалобы» Вашего сервера:<br> [B]Сервер №1 | Red[/B] → https://clck.ru/PF2bE <br>[B]Сервер №2 | Green[/B] → https://clck.ru/PF2bx<br>[B]Сервер №3 | Blue[/B] → https://clck.ru/PF2cm<br>[B]Сервер №4 | Yellow[/B] → https://clck.ru/PF2dJ<br>[B]Сервер №5 | Orange[/B] → https://clck.ru/RWXCM<br>[B]Сервер №6 | Purple[/B] → https://clck.ru/RcNbF<br>[B]Сервер №7 | Lime[/B] → https://clck.ru/RreiU<br>[B]Сервер №8 | Pink[/B] → https://clck.ru/SKwPq<br>[B]Сервер №9 | Cherry[/B] → https://clck.ru/SRgaB<br>[B]Сервер №10 | Black[/B] → https://clck.ru/Seu43<br>[B]Сервер №11 | Indigo[/B] → https://clck.ru/TUaQP<br>[B] Сервер №12 | White[/B] → https://clck.ru/TqE9R <br>[B] Сервер №13 | Magenta[/B] → https://vk.cc/c1N4Pl" +
		'[CENTER]Отказано, закрыто.[/CENTER][/FONT]',
	},
	{
	  title: 'Нет нужных докв для возврата',
	  content:
		'[FONT=Courier New][CENTER]{{ greeting }}, уважаемый {{ user.mention }}[/CENTER]<br>' +
		"[CENTER]Нам очень жаль, но без доказательств (в частности скриншот или видео) – вернуть ничего не получится. Если таковы найдутся, создайте новую тему, приложив доказательства и ссылку на тему, в которой написано данное сообщение.<br><br>" +
		'[CENTER]Не расстраивайтесь, и всего доброго! Отказано, закрыто.[/CENTER][/FONT]',
	},
	{
	  title: 'Сбор пароля',
	  content:
		'[FONT=Courier New][CENTER]{{ greeting }}, уважаемый {{ user.mention }}[/CENTER]<br>' +
		"[CENTER]Если Вы обезопасили Ваш аккаунт и привязали его к странице во ВКонтакте или к почте, то сбросить пароль Вы всегда сможете обратившись в официальное сообщество проекта - https://vk.com/blackrussia.online. Напишите 'Начать' в личные сообщения группы, затем выберите нужные Вам функции.<br><br>" +
		'[CENTER]К сожалению, иногда решение подобных вопросов требует много времени. А пока что ждём вместе с Вами восстановления пароля – Рассмотрено.[/CENTER][/FONT]',
	},
];

$(document).ready(() => {
// Загрузка скрипта для обработки шаблонов
$('body').append('<script src="https://cdn.jsdelivr.net/npm/handlebars@latest/dist/handlebars.js"></script>');

// Добавление кнопок при загрузке страницы
addButton('На рассмотрение', 'pin');
addButton('КП', 'teamProject');
addButton('Отказ', 'unaccept');
addButton('Рассмотрено', 'watched');
addButton('Одобрено', 'accepted');
addButton('Закрыто', 'closed');
addButton('Ответы', 'selectAnswer');

// Поиск информации о теме
const threadData = getThreadData();

$('button#unaccept').click(() => editThreadData(UNACCEPT_PREFIX, false));
$('button#pin').click(() => editThreadData(PIN_PREFIX, true));
$('button#accepted').click(() => editThreadData(ACCEPT_PREFIX, false));
$('button#teamProject').click(() => editThreadData(COMMAND_PREFIX, true));
$('button#watched').click(() => editThreadData(WATCHED_PREFIX, false));
$('button#closed').click(() => editThreadData(CLOSE_PREFIX, false));

$(`button#selectAnswer`).click(() => {
  XF.alert(buttonsMarkup(buttons), null, 'Выберите ответ:');
  buttons.forEach((btn, id) => {
	$(`button#answers-${id}`).click(() => pasteContent(id, threadData));
  });
});
});

function addButton(name, id) {
$('.button--icon--reply').before(
  `<button type="button" class="button rippleButton" id="${id}" style="margin: 3px;">${name}</button>`,
);
}

function buttonsMarkup(buttons) {
return `<div class="select_answer">${buttons
  .map(
	(btn, i) =>
	  `<button id="answers-${i}" class="button--primary button ` +
	  `rippleButton" style="margin:5px"><span class="button-text">${btn.title}</span></button>`,
  )
  .join('')}</div>`;
}

function pasteContent(id, data = {}) {
const template = Handlebars.compile(buttons[id].content);
if ($('.fr-element.fr-view p').text() === '') $('.fr-element.fr-view p').empty();

$('span.fr-placeholder').empty();
$('div.fr-element.fr-view p').append(template(data));
$('a.overlay-titleCloser').trigger('click');
}

function getThreadData() {
const authorID = $('a.username')[0].attributes['data-user-id'].nodeValue;
const authorName = $('a.username').html();
const hours = new Date().getHours();
return {
  user: {
	id: authorID,
	name: authorName,
	mention: `[USER=${authorID}]${authorName}[/USER]`,
  },
  greeting: () =>
	4 < hours && hours <= 11
	  ? 'Доброе утро'
	  : 11 < hours && hours <= 15
	  ? 'Добрый день'
	  : 15 < hours && hours <= 21
	  ? 'Добрый вечер'
	  : 'Доброй ночи',
};
}

function editThreadData(prefix, pin = false) {
// Получаем заголовок темы, так как он необходим при запросе
	const threadTitle = $('.p-title-value')[0].lastChild.textContent;

	if(pin == false){
		fetch(`${document.URL}edit`, {
		  method: 'POST',
		  body: getFormData({
			prefix_id: prefix,
			title: threadTitle,
			_xfToken: XF.config.csrf,
			_xfRequestUri: document.URL.split(XF.config.url.fullBase)[1],
			_xfWithData: 1,
			_xfResponseType: 'json',
		  }),
		}).then(() => location.reload());
	} else  {
		fetch(`${document.URL}edit`, {
		  method: 'POST',
		  body: getFormData({
			prefix_id: prefix,
			title: threadTitle,
			pin: 1,
			_xfToken: XF.config.csrf,
			_xfRequestUri: document.URL.split(XF.config.url.fullBase)[1],
			_xfWithData: 1,
			_xfResponseType: 'json',
		  }),
		}).then(() => location.reload());
	}
	if(prefix == UNACCEPT_PREFIX || prefix == ACCEPT_PREFIX || prefix == CLOSE_PREFIX || prefix == WATCHED_PREFIX) {
		moveThread(prefix, 230);
	}
}

function moveThread(prefix, type) {
// Получаем заголовок темы, так как он необходим при запросе
const threadTitle = $('.p-title-value')[0].lastChild.textContent;

fetch(`${document.URL}move`, {
  method: 'POST',
  body: getFormData({
	prefix_id: prefix,
	title: threadTitle,
	target_node_id: type,
	redirect_type: 'none',
	notify_watchers: 1,
	starter_alert: 1,
	starter_alert_reason: "",
	_xfToken: XF.config.csrf,
	_xfRequestUri: document.URL.split(XF.config.url.fullBase)[1],
	_xfWithData: 1,
	_xfResponseType: 'json',
  }),
}).then(() => location.reload());
}

function getFormData(data) {
	const formData = new FormData();
	Object.entries(data).forEach(i => formData.append(i[0], i[1]));
	return formData;
  }
})();