// ==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
}
})