您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
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.10 // @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" />` 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 */ 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') } } }) } // Adding "add to torrServer" button on details page const addTorrServerButtonOnDetailPage = () => { const downloadSection = $('#download') const magnetLink = downloadSection.find('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 }) }) } // 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', () => { 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() addTorrServerButtonOnDetailPage() sorting() })()