您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Magnet link-icon maker for kinozal.(tv|me|guru) + "Add to TorrServer" button
// ==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  // ==/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 = ` <svg width="25" height="25" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" viewBox="0 0 59 59" xml:space="preserve"> <path style="fill:#424A60;" d="M46,41.5H26c-6.627,0-12-5.373-12-12v0c0-6.627,5.373-12,12-12h20v-14H26c-14.359,0-26,11.641-26,26 v0c0,14.359,11.641,26,26,26h20V41.5z"/> <g> <path style="fill:#C7CAC7;" d="M53,7.5h1c0.552,0,1-0.447,1-1s-0.448-1-1-1h-1c-0.552,0-1,0.447-1,1S52.448,7.5,53,7.5z"/> <path style="fill:#C7CAC7;" d="M49,7.5h1c0.552,0,1-0.447,1-1s-0.448-1-1-1h-1c-0.552,0-1,0.447-1,1S48.448,7.5,49,7.5z"/> <path style="fill:#C7CAC7;" d="M57,7.5h1c0.552,0,1-0.447,1-1s-0.448-1-1-1h-1c-0.552,0-1,0.447-1,1S56.448,7.5,57,7.5z"/> <path style="fill:#C7CAC7;" d="M54,13.5h-1c-0.552,0-1,0.447-1,1s0.448,1,1,1h1c0.552,0,1-0.447,1-1S54.552,13.5,54,13.5z"/> <path style="fill:#C7CAC7;" d="M49,15.5h1c0.552,0,1-0.447,1-1s-0.448-1-1-1h-1c-0.552,0-1,0.447-1,1S48.448,15.5,49,15.5z"/> <path style="fill:#C7CAC7;" d="M58,13.5h-1c-0.552,0-1,0.447-1,1s0.448,1,1,1h1c0.552,0,1-0.447,1-1S58.552,13.5,58,13.5z"/> <path style="fill:#C7CAC7;" d="M50,10.5c0,0.553,0.448,1,1,1h1c0.552,0,1-0.447,1-1s-0.448-1-1-1h-1C50.448,9.5,50,9.947,50,10.5z"/> <path style="fill:#C7CAC7;" d="M54,10.5c0,0.553,0.448,1,1,1h1c0.552,0,1-0.447,1-1s-0.448-1-1-1h-1C54.448,9.5,54,9.947,54,10.5z"/> <path style="fill:#C7CAC7;" d="M54,44.5h-1c-0.552,0-1,0.447-1,1s0.448,1,1,1h1c0.552,0,1-0.447,1-1S54.552,44.5,54,44.5z"/> <path style="fill:#C7CAC7;" d="M49,46.5h1c0.552,0,1-0.447,1-1s-0.448-1-1-1h-1c-0.552,0-1,0.447-1,1S48.448,46.5,49,46.5z"/> <path style="fill:#C7CAC7;" d="M58,44.5h-1c-0.552,0-1,0.447-1,1s0.448,1,1,1h1c0.552,0,1-0.447,1-1S58.552,44.5,58,44.5z"/> <path style="fill:#C7CAC7;" d="M54,52.5h-1c-0.552,0-1,0.447-1,1s0.448,1,1,1h1c0.552,0,1-0.447,1-1S54.552,52.5,54,52.5z"/> <path style="fill:#C7CAC7;" d="M50,52.5h-1c-0.552,0-1,0.447-1,1s0.448,1,1,1h1c0.552,0,1-0.447,1-1S50.552,52.5,50,52.5z"/> <path style="fill:#C7CAC7;" d="M58,52.5h-1c-0.552,0-1,0.447-1,1s0.448,1,1,1h1c0.552,0,1-0.447,1-1S58.552,52.5,58,52.5z"/> <path style="fill:#C7CAC7;" d="M53,49.5c0-0.553-0.448-1-1-1h-1c-0.552,0-1,0.447-1,1s0.448,1,1,1h1C52.552,50.5,53,50.053,53,49.5z"/> <path style="fill:#C7CAC7;" d="M57,49.5c0-0.553-0.448-1-1-1h-1c-0.552,0-1,0.447-1,1s0.448,1,1,1h1C56.552,50.5,57,50.053,57,49.5z"/> </g> <rect x="32" y="3.5" style="fill:#EBBA16;" width="14" height="14"/> <rect x="32" y="41.5" style="fill:#EBBA16;" width="14" height="14"/> </svg> ` // TorrServer icon const torrServerIcon = `<img src="" width="25px" height="25px" alt="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 = $('<li><a href="javascript:;" title="Настройки скрипта Kinozal Magnetizer"><div>Настройки</div></a></li>') $tab.click(toggleSettings) $('.menu > ul').append($tab) const modal = $(` <div id="mserj_settings" style="display: none"> <div class="header">Настройка скрипта</div> <div class="fields"> <div class="row"> <label class="label"> <input type="checkbox" id="mserj_showMagnetButton"> <span>Показывать кнопку "magnet"</span> ${magnetIcon} </label> </div> <div class="row"> <label class="label"> <input type="checkbox" id="mserj_showDownloadButton"> <span>Показывать кнопку "скачать"</span> <button class="mserj-download-btn"></button> </label> </div> <div class="row"> <label class="label"> <input type="checkbox" id="mserj_showAddToTorrServerButton"> <span>Показывать кнопку "TorrServer"</span> ${torrServerIcon} </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="justify-content: center"> <input type="button" value="Сохранить настройки" id="mserj_save_settings" /> </div> </div> </div> `) $('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('<td class="z"></td>') settings.showMagnetButton && tableHeader.prepend('<td class="z"></td>') settings.showAddToTorrServerButton && tableHeader.prepend('<td class="z"></td>') // 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', `<td style="width: 30px;"><a title="Magnet-ссылка" href="magnet:?xt=urn:btih:${torrentHash}" style="display: block; font-size: 0;">${magnetIcon}</a></td>` ) } // Adding "add to torrServer" button to the page. if (settings.showAddToTorrServerButton) { downloadCell.insertAdjacentHTML( 'beforebegin', `<td style="width: 30px;"><button class="add_to_torrserver" title="Добавить в TorrServer" style="display: block; font-size: 0; border: none; padding: 0; cursor: pointer;">${torrServerIcon}</button></td>` ) 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 } })