您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
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/preact@10.5.7/dist/preact.umd.js
- // @require https://unpkg.com/preact@10.5.7/hooks/dist/hooks.umd.js
- // @license GPLv2
- // @noframes
- // @author buzz
- // @version 2.0.8
- // @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.8";
- 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_results: 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_RESULTS: 1,
- RESULTS_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__8dIDU {\n margin-top: 10px;\n}\n\n .Options_options__8dIDU > label > span {\n margin-left: 10px;\n}\n";
- var css$6 = {"options":"Options_options__8dIDU"};
- 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__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";
- 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"};
- 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.noResultsMatcher ? preact.h(Icon, {
- className: css$5.extraIcon,
- title: "Site supports fetching of results",
- 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.resultCount
- }, "Showing ", preact.h("span", null, total), " sites.")), preact.h("div", {
- className: css$5.siteList
- }, cats));
- };
- 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";
- var css$4 = {"about":"About_about__3lHx7","top":"About_top__3XyCB","content":"About_content__1xMTu"};
- 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__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 + 250px);\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: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(250px - 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";
- var css$3 = {"popover":"Config_popover__3RK3L","inner":"Config_inner__2Sbjz","top":"Config_top__2kgQ3","link":"Config_link__3aqRB","active":"Config_active__iBK3y","body":"Config_body__2JuhF","controls":"Config_controls__3hBBQ"};
- styleInject(css_248z$3);
- const OPTIONS = [['show_category_captions', 'Show category captions'], ['open_blank', 'Open links in new tab'], ['fetch_results', 'Automatically fetch results']];
- const Config = ({
- config,
- 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 fetchResultsArr = hooks.useState(config.fetch_results);
- const [showCategoryCaptions, setShowCategoryCaptions] = showCategoryCaptionsArr;
- const [openBlank, setOpenBlank] = openBlankArr;
- const [fetchResults, setFetchResults] = fetchResultsArr;
- const optStates = [showCategoryCaptionsArr, openBlankArr, fetchResultsArr];
- 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);
- setFetchResults(config.fetch_results);
- setOpenBlank(config.open_blank);
- setShowCategoryCaptions(config.show_category_captions);
- };
- const onClickSave = () => {
- setConfig({
- enabled_sites: enabledSites,
- fetch_results: fetchResults,
- open_blank: openBlank,
- show_category_captions: showCategoryCaptions
- });
- setShow(false);
- };
- return preact.h("div", {
- className: css$3.popover,
- 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() {
- _extends = Object.assign || function (target) {
- for (var i = 1; i < arguments.length; i++) {
- var source = arguments[i];
- for (var key in source) {
- if (Object.prototype.hasOwnProperty.call(source, key)) {
- target[key] = source[key];
- }
- }
- }
- return target;
- };
- return _extends.apply(this, 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 results
- if (Array.isArray(site.noResultsMatcher)) {
- // Advanced ways of checking, currently only EL_COUNT is supported
- const [checkType, selector, compType, number] = site.noResultsMatcher;
- 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 result;
- try {
- result = doc.querySelectorAll(selector);
- } catch (err) {
- console.error(err);
- return FETCH_STATE.ERROR;
- }
- if (compType === 'GT') {
- if (result.length > number) {
- return FETCH_STATE.RESULTS_FOUND;
- }
- }
- if (compType === 'LT') {
- if (result.length < number) {
- return FETCH_STATE.RESULTS_FOUND;
- }
- }
- break;
- }
- }
- return FETCH_STATE.NO_RESULTS;
- }
- const matchStrings = Array.isArray(site.noResultsMatcher) ? site.noResultsMatcher : [site.noResultsMatcher];
- if (matchStrings.some(matchString => resp.responseText.includes(matchString))) {
- return FETCH_STATE.NO_RESULTS;
- }
- return FETCH_STATE.RESULTS_FOUND;
- };
- const useResultFetcher = (imdbInfo, site) => {
- const [fetchState, setFetchState] = hooks.useState(null);
- hooks.useEffect(() => {
- let xhr;
- if (site.noResultsMatcher) {
- // Site supports result 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 results 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__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";
- var css$2 = {"linkWrapper":"SiteLink_linkWrapper__2uDyT","resultsIcon":"SiteLink_resultsIcon__3_V-k"};
- styleInject(css_248z$2);
- const ResultsIndicator = ({
- imdbInfo,
- site
- }) => {
- const fetchState = useResultFetcher(imdbInfo, site);
- let iconType;
- let title;
- switch (fetchState) {
- case FETCH_STATE.LOADING:
- iconType = 'spinner';
- title = 'Loading…';
- break;
- case FETCH_STATE.NO_RESULTS:
- iconType = 'x';
- title = 'No Results found!';
- break;
- case FETCH_STATE.RESULTS_FOUND:
- iconType = 'tick';
- title = 'Results 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 results! (See dev console for details)';
- break;
- default:
- return null;
- }
- return preact.h(Icon, {
- className: css$2.resultsIcon,
- title: title,
- type: iconType
- });
- };
- 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)), preact.h(ResultsIndicator, {
- imdbInfo: imdbInfo,
- site: site
- }), last ? null : preact.h(Sep, null));
- };
- 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";
- var css$1 = {"linkList":"LinkList_linkList__rlGOn","h4":"LinkList_h4__2axTi"};
- 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__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";
- var css = {"configWrapper":"App_configWrapper__2KuAE"};
- styleInject(css_248z);
- 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));
- }, []);
- 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 (!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,
- setConfig: setConfig,
- setShow: setShowConfig,
- sites: sites,
- show: showConfig
- })), preact.h(LinkList, {
- config: config,
- imdbInfo: imdbInfo,
- sites: sites
- }));
- };
- 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('[data-testid="hero-title-block__title"]')) {
- return ['redesign2020', 'title', '[class*=TitleMainBelowTheFold]'];
- } // 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 = /^\/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();
- let injectionEl = document.querySelector(containerSelector);
- if (!injectionEl) {
- throw new Error('LTA: Could not find target container!');
- }
- if (imdbInfo.layout === 'redesign2020') {
- injectionEl = injectionEl.parentElement;
- }
- const container = document.createElement('div');
- container.style.position = 'relative';
- if (imdbInfo.layout === 'redesign2020') {
- container.style.padding = '0 var(--ipt-pageMargin)';
- container.style.minHeight = '50px';
- injectionEl.insertBefore(container, injectionEl.firstChild);
- } else {
- container.classList.add('article');
- injectionEl.appendChild(container);
- }
- preact.render(preact.h(App, {
- imdbInfo: imdbInfo
- }), container);
- }(preact, preactHooks));