Adds all kinds of links to IMDb, customizable!
// ==UserScript== // @name IMDb: Link 'em all! // @description Adds all kinds of links to IMDb, customizable! // @namespace https://greasyfork.org/en/users/8981-buzz // @match *://*.imdb.com/*title/tt*/* // @connect * // @require https://greasemonkey.github.io/gm4-polyfill/gm4-polyfill.js // @require https://unpkg.com/[email protected]/dist/preact.umd.js // @require https://unpkg.com/[email protected]/hooks/dist/hooks.umd.js // @license GPLv2 // @noframes // @author buzz // @version 2.0.15 // @grant GM_getValue // @grant GM_setValue // @grant GM_xmlhttpRequest // @grant GM.getValue // @grant GM.setValue // @grant GM.xmlHttpRequest // ==/UserScript== (function (preact, hooks) { 'use strict'; var version = "2.0.15"; var description = "Adds all kinds of links to IMDb, customizable!"; var homepage = "https://github.com/buzz/imdb-link-em-all#readme"; const DESCRIPTION = description; const HOMEPAGE = homepage; const NAME_VERSION = `Link 'em all! v${version}`; const SITES_URL = 'https://raw.githubusercontent.com/buzz/imdb-link-em-all/master/sites.json'; // gets replaced by rollup! const GM_CONFIG_KEY = 'config'; const GREASYFORK_URL = 'https://greasyfork.org/scripts/17154-imdb-link-em-all'; const DEFAULT_CONFIG = { enabled_sites: [], fetch_r###lts: true, first_run: true, open_blank: true, show_category_captions: true }; const CATEGORIES = { search: 'Search', movie_site: 'Movie sites', pub_tracker: 'Public trackers', priv_tracker: 'Private trackers', streaming: 'Streaming', filehoster: 'Filehosters', subtitles: 'Subtitles', tv: 'TV' }; const FETCH_STATE = { LOADING: 0, NO_R###LTS: 1, R###LTS_FOUND: 2, NO_ACCESS: 3, TIMEOUT: 4, ERROR: 5 }; var img$8 = ""; var img$7 = ""; var img$6 = ""; var img$5 = ""; var img$4 = ""; var img$3 = ""; var img$2 = ""; var img$1 = ""; var img = ""; const iconSrcs = { cog: img$8, error: img$7, info: img$6, lock: img$5, tick: img$4, timeout: img$3, world: img$1, x: img$2, spinner: img }; const Icon = ({ className, title, type }) => preact.h("img", { alt: `${type} icon`, className: className, src: iconSrcs[type], title: title }); function styleInject(css, ref) { if ( ref === void 0 ) ref = {}; var insertAt = ref.insertAt; if (!css || typeof document === 'undefined') { return; } var head = document.head || document.getElementsByTagName('head')[0]; var style = document.createElement('style'); style.type = 'text/css'; if (insertAt === 'top') { if (head.firstChild) { head.insertBefore(style, head.firstChild); } else { head.appendChild(style); } } else { head.appendChild(style); } if (style.styleSheet) { style.styleSheet.cssText = css; } else { style.appendChild(document.createTextNode(css)); } } var css_248z$6 = ".Options_options__5TB2e {\n margin-top: 10px;\n}\n\n .Options_options__5TB2e > label > span {\n margin-left: 10px;\n}\n"; var css$6 = {"options":"Options_options__5TB2e"}; styleInject(css_248z$6); const Options = ({ options }) => { const optionLabels = options.map(([key, title, val, setter]) => preact.h("label", { key: key }, preact.h("input", { checked: val, onInput: ev => setter(ev.target.checked), type: "checkbox" }), preact.h("span", null, title), preact.h("br", null))); return preact.h("div", { className: css$6.options }, optionLabels); }; const SiteIcon = ({ className, site, title }) => site.icon ? preact.h("img", { alt: site.title, className: className, src: site.icon, title: title }) : null; var css_248z$5 = ".Sites_searchBar__omy0k {\n display: flex;\n flex-direction: row;\n margin-bottom: 1em;\n}\n\n .Sites_searchBar__omy0k .Sites_searchInput__0o5oY {\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__omy0k .Sites_searchInput__0o5oY: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__omy0k .Sites_searchInput__0o5oY > * {\n background-color: transparent;\n border: none;\n height: 16px;\n}\n\n .Sites_searchBar__omy0k .Sites_searchInput__0o5oY > button {\n margin: 0 0 0 0.7em;\n padding: 0;\n}\n\n .Sites_searchBar__omy0k .Sites_searchInput__0o5oY > input {\n flex-grow: 1;\n outline: none;\n padding: 0 0 0 0.5em;\n}\n\n .Sites_searchBar__omy0k .Sites_r###ltCount__xMc-y {\n font-weight: bold;\n margin-left: 2em;\n min-width: 140px;\n text-align: right;\n}\n\n .Sites_searchBar__omy0k .Sites_r###ltCount__xMc-y > span {\n color: black;\n}\n\n.Sites_siteList__4rCbT .Sites_catList__Fv8G0 {\n display: flex;\n flex-wrap: wrap;\n}\n\n.Sites_siteList__4rCbT .Sites_catList__Fv8G0 h4 {\n width: 100%;\n}\n\n.Sites_siteList__4rCbT .Sites_catList__Fv8G0 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__4rCbT .Sites_catList__Fv8G0 label:hover {\n color: #222;\n}\n\n.Sites_siteList__4rCbT .Sites_catList__Fv8G0 label.Sites_checked__nqnSg span {\n color: black;\n}\n\n.Sites_siteList__4rCbT .Sites_catList__Fv8G0 label .Sites_title__4rEy0 {\n flex-grow: 1;\n overflow: hidden;\n text-overflow: ellipsis;\n}\n\n.Sites_siteList__4rCbT .Sites_catList__Fv8G0 label input {\n margin-right: 4px;\n}\n\n.Sites_siteList__4rCbT .Sites_catList__Fv8G0 label .Sites_extraIcon__YYfVy {\n height: 12px;\n margin-left: 4px;\n width: 12px;\n}\n\n.Sites_siteList__4rCbT .Sites_catList__Fv8G0 label .Sites_siteIcon__GRVSj {\n flex-shrink: 0;\n height: 16px;\n margin-right: 6px;\n width: 16px;\n}\n"; var css$5 = {"searchBar":"Sites_searchBar__omy0k","searchInput":"Sites_searchInput__0o5oY","r###ltCount":"Sites_r###ltCount__xMc-y","siteList":"Sites_siteList__4rCbT","catList":"Sites_catList__Fv8G0","checked":"Sites_checked__nqnSg","title":"Sites_title__4rEy0","extraIcon":"Sites_extraIcon__YYfVy","siteIcon":"Sites_siteIcon__GRVSj"}; styleInject(css_248z$5); const SearchInput = ({ q, setQ }) => preact.h("div", { className: css$5.searchInput }, preact.h("span", null, "\uD83D\uDD0D"), preact.h("input", { onInput: e => { setQ(e.target.value.toLowerCase().trim()); }, placeholder: "Search", value: q }), preact.h("button", { style: { display: q.length ? 'unset' : 'none' }, title: "Clear", type: "button", onClick: () => setQ('') }, preact.h(Icon, { type: "x" }))); const DummyIcon = ({ size }) => { const sizePx = `${size}px`; const style = { display: 'inline-block', height: sizePx, width: sizePx }; return preact.h("div", { className: css$5.siteIcon, style: style }); }; const SiteLabel = ({ checked, setEnabled, site }) => { const input = preact.h("input", { checked: checked, onInput: e => setEnabled(prev => e.target.checked ? [...prev, site.id] : prev.filter(id => id !== site.id)), type: "checkbox" }); const icon = site.icon ? preact.h(SiteIcon, { className: css$5.siteIcon, site: site, title: site.title }) : preact.h(DummyIcon, { size: 16 }); const title = preact.h("span", { className: css$5.title, title: site.title }, site.title); const extraIcons = [site.noAccessMatcher ? preact.h(Icon, { className: css$5.extraIcon, title: "Access restricted", type: "lock" }) : null, site.noR###ltsMatcher ? preact.h(Icon, { className: css$5.extraIcon, title: "Site supports fetching of r###lts", type: "tick" }) : null]; return preact.h("label", { className: checked ? css$5.checked : null }, input, icon, " ", title, " ", extraIcons); }; const CategoryList = ({ enabled, name, setEnabled, sites }) => { const siteLabels = sites.map(site => preact.h(SiteLabel, { checked: enabled.includes(site.id), setEnabled: setEnabled, site: site })); return preact.h("div", { className: css$5.catList }, preact.h("h4", null, name, " ", preact.h("span", null, "(", sites.length, ")")), siteLabels); }; const Sites = ({ enabledSites, setEnabledSites, sites }) => { const [q, setQ] = hooks.useState(''); const catSites = Object.keys(CATEGORIES).map(cat => { const s = sites.filter(site => site.category === cat); if (q.length) { return s.filter(site => site.title.toLowerCase().includes(q)); } return s; }); const cats = Object.entries(CATEGORIES).map(([cat, catName], i) => catSites[i].length ? preact.h(CategoryList, { enabled: enabledSites, key: cat, name: catName, setEnabled: setEnabledSites, sites: catSites[i] }) : null); const total = catSites.reduce((acc, s) => acc + s.length, 0); return preact.h(preact.Fragment, null, preact.h("div", { className: css$5.searchBar }, preact.h(SearchInput, { q: q, setQ: setQ }), preact.h("div", { className: css$5.r###ltCount }, "Showing ", preact.h("span", null, total), " sites.")), preact.h("div", { className: css$5.siteList }, cats)); }; var css_248z$4 = ".About_about__wuWQp {\n padding: 1em 0;\n position: relative;\n}\n\n .About_about__wuWQp ul > li {\n margin-bottom: 0;\n}\n\n .About_about__wuWQp h2 {\n font-size: 20px;\n margin: 0.5em 0;\n}\n\n .About_about__wuWQp > *:last-child {\n margin-bottom: 0;\n}\n\n .About_about__wuWQp .About_top__jQHYs {\n text-align: center;\n}\n\n .About_about__wuWQp .About_content__hReHO {\n width: 61.8%;\n margin: 0 auto;\n}\n"; var css$4 = {"about":"About_about__wuWQp","top":"About_top__jQHYs","content":"About_content__hReHO"}; styleInject(css_248z$4); const About = () => preact.h("div", { className: css$4.about }, preact.h("div", { className: css$4.top }, preact.h("h3", null, "\uD83C\uDFA5 ", NAME_VERSION), preact.h("p", null, DESCRIPTION)), preact.h("div", { className: css$4.content }, preact.h("h2", null, "\uD83D\uDD17 Links"), preact.h("ul", null, preact.h("li", null, preact.h("a", { target: "_blank", rel: "noreferrer", href: HOMEPAGE }, "GitHub")), preact.h("li", null, preact.h("a", { target: "_blank", rel: "noreferrer", href: GREASYFORK_URL }, "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", { target: "_blank", rel: "noreferrer", href: "https://github.com/buzz/imdb-link-em-all/issues/new" }, "Open a GitHub issue"), ' ', "or"), preact.h("li", null, preact.h("a", { target: "_blank", rel: "noreferrer", href: "https://greasyfork.org/en/scripts/17154-imdb-link-em-all/feedback" }, "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", { target: "_blank", rel: "noreferrer", href: "https://github.com/buzz/imdb-link-em-all/blob/master/LICENSE" }, "GPL-2.0 License"), "."))); var css_248z$3 = ".Config_popover__qMfu9 {\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__qMfu9.Config_layout-legacy__M6fyd {\n left: calc(-800px + 235px);\n}\n.Config_popover__qMfu9.Config_layout-legacy__M6fyd:before {\n right: calc(235px - 2 * 8px);\n}\n.Config_popover__qMfu9: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__qMfu9 .Config_inner__oVRAK {\n display: flex;\n flex-direction: column;\n text-shadow: 0 1px 0 rgba(255, 255, 255, 0.2);\n}\n.Config_popover__qMfu9 .Config_inner__oVRAK .Config_top__6DKJ8 {\n display: flex;\n flex-direction: row;\n}\n.Config_popover__qMfu9 .Config_inner__oVRAK .Config_top__6DKJ8 .Config_link__GTbGq {\n flex-grow: 1;\n text-align: right;\n}\n.Config_popover__qMfu9 .Config_inner__oVRAK .Config_top__6DKJ8 .Config_link__GTbGq > a {\n color: #333;\n margin-left: 12px;\n margin-right: 4px;\n}\n.Config_popover__qMfu9 .Config_inner__oVRAK .Config_top__6DKJ8 .Config_link__GTbGq > a:visited {\n color: #333;\n}\n.Config_popover__qMfu9 .Config_inner__oVRAK .Config_top__6DKJ8 > 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__qMfu9 .Config_inner__oVRAK .Config_top__6DKJ8 > button:hover {\n background-color: rgba(0, 0, 0, 0.1);\n color: #222;\n}\n.Config_popover__qMfu9 .Config_inner__oVRAK .Config_top__6DKJ8 > button.Config_active__vD-Fl {\n background-color: #c2c2c2;\n color: #222;\n}\n.Config_popover__qMfu9 .Config_inner__oVRAK .Config_top__6DKJ8 > button:last-child {\n margin-right: 0;\n}\n.Config_popover__qMfu9 .Config_inner__oVRAK .Config_top__6DKJ8 > button > img {\n vertical-align: text-bottom;\n}\n.Config_popover__qMfu9 .Config_inner__oVRAK .Config_body__wtDKH {\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__qMfu9 .Config_inner__oVRAK .Config_body__wtDKH > div {\n overflow: hidden;\n}\n.Config_popover__qMfu9 .Config_inner__oVRAK .Config_body__wtDKH > div > *:first-child {\n margin-top: 0;\n}\n.Config_popover__qMfu9 .Config_inner__oVRAK .Config_body__wtDKH > div > *:last-child {\n margin-bottom: 0;\n}\n.Config_popover__qMfu9 .Config_inner__oVRAK .Config_controls__-N2ev {\n display: flex;\n flex-direction: row;\n margin-top: 10px;\n}\n.Config_popover__qMfu9 .Config_inner__oVRAK .Config_controls__-N2ev > div:first-child {\n flex-grow: 1;\n}\n.Config_popover__qMfu9 .Config_inner__oVRAK .Config_controls__-N2ev button {\n padding-bottom: 0;\n padding-top: 0;\n margin-right: 12px;\n}\n"; var css$3 = {"popover":"Config_popover__qMfu9","layout-legacy":"Config_layout-legacy__M6fyd","inner":"Config_inner__oVRAK","top":"Config_top__6DKJ8","link":"Config_link__GTbGq","active":"Config_active__vD-Fl","body":"Config_body__wtDKH","controls":"Config_controls__-N2ev"}; styleInject(css_248z$3); const OPTIONS = [['show_category_captions', 'Show category captions'], ['open_blank', 'Open links in new tab'], ['fetch_r###lts', 'Automatically fetch r###lts']]; const Config = ({ config, layout, setConfig, setShow, show, sites }) => { const [enabledSites, setEnabledSites] = hooks.useState(config.enabled_sites); const showCategoryCaptionsArr = hooks.useState(config.show_category_captions); const openBlankArr = hooks.useState(config.open_blank); const fetchR###ltsArr = hooks.useState(config.fetch_r###lts); const [showCategoryCaptions, setShowCategoryCaptions] = showCategoryCaptionsArr; const [openBlank, setOpenBlank] = openBlankArr; const [fetchR###lts, setFetchR###lts] = fetchR###ltsArr; const optStates = [showCategoryCaptionsArr, openBlankArr, fetchR###ltsArr]; const options = OPTIONS.map((opt, i) => [...opt, ...optStates[i]]); const [tab, setTab] = hooks.useState(0); const tabs = [{ title: 'Sites', icon: 'world', comp: preact.h(Sites, { enabledSites: enabledSites, setEnabledSites: setEnabledSites, sites: sites }) }, { title: 'Options', icon: 'cog', comp: preact.h(Options, { options: options }) }, { title: 'About', icon: 'info', comp: preact.h(About, null) }]; const onClickCancel = () => { setShow(false); // Restore state setEnabledSites(config.enabled_sites); setFetchR###lts(config.fetch_r###lts); setOpenBlank(config.open_blank); setShowCategoryCaptions(config.show_category_captions); }; const onClickSave = () => { setConfig({ enabled_sites: enabledSites, fetch_r###lts: fetchR###lts, open_blank: openBlank, show_category_captions: showCategoryCaptions }); setShow(false); }; return preact.h("div", { className: `${css$3.popover} ${css$3['layout-' + layout]}`, style: { display: show ? 'block' : 'none' } }, preact.h("div", { className: css$3.inner }, preact.h("div", { className: css$3.top }, tabs.map(({ title, icon }, i) => preact.h("button", { className: tab === i ? css$3.active : null, type: "button", onClick: () => setTab(i) }, preact.h(Icon, { title: title, type: icon }), " ", title)), preact.h("div", { className: css$3.link }, preact.h("a", { target: "_blank", rel: "noreferrer", href: HOMEPAGE }, "\uD83C\uDFA5 ", NAME_VERSION))), preact.h("div", { className: css$3.body }, tabs.map(({ comp }, i) => preact.h("div", { style: { display: tab === i ? 'block' : 'none' } }, comp))), preact.h("div", { className: css$3.controls }, preact.h("div", null, preact.h("button", { className: "btn primary small", onClick: onClickSave, type: "button" }, "OK"), preact.h("button", { className: "btn small", onClick: onClickCancel, type: "button" }, "Cancel"))))); }; function _extends() { return _extends = Object.assign ? Object.assign.bind() : function (n) { for (var e = 1; e < arguments.length; e++) { var t = arguments[e]; for (var r in t) ({}).hasOwnProperty.call(t, r) && (n[r] = t[r]); } return n; }, _extends.apply(null, arguments); } const replaceFields = (str, { id, title, year }, 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); const checkResponse = (resp, site) => { // Likely a redirect to login page if (resp.responseHeaders && resp.responseHeaders.includes('Refresh: 0; url=')) { return FETCH_STATE.NO_ACCESS; } // There should be a responseText if (!resp.responseText) { return FETCH_STATE.ERROR; } // Detect Blogger content warning if (resp.responseText.includes('The blog that you are about to view may contain content only suitable for adults.')) { return FETCH_STATE.NO_ACCESS; } // Detect CloudFlare anti DDOS page if (resp.responseText.includes('Checking your browser before accessing')) { return FETCH_STATE.NO_ACCESS; } // Check site access if (site.noAccessMatcher) { const matchStrings = Array.isArray(site.noAccessMatcher) ? site.noAccessMatcher : [site.noAccessMatcher]; if (matchStrings.some(matchString => resp.responseText.includes(matchString))) { return FETCH_STATE.NO_ACCESS; } } // Check r###lts if (Array.isArray(site.noR###ltsMatcher)) { // Advanced ways of checking, currently only EL_COUNT is supported const [checkType, selector, compType, number] = site.noR###ltsMatcher; const m = resp.responseHeaders.match(/content-type:\s([^\s;]+)/); const contentType = m ? m[1] : 'text/html'; let doc; try { const parser = new DOMParser(); doc = parser.parseFromString(resp.responseText, contentType); } catch (e) { console.error('Could not parse document!'); return FETCH_STATE.ERROR; } switch (checkType) { case 'EL_COUNT': { let r###lt; try { r###lt = doc.querySelectorAll(selector); } catch (err) { console.error(err); return FETCH_STATE.ERROR; } if (compType === 'GT') { if (r###lt.length > number) { return FETCH_STATE.R###LTS_FOUND; } } if (compType === 'LT') { if (r###lt.length < number) { return FETCH_STATE.R###LTS_FOUND; } } break; } } return FETCH_STATE.NO_R###LTS; } const matchStrings = Array.isArray(site.noR###ltsMatcher) ? site.noR###ltsMatcher : [site.noR###ltsMatcher]; if (matchStrings.some(matchString => resp.responseText.includes(matchString))) { return FETCH_STATE.NO_R###LTS; } return FETCH_STATE.R###LTS_FOUND; }; const useR###ltFetcher = (imdbInfo, site) => { const [fetchState, setFetchState] = hooks.useState(null); hooks.useEffect(() => { let xhr; if (site.noR###ltsMatcher) { // Site supports r###lt fetching const { url } = site; const isPost = Array.isArray(url); const opts = { timeout: 20000, onload: resp => setFetchState(checkResponse(resp, site)), onerror: resp => { console.error(`Failed to fetch r###lts from URL '${url}': ${resp.statusText}`); setFetchState(FETCH_STATE.ERROR); }, ontimeout: () => setFetchState(FETCH_STATE.TIMEOUT) }; if (isPost) { const [postUrl, fields] = url; opts.method = 'POST'; opts.url = postUrl; opts.headers = { 'Content-Type': 'application/x-www-form-urlencoded' }; opts.data = Object.keys(fields).map(key => { const val = replaceFields(fields[key], imdbInfo, false); return `${key}=${val}`; }).join('&'); } else { opts.method = 'GET'; opts.url = replaceFields(url, imdbInfo); } xhr = GM.xmlHttpRequest(opts); setFetchState(FETCH_STATE.LOADING); } return () => { if (xhr && xhr.abort) { xhr.abort(); } }; }, [imdbInfo, site]); return fetchState; }; var css_248z$2 = ".SiteLink_linkWrapper__wGnJ- {\n display: inline-block;\n margin-right: 4px;\n}\n\n .SiteLink_linkWrapper__wGnJ- img {\n vertical-align: baseline;\n}\n\n .SiteLink_linkWrapper__wGnJ- a {\n white-space: pre-line;\n}\n\n .SiteLink_linkWrapper__wGnJ- a > img {\n height: 16px;\n width: 16px;\n margin-right: 4px;\n}\n\n .SiteLink_linkWrapper__wGnJ- .SiteLink_r###ltsIcon__mjHYM {\n margin-left: 4px;\n}\n"; var css$2 = {"linkWrapper":"SiteLink_linkWrapper__wGnJ-","r###ltsIcon":"SiteLink_r###ltsIcon__mjHYM"}; styleInject(css_248z$2); const R###ltsIndicator = ({ imdbInfo, site }) => { const fetchState = useR###ltFetcher(imdbInfo, site); let iconType; let title; switch (fetchState) { case FETCH_STATE.LOADING: iconType = 'spinner'; title = 'Loading…'; break; case FETCH_STATE.NO_R###LTS: iconType = 'x'; title = 'No R###lts found!'; break; case FETCH_STATE.R###LTS_FOUND: iconType = 'tick'; title = 'R###lts found!'; break; case FETCH_STATE.NO_ACCESS: iconType = 'lock'; title = 'You have to login to this site!'; break; case FETCH_STATE.TIMEOUT: iconType = 'timeout'; title = 'You have to login to this site!'; break; case FETCH_STATE.ERROR: iconType = 'error'; title = 'Error fetching r###lts! (See dev console for details)'; break; default: return null; } return preact.h(Icon, { className: css$2.r###ltsIcon, title: title, type: iconType }); }; // As it is not possible to open links with POST request we need a trick const usePostLink = (url, openBlank, imdbInfo) => { const formEl = hooks.useRef(); const isPost = Array.isArray(url); const href = isPost ? url[0] : replaceFields(url, imdbInfo, false); const onClick = event => { if (isPost && formEl.current) { event.preventDefault(); formEl.current.submit(); } }; hooks.useEffect(() => { if (isPost) { const [postUrl, fields] = url; const form = document.createElement('form'); form.action = postUrl; form.method = 'POST'; form.style.display = 'none'; form.target = openBlank ? '_blank' : '_self'; Object.keys(fields).forEach(key => { const input = document.createElement('input'); input.type = 'text'; input.name = key; input.value = replaceFields(fields[key], imdbInfo, false); form.appendChild(input); }); document.body.appendChild(form); formEl.current = form; } return () => { if (formEl.current) { formEl.current.remove(); } }; }); return [href, onClick]; }; const Sep = () => preact.h(preact.Fragment, null, "\xA0", preact.h("span", { className: "ghost" }, "|")); const SiteLink = ({ config, imdbInfo, last, site }) => { const extraAttrs = config.open_blank ? { target: '_blank', rel: 'noreferrer' } : {}; const [href, onClick] = usePostLink(site.url, config.open_blank, imdbInfo); return preact.h("span", { className: css$2.linkWrapper }, preact.h("a", _extends({ className: "ipc-link ipc-link--base", href: href, onClick: onClick }, extraAttrs), preact.h(SiteIcon, { site: site }), preact.h("span", null, site.title)), config.fetch_r###lts ? preact.h(R###ltsIndicator, { imdbInfo: imdbInfo, site: site }) : null, last ? null : preact.h(Sep, null)); }; var css_248z$1 = ".LinkList_linkList__beWAL {\n line-height: 1.6rem\n}\n\n.LinkList_h4__OVHW- {\n margin-top: 0.5rem\n}\n"; var css$1 = {"linkList":"LinkList_linkList__beWAL","h4":"LinkList_h4__OVHW-"}; styleInject(css_248z$1); const LinkList = ({ config, imdbInfo, sites }) => Object.entries(CATEGORIES).map(([category, categoryName]) => { const catSites = sites.filter(site => site.category === category && config.enabled_sites.includes(site.id)); if (!catSites.length) { return null; } const caption = config.show_category_captions ? preact.h("h4", { className: css$1.h4 }, categoryName) : null; return preact.h(preact.Fragment, null, caption, preact.h("div", { className: css$1.linkList }, catSites.map((site, i) => preact.h(SiteLink, { config: config, imdbInfo: imdbInfo, last: i === catSites.length - 1, site: site })))); }); var css_248z = ".App_configWrapper__bVP2M {\n position: absolute;\n right: 20px;\n top: 20px;\n}\n\n .App_configWrapper__bVP2M > button {\n background: transparent;\n border: none;\n cursor: pointer;\n outline: none;\n padding: 0;\n}\n\n .App_configWrapper__bVP2M > button > img {\n vertical-align: baseline;\n}\n"; var css = {"configWrapper":"App_configWrapper__bVP2M"}; styleInject(css_248z); // Note: GM.* only work in async functions const restoreConfig = async () => JSON.parse(await GM.getValue(GM_CONFIG_KEY)); const saveConfig = async config => GM.setValue(GM_CONFIG_KEY, JSON.stringify(config)); const useConfig = () => { const [config, setConfig] = hooks.useState(); hooks.useEffect(() => { restoreConfig().then(c => setConfig(c)).catch(() => setConfig(DEFAULT_CONFIG)); }, []); hooks.useEffect(() => { if (config) { saveConfig(config); } }, [config]); return { config, setConfig }; }; const loadSites = () => new Promise((resolve, reject) => GM.xmlHttpRequest({ method: 'GET', url: SITES_URL, nocache: true, onload({ response, status, statusText }) { if (status === 200) { try { resolve(JSON.parse(response).sort((a, b) => a.title.localeCompare(b.title))); } catch (e) { reject(e); } } else { reject(new Error(`LTA: Could not load sites (URL=${SITES_URL}): ${status} ${statusText}`)); } }, onerror({ status, statusText }) { reject(new Error(`LTA: Could not load sites (URL=${SITES_URL}): ${status} ${statusText}`)); } })); const useSites = () => { const [sites, setSites] = hooks.useState([]); hooks.useEffect(() => { loadSites().then(s => setSites(s)).catch(err => setSites(err.message)); }, []); return sites; }; const App = ({ imdbInfo }) => { const { config, setConfig } = useConfig(); const sites = useSites(); const [showConfig, setShowConfig] = hooks.useState(false); hooks.useEffect(() => { if (config && config.first_run) { setShowConfig(true); setConfig(prev => ({ ...prev, first_run: false })); } }, [config]); if (typeof sites === 'string') { return sites; // Display error message } if (!config || !sites.length) { return null; } return preact.h(preact.Fragment, null, imdbInfo.layout === 'legacy' ? preact.h("hr", null) : null, preact.h("div", { className: css.configWrapper }, preact.h("button", { onClick: () => setShowConfig(cur => !cur), title: "Configure", type: "button" }, preact.h(Icon, { type: "cog" })), preact.h(Config, { config: config, layout: imdbInfo.layout, setConfig: setConfig, setShow: setShowConfig, sites: sites, show: showConfig })), preact.h(LinkList, { config: config, imdbInfo: imdbInfo, sites: sites })); }; const divId = '__LTA__'; const detectLayout = mUrl => { // Currently there seem to be 3 different IMDb layouts: // 1) "legacy": URL ends with '/reference' if (['reference', 'combined'].includes(mUrl[2])) { return ['legacy', 'h3[itemprop=name]', '.titlereference-section-overview > *:last-child']; } // 2) "redesign2020": Redesign 2020 // https://www.imdb.com/preferences/beta-control?e=tmd&t=in&u=/title/tt0163978/ if (document.querySelector('main section > .ipc-page-content-container')) { return ['redesign2020', 'title', 'main > * > section > div']; } // 3) "new": The old default (has been around for many years) return ['new', 'h1', '.title-overview']; }; const parseImdbInfo = () => { // TODO: extract type (TV show, movie, ...) // Parse IMDb number and layout const mUrl = /^\/(?:[a-z]{2}\/)?title\/tt([0-9]{7,8})\/([a-z]*)/.exec(window.location.pathname); if (!mUrl) { throw new Error('LTA: Could not parse IMDb URL!'); } const [layout, titleSelector, containerSelector] = detectLayout(mUrl); const info = { id: mUrl[1], layout }; info.title = document.querySelector(titleSelector).innerText.trim(); const mTitle = /^(.+)\s+\((\d+)\)/.exec(info.title); if (mTitle) { info.title = mTitle[1].trim(); info.year = parseInt(mTitle[2].trim(), 10); } return [info, containerSelector]; }; const [imdbInfo, containerSelector] = parseImdbInfo(); const injectAndStart = () => { let injectionEl = document.querySelector(containerSelector); if (!injectionEl) { throw new Error('LTA: Could not find target container!'); } const container = document.createElement('div'); container.id = divId; container.style.position = 'relative'; if (imdbInfo.layout === 'redesign2020') { container.className = 'ipc-page-content-container ipc-page-content-container--center'; container.style.backgroundColor = 'white'; container.style.padding = '0 var(--ipt-pageMargin)'; container.style.minHeight = '50px'; injectionEl.prepend(container); } else { container.classList.add('article'); injectionEl.appendChild(container); } preact.render(preact.h(App, { imdbInfo: imdbInfo }), container); }; const containerWatchdog = () => { const container = document.querySelector(`#${divId}`); if (container === null) { injectAndStart(); } window.setTimeout(containerWatchdog, 1000); }; window.setTimeout(containerWatchdog, 500); })(preact, preactHooks);