Kinopoisk+

Adds links to search for popular torrent sites

当前为 2020-12-20 提交的版本,查看 最新版本

  1. /* globals GM, GM.xmlHttpRequest, GM_setValue, GM_getValue, GM_info */
  2. // ==UserScript==
  3. // @name Kinopoisk+
  4. // @name:ru Кинопоиск+
  5. // @description Adds links to search for popular torrent sites
  6. // @description:ru Добавляет ссылки для поиска по популярным торрент-сайтам
  7. // @namespace kp.user.js
  8. // @version 1.1.0
  9. // @author Xant1k@bt (2015-2017), askornot (2020)
  10. // @license MIT
  11. // @icon https://icons.duckduckgo.com/ip9/kinopoisk.ru.ico
  12. // @homepageURL https://greasyfork.org/en/scripts/418547-kinopoisk
  13. // @supportURL https://greasyfork.org/en/scripts/418547-kinopoisk/feedback
  14. // @match https://www.kinopoisk.ru/*
  15. // @grant GM.xmlHttpRequest
  16. // @grant GM_getValue
  17. // @grant GM_setValue
  18. // @connect www.google.com
  19. // @run-at document-end
  20. // @compatible chrome Violentmonkey 2.12.7
  21. // @compatible firefox Greasemonkey 4.10.0
  22. // @compatible firefox Tampermonkey 4.11.6120
  23. // @noframes
  24. // ==/UserScript==
  25.  
  26. 'use strict';
  27.  
  28. const css = String.raw`
  29. <style type="text/css">
  30. .resources { padding: 4px; }
  31. .resources a { display: inline-block; margin: 2px; }
  32. .resources a img { width: 16px; height: 16px; }
  33. .iface__resources { display: none; }
  34. .iface__resources__active { display: block; }
  35. .plus__square {
  36. background: none; vertical-align: top;
  37. border: none; color: rgba(31,31,31,.5); padding: 1px;
  38. }
  39. .plus__square:hover { color: #1f1f1f; }
  40. .plus__square:before { content: "\02795"; }
  41. .minus__square:before { content: "\2796"; }
  42. .input__resource { width: 80%; }
  43. label[for="input__resource"] {
  44. color: #393939;
  45. font-weight: 400; font-size: 12px;
  46. }
  47. </style>`;
  48.  
  49. const DEFAULT_RESOURCES = [
  50. 'https://rutracker.org/forum/tracker.php?nm=%text %year',
  51. 'http://kinozal.tv/browse.php?s=%text&d=%year',
  52. 'http://rutor.info/search/%text %year',
  53. 'https://teamhd.org/browse?search=%text&year=%YEAR',
  54. 'https://nnmclub.to/forum/tracker.php?nm=%text %year',
  55. 'https://www.imdb.com/search/title/?title=%engtext&release_date=%year,%endyear',
  56. 'https://www.youtube.com/results?search_query=%text %year'
  57. ];
  58.  
  59. const HINT = (
  60. 'Шаблоны для составления запроса %text %engtext %year %endyear'
  61. );
  62.  
  63. const LOADING_IMG = '';
  64. const WIDTH = 1;
  65.  
  66. const STORAGE_KEY = '__kp_resources';
  67. const USER_RESOURCES = [];
  68. const QUERY_DATA = {};
  69.  
  70. let panelResources, controlResources;
  71.  
  72. const blobToBase64 = (blob, fn) => {
  73. const reader = new FileReader();
  74. reader.onloadend = ({ target }) => fn(target.result);
  75. reader.readAsDataURL(blob);
  76. };
  77.  
  78. const favicon = ({ target }) => {
  79. if (target.width === WIDTH) {
  80. target.setAttribute('src', LOADING_IMG);
  81. GM.xmlHttpRequest({
  82. url: 'https://www.google.com/s2/favicons?domain=' + target.alt,
  83. method: 'GET',
  84. onload: ({ status, response }) => {
  85. if (status === 200) {
  86. blobToBase64(response, (base64) => {
  87. target.setAttribute('src', base64);
  88. });
  89. }
  90. },
  91. responseType: 'blob'
  92. });
  93. }
  94. };
  95.  
  96. const safeURL = (str) => {
  97. try {
  98. return new URL(str);
  99. } catch (e) {
  100. return {};
  101. }
  102. };
  103.  
  104. const querystring = (str) => (
  105. str.replace(/%(\w+)/g, (str, word) => {
  106. word = word.toLowerCase();
  107. return word in QUERY_DATA
  108. ? encodeURIComponent(QUERY_DATA[word])
  109. : str;
  110. })
  111. );
  112.  
  113. const extractQueryData = () => {
  114. try {
  115. const script = document.querySelector('#__NEXT_DATA__');
  116. const { props, query } = JSON.parse(script.textContent);
  117. const { apolloState: { data } } = props;
  118. const { id } = query;
  119. const { releaseYears, productionYear, title } = (
  120. data[`TvSeries:${id}`] ||
  121. data[`Film:${id}`]
  122. );
  123. const [ year ] = Array.isArray(releaseYears)
  124. ? releaseYears
  125. : [ productionYear ];
  126. const { start, end } = typeof year === 'object'
  127. ? year
  128. : { start: year, end: year };
  129. Object.assign(QUERY_DATA, {
  130. year: start,
  131. endyear: end,
  132. engtext: title.original,
  133. text: title.russian
  134. });
  135. } catch {}
  136. };
  137.  
  138. const addResource = (host, href) => {
  139. const a = document.createElement('a');
  140. const img = document.createElement('img');
  141. const query = querystring(href);
  142. a.setAttribute('target', '_blank');
  143. a.setAttribute('rel', 'noopener noreferrer');
  144. a.setAttribute('title', host);
  145. a.setAttribute('href', query);
  146. img.setAttribute('src', 'https://favicon.yandex.net/favicon/' + host);
  147. img.setAttribute('alt', host);
  148. img.addEventListener('load', favicon, { once: true });
  149. img.addEventListener('error', favicon, { once: true });
  150. a.append(img);
  151. panelResources.insertAdjacentElement('afterbegin', a);
  152. };
  153.  
  154. const addResourceClick = ({ target }) => {
  155. const { previousSibling: input } = target;
  156. const { host, href } = safeURL(input.value);
  157. if (host === undefined) return false;
  158. addResource(host, href);
  159. USER_RESOURCES.push(href);
  160. input.value = '';
  161. };
  162.  
  163. const controlClick = ({ target }) => {
  164. target.classList.toggle('minus__square');
  165. controlResources.classList.toggle('iface__resources__active');
  166. };
  167.  
  168. const initInterface = () => {
  169. const label = document.createElement('label');
  170. const input = document.createElement('input');
  171. const button = document.createElement('button');
  172. const span = document.createElement('span');
  173. span.classList.add('error__resource');
  174. input.classList.add('input__resource');
  175. label.textContent = HINT;
  176. button.textContent = '+';
  177. label.setAttribute('for', 'input__resource');
  178. input.setAttribute('id', 'input__resource');
  179. button.addEventListener('click', addResourceClick);
  180. controlResources.append(label);
  181. controlResources.append(input);
  182. controlResources.append(button);
  183. controlResources.append(span);
  184. panelResources.insertAdjacentElement('afterend', controlResources);
  185. };
  186.  
  187. const initControl = () => {
  188. const button = document.createElement('button');
  189. const i = document.createElement('i');
  190. button.className = 'plus__square';
  191. button.setAttribute('role', 'button');
  192. button.setAttribute('title', 'Добавить новый ресурс');
  193. button.addEventListener('click', controlClick);
  194. button.append(i);
  195. panelResources.append(button);
  196. window.addEventListener('beforeunload', (event) => {
  197. GM_setValue(STORAGE_KEY, USER_RESOURCES);
  198. delete event.returnValue;
  199. });
  200. };
  201.  
  202. const initResources = (resources) => {
  203. for (const resource of resources) {
  204. const { host, href } = safeURL(resource);
  205. if (host === undefined) continue;
  206. addResource(host, href);
  207. }
  208. };
  209.  
  210. panelResources = document.createElement('div');
  211. controlResources = document.createElement('div');
  212. panelResources.classList.add('resources');
  213. controlResources.classList.add('iface__resources');
  214. extractQueryData();
  215. if (Object.keys(QUERY_DATA).length === 0) return;
  216. initResources(DEFAULT_RESOURCES);
  217. setTimeout(() => {
  218. document.head.insertAdjacentHTML('beforeend', css);
  219. document
  220. .querySelector('.styles_posterContainer__1w5Ik')
  221. .insertAdjacentElement('beforeend', panelResources);
  222. if (GM_info.scriptHandler !== 'Greasemonkey') {
  223. const resources = GM_getValue(STORAGE_KEY, USER_RESOURCES);
  224. USER_RESOURCES.push(...resources);
  225. initResources(resources);
  226. initControl();
  227. initInterface();
  228. }
  229. }, 2000);