Mejora la navegación en ForoCoches.
// ==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");
};
})();