Trakt.tv | Custom Links (Watch-Now + External)

Adds custom links to all the "Watch-Now" and "External" sections (for titles and people). The defaults include Letterboxd, Stremio, streaming sites (e.g. P-Stream), torrent aggregators (e.g. EXT, Knaben) and more. Easily customizable. See README for details.

// ==UserScript==
// @name         Trakt.tv | Custom Links (Watch-Now + External)
// @description  Adds custom links to all the "Watch-Now" and "External" sections (for titles and people). The defaults include Letterboxd, Stremio, streaming sites (e.g. P-Stream), torrent aggregators (e.g. EXT, Knaben) and more. Easily customizable. See README for details.
// @version      0.5.1
// @namespace    https://github.com/Fenn3c401
// @author       Fenn3c401
// @license      GPL-3.0-or-later
// @homepageURL  https://github.com/Fenn3c401/Trakt.tv-Userscript-Collection#readme
// @supportURL   https://github.com/Fenn3c401/Trakt.tv-Userscript-Collection/issues
// @icon         
// @match        https://trakt.tv/*
// @run-at       document-start
// @resource     cineby.app     https://www.cineby.app/logo.png
// @resource     bitcine.app    https://www.bitcine.app/logo.svg
// @resource     hexa.watch     https://hexa.watch/hexa-logo.png
// @grant        unsafeWindow
// @grant        GM_addStyle
// @grant        GM_getResourceURL
// @grant        GM.cookie
// ==/UserScript==

/* README
> Inspired by Tusky's [Trakt Watchlist Downloader](https://greasyfork.org/scripts/17991) and Accus1958's [trakt.tv Streaming Services Integration](https://greasyfork.org/scripts/486706) userscripts.

### General
- The installation of the [Trakt.tv | Trakt API Module](f785bub0.md) userscript is optional, as there is a scraping-based fallback, but very much recommended. Scraping is comparatively slow,
    resource-intensive and error prone, with a heavily reduced set of available item-data. You can see what data is available by scraping alone, in the `scrapeFromSummaryPage()` function.
- This script also makes the color of watch-now buttons correspond to the title's digital release status. White means the title is available on a streaming service for your selected watch-now country,
    light-grey means the title is available on a streaming service of another country and dark-grey means that the titles is not available on any streaming service.
    Keep in mind that the data-source-counts attribute (which this is based on) can be unreliable, the attribute can be empty despite the title being available for streaming or
    contain sources which don't actually have the title available yet.
- `DEFAULT_WNLINK_ADDITIONS` controls how many custom watch-now links are added to the two-slot watch-now preview on title summary pages and header search results. Can be 0-2.

### Adding/Modifying Custom Links
- Custom links are always inserted above the default links, in the order in which they appear in the respective configuration array.
- Item-data can be of five different types: `movies`, `shows`, `seasons`, `episodes` and `people`. `people` item-data is never provided to watch-now links. `seasons` and `episodes` item-data
    is the same as that for `shows`, except for it having an additional season/episode number property. You can see what data was fetched by checking `window.userscriptItemDataCache` in the console.
- There are eight properties that can be assigned to a custom link, all of which are technically optional, though at least a `name` and either `buildUrl` or `evalOnClick` should be provided:

| *LINK_TYPE* | *PROP_NAME*   | *DESCRIPTION* |
| :---------- | :------------ | :------------ |
| both        | `name`        | The name of the link. For watch-now links this will replace the logo if none is provided. |
| both        | `buildUrl`    | A function which takes item-data and returns an absolute url, which will be opened in a new tab on click. Defaults to current page url + '#' (which does nothing). |
| both        | `evalOnClick` | Will be set as onclick attribute if provided. Useful for e.g. cross-script interactions. Still allows for page navigation if `buildUrl` was set, unless you return `false`. |
| both        | `includeIf`   | A predicate function which takes item-data (and returns a boolean) to decide on whether to include this link for the current item. Defaults to `true`. Useful for e.g. only including a watch-now link if the title is of a specific genre OR not including an external link on `/people` pages. |
| watch_now   | `category`    | A link category to be displayed in a second line below the name. See the `watchNowCategories` object, though any string will work. The category line is omitted if no category is provided. |
| watch_now   | `bgColor`     | The background color of the link button. Defaults to `DEFAULT_WNLINK_BGCOLOR`. |
| watch_now   | `textColor`   | The text color of the link button, which is used for the name that is displayed instead of the logo if none is provided. Defaults to `DEFAULT_WNLINK_TEXTCOLOR`. |
| watch_now   | `logo`        | A logo for the link. Should have a transparent background. Can be a data uri (base64 encoded image, see "Tools" section below) or a regular url. Some external sites may disallow hotlinking of images, in that case use a data uri. Or they may have a restrictive image caching policy (then logo has to constantly be re-fetched, which results in a noticeable loading delay), in that case use `GM_getResourceURL` to have the userscript manager handle caching. |

### Tools
- [https://base64.guru/converter/encode/image](https://base64.guru/converter/encode/image)
- [https://compresspng.com](https://compresspng.com)
- [https://www.svgviewer.dev/](https://www.svgviewer.dev/)
*/


'use strict';

const DEFAULT_WNLINK_ADDITIONS = 1;
const DEFAULT_WNLINK_BGCOLOR = '#333';
const DEFAULT_WNLINK_TEXTCOLOR = '#fff';

const watchNowCategories = {
  regular: 'Regular',
  streamingSiteDirectLink: 'Streaming Direct',
  streamingSiteSearchLink: 'Streaming Search',
  torrentAggregator: 'Aggregator',
  torrentTracker: 'Tracker',
  usenetIndexer: 'Indexer',
};

const buildUrlTemplates = {
  torrentsDefault: (i) => `${encodeURIComponent(i.title)}${i.type === 'movies' && i.year ? ` ${i.year}` : ''}${i.season !== undefined ? ` s${String(i.season).padStart(2, '0')}${i.episode ? `e${String(i.episode).padStart(2, '0')}` : ''}` : ''}`,
  streamingDirectPathDefault: (i) => `/${i.type === 'movies' ? `movie/${i.ids.tmdb}` : `tv/${i.ids.tmdb}/${i.season !== undefined ? i.season : '1'}/${i.episode ? i.episode : '1'}`}`,
};

const customWatchNowLinks = [
  {
    name: 'EXT',
    buildUrl: (i) => `https://ext.to/browse/?q=${buildUrlTemplates.torrentsDefault(i)} 1080p 265&with_adult=1`,
    category: watchNowCategories.torrentAggregator,
    bgColor: '#262a33',
    logo: '',
  },
  {
    name: 'Knaben Database',
    buildUrl: (i) => `https://knaben.org/search/${buildUrlTemplates.torrentsDefault(i)} 1080p 265/${i.type === 'movies' ? '3000000' : i.genres.includes('anime') ? '6000000' : '2000000'}/1/seeders`,
    category: watchNowCategories.torrentAggregator,
    bgColor: '#2c2f31',
    logo: '',
  },
  {
    name: 'BTDigg',
    buildUrl: (i) => `https://btdig.com/search?q=${buildUrlTemplates.torrentsDefault(i)} 1080p x265`,
    category: watchNowCategories.torrentAggregator,
    bgColor: '#0e2130',
    logo: '',
  },
  {
    name: 'Stremio',
    buildUrl: (i) => `https://web.stremio.com/#/detail/${i.type === 'movies' ? 'movie' : 'series'}/${i.ids.imdb}/${i.ids.imdb}${i.type === 'movies' ? '' : `%3A${i.season !== undefined ? i.season : '1'}%3A${i.episode ? i.episode : '1'}`}`,
    category: watchNowCategories.streamingSiteDirectLink,
    bgColor: '#19163a',
    logo: 'https://web.stremio.com/images/stremio_symbol.png',
  },
  {
    name: 'AnimeKAI',
    buildUrl: (i) => `https://animekai.to/browser?keyword=${i.title}`,
    includeIf: (i) => i.genres.includes('anime'),
    category: watchNowCategories.streamingSiteSearchLink,
    bgColor: '#0d1116',
    logo: 'https://animekai.to/assets/uploads/37585a39fe8c8d8fafaa2c7bfbf5374ecac859ea6a0886b7dc.png',
  },
  {
    name: 'P-Stream',
    buildUrl: (i) => `https://iframe.pstream.mov/embed/tmdb-${i.type === 'movies' ? `movie-${i.ids.tmdb}` : `tv-${i.ids.tmdb}/${i.season !== undefined ? i.season : '1'}/${i.episode ? i.episode : '1'}`}`,
    category: watchNowCategories.streamingSiteDirectLink,
    bgColor: '#080914',
    logo: '',
  },
  {
    name: 'Cineby',
    buildUrl: (i) => `https://www.cineby.app${buildUrlTemplates.streamingDirectPathDefault(i)}`,
    category: watchNowCategories.streamingSiteDirectLink,
    bgColor: '#520000',
    logo: GM_getResourceURL('cineby.app'),
  },
  {
    name: 'Hexa Watch',
    buildUrl: (i) => `https://hexa.watch/watch${buildUrlTemplates.streamingDirectPathDefault(i)}`,
    category: watchNowCategories.streamingSiteDirectLink,
    bgColor: '#111317',
    logo: GM_getResourceURL('hexa.watch'),
  },
  {
    name: 'Fmovies+',
    buildUrl: (i) => `https://www.fmovies.gd/watch${buildUrlTemplates.streamingDirectPathDefault(i)}`,
    category: watchNowCategories.streamingSiteDirectLink,
    bgColor: '#111a1e',
    textColor: '#0ea6c4',
  },
  {
    name: 'Bitcine',
    buildUrl: (i) => `https://www.bitcine.app${buildUrlTemplates.streamingDirectPathDefault(i)}?play=true`,
    category: watchNowCategories.streamingSiteDirectLink,
    bgColor: '#1f0a37',
    logo: GM_getResourceURL('bitcine.app'),
  },
  {
    name: 'SceneNZBs',
    buildUrl: (i) => `https://scenenzbs.com/search/${buildUrlTemplates.torrentsDefault(i)} 1080p 265`,
    category: watchNowCategories.usenetIndexer,
    bgColor: '#212529',
    logo: '',
  },
];

const customExternalLinks = [
  {
    name: 'Reddit',
    buildUrl: (i) => `https://google.com/search?q=site:reddit.com discussion ${encodeURIComponent(i.title)}${i.type === 'movies' && i.year ? ` ${i.year}` : ''}${i.season !== undefined ? ` season ${i.season}${i.episode ? ` episode ${i.episode}` : ''}` : ''}`,
    includeIf: (i) => i.type !== 'people',
  },
  {
    name: 'LBXD',
    buildUrl: (i) => `https://letterboxd.com/search/films/tmdb:${i.ids.tmdb}/`,
    includeIf: (i) => i.type === 'movies',
  },
  {
    name: 'Fandom',
    buildUrl: (i) => `https://${i.title.toLowerCase().replaceAll(/[^a-z0-9]/g, '')}.fandom.com/wiki/`,
    includeIf: (i) => i.type !== 'people',
  },
  {
    name: 'Spotify',
    buildUrl: (i) => `https://open.spotify.com/search/${i.title} Soundtrack`,
    includeIf: (i) => i.type !== 'people',
  },
  {
    name: 'YouGlish',
    buildUrl: (i) => `https://youglish.com/pronounce/${i.name.replaceAll(' ', '_')}/english`,
    includeIf: (i) => i.type === 'people',
  },
  {
    name: 'Forvo',
    buildUrl: (i) => `https://forvo.com/search/${i.name}/`,
    includeIf: (i) => i.type === 'people',
  },
  {
    name: 'Bacon°',
    buildUrl: (i) => `https://oracleofbacon.org/graph.php?who=${i.name.replaceAll(' ', '+')}`,
    includeIf: (i) => i.type === 'people',
  },
  {
    name: 'AZN',
    buildUrl: (i) => `https://aznude.com/${i.title ? `search.html?q=${encodeURIComponent(i.title)}` : `view/celeb/${i.name.toLowerCase()[0]}/${i.name.toLowerCase().replaceAll(/[^a-z]/g, '')}.html`}`,
    includeIf: (i) => i.type !== 'people' || /female|non_binary/.test(i.gender) && i.birthday && Date.now() - new Date(i.birthday) > 18 * 365.25 * 24 * 60 * 60 * 1000,
  },
  {
    name: 'R34',
    buildUrl: (i) => `https://rule34.xxx/index.php?page=post&s=list&tags=sort:score ${(i.title ?? i.name).toLowerCase().replaceAll(/[^a-z0-9-:; ]/g, '').replaceAll(' ', '_')}`,
    includeIf: (i) => i.type !== 'people' || /female|non_binary/.test(i.gender) && i.birthday && Date.now() - new Date(i.birthday) > 18 * 365.25 * 24 * 60 * 60 * 1000,
  },
];

///////////////////////////////////////////////////////////////////////////////////////////////

let $, trakt;
const itemDataCache = unsafeWindow.userscriptItemDataCache = {};


addStyles();

document.addEventListener('turbo:load', async () => {
  $ ??= unsafeWindow.jQuery;
  trakt ??= unsafeWindow.userscriptTraktAPIModule?.isFulfilled ? await unsafeWindow.userscriptTraktAPIModule : null;
  if (!$) return;

  const itemUrl = $('.notable-summary').attr('data-url') || $('.sidebar').attr('data-url'),
        itemData = /^\/(movies|shows|seasons|episodes|people)/.test(itemUrl) ? await getItemData(itemUrl) : undefined;

  if (customExternalLinks && itemData) {
    addExternalLinksToSidebar(itemData);
    addExternalLinksToAdditionalStats(itemData);
  }

  if (customWatchNowLinks) {
    if (itemData && itemData.type !== 'people') {
      addWatchNowLinksToSidebar(itemData);
      addWatchNowLinksToActionButtons(itemData);
    }

    const $searchResults = $('#header-search-autocomplete-results');
    if ($searchResults.length) {
      $(document).off('ajaxSuccess.userscript83278').on('ajaxSuccess.userscript83278', (_evt, _xhr, opt) => {
        if (/^\/search\/autocomplete(?!\/(people|lists|users))/.test(opt.url)) addWatchNowLinksToSearchResults($searchResults);
      });
    }

    const $watchNowContent = $('#watch-now-content');
    if ($watchNowContent.length) {
      if ($watchNowContent.has('.streaming-links').length) addWatchNowLinksToModal($watchNowContent);

      $(document).off('ajaxSuccess.userscript79689').on('ajaxSuccess.userscript79689', (_evt, _xhr, opt) => {
        if (opt.url.includes('/streaming_links?country=')) addWatchNowLinksToModal($watchNowContent);
      });
    }
  }
}, { capture: true });

///////////////////////////////////////////////////////////////////////////////////////////////

const newExternalLinkElem = (l, itemData) =>
  `<a target="_blank" id="" href="${l.buildUrl ? l.buildUrl(itemData) : '#'}"` +
  `${l.evalOnClick ? ` onclick="${l.evalOnClick(itemData)}; return $(this).attr('href') !== '#';"` : ''} data-original-title="" title="">${l.name}</a>`;

function addExternalLinksToSidebar(itemData) {
  $('#info-wrapper .sidebar .external > li').prepend(
    customExternalLinks
      .filter((l) => l.includeIf ? l.includeIf(itemData) : true)
      .map((l) => newExternalLinkElem(l, itemData))
      .join('')
  );
}

function addExternalLinksToAdditionalStats(itemData) {
  $('.additional-stats.with-external-links label:contains("Links")').after(
    customExternalLinks
      .filter((l) => l.includeIf ? l.includeIf(itemData) : true)
      .map((l) => newExternalLinkElem(l, itemData) + ', ')
      .join('')
  );
}

///////////////////////////////////////////////////////////////////////////////////////////////

const newWatchNowLinkElem = (l, itemData) =>
  `<a class="${/ (4k|uhd)/i.test(l.buildUrl) ? 'has-uhd' : ''}" target="_blank" rel="nofollow" data-source="custom_links_userscript" data-country="" ` + // no link attr to prevent href override
  `href="${l.buildUrl ? l.buildUrl(itemData) : '#'}"${l.evalOnClick ? ` onclick="${l.evalOnClick(itemData)}; return $(this).attr('href') !== '#';"` : ''} data-original-title="" title="">` +
    `<div class="icon btn-custom" data-country="" style="${l.bgColor ? `--btn-custom-bg-color: ${l.bgColor};` : ''}">` +
      (l.logo ? `<img class="lazy" data-original="" src="${l.logo}" alt="${l.name}">` : `<div class="text" style="${l.textColor ? `--btn-custom-text-color: ${l.textColor};` : ''}">${l.name?.replace(' ', '<br>')}</div>`) +
    `</div>` +
  `</a>`;

function addWatchNowLinksToSidebar(itemData) {
  const $sidebar = $('#info-wrapper .sidebar');
  let linkAdditions = DEFAULT_WNLINK_ADDITIONS;

  if ($sidebar.has('.btn-watch-now').length && !$sidebar.has('.streaming-links').length) {
    $sidebar.find('.btn-watch-now').before(
      `<div class="streaming-links">` +
        `<div class="sources"></div>` +
      `</div>`
    );
    linkAdditions = 2;
  }

  $sidebar
    .find('.streaming-links .sources').prepend(
      customWatchNowLinks
        .filter((l) => l.includeIf ? l.includeIf(itemData) : true)
        .slice(0, linkAdditions)
        .map((l) => newWatchNowLinkElem(l, itemData))
        .join('')
    )
    .find('a').slice(2).remove();
}

function addWatchNowLinksToActionButtons(itemData) {
  const $actionButtons = $('#overview .action-buttons');
  let linkAdditions = DEFAULT_WNLINK_ADDITIONS;

  if ($actionButtons.length && !$actionButtons.has('.btn-watch-now').length) {
    const $sidebarBtnWatchNow = $('#info-wrapper .sidebar .btn-watch-now'),
          dataSourceCounts = $sidebarBtnWatchNow.attr('data-source-counts'),
          itemUrl = $sidebarBtnWatchNow.attr('data-url');
    if (!dataSourceCounts || !itemUrl) return;

    $actionButtons.prepend(
      `<div class="streaming-links">` +
        `<div class="sources"></div>` +
      `</div>` +
      `<a class="btn btn-block btn-summary btn-watch-now visible-xs selected" data-source-counts="${dataSourceCounts}" data-target="#watch-now-modal" data-toggle="modal" data-url="${itemUrl}" href="#">` +
        `<i class="fa fa-fw fa-solid fa-play"></i>` +
        `<div class="text">` +
          `<div class="main-info">Watch Now</div>` +
          `<div class="under-info">0 streaming services</div>` +
        `</div>` +
      `</a>`
    );
    linkAdditions = 2;
  }

  $actionButtons
    .find('.sources').prepend(
      customWatchNowLinks
        .filter((l) => l.includeIf ? l.includeIf(itemData) : true)
        .slice(0, linkAdditions)
        .map((l) => $(newWatchNowLinkElem(l, itemData)).removeAttr('rel link dataSource'))
    )
    .find('a').slice(2).remove();
}

async function addWatchNowLinksToSearchResults($searchResults) {
  $searchResults.find('> .search-result').each(async function() {
    const itemData = await getItemData($(this).attr('data-url'));
    let linkAdditions = DEFAULT_WNLINK_ADDITIONS;

    if (!$(this).has('.streaming-links').length) {
      $(this).append(
        `<div class="streaming-links">` +
          `<div class="sources"></div>` +
        `</div>`
      );
      linkAdditions = 2;
    }

    $(this)
      .find('.streaming-links .sources').prepend(
        customWatchNowLinks
          .filter((l) => l.includeIf ? l.includeIf(itemData) : true)
          .slice(0, linkAdditions)
          .map((l) => $(newWatchNowLinkElem(l, itemData))
            .removeAttr('data-original-title title')
            .on('click', (evt) => evt.stopPropagation()) // don't trigger default click handler on .search-result
          )
      )
      .find('a').slice(2).remove();
  });
}

async function addWatchNowLinksToModal($watchNowContent) {
  const itemData = await getItemData($watchNowContent.attr('data-url'));

  $watchNowContent
    .find('> .streaming-links').prepend(
      `<div class="title">Custom Links</div>` +
      `<div class="section"></div>` +
      ($watchNowContent.has('.no-links').length ? `<div class="title"></div>` : '')
    )
    .find('.section').first().append(
      customWatchNowLinks
        .filter((l) => l.includeIf ? l.includeIf(itemData) : true)
        .map((l) => $(newWatchNowLinkElem(l, itemData)).append(`<div class="price">${l.name}${l.category ? `<br><i>(${l.category})</i>` : ''}</div>`))
    );
}

///////////////////////////////////////////////////////////////////////////////////////////////

async function getItemData(itemUrl) {
  const fetchFromApi = async () => {
    let pathSplit = itemUrl.split('/').filter(Boolean),
        id = pathSplit[1]; // is trakt-id for seasons + eps and slug for shows + movs + people (can be numeric though e.g. /shows/1883 which gets interpreted as trakt-id by api)

    if (!isNaN(id)) {
      const resp = await fetch(itemUrl);
      if (!resp.ok) throw new Error(`getItemData: Fetching ${resp.url} failed with status: ${resp.status}`);

      const replaceWithShowSlug = /seasons|episodes/.test(pathSplit[0]); // use show data for seasons + eps
      if (replaceWithShowSlug) {
        pathSplit = new URL(resp.url).pathname.split('/').filter(Boolean);
        id = pathSplit[1];
      }

      const convertNumericSlugToTraktId = !isNaN(pathSplit[1]);
      if (convertNumericSlugToTraktId) {
        const itemDoc = new DOMParser().parseFromString(await resp.text(), 'text/html');
        id = $(itemDoc).find('.summary-user-rating').attr(`data-${pathSplit[0].slice(0, -1)}-id`);
      }
    }

    return {
      itemUrl,
      type: pathSplit[0],
      ...(await trakt[pathSplit[0]].summary({ id, extended: 'full' })),
      ...(pathSplit[3] && { season: +pathSplit[3] }),
      ...(pathSplit[5] && { episode: +pathSplit[5] }),
    }
  };

  // will return a subset of the data returned by fetchFromApi(), with some inconsistencies e.g. "year" and "original_title" of shows are not embedded on season + ep summary pages // TODO
  const scrapeFromSummaryPage = async () => {
    let itemDoc = document;
    if (location.pathname !== itemUrl) {
      const resp = await fetch(itemUrl);
      if (!resp.ok) throw new Error(`getItemData: Fetching ${resp.url} failed with status: ${resp.status}`);
      itemDoc = new DOMParser().parseFromString(await resp.text(), 'text/html');
    }

    const type = itemUrl.split('/').filter(Boolean)[0],
          $additionalStatsLi = $(itemDoc).find('.additional-stats > li'),
          $notableSummary = $(itemDoc).find('.notable-summary'),
          filterStatsElemsByLabel = (labelText) => $additionalStatsLi.filter((_, e) => $(e).find('label').text().toLowerCase() === labelText);

    const itemData = {
      itemUrl,
      type,
      ids: {
        trakt: +($notableSummary.attr('data-movie-id') ?? $notableSummary.attr('data-show-id') ?? $notableSummary.attr('data-person-id')),
        imdb: $(itemDoc).find('#external-link-imdb').attr('href')?.match(/(?:tt|nm)\d+/)?.[0], // tt = 'title type', nm = 'name'
        tmdb: +$(itemDoc).find('#external-link-tmdb').attr('href')?.match(/\d+/)?.[0] || undefined,
      },
      ...(type !== 'people' && { title: $(itemDoc).find(':is(body > [itemtype$="Movie"], body > [itemtype$="TVSeries"], body > [itemtype] > [itemtype$="TVSeries"]) > meta[itemprop="name"]')
                                                  .attr('content')?.match(/(.+?)(?: \(\d{4}\))?$/)?.[1] }),
      ...(/shows|movies/.test(type) && { original_title: filterStatsElemsByLabel('original title').contents().get(-1)?.textContent }),
      ...(/shows|movies/.test(type) && { year: +$(itemDoc).find('#summary-wrapper .mobile-title .year').get(0)?.textContent || undefined }),
      ...(type !== 'people' && { genres: $additionalStatsLi.find('[itemprop="genre"]').map((_, e) => $(e).text().toLowerCase()).get() }),
      ...(/seasons|episodes/.test(type) && { season: +$notableSummary.attr('data-season-number') }),
      ...(type === 'episodes' && { episode: +$notableSummary.attr('data-episode-number') }),
      ...(type === 'people' && { name: $(itemDoc).find('body > [itemtype$="Person"] > meta[itemprop="name"]').attr('content') }),
      ...(type === 'people' && { gender: filterStatsElemsByLabel('gender').contents().get(-1)?.textContent.toLowerCase().replace('-', '_') }),
      ...(type === 'people' && { birthday: filterStatsElemsByLabel('birthday').children().last().attr('data-date') }),
    };
    if (Object.hasOwn(itemData, 'original_title')) itemData.original_title ??= itemData.title;

    return itemData;
  }

  if (!itemDataCache[itemUrl]) itemDataCache[itemUrl] = await (trakt ? fetchFromApi : scrapeFromSummaryPage)();
  return itemDataCache[itemUrl];
}

///////////////////////////////////////////////////////////////////////////////////////////////

async function addStyles() {
  const watchNowCountry = (await GM.cookie.list({ name: 'watchnow_country' }))[0]?.value ?? new Intl.Locale(navigator.language).region.toLowerCase();

  GM_addStyle(`
    .no-watchnow-sources:not([data-url^="/people"]) {
      display: block !important;
    }
    [data-source-counts] > .fa-play::before {
      color: #777 !important;
    }
    [data-source-counts="{}"] > .fa-play::before {
      color: #333 !important;
    }
    [data-source-counts*="${watchNowCountry}"] > .fa-play::before {
      color: #ccc !important;
    }
    [data-source-counts] > .fa-play:hover::before {
      color: #fff !important;
    }

    :root {
      --btn-custom-bg-color: ${DEFAULT_WNLINK_BGCOLOR};
      --btn-custom-text-color: ${DEFAULT_WNLINK_TEXTCOLOR};
    }
    .streaming-links a > .icon.btn-custom {
      display: flex !important;
      justify-content: center !important;
      border-width: 0px !important;
      padding: 4px 6px !important;
      background-color: var(--btn-custom-bg-color) !important;
    }
    .streaming-links a > .icon.btn-custom:hover {
      background-color: color-mix(in oklab, var(--btn-custom-bg-color), black 20%) !important;
    }
    .streaming-links a > .icon.btn-custom > img {
      object-fit: contain !important;
    }
    .streaming-links a > .icon.btn-custom > .text {
      color: var(--btn-custom-text-color) !important;
      font-weight: 1000 !important;
    }


    #watch-now-modal {
      top: 37.5px !important;
    }
    #watch-now-modal #watch-now-content .streaming-links {
      max-height: calc(100vh - 190px) !important;
      overflow: auto !important;
      scrollbar-width: none !important;
    }
  `);
}