Sleazy Fork is available in English.

indexxx

Adds useful information and tools to indexxx pages

// ==UserScript==
// @name        indexxx
// @author      peolic
// @version     3.15
// @description Adds useful information and tools to indexxx pages
// @icon        https://www.indexxx.com/apple-touch-icon.png
// @namespace   https://github.com/peolic
// @match       https://www.indexxx.com/*
// @grant       none
// @homepageURL https://gist.github.com/peolic/6aa2cef8fafa377cb5848a473c0e3b30
// ==/UserScript==

//@ts-check

/** @type {{ name: string; url: string; } | null} */
let currentModel = null;

function main() {
  // Model page
  if (/^\/m\/(.+)/.test(window.location.pathname)) {
    new ModelPage();
  } else if (/^\/websites\/(.+)\/sets\//.test(window.location.pathname)) {
    new WebsiteSetsPage();
  }
}

const NO_ALIAS = '[no alias]';

class ModelPage {

  constructor() {
    currentModel = {
      name: /** @type {HTMLHeadingElement} */ (document.querySelector('h1#model-name')).innerText,
      url: /** @type {HTMLLinkElement} */ (document.querySelector('link[rel="canonical"]')).href,
    };

    this.websitesBox = /** @type {HTMLDivElement} */ (document.querySelector('#model-websites-box'));
    this.modelHeader = /** @type {HTMLDivElement} */ (document.querySelector('#model-header'));
    this.portfolioHeader = /** @type {HTMLHeadingElement} */ (this.modelHeader.querySelector(':scope > .block1 > h2'));

    /** @type {IndexxxSet[]} */
    this.allSets =
      /** @type {HTMLDivElement[]} */
      (Array.from(document.querySelectorAll('.pset.card'))).map(parseSetCard);

    this.uniqueSets = this.allSets
      .filter((set, i, self) => {
        const { outUrl, display } = set.site;
        const alias = set.models[0].name;
        return i === self.findIndex((other) =>
          (outUrl ? outUrl === other.site.outUrl : display === other.site.display)
          && alias === other.models[0].name
        );
      });

    this.aliases = this.aliasesUsageCount();

    const boxAliases = this.aliasUsageCountBox();
    const boxSites = this.sitesForAliasBox();
    this.websitesBox.after(boxAliases, boxSites);

    tryResizable(boxAliases, { handles: 'e', minWidth: 130 });
    tryResizable(boxSites, { handles: 'e', minWidth: 204 });

    this.setUpToggle();
    this.exportTarget = this.setUpExport();
  }

  style = {
    box: {
      borderColor: '#006ccc',
      backgroundColor: '#8bbeff',
      width: '204px',
    },
    title: {
      backgroundColor: '#006ccc',
    },
    body: {
      backgroundColor: '#e5eff9',
      wordBreak: 'break-word',
      padding: '0.1rem',
    },
    csv: {
      userSelect: 'all',
      overflow: 'hidden',
      textOverflow: 'ellipsis',
      whiteSpace: 'nowrap',
      fontSize: '10px',
    },
  }

  // Alias Usage Count
  aliasUsageCountBox() {
    const box = document.createElement('div');
    box.classList.add('model-snippet-box');
    setStyles(box, this.style.box);
    const boxTitle = document.createElement('div');
    boxTitle.classList.add('box-title', 'd-flex', 'justify-content-between', 'px-1');
    boxTitle.innerText = 'Alias Usage Count';
    setStyles(boxTitle, this.style.title);
    box.appendChild(boxTitle);
    const titleAliasCount = document.createElement('span');
    boxTitle.appendChild(titleAliasCount);
    const boxBody = document.createElement('div');
    boxBody.classList.add('box-body');
    setStyles(boxBody, this.style.body);
    this.aliases
      .sort(({ count: a }, { count: b }) => {
        if (a < b) return 1;
        if (a > b) return -1;
        return 0;
      })
      .forEach(({ alias, count, listed }) => {
        const div = document.createElement('div');
        const searchSpan = document.createElement('span');
        setStyles(searchSpan, { marginRight: '.25rem', cursor: 'pointer', userSelect: 'none' });
        searchSpan.innerText = '🔎';
        searchSpan.addEventListener('click', () => {
          const input = /** @type {HTMLInputElement} */ (document.querySelector('input#aliasSitesInput'));
          input.value = alias;
          input.dispatchEvent(new Event('input'));
        });
        const aliasSpan = document.createElement('span');
        setStyles(aliasSpan, {
          fontWeight: alias === NO_ALIAS ? 'bold' : undefined,
          userSelect: 'all',
        });
        if (!listed) {
          aliasSpan.classList.add('text-danger');
          aliasSpan.title = `"${alias}" is not listed as an alias.`;
        }
        aliasSpan.innerText = alias;
        div.append(searchSpan, aliasSpan, ` = ${count >= 0 ? count : '?'}`);
        boxBody.appendChild(div);
      });
    box.appendChild(boxBody);
    const csv = document.createElement('div');
    setStyles(csv, this.style.csv);
    const validAliases = this.aliases
      .reduce(
        (r, { alias }) => alias !== NO_ALIAS ? r.concat(alias) : r,
        /** @type {string[]} */ ([])
      );
    csv.innerText = validAliases.sort().join(', ');
    titleAliasCount.innerText = `${validAliases.length}`;
    box.appendChild(csv);
    return box;
  }

  // Sites For Alias
  sitesForAliasBox() {
    const box = document.createElement('div');
    box.classList.add('model-snippet-box');
    setStyles(box, this.style.box);
    const boxTitle = document.createElement('div');
    boxTitle.classList.add('box-title', 'd-flex', 'justify-content-between', 'px-1');
    setStyles(boxTitle, this.style.title);
    box.appendChild(boxTitle);
    const boxDefaultTitle = 'Websites By Alias';
    const boxTitleNode = document.createTextNode(boxDefaultTitle);
    boxTitle.appendChild(boxTitleNode);
    const titleSetCount = document.createElement('span');
    boxTitle.appendChild(titleSetCount);
    const boxBody = document.createElement('div');
    boxBody.classList.add('box-body');
    setStyles(boxBody, this.style.body);
    box.appendChild(boxBody);
    const results = document.createElement('div');
    results.innerText = 'Results will show up here';
    boxBody.appendChild(results);
    const filterDiv = document.createElement('div');
    filterDiv.classList.add('d-flex');
    boxBody.prepend(filterDiv);
    const input = document.createElement('input');
    input.id = 'aliasSitesInput';
    input.type = 'text';
    input.setAttribute('placeholder', 'Click 🔎 or enter an alias...');
    input.setAttribute('list', 'modelAliases');
    setStyles(input, { flex: 'auto', marginBottom: '0.2rem' });
    filterDiv.append(input);
    const clearWrapper = document.createElement('span');
    clearWrapper.style.position = 'relative';
    const clear = document.createElement('span');
    setStyles(clear, {
      position: 'absolute',
      top: '-0.22em',
      right: '.1em',
      padding: '0 .1em',
      color: this.style.title.backgroundColor,
      fontWeight: '800',
      fontFamily: 'sans-serif',
      fontSize: '1.2rem',
      cursor: 'pointer',
      userSelect: 'none',
    })
    clear.title = 'Clear';
    clear.innerText = 'x';
    clear.addEventListener('click', () => {
      input.value = '';
      input.dispatchEvent(new Event('input'));
    });
    clearWrapper.appendChild(clear);
    filterDiv.append(clearWrapper);
    const datalist = document.createElement('datalist');
    datalist.id = 'modelAliases';
    for (const alias of this.aliases.map(({ alias }) => alias).sort()) {
      const option = document.createElement('option');
      option.innerText = alias;
      datalist.appendChild(option);
    }
    input.after(datalist);
    const csv = document.createElement('div');
    setStyles(csv, this.style.csv);
    box.appendChild(csv);
    const handleInput = () => {
      const value = input.value.trim();
      filterAll.disabled = !value;
      filterAll.checked = false;
      restoreAllSets();

      let sites = this.sitesForName(value, true)
        .sort((a, b) => (a.name || a.display).localeCompare((b.name || b.display), undefined, { sensitivity: 'accent' }));
      if (!value)
        sites = sites.filter(({ setCount }) => setCount > 0);

      if (sites.length === 0) {
        boxTitleNode.textContent = boxDefaultTitle;
        titleSetCount.innerText = '';
        csv.innerText = '';
        if (value) {
          results.innerText = 'No results';
          filterAll.disabled = true;
          filterAll.checked = false;
        } else {
          results.innerText = 'Results will show up here';
        }
        return;
      }

      const totalSetCount = sites.reduce((sum, { setCount }) => sum + setCount, 0);
      // Disable if there are no sets
      filterAll.disabled = totalSetCount === 0;

      boxTitleNode.textContent = value ? boxDefaultTitle : 'Sets By Website';
      results.innerText = '';
      results.append(...sites.map(makeSiteDiv));

      csv.innerText = sites
        .map((s) => s.name || s.display)
        .filter((s, i, self) => self.indexOf(s) === i)
        .join(',');

      titleSetCount.innerText = `${totalSetCount} sets`;
    };
    input.addEventListener('input', handleInput);

    const filterAll = document.createElement('input');
    filterAll.type = 'checkbox';
    filterAll.checked = false;
    filterAll.disabled = true;
    filterAll.title = 'Select/unselect all sites below';
    setStyles(filterAll, { margin: '0 2px 2px 0' });
    filterDiv.prepend(filterAll);
    filterAll.addEventListener('input', () => {
      const checkboxes = getFilterCheckboxes();
      checkboxes.forEach((cb) => {
        cb.checked = filterAll.checked;
      });
      filterSets();
    });

    /** @param {SiteFilter} site */
    const makeSiteDiv = (site) => {
      const { name, display, listedCount, setCount } = site;
      const div = document.createElement('div');
      const filter = document.createElement('input');
      filter.type = 'checkbox';
      filter.name = 'siteSetFilter';
      filter.value = JSON.stringify(site);
      filter.disabled = setCount === 0;
      filter.title = 'Filter sets by this sites';
      setStyles(filter, { marginRight: '.25rem' });
      filter.addEventListener('input', () => {
        const checkboxes = getFilterCheckboxes();
        filterAll.checked = checkboxes.every((cb) => cb.checked);
        filterSets();
      });
      const siteSpan = document.createElement('abbr');
      setStyles(siteSpan, { userSelect: 'all', wordBreak: 'break-all' });
      siteSpan.innerText = name || display;
      if (name && name !== display)
        siteSpan.title = display;
      const countEl = document.createElement('span');
      countEl.innerText = ` (${setCount || listedCount})`;
      const currentFilter = input.value.trim();
      if (currentFilter && listedCount && setCount === 0) {
        countEl.classList.add('text-danger');
        countEl.title = `No sets as "${currentFilter}" found.`;
      }
      div.append(filter, siteSpan, countEl);
      return div;
    };

    /** @type {IndexxxSet["el"][]} */
    const filteredSets = [];

    const restoreAllSets = () => {
      filteredSets.forEach((el) => {
        el.classList.toggle('d-none', false);
      });
      filteredSets.length = 0;
    }

    const getFilterCheckboxes = () =>
      /** @type {HTMLInputElement[]} */
      (Array.from(document.querySelectorAll('input[name="siteSetFilter"]:not(:disabled)')))

    const filterSets = () => {
      const checkboxes = getFilterCheckboxes();
      /** @type {string[]} */
      const selected = [];
      for (const cb of checkboxes) {
        if (cb.checked) {
          /** @type {SiteFilter} */
          const site = JSON.parse(cb.value);
          selected.push(site.display);
        }
      }
      // console.debug('selected sites', selected);

      this.allSets.forEach((set) => {
        const isSelected = selected.length === 0 || selected.includes(set.site.display);
        const currentFilter = input.value.trim();
        const isMatch = isSelected && (!currentFilter || set.models[0].name === currentFilter);
        set.el.classList.toggle('d-none', !isMatch);
        if (!isMatch) return filteredSets.push(set.el);
        const index = filteredSets.indexOf(set.el);
        if (index !== -1) filteredSets.splice(index, 1);
      });
    };

    // initial state
    handleInput();

    return box;
  }

  _createActions() {
    const modelTitleSection = /** @type {HTMLDivElement */ (document.querySelector('#modelTitleSection'));

    const atid = 'userscript-actions-top';
    /** @type {HTMLDivElement | null} */
    let actionsTop = modelTitleSection.querySelector(`#${atid}`);
    if (!actionsTop) {
      actionsTop = document.createElement('div');
      actionsTop.id = atid;
      setStyles(actionsTop, { fontSize: '2em', width: '3.3em', padding: '0 0.25rem' });
      actionsTop.classList.add('d-flex', 'justify-content-between');
      modelTitleSection.appendChild(actionsTop);
    }

    const abid = 'userscript-actions-bottom';
    /** @type {HTMLSpanElement | null} */
    let actionsBottom = this.portfolioHeader?.querySelector(`#${abid}`);
    if (this.portfolioHeader && !actionsBottom) {
      actionsBottom = document.createElement('span');
      actionsBottom.id = abid;
      actionsBottom.classList.add('ml-2');
      this.portfolioHeader.appendChild(actionsBottom);
    }

    return {
      titleActions: actionsTop,
      portfolioActions: actionsBottom,
    };
  }

  setUpToggle() {
    const { titleActions } = this._createActions();
    const toggleButton = document.createElement('div');
    setStyles(toggleButton, { cursor: 'pointer' });
    toggleButton.title = 'Toggle all info boxes';
    toggleButton.innerText = '🔘';
    titleActions.append(toggleButton);

    toggleButton.addEventListener('click', () => {
      const anyHidden =
        /** @type {HTMLDivElement[]} */
        (Array.from(this.modelHeader.querySelectorAll('.box-title + .box-body')))
          .some((el) => el.style.display === 'none');

      /** @type {HTMLDivElement[]} */
      (Array.from(this.modelHeader.querySelectorAll('.box-title + .box-body')))
        .forEach((el) => el.style.display = anyHidden ? '' : 'none');

      const twitter = /** @type {HTMLDivElement} */ (this.modelHeader.querySelector(':scope > .twitter'));
      if (twitter)
        twitter.classList.toggle('d-none', !anyHidden);

      const modelImage = /** @type {HTMLImageElement} */ (this.modelHeader.querySelector('img.model-img'));
      if (modelImage.dataset.maxHeight) {
        modelImage.style.maxHeight = modelImage.dataset.maxHeight;
        modelImage.dataset.maxHeight = '';
      } else {
        modelImage.dataset.maxHeight = modelImage.style.maxHeight;
        modelImage.style.maxHeight = '350px';
      }
    });
  }

  /** @type {Map<string, (set: IndexxxSet) => HTMLTableCellElement>} */
  ExportFields = new Map([
    ['URL', (set) => {
      const cell = document.createElement('td');
      cell.style.maxWidth = '500px';
      const a = document.createElement('a');
      a.href = a.innerText = set.url;
      a.target = '_blank';
      cell.appendChild(a);
      return cell;
    }],
    ['Site', (set) => {
      const cell = document.createElement('td');
      cell.innerText = set.site.name || set.site.display;
      return cell;
    }],
    ['Date', (set) => {
      const cell = document.createElement('td');
      cell.innerText = set.date;
      return cell;
    }],
    ['Title', (set) => {
      const cell = document.createElement('td');
      cell.style.maxWidth = '400px';
      cell.innerText = set.title;
      return cell;
    }],
    ['Models', (set) => {
      const cell = document.createElement('td');
      cell.style.maxWidth = '400px';
      set.models.forEach((model, idx) => {
        if (idx > 0) cell.append(' | ');
        const a = document.createElement('a');
        a.classList.add('d-inline-block');
        a.innerText =
          model.actualName && model.actualName !== model.name
            ? `${model.name} (${model.actualName})`
            : (model.actualName || model.name);
        a.href = model.url;
        cell.append(a);
      });
      return cell;
    }],
  ]);

  exportColumnOrder = Array.from(this.ExportFields.keys());

  setUpExport() {
    const { portfolioActions } = this._createActions();

    if (!portfolioActions)
      return;

    const exportButton = document.createElement('span');
    setStyles(exportButton, { cursor: 'pointer' });
    exportButton.title = 'Export visible sets';
    exportButton.innerText = '📤';
    portfolioActions.append(exportButton);

    const exportTarget = document.createElement('div');
    exportTarget.id = 'export-target';
    this.portfolioHeader.after(exportTarget);

    exportButton.addEventListener('click', () => this.onExport());

    return exportTarget;
  }

  onExport() {
    const table = document.createElement('table');
    table.classList.add('table', 'table-sm', 'table-bordered', 'table-striped', 'w-auto');
    const thead = document.createElement('thead');
    const tbody = document.createElement('tbody');
    table.append(thead, tbody);

    const theadrow = document.createElement('tr');
    theadrow.classList.add('table-primary');
    this.exportColumnOrder.forEach((key, column) => {
      const cell = document.createElement('th');
      cell.innerText = key;
      cell.style.cursor = 'pointer';
      cell.title = 'Hide values in column';
      cell.addEventListener('click', () => {
        for (const row of tbody.rows) {
          row.cells[column].classList.toggle('invisible');
        }
      });
      theadrow.appendChild(cell);
    });
    thead.append(theadrow);

    const visibleSets = this.allSets.filter(({ el }) => !el.classList.contains('d-none'));

    const rows = visibleSets.map((set) => {
      const row = document.createElement('tr');
      for (const fieldFunc of this.ExportFields.values()) {
        row.appendChild(fieldFunc(set));
      }
      return row;
    });
    tbody.append(...rows);

    this.exportTarget.innerHTML = '';
    this.exportTarget.append(table);

    const { portfolioActions } = this._createActions();

    const bodyRect = document.body.getBoundingClientRect();
    const targetRect = this.exportTarget.getBoundingClientRect();
    window.scrollTo({
      behavior: 'smooth',
      top: targetRect.top - bodyRect.top - 50,
      left: 0,
    });

    /** @type {HTMLSpanElement | null} */
    let buttons = this.portfolioHeader.querySelector('#userscript-export-buttons');
    if (!buttons) {
      buttons = document.createElement('span');
      buttons.id = 'userscript-export-buttons';
      buttons.classList.add('ml-2');
      portfolioActions.append(buttons);
    }

    const scrollUp = () =>
      window.scrollTo({ behavior: 'smooth', top: bodyRect.top, left: 0 });

    const titleUp = 'Go up';
    if (!buttons.querySelector(`[title="${titleUp}"]`)) {
      const up = document.createElement('span');
      setStyles(up, { cursor: 'pointer' });
      up.title = titleUp;
      up.innerText = '🔝';
      up.addEventListener('click', scrollUp);
      buttons.append(up);
    }

    const titleClose = 'Close';
    if (!buttons.querySelector(`[title="${titleClose}"]`)) {
      const close = document.createElement('span');
      setStyles(close, { cursor: 'pointer' });
      close.title = titleClose;
      close.innerText = '❌';
      close.addEventListener('click', () => {
        this.exportTarget.innerHTML = '';
        close.remove();
      });
      buttons.append(close);
    }
  }

  /**
   * Number of uses for each alias (sets or listed)
   */
  aliasesUsageCount() {
    /** @type {string[]} */
    const sites = [];

    /** @type {{ alias: string, count: number, listed: boolean }[]} */
    const results = [];

    /** @type {HTMLSpanElement[]} */
    const aliasSpans = (Array.from(this.websitesBox.querySelectorAll('li span.alias')));

    aliasSpans.forEach((aliasSpan) => {
      let name = aliasSpan.innerText.trim();
      if (!name) name = NO_ALIAS;

      const siteEl = closestWebsiteLink(aliasSpan);
      if (!siteEl) {
        console.error('error getting site for', aliasSpan);
        return;
      }

      const site = siteEl.innerText;
      if (site) {
        const key = [site, name].join('|');
        if (sites.includes(key)) {
          console.debug(`skipping duplicate site/alias: ${key}`);
          return;
        }
        sites.push(key);
      }

      const { length: setCount } = this.setsBySite(siteEl, aliasSpan.innerText);
      const listedCount = this.getAliasListedCount(aliasSpan);

      if (listedCount === null && !setCount) return;

      const minSetCount = setCount || listedCount || 1;

      const result = results.find(({ alias }) => alias === name);
      if (!result) results.push({ alias: name, count: minSetCount, listed: true });
      else result.count += minSetCount;
    });

    this.uniqueSets
      .forEach((set) => {
        const { name } = set.models[0];
        if (results.find(({ alias }) => alias === name) === undefined) {
          const { display: siteDisplay, el: siteEl } = set.site;
          const key = [siteDisplay, name].join('|');
          if (sites.includes(key)) {
            // console.debug(`skipping duplicate site/alias: ${key}`);
            return;
          }
          sites.push(key);

          const { length: count } = this.setsBySite(siteEl, name);
          results.push({ alias: name, count, listed: false });
        }
      });

    return results;
  }

  /**
   * @typedef SiteFilter
   * @property {string | null} name
   * @property {string} display
   * @property {number} listedCount
   * @property {number} setCount
   */
  /**
   * List of sites for alias
   * @param {string} [name]
   * @param {boolean} [exact=false]
   * @returns {SiteFilter[]}
   */
  sitesForName(name, exact=false) {
    /** @type {SiteFilter[]} */
    const sites = [];

    /**
     * @param {SiteFilter} other
     */
    const findExistingSite = (other) =>
      sites.find((site) => site.name == other.name && site.display == other.display);

    /**
     * @param {string} name
     * @param {string} display
     * @param {number} count
     * @param {boolean} fromList
     */
    const push = (name, display, count, fromList) => {
      const countTarget = (c = fromList) => c ? 'listedCount' : 'setCount';
      const final = /** @type {SiteFilter} */ ({
        name,
        display,
        [countTarget()]: count,
        [countTarget(!fromList)]: 0,
      });
      const existingSite = findExistingSite(final);
      if (existingSite)
        existingSite[countTarget()] += count;
      else
        sites.push(final);
    };

    /**
     * @param {string} alias
     */
    const shouldHandle = (alias) => (
      !name
      || (name === NO_ALIAS && !alias)
      || alias.localeCompare(name, undefined, { sensitivity: exact ? 'variant' : 'accent' }) === 0
    );

    /** @type {HTMLSpanElement[]} */
    (Array.from(this.websitesBox.querySelectorAll('li span.alias')))
      .forEach((aliasSpan) => {
        const alias = aliasSpan.innerText.trim();

        if (shouldHandle(alias)) {
          const siteEl = closestWebsiteLink(aliasSpan);
          if (!siteEl) {
            console.error('error getting site for', aliasSpan);
            sites.push({ name: null, display: '???', listedCount: 0, setCount: 0 });
            return;
          }

          const siteSets = this.setsBySite(siteEl);
          const siteName = siteSets[0]?.site.name || '';
          const siteDisplay = siteEl.innerText;

          const hasSetsAsAlias = !!siteSets.find((set) => set.models[0].name === alias);
          if (!hasSetsAsAlias) {
            const listedCount = (name ? this.getAliasListedCount(aliasSpan) : 0);
            if (listedCount !== null)
              push(siteName, siteDisplay, listedCount, true);
          }
        }
      });

    this.uniqueSets
      .forEach((set) => {
        const alias = set.models[0].name;
        if (shouldHandle(alias)) {
          const { name: siteName, display: siteDisplay, el: siteEl } = set.site;
          const { length: setCount } = this.setsBySite(siteEl, alias);
          push(siteName, siteDisplay, setCount, false);
        }
      });

    return sites;
  }

  /**
   * Get only the listed count of aliases.
   * @param {HTMLSpanElement} aliasSpan
   * @returns {number | null}
   */
  getAliasListedCount(aliasSpan) {
    let nextSpan = /** @type {HTMLElement | undefined} */ (aliasSpan.nextElementSibling);

    if (!nextSpan || nextSpan.matches('a[title="edit"]')) {
      // get parent site (multiple names for one site, set count on site row)
      const site = closestWebsiteLink(aliasSpan);
      if (!site) {
        console.error('error getting site for', aliasSpan);
        return 0;
      }

      const siteParent = /** @type {HTMLElement} */ (site.parentElement);
      nextSpan = /** @type {HTMLSpanElement} */ (siteParent.querySelector(':scope > span > span'));
      if (!nextSpan)
        return 0;
    }

    if (nextSpan.classList.contains('count')) {
      return Number(nextSpan.innerText.trim()) || 0;
    }

    if (nextSpan.classList.contains('descr')) {
      const text = nextSpan.innerText.trim();
      if (/\b(delete|remove)\b/i.test(text))
        return null;
      const match = text.match(/^(\d+) *;.+/);
      if (match)
        return Number(match[0]);
    }

    return 0;
  }

  /**
   * @param {HTMLAnchorElement | HTMLSpanElement} siteEl
   * @param {string} [alias]
   * @returns {IndexxxSet[]}
   */
  setsBySite(siteEl, alias) {
    const site = siteEl.getAttribute('href') || siteEl.innerText;

    if (!site)
      return [];

    const sets = this.allSets.filter((set) => {
      if (alias && set.models[0].name !== alias)
        return false;

      if (/^https?:\/\//.test(site) || site.startsWith('/')) // outUrl
        return set.site.outUrl === site;
      else
        return set.site.display === site;
    });

    return sets;
  }

} // ModelPage

class WebsiteSetsPage {

  constructor() {
    /** @type {IndexxxSet[]} */
    this.allSets =
      /** @type {HTMLDivElement[]} */
      (Array.from(document.querySelectorAll('.pset.card'))).map(parseSetCard);
  }

} // WebsiteSetsPage

/**
 * @param {HTMLElement} start
 * @returns {HTMLAnchorElement | null}
 */
function closestWebsiteLink(start) {
  const selector = ':scope > a.websiteLink, :scope > span.websiteLink';
  let current = /** @type {HTMLElement} */ (start.parentElement).closest('li');
  while (current && !current.querySelector(selector)) {
    current = /** @type {HTMLElement} */ (current.parentElement).closest('li');
  }
  if (!current) return null;
  return current.querySelector(selector);
}

/**
 * @typedef IndexxxSetModel
 * @property {string} name
 * @property {string} [actualName]
 * @property {string} url
 * @property {HTMLAnchorElement | HTMLSpanElement | null} el
 */
/**
 * @typedef IndexxxSetSite
 * @property {string} name (title)
 * @property {string} display (innerText)
 * @property {string | null} outUrl (href)
 * @property {HTMLAnchorElement | HTMLSpanElement} el
 */
/**
 * @typedef IndexxxSet
 * @property {string} title
 * @property {string} date
 * @property {string} url
 * @property {[self: IndexxxSetModel, ...other: IndexxxSetModel[]]} models
 * @property {IndexxxSetSite} site
 * @property {HTMLDivElement} el
 */
/**
 * @param {HTMLDivElement} setCard
 * @returns {IndexxxSet}
 */
function parseSetCard(setCard) {
  const setLink = /** @type {HTMLAnchorElement} */ (setCard.querySelector('.psetInfo > div:nth-of-type(2) > a'));
  const url = setLink.href;

  const date = /** @type {HTMLTimeElement} */ (setCard.querySelector('time')).innerText.trim();

  // alt="Lily Jordan, Kylie Page in Kylie Page fucking in the bed with her small tits,  at 2chickssametime.com"
  // alt="Lily Jordan in  at amkingdom.com"
  // alt="Michelle B (Lenka) in Michelle B 1,  at domai.com"
  // alt="Eva Fenix in ,  at atkexotics.com"
  // [website sets - no models] alt="in The Pro,  at brandibelle.com"
  // https://regex101.com/r/SqnoyK/3
  const alt = /** @type {HTMLImageElement} */ (setCard.querySelector('img')).alt.trim();
  const altMatch = alt.match(/^(?:(.*?) )?in (?:(.+)?, )? at .+$/);
  if (!altMatch) {
    console.debug(setCard);
    throw new Error(`Failed to parse alt value: ${alt}`);
  }

  const [, rawActualNames, rawTitle] = altMatch;
  if (currentModel && !rawActualNames)
    throw new Error(`Failed to parse alt value: ${alt} (no models)`);

  const title = rawTitle || '[no title]';
  setLink.title = title;

  /** @type {{ [index: number]: string | undefined }} */
  const actualNames = {};
  rawActualNames?.split(/, /g).forEach((name, i) => {
    // https://regex101.com/r/fO11Ht/1
    const nameMatch = name.match(/^(.+?)(?: \((.+)\))?$/);
    if (!nameMatch) {
      console.error(`Failed to parse name: ${name}`, setCard);
      return;
    }
    const [, aliasOrName, actualName] = nameMatch;
    actualNames[i] = actualName || aliasOrName || undefined;
  });

  /** @type {IndexxxSetModel[]} */
  const allModels = (
    /** @type {(HTMLAnchorElement | HTMLSpanElement)[]} */
    (Array.from(setCard.querySelectorAll('.models li .modelLink')))
      .map((modelLink, i) => ({
        name: modelLink.innerText.trim(),
        actualName: actualNames[i],
        url: (modelLink instanceof HTMLAnchorElement
          ? modelLink
          : /** @type {HTMLLinkElement} */ (modelLink.nextElementSibling)
        ).href,
        el: modelLink,
      }))
  );
  let thisModel = allModels.find(({ el }) => el instanceof HTMLSpanElement);
  if (thisModel) {
    allModels.splice(allModels.indexOf(thisModel), 1);
  } else {
    if (currentModel) {
      console.info('unexpected missing set model', setCard);
      /** @type {IndexxxSetModel} */
      thisModel = ({ ...currentModel, el: null });
    }
  }
  // FIXME: TypeScript error
  /** @type {IndexxxSet["models"]} */
  const models = [thisModel, ...allModels];

  const siteEl = /** @type {HTMLAnchorElement | HTMLSpanElement} */
    (setCard.querySelector('.psetInfo > .psWebsite > :first-child'));
  const { innerText: display, title: siteTitle } = siteEl;
  const outUrl = siteEl.getAttribute('href');

  const siteName = siteTitle?.replace(/^Go to: /, '');

  /** @type {IndexxxSet["site"]} */
  const site = { name: siteName, display, outUrl, el: siteEl };

  return { date, url, models, site, title, el: setCard };
}

/**
 * @template {HTMLElement} E
 * @param {E} el
 * @param {Partial<CSSStyleDeclaration>} styles
 * @returns {E}
 */
function setStyles(el, styles) {
  Object.assign(el.style, styles);
  return el;
}

/**
 * @template {HTMLElement} E
 * @param {E} el
 * @param {Object} opts
 * @returns {E}
 */
function tryResizable(el, opts) {
  try {
    //@ts-expect-error
    jQuery(el).resizable(opts);
  } catch (error) {}
  return el;
}

main();