// ==UserScript== // @name Kinozal Magnetizer + TorrServer // @description Magnet link-icon maker for kinozal.(tv|me|guru) + "Add to TorrServer" button // @version 1.12 // @match *://kinozal.tv/details.php* // @match *://kinozal.me/details.php* // @match *://kinozal.guru/details.php* // @match *://kinozal.tv/browse.php* // @match *://kinozal.me/browse.php* // @match *://kinozal.guru/browse.php* // @run-at document-end // @grant GM_getValue // @grant GM_setValue // @grant GM_addStyle // @copyright 2024, MSerj // @license MIT // @namespace https://greasyfork.org/en/users/1321619-mserj // @icon  // @downloadURL https://update.greasyfork.icu/scripts/498528/Kinozal%20Magnetizer%20%2B%20TorrServer.user.js // @updateURL https://update.greasyfork.icu/scripts/498528/Kinozal%20Magnetizer%20%2B%20TorrServer.meta.js // ==/UserScript== // Styles for the download button GM_addStyle(`.mserj-download-btn { display: inline-block; height: 32px; width: 32px; border: none; background-image: url(); background-size: 32px 32px; }`) // Styles for setting modal 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: #f1d29c;\tpadding: 10px;\tfont-weight: bold; text-align: center; }`) GM_addStyle('#mserj_settings .fields { padding: 5px; }') 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; }') // Magnet icon SVG data const magnetIcon = ` ` // TorrServer icon const torrServerIcon = `TorrServer` /** * Settings stuff */ let settings = {} const loadSettings = () => { settings = { showMagnetButton: GM_getValue('showMagnetButton', true), showDownloadButton: GM_getValue('showDownloadButton', false), showAddToTorrServerButton: GM_getValue('showAddToTorrServerButton', false), torrServerIp: GM_getValue('torrServerIp', 'localhost'), torrServerPort: GM_getValue('torrServerPort', 8090), torrServerLogin: GM_getValue('torrServerLogin', ''), torrServerPassword: GM_getValue('torrServerPassword', '') } } // modal to configure settings const toggleSettings = () => { const $sett_wnd = $('#mserj_settings'), x = parseInt(($(window).width() - $sett_wnd.width()) / 2), y = parseInt(($(window).height() - $sett_wnd.height()) / 2) $('#mserj_showMagnetButton').attr('checked', !!settings.showMagnetButton) $('#mserj_showDownloadButton').attr('checked', !!settings.showDownloadButton) $('#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') } const attachSettingsModal = () => { const $tab = $('
  • Настройки
  • ') $tab.click(toggleSettings) $('.menu > ul').append($tab) const modal = $(` `) $('body').append(modal) $('#mserj_save_settings').on('click', () => { GM_setValue('showMagnetButton', $('#mserj_showMagnetButton').is(':checked')) GM_setValue('showDownloadButton', $('#mserj_showDownloadButton').is(':checked')) 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()) loadSettings() $('#mserj_settings').toggle('fast') location.reload() }) } /** * TorrServer stuff */ function addToTorrServer(data) { $.ajax({ method: 'POST', url: `${settings.torrServerIp}:${settings.torrServerPort}/torrents`, dataType: 'json', data: JSON.stringify({ action: 'add', save_to_db: true, ...data }), headers: { 'Content-Type': 'application/json', ...(settings.torrServerLogin && settings.torrServerPassword && { Authorization: 'Basic ' + btoa(settings.torrServerLogin + ':' + settings.torrServerPassword) }) }, success: () => { alert('Успешно добавлено в TorrServer') }, error: response => { if (response.status === 401) { alert('Авторизация не удалась! Проверьте ( соединение / логин / пароль )') } else { alert('Не удалось отправить запрос на TorrServer') } } }) } // 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('img.p200') if (poster) { return poster.src } else { return null } } catch (error) { console.error('Error fetching or parsing HTML:', error) return null } } /** * handle search page case */ const processSearchPage = () => { // Function to fetch torrent hash and add download/magnet links async function processTorrentRow(row) { const torrentUrl = $(row).find('.nam a').attr('href') const uArgs = torrentUrl.split('?')[1].split('&') // Find torrent id let id = uArgs.find(el => el.startsWith('id='))?.split('=')[1] if (id) { if (settings.showDownloadButton) { // Create download button const downloadCell = document.createElement('td') const link = document.createElement('a') link.className = 'mserj-download-btn' link.href = `${location.origin}/download.php?id=${id}` downloadCell.appendChild(link) row.insertBefore(downloadCell, row.firstChild) } if (settings.showMagnetButton || settings.showAddToTorrServerButton) { // Fetch torrent hash const response = await fetch(`/get_srv_details.php?id=${id}&action=2`) const html = await response.text() const dom = new DOMParser().parseFromString(html, 'text/html') const torrentHash = dom.querySelector('ul > li:first-child')?.innerText.substr(10) if (settings.showMagnetButton && torrentHash) { // Create magnet link const magnetCell = document.createElement('td') const magnetLink = document.createElement('a') magnetLink.title = 'Magnet-ссылка' // Assuming 'ссылка' means 'link' magnetLink.href = `magnet:?xt=urn:btih:${torrentHash}` magnetLink.style.display = 'block' magnetLink.style.fontSize = '0px' magnetLink.innerHTML = magnetIcon magnetCell.appendChild(magnetLink) row.insertBefore(magnetCell, row.firstChild) } // Adding "add to torrServer" button to the page. if (settings.showAddToTorrServerButton && torrentHash) { // Create torrServer button const torrServerCell = document.createElement('td') const torrServerButton = document.createElement('button') torrServerButton.id = `add_to_torrserver-${id}` torrServerButton.title = 'Добавить в TorrServer' torrServerButton.style.display = 'block' torrServerButton.style.fontSize = '0px' torrServerButton.style.border = 'none' torrServerButton.style.padding = '0px' torrServerButton.style.cursor = 'pointer' torrServerButton.innerHTML = torrServerIcon torrServerCell.appendChild(torrServerButton) if (settings.showMagnetButton) { row.firstChild.parentNode.insertBefore(torrServerCell, row.firstChild.nextSibling) // something like row.insertAfter(torrServerCell, row.firstChild) } else { row.insertBefore(torrServerCell, row.firstChild) } $(`#add_to_torrserver-${id}`).on('click', () => { ;(async () => { const poster = await fetchTorrentPoster(torrentUrl) addToTorrServer({ link: `magnet:?xt=urn:btih:${torrentHash}`, poster }) })() }) } } } } const table = $('.t_peer') const tableHeader = table.find('.mn') // Add empty cells for download and magnet links in the table header settings.showDownloadButton && tableHeader.prepend('') settings.showMagnetButton && tableHeader.prepend('') settings.showAddToTorrServerButton && tableHeader.prepend('') // Process each row in the table (excluding the header row) table .find('tr') .not(tableHeader) .each((i, row) => { processTorrentRow(row) }) } /** * handle details page case */ const processDetailsPage = async () => { // Finding download button cell. const downloadCell = document.querySelector('.w100p td:first-of-type') // Fetching torrent hash string. const response = await (await fetch(`/get_srv_details.php?id=${new URL(location.href).searchParams.get('id')}&action=2`)).text() // Converting response text to dom element, so we can easily traverse and extract torrent hash with querySelector. const dom = new DOMParser().parseFromString(response, 'text/html') const torrentHash = dom.documentElement.querySelector('ul > li:first-child').innerText.substr(10) // Adding magnet link to the page. if (settings.showMagnetButton) { downloadCell.insertAdjacentHTML( 'beforebegin', `${magnetIcon}` ) } // Adding "add to torrServer" button to the page. if (settings.showAddToTorrServerButton) { downloadCell.insertAdjacentHTML( 'beforebegin', `` ) const poster = $('.p200').attr('src') $('.add_to_torrserver').on('click', e => { addToTorrServer({ link: `magnet:?xt=urn:btih:${torrentHash}`, poster: poster.startsWith('http') ? poster : `${location.origin}/${poster}` }) }) } } /** * Main script starts here after page is ready */ $(document).ready(function () { loadSettings() attachSettingsModal() switch (location.pathname) { case '/details.php': processDetailsPage() break case '/browse.php': processSearchPage() break } })