Greasy Fork is available in English.

RuTor popular torrents highlight + TorrServer integration

Highlight popular torrents (based on peers) + Add to TorrServer button

// ==UserScript==
// @name            RuTor popular torrents highlight + TorrServer integration
// @description     Highlight popular torrents (based on peers) + Add to TorrServer button
// @version         1.11
// @match           *://rutor.is/*
// @match           *://rutor.org/*
// @match           *://rutor.info/*
// @run-at          document-end
// @grant           none
// @copyright       2024, MSerj
// @license         MIT
//
// @grant           GM_getValue
// @grant           GM_setValue
// @grant           GM_addStyle

// @icon 

// @namespace       https://greasyfork.org/en/users/1321619-mserj
// ==/UserScript==

// General styles
GM_addStyle('table tr.backgr td { white-space: nowrap; }')
GM_addStyle('#mserj_settings { width: 400px; min-height: 150px; position: fixed; left: 0; top: 0; background-color: #fff; border: 1px solid #a00; }')
GM_addStyle(`#mserj_settings .header {\tbackground: #ffde00;\tpadding: 10px;\tfont-weight: bold; text-align: center; }`)
GM_addStyle('#mserj_settings .fields { padding: 5px; }')
GM_addStyle('#mserj_settings .fields .row { clear: both; height: 30px; }')
GM_addStyle('#mserj_settings .fields .row .col1 { width: 300px; float: left; }')
GM_addStyle('#mserj_settings .fields .row .col2 { width: 90px; float: left; }')
GM_addStyle('#mserj_settings .mserj-color { max-width: 70px; max-height: 20px; }')

// GM_addStyle('#mserj_settings .fields .row { display: flex; margin-bottom: 10px; }')
GM_addStyle('#mserj_settings .fields .row .label { display: flex; align-items: center; }')
GM_addStyle('#mserj_settings .fields .row .label span { margin-right: 10px; }')
GM_addStyle('#mserj_settings .fields .row .label span:first-child { width: 100px; }')

// TorrServer icon
const getTorrServerIcon = (size = 25) =>
	`<img src="" width="${size}px" height="${size}px"  alt="TorrServer" />`

const createTorrServerButton = ({ id, size }) => {
	const torrServerButton = document.createElement('button')
	torrServerButton.id = `add_to_torrserver-${id}`
	torrServerButton.title = 'Добавить в TorrServer'
	torrServerButton.style.fontSize = '0px'
	torrServerButton.style.border = 'none'
	torrServerButton.style.padding = '0px'
	torrServerButton.style.cursor = 'pointer'
	torrServerButton.style.marginRight = '5px'
	torrServerButton.innerHTML = getTorrServerIcon(size)

	return torrServerButton
}

// Main script
;(() => {
	let settings = {}

	const loadSettings = () => {
		settings = {
			line_color: GM_getValue('line_color', '#ff0000'),
			mark_repack: GM_getValue('mark_repack', false),
			repack_color: GM_getValue('repack_color', '#ddffdd'),
			mark_fitGirl: GM_getValue('mark_fitGirl', true),
			fitGirl_color: GM_getValue('fitGirl_color', '#ddddff'),
			mark_highQuality: GM_getValue('mark_highQuality', true),
			highQuality_color: GM_getValue('highQuality_color', '#f4ddff'),
			mark_hidden: GM_getValue('mark_hidden', true),
			hidden_opacity: GM_getValue('hidden_opacity', 0.1),
			hidden_words: GM_getValue('hidden_words', 'МР3,FLAC,flac,КПК,Футбол,UFC,книг,книги,MP3'),

			showAddToTorrServerButton: GM_getValue('showAddToTorrServerButton', false),
			torrServerIp: GM_getValue('torrServerIp', 'localhost'),
			torrServerPort: GM_getValue('torrServerPort', 8090),
			torrServerLogin: GM_getValue('torrServerLogin', ''),
			torrServerPassword: GM_getValue('torrServerPassword', '')
		}
	}

	const setStyles = () => {
		GM_addStyle('.mserj-line { height: 3px; background-color: ' + settings.line_color + '; }')
		GM_addStyle('tr.mserj-repack td { background-color: ' + settings.repack_color + '; }')
		GM_addStyle('tr.mserj-fitGirl td { background-color: ' + settings.fitGirl_color + '; }')
		GM_addStyle('tr.mserj-4K td { background-color: ' + settings.highQuality_color + '; }')
		GM_addStyle(`tr.mserj-hidden { ${settings.hidden_opacity ? `opacity: ${settings.hidden_opacity}; display: table-row` : 'display: none'}; }`)
	}

	const toggleSettings = () => {
		const $sett_wnd = $('#mserj_settings'),
			x = parseInt(($(window).width() - $sett_wnd.width()) / 2),
			y = parseInt(($(window).height() - $sett_wnd.height()) / 2)

		$('#mserj_line_color').val(settings.line_color)

		$('#mserj_mark_repack').attr('checked', !!settings.mark_repack)
		$('#mserj_repack_color').val(settings.repack_color)

		$('#mserj_mark_fitGirl').attr('checked', !!settings.mark_fitGirl)
		$('#mserj_fitGirl_color').val(settings.fitGirl_color)

		$('#mserj_mark_highQuality').attr('checked', !!settings.mark_highQuality)
		$('#mserj_highQuality_color').val(settings.highQuality_color)

		$('#mserj_mark_hidden').attr('checked', !!settings.mark_hidden)
		$('#mserj_hidden_opacity').val(settings.hidden_opacity * 10)
		$('#mserj_hidden_words').val(settings.hidden_words)

		$('#mserj_showAddToTorrServerButton').attr('checked', !!settings.showAddToTorrServerButton)
		$('#mserj_torrServerIp').val(settings.torrServerIp)
		$('#mserj_torrServerPort').val(settings.torrServerPort)
		$('#mserj_torrServerLogin').val(settings.torrServerLogin)
		$('#mserj_torrServerPassword').val(settings.torrServerPassword)

		$('#mserj_settings').css({ left: x, top: y }).toggle('fast')
	}

	// Adding settings button & modal
	const addSettings = () => {
		const $tab = $('<a href="javascript:;" class="menu_b"><div>Настройки</div></a>')

		$tab.click(toggleSettings)
		$('#menu').append($tab)

		const $wnd = $(`
		<div id="mserj_settings" style="display: none">
		  <div class="header">Настройка скрипта</div>
		  <div class="fields">
		    <div class="row">
		      <div class="col1">Цвет полоски популярности раздачи:</div>
		      <div class="col2"><input type="color" class="mserj-color" id="mserj_line_color" /></div>
		    </div>
		    <div class="row">
		      <div class="col1"><input type="checkbox" id="mserj_mark_repack">Выделять репаки</div>
		      <div class="col2"><input type="color" class="mserj-color" id="mserj_repack_color" /></div>
		    </div>
		    <div class="row">
		    	<div class="col1"><input type="checkbox" id="mserj_mark_fitGirl">Выделять репаки от FitGirl</div>
		    	<div class="col2"><input type="color" class="mserj-color" id="mserj_fitGirl_color" /></div>
	        </div>
		  	<div class="row">
		  		<div class="col1"><input type="checkbox" id="mserj_mark_highQuality">Выделять 4K раздачи</div>
		  		<div class="col2"><input type="color" class="mserj-color" id="mserj_highQuality_color" /></div>
	        </div>
	        <div class="row">
	        	<div class="col1"><input type="checkbox" id="mserj_mark_hidden">Скрывать не интересные раздачи</div>
	        	<div class="col2"><input type="range" class="mserj-color" min="0" max="10" id="mserj_hidden_opacity" /></div>
	        </div>
		    <div class="row">
			    <div class="col1">Скрыть раздачи включающие (через запятую):</div>
			    <div class="col2"><input type="text" class="mserj-color" id="mserj_hidden_words" /></div>
		    </div>
		    <hr />
		    <div class="row">
		       <label class="label">
		         <input type="checkbox" id="mserj_showAddToTorrServerButton">
		         <span>Показывать кнопку "TorrServer"</span>
		         ${getTorrServerIcon()}
		       </label>
		     </div>
		     <div class="row">
		       <label class="label">
		         <span>TorrServer IP</span>
		         <input type="text" id="mserj_torrServerIp">
		       </label>
		     </div>
		     <div class="row">
		       <label class="label">
		         <span>TorrServer Port</span>
		         <input type="text" id="mserj_torrServerPort">
		       </label>
		     </div>
		     <div class="row">
		       <label class="label">
		         <span>TorrServer Login</span>
		         <input type="text" id="mserj_torrServerLogin">
		       </label>
		     </div>
		     <div class="row">
		       <label class="label">
		         <span>TorrServer Password</span>
		         <input type="password" id="mserj_torrServerPassword">
		       </label>
		     </div>
		    <div class="row" style="margin-top: 10px; text-align: center"><input type="button" value="Сохранить настройки" id="mserj_save_settings" /></div>
		  </div>
		</div>
		`)

		$('body').append($wnd)

		$('#mserj_save_settings').live('click', () => {
			GM_setValue('line_color', $('#mserj_line_color').val())
			GM_setValue('mark_repack', $('#mserj_mark_repack').is(':checked'))
			GM_setValue('repack_color', $('#mserj_repack_color').val())
			GM_setValue('mark_fitGirl', $('#mserj_mark_fitGirl').is(':checked'))
			GM_setValue('fitGirl_color', $('#mserj_fitGirl_color').val())
			GM_setValue('mark_highQuality', $('#mserj_mark_highQuality').is(':checked'))
			GM_setValue('highQuality_color', $('#mserj_highQuality_color').val())
			GM_setValue('mark_hidden', $('#mserj_mark_hidden').is(':checked'))
			GM_setValue('hidden_opacity', $('#mserj_hidden_opacity').val() * 0.1)
			GM_setValue('hidden_words', $('#mserj_hidden_words').val())

			GM_setValue('showAddToTorrServerButton', $('#mserj_showAddToTorrServerButton').is(':checked'))
			GM_setValue('torrServerIp', $('#mserj_torrServerIp').val())
			GM_setValue('torrServerPort', $('#mserj_torrServerPort').val())
			GM_setValue('torrServerLogin', $('#mserj_torrServerLogin').val())
			GM_setValue('torrServerPassword', $('#mserj_torrServerPassword').val())

			location.reload()
		})
	}

	/**
	 * TorrServer stuff
	 */
	async function addToTorrServer(data) {
		try {
			const response = await fetch(`${settings.torrServerIp}:${settings.torrServerPort}/torrents`, {
				method: 'POST',
				headers: {
					'Content-Type': 'application/json',
					Authorization:
						settings.torrServerLogin && settings.torrServerPassword
							? `Basic ${btoa(settings.torrServerLogin + ':' + settings.torrServerPassword)}`
							: undefined
				},
				body: JSON.stringify({ action: 'add', save_to_db: true, ...data })
			})
			if (response.ok) {
				alert('Успешно добавлено в TorrServer')
			} else {
				if (response.status === 401) {
					alert('Авторизация не удалась! Проверьте ( соединение / логин / пароль )')
				} else {
					alert('Не удалось отправить запрос на TorrServer')
				}
			}
		} catch (error) {
			alert('Не удалось отправить запрос на TorrServer')
		}
	}

	// Adding "add to torrServer" button on details page
	const addTorrServerButtonOnDetailPage = () => {
		const downloadSection = document.getElementById('download')

		if (downloadSection) {
			const magnetLink = downloadSection.querySelectorAll('a')[0].href

			const id = location.href.split('torrent/')[1].split('/')[0]
			const torrServerButton = createTorrServerButton({ id, size: 30 })
			downloadSection.prepend(torrServerButton)

			$(`#add_to_torrserver-${id}`).bind('click', () => {
				addToTorrServer({
					link: magnetLink.split('&dn')[0],
					poster: $('#details img')[0].src
				})
			})
		}
	}

	// Fetch torrent poster
	async function fetchTorrentPoster(url) {
		try {
			const response = await fetch(url)
			if (!response.ok) {
				throw new Error(`HTTP error! status: ${response.status}`)
			}
			const htmlString = await response.text()
			const parser = new DOMParser()
			const doc = parser.parseFromString(htmlString, 'text/html')

			const poster = doc.querySelector('#details img')
			if (poster) {
				return poster.src
			} else {
				return null
			}
		} catch (error) {
			console.error('Error fetching or parsing HTML:', error)
			return null
		}
	}

	// Mark/highlight lines
	const markLines = () => {
		const max_width = $(window).width() - 280 - 214

		$('tr.gai, tr.tum').each(function () {
			const cells = $(this).find('td'),
				links_cell = cells.get(1),
				peers_spans = $(cells.get().pop()).find('span'),
				links = $(links_cell).find('a'),
				magnetLink = links.get(1).href,
				titleLink = links.length === 2 ? links.get(1) : links.get(2)
			const count = (parseInt($.trim($(peers_spans.get(0)).text())) + parseInt($.trim($(peers_spans.get(1)).text()))) * 1.3

			$(links_cell).append('<div class="mserj-line" style="width: ' + Math.min(max_width, parseInt(count / 1)) + 'px"></div>')

			// Adding "add to torrServer" button the row
			if (settings.showAddToTorrServerButton && magnetLink) {
				const id = titleLink.href.split('torrent/')[1].split('/')[0]
				const torrServerButton = createTorrServerButton({ id, size: 13 })
				links_cell.insertBefore(torrServerButton, titleLink)

				$(`#add_to_torrserver-${id}`).bind('click', () => {
					;(async () => {
						const poster = await fetchTorrentPoster(titleLink)
						addToTorrServer({
							link: magnetLink.split('&dn')[0],
							poster
						})
					})()
				})
			}

			if (settings.mark_repack && (titleLink.innerHTML.includes('RePack') || titleLink.innerHTML.includes('repack'))) {
				$(this).addClass('mserj-repack')
			}
			if (settings.mark_fitGirl && (titleLink.innerHTML.includes('FitGirl') || titleLink.innerHTML.includes('fitgirl'))) {
				$(this).addClass('mserj-fitGirl')
			}
			if (settings.mark_highQuality && (titleLink.innerHTML.includes(' 4K') || titleLink.innerHTML.includes('2160p'))) {
				$(this).addClass('mserj-4K')
			}
			if (settings.mark_hidden && settings.hidden_words.split(',').some(word => titleLink.innerHTML.includes(word))) {
				$(this).addClass('mserj-hidden')
			}
		})
	}

	/**
	 * Sorting functionality
	 */
	function sortByColumn(sortWhat, type, field, btnIndex) {
		let dataClicked = sortWhat.sorti[btnIndex].press,
			press = this

		sortWhat = Object.assign({}, sortWhat)

		if (type === 0) {
			sortWhat.razd.sort(function (a, b) {
				const an = a[field],
					bn = b[field]
				return an - bn
			})
		} else if (type === 1) {
			sortWhat.razd.sort(function (a, b) {
				const x = a[field].toLowerCase()
				const y = b[field].toLowerCase()

				if (x < y) return -1
				if (x > y) return 1
				return 0
			})
		}

		if (dataClicked) sortWhat.razd.reverse()

		for (let i = 0; i < sortWhat.razd.length; i++) {
			let elDetach = $(sortWhat.razd[i].es),
				childs = null

			if (elDetach.next().next().is('.my_tr')) childs = [elDetach.next(), elDetach.next().next()]

			elDetach.detach().appendTo(sortWhat.category)

			if (childs != null) {
				$(childs[1]).detach().insertAfter(elDetach)
				$(childs[0]).detach().insertAfter(elDetach)
			}
		}

		if (btnIndex === 3 || btnIndex === 4) {
			press = $(sortWhat.sorti[btnIndex].el_img)
		}

		sortWhat.sorti.map(function (currArr, indexArr) {
			if (indexArr !== btnIndex) {
				currArr.press = false

				if ($(currArr.el_img).is('img')) $(currArr.el_img).css('transform', 'scaleY(1)')
				else $(currArr.el_img).find('img').css('transform', 'scaleY(1)')
			}
		})

		if ($(press).is('img')) {
			$(press).css('transform', 'scaleY(' + (dataClicked ? '1' : '-1') + ')')
		} else {
			$(press)
				.find('img[width^=15]')
				.css('transform', 'scaleY(' + (dataClicked ? '1' : '-1') + ')')
		}

		sortWhat.sorti[btnIndex].press = !sortWhat.sorti[btnIndex].press
	}

	// Header events
	function setEventHeaderTitle(massiv) {
		let titleSort = [],
			titles = $(this)
				.find('.backgr > td')
				.each(function (indexEl, el) {
					if (indexEl === 3 && el.textContent === 'Пиры') {
						let img = $('<img>')
								.attr({ src: 'https://raw.githubusercontent.com/AlekPet/Rutor-Preview-Ajax/master/assets/images/arrow_icon.gif', width: '15' })
								.css({ position: 'relative', top: '3px', cursor: 'pointer' })
								.attr({ title: 'Сортировать по Раздающим', id: '_Up' }),
							img_clone = img.clone(false).attr({ title: 'Сортировать по Качающим', id: '_Down' })
						$(el)
							.css({ width: '90px' })
							.append($('<span class="green">').text(' Р').css({ cursor: 'pointer' }).attr({ title: 'Сортировать по Раздающим', id: '_Up' }))
							.append(img)
							.append($('<span class="red">>').text('К').css({ cursor: 'pointer' }).attr({ title: 'Сортировать по Качающим', id: '_Down' }))
							.append(img_clone)
						titleSort.push(
							{
								el_img: img,
								index: indexEl,
								press: false
							},
							{
								el_img: img_clone,
								index: indexEl + 1,
								press: false
							}
						)
					} else {
						let img = $('<img>')
							.attr({ src: 'https://raw.githubusercontent.com/AlekPet/Rutor-Preview-Ajax/master/assets/images/arrow_icon.gif', width: '15' })
							.css({ position: 'relative', top: $(el).children().first().is('img') ? '-10px' : '3px' })
						$(el)
							.css({ width: '80px', cursor: 'pointer' })
							.attr('title', 'Сортировать по "' + ($(el).children().first().is('img') ? 'Добавлено' : $(el).text()) + '"')
							.append(img)
						titleSort.push({
							el_img: el,
							index: indexEl,
							press: false
						})
					}
				})
		massiv.sorti = titleSort
		// By date
		titles.eq(0).click(function () {
			sortByColumn.call(this, massiv, 0, 'date', 0)
		})
		// By name
		titles.eq(1).click(function () {
			sortByColumn.call(this, massiv, 1, 'name', 1)
		})
		// By size
		titles.eq(2).click(function () {
			sortByColumn.call(this, massiv, 0, 'size', 2)
		})
		// By Up/Down
		titles
			.eq(3)
			.find('div, img')
			.each(function (index, el) {
				$(this).click(function () {
					if (el.id === '_Up') {
						sortByColumn.call(el, massiv, 0, 'up', 3)
					} else {
						sortByColumn.call(el, massiv, 0, 'down', 4)
					}
				})
			})
	}

	function sorting() {
		if (!location.href.includes('/torrent/')) {
			// Ищим классы для получения данных
			let massivT = [],
				month = ['Январь', 'Февраль', 'Март', 'Апрель', 'Май', 'Июнь', 'Июль', 'Август', 'Сентябрь', 'Октябрь', 'Ноябрь', 'Декабрь'],
				razmeronosti = ['kB', 'MB', 'GB']

			$('#index > table').each(function (idx) {
				let objCat = { category: this, name: $(this).prev().text(), razd: [] }

				$(this)
					.find('.gai, .tum')
					.each(function () {
						const rowWithComments = this.children.length === 5

						const dateCell = this.children[0]
						const nameCell = this.children[1]
						const sizeCell = this.children[rowWithComments ? 3 : 2]
						const peersCell = this.children[rowWithComments ? 4 : 3]

						const name = nameCell.children[2].textContent || nameCell.children[0].getAttribute('title')
						const size = sizeCell.textContent
						const colR = peersCell.children[0].textContent
						const colU = peersCell.children[2].textContent

						// Date
						let dateT = dateCell.textContent
						dateT = dateT.split(/\s+/)

						$.each(month, function (idx, val) {
							if (dateT[1] === val.substr(0, 3)) dateT[1] = idx
						})

						dateT = new Date(parseInt('20' + dateT[2]), dateT[1], dateT[0], 0, 0, 0)

						// Sizes
						let complSize
						$.each(razmeronosti, function (idx, val) {
							if (size.includes(val)) {
								complSize = size.substr(0, size.indexOf(razmeronosti[idx])) * 1
								if (idx === 1) {
									complSize = complSize * 1000
								} else if (idx === 2) {
									complSize = complSize * 1000000
								}
							} else {
								complSize = parseFloat(size)
							}
						})

						objCat.razd.push({ es: this, date: dateT, name, size: complSize, up: colR, down: colU })
					})
				massivT.push(objCat)
				setEventHeaderTitle.call(this, massivT[idx])
			})
		}
	}

	// settings
	loadSettings()
	setStyles()
	addSettings()
	markLines()
	addTorrServerButtonOnDetailPage()

	sorting()
})()