IMDb List/Watchlist → FLAM Launcher

Letterboxd-style launcher

// ==UserScript==
// @name         IMDb List/Watchlist → FLAM Launcher
// @namespace    http://tampermonkey.net/
// @version      3.2.0
// @description  Letterboxd-style launcher
// @match        https://*.imdb.com/list/ls*
// @match        https://*.imdb.com/list/ls*/*
// @match        https://*.imdb.com/user/*/watchlist*
// @grant        GM_xmlhttpRequest
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_addStyle
// @run-at       document-idle
// @connect      *
// @connect      api.themoviedb.org
// @connect      api.github.com
// @connect      raw.githubusercontent.com
// ==/UserScript==

;(async function () {
  'use strict';

  // ────────────────────────────────────────────────────────────────────────
  // Config
  // ────────────────────────────────────────────────────────────────────────
  const TMDB_API_KEY = 'f090bb54758cabf231fb605d3e3e0468';

  // Base cache (gzipped u32 pairs: imdbNumeric → packed)
  const BASE_GZ_URL = 'https://raw.githubusercontent.com/hcgiub001/letterboxd-tmdb-cache/main/imdb_tmdb_pairs_u32.bin.gz';

  // Resolver & fetch limits
  const CONCURRENCY_TMDB_FIND = 50;
  const CONCURRENCY_PAGE_FETCH = 3;   // keep the 3-thread pool for pagination
  const MAX_LIST_ITEMS = 5000;
  const URL_LIMIT_BYTES = 65536;
  const THIRTY_DAYS_MS = 30 * 24 * 60 * 60 * 1000;
  const NEGATIVE_TTL_MS = THIRTY_DAYS_MS; // 30 days

  // IndexedDB stores/keys
  const CACHE_DB_NAME = 'FenlightCacheDB';
  const CACHE_DB_VERSION = 2;
  const OVERRIDES_STORE = 'tmdbCache';
  const BASE_STORE = 'tmdbBase';

  // Overrides & negative caches
  const OVERRIDES_KEY = 'imdb_tmdb_cache_overrides';          // { 'tt…': packed }
  const OVERRIDES_BACKUP_KEY = 'imdb_tmdb_cache_backup';
  const NEGATIVE_KEY = 'imdb_tmdb_cache_negative';            // { 'tt…': lastFailedTs }
  const NEGATIVE_BACKUP_KEY = 'imdb_tmdb_cache_negative_bak';

  // Base cache metadata keys
  const BASE_BIN_KEY  = 'imdb_base_bin';     // ArrayBuffer: unzipped .bin (u32 pairs)
  const BASE_ETAG_KEY = 'imdb_base_gz_etag'; // last ETag of gz
  const BASE_COUNT_KEY = 'imdb_base_count';  // pair count (info only)

  // URL/UX defaults
  const MEDIA_TYPE_DEFAULT_FIXED = 'm';
  const DEFAULT_INDICATOR = 'progress';

  // Artwork defaults (match Letterboxd)
  const POSTER_ENABLE_DEFAULT = true;
  const POSTER_STRATEGY_DEFAULT = 'first_4';
  const FANART_ENABLE_DEFAULT = true;
  const FANART_STRATEGY_DEFAULT = 'first_4';
  const FANART_FALLBACK_DEFAULT = 'first_4';

  // Info button + last stats keys
  const INFO_BTN_ENABLE_KEY = 'showInfoButton';
  const LAST_STATS_KEY = 'lastSendStats';

  // Description defaults
  const DEFAULT_DESCRIPTION = '';

  // ────────────────────────────────────────────────────────────────────────
  // Tiny utils & text helpers (match LB)
  // ────────────────────────────────────────────────────────────────────────
  const sleep = (ms) => new Promise(r => setTimeout(r, ms));
  const utf8ByteLen = (s) => new TextEncoder().encode(s).length;
  const gmGetOrSet = (k, d) => (typeof GM_getValue(k) === 'undefined' ? d : GM_getValue(k));

  // Basic page-type detection
  function isWatchlistPath() {
    // /user/{id or name}/watchlist/ (with or without query)
    return /^\/user\/[^/]+\/watchlist\/?$/i.test(location.pathname) || /^\/user\/[^/]+\/watchlist\/\S*/i.test(location.pathname);
  }

  function htmlToTextWithBreaks(html) {
    if (!html) return '';
    let t = html;
    t = t.replace(/<br\s*\/?>/gi, '\n');
    t = t.replace(/<\/p>\s*<p[^>]*>/gi, '\n\n');
    t = t.replace(/<\/?p[^>]*>/gi, '');
    t = t.replace(/<li[^>]*>\s*/gi, '• ');
    t = t.replace(/<\/li>\s*/gi, '\n');
    t = t.replace(/<\/?ul[^>]*>/gi, '');
    t = t.replace(/<\/?ol[^>]*>/gi, '');
    t = t.replace(/<[^>]+>/g, '');
    const ta = document.createElement('textarea');
    ta.innerHTML = t.replace(/&nbsp;/gi, '\u00A0');
    t = ta.value;
    t = t.replace(/\u00A0/g, ' ');
    t = t.replace(/\r\n/g, '\n').replace(/^\s+/, '').replace(/\s+$/, '');
    return t;
  }
  function sanitizeDescPreservingBreaks(s) {
    if (!s) return '';
    return s.replace(/\u00A0/g, ' ')
            .replace(/\r\n/g, '\n')
            .replace(/\t/g, '  ')
            .trim();
  }

  // ────────────────────────────────────────────────────────────────────────
  // IMDb scraping (pagination + metadata) — LISTS + WATCHLISTS
  // ────────────────────────────────────────────────────────────────────────

  // Shared helpers
  function parseNextDataDoc(doc = document) {
    const el = doc.getElementById('__NEXT_DATA__');
    if (!el || !el.textContent) return null;
    try { return JSON.parse(el.textContent); } catch { return null; }
  }
  function parseNextDataFromHTML(html) {
    // fast/robust: find the script tag content
    const m = html.match(/<script[^>]*id="__NEXT_DATA__"[^>]*>([\s\S]*?)<\/script>/i);
    if (!m) return null;
    try { return JSON.parse(m[1]); } catch { return null; }
  }

  // ----- METADATA (name/author/desc) -----
  function getListName() {
    // LISTS: as before
    let el = document.querySelector('span.hero__primary-text[data-testid="hero__primary-text"]');
    if (el && el.textContent) return el.textContent.trim();

    // WATCHLISTS: try generic h1 / header
    const h1 = document.querySelector('h1, [data-testid="list-page-atf-title-block"] h1, [data-testid="list-page-atf-title-block"] [data-testid="hero__primary-text"]');
    if (h1 && h1.textContent) return h1.textContent.trim();

    // Fallback: __NEXT_DATA__
    const nd = parseNextDataDoc();
    const nameFromND =
      nd?.props?.pageProps?.pageTitle ||
      nd?.props?.pageProps?.listTitle ||
      nd?.props?.pageProps?.seo?.metadata?.title ||
      nd?.props?.pageProps?.seo?.meta?.title ||
      '';
    if (nameFromND) return String(nameFromND).replace(/\s*-\s*IMDb\s*$/i, '').trim();

    // Final fallback: document.title
    return (document.title || 'IMDb List').replace(/\s*-\s*IMDb\s*$/i, '').trim();
  }

  function getAuthor() {
    // LISTS: as before
    const a = document.querySelector('[data-testid="list-author-link"]');
    if (a && a.textContent) return a.textContent.trim();

    // WATCHLISTS: attempt to locate an author/owner link
    // Common patterns: link to /user/{id}/ …
    const owner =
      document.querySelector('a[href^="/user/"][data-testid*="author"], a[href^="/user/"][data-testid*="owner"]') ||
      document.querySelector('[data-testid="list-owner"] a[href^="/user/"]') ||
      document.querySelector('.ipc-title__subtitle a[href^="/user/"]') ||
      document.querySelector('a[href^="/user/"]:not([href*="watchlist"])');
    if (owner && owner.textContent) return owner.textContent.trim();

    // Fallback: __NEXT_DATA__ (if any)
    const nd = parseNextDataDoc();
    const ownerName =
      nd?.props?.pageProps?.owner?.name ||
      nd?.props?.pageProps?.user?.name ||
      nd?.props?.pageProps?.userName ||
      '';
    if (ownerName) return String(ownerName).trim();

    return '';
  }

  function getDescriptionRawHTML() {
    // LISTS: current selector (keep)
    const el = document.querySelector('.list-description-content .ipc-html-content-inner-div');
    if (el && el.innerHTML) return el.innerHTML;

    // WATCHLISTS: often no description; try generic content area
    const w = document.querySelector('[data-testid="list-page-description"], [data-testid="list-page-atf-title-block"] .ipc-html-content-inner-div');
    if (w && w.innerHTML) return w.innerHTML;

    return '';
  }
  function getDescriptionText() {
    return htmlToTextWithBreaks(getDescriptionRawHTML() || '');
  }

  // ----- LIST PAGES (unchanged logic) -----
  function extractItemList(html) {
    // robustly parse the item list from <script type="application/ld+json">
    const re = /<script\s+type="application\/ld\+json">([\s\S]*?)<\/script>/g;
    let m;
    while ((m = re.exec(html))) {
      try {
        const j = JSON.parse(m[1]);
        if (j['@type'] === 'ItemList') return j.itemListElement;
      } catch {}
    }
    return [];
  }

  async function gatherItemsFromListPages() {
    const origin = location.origin;
    const basePath = location.pathname.replace(/\?.*$/,'');
    const sel = document.getElementById('listPagination');
    const total = sel ? sel.options.length : 1;

    // page 1
    const scripts = Array.from(document.querySelectorAll('script[type="application/ld+json"]'));
    let page1 = null;
    for (const s of scripts) {
      try {
        const j = JSON.parse(s.textContent);
        if (j['@type'] === 'ItemList') { page1 = j.itemListElement; break; }
      } catch {}
    }
    if (!page1) { alert('⚠️ Failed to parse page 1'); return null; }

    // rest
    const urls = [];
    for (let p = 2; p <= total; p++) urls.push(`${origin+basePath}?page=${p}`);

    const rest = await (async function limitedFetch(urls, limit = CONCURRENCY_PAGE_FETCH) {
      const out = new Array(urls.length); let next = 0, active = 0;
      return new Promise((resolve) => {
        const launch = () => {
          while (active < limit && next < urls.length) {
            const i = next++, u = urls[i]; active++;
            fetch(u, { credentials: 'include' })
              .then(r => r.ok ? r.text() : Promise.reject(new Error('HTTP '+r.status)))
              .then(html => { out[i] = extractItemList(html); })
              .catch(() => { out[i] = []; })
              .finally(() => { active--; (next < urls.length) ? launch() : (active === 0 && resolve(out)); });
          }
        };
        urls.length ? launch() : resolve(out);
      });
    })(urls, CONCURRENCY_PAGE_FETCH);

    const all = page1.concat(...rest);

    return all
      .map(e => {
        const idMatch = (e.item && e.item.url) ? e.item.url.match(/tt\d+/) : null;
        const t = (e.item && e.item['@type']) ? String(e.item['@type']) : '';
        const mtHint = (t === 'Movie') ? 'm' : (/^TV/i.test(t) ? 'tv' : 'm');
        return idMatch ? { imdbId: idMatch[0], mtHint, listIndex: e.position || 0 } : null;
      })
      .filter(Boolean);
  }

  // ----- WATCHLIST PAGES (new) -----
  function extractWatchlistEdgesFromND(nd) {
    const mcd = nd?.props?.pageProps?.mainColumnData;
    const container = mcd?.predefinedList || mcd?.list || mcd?.titleList || null;
    const edges = container?.titleListItemSearch?.edges;
    return Array.isArray(edges) ? edges : [];
  }
  function edgesToItems(edges) {
    const items = [];
    for (const e of edges) {
      const imdbId = e?.listItem?.id; // "tt…"
      if (!imdbId) continue;
      const pos = e?.node?.absolutePosition || e?.node?.rank || 0;
      const tt = (e?.listItem?.titleType?.id || e?.listItem?.titleType?.text || '').toString().toLowerCase();
      const mtHint = tt.startsWith('tv') ? 'tv' : 'm';
      items.push({ imdbId, mtHint, listIndex: pos });
    }
    return items;
  }
  function getWatchlistPageCountFromDoc(doc) {
    // Try dropdown first
    const sel = doc.querySelector('#listPagination');
    if (sel?.options?.length) return sel.options.length;

    // Try __NEXT_DATA__ totalItems
    const nd = parseNextDataDoc(doc);
    const totalItems = nd?.props?.pageProps?.totalItems;
    if (Number.isFinite(totalItems)) return Math.max(1, Math.ceil(totalItems / 250));

    // Fallback: 1
    return 1;
  }

  async function gatherItemsFromWatchlist() {
    const base = new URL(location.href);
    base.searchParams.set('page', '1');

    // Use current DOM for page 1
    const firstND = parseNextDataDoc(document);
    if (!firstND) { alert('⚠️ Could not read watchlist data on page 1'); return null; }
    const maxPages = getWatchlistPageCountFromDoc(document);

    const resultsByPage = new Map();
    resultsByPage.set(1, edgesToItems(extractWatchlistEdgesFromND(firstND)));

    // Build remaining page URLs
    const urls = [];
    for (let p = 2; p <= maxPages; p++) {
      const u = new URL(base.toString());
      u.searchParams.set('page', String(p));
      urls.push({ p, url: u.toString() });
    }

    // Pool=3 fetch of pages 2..N, parse __NEXT_DATA__, convert to items
    await (async function poolFetchWatchlist(pairs, limit = CONCURRENCY_PAGE_FETCH) {
      let i = 0, active = 0;
      return new Promise((resolve) => {
        const launch = () => {
          while (active < limit && i < pairs.length) {
            const idx = i++; active++;
            const { p, url } = pairs[idx];
            fetch(url, { credentials: 'include' })
              .then(r => r.ok ? r.text() : Promise.reject(new Error('HTTP '+r.status)))
              .then(html => {
                const nd = parseNextDataFromHTML(html);
                const edges = extractWatchlistEdgesFromND(nd || {});
                resultsByPage.set(p, edgesToItems(edges));
              })
              .catch(() => { resultsByPage.set(p, []); })
              .finally(() => {
                active--;
                (i < pairs.length) ? launch() : (active === 0 && resolve());
              });
          }
        };
        pairs.length ? launch() : resolve();
      });
    })(urls, CONCURRENCY_PAGE_FETCH);

    // Combine in order
    const all = [];
    for (let p = 1; p <= maxPages; p++) {
      const arr = resultsByPage.get(p);
      if (Array.isArray(arr)) all.push(...arr);
    }
    return all;
  }

  // Unified entry (keeps all prior behavior, just adds watchlists)
  async function gatherItems() {
    if (isWatchlistPath()) {
      const items = await gatherItemsFromWatchlist();
      return items || [];
    }
    // default: list pages
    const items = await gatherItemsFromListPages();
    return items || [];
  }

  // Try to provide an "author_fanart" equivalent on IMDb: use og:image if present
  function getAuthorFanartUrl_IMDb() {
    const og = document.querySelector('meta[property="og:image"]')?.getAttribute('content') || '';
    return og || '';
  }

  // ────────────────────────────────────────────────────────────────────────
  // TMDB resolver (/find/{imdb_id}) with 50-concurrency (kept)
  // ────────────────────────────────────────────────────────────────────────
  async function tmdbFindByImdb(ttid) {
    const url = `https://api.themoviedb.org/3/find/${ttid}?` + new URLSearchParams({
      api_key: TMDB_API_KEY,
      external_source: 'imdb_id'
    });
    for (let i = 0; i < 3; i++) {
      try {
        const r = await fetch(url);
        if (r.ok) {
          const j = await r.json();
          const mov = Array.isArray(j.movie_results) ? j.movie_results : [];
          const tv  = Array.isArray(j.tv_results) ? j.tv_results : [];
          const ep  = Array.isArray(j.tv_episode_results) ? j.tv_episode_results : [];
          if (mov.length) return { tmdbId: mov[0].id, isTv: false };
          if (tv.length)  return { tmdbId: tv[0].id,  isTv: true  };
          if (ep.length) {
            const showId = ep[0].show_id || ep[0].id || null;
            if (showId) return { tmdbId: showId, isTv: true };
          }
          return null;
        }
        if (r.status === 429) await sleep(250 * (i + 1));
        else return null;
      } catch {
        await sleep(200 * (i + 1));
      }
    }
    return null;
  }

  // ────────────────────────────────────────────────────────────────────────
  // IndexedDB helpers + base cache load/refresh (kept & aligned)
  // ────────────────────────────────────────────────────────────────────────
  function openDB() {
    return new Promise((resolve, reject) => {
      const rq = indexedDB.open(CACHE_DB_NAME, CACHE_DB_VERSION);
      rq.onupgradeneeded = () => {
        const db = rq.result;
        if (!db.objectStoreNames.contains(OVERRIDES_STORE)) db.createObjectStore(OVERRIDES_STORE);
        if (!db.objectStoreNames.contains(BASE_STORE)) db.createObjectStore(BASE_STORE);
      };
      rq.onsuccess = () => resolve(rq.result);
      rq.onerror = () => reject(rq.error);
    });
  }
  async function idbGet(store, key) {
    const db = await openDB();
    return new Promise((res, rej) => {
      const tx = db.transaction(store, 'readonly');
      const rq = tx.objectStore(store).get(key);
      rq.onsuccess = () => res(rq.result);
      rq.onerror = () => rej(rq.error);
    });
  }
  async function idbPut(store, key, val) {
    const db = await openDB();
    return new Promise((res, rej) => {
      const tx = db.transaction(store, 'readwrite');
      const rq = tx.objectStore(store).put(val, key);
      rq.onsuccess = () => res();
      rq.onerror = () => rej(rq.error);
    });
  }
  async function idbDel(store, key) {
    const db = await openDB();
    return new Promise((res, rej) => {
      const tx = db.transaction(store, 'readwrite');
      const rq = tx.objectStore(store).delete(key);
      rq.onsuccess = () => res();
      rq.onerror = () => rej(rq.error);
    });
  }

  let BASE_PAIRS_U32 = null;   // Uint32Array (k0, v0, k1, v1, ...)
  let BASE_PAIR_COUNT = 0;

  function isLittleEndian() {
    const b = new ArrayBuffer(4);
    new DataView(b).setUint32(0, 0x11223344, true);
    return new Uint8Array(b)[0] === 0x44;
  }
  const HOST_LE = isLittleEndian();

  function parseBaseBinToU32(buf) {
    if (!buf || !buf.byteLength) return false;
    if (buf.byteLength % 8 !== 0) return false;
    if (HOST_LE) {
      BASE_PAIRS_U32 = new Uint32Array(buf);
    } else {
      const view = new DataView(buf);
      const out = new Uint32Array(buf.byteLength / 4);
      for (let i = 0; i < out.length; i++) out[i] = view.getUint32(i * 4, true);
      BASE_PAIRS_U32 = out;
    }
    if (BASE_PAIRS_U32.length % 2 !== 0) return false;
    BASE_PAIR_COUNT = BASE_PAIRS_U32.length >>> 1;
    return true;
  }

  function baseLookupPackedByNumericImdb(imdbNumeric) {
    if (!BASE_PAIRS_U32) return undefined;
    let lo = 0, hi = BASE_PAIR_COUNT - 1;
    while (lo <= hi) {
      const mid = (lo + hi) >>> 1;
      const k = BASE_PAIRS_U32[mid << 1];
      if (k === imdbNumeric) return BASE_PAIRS_U32[(mid << 1) + 1];
      if (k < imdbNumeric) lo = mid + 1; else hi = mid - 1;
    }
    return undefined;
  }

  function unpackPacked(packed) {
    return { isTv: !!(packed & 1), tmdbId: packed >>> 1 };
  }

  function gmFetchArrayBuffer(url, headers = {}) {
    return new Promise((resolve, reject) => {
      GM_xmlhttpRequest({
        method: 'GET',
        url,
        headers,
        responseType: 'arraybuffer',
        onload: (res) => {
          if (res.status >= 200 && res.status < 300) resolve({ buf: res.response, headers: res.responseHeaders || '', status: res.status });
          else if (res.status === 304) resolve({ buf: null, headers: res.responseHeaders || '', status: 304 });
          else reject(new Error(`HTTP ${res.status}`));
        },
        onerror: (e) => reject(e.error || e),
        ontimeout: () => reject(new Error('Timeout')),
      });
    });
  }

  async function unzipGzipToArrayBuffer(gzBuf) {
    if (typeof DecompressionStream !== 'function') {
      throw new Error('DecompressionStream(gzip) not supported by this browser.');
    }
    const ds = new DecompressionStream('gzip');
    return await new Response(new Blob([gzBuf]).stream().pipeThrough(ds)).arrayBuffer();
  }

  async function loadBaseFromIDB() {
    const buf = await idbGet(BASE_STORE, BASE_BIN_KEY);
    const count = await idbGet(BASE_STORE, BASE_COUNT_KEY);
    if (!buf || !buf.byteLength) return false;
    if (!parseBaseBinToU32(buf)) return false;
    if (typeof count === 'number') BASE_PAIR_COUNT = count;
    return true;
  }

  async function saveBaseToIDB(buf, etag) {
    await idbPut(BASE_STORE, BASE_BIN_KEY, buf);
    await idbPut(BASE_STORE, BASE_ETAG_KEY, etag || '');
    await idbPut(BASE_STORE, BASE_COUNT_KEY, BASE_PAIR_COUNT);
  }

  async function refreshBaseCacheNow(showAlerts = true) {
    try {
      const prevEtag = (await idbGet(BASE_STORE, BASE_ETAG_KEY)) || '';
      const headers = prevEtag ? { 'If-None-Match': prevEtag } : {};
      const { buf: gzBuf, headers: respHeaders, status } = await gmFetchArrayBuffer(BASE_GZ_URL, headers);
      if (status === 304) {
        GM_setValue('baseLastRevalidateTs', Date.now());
        if (showAlerts) alert('✅ Base cache already up to date.');
        return true;
      }

      const m = String(respHeaders || '').match(/etag:\s*([^\r\n]+)/i);
      const etag = m ? m[1].trim() : '';

      const bin = await unzipGzipToArrayBuffer(gzBuf);
      if (!parseBaseBinToU32(bin)) throw new Error('Invalid base cache (not u32 pairs).');

      await saveBaseToIDB(bin, etag);
      GM_setValue('baseLastRevalidateTs', Date.now());
      if (showAlerts) alert(`✅ Base cache refreshed (${BASE_PAIR_COUNT.toLocaleString()} pairs).`);
      return true;
    } catch (e) {
      console.error('[Base cache] refresh failed:', e);
      if (showAlerts) alert('❌ Base cache refresh failed: ' + e.message);
      return false;
    }
  }

  async function maybeRevalidateBaseMonthly() {
    try {
      const last = gmGetOrSet('baseLastRevalidateTs', 0);
      const now = Date.now();
      if (now - last >= THIRTY_DAYS_MS) {
        await refreshBaseCacheNow(false);
        GM_setValue('baseLastRevalidateTs', Date.now());
      }
    } catch {}
  }

  async function ensureBaseLoadedOnce() {
    if (await loadBaseFromIDB()) return true;
    await refreshBaseCacheNow(false); // first-time fetch
    return await loadBaseFromIDB();
  }

  // ────────────────────────────────────────────────────────────────────────
  // Overrides + Negative caches (kept)
  // ────────────────────────────────────────────────────────────────────────
  let LOCAL_OVERRIDES = null;
  let DIRTY_OVERRIDES = false;

  let LOCAL_NEGATIVE = null;
  let DIRTY_NEGATIVE = false;

  async function overridesLoadOnce() {
    if (!LOCAL_OVERRIDES) LOCAL_OVERRIDES = (await idbGet(OVERRIDES_STORE, OVERRIDES_KEY)) || {};
  }
  async function overridesPersist() {
    if (!DIRTY_OVERRIDES) return;
    await idbPut(OVERRIDES_STORE, OVERRIDES_KEY, LOCAL_OVERRIDES);
    try { await idbPut(OVERRIDES_STORE, OVERRIDES_BACKUP_KEY, JSON.parse(JSON.stringify(LOCAL_OVERRIDES))); } catch {}
    DIRTY_OVERRIDES = false;
  }
  async function overridesClear() {
    await idbDel(OVERRIDES_STORE, OVERRIDES_KEY);
    await idbDel(OVERRIDES_STORE, OVERRIDES_BACKUP_KEY);
    LOCAL_OVERRIDES = {};
    DIRTY_OVERRIDES = false;
  }

  async function negativeLoadOnce() {
    if (!LOCAL_NEGATIVE) LOCAL_NEGATIVE = (await idbGet(OVERRIDES_STORE, NEGATIVE_KEY)) || {};
  }
  function negativeGetTs(ttid) { return LOCAL_NEGATIVE ? LOCAL_NEGATIVE[ttid] : undefined; }
  function negativeIsFresh(ttid, now = Date.now()) {
    const ts = negativeGetTs(ttid);
    if (!Number.isFinite(ts)) return false;
    return (now - ts) < NEGATIVE_TTL_MS;
  }
  function negativeSetNow(ttid, now = Date.now()) {
    if (!LOCAL_NEGATIVE) LOCAL_NEGATIVE = {};
    LOCAL_NEGATIVE[ttid] = now;
    DIRTY_NEGATIVE = true;
  }
  async function negativePersist() {
    if (!DIRTY_NEGATIVE) return;
    await idbPut(OVERRIDES_STORE, NEGATIVE_KEY, LOCAL_NEGATIVE);
    try { await idbPut(OVERRIDES_STORE, NEGATIVE_BACKUP_KEY, JSON.parse(JSON.stringify(LOCAL_NEGATIVE))); } catch {}
    DIRTY_NEGATIVE = false;
  }
  async function negativeClear() {
    await idbDel(OVERRIDES_STORE, NEGATIVE_KEY);
    await idbDel(OVERRIDES_STORE, NEGATIVE_BACKUP_KEY);
    LOCAL_NEGATIVE = {};
    DIRTY_NEGATIVE = false;
  }

  function overrideGetPacked(ttid) {
    const v = LOCAL_OVERRIDES ? LOCAL_OVERRIDES[ttid] : undefined;
    return Number.isFinite(v) ? v : undefined;
  }
  function overrideSetPacked(ttid, tmdbId, isTv) {
    if (!LOCAL_OVERRIDES) LOCAL_OVERRIDES = {};
    const packed = (tmdbId << 1) | (isTv ? 1 : 0);
    if (LOCAL_OVERRIDES[ttid] !== packed) {
      LOCAL_OVERRIDES[ttid] = packed;
      DIRTY_OVERRIDES = true;
    }
  }
  function getFromAnyCachePacked(ttid) {
    // 1) overrides by 'tt…'
    const ov = overrideGetPacked(ttid);
    if (ov !== undefined) return ov;

    // 2) base by numeric key
    const m = ttid.match(/^tt(\d+)$/);
    if (m) {
      const num = Number(m[1]);
      if (Number.isFinite(num)) {
        const b = baseLookupPackedByNumericImdb(num);
        if (b !== undefined) return b;
      }
    }
    return undefined;
  }

  // ────────────────────────────────────────────────────────────────────────
  // Gzip+Base64 builder (B-style)
  // ────────────────────────────────────────────────────────────────────────
  async function gzipBase64(str) {
    try {
      if (typeof CompressionStream !== 'function') return null;
      const enc = new TextEncoder().encode(str);
      const cs = new CompressionStream('gzip');
      const gzStream = new Blob([enc]).stream().pipeThrough(cs);
      const ab = await new Response(gzStream).arrayBuffer();
      const u8 = new Uint8Array(ab);
      let bin = '';
      const chunk = 0x8000;
      for (let i = 0; i < u8.length; i += chunk) {
        const sub = u8.subarray(i, i + chunk);
        bin += String.fromCharCode.apply(null, sub);
      }
      return btoa(bin);
    } catch {
      return null;
    }
  }

  // ────────────────────────────────────────────────────────────────────────
  // Build Fenlight URL — Artwork omitted when action === 'view'
  // ────────────────────────────────────────────────────────────────────────
  async function buildFenUrlTMDB_Art(resolvedItems, opts) {
    const {
      listName, author, description, action = '',
      posterEnabled, posterStrategy,
      fanartEnabled, fanartStrategy, fanartFallback
    } = opts;

    const listItems = resolvedItems.map(it => it.isTv ? ({ id: it.tmdbId, mt: 'tv' }) : ({ id: it.tmdbId }));

    const base = 'plugin://plugin.video.fenlight/?';
    const common = [
      ['mode', 'personal_lists.external'],
      ...(action ? [['action', action]] : []),
      ['list_type', 'tmdb'],
      ['list_name', listName || 'IMDb List'],
      ['author', author || ''],
      ['media_type_default', MEDIA_TYPE_DEFAULT_FIXED],
      ['busy_indicator', gmGetOrSet('indicator', DEFAULT_INDICATOR)],
      ...(description ? [['description', description]] : []),
    ];

    // IMPORTANT CHANGE: If action is 'view', do NOT include poster/fanart keys
    const artworkAllowed = (String(action).toLowerCase() !== 'view');

    // Artwork (match LB exactly, but only if allowed)
    if (artworkAllowed && posterEnabled) {
      common.push(['poster', posterStrategy]); // 'first_4' or 'random'
    }
    if (artworkAllowed && fanartEnabled) {
      if (fanartStrategy === 'author_fanart') {
        let url = getAuthorFanartUrl_IMDb();
        if (url) {
          common.push(['fanart', url]);
        } else {
          if (fanartFallback === 'first_4' || fanartFallback === 'random') {
            common.push(['fanart', fanartFallback]);
          }
          // 'none' ⇒ omit fanart key
        }
      } else {
        // direct strategy: 'first_4' or 'random'
        common.push(['fanart', fanartStrategy]);
      }
    }

    const rawJson = JSON.stringify(listItems);
    const encode = (arr) => arr.map(([k,v]) => `${encodeURIComponent(k)}=${encodeURIComponent(v)}`).join('&');

    // Raw
    const rawParts = [...common, ['list_items', rawJson]];
    const rawUrl = base + encode(rawParts);
    const rawBytes = utf8ByteLen(rawUrl);

    // Gzip+Base64 (if supported)
    let gzB64 = await gzipBase64(rawJson);
    let gzUrl = null, gzBytes = null;
    if (gzB64) {
      const gzParts = [...common, ['base64_items', gzB64]];
      gzUrl = base + encode(gzParts);
      gzBytes = utf8ByteLen(gzUrl);
    }

    // Choose
    const url = gzUrl || rawUrl;
    const urlBytes = (gzBytes != null) ? gzBytes : rawBytes;

    return { url, urlBytes, rawUrl, rawBytes, gzUrl, gzBytes };
  }

  // ────────────────────────────────────────────────────────────────────────
  // Kodi JSON-RPC + trust preflight (kept)
  // ────────────────────────────────────────────────────────────────────────
  function preflightKodiPermission() {
    const ip   = GM_getValue('kodiIp','').trim();
    const port = GM_getValue('kodiPort','').trim();
    const user = GM_getValue('kodiUser','');
    const pass = gmGetOrSet('kodiPass','');

    if (!ip || !port) return; // nothing to preflight

    try {
      GM_xmlhttpRequest({
        method: 'POST',
        url:    `http://${ip}:${port}/jsonrpc`,
        headers: { 'Content-Type':'application/json', 'Authorization':'Basic ' + btoa(`${user}:${pass}`) },
        data: JSON.stringify({ jsonrpc:'2.0', id:0, method:'JSONRPC.Ping' }),
        timeout: 7000,
        onload: () => {}, onerror: () => {}, ontimeout: () => {}
      });
    } catch {}
  }

  function sendToKodi(url) {
    const ip   = GM_getValue('kodiIp','').trim();
    const port = GM_getValue('kodiPort','').trim();
    const user = GM_getValue('kodiUser','');
    const pass = gmGetOrSet('kodiPass','');
    return new Promise((resolve) => {
      if (!ip || !port) { alert('⚠️ Please configure Kodi IP & port in settings.'); resolve(false); return; }
      GM_xmlhttpRequest({
        method: 'POST',
        url:    `http://${ip}:${port}/jsonrpc`,
        headers: { 'Content-Type':'application/json', 'Authorization':'Basic ' + btoa(`${user}:${pass}`) },
        data: JSON.stringify({ jsonrpc:'2.0', id:1, method:'Player.Open', params:{ item:{ file:url } } }),
        timeout: 15000,
        onload: (res) => resolve(res.status >= 200 && res.status < 300),
        onerror: () => resolve(false),
        ontimeout: () => resolve(false),
      });
    });
  }

  // ────────────────────────────────────────────────────────────────────────
  // UI: bubbles, overlays, and ASK — EXACT same visual as Letterboxd
  // ────────────────────────────────────────────────────────────────────────
  function showSideInfoNearEl(el, lines, ms = 5000) {
    try { const prev = document.getElementById('kodi-send-info'); if (prev) prev.remove(); } catch {}
    const box = document.createElement('div');
    box.id = 'kodi-send-info';
    Object.assign(box.style, {
      position:'fixed', top:'0px', right:'130px',
      background:'#1b1b1b', color:'#eee', border:'1px solid #333',
      borderRadius:'6px', padding:'8px 10px', fontSize:'12px',
      lineHeight:'1.4', zIndex:2147483647, boxShadow:'0 2px 10px rgba(0,0,0,.35)',
      maxWidth:'360px', whiteSpace:'pre-wrap', pointerEvents:'none'
    });
    box.innerText = lines.join('\n');
    document.body.append(box);
    let topPx = 150;
    try {
      const rect = el.getBoundingClientRect();
      const boxH = box.offsetHeight;
      topPx = Math.max(10, Math.min(window.innerHeight - boxH - 10, Math.round(rect.top + 40)));
    } catch {}
    box.style.top = `${topPx}px`;
    setTimeout(() => { try { box.remove(); } catch {} }, ms);
  }

  let PROCESS_BUBBLE_EL = null;
  function showOrUpdateProcessingBubble(targetEl, lines, persist = false) {
    const GAP = 10;
    if (!PROCESS_BUBBLE_EL) {
      PROCESS_BUBBLE_EL = document.createElement('div');
      PROCESS_BUBBLE_EL.id = 'kodi-processing-bubble';
      Object.assign(PROCESS_BUBBLE_EL.style, {
        position:'fixed', background:'#1b1b1b', color:'#eee',
        border:'1px solid #333', borderRadius:'8px', padding:'8px 10px',
        fontSize:'12px', lineHeight:'1.4', zIndex:2147483647,
        boxShadow:'0 2px 10px rgba(0,0,0,.35)', maxWidth:'360px', whiteSpace:'pre-wrap'
      });
      document.body.append(PROCESS_BUBBLE_EL);
    }
    PROCESS_BUBBLE_EL.textContent = lines.join('\n');

    const rect = targetEl.getBoundingClientRect();
    const topPx = Math.max(0, Math.round(rect.top));
    const rightPx = Math.max(0, Math.round(window.innerWidth - rect.left + GAP));
    PROCESS_BUBBLE_EL.style.top = `${topPx}px`;
    PROCESS_BUBBLE_EL.style.right = `${rightPx}px`;
    PROCESS_BUBBLE_EL.dataset.persist = persist ? '1' : '0';
    PROCESS_BUBBLE_EL.style.display = 'block';
  }
  function hideProcessingBubble(force = false) {
    if (PROCESS_BUBBLE_EL) {
      if (force || PROCESS_BUBBLE_EL.dataset.persist !== '1') {
        PROCESS_BUBBLE_EL.remove();
        PROCESS_BUBBLE_EL = null;
      }
    }
  }

  // Action chooser — FLAM-styled (MOBILE-SAFE) — same as LB
  function askForActionOverlay() {
    return new Promise((resolve) => {
      const overlay = document.createElement('div');
      overlay.id = 'flam-action-overlay';
      const SAFE_INSET = 16;
      Object.assign(overlay.style, {
        position: 'fixed',
        inset: 0,
        background: 'rgba(0,0,0,0.66)',
        display: 'grid',
        placeItems: 'center',
        zIndex: 2147483647,
        padding: `${SAFE_INSET}px`,
        boxSizing: 'border-box'
      });

      const panel = document.createElement('div');
      panel.className = 'flam-action-panel';
      Object.assign(panel.style, {
        position: 'relative',
        width: `min(760px, calc(100vw - ${SAFE_INSET * 2}px))`,
        maxWidth: `calc(100vw - ${SAFE_INSET * 2}px)`,
        maxHeight: `calc(100vh - ${SAFE_INSET * 2}px)`,
        overflow: 'auto',
        background: 'linear-gradient(180deg,#2a2a2a 0,#1f1f1f 100%)',
        color: '#e8e8e8',
        border: '1px solid rgba(255,255,255,.85)',
        borderRadius: '22px',
        boxShadow: '0 18px 44px rgba(0,0,0,.55)',
        padding: '28px 22px 20px',
        boxSizing: 'border-box'
      });

      const css = document.createElement('style');
      css.textContent = `
        #flam-action-overlay .flam-title {
          text-transform: uppercase;
          letter-spacing: .08em;
          text-align: center;
          font-weight: 700;
          color: #dcdcdc;
          font-size: 18px;
          margin: 0 0 12px;
        }
        #flam-action-overlay .flam-sub {
          text-align: center;
          color: #bdbdbd;
          font-size: 14px;
          margin-bottom: 18px;
        }
        #flam-action-overlay .flam-logo {
          position: absolute;
          top: 12px;
          left: 12px;
          width: 46px;
          height: 46px;
          border-radius: 12px;
          object-fit: contain;
          background: #111;
          border: 1px solid #2b2b2b;
          box-shadow: 0 2px 10px rgba(0,0,0,.35);
        }
        #flam-action-overlay .choices {
          display: grid;
          grid-template-columns: repeat(3, minmax(0,1fr));
          gap: 14px;
        }
        @media (max-width: 720px) {
          #flam-action-overlay .choices { grid-template-columns: 1fr; }
        }
        #flam-action-overlay .choice-btn {
          appearance: none;
          border: 1px solid rgba(255,255,255,.08);
          border-radius: 20px;
          padding: 18px 12px;
          background: #4b444b;
          color: #ffffff;
          font-size: 16px;
          font-weight: 700;
          box-shadow: inset 0 -2px 0 rgba(0,0,0,.12), 0 2px 8px rgba(0,0,0,.25);
          cursor: pointer;
          transition: background .12s ease, color .12s ease, transform .06s ease, outline-color .12s ease;
          outline: none;
          width: 100%;
        }
        #flam-action-overlay .choice-btn:focus-visible {
          background: #4b444b;
          color: #ffffff;
          outline: 2px solid rgba(255,255,255,.9);
          outline-offset: 2px;
        }
        #flam-action-overlay .choice-btn:hover,
        #flam-action-overlay .choice-btn.touching {
          background: #ffffff;
          color: #0b0b0b;
        }
        #flam-action-overlay .choice-btn:active { transform: translateY(1px); }
        #flam-action-overlay .footer { display:flex; justify-content:flex-end; margin-top:16px; }
        #flam-action-overlay .cancel-btn {
          background:#3a3a3a; color:#fff; border:none; border-radius:10px; padding:8px 12px; cursor:pointer;
        }
      `;
      panel.append(css);

      const logo = document.createElement('img');
      logo.className = 'flam-logo';
      try { if (typeof mainIcon !== 'undefined' && mainIcon && mainIcon.src) logo.src = mainIcon.src; } catch {}
      panel.append(logo);

      const title = document.createElement('div');
      title.className = 'flam-title';
      title.textContent = 'Send to FLAM';

      const sub = document.createElement('div');
      sub.className = 'flam-sub';
      sub.textContent = 'Choose what you want to do with this list';

      const choicesWrap = document.createElement('div');
      choicesWrap.className = 'choices';

      function mkBtn(label, value) {
        const b = document.createElement('button');
        b.type = 'button';
        b.className = 'choice-btn';
        b.textContent = label;
        b.addEventListener('pointerdown', () => b.classList.add('touching'), { passive: true });
        b.addEventListener('pointerup', () => b.classList.remove('touching'));
        b.addEventListener('pointercancel', () => b.classList.remove('touching'));
        b.addEventListener('click', () => cleanup(value));
        return b;
      }

      const btnView = mkBtn('View', 'view');
      const btnImport = mkBtn('Import', 'import');
      const btnBoth = mkBtn('Import + View', 'import_view');
      choicesWrap.append(btnView, btnImport, btnBoth);

      const footer = document.createElement('div');
      footer.className = 'footer';
      const cancel = document.createElement('button');
      cancel.className = 'cancel-btn';
      cancel.textContent = 'Cancel';
      cancel.addEventListener('click', () => cleanup(null));
      footer.append(cancel);

      panel.append(title, sub, choicesWrap, footer);
      overlay.append(panel);
      document.body.append(overlay);

      function onKey(e) { if (e.key === 'Escape') cleanup(null); }
      function onBackdrop(e) { if (e.target === overlay) cleanup(null); }
      overlay.addEventListener('click', onBackdrop);
      document.addEventListener('keydown', onKey);

      btnView.focus();

      function cleanup(val) {
        document.removeEventListener('keydown', onKey);
        try { overlay.remove(); } catch {}
        resolve(val);
      }
    });
  }

  function showTooManyItemsOverlay(totalCount) {
    const existing = document.getElementById('kodi-too-many-items'); if (existing) try { existing.remove(); } catch {}
    const overlay = document.createElement('div');
    overlay.id = 'kodi-too-many-items';
    Object.assign(overlay.style, {
      position:'fixed', inset:0, background:'rgba(0,0,0,0.7)',
      display:'flex', alignItems:'center', justifyContent:'center',
      zIndex:2147483647, padding:'20px', boxSizing:'border-box'
    });
    const panel = document.createElement('div');
    Object.assign(panel.style, {
      background:'#222', color:'#eee', borderRadius:'10px',
      width:'560px', maxWidth:'95vw', padding:'20px',
      boxShadow:'0 10px 30px rgba(0,0,0,0.45)', border:'1px solid #333'
    });
    panel.innerHTML = `
      <div style="display:flex;align-items:center;gap:10px;margin-bottom:8px">
        <div style="background:#e50914;width:28px;height:28px;border-radius:6px;display:flex;align-items:center;justify-content:center;font-weight:800">!</div>
        <h2 style="margin:0;font-size:18px;">This list is a little too epic to send in one go</h2>
      </div>
      <div style="font-size:13px; color:#ccc; line-height:1.55;">
        Your list contains <strong>${totalCount.toLocaleString()}</strong> items. For reliability, sending lists over
        <strong>${MAX_LIST_ITEMS.toLocaleString()}</strong> items isn’t supported in a single run.
        <br><br>
        Try sending it in smaller parts (for example, a few pages at a time) or filter the list and send again.
      </div>
      <div style="margin-top:14px;text-align:right;">
        <button id="kodi-too-many-items-ok" style="background:#e50914;color:#fff;border:none;border-radius:6px;padding:8px 12px;cursor:pointer">Got it</button>
      </div>
    `;
    overlay.append(panel);
    panel.querySelector('#kodi-too-many-items-ok').onclick = () => { try { overlay.remove(); } catch {} };
    document.body.append(overlay);
  }

  // Description editor (same as LB)
  function editDescriptionOverlay(defaultDesc) {
    return new Promise(resolve => {
      const overlay = document.createElement('div');
      overlay.id = 'kodieditdesc';
      Object.assign(overlay.style, {
        position: 'fixed', top: 0, left: 0, width: '100vw', height: '100vh',
        background: 'rgba(0,0,0,0.7)', display: 'flex', alignItems: 'center',
        justifyContent: 'center', zIndex: 2147483647,
        padding: '20px', boxSizing: 'border-box'
      });
      const panel = document.createElement('div');
      panel.innerHTML = `
        <h2 style="margin:0 0 12px; color:#fff; font-size:18px;">Edit Description</h2>
        <textarea id="kodieditdesc_text" style="width:600px; max-width:95vw; height:300px; font-size:14px; box-sizing:border-box; white-space:pre-wrap;">${defaultDesc}</textarea>
        <div style="margin-top:12px; text-align:right;">
          <button id="kodieditdesc_save" style="margin-right:8px; padding:6px 12px; font-size:14px;">Save</button>
          <button id="kodieditdesc_cancel" style="padding:6px 12px; font-size:14px;">Cancel</button>
        </div>
      `;
      Object.assign(panel.style, { background: '#222', padding: '20px', borderRadius: '6px' });
      overlay.append(panel);
      document.body.append(overlay);

      panel.querySelector('#kodieditdesc_save').onclick = () => {
        const val = panel.querySelector('#kodieditdesc_text').value;
        overlay.remove();
        resolve(val);
      };
      panel.querySelector('#kodieditdesc_cancel').onclick = () => {
        overlay.remove();
        resolve(null);
      };
    });
  }

  // ────────────────────────────────────────────────────────────────────────
  // Processing core (partition, resolve, persist)
  // ────────────────────────────────────────────────────────────────────────
  async function processAll(ttItems) {
    const baseHits = [];
    const overrideHits = [];
    const negativeHits = [];
    const toResolve = [];

    const now = Date.now();

    // Partition by caches
    for (const it of ttItems) {
      const packed = getFromAnyCachePacked(it.imdbId);
      if (packed !== undefined) {
        const { tmdbId, isTv } = unpackPacked(packed);
        if (overrideGetPacked(it.imdbId) !== undefined) overrideHits.push({ ...it, tmdbId, isTv });
        else baseHits.push({ ...it, tmdbId, isTv });
        continue;
      }
      if (negativeIsFresh(it.imdbId, now)) {
        negativeHits.push({ ...it });
      } else {
        toResolve.push(it);
      }
    }

    const preCounts = {
      base: baseHits.length,
      overrides: overrideHits.length,
      cachedNoId: negativeHits.length,
      uncached: toResolve.length
    };

    // Resolve uncached via TMDB (pool)
    if (toResolve.length) {
      const launchEl = mainIcon || document.body;
      showOrUpdateProcessingBubble(launchEl, [
        'Resolving via TMDB…',
        `Uncached: ${toResolve.length}`,
        `Pool: ${CONCURRENCY_TMDB_FIND}`
      ], true);

      const results = await (async function resolveUncachedWithPool(ttids, onProgress) {
        let idx = 0, active = 0, done = 0;
        const results = Object.create(null);

        return new Promise((resolve) => {
          const launch = () => {
            while (active < CONCURRENCY_TMDB_FIND && idx < ttids.length) {
              const ttid = ttids[idx++]; active++;
              (async () => {
                let tries = 0, res = null;
                while (tries < 3 && res === null) {
                  tries++;
                  try { res = await tmdbFindByImdb(ttid); }
                  catch { res = null; await sleep(300 * tries); }
                }
                results[ttid] = res;
              })().finally(() => {
                active--; done++;
                if (typeof onProgress === 'function') onProgress(done, ttids.length);
                if (idx < ttids.length) launch(); else if (active === 0) resolve(results);
              });
            }
          };
          ttids.length ? launch() : resolve(results);
        });
      })(toResolve.map(x => x.imdbId), (d, total) => {
        showOrUpdateProcessingBubble(launchEl, [
          'Resolving via TMDB…',
          `Progress: ${d} / ${total}`,
          `Pool: ${CONCURRENCY_TMDB_FIND}`
        ], true);
      });

      hideProcessingBubble(true);

      for (const it of toResolve) {
        const r = results[it.imdbId];
        if (r && r.tmdbId) {
          overrideSetPacked(it.imdbId, r.tmdbId, !!r.isTv);
          overrideHits.push({ ...it, tmdbId: r.tmdbId, isTv: !!r.isTv });
        } else {
          negativeSetNow(it.imdbId, now);
          negativeHits.push({ ...it });
        }
      }
      await overridesPersist();
      await negativePersist();
    }

    const resolved = baseHits.concat(overrideHits);
    return {
      resolved,
      counts: preCounts
    };
  }

  // ────────────────────────────────────────────────────────────────────────
  // Launcher ICON + Settings + optional INFO button (LETTERBOXD-STYLE)
  // ────────────────────────────────────────────────────────────────────────
  GM_addStyle(`
    #lb-tmdb-icon-wrap {
      position: fixed; bottom: 10px; right: 10px; display: flex; gap: 8px; align-items: stretch;
      z-index: 2147483647;
      font-family: system-ui, -apple-system, Segoe UI, Roboto, Ubuntu, Cantarell, Noto Sans, Arial;
    }
    #lb-tmdb-main-icon {
      width: var(--lb-size, 64px); height: var(--lb-size, 64px);
      object-fit: contain; border-radius: 10px; background: rgba(0,0,0,0.03);
      box-shadow: 0 2px 10px rgba(0,0,0,0.12); cursor: pointer; user-select: none;
    }
    #lb-tmdb-main-icon:active { transform: translateY(1px); }

    /* Right column that matches the icon's full height */
    #lb-side-buttons {
      display: flex; flex-direction: column; gap: 8px;
      height: var(--lb-size, 64px);
      width: 36px;
    }

    #lb-settings-btn, #lb-info-btn {
      flex: 1; width: 36px; display: grid; place-items: center;
      background: #444; color: #fff; border: none; border-radius: 10px; cursor: pointer;
      box-shadow: 0 2px 10px rgba(0,0,0,0.2); padding: 0;
    }
    #lb-settings-btn:active, #lb-info-btn:active { transform: translateY(1px); }

    #lb-settings-btn svg, #lb-info-btn svg { display:block }

    @media (prefers-color-scheme: dark) {
      #lb-tmdb-main-icon { background: rgba(255,255,255,0.06); }
      #lb-settings-btn, #lb-info-btn { background: #333; }
    }
  `);

  const topRightBar = document.createElement('div');
  Object.assign(topRightBar.style, { position: 'fixed', top: '10px', right: '10px', display: 'flex', gap: '8px', zIndex: 2147483647 });
  document.body.append(topRightBar);

  const iconWrap = document.createElement('div');
  iconWrap.id = 'lb-tmdb-icon-wrap';
  iconWrap.style.setProperty('--lb-size', `${gmGetOrSet('lb_icon_size', 64)}px`);

  const mainIcon = document.createElement('img');
  mainIcon.id = 'lb-tmdb-main-icon';
  mainIcon.alt = 'FLAM Launcher';
  mainIcon.src = '';

  // Right-side vertical buttons column
  const sideButtons = document.createElement('div');
  sideButtons.id = 'lb-side-buttons';

  const settingsBtn = document.createElement('button');
  settingsBtn.id = 'lb-settings-btn';
  settingsBtn.title = 'Settings';
  settingsBtn.innerHTML = `
    <svg xmlns="http://www.w3.org/2000/svg" width="18" height="18" viewBox="0 0 24 24" fill="none">
      <path d="M12 15.5a3.5 3.5 0 1 0 0-7 3.5 3.5 0 0 0 0 7Z" fill="#fff"/>
      <path d="M20 13.1v-2.2l-2.02-.62a6.93 6.93 0 0 0-.52-1.25l1.1-1.84-1.56-1.56-1.84 1.1c-.4-.2-.82-.37-1.25-.52L13.1 4h-2.2l-.61 2.02c-.43.14-.85.31-1.25.52l-1.84-1.1L5.64 7 6.74 8.84c-.2.4-.37.82-.52 1.25L4 10.9v2.2l2.02.62c.14.43.31.85.52 1.25L5.43 16.8l1.56 1.56 1.84-1.1c.4.2.82.37 1.25.52l.62 2.02h2.2l.62-2.02c.43-.14.85-.31 1.25-.52l1.84 1.1 1.56-1.56-1.1-1.84c.2-.4.37-.82.52-1.25L20 13.1Z" fill="#fff"/>
    </svg>
  `;

  // NEW: Info button (simple white "i" silhouette) — toggled by setting
  const infoBtn = document.createElement('button');
  infoBtn.id = 'lb-info-btn';
  infoBtn.title = 'Last Send Info';
  infoBtn.innerHTML = `
    <svg xmlns="http://www.w3.org/2000/svg" width="18" height="18" viewBox="0 0 24 24" fill="none" aria-hidden="true">
      <path d="M12 22c5.523 0 10-4.477 10-10S17.523 2 12 2 2 6.477 2 12s4.477 10 10 10Z" fill="#fff" opacity=".12"/>
      <path d="M12 6.75a1.25 1.25 0 1 1 0 2.5 1.25 1.25 0 0 1 0-2.5Zm-1.25 4.25a1 1 0 0 0-1 1v5.25a1 1 0 1 0 2 0V12h1.25a1 1 0 1 0 0-2H10.75Z" fill="#fff"/>
      <circle cx="12" cy="12" r="9" stroke="#fff" stroke-width="1.5" opacity=".85"/>
    </svg>
  `;

  sideButtons.append(settingsBtn, infoBtn);
  iconWrap.append(mainIcon, sideButtons);
  topRightBar.append(iconWrap);

  // Icon DB + Picker (same as LB)
  const ICONS_API_URL = 'https://api.github.com/repos/hcgiub001/letterboxd-tmdb-cache/contents/addon_icons';
  const ICON_DB_KEY = 'lb_tmdb_icon_db_v1';
  const ICON_SELECTED_NAME_KEY = 'lb_tmdb_icon_selected_name';
  const ICON_SIZE_KEY = 'lb_icon_size';

  function gmFetchJSON(url) {
    return new Promise((resolve, reject) => {
      GM_xmlhttpRequest({
        method: 'GET', url,
        headers: { 'Accept': 'application/vnd.github+json' },
        onload: (res) => {
          if (res.status >= 200 && res.status < 300) {
            try { resolve(JSON.parse(res.responseText)); } catch (e) { reject(e); }
          } else reject(new Error(`HTTP ${res.status}`));
        },
        onerror: reject
      });
    });
  }
  function gmFetchArrayBuffer2(url) {
    return new Promise((resolve, reject) => {
      GM_xmlhttpRequest({
        method: 'GET', url, responseType: 'arraybuffer',
        onload: (res) => {
          if (res.status >= 200 && res.status < 300) resolve(res.response);
          else reject(new Error(`HTTP ${res.status}`));
        },
        onerror: reject
      });
    });
  }
  function arrayBufferToDataURL(buf, mime = 'image/png') {
    const bytes = new Uint8Array(buf);
    const chunk = 0x8000;
    let binary = '';
    for (let i = 0; i < bytes.length; i += chunk) {
      const sub = bytes.subarray(i, i + chunk);
      binary += String.fromCharCode.apply(null, sub);
    }
    return `data:${mime};base64,` + btoa(binary);
  }
  async function buildIconDBFromGitHub() {
    const list = await gmFetchJSON(ICONS_API_URL);
    const files = (list || [])
      .filter(x => x && x.type === 'file' && /\.png$/i.test(x.name) && x.download_url)
      .sort((a, b) => a.name.localeCompare(b.name));
    if (!files.length) throw new Error('No PNG files found in addon_icons.');
    const downloads = await Promise.all(files.map(async (f) => {
      const buf = await gmFetchArrayBuffer2(f.download_url);
      const dataUrl = arrayBufferToDataURL(buf, 'image/png');
      return { name: f.name, dataUrl };
    }));
    const newDb = { icons: downloads, lastSync: new Date().toISOString() };
    GM_setValue(ICON_DB_KEY, newDb);
    return newDb;
  }

  let ICON_DB = GM_getValue(ICON_DB_KEY, null);
  let ICON_SELECTED_NAME = gmGetOrSet(ICON_SELECTED_NAME_KEY, '');
  let ICON_SIZE = parseInt(gmGetOrSet(ICON_SIZE_KEY, 64), 10) || 64;

  async function ensureIconDB() {
    if (!ICON_DB || !Array.isArray(ICON_DB.icons) || !ICON_DB.icons.length) {
      try { ICON_DB = await buildIconDBFromGitHub(); }
      catch (e) {
        console.error('[Icon DB] Initial build failed:', e);
        ICON_DB = { icons: [{ name: 'fallback.png', dataUrl: '' }], lastSync: new Date().toISOString() };
        GM_setValue(ICON_DB_KEY, ICON_DB);
      }
    }
  }
  function setIconSize(px) {
    ICON_SIZE = Math.max(16, Math.min(512, +px || 64));
    iconWrap.style.setProperty('--lb-size', `${ICON_SIZE}px`);
    GM_setValue(ICON_SIZE_KEY, ICON_SIZE);
  }
  function setSelectedIconByName(name) {
    ICON_SELECTED_NAME = name || '';
    GM_setValue(ICON_SELECTED_NAME_KEY, ICON_SELECTED_NAME);
    const entry = ICON_DB?.icons?.find(i => i.name === ICON_SELECTED_NAME) || ICON_DB?.icons?.[0];
    if (entry) mainIcon.src = entry.dataUrl;
  }
  await ensureIconDB();
  if (!ICON_SELECTED_NAME && ICON_DB?.icons?.length) {
    ICON_SELECTED_NAME = ICON_DB.icons[0].name;
    GM_setValue(ICON_SELECTED_NAME_KEY, ICON_SELECTED_NAME);
  }
  setIconSize(ICON_SIZE);
  setSelectedIconByName(ICON_SELECTED_NAME);

  // NEW: Info button visibility control
  function updateInfoButtonVisibility() {
    const show = gmGetOrSet(INFO_BTN_ENABLE_KEY, true);
    infoBtn.style.display = show ? 'grid' : 'none';
    // If hidden, let settings fill full height (already flex:1) — done via CSS layout.
  }
  updateInfoButtonVisibility();

  // ────────────────────────────────────────────────────────────────────────
  // Run control
  // ────────────────────────────────────────────────────────────────────────
  async function startRun() {
    await ensureBaseLoadedOnce();
    await overridesLoadOnce();
    await negativeLoadOnce();
    setTimeout(() => { maybeRevalidateBaseMonthly(); }, 0);
  }

  async function resolveActionForThisRun() {
    const stored = GM_getValue('kodiAction', 'ask');
    if (stored === 'ask') return await askForActionOverlay();
    return stored;
  }

  // ────────────────────────────────────────────────────────────────────────
  // Show URL (Tools) — LB-style panel and bytes/savings readout
  // ────────────────────────────────────────────────────────────────────────
  async function handleShowUrl(btnEl) {
    if (btnEl) btnEl.disabled = true;

    await startRun();

    let items = await gatherItems();
    if (!items || !items.length) { alert('No items'); if (btnEl) btnEl.disabled = false; return; }
    if (items.length > MAX_LIST_ITEMS) { showTooManyItemsOverlay(items.length); if (btnEl) btnEl.disabled = false; return; }

    let how = prompt('How many items? number or "all":', 'all');
    if (how === null) { if (btnEl) btnEl.disabled = false; return; }
    how = how.trim().toLowerCase();
    if (how !== 'all') {
      const n = parseInt(how, 10);
      if (isNaN(n) || n < 1) { alert('Invalid number'); if (btnEl) btnEl.disabled = false; return; }
      items = items.slice(0, n);
    }

    const { resolved, counts } = await processAll(items);

    const listName = getListName();
    const author = getAuthor();

    // Description (same behavior as Letterboxd = optional edit)
    let descriptionText = '';
    if (gmGetOrSet('descEnable', true)) {
      const pageDesc = getDescriptionText() || DEFAULT_DESCRIPTION;
      if (gmGetOrSet('descMode', 'send') === 'edit') {
        const edited = await editDescriptionOverlay(pageDesc);
        if (edited === null) { if (btnEl) btnEl.disabled = false; return; }
        descriptionText = sanitizeDescPreservingBreaks(edited);
      } else {
        descriptionText = pageDesc;
      }
    }

    // Action (ASK matches LB)
    let action = GM_getValue('kodiAction', 'ask');
    if (action === 'ask') {
      const chosen = await askForActionOverlay();
      if (!chosen && chosen !== '') { if (btnEl) btnEl.disabled = false; return; } // cancel
      action = chosen || '';
    }

    // Artwork options (same keys & behavior as LB)
    const posterEnabled = gmGetOrSet('posterEnable', POSTER_ENABLE_DEFAULT);
    const posterStrategy = gmGetOrSet('posterStrategy', POSTER_STRATEGY_DEFAULT);
    const fanartEnabled = gmGetOrSet('fanartEnable', FANART_ENABLE_DEFAULT);
    const fanartStrategy = gmGetOrSet('fanartStrategy', FANART_STRATEGY_DEFAULT);
    const fanartFallback = gmGetOrSet('fanartFallback', FANART_FALLBACK_DEFAULT);

    const fen = await buildFenUrlTMDB_Art(resolved, {
      listName, author, description: descriptionText, action,
      posterEnabled, posterStrategy,
      fanartEnabled, fanartStrategy, fanartFallback
    });

    const pct = Math.min(100, Math.round((fen.urlBytes / URL_LIMIT_BYTES) * 1000) / 10);
    const savings = (typeof fen.gzBytes === 'number') ? (fen.rawBytes - fen.gzBytes) : 0;

    const existing = document.getElementById('kodishowurl'); if (existing) existing.remove();
    const overlay = document.createElement('div');
    overlay.id = 'kodishowurl';
    Object.assign(overlay.style, {
      position:'fixed', top:0, left:0, width:'100vw', height:'100vh',
      background:'rgba(0,0,0,0.7)', display:'flex', alignItems:'center', justifyContent:'center',
      zIndex:2147483647, padding:'20px', boxSizing:'border-box'
    });
    const panel = document.createElement('div');
    Object.assign(panel.style, { background:'#222', padding:'20px', borderRadius:'6px', maxWidth:'95vw' });
    panel.innerHTML = `
      <h2 style="margin:0 0 12px; color:#fff; font-size:18px;">URL (${fen.gzUrl ? 'gzip+base64' : 'raw JSON'})</h2>
      <textarea id="kodishowurl_text" style="width:600px; max-width:95vw; height:200px; font-size:14px; box-sizing:border-box; white-space:pre-wrap;">${fen.url}</textarea>
      <div style="margin-top:10px; padding:8px; background:#1b1b1b; border:1px solid #333; border-radius:6px; color:#ddd; font-size:12px; line-height:1.5">
        <div><strong>Raw URL:</strong> ${fen.rawBytes} bytes</div>
        <div><strong>Gzip+Base64 URL:</strong> ${typeof fen.gzBytes === 'number' ? fen.gzBytes + ' bytes' : 'n/a (compression unavailable)'}</div>
        <div><strong>Savings:</strong> ${typeof fen.gzBytes === 'number' ? (savings + ' bytes (' + (fen.rawBytes ? Math.round((savings / fen.rawBytes) * 100) : 0) + '%)') : '—'}</div>
        <div><strong>Using:</strong> ${fen.gzUrl ? 'base64_items (gzip+base64)' : 'list_items (raw JSON)'}</div>
        <div><strong>Limit usage:</strong> ${fen.urlBytes} / ${URL_LIMIT_BYTES} (${pct}%)</div>
        <div><strong>Cache:</strong> base ${counts.base} • overrides ${counts.overrides} • cached_no_id ${counts.cachedNoId} • uncached ${counts.uncached} (Σ=${counts.base+counts.overrides+counts.cachedNoId+counts.uncached} / ${items.length})</div>
      </div>
      <div style="margin-top:12px; text-align:right;">
        <button id="kodishowurl_copy" style="margin-right:8px; padding:6px 12px; font-size:14px;">Copy URL</button>
        <button id="kodishowurl_close" style="padding:6px 12px; font-size:14px;">Close</button>
      </div>
    `;
    overlay.append(panel);
    document.body.append(overlay);
    panel.querySelector('#kodishowurl_copy').onclick = () => {
      const ta = panel.querySelector('#kodishowurl_text'); ta.select(); document.execCommand('copy');
      panel.querySelector('#kodishowurl_copy').textContent = 'Copied!';
    };
    panel.querySelector('#kodishowurl_close').onclick = () => overlay.remove();

    if (btnEl) btnEl.disabled = false;
  }

  // ────────────────────────────────────────────────────────────────────────
  // Main icon click → EARLY ASK + TRUST PREFLIGHT + full run (LB styling)
  // ────────────────────────────────────────────────────────────────────────
  let ICON_BUSY = false;
  mainIcon.addEventListener('click', async function () {
    if (ICON_BUSY) return;
    ICON_BUSY = true;

    const startTime = performance.now();

    // Trigger TM trust prompt immediately (harmless ping)
    preflightKodiPermission();

    showOrUpdateProcessingBubble(mainIcon, ['Processing…'], true);

    await startRun();

    // Show ASK immediately (if configured) while processing runs
    const stored = GM_getValue('kodiAction', 'ask');
    const actionPromise = (stored === 'ask') ? askForActionOverlay() : Promise.resolve(stored);

    // Process in background
    const processingPromise = (async () => {
      let items;
      try { items = await gatherItems(); }
      catch { alert('Scrape failed'); hideProcessingBubble(true); ICON_BUSY = false; return null; }
      if (!items || !items.length) { alert('No items'); hideProcessingBubble(true); ICON_BUSY = false; return null; }

      if (items.length > MAX_LIST_ITEMS) {
        showTooManyItemsOverlay(items.length);
        hideProcessingBubble(true);
        ICON_BUSY = false;
        return null;
      }

      // Pre-cache stats
      const overrides0 = (await idbGet(OVERRIDES_STORE, OVERRIDES_KEY)) || {};
      let baseHitCount = 0, overrideHitCount = 0, uncachedCount = 0;
      for (const it of items) {
        const fid = String(it.imdbId || '');
        const inOverrides = Number.isFinite(overrides0[fid]);
        if (inOverrides) { overrideHitCount++; continue; }
        const m = fid.match(/^tt(\d+)$/);
        const basePacked = m ? baseLookupPackedByNumericImdb(Number(m[1])) : undefined;
        if (basePacked !== undefined) baseHitCount++;
        else uncachedCount++;
      }
      const totalPlanned = baseHitCount + overrideHitCount + uncachedCount;

      showOrUpdateProcessingBubble(mainIcon, [
        'Processing…',
        `Items selected: ${items.length}`,
        `Cache — base: ${baseHitCount} • overrides: ${overrideHitCount} • uncached: ${uncachedCount}`,
        `Sum check: ${totalPlanned} / ${items.length}`
      ], true);

      const { resolved, counts } = await processAll(items);

      // Description (same as LB)
      let descriptionText = '';
      if (gmGetOrSet('descEnable', true)) {
        const pageDesc = getDescriptionText() || DEFAULT_DESCRIPTION;
        if (gmGetOrSet('descMode', 'send') === 'edit') {
          const edited = await editDescriptionOverlay(pageDesc);
          if (edited === null) { hideProcessingBubble(true); ICON_BUSY = false; return null; }
          descriptionText = sanitizeDescPreservingBreaks(edited);
        } else {
          descriptionText = pageDesc;
        }
      }

      const listName = getListName();
      const author = getAuthor();

      return { resolved, counts, listName, author, descriptionText, totalItems: items.length };
    })();

    const action = await actionPromise;
    const proc = await processingPromise;
    if (!proc) { ICON_BUSY = false; return; }

    if (stored === 'ask' && !action) { hideProcessingBubble(true); ICON_BUSY = false; return; } // canceled

    // Artwork options pulled from settings (LB parity)
    const posterEnabled = gmGetOrSet('posterEnable', POSTER_ENABLE_DEFAULT);
    const posterStrategy = gmGetOrSet('posterStrategy', POSTER_STRATEGY_DEFAULT);
    const fanartEnabled = gmGetOrSet('fanartEnable', FANART_ENABLE_DEFAULT);
    const fanartStrategy = gmGetOrSet('fanartStrategy', FANART_STRATEGY_DEFAULT);
    const fanartFallback = gmGetOrSet('fanartFallback', FANART_FALLBACK_DEFAULT);

    showOrUpdateProcessingBubble(mainIcon, [
      'Building URL…',
      (String(action).toLowerCase() === 'view') ? `Artwork disabled for 'view'` :
      (posterEnabled ? `Poster: ${posterStrategy}` : `Poster: disabled`),
      (String(action).toLowerCase() === 'view') ? `` :
      (fanartEnabled ? `Fanart: ${fanartStrategy}${fanartStrategy==='author_fanart' ? ` (fallback: ${fanartFallback})` : ''}` : `Fanart: disabled`)
    ].filter(Boolean), true);

    const fen = await buildFenUrlTMDB_Art(proc.resolved, {
      listName: proc.listName,
      author: proc.author,
      description: proc.descriptionText,
      action: action || '',
      posterEnabled, posterStrategy,
      fanartEnabled, fanartStrategy, fanartFallback
    });

    showOrUpdateProcessingBubble(mainIcon, [
      'Sending to Kodi…',
      `URL bytes: ${fen.urlBytes} ${fen.gzUrl ? '(gzip+base64)' : '(raw JSON)'}`
    ], true);

    const ok = await sendToKodi(fen.url);
    const elapsedMs = Math.max(0, Math.round(performance.now() - startTime));

    hideProcessingBubble(true);

    const pct = Math.min(100, Math.round((fen.urlBytes / URL_LIMIT_BYTES) * 1000) / 10);
    const lines = [
      `Items sent: ${proc.resolved.length}`,
      `📦 Cache — base: ${proc.counts.base} • overrides: ${proc.counts.overrides} • cached_no_id: ${proc.counts.cachedNoId} • uncached: ${proc.counts.uncached}`,
      `Σ = ${proc.counts.base + proc.counts.overrides + proc.counts.cachedNoId + proc.counts.uncached} / ${proc.totalItems}`,
      `📨 Bytes: ${fen.urlBytes} / ${URL_LIMIT_BYTES} (${pct}%) ${fen.gzUrl ? '[gzip+base64]' : '[raw JSON]'}`,
      `⏱ Elapsed: ${(elapsedMs/1000).toFixed(1)}s`,
      ok ? '✅ Sent' : '❌ Failed'
    ];
    showSideInfoNearEl(mainIcon, lines, 7000);

    // NEW: persist last send stats for Info button
    try {
      GM_setValue(LAST_STATS_KEY, { lines, when: Date.now() });
    } catch {}

    ICON_BUSY = false;
  });

  // NEW: Info button handler — show last saved stats
  infoBtn.addEventListener('click', () => {
    const last = gmGetOrSet(LAST_STATS_KEY, null);
    if (!last || !Array.isArray(last.lines)) {
      showSideInfoNearEl(mainIcon, ['No previous send stats found.', 'Send a list to populate this.'], 5000);
      return;
    }
    const when = new Date(last.when || Date.now());
    const whenStr = `${when.toLocaleDateString()} ${when.toLocaleTimeString()}`;
    const lines = [...last.lines, `🕒 Last sent: ${whenStr}`];
    showSideInfoNearEl(mainIcon, lines, 8000);
  });

  // ────────────────────────────────────────────────────────────────────────
  // Settings (Letterboxd-style): General + Tools tabs (kept, plus info toggle)
  // ────────────────────────────────────────────────────────────────────────
  settingsBtn.addEventListener('click', showSettings);

  function showSettings() {
    if (document.getElementById('kodisettings')) return;

    const overlay = document.createElement('div');
    overlay.id = 'kodisettings';
    Object.assign(overlay.style, {
      position:'fixed', top:0, left:0, width:'100vw', height:'100vh',
      background:'rgba(0,0,0,0.7)', display:'flex', alignItems:'center',
      justifyContent:'center', zIndex:2147483647, padding:'20px', boxSizing:'border-box'
    });

    const panel = document.createElement('div');

    panel.innerHTML = `
      <div style="display:flex; gap:8px; align-items:center; justify-content:space-between; margin-bottom:8px;">
        <h2 style="color:#fff;margin:0">Kodi Settings</h2>
        <div style="display:flex; gap:6px;">
          <button class="kodi-tab-btn" data-tab="general" style="background:#e50914;color:#fff;border:none;border-radius:4px;padding:6px 10px;cursor:pointer">General</button>
          <button class="kodi-tab-btn" data-tab="tools" style="background:#444;color:#fff;border:none;border-radius:4px;padding:6px 10px;cursor:pointer">Tools</button>
        </div>
      </div>

      <div id="kodi-tab-general" class="kodi-tab" style="display:block">

        <h3 style="color:#fff;margin:10px 0 6px;">Launcher Icon</h3>
        <div style="background:#1b1b1b;border:1px solid #333;border-radius:10px;padding:10px;margin-bottom:12px;color:#ddd">
          <div style="display:flex;align-items:center;justify-content:space-between;gap:8px;margin-bottom:8px">
            <div style="display:flex;align-items:center;gap:10px">
              <img id="iconPreview" alt="Icon preview" style="width:48px;height:48px;border-radius:10px;border:1px solid #2a2a2a;object-fit:contain; background:#111"/>
              <div style="font-size:12px;line-height:1.4">
                <div><strong>Selected:</strong> <span id="iconSelectedName">—</span></div>
                <div id="iconMeta" style="opacity:.8">Cached icons: — • Last refresh: —</div>
              </div>
            </div>
            <button id="refreshIconsBtn" style="font-size:12px;padding:6px 10px;border-radius:8px;background:#000;color:#fff;border:none;cursor:pointer">Refresh from GitHub</button>
          </div>

          <div id="iconGrid" style="display:grid;grid-template-columns:repeat(auto-fill,minmax(48px,1fr));gap:8px;max-height:220px;overflow:auto;padding-right:2px"></div>

          <div style="display:grid;grid-template-columns:1fr auto;gap:8px;align-items:center;margin-top:10px">
            <div>
              <label style="font-size:13px">Length (px):</label>
              <input id="iconSizeRange" type="range" min="16" max="512" step="1">
            </div>
            <input id="iconSizeNumber" type="number" min="16" max="512" step="1" style="width:90px;padding:6px 8px;border-radius:8px;border:1px solid #444;background:#111;color:#eee">
          </div>
        </div>

        <div style="background:#1b1b1b;border:1px solid #333;border-radius:10px;padding:10px;margin-bottom:12px;color:#ddd">
          <label style="display:flex;align-items:center;gap:8px;">
            <input type="checkbox" id="showInfoBtn">
            <span>Show Info Button (shows last send stats)</span>
          </label>
          <div style="font-size:12px;color:#bbb;margin-top:6px;">When enabled, an Info button appears below the settings button. Clicking it shows the last stats bubble from your most recent send.</div>
        </div>

        <label style="color:#fff">Kodi IP:</label>
        <input id="kodiIp" style="width:100%;margin-bottom:8px"/>

        <label style="color:#fff">Kodi Port:</label>
        <input id="kodiPort" style="width:100%;margin-bottom:8px"/>

        <label style="color:#fff">Kodi User:</label>
        <input id="kodiUser" style="width:100%;margin-bottom:8px"/>

        <label style="color:#fff">Kodi Pass:</label>
        <input id="kodiPass" type="password" style="width:100%;margin-bottom:12px"/>

        <label style="color:#fff">Default Action:</label>
        <select id="kodiAction" style="width:100%;margin-bottom:16px">
          <option value="">Omit Action key</option>
          <option value="view">view</option>
          <option value="import">import</option>
          <option value="import_view">import_view</option>
          <option value="ask">Ask each time</option>
        </select>

        <label style="color:#fff">Busy indicator:</label>
        <select id="indicator" style="width:100%;margin-bottom:16px">
          <option value="none">none</option>
          <option value="busy">busy</option>
          <option value="progress">progress</option>
        </select>

        <label style="color:#fff"><input type="checkbox" id="descEnable"/> Add Description</label>
        <div id="descOptions" style="margin:8px 0;display:none;color:#fff">
          <label>Edit mode:</label>
          <select id="descMode" style="width:100%;margin-bottom:8px">
            <option value="send">Send without editing</option>
            <option value="edit">Edit before sending</option>
          </select>
        </div>

        <hr style="border-color:#444;margin:14px 0">

        <div>
          <h3 style="color:#fff;margin:0 0 8px;">Artwork Options</h3>
          <div style="display:grid;grid-template-columns:1fr 1fr;gap:10px;margin-bottom:10px">
            <div style="border:1px solid #333;border-radius:6px;padding:10px;">
              <label style="color:#fff"><input type="checkbox" id="posterEnable"/> Enable Poster</label>
              <div style="color:#bbb;font-size:12px;margin:6px 0 8px">Include the <code>poster</code> key in the URL.<br>(Ignored when action=<code>view</code>)</div>
              <label style="color:#fff">Poster selection:</label>
              <select id="posterStrategy" style="width:100%;margin-top:4px">
                <option value="first_4">first_4</option>
                <option value="random">random</option>
              </select>
            </div>
            <div style="border:1px solid #333;border-radius:6px;padding:10px;">
              <label style="color:#fff"><input type="checkbox" id="fanartEnable"/> Enable Fanart</label>
              <div style="color:#bbb;font-size:12px;margin:6px 0 8px">The <code>fanart</code> key accepts either a strategy or a direct URL.<br>(Ignored when action=<code>view</code>)</div>
              <label style="color:#fff">Fanart selection:</label>
              <select id="fanartStrategy" style="width:100%;margin-top:4px">
                <option value="author_fanart">author_fanart (use page og:image)</option>
                <option value="first_4">first_4</option>
                <option value="random">random</option>
              </select>
              <div id="fanartFallbackBox" style="margin-top:8px; display:none;">
                <label style="color:#fff">If author_fanart missing, fallback to:</label>
                <select id="fanartFallback" style="width:100%;margin-top:4px">
                  <option value="none">none</option>
                  <option value="first_4">first_4</option>
                  <option value="random">random</option>
                </select>
              </div>
              <div style="color:#888; font-size:12px; margin-top:6px;">
                On IMDb, “author fanart” uses the page’s OpenGraph image if present; otherwise the chosen fallback strategy.
              </div>
            </div>
          </div>
        </div>

        <hr style="border-color:#444;margin:14px 0">

        <div>
          <h3 style="color:#fff;margin:0 0 8px;">Cache Tools</h3>
          <div style="display:flex;gap:8px;flex-wrap:wrap;margin-bottom:6px">
            <button id="refreshBaseCacheBtn">Refresh Base Cache</button>
          </div>
          <div id="cacheToolMsg" style="color:#bbb;font-size:12px;margin-top:8px;"></div>
        </div>
      </div>

      <div id="kodi-tab-tools" class="kodi-tab" style="display:none">
        <div style="display:flex; flex-wrap:wrap; gap:8px; margin-bottom:12px;">
          <button id="btnShowUrl" title="Build & view plugin URL">Show URL</button>
          <button id="btnClearOverrides" title="Clear overrides">Clear Overrides</button>
        </div>
        <div style="color:#bbb; font-size:12px;">
          Use the bottom-right icon to send. Tools let you preview the URL or clear positive overrides.
        </div>
      </div>

      <div style="text-align:right;margin-top:12px; position:sticky; bottom:0; background:#222; padding-top:8px;">
        <button id="kodisave" style="margin-right:8px">Save</button>
        <button id="kodicancel">Close</button>
      </div>
    `;

    Object.assign(panel.style, {
      background: '#222', padding: '16px', borderRadius: '8px',
      width: '720px', maxWidth: '95vw', maxHeight: '90vh',
      overflowY: 'auto', boxSizing: 'border-box', color:'#fff'
    });

    overlay.append(panel);
    document.body.append(overlay);

    const tabBtns = panel.querySelectorAll('.kodi-tab-btn');
    function activateTab(name) {
      panel.querySelectorAll('.kodi-tab').forEach(el => el.style.display = 'none');
      const btns = panel.querySelectorAll('.kodi-tab-btn');
      btns.forEach(b => { b.style.background = (b.dataset.tab === name) ? '#e50914' : '#444'; });
      const shown = panel.querySelector(`#kodi-tab-${name}`);
      if (shown) shown.style.display = 'block';
    }
    tabBtns.forEach(b => b.addEventListener('click', () => activateTab(b.dataset.tab)));
    activateTab('general');

    // hydrate
    panel.querySelector('#iconPreview').src = mainIcon.src;

    panel.querySelector('#kodiIp').value = GM_getValue('kodiIp','');
    panel.querySelector('#kodiPort').value = GM_getValue('kodiPort','');
    panel.querySelector('#kodiUser').value = GM_getValue('kodiUser','');
    panel.querySelector('#kodiPass').value = gmGetOrSet('kodiPass','');
    panel.querySelector('#kodiAction').value = GM_getValue('kodiAction','ask');
    panel.querySelector('#indicator').value = gmGetOrSet('indicator', DEFAULT_INDICATOR);

    panel.querySelector('#descEnable').checked = gmGetOrSet('descEnable', true);
    panel.querySelector('#descMode').value = gmGetOrSet('descMode', 'send');
    const descOpts = panel.querySelector('#descOptions');
    panel.querySelector('#descEnable').addEventListener('change', e => {
      descOpts.style.display = e.target.checked ? 'block' : 'none';
    });
    if (panel.querySelector('#descEnable').checked) descOpts.style.display = 'block';

    // Info toggle
    panel.querySelector('#showInfoBtn').checked = gmGetOrSet(INFO_BTN_ENABLE_KEY, true);

    panel.querySelector('#posterEnable').checked = gmGetOrSet('posterEnable', POSTER_ENABLE_DEFAULT);
    panel.querySelector('#posterStrategy').value = gmGetOrSet('posterStrategy', POSTER_STRATEGY_DEFAULT);
    panel.querySelector('#fanartEnable').checked = gmGetOrSet('fanartEnable', FANART_ENABLE_DEFAULT);
    panel.querySelector('#fanartStrategy').value = gmGetOrSet('fanartStrategy', FANART_STRATEGY_DEFAULT);
    panel.querySelector('#fanartFallback').value = gmGetOrSet('fanartFallback', FANART_FALLBACK_DEFAULT);
    function refreshFanartFallbackVisibility() {
      const strat = panel.querySelector('#fanartStrategy').value;
      panel.querySelector('#fanartFallbackBox').style.display = (strat === 'author_fanart') ? 'block' : 'none';
    }
    panel.querySelector('#fanartStrategy').addEventListener('change', refreshFanartFallbackVisibility);
    refreshFanartFallbackVisibility();

    // Icon controls
    const iconNameEl = panel.querySelector('#iconSelectedName');
    const iconMeta = panel.querySelector('#iconMeta');
    const iconGrid = panel.querySelector('#iconGrid');
    const sizeRange = panel.querySelector('#iconSizeRange');
    const sizeNumber = panel.querySelector('#iconSizeNumber');
    const refreshIconsBtn = panel.querySelector('#refreshIconsBtn');

    function updateIconMeta() {
      const count = ICON_DB?.icons?.length || 0;
      const when = ICON_DB?.lastSync ? new Date(ICON_DB.lastSync).toLocaleString() : 'never';
      iconMeta.textContent = `Cached icons: ${count} • Last refresh: ${when}`;
    }
    function populateIconGrid() {
      iconGrid.innerHTML = '';
      if (!ICON_DB?.icons?.length) {
        iconGrid.textContent = 'No cached icons. Click “Refresh from GitHub”.';
        return;
      }
      for (const it of ICON_DB.icons) {
        const img = document.createElement('img');
        img.src = it.dataUrl;
        img.title = it.name;
        img.alt = it.name;
        Object.assign(img.style, {
          width: '100%', aspectRatio: '1 / 1', objectFit: 'contain',
          background: '#111', border: '1px solid #2a2a2a', borderRadius: '10px', cursor: 'pointer'
        });
        if (it.name === ICON_SELECTED_NAME) {
          img.style.outline = '2px solid #fff';
          img.style.outlineOffset = '2px';
        }
        img.addEventListener('click', () => {
          ICON_SELECTED_NAME = it.name;
          setSelectedIconByName(ICON_SELECTED_NAME);
          iconNameEl.textContent = ICON_SELECTED_NAME;
          iconGrid.querySelectorAll('img').forEach(g => { g.style.outline = 'none'; g.style.outlineOffset = '0'; });
          img.style.outline = '2px solid #fff';
          img.style.outlineOffset = '2px';
          panel.querySelector('#iconPreview').src = mainIcon.src;
        });
        iconGrid.appendChild(img);
      }
    }
    function syncSizeInputs() {
      sizeRange.value = String(ICON_SIZE);
      sizeNumber.value = String(ICON_SIZE);
    }

    iconNameEl.textContent = ICON_SELECTED_NAME || '—';
    updateIconMeta();
    populateIconGrid();
    syncSizeInputs();

    sizeRange.addEventListener('input', (e) => { setIconSize(e.target.value); syncSizeInputs(); });
    sizeNumber.addEventListener('input', (e) => { setIconSize(e.target.value); syncSizeInputs(); });

    refreshIconsBtn.addEventListener('click', async () => {
      const prev = refreshIconsBtn.textContent;
      refreshIconsBtn.disabled = true; refreshIconsBtn.textContent = 'Refreshing…';
      try {
        ICON_DB = await buildIconDBFromGitHub();
        if (!ICON_DB.icons.find(i => i.name === ICON_SELECTED_NAME)) {
          ICON_SELECTED_NAME = ICON_DB.icons[0].name;
          GM_setValue(ICON_SELECTED_NAME_KEY, ICON_SELECTED_NAME);
        }
        setSelectedIconByName(ICON_SELECTED_NAME);
        panel.querySelector('#iconPreview').src = mainIcon.src;
        iconNameEl.textContent = ICON_SELECTED_NAME;
        updateIconMeta();
        populateIconGrid();
        refreshIconsBtn.textContent = 'Refreshed ✓';
        await sleep(700);
      } catch (e) {
        console.error('[Icon DB] Refresh failed:', e);
        refreshIconsBtn.textContent = 'Refresh failed';
        await sleep(1200);
      } finally {
        refreshIconsBtn.textContent = prev;
        refreshIconsBtn.disabled = false;
      }
    });

    // Cache tools (base)
    const cacheMsg = panel.querySelector('#cacheToolMsg');
    const setCacheMsg = (txt, ok=false) => { cacheMsg.textContent = txt || ''; cacheMsg.style.color = ok ? '#8ee6a4' : '#bbb'; };
    panel.querySelector('#refreshBaseCacheBtn').onclick = async () => {
      setCacheMsg('Refreshing base cache…');
      const ok = await refreshBaseCacheNow(true);
      setCacheMsg(ok ? `Base cache ready (${BASE_PAIR_COUNT.toLocaleString()} pairs).` : 'Base cache refresh failed.', ok);
    };

    // Tools tab buttons
    panel.querySelector('#btnShowUrl').onclick = function () { handleShowUrl(this); };
    panel.querySelector('#btnClearOverrides').onclick = async () => { await overridesClear(); alert('Overrides cleared'); };

    // Save/Close
    panel.querySelector('#kodisave').onclick = () => {
      GM_setValue('kodiIp', panel.querySelector('#kodiIp').value.trim());
      GM_setValue('kodiPort', panel.querySelector('#kodiPort').value.trim());
      GM_setValue('kodiUser', panel.querySelector('#kodiUser').value);
      GM_setValue('kodiPass', panel.querySelector('#kodiPass').value);
      GM_setValue('kodiAction', panel.querySelector('#kodiAction').value);
      GM_setValue('indicator', panel.querySelector('#indicator').value);

      GM_setValue('descEnable', panel.querySelector('#descEnable').checked);
      GM_setValue('descMode', panel.querySelector('#descMode').value);

      GM_setValue('posterEnable', panel.querySelector('#posterEnable').checked);
      GM_setValue('posterStrategy', panel.querySelector('#posterStrategy').value);
      GM_setValue('fanartEnable', panel.querySelector('#fanartEnable').checked);
      GM_setValue('fanartStrategy', panel.querySelector('#fanartStrategy').value);
      GM_setValue('fanartFallback', panel.querySelector('#fanartFallback').value);

      GM_setValue(INFO_BTN_ENABLE_KEY, panel.querySelector('#showInfoBtn').checked);
      updateInfoButtonVisibility();

      overlay.remove();
      alert('✅ Settings saved');
    };
    panel.querySelector('#kodicancel').onclick = () => overlay.remove();
  }

})();