Kinopoisk+

Adds links to search for popular torrent sites

/* globals GM, GM.xmlHttpRequest, GM_setValue, GM_getValue, GM_info */
// ==UserScript==
// @name            Kinopoisk+
// @name:ru         Кинопоиск+
// @description     Adds links to search for popular torrent sites
// @description:ru  Добавляет ссылки для поиска по популярным торрент-сайтам
// @namespace       kp.user.js
// @version         1.1.8
// @author          Xant1k@bt (2015-2017), askornot (2020-2022)
// @license         MIT
// @icon            https://icons.duckduckgo.com/ip9/kinopoisk.ru.ico
// @homepageURL     https://greasyfork.org/en/scripts/418547-kinopoisk
// @supportURL      https://greasyfork.org/en/scripts/418547-kinopoisk/feedback
// @match           https://www.kinopoisk.ru/*
// @grant           GM.xmlHttpRequest
// @grant           GM_getValue
// @grant           GM_setValue
// @connect         www.google.com
// @run-at          document-end
// @compatible      chrome     Violentmonkey 2.12.7
// @compatible      firefox    Greasemonkey  4.10.0
// @compatible      firefox    Tampermonkey  4.11.6120
// @noframes
// ==/UserScript==

'use strict';

const css = String.raw`
<style type="text/css">
  .resources { padding: 4px; }
  .resources a { display: inline-block; margin: 2px; }
  .resources a img { width: 16px; height: 16px; }
  .iface__resources { display: none; }
  .iface__resources__active { display: block; }
  .plus__square { 
    background: none; vertical-align: top;
    border: none; color: rgba(31,31,31,.5); padding: 1px; 
  }
  .plus__square:hover { color: #1f1f1f; }
  .plus__square:before { content: "\02795"; }
  .minus__square:before { content: "\2796"; }
  .input__resource { width: 80%; }
  label[for="input__resource"] { 
    color: #393939;
    font-weight: 400; font-size: 12px;
  }
</style>`;

const DEFAULT_RESOURCES = [
  'https://reyohoho.github.io/reyohoho/#%id',
  'https://rutracker.org/forum/tracker.php?nm=%text %year',
  'http://kinozal.tv/browse.php?s=%text&d=%year',
  'http://rutor.info/search/0/0/100/0/%text %year',
  'https://teamhd.org/browse?search=%text&year=%YEAR',
  'https://nnmclub.to/forum/tracker.php?nm=%text %year',
  'https://www.imdb.com/search/title/?title=%engtext&release_date=%year,%endyear',
  'https://www.youtube.com/results?search_query=%text %year'
];

const HINT = (
  'Шаблоны для составления запроса %id, %text %engtext %year %endyear'
);

const LOADING_IMG = '';
const CONTAINER_WAITING_TIME = 1000;
const ONE_PIXEL = 1;

const STORAGE_KEY = '__kp_resources';
const USER_RESOURCES = [];
const QUERY_DATA = {};

let containerResources, controlResources;

const blobToBase64 = (blob, fn) => {
  const reader = new FileReader();
  reader.readAsDataURL(blob);
  reader.onloadend = () => fn(reader.result);
};

const favicon = ({ target }) => {
  if (target.naturalWidth === ONE_PIXEL) {
    target.setAttribute('src', LOADING_IMG);
    GM.xmlHttpRequest({
      url: 'https://www.google.com/s2/favicons?domain=' + target.alt,
      method: 'GET',
      onload: ({ status, response }) => {
        if (status === 200) {
          blobToBase64(response, (base64) => {
            target.setAttribute('src', base64);
          });
        }
      },
      responseType: 'blob'
    });
  }
};

const safeURL = (str) => {
  try {
    return new URL(str);
  } catch {
    return {};
  }
};

const querystring = (str) => (
  str.replace(/(?:%(\w+)?)/g, (str, word) => {
    if (word === undefined) return '';
    word = word.toLowerCase();
    return word in QUERY_DATA
      ? encodeURIComponent(QUERY_DATA[word])
      : str;
  })
);

const extractQueryData = () => {
  try {
    const script = document.querySelector('#__NEXT_DATA__');
    const { props, query } = JSON.parse(script.textContent);
    const { apolloState: { data } } = props;
    const { id } = query;
    const { releaseYears, productionYear, title } = (
      data[`TvSeries:${id}`] ||
      data[`Film:${id}`]
    );
    const [ year ] = Array.isArray(releaseYears)
      ? releaseYears
      : [ productionYear ];
    const { start, end } = typeof year === 'object'
      ? year
      : { start: year, end: year };
    Object.assign(QUERY_DATA, {
      id,
      year: start,
      endyear: end,
      engtext: title.original || title.russian,
      text: title.russian
    });
  } catch {}
};

const addResource = (host, href) => {
  const a = document.createElement('a');
  const img = document.createElement('img');
  const query = querystring(href);
  a.setAttribute('target', '_blank');
  a.setAttribute('rel', 'noopener noreferrer');
  a.setAttribute('title', host);
  a.setAttribute('href', query);
  img.setAttribute('src', 'https://favicon.yandex.net/favicon/' + host);
  img.setAttribute('alt', host);
  img.addEventListener('load', favicon, { once: true });
  img.addEventListener('error', favicon, { once: true });
  a.append(img);
  containerResources.insertAdjacentElement('afterbegin', a);
};

const addResourceClick = ({ target }) => {
  const { previousSibling: input } = target;
  const { host, href } = safeURL(input.value);
  if (host === undefined) return false;
  addResource(host, href);
  USER_RESOURCES.push(href);
  input.value = '';
};

const controlClick = ({ target }) => {
  target.classList.toggle('minus__square');
  controlResources.classList.toggle('iface__resources__active');
};

const initInterface = () => {
  const label = document.createElement('label');
  const input = document.createElement('input');
  const button = document.createElement('button');
  const span = document.createElement('span');
  span.classList.add('error__resource');
  input.classList.add('input__resource');
  label.textContent = HINT;
  button.textContent = '+';
  label.setAttribute('for', 'input__resource');
  input.setAttribute('id', 'input__resource');
  button.addEventListener('click', addResourceClick);
  controlResources.append(label);
  controlResources.append(input);
  controlResources.append(button);
  controlResources.append(span);
  containerResources.insertAdjacentElement('afterend', controlResources);
};

const initControl = () => {
  const button = document.createElement('button');
  const i = document.createElement('i');
  button.className = 'plus__square';
  button.setAttribute('role', 'button');
  button.setAttribute('title', 'Добавить новый ресурс');
  button.addEventListener('click', controlClick);
  button.append(i);
  containerResources.append(button);
};

const initResources = (resources) => {
  for (const resource of resources) {
    const { host, href } = safeURL(resource);
    if (host === undefined) continue;
    addResource(host, href);
  }
};

extractQueryData();
if (Object.keys(QUERY_DATA).length === 0) return;
containerResources = document.createElement('div');
containerResources.classList.add('resources');
initResources(DEFAULT_RESOURCES);
const timer = setInterval(() => {
  const container = document.querySelector('.styles_posterContainer__F02wH');
  if (container) clearInterval(timer);
  document.head.insertAdjacentHTML('beforeend', css);
  container.insertAdjacentElement('beforeend', containerResources);
  if (GM_info.scriptHandler !== 'Greasemonkey') {
    controlResources = document.createElement('div');
    controlResources.classList.add('iface__resources');
    const resources = GM_getValue(STORAGE_KEY, USER_RESOURCES);
    USER_RESOURCES.push(...resources);
    initResources(resources);
    initControl();
    initInterface();
    window.onbeforeunload = (event) => {
      GM_setValue(STORAGE_KEY, USER_RESOURCES);
      delete event.returnValue;
    };
  }
}, CONTAINER_WAITING_TIME);