FCX

Mejora la navegación en ForoCoches.

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         FCX
// @namespace    https://github.com/mdmrk/fcx
// @version      0.1.0
// @description  Mejora la navegación en ForoCoches.
// @license      GNU General Public License v3.0
// @author       mdmrk
// @match        https://forocoches.com/*
// @icon         
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_addStyle
// @grant        GM_registerMenuCommand
// @run-at       document-end
// @noframes
// ==/UserScript==

// src/style.css
var style_default = ".infinite-scroll-separator{justify-content:center;align-items:center;margin:20px 0;display:flex;position:relative}.infinite-scroll-separator-line{background-color:var(--border-color,#e5e7eb);z-index:1;height:1px;position:absolute;left:0;right:0}.infinite-scroll-separator-text{background-color:var(--bg-color,#fff);color:var(--text-secondary,#6b7280);z-index:2;border:1px solid var(--border-color,#e5e7eb);border-radius:9999px;padding:0 16px;font-size:.875rem;font-weight:500}#fcx-config-backdrop{z-index:9998;background:#00000080;width:100%;height:100%;position:fixed;top:0;left:0}#fcx-config-panel{color:#ccc;z-index:9999;opacity:0;background:#1e1e1e;border:1px solid #444;border-radius:4px;flex-direction:column;width:900px;max-width:95%;max-height:90vh;font-size:13px;display:flex;position:fixed;top:50%;left:50%;transform:translate(-50%,-50%)scale(.95);box-shadow:0 10px 25px #00000080}#fcx-config-header{color:#fff;background:linear-gradient(#2a2a2a,#1a1a1a);border-bottom:1px solid #000;padding:8px 10px;font-weight:700}#fcx-config-content{flex:1;padding:0;overflow-y:auto}#fcx-config-table{border-collapse:collapse;width:100%}#fcx-config-table th,#fcx-config-table td{text-align:left;border-bottom:1px solid #333;padding:10px}#fcx-config-table th{color:#fe5e00;z-index:10;background:#252525;font-weight:700;position:sticky;top:0}#fcx-config-table th.fcx-col-opt{width:auto}#fcx-config-table th.fcx-col-chk{text-align:center;width:80px}#fcx-config-table td.fcx-col-chk{text-align:center}.fcx-desc{color:#888;margin-top:4px;font-size:11px;display:block}.fcx-label{color:#eee;font-weight:700}.fcx-disabled{opacity:.3;pointer-events:none}#fcx-footer{text-align:right;color:#666;background:#1a1a1a;border-top:1px solid #333;padding:5px 10px;font-size:11px}#fcx-footer a{color:#888;cursor:pointer;margin-left:10px;text-decoration:none}#fcx-footer a:hover{color:#ccc}";

// src/utils/inject-console.ts
var consoleMethodsThatDontBreakWhenArgumentIsString = [
  "log",
  "error",
  "warn",
  "info",
  "debug",
  "trace"
];
function injectConsole(prefix) {
  const originalConsole = globalThis.console;
  globalThis.console = new Proxy(originalConsole, {
    get(target, prop, receiver) {
      const method = Reflect.get(target, prop, receiver);
      return consoleMethodsThatDontBreakWhenArgumentIsString.some((propName) => propName === prop) ? method.bind(target, `[${prefix}]
`) : method;
    }
  });
}

// src/config.ts
var devMode = false;
var theme = {
  enhancedCommentClass: "enhanced-comment"
};
var oldSelectors = {
  feedContainer: "#posts",
  commentItem: "table[id^='post']",
  avatar: ".avatar",
  authorName: ".bigusername",
  content: "div[id^='post_message_']",
  nextPageLink: "a[rel='next']",
  prevPageLink: "a[rel='prev']",
  activePage: "td.alt2 .mfont strong"
};
var newSelectors = {
  feedContainer: "#posts",
  commentItem: ".postbit_wrapper",
  avatar: ".thread-profile-image",
  authorName: "div[id^='postmenu_'] > a",
  content: "div[id^='post_message_']",
  nextPageLink: "a:has(span[style*='--next-right-icon'])",
  prevPageLink: "a:has(span[style*='--next-left-icon'])",
  activePage: "span[title*='Mostrando resultados'] > strong"
};

// src/utils/logger.ts
var logger = {
  log: (...args) => {
    if (devMode)
      console.log(...args);
  },
  warn: (...args) => {
    if (devMode)
      console.warn(...args);
  },
  error: (...args) => {
    if (devMode)
      console.error(...args);
  },
  info: (...args) => {
    if (devMode)
      console.info(...args);
  }
};

// src/utils/detect-interface.ts
var isNewInterface = () => {
  return !!document.getElementById("fc-desktop-version-tag-for-monitoring");
};

// src/utils/page-state.ts
var detectPageType = () => {
  const url = window.location.href;
  if (url.includes("showthread.php")) {
    return "THREAD" /* THREAD */;
  }
  if (url.includes("forumdisplay.php")) {
    return "CATEGORY" /* CATEGORY */;
  }
  if (/\/foro\/?($|\?|#)/.test(url)) {
    return "HOME" /* HOME */;
  }
  if (/^https?:\/\/[^/]+\/?(?:$|\?|#)/.test(url)) {
    return "HOME" /* HOME */;
  }
  return "UNKNOWN" /* UNKNOWN */;
};
var currentPageType = detectPageType();

// src/config-registry.ts
var CONFIG_KEYS = {
  REMOVE_SIDEBAR: "remove_sidebar",
  INFINITE_SCROLL: "infinite_scroll",
  REMOVE_BANNERS: "remove_banners"
};
var configs = [
  {
    key: CONFIG_KEYS.REMOVE_SIDEBAR,
    label: "Eliminar Barra Lateral",
    description: "Elimina la barra lateral derecha y expande el área de contenido principal.",
    defaultValue: false,
    type: "checkbox",
    scopes: ["new"]
  },
  {
    key: CONFIG_KEYS.INFINITE_SCROLL,
    label: "Scroll Infinito",
    description: "Carga automáticamente la siguiente página de hilos al llegar al final.",
    defaultValue: true,
    type: "checkbox"
  },
  {
    key: CONFIG_KEYS.REMOVE_BANNERS,
    label: "Eliminar Publicidad",
    description: "Oculta los banners de publicidad.",
    defaultValue: true,
    type: "checkbox"
  }
];

// src/utils/storage.ts
var getConfig = (key, defaultValue) => {
  if (typeof GM_getValue !== "undefined") {
    return GM_getValue(key, defaultValue);
  }
  const val = localStorage.getItem(`fcx_${key}`);
  if (val === null)
    return defaultValue;
  try {
    return JSON.parse(val);
  } catch {
    return val;
  }
};
var setConfig = (key, value) => {
  if (typeof GM_setValue !== "undefined") {
    GM_setValue(key, value);
    return;
  }
  localStorage.setItem(`fcx_${key}`, JSON.stringify(value));
};
var resetConfig = (key, defaultValue) => {
  setConfig(key, defaultValue);
};
var getScopedConfigKey = (key, scope) => {
  return `${key}_${scope}`;
};
var getEffectiveConfig = (key, scope) => {
  const general = getConfig(key, false);
  if (general === true)
    return true;
  const scopedKey = getScopedConfigKey(key, scope);
  return getConfig(scopedKey, false);
};

// src/ui/config-panel.ts
var createCheckbox = (checked, onChange) => {
  const input = document.createElement("input");
  input.type = "checkbox";
  input.checked = checked;
  input.onchange = (e) => {
    onChange(e.target.checked);
  };
  return input;
};
var renderRow = (item) => {
  const tr = document.createElement("tr");
  const tdName = document.createElement("td");
  const label = document.createElement("div");
  label.className = "fcx-label";
  label.textContent = item.label;
  const desc = document.createElement("span");
  desc.className = "fcx-desc";
  desc.textContent = item.description;
  tdName.append(label, desc);
  const tdGeneral = document.createElement("td");
  tdGeneral.className = "fcx-col-chk";
  const generalChecked = getConfig(item.key, item.defaultValue);
  const updateRowState = (isGeneralChecked) => {
    if (isGeneralChecked) {
      tdNew.classList.add("fcx-disabled");
      tdOld.classList.add("fcx-disabled");
    } else {
      tdNew.classList.remove("fcx-disabled");
      tdOld.classList.remove("fcx-disabled");
    }
  };
  const checkGeneral = createCheckbox(generalChecked, (val) => {
    setConfig(item.key, val);
    updateRowState(val);
  });
  tdGeneral.append(checkGeneral);
  const tdNew = document.createElement("td");
  tdNew.className = "fcx-col-chk";
  if (!item.scopes || item.scopes.includes("new")) {
    const keyNew = getScopedConfigKey(item.key, "new");
    const checkNew = createCheckbox(getConfig(keyNew, false), (val) => setConfig(keyNew, val));
    tdNew.append(checkNew);
  }
  const tdOld = document.createElement("td");
  tdOld.className = "fcx-col-chk";
  if (!item.scopes || item.scopes.includes("old")) {
    const keyOld = getScopedConfigKey(item.key, "old");
    const checkOld = createCheckbox(getConfig(keyOld, false), (val) => setConfig(keyOld, val));
    tdOld.append(checkOld);
  }
  updateRowState(generalChecked);
  tr.append(tdName, tdGeneral, tdNew, tdOld);
  return tr;
};
var toggleConfigPanel = () => {
  if (document.getElementById("fcx-config-panel")) {
    closePanel();
    return;
  }
  openPanel();
};
var openPanel = () => {
  const backdrop = document.createElement("div");
  backdrop.id = "fcx-config-backdrop";
  backdrop.onclick = closePanel;
  const panel = document.createElement("div");
  panel.id = "fcx-config-panel";
  const header = document.createElement("div");
  header.id = "fcx-config-header";
  header.textContent = "Configuración FCX";
  const content = document.createElement("div");
  content.id = "fcx-config-content";
  const table = document.createElement("table");
  table.id = "fcx-config-table";
  const thead = document.createElement("thead");
  const trHead = document.createElement("tr");
  const thName = document.createElement("th");
  thName.className = "fcx-col-opt";
  thName.textContent = "Opción";
  const thGen = document.createElement("th");
  thGen.className = "fcx-col-chk";
  thGen.textContent = "General";
  const thNew = document.createElement("th");
  thNew.className = "fcx-col-chk";
  thNew.textContent = "Nuevo";
  const thOld = document.createElement("th");
  thOld.className = "fcx-col-chk";
  thOld.textContent = "Antiguo";
  trHead.append(thName, thGen, thNew, thOld);
  thead.append(trHead);
  table.append(thead);
  const tbody = document.createElement("tbody");
  configs.forEach((item) => {
    if (item.type === "checkbox") {
      tbody.append(renderRow(item));
    }
  });
  table.append(tbody);
  content.append(table);
  const footer = document.createElement("div");
  footer.id = "fcx-footer";
  const closeLink = document.createElement("a");
  closeLink.textContent = "Cerrar";
  closeLink.onclick = closePanel;
  const resetLink = document.createElement("a");
  resetLink.textContent = "Restaurar todo";
  resetLink.onclick = () => {
    if (confirm("¿Restaurar toda la configuración?")) {
      configs.forEach((c) => {
        resetConfig(c.key, c.defaultValue);
        if (!c.scopes || c.scopes.includes("new")) {
          resetConfig(getScopedConfigKey(c.key, "new"), false);
        }
        if (!c.scopes || c.scopes.includes("old")) {
          resetConfig(getScopedConfigKey(c.key, "old"), false);
        }
      });
      closePanel();
      openPanel();
    }
  };
  footer.append(resetLink, document.createTextNode(" | "), closeLink);
  panel.append(header, content, footer);
  document.body.append(backdrop, panel);
  requestAnimationFrame(() => {
    backdrop.style.opacity = "1";
    panel.style.opacity = "1";
    panel.style.transform = "translate(-50%, -50%) scale(1)";
  });
};
var closePanel = () => {
  const b = document.getElementById("fcx-config-backdrop");
  const p = document.getElementById("fcx-config-panel");
  if (b && p) {
    b.style.opacity = "0";
    p.style.opacity = "0";
    setTimeout(() => {
      b.remove();
      p.remove();
    }, 200);
  }
};

// src/lib/style-comment.ts
var styleComment = (element, _selectors) => {
  if (element.dataset.processed === "true")
    return;
  element.dataset.processed = "true";
  element.classList.add(theme.enhancedCommentClass);
};

// src/lib/watch-feed.ts
var watchFeed = (selectors) => {
  if (!window.location.href.includes("showthread.php")) {
    logger.log("Watch Feed: Not a thread page.");
    return () => {};
  }
  const feed = document.querySelector(selectors.feedContainer);
  if (!feed) {
    logger.warn("Feed container not found, retrying in 1s...");
    const timer = setTimeout(() => watchFeed(selectors), 1000);
    return () => clearTimeout(timer);
  }
  logger.log("Feed found, starting watcher...");
  const existing = feed.querySelectorAll(selectors.commentItem);
  existing.forEach((el) => {
    styleComment(el, selectors);
  });
  const observer = new MutationObserver((mutations) => {
    for (const mutation of mutations) {
      for (const node of mutation.addedNodes) {
        if (node instanceof HTMLElement) {
          if (node.matches(selectors.commentItem)) {
            styleComment(node, selectors);
          } else {
            const children = node.querySelectorAll(selectors.commentItem);
            children.forEach((el) => {
              styleComment(el, selectors);
            });
          }
        }
      }
    }
  });
  observer.observe(feed, { childList: true, subtree: true });
  return () => {
    observer.disconnect();
    logger.log("Feed watcher stopped");
  };
};

// src/lib/infinite-scroll.ts
var isLoading = false;
var nextUrl = null;
var prevUrl = null;
var currentSelectors = null;
var initInfiniteScroll = (selectors) => {
  currentSelectors = selectors;
  if (currentPageType !== "THREAD" /* THREAD */) {
    logger.log("Infinite Scroll: Not a thread page.");
    return;
  }
  const prevLink = document.querySelector(selectors.prevPageLink);
  if (prevLink) {
    prevUrl = prevLink.href;
    logger.log("Infinite Scroll: Prev page is", prevUrl);
    const topSentry = document.createElement("div");
    topSentry.id = "infinite-scroll-top-sentry";
    topSentry.innerHTML = "<p style='color: #666; font-weight: bold;'>Cargando página anterior...</p>";
    topSentry.style.textAlign = "center";
    topSentry.style.padding = "40px";
    const feed = document.querySelector(selectors.feedContainer);
    if (feed)
      feed.before(topSentry);
    const topObserver = new IntersectionObserver((entries) => {
      const entry = entries[0];
      if (entry.isIntersecting && !isLoading && prevUrl) {
        loadPrevPage();
      }
    }, { rootMargin: "300px" });
    topObserver.observe(topSentry);
  }
  const nextLink = document.querySelector(selectors.nextPageLink);
  if (!nextLink) {
    logger.log("Infinite Scroll: No next page found.");
  } else {
    nextUrl = nextLink.href;
    logger.log("Infinite Scroll: Next page is", nextUrl);
    const bottomSentry = document.createElement("div");
    bottomSentry.id = "infinite-scroll-bottom-sentry";
    bottomSentry.innerHTML = "<p style='color: #666; font-weight: bold;'>Cargando página siguiente...</p>";
    bottomSentry.style.textAlign = "center";
    bottomSentry.style.padding = "40px";
    const feed = document.querySelector(selectors.feedContainer);
    if (feed)
      feed.after(bottomSentry);
    const bottomObserver = new IntersectionObserver((entries) => {
      const entry = entries[0];
      if (entry.isIntersecting && !isLoading && nextUrl) {
        loadNextPage();
      }
    }, { rootMargin: "300px" });
    bottomObserver.observe(bottomSentry);
  }
};
var getPageNumber = (url) => {
  try {
    const urlObj = new URL(url);
    const params = new URLSearchParams(urlObj.search);
    return params.get("page");
  } catch (e) {
    console.error("Error parsing URL for page number", e);
    return null;
  }
};
var getPageNumberFromDoc = (doc) => {
  if (currentSelectors?.activePage) {
    const activePageEl = doc.querySelector(currentSelectors.activePage);
    if (activePageEl?.textContent) {
      return activePageEl.textContent.trim();
    }
  }
  return null;
};
var createSeparator = (pageNumber) => {
  const div = document.createElement("div");
  div.classList.add("infinite-scroll-separator");
  const line = document.createElement("div");
  line.classList.add("infinite-scroll-separator-line");
  const span = document.createElement("span");
  span.textContent = `Página ${pageNumber}`;
  span.classList.add("infinite-scroll-separator-text");
  div.appendChild(line);
  div.appendChild(span);
  return div;
};
var loadPrevPage = async () => {
  if (!prevUrl)
    return;
  isLoading = true;
  try {
    const response = await fetch(prevUrl);
    const text = await response.text();
    const parser = new DOMParser;
    const doc = parser.parseFromString(text, "text/html");
    const newFeed = doc.querySelector(currentSelectors.feedContainer);
    const currentFeed = document.querySelector(currentSelectors.feedContainer);
    if (newFeed && currentFeed) {
      const oldScrollHeight = document.documentElement.scrollHeight;
      const oldScrollTop = document.documentElement.scrollTop;
      Array.from(newFeed.children).reverse().forEach((child) => {
        const importedNode = document.importNode(child, true);
        currentFeed.insertBefore(importedNode, currentFeed.firstChild);
      });
      const pageNum = getPageNumberFromDoc(doc) || getPageNumber(prevUrl);
      if (pageNum) {
        const separator = createSeparator(pageNum);
        currentFeed.insertBefore(separator, currentFeed.firstChild);
      }
      const newScrollHeight = document.documentElement.scrollHeight;
      document.documentElement.scrollTop = oldScrollTop + (newScrollHeight - oldScrollHeight);
    }
    const prevLink = doc.querySelector(currentSelectors.prevPageLink);
    if (prevLink) {
      prevUrl = prevLink.href;
    } else {
      prevUrl = null;
      document.querySelector("#infinite-scroll-top-sentry")?.remove();
    }
  } catch (err) {
    console.error("Infinite Scroll (Prev) Error:", err);
  } finally {
    isLoading = false;
  }
};
var loadNextPage = async () => {
  if (!nextUrl)
    return;
  isLoading = true;
  try {
    const response = await fetch(nextUrl);
    const text = await response.text();
    const parser = new DOMParser;
    const doc = parser.parseFromString(text, "text/html");
    const newFeed = doc.querySelector(currentSelectors.feedContainer);
    const currentFeed = document.querySelector(currentSelectors.feedContainer);
    if (newFeed && currentFeed) {
      const pageNum = getPageNumberFromDoc(doc) || getPageNumber(nextUrl);
      if (pageNum) {
        const separator = createSeparator(pageNum);
        currentFeed.appendChild(separator);
      }
      Array.from(newFeed.children).forEach((child) => {
        const importedNode = document.importNode(child, true);
        currentFeed.appendChild(importedNode);
      });
    }
    const nextLink = doc.querySelector(currentSelectors.nextPageLink);
    if (nextLink) {
      nextUrl = nextLink.href;
    } else {
      nextUrl = null;
      document.querySelector("#infinite-scroll-bottom-sentry")?.remove();
    }
  } catch (err) {
    console.error("Infinite Scroll Error:", err);
  } finally {
    isLoading = false;
  }
};

// src/lib/remove-banners.ts
var removeBanners = (scope) => {
  const shouldRemove = getEffectiveConfig(CONFIG_KEYS.REMOVE_BANNERS, scope);
  if (!shouldRemove)
    return;
  const banner = document.getElementById("notices-wrapper");
  if (banner) {
    banner.remove();
  }
};

// src/adapters/new-adapter.ts
class NewSiteAdapter {
  name = "New Interface";
  selectors;
  constructor() {
    this.selectors = newSelectors;
  }
  init() {
    logger.log(`Initializing ${this.name} adapter...`);
    this.removeSidebar();
    removeBanners("new");
  }
  removeSidebar() {
    const shouldRemove = getEffectiveConfig(CONFIG_KEYS.REMOVE_SIDEBAR, "new");
    if (!shouldRemove)
      return;
    const sidebar = document.querySelector("#sidebar");
    if (sidebar)
      sidebar.remove();
    const main = document.querySelector("main");
    if (main)
      main.style.display = "block";
  }
  setupFeatures() {
    watchFeed(this.selectors);
    if (getEffectiveConfig(CONFIG_KEYS.INFINITE_SCROLL, "new")) {
      initInfiniteScroll(this.selectors);
    }
  }
}

// src/adapters/old-adapter.ts
class OldSiteAdapter {
  name = "Old Interface";
  selectors;
  constructor() {
    this.selectors = oldSelectors;
  }
  init() {
    logger.log(`Initializing ${this.name} adapter...`);
    removeBanners("old");
  }
  setupFeatures() {
    watchFeed(this.selectors);
    if (getEffectiveConfig(CONFIG_KEYS.INFINITE_SCROLL, "old")) {
      initInfiniteScroll(this.selectors);
    }
  }
}

// src/index.ts
(() => {
  injectConsole("FCX");
  GM_addStyle(style_default);
  const isNew = isNewInterface();
  logger.log("Script initializing...");
  logger.log(`Interface: ${isNew ? "New" : "Old"}`);
  logger.log(`Page Type: ${currentPageType}`);
  const adapter = isNew ? new NewSiteAdapter : new OldSiteAdapter;
  adapter.init();
  adapter.setupFeatures();
  GM_registerMenuCommand("Configuración", toggleConfigPanel);
  return () => {
    logger.log("Script unloaded");
  };
})();