Greasy Fork is available in English.

Kinozal Magnetizer + TorrServer

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