FCX

Mejora la navegación en ForoCoches.

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 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");
  };
})();