// ==UserScript==
// @name RuTor popular torrents highlight + TorrServer integration
// @description Highlight popular torrents (based on peers), + Add to TorrServer button
// @version 1.02
// @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('#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" />`
// 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
*/
function addToTorrServer(data) {
$.ajax({
type: 'POST',
url: `${settings.torrServerIp}:${settings.torrServerPort}/torrents`,
dataType: 'json',
data: JSON.stringify({ action: 'add', save_to_db: true, ...data }),
contentType: 'application/json',
beforeSend: function (xhr) {
if (settings.torrServerLogin && settings.torrServerPassword) {
xhr.setRequestHeader('Authorization', 'Basic ' + btoa(settings.torrServerLogin + ':' + settings.torrServerPassword))
}
},
success: ok => {
alert('Успешно добавлено в TorrServer')
},
error: response => {
if (response.status === 401) {
alert('Авторизация не удалась! Проверьте ( соединение / логин / пароль )')
} else {
alert('Не удалось отправить запрос на TorrServer')
}
}
})
}
// 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 to the page.
if (settings.showAddToTorrServerButton && magnetLink) {
// Create torrServer button
const id = titleLink.href.split('torrent/')[1].split('/')[0]
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(13)
links_cell.insertBefore(torrServerButton, titleLink)
$(`#add_to_torrserver-${id}`).bind('click', () => {
addToTorrServer({
link: magnetLink.split('&dn')[0]
})
})
}
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 (var 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()
sorting()
})()