IMDb: Link 'em all!

Adds all kinds of links to IMDb, customizable!

当前为 2021-12-17 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name IMDb: Link 'em all!
  3. // @description Adds all kinds of links to IMDb, customizable!
  4. // @namespace https://greasyfork.org/en/users/8981-buzz
  5. // @match *://*.imdb.com/title/tt*/*
  6. // @connect *
  7. // @require https://greasemonkey.github.io/gm4-polyfill/gm4-polyfill.js
  8. // @require https://unpkg.com/preact@10.5.7/dist/preact.umd.js
  9. // @require https://unpkg.com/preact@10.5.7/hooks/dist/hooks.umd.js
  10. // @license GPLv2
  11. // @noframes
  12. // @author buzz
  13. // @version 2.0.10
  14. // @grant GM_getValue
  15. // @grant GM_setValue
  16. // @grant GM_xmlhttpRequest
  17. // @grant GM.getValue
  18. // @grant GM.setValue
  19. // @grant GM.xmlHttpRequest
  20. // ==/UserScript==
  21. (function (preact, hooks) {
  22. 'use strict';
  23.  
  24. var version = "2.0.10";
  25. var description = "Adds all kinds of links to IMDb, customizable!";
  26. var homepage = "https://github.com/buzz/imdb-link-em-all#readme";
  27.  
  28. const DESCRIPTION = description;
  29. const HOMEPAGE = homepage;
  30. const NAME_VERSION = `Link 'em all! v${version}`;
  31. const SITES_URL = 'https://raw.githubusercontent.com/buzz/imdb-link-em-all/master/sites.json'; // gets replaced by rollup!
  32.  
  33. const GM_CONFIG_KEY = 'config';
  34. const GREASYFORK_URL = 'https://greasyfork.org/scripts/17154-imdb-link-em-all';
  35. const DEFAULT_CONFIG = {
  36. enabled_sites: [],
  37. fetch_results: true,
  38. first_run: true,
  39. open_blank: true,
  40. show_category_captions: true
  41. };
  42. const CATEGORIES = {
  43. search: 'Search',
  44. movie_site: 'Movie sites',
  45. pub_tracker: 'Public trackers',
  46. priv_tracker: 'Private trackers',
  47. streaming: 'Streaming',
  48. filehoster: 'Filehosters',
  49. subtitles: 'Subtitles',
  50. tv: 'TV'
  51. };
  52. const FETCH_STATE = {
  53. LOADING: 0,
  54. NO_RESULTS: 1,
  55. RESULTS_FOUND: 2,
  56. NO_ACCESS: 3,
  57. TIMEOUT: 4,
  58. ERROR: 5
  59. };
  60.  
  61. var img$8 = "";
  62.  
  63. var img$7 = "";
  64.  
  65. var img$6 = "";
  66.  
  67. var img$5 = "";
  68.  
  69. var img$4 = "";
  70.  
  71. var img$3 = "";
  72.  
  73. var img$2 = "";
  74.  
  75. var img$1 = "";
  76.  
  77. var img = "";
  78.  
  79. const iconSrcs = {
  80. cog: img$8,
  81. error: img$7,
  82. info: img$6,
  83. lock: img$5,
  84. tick: img$4,
  85. timeout: img$3,
  86. world: img$1,
  87. x: img$2,
  88. spinner: img
  89. };
  90.  
  91. const Icon = ({
  92. className,
  93. title,
  94. type
  95. }) => preact.h("img", {
  96. alt: `${type} icon`,
  97. className: className,
  98. src: iconSrcs[type],
  99. title: title
  100. });
  101.  
  102. function styleInject(css, ref) {
  103. if ( ref === void 0 ) ref = {};
  104. var insertAt = ref.insertAt;
  105.  
  106. if (!css || typeof document === 'undefined') { return; }
  107.  
  108. var head = document.head || document.getElementsByTagName('head')[0];
  109. var style = document.createElement('style');
  110. style.type = 'text/css';
  111.  
  112. if (insertAt === 'top') {
  113. if (head.firstChild) {
  114. head.insertBefore(style, head.firstChild);
  115. } else {
  116. head.appendChild(style);
  117. }
  118. } else {
  119. head.appendChild(style);
  120. }
  121.  
  122. if (style.styleSheet) {
  123. style.styleSheet.cssText = css;
  124. } else {
  125. style.appendChild(document.createTextNode(css));
  126. }
  127. }
  128.  
  129. var css_248z$6 = ".Options_options__8dIDU {\n margin-top: 10px;\n}\n\n .Options_options__8dIDU > label > span {\n margin-left: 10px;\n}\n";
  130. var css$6 = {"options":"Options_options__8dIDU"};
  131. styleInject(css_248z$6);
  132.  
  133. const Options = ({
  134. options
  135. }) => {
  136. const optionLabels = options.map(([key, title, val, setter]) => preact.h("label", {
  137. key: key
  138. }, preact.h("input", {
  139. checked: val,
  140. onInput: ev => setter(ev.target.checked),
  141. type: "checkbox"
  142. }), preact.h("span", null, title), preact.h("br", null)));
  143. return preact.h("div", {
  144. className: css$6.options
  145. }, optionLabels);
  146. };
  147.  
  148. const SiteIcon = ({
  149. className,
  150. site,
  151. title
  152. }) => site.icon ? preact.h("img", {
  153. alt: site.title,
  154. className: className,
  155. src: site.icon,
  156. title: title
  157. }) : null;
  158.  
  159. var css_248z$5 = ".Sites_searchBar__1cpJl {\n display: flex;\n flex-direction: row;\n margin-bottom: 1em;\n}\n\n .Sites_searchBar__1cpJl .Sites_searchInput__1iJDL {\n background-color: rgba(255, 255, 255, 0.9);\n border-radius: 3px;\n border-top-color: #949494;\n border: 1px solid #a6a6a6;\n box-shadow: 0 1px 0 rgba(0, 0, 0, .07) inset;\n display: flex;\n flex-direction: row;\n height: 24px;\n line-height: normal;\n outline: 0;\n padding: 3px 7px;\n transition: all 100ms linear;\n width: 100%;\n}\n\n .Sites_searchBar__1cpJl .Sites_searchInput__1iJDL:focus-within {\n background-color: #fff;\n border-color: #e77600;\n box-shadow: 0 0 2px 2px rgba(228, 121, 17, 0.25);\n}\n\n .Sites_searchBar__1cpJl .Sites_searchInput__1iJDL > * {\n background-color: transparent;\n border: none;\n height: 16px;\n}\n\n .Sites_searchBar__1cpJl .Sites_searchInput__1iJDL > button {\n margin: 0 0 0 0.7em;\n padding: 0;\n}\n\n .Sites_searchBar__1cpJl .Sites_searchInput__1iJDL > input {\n flex-grow: 1;\n outline: none;\n padding: 0 0 0 0.5em;\n}\n\n .Sites_searchBar__1cpJl .Sites_resultCount__2p4vG {\n font-weight: bold;\n margin-left: 2em;\n min-width: 140px;\n text-align: right;\n}\n\n .Sites_searchBar__1cpJl .Sites_resultCount__2p4vG > span {\n color: black;\n}\n\n.Sites_siteList__1Y3wR .Sites_catList__6txMX {\n display: flex;\n flex-wrap: wrap;\n}\n\n.Sites_siteList__1Y3wR .Sites_catList__6txMX h4 {\n width: 100%;\n}\n\n.Sites_siteList__1Y3wR .Sites_catList__6txMX label {\n align-items: center;\n color: #444;\n display: flex;\n flex-flow: row;\n padding: 0 6px;\n transition: color 100ms;\n width: 25%;\n}\n\n.Sites_siteList__1Y3wR .Sites_catList__6txMX label:hover {\n color: #222;\n}\n\n.Sites_siteList__1Y3wR .Sites_catList__6txMX label.Sites_checked__3D9QY span {\n color: black;\n}\n\n.Sites_siteList__1Y3wR .Sites_catList__6txMX label .Sites_title__1Gu_F {\n flex-grow: 1;\n overflow: hidden;\n text-overflow: ellipsis;\n}\n\n.Sites_siteList__1Y3wR .Sites_catList__6txMX label input {\n margin-right: 4px;\n}\n\n.Sites_siteList__1Y3wR .Sites_catList__6txMX label .Sites_extraIcon__jwLPa {\n height: 12px;\n margin-left: 4px;\n width: 12px;\n}\n\n.Sites_siteList__1Y3wR .Sites_catList__6txMX label .Sites_siteIcon__3uzGl {\n flex-shrink: 0;\n margin-right: 6px;\n}\n";
  160. var css$5 = {"searchBar":"Sites_searchBar__1cpJl","searchInput":"Sites_searchInput__1iJDL","resultCount":"Sites_resultCount__2p4vG","siteList":"Sites_siteList__1Y3wR","catList":"Sites_catList__6txMX","checked":"Sites_checked__3D9QY","title":"Sites_title__1Gu_F","extraIcon":"Sites_extraIcon__jwLPa","siteIcon":"Sites_siteIcon__3uzGl"};
  161. styleInject(css_248z$5);
  162.  
  163. const SearchInput = ({
  164. q,
  165. setQ
  166. }) => preact.h("div", {
  167. className: css$5.searchInput
  168. }, preact.h("span", null, "\uD83D\uDD0D"), preact.h("input", {
  169. onInput: e => {
  170. setQ(e.target.value.toLowerCase().trim());
  171. },
  172. placeholder: "Search",
  173. value: q
  174. }), preact.h("button", {
  175. style: {
  176. display: q.length ? 'unset' : 'none'
  177. },
  178. title: "Clear",
  179. type: "button",
  180. onClick: () => setQ('')
  181. }, preact.h(Icon, {
  182. type: "x"
  183. })));
  184.  
  185. const DummyIcon = ({
  186. size
  187. }) => {
  188. const sizePx = `${size}px`;
  189. const style = {
  190. display: 'inline-block',
  191. height: sizePx,
  192. width: sizePx
  193. };
  194. return preact.h("div", {
  195. className: css$5.siteIcon,
  196. style: style
  197. });
  198. };
  199.  
  200. const SiteLabel = ({
  201. checked,
  202. setEnabled,
  203. site
  204. }) => {
  205. const input = preact.h("input", {
  206. checked: checked,
  207. onInput: e => setEnabled(prev => e.target.checked ? [...prev, site.id] : prev.filter(id => id !== site.id)),
  208. type: "checkbox"
  209. });
  210. const icon = site.icon ? preact.h(SiteIcon, {
  211. className: css$5.siteIcon,
  212. site: site,
  213. title: site.title
  214. }) : preact.h(DummyIcon, {
  215. size: 16
  216. });
  217. const title = preact.h("span", {
  218. className: css$5.title,
  219. title: site.title
  220. }, site.title);
  221. const extraIcons = [site.noAccessMatcher ? preact.h(Icon, {
  222. className: css$5.extraIcon,
  223. title: "Access restricted",
  224. type: "lock"
  225. }) : null, site.noResultsMatcher ? preact.h(Icon, {
  226. className: css$5.extraIcon,
  227. title: "Site supports fetching of results",
  228. type: "tick"
  229. }) : null];
  230. return preact.h("label", {
  231. className: checked ? css$5.checked : null
  232. }, input, icon, " ", title, " ", extraIcons);
  233. };
  234.  
  235. const CategoryList = ({
  236. enabled,
  237. name,
  238. setEnabled,
  239. sites
  240. }) => {
  241. const siteLabels = sites.map(site => preact.h(SiteLabel, {
  242. checked: enabled.includes(site.id),
  243. setEnabled: setEnabled,
  244. site: site
  245. }));
  246. return preact.h("div", {
  247. className: css$5.catList
  248. }, preact.h("h4", null, name, " ", preact.h("span", null, "(", sites.length, ")")), siteLabels);
  249. };
  250.  
  251. const Sites = ({
  252. enabledSites,
  253. setEnabledSites,
  254. sites
  255. }) => {
  256. const [q, setQ] = hooks.useState('');
  257. const catSites = Object.keys(CATEGORIES).map(cat => {
  258. const s = sites.filter(site => site.category === cat);
  259.  
  260. if (q.length) {
  261. return s.filter(site => site.title.toLowerCase().includes(q));
  262. }
  263.  
  264. return s;
  265. });
  266. const cats = Object.entries(CATEGORIES).map(([cat, catName], i) => catSites[i].length ? preact.h(CategoryList, {
  267. enabled: enabledSites,
  268. key: cat,
  269. name: catName,
  270. setEnabled: setEnabledSites,
  271. sites: catSites[i]
  272. }) : null);
  273. const total = catSites.reduce((acc, s) => acc + s.length, 0);
  274. return preact.h(preact.Fragment, null, preact.h("div", {
  275. className: css$5.searchBar
  276. }, preact.h(SearchInput, {
  277. q: q,
  278. setQ: setQ
  279. }), preact.h("div", {
  280. className: css$5.resultCount
  281. }, "Showing ", preact.h("span", null, total), " sites.")), preact.h("div", {
  282. className: css$5.siteList
  283. }, cats));
  284. };
  285.  
  286. var css_248z$4 = ".About_about__3lHx7 {\n padding: 1em 0;\n position: relative;\n}\n\n .About_about__3lHx7 ul > li {\n margin-bottom: 0;\n}\n\n .About_about__3lHx7 h2 {\n font-size: 20px;\n margin: 0.5em 0;\n}\n\n .About_about__3lHx7 > *:last-child {\n margin-bottom: 0;\n}\n\n .About_about__3lHx7 .About_top__3XyCB {\n text-align: center;\n}\n\n .About_about__3lHx7 .About_content__1xMTu {\n width: 61.8%;\n margin: 0 auto;\n}\n";
  287. var css$4 = {"about":"About_about__3lHx7","top":"About_top__3XyCB","content":"About_content__1xMTu"};
  288. styleInject(css_248z$4);
  289.  
  290. const About = () => preact.h("div", {
  291. className: css$4.about
  292. }, preact.h("div", {
  293. className: css$4.top
  294. }, preact.h("h3", null, "\uD83C\uDFA5 ", NAME_VERSION), preact.h("p", null, DESCRIPTION)), preact.h("div", {
  295. className: css$4.content
  296. }, preact.h("h2", null, "\uD83D\uDD17 Links"), preact.h("ul", null, preact.h("li", null, preact.h("a", {
  297. target: "_blank",
  298. rel: "noreferrer",
  299. href: HOMEPAGE
  300. }, "GitHub")), preact.h("li", null, preact.h("a", {
  301. target: "_blank",
  302. rel: "noreferrer",
  303. href: GREASYFORK_URL
  304. }, "Greasy Fork"))), preact.h("h2", null, "\u2728 Contributions"), preact.h("p", null, "Add new sites or update existing entries."), preact.h("ul", null, preact.h("li", null, preact.h("a", {
  305. target: "_blank",
  306. rel: "noreferrer",
  307. href: "https://github.com/buzz/imdb-link-em-all/issues/new"
  308. }, "Open a GitHub issue"), ' ', "or"), preact.h("li", null, preact.h("a", {
  309. target: "_blank",
  310. rel: "noreferrer",
  311. href: "https://greasyfork.org/en/scripts/17154-imdb-link-em-all/feedback"
  312. }, "Give feedback"), ' ', "on Greasy Fork.")), preact.h("p", null, preact.h("em", null, "Thanks to all the contributors!"), " \uD83D\uDC4D"), preact.h("h2", null, "\u2696 License"), preact.h("p", null, "This script is licensed under the terms of the", ' ', preact.h("a", {
  313. target: "_blank",
  314. rel: "noreferrer",
  315. href: "https://github.com/buzz/imdb-link-em-all/blob/master/LICENSE"
  316. }, "GPL-2.0 License"), ".")));
  317.  
  318. var css_248z$3 = ".Config_popover__3RK3L {\n background-color: #a5a5a5;\n border-radius: 4px;\n box-shadow: 0 0 2em rgba(0, 0, 0, 0.1);\n color: #333;\n display: block;\n font-family: Verdana, Arial, sans-serif;\n font-size: 11px;\n left: calc(-800px + 35px);\n line-height: 1.5rem;\n padding: 10px;\n position: absolute;\n top: calc(20px + 8px);\n white-space: nowrap;\n width: 800px;\n z-index: 100;\n}\n.Config_popover__3RK3L.Config_layout-legacy__6Cdsp {\n left: calc(-800px + 235px);\n}\n.Config_popover__3RK3L.Config_layout-legacy__6Cdsp:before {\n right: calc(235px - 2 * 8px);\n}\n.Config_popover__3RK3L:before {\n border-bottom: 8px solid #a5a5a5;\n border-left: 8px solid transparent;\n border-right: 8px solid transparent;\n border-top: 8px solid transparent;\n content: \"\";\n display: block;\n height: 8px;\n right: calc(35px - 2 * 8px);\n position: absolute;\n top: calc(-2 * 8px);\n width: 0;\n}\n.Config_popover__3RK3L .Config_inner__2Sbjz {\n display: flex;\n flex-direction: column;\n text-shadow: 0 1px 0 rgba(255, 255, 255, 0.2);\n}\n.Config_popover__3RK3L .Config_inner__2Sbjz .Config_top__2kgQ3 {\n display: flex;\n flex-direction: row;\n}\n.Config_popover__3RK3L .Config_inner__2Sbjz .Config_top__2kgQ3 .Config_link__3aqRB {\n flex-grow: 1;\n text-align: right;\n}\n.Config_popover__3RK3L .Config_inner__2Sbjz .Config_top__2kgQ3 .Config_link__3aqRB > a {\n color: #333;\n margin-left: 12px;\n margin-right: 4px;\n}\n.Config_popover__3RK3L .Config_inner__2Sbjz .Config_top__2kgQ3 .Config_link__3aqRB > a:visited {\n color: #333;\n}\n.Config_popover__3RK3L .Config_inner__2Sbjz .Config_top__2kgQ3 > button {\n background-color: rgba(0, 0, 0, 0.05);\n border-bottom-left-radius: 0;\n border-bottom-right-radius: 0;\n border-bottom: transparent;\n border-left: 1px solid rgba(0, 0, 0, 0.25);\n border-right: 1px solid rgba(0, 0, 0, 0.25);\n border-top-left-radius: 2px;\n border-top-right-radius: 2px;\n border-top: 1px solid rgba(0, 0, 0, 0.25);\n color: #424242;\n font-size: 12px;\n margin: 0 6px 0 0;\n outline: none;\n padding: 0 6px;\n transform: translateY(1px);\n text-shadow: 0 1px 0 rgba(255, 255, 255, 0.2);\n}\n.Config_popover__3RK3L .Config_inner__2Sbjz .Config_top__2kgQ3 > button:hover {\n background-color: rgba(0, 0, 0, 0.1);\n color: #222;\n}\n.Config_popover__3RK3L .Config_inner__2Sbjz .Config_top__2kgQ3 > button.Config_active__iBK3y {\n background-color: #c2c2c2;\n color: #222;\n}\n.Config_popover__3RK3L .Config_inner__2Sbjz .Config_top__2kgQ3 > button:last-child {\n margin-right: 0;\n}\n.Config_popover__3RK3L .Config_inner__2Sbjz .Config_top__2kgQ3 > button > img {\n vertical-align: text-bottom;\n}\n.Config_popover__3RK3L .Config_inner__2Sbjz .Config_body__2JuhF {\n background-color: #c2c2c2;\n border-bottom-left-radius: 2px;\n border-bottom-right-radius: 2px;\n border-top-right-radius: 2px;\n border: 1px solid rgba(0, 0, 0, 0.25);\n padding: 12px 10px 12px;\n}\n.Config_popover__3RK3L .Config_inner__2Sbjz .Config_body__2JuhF > div {\n overflow: hidden;\n}\n.Config_popover__3RK3L .Config_inner__2Sbjz .Config_body__2JuhF > div > *:first-child {\n margin-top: 0;\n}\n.Config_popover__3RK3L .Config_inner__2Sbjz .Config_body__2JuhF > div > *:last-child {\n margin-bottom: 0;\n}\n.Config_popover__3RK3L .Config_inner__2Sbjz .Config_controls__3hBBQ {\n display: flex;\n flex-direction: row;\n margin-top: 10px;\n}\n.Config_popover__3RK3L .Config_inner__2Sbjz .Config_controls__3hBBQ > div:first-child {\n flex-grow: 1;\n}\n.Config_popover__3RK3L .Config_inner__2Sbjz .Config_controls__3hBBQ button {\n padding-bottom: 0;\n padding-top: 0;\n margin-right: 12px;\n}\n";
  319. var css$3 = {"popover":"Config_popover__3RK3L","layout-legacy":"Config_layout-legacy__6Cdsp","inner":"Config_inner__2Sbjz","top":"Config_top__2kgQ3","link":"Config_link__3aqRB","active":"Config_active__iBK3y","body":"Config_body__2JuhF","controls":"Config_controls__3hBBQ"};
  320. styleInject(css_248z$3);
  321.  
  322. const OPTIONS = [['show_category_captions', 'Show category captions'], ['open_blank', 'Open links in new tab'], ['fetch_results', 'Automatically fetch results']];
  323.  
  324. const Config = ({
  325. config,
  326. layout,
  327. setConfig,
  328. setShow,
  329. show,
  330. sites
  331. }) => {
  332. const [enabledSites, setEnabledSites] = hooks.useState(config.enabled_sites);
  333. const showCategoryCaptionsArr = hooks.useState(config.show_category_captions);
  334. const openBlankArr = hooks.useState(config.open_blank);
  335. const fetchResultsArr = hooks.useState(config.fetch_results);
  336. const [showCategoryCaptions, setShowCategoryCaptions] = showCategoryCaptionsArr;
  337. const [openBlank, setOpenBlank] = openBlankArr;
  338. const [fetchResults, setFetchResults] = fetchResultsArr;
  339. const optStates = [showCategoryCaptionsArr, openBlankArr, fetchResultsArr];
  340. const options = OPTIONS.map((opt, i) => [...opt, ...optStates[i]]);
  341. const [tab, setTab] = hooks.useState(0);
  342. const tabs = [{
  343. title: 'Sites',
  344. icon: 'world',
  345. comp: preact.h(Sites, {
  346. enabledSites: enabledSites,
  347. setEnabledSites: setEnabledSites,
  348. sites: sites
  349. })
  350. }, {
  351. title: 'Options',
  352. icon: 'cog',
  353. comp: preact.h(Options, {
  354. options: options
  355. })
  356. }, {
  357. title: 'About',
  358. icon: 'info',
  359. comp: preact.h(About, null)
  360. }];
  361.  
  362. const onClickCancel = () => {
  363. setShow(false); // Restore state
  364.  
  365. setEnabledSites(config.enabled_sites);
  366. setFetchResults(config.fetch_results);
  367. setOpenBlank(config.open_blank);
  368. setShowCategoryCaptions(config.show_category_captions);
  369. };
  370.  
  371. const onClickSave = () => {
  372. setConfig({
  373. enabled_sites: enabledSites,
  374. fetch_results: fetchResults,
  375. open_blank: openBlank,
  376. show_category_captions: showCategoryCaptions
  377. });
  378. setShow(false);
  379. };
  380.  
  381. return preact.h("div", {
  382. className: `${css$3.popover} ${css$3['layout-' + layout]}`,
  383. style: {
  384. display: show ? 'block' : 'none'
  385. }
  386. }, preact.h("div", {
  387. className: css$3.inner
  388. }, preact.h("div", {
  389. className: css$3.top
  390. }, tabs.map(({
  391. title,
  392. icon
  393. }, i) => preact.h("button", {
  394. className: tab === i ? css$3.active : null,
  395. type: "button",
  396. onClick: () => setTab(i)
  397. }, preact.h(Icon, {
  398. title: title,
  399. type: icon
  400. }), " ", title)), preact.h("div", {
  401. className: css$3.link
  402. }, preact.h("a", {
  403. target: "_blank",
  404. rel: "noreferrer",
  405. href: HOMEPAGE
  406. }, "\uD83C\uDFA5 ", NAME_VERSION))), preact.h("div", {
  407. className: css$3.body
  408. }, tabs.map(({
  409. comp
  410. }, i) => preact.h("div", {
  411. style: {
  412. display: tab === i ? 'block' : 'none'
  413. }
  414. }, comp))), preact.h("div", {
  415. className: css$3.controls
  416. }, preact.h("div", null, preact.h("button", {
  417. className: "btn primary small",
  418. onClick: onClickSave,
  419. type: "button"
  420. }, "OK"), preact.h("button", {
  421. className: "btn small",
  422. onClick: onClickCancel,
  423. type: "button"
  424. }, "Cancel")))));
  425. };
  426.  
  427. function _extends() {
  428. _extends = Object.assign || function (target) {
  429. for (var i = 1; i < arguments.length; i++) {
  430. var source = arguments[i];
  431.  
  432. for (var key in source) {
  433. if (Object.prototype.hasOwnProperty.call(source, key)) {
  434. target[key] = source[key];
  435. }
  436. }
  437. }
  438.  
  439. return target;
  440. };
  441.  
  442. return _extends.apply(this, arguments);
  443. }
  444.  
  445. const replaceFields = (str, {
  446. id,
  447. title,
  448. year
  449. }, encode = true) => str.replace(new RegExp('{{IMDB_TITLE}}', 'g'), encode ? encodeURIComponent(title) : title).replace(new RegExp('{{IMDB_ID}}', 'g'), id).replace(new RegExp('{{IMDB_YEAR}}', 'g'), year);
  450.  
  451. const checkResponse = (resp, site) => {
  452. // Likely a redirect to login page
  453. if (resp.responseHeaders && resp.responseHeaders.includes('Refresh: 0; url=')) {
  454. return FETCH_STATE.NO_ACCESS;
  455. } // There should be a responseText
  456.  
  457.  
  458. if (!resp.responseText) {
  459. return FETCH_STATE.ERROR;
  460. } // Detect Blogger content warning
  461.  
  462.  
  463. if (resp.responseText.includes('The blog that you are about to view may contain content only suitable for adults.')) {
  464. return FETCH_STATE.NO_ACCESS;
  465. } // Detect CloudFlare anti DDOS page
  466.  
  467.  
  468. if (resp.responseText.includes('Checking your browser before accessing')) {
  469. return FETCH_STATE.NO_ACCESS;
  470. } // Check site access
  471.  
  472.  
  473. if (site.noAccessMatcher) {
  474. const matchStrings = Array.isArray(site.noAccessMatcher) ? site.noAccessMatcher : [site.noAccessMatcher];
  475.  
  476. if (matchStrings.some(matchString => resp.responseText.includes(matchString))) {
  477. return FETCH_STATE.NO_ACCESS;
  478. }
  479. } // Check results
  480.  
  481.  
  482. if (Array.isArray(site.noResultsMatcher)) {
  483. // Advanced ways of checking, currently only EL_COUNT is supported
  484. const [checkType, selector, compType, number] = site.noResultsMatcher;
  485. const m = resp.responseHeaders.match(/content-type:\s([^\s;]+)/);
  486. const contentType = m ? m[1] : 'text/html';
  487. let doc;
  488.  
  489. try {
  490. const parser = new DOMParser();
  491. doc = parser.parseFromString(resp.responseText, contentType);
  492. } catch (e) {
  493. console.error('Could not parse document!');
  494. return FETCH_STATE.ERROR;
  495. }
  496.  
  497. switch (checkType) {
  498. case 'EL_COUNT':
  499. {
  500. let result;
  501.  
  502. try {
  503. result = doc.querySelectorAll(selector);
  504. } catch (err) {
  505. console.error(err);
  506. return FETCH_STATE.ERROR;
  507. }
  508.  
  509. if (compType === 'GT') {
  510. if (result.length > number) {
  511. return FETCH_STATE.RESULTS_FOUND;
  512. }
  513. }
  514.  
  515. if (compType === 'LT') {
  516. if (result.length < number) {
  517. return FETCH_STATE.RESULTS_FOUND;
  518. }
  519. }
  520.  
  521. break;
  522. }
  523. }
  524.  
  525. return FETCH_STATE.NO_RESULTS;
  526. }
  527.  
  528. const matchStrings = Array.isArray(site.noResultsMatcher) ? site.noResultsMatcher : [site.noResultsMatcher];
  529.  
  530. if (matchStrings.some(matchString => resp.responseText.includes(matchString))) {
  531. return FETCH_STATE.NO_RESULTS;
  532. }
  533.  
  534. return FETCH_STATE.RESULTS_FOUND;
  535. };
  536.  
  537. const useResultFetcher = (imdbInfo, site) => {
  538. const [fetchState, setFetchState] = hooks.useState(null);
  539. hooks.useEffect(() => {
  540. let xhr;
  541.  
  542. if (site.noResultsMatcher) {
  543. // Site supports result fetching
  544. const {
  545. url
  546. } = site;
  547. const isPost = Array.isArray(url);
  548. const opts = {
  549. timeout: 20000,
  550. onload: resp => setFetchState(checkResponse(resp, site)),
  551. onerror: resp => {
  552. console.error(`Failed to fetch results from URL '${url}': ${resp.statusText}`);
  553. setFetchState(FETCH_STATE.ERROR);
  554. },
  555. ontimeout: () => setFetchState(FETCH_STATE.TIMEOUT)
  556. };
  557.  
  558. if (isPost) {
  559. const [postUrl, fields] = url;
  560. opts.method = 'POST';
  561. opts.url = postUrl;
  562. opts.headers = {
  563. 'Content-Type': 'application/x-www-form-urlencoded'
  564. };
  565. opts.data = Object.keys(fields).map(key => {
  566. const val = replaceFields(fields[key], imdbInfo, false);
  567. return `${key}=${val}`;
  568. }).join('&');
  569. } else {
  570. opts.method = 'GET';
  571. opts.url = replaceFields(url, imdbInfo);
  572. }
  573.  
  574. xhr = GM.xmlHttpRequest(opts);
  575. setFetchState(FETCH_STATE.LOADING);
  576. }
  577.  
  578. return () => {
  579. if (xhr && xhr.abort) {
  580. xhr.abort();
  581. }
  582. };
  583. }, [imdbInfo, site]);
  584. return fetchState;
  585. };
  586.  
  587. var css_248z$2 = ".SiteLink_linkWrapper__2uDyT {\n display: inline-block;\n margin-right: 4px;\n}\n\n .SiteLink_linkWrapper__2uDyT img {\n vertical-align: baseline;\n}\n\n .SiteLink_linkWrapper__2uDyT a {\n white-space: pre-line;\n}\n\n .SiteLink_linkWrapper__2uDyT a > img {\n height: 16px;\n width: 16px;\n margin-right: 4px;\n}\n\n .SiteLink_linkWrapper__2uDyT .SiteLink_resultsIcon__3_V-k {\n margin-left: 4px;\n}\n";
  588. var css$2 = {"linkWrapper":"SiteLink_linkWrapper__2uDyT","resultsIcon":"SiteLink_resultsIcon__3_V-k"};
  589. styleInject(css_248z$2);
  590.  
  591. const ResultsIndicator = ({
  592. imdbInfo,
  593. site
  594. }) => {
  595. const fetchState = useResultFetcher(imdbInfo, site);
  596. let iconType;
  597. let title;
  598.  
  599. switch (fetchState) {
  600. case FETCH_STATE.LOADING:
  601. iconType = 'spinner';
  602. title = 'Loading…';
  603. break;
  604.  
  605. case FETCH_STATE.NO_RESULTS:
  606. iconType = 'x';
  607. title = 'No Results found!';
  608. break;
  609.  
  610. case FETCH_STATE.RESULTS_FOUND:
  611. iconType = 'tick';
  612. title = 'Results found!';
  613. break;
  614.  
  615. case FETCH_STATE.NO_ACCESS:
  616. iconType = 'lock';
  617. title = 'You have to login to this site!';
  618. break;
  619.  
  620. case FETCH_STATE.TIMEOUT:
  621. iconType = 'timeout';
  622. title = 'You have to login to this site!';
  623. break;
  624.  
  625. case FETCH_STATE.ERROR:
  626. iconType = 'error';
  627. title = 'Error fetching results! (See dev console for details)';
  628. break;
  629.  
  630. default:
  631. return null;
  632. }
  633.  
  634. return preact.h(Icon, {
  635. className: css$2.resultsIcon,
  636. title: title,
  637. type: iconType
  638. });
  639. };
  640.  
  641. const usePostLink = (url, openBlank, imdbInfo) => {
  642. const formEl = hooks.useRef();
  643. const isPost = Array.isArray(url);
  644. const href = isPost ? url[0] : replaceFields(url, imdbInfo, false);
  645.  
  646. const onClick = event => {
  647. if (isPost && formEl.current) {
  648. event.preventDefault();
  649. formEl.current.submit();
  650. }
  651. };
  652.  
  653. hooks.useEffect(() => {
  654. if (isPost) {
  655. const [postUrl, fields] = url;
  656. const form = document.createElement('form');
  657. form.action = postUrl;
  658. form.method = 'POST';
  659. form.style.display = 'none';
  660. form.target = openBlank ? '_blank' : '_self';
  661. Object.keys(fields).forEach(key => {
  662. const input = document.createElement('input');
  663. input.type = 'text';
  664. input.name = key;
  665. input.value = replaceFields(fields[key], imdbInfo, false);
  666. form.appendChild(input);
  667. });
  668. document.body.appendChild(form);
  669. formEl.current = form;
  670. }
  671.  
  672. return () => {
  673. if (formEl.current) {
  674. formEl.current.remove();
  675. }
  676. };
  677. });
  678. return [href, onClick];
  679. };
  680.  
  681. const Sep = () => preact.h(preact.Fragment, null, "\xA0", preact.h("span", {
  682. className: "ghost"
  683. }, "|"));
  684.  
  685. const SiteLink = ({
  686. config,
  687. imdbInfo,
  688. last,
  689. site
  690. }) => {
  691. const extraAttrs = config.open_blank ? {
  692. target: '_blank',
  693. rel: 'noreferrer'
  694. } : {};
  695. const [href, onClick] = usePostLink(site.url, config.open_blank, imdbInfo);
  696. return preact.h("span", {
  697. className: css$2.linkWrapper
  698. }, preact.h("a", _extends({
  699. className: "ipc-link ipc-link--base",
  700. href: href,
  701. onClick: onClick
  702. }, extraAttrs), preact.h(SiteIcon, {
  703. site: site
  704. }), preact.h("span", null, site.title)), preact.h(ResultsIndicator, {
  705. imdbInfo: imdbInfo,
  706. site: site
  707. }), last ? null : preact.h(Sep, null));
  708. };
  709.  
  710. var css_248z$1 = ".LinkList_linkList__rlGOn {\n line-height: 1.6rem\n}\n\n.LinkList_h4__2axTi {\n margin-top: 0.5rem\n}\n";
  711. var css$1 = {"linkList":"LinkList_linkList__rlGOn","h4":"LinkList_h4__2axTi"};
  712. styleInject(css_248z$1);
  713.  
  714. const LinkList = ({
  715. config,
  716. imdbInfo,
  717. sites
  718. }) => Object.entries(CATEGORIES).map(([category, categoryName]) => {
  719. const catSites = sites.filter(site => site.category === category && config.enabled_sites.includes(site.id));
  720.  
  721. if (!catSites.length) {
  722. return null;
  723. }
  724.  
  725. const caption = config.show_category_captions ? preact.h("h4", {
  726. className: css$1.h4
  727. }, categoryName) : null;
  728. return preact.h(preact.Fragment, null, caption, preact.h("div", {
  729. className: css$1.linkList
  730. }, catSites.map((site, i) => preact.h(SiteLink, {
  731. config: config,
  732. imdbInfo: imdbInfo,
  733. last: i === catSites.length - 1,
  734. site: site
  735. }))));
  736. });
  737.  
  738. var css_248z = ".App_configWrapper__2KuAE {\n position: absolute;\n right: 20px;\n top: 20px;\n}\n\n .App_configWrapper__2KuAE > button {\n background: transparent;\n border: none;\n cursor: pointer;\n outline: none;\n padding: 0;\n}\n\n .App_configWrapper__2KuAE > button > img {\n vertical-align: baseline;\n}\n";
  739. var css = {"configWrapper":"App_configWrapper__2KuAE"};
  740. styleInject(css_248z);
  741.  
  742. const restoreConfig = async () => JSON.parse(await GM.getValue(GM_CONFIG_KEY));
  743.  
  744. const saveConfig = async config => GM.setValue(GM_CONFIG_KEY, JSON.stringify(config));
  745.  
  746. const useConfig = () => {
  747. const [config, setConfig] = hooks.useState();
  748. hooks.useEffect(() => {
  749. restoreConfig().then(c => setConfig(c)).catch(() => setConfig(DEFAULT_CONFIG));
  750. }, []);
  751. hooks.useEffect(() => {
  752. if (config) {
  753. saveConfig(config);
  754. }
  755. }, [config]);
  756. return {
  757. config,
  758. setConfig
  759. };
  760. };
  761.  
  762. const loadSites = () => new Promise((resolve, reject) => GM.xmlHttpRequest({
  763. method: 'GET',
  764. url: SITES_URL,
  765. nocache: true,
  766.  
  767. onload({
  768. response,
  769. status,
  770. statusText
  771. }) {
  772. if (status === 200) {
  773. try {
  774. resolve(JSON.parse(response).sort((a, b) => a.title.localeCompare(b.title)));
  775. } catch (e) {
  776. reject(e);
  777. }
  778. } else {
  779. reject(new Error(`LTA: Could not load sites (URL=${SITES_URL}): ${status} ${statusText}`));
  780. }
  781. },
  782.  
  783. onerror({
  784. status,
  785. statusText
  786. }) {
  787. reject(new Error(`LTA: Could not load sites (URL=${SITES_URL}): ${status} ${statusText}`));
  788. }
  789.  
  790. }));
  791.  
  792. const useSites = () => {
  793. const [sites, setSites] = hooks.useState([]);
  794. hooks.useEffect(() => {
  795. loadSites().then(s => setSites(s));
  796. }, []);
  797. return sites;
  798. };
  799.  
  800. const App = ({
  801. imdbInfo
  802. }) => {
  803. const {
  804. config,
  805. setConfig
  806. } = useConfig();
  807. const sites = useSites();
  808. const [showConfig, setShowConfig] = hooks.useState(false);
  809. hooks.useEffect(() => {
  810. if (config && config.first_run) {
  811. setShowConfig(true);
  812. setConfig(prev => ({ ...prev,
  813. first_run: false
  814. }));
  815. }
  816. }, [config]);
  817.  
  818. if (!config || !sites.length) {
  819. return null;
  820. }
  821.  
  822. return preact.h(preact.Fragment, null, imdbInfo.layout === 'legacy' ? preact.h("hr", null) : null, preact.h("div", {
  823. className: css.configWrapper
  824. }, preact.h("button", {
  825. onClick: () => setShowConfig(cur => !cur),
  826. title: "Configure",
  827. type: "button"
  828. }, preact.h(Icon, {
  829. type: "cog"
  830. })), preact.h(Config, {
  831. config: config,
  832. layout: imdbInfo.layout,
  833. setConfig: setConfig,
  834. setShow: setShowConfig,
  835. sites: sites,
  836. show: showConfig
  837. })), preact.h(LinkList, {
  838. config: config,
  839. imdbInfo: imdbInfo,
  840. sites: sites
  841. }));
  842. };
  843.  
  844. const divId = '__LTA__';
  845.  
  846. const detectLayout = mUrl => {
  847. // Currently there seem to be 3 different IMDb layouts:
  848. // 1) "legacy": URL ends with '/reference'
  849. if (['reference', 'combined'].includes(mUrl[2])) {
  850. return ['legacy', 'h3[itemprop=name]', '.titlereference-section-overview > *:last-child'];
  851. } // 2) "redesign2020": Redesign 2020
  852. // https://www.imdb.com/preferences/beta-control?e=tmd&t=in&u=/title/tt0163978/
  853.  
  854.  
  855. if (document.querySelector('[data-testid="hero-title-block__title"]')) {
  856. return ['redesign2020', 'title', '[class*=TitleMainHeroGroup]'];
  857. } // 3) "new": The old default (has been around for many years)
  858.  
  859.  
  860. return ['new', 'h1', '.title-overview'];
  861. };
  862.  
  863. const parseImdbInfo = () => {
  864. // TODO: extract type (TV show, movie, ...)
  865. // Parse IMDb number and layout
  866. const mUrl = /^\/title\/tt([0-9]{7,8})\/([a-z]*)/.exec(window.location.pathname);
  867.  
  868. if (!mUrl) {
  869. throw new Error('LTA: Could not parse IMDb URL!');
  870. }
  871.  
  872. const [layout, titleSelector, containerSelector] = detectLayout(mUrl);
  873. const info = {
  874. id: mUrl[1],
  875. layout
  876. };
  877. info.title = document.querySelector(titleSelector).innerText.trim();
  878. const mTitle = /^(.+)\s+\((\d+)\)/.exec(info.title);
  879.  
  880. if (mTitle) {
  881. info.title = mTitle[1].trim();
  882. info.year = parseInt(mTitle[2].trim(), 10);
  883. }
  884.  
  885. return [info, containerSelector];
  886. };
  887.  
  888. const [imdbInfo, containerSelector] = parseImdbInfo();
  889.  
  890. const injectAndStart = () => {
  891. let injectionEl = document.querySelector(containerSelector);
  892.  
  893. if (!injectionEl) {
  894. throw new Error('LTA: Could not find target container!');
  895. }
  896.  
  897. const container = document.createElement('div');
  898. container.id = divId;
  899. container.style.position = 'relative';
  900.  
  901. if (imdbInfo.layout === 'redesign2020') {
  902. container.className = 'ipc-page-content-container ipc-page-content-container--center';
  903. container.style.padding = '0 var(--ipt-pageMargin)';
  904. container.style.minHeight = '50px';
  905. const targetEl = injectionEl.nextSibling;
  906. injectionEl = injectionEl.parentElement;
  907. injectionEl.insertBefore(container, targetEl);
  908. } else {
  909. container.classList.add('article');
  910. injectionEl.appendChild(container);
  911. }
  912.  
  913. preact.render(preact.h(App, {
  914. imdbInfo: imdbInfo
  915. }), container);
  916. };
  917.  
  918. const containerWatchdog = () => {
  919. const container = document.querySelector(`#${divId}`);
  920.  
  921. if (container === null) {
  922. injectAndStart();
  923. }
  924.  
  925. window.setTimeout(containerWatchdog, 1000);
  926. };
  927.  
  928. window.setTimeout(containerWatchdog, 500);
  929.  
  930. }(preact, preactHooks));