// ==UserScript== // @name Spotless for eBay // @namespace https://github.com/OsborneLabs // @version 2.2.0 // @description Hides sponsored listings, cleans urls, and removes site telemetry // @author Osborne Labs // @license GPL-3.0-only // @homepageURL https://github.com/OsborneLabs/Spotless // @icon  // @match https://*.ebay.com/* // @match https://*.ebay.at/* // @match https://*.ebay.be/* // @match https://*.ebay.ca/* // @match https://*.ebay.ch/* // @match https://*.ebay.com.au/* // @match https://*.ebay.com.hk/* // @match https://*.ebay.com.my/* // @match https://*.ebay.com.sg/* // @match https://*.ebay.co.uk/* // @match https://*.ebay.de/* // @match https://*.ebay.es/* // @match https://*.ebay.fr/* // @match https://*.ebay.ie/* // @match https://*.ebay.it/* // @match https://*.ebay.nl/* // @match https://*.ebay.pl/* // @run-at document-start // @supportURL https://github.com/OsborneLabs/Spotless/issues // @grant none // @downloadURL https://update.greasyfork.icu/scripts/541981/Spotless%20for%20eBay.user.js // @updateURL https://update.greasyfork.icu/scripts/541981/Spotless%20for%20eBay.meta.js // ==/UserScript== /* jshint esversion: 11 */ (function() { "use strict"; const APP_NAME = "Spotless"; const APP_NAME_DEBUG_MODE = "SPOTLESS FOR EBAY"; const APP_KEY_HIDE_SPONSORED_CONTENT = "hideSponsoredContent"; const APP_ICONS = { locked: ``, unlocked: ``, arrow: ``, heart: `` }; const state = { ui: { hidingEnabled: localStorage.getItem(APP_KEY_HIDE_SPONSORED_CONTENT) !== "false", highlightedSponsoredContent: [], isContentProcessing: false, updateScheduled: false, observerInitialized: false, classFontCache: new Map() }, banner: { bannerUpdateScheduled: false }, carousel: { carouselObserver: null, carouselObserverInitialized: false } }; function createStyles() { const style = document.createElement("style"); style.textContent = ` :root { --color-app-bubble: #E74C3C; --color-app-icon-heart-hover: red; --color-app-icon: white; --color-app-switch-off: #CCC; --color-app-switch-on: #2AA866; --color-app-switch-thumb: white; --color-highlight-background: rgba(255, 230, 230, 0.30); --color-highlight-border: #D95C5C; --color-panel-divider: rgba(255, 255, 255, 0.1); --color-panel-row: rgba(20, 30, 45, 0.5); --color-panel-shadow: 0 8px 20px rgba(0, 0, 0, 0.2); --color-panel: rgba(34, 50, 70, 0.85); --color-text-link-default: var(--color-text-link-app-hover); --color-text-link-app-hover: lightblue; --color-text-link-ebay-hover: #2854D9; --color-text-default: white; --size-text-body-default: 14px; --size-text-body-error: 19px; --size-text-footer: 12px; --size-text-header-title: 23px; --size-thickness-highlight-border: 2px; } @media (max-width: 768px) { #panelWrapper { position: fixed; bottom: 5px; left: 50%; right: unset; transform: translateX(-50%); width: 90% !important; padding: 0 16px; } #panelHeader h2.panel-title { position: static; bottom: auto; } } #panelWrapper, #panelBox, .lock-icon-animation, .lock-icon-animation.active { box-sizing: border-box; } #panelWrapper { position: fixed; bottom: 10px; right: 5px; z-index: 9999; width: 100%; max-width: 320px; padding: 0 16px; font-family: "Segoe UI", sans-serif; } #panelBox { display: none; } #panelBox.show { display: flex; flex-direction: column; gap: 0px; background: var(--color-panel); backdrop-filter: blur(10px); color: var(--color-text-default); padding: 16px; border-radius: 14px; width: 100%; box-shadow: var(--color-panel-shadow); transition: transform 0.2s ease; } #panelBox:hover { transform: translateY(-2px); } #panelBox.minimized #arrowIcon { transform: rotate(180deg); } #panelBox.minimized { padding: 12px; overflow: hidden; } #panelHeader { display: flex; align-items: center; height: 30px; gap: 8px; } #panelHeader h2.panel-title { position: relative; bottom: 1px; font-size: var(--size-text-header-title); font-weight: 600; margin: 0; color: var(--color-text-default); } h2.panel-title, .panel-body-row, .panel-footer, .error-page { user-select: none; } .panel-body-row { margin: 0; font-size: var(--size-text-body-default); display: flex; align-items: center; justify-content: space-between; background: var(--color-panel-row); backdrop-filter: blur(12px); padding: 12px 16px; border-radius: 14px; } .panel-body-row + .panel-body-row { margin-top: 5px; } .panel-footer { height: 10px; display: flex; align-items: center; justify-content: flex-end; gap: 6px; font-size: var(--size-text-footer); color: var(--color-text-default); } .panel-page-container { position: relative; width: 100%; } hr.section-divider { flex-grow: 1; border: none; border-top: 1px solid var(--color-panel-divider); margin: 12px 0; } #minimizePanelButton { width: 28px; height: 28px; margin-left: auto; padding: 2px; border: none; background: none; display: flex; align-items: center; justify-content: center; cursor: pointer; box-sizing: content-box; } .lock-icon { width: 28px; height: 28px; padding: 4px; border-radius: 50%; fill: var(--color-app-icon); } .lock-icon-animation { position: absolute; top: 0; left: 0; width: 28px; height: 28px; opacity: 0; transition: opacity 0.4s ease, transform 0.4s ease; transform: rotate(0deg); } .lock-icon-animation.active { opacity: 1; transform: rotate(360deg); } #lockIconContainer { position: relative; width: 28px; height: 28px; } #arrowIcon { width: 28px; height: 28px; fill: var(--color-app-icon); transition: transform 0.3s ease; } .heart-icon { width: 10px; height: 10px; vertical-align: middle; fill: var(--color-app-icon); } .heart-icon:hover { fill: var(--color-app-icon-heart-hover); } #countBubble { background-color: var(--color-app-bubble); color: var(--color-text-default); font-size: 12px; font-weight: bold; padding: 3px 8px; border-radius: 999px; min-width: 20px; text-align: center; } .switch { position: relative; display: inline-block; width: 42px; height: 22px; } .switch input { opacity: 0; width: 0; height: 0; } .switch-label { margin-right: 10px; } .slider { position: absolute; cursor: pointer; top: 0; left: 0; right: 0; bottom: 0; background-color: var(--color-app-switch-off); transition: 0.3s; border-radius: 34px; } .slider:before { position: absolute; content: ""; height: 18px; width: 18px; top: 2px; left: 2px; background-color: var(--color-app-switch-thumb); transition: 0.3s; border-radius: 50%; } input:checked + .slider { background-color: var(--color-app-switch-on); } input:checked + .slider:before { transform: translateX(20px); } #creatorPage { color: var(--color-text-default); transition: color 0.3s ease; } #creatorPage:hover, .outbound-status-page:hover, .outbound-update-page:hover { color: var(--color-text-link-app-hover); } .error-page { text-align: center; font-size: var(--size-text-body-error); padding: 9px 0; } .error-page p:first-child { margin-bottom: 5px; } .error-page p:last-child { margin-top: 0; } .outbound-status-page, .outbound-update-page { text-decoration: underline; color: var(--color-text-link-default); } .outbound-status-page:visited, .outbound-update-page:visited { color: var(--color-text-link-default); } .seller-linked:hover { color: var(--color-text-link-ebay-hover) !important; } .sponsored-highlight { border: var(--size-thickness-highlight-border) dashed var(--color-highlight-border) !important; background-color: var(--color-highlight-background); } .sponsored-hidden { display: none !important; } .sponsored-hidden-banner { display: none !important; } .sponsored-hidden-carousel { display: none !important; } `; document.head.appendChild(style); } async function init() { observeURLMutation(); createStyles(); buildPanel(); hideOrShowPanel(); if (determineCarouselDetection()) { initCarouselObserver(); removeSponsoredCarousels(); } if (isResultsPage()) { initBannerObserver(); } await processSponsoredContent(); } function validateCurrentPage() { const url = new URL(location.href); const params = url.searchParams; const isSearchPage = /^https:\/\/([a-z0-9-]+\.)*ebay\.[a-z.]+\/(sch|shop)\//i.test(url.href); const isAdvancedSearchPage = url.href.includes("ebayadvsearch"); const isSellerPage = params.has("_ssn"); const isVisuallySimilarPage = params.get("_vss") === "1"; const isCompletedPage = params.get("LH_Complete") === "1"; const isSoldPage = params.get("LH_Sold") === "1"; return isSearchPage && !isAdvancedSearchPage && !isVisuallySimilarPage && !isSellerPage && !isCompletedPage && !isSoldPage; } function isResultsPage() { return /^https:\/\/([a-z0-9-]+\.)*ebay\.[a-z.]+\/sch\//i.test(location.href); } function isListingPage() { return /^https:\/\/([a-z0-9-]+\.)*ebay\.[a-z.]+\/itm\/\d+/.test(location.href); } function isCheckoutPage() { return /^https:\/\/cart(\.[a-z0-9-]+)?\.ebay\.[a-z.]+/.test(location.href); } function initObserver() { if (state.ui.observerInitialized) return; observer.observe(document.body, { childList: true, subtree: true, }); state.ui.observerInitialized = true; } function initBannerObserver() { const observer = new MutationObserver(() => { if (state.banner.bannerUpdateScheduled) return; state.banner.bannerUpdateScheduled = true; requestAnimationFrame(() => { state.banner.bannerUpdateScheduled = false; removeSponsoredBanners(document); }); }); observer.observe(document.body, { childList: true, subtree: true }); } function initCarouselObserver() { const carouselState = state.carousel; if (carouselState.carouselObserverInitialized) return; carouselState.carouselObserverInitialized = true; const targetContainer = document.querySelector('[data-results-container], main, .srp-results') || document.body; if (!targetContainer) return; const observer = new MutationObserver(() => { removeSponsoredCarousels(); }); observer.observe(targetContainer, { childList: true, subtree: true }); carouselState.carouselObserver = observer; } function initCleanListingObserver() { const observer = new MutationObserver(() => { cleanListingURLs(); }); observer.observe(document.body, { childList: true, subtree: true, attributes: true, attributeFilter: ['href'], }); } function observeURLMutation() { let previousURL = location.href; const handleURLChange = () => { const currentURL = location.href; if (currentURL !== previousURL) { previousURL = currentURL; hideOrShowPanel(); scheduleHighlightUpdate(); } }; ["pushState", "replaceState"].forEach(method => { const original = history[method]; history[method] = function(...args) { const result = original.apply(this, args); handleURLChange(); return result; }; }); window.addEventListener("popstate", handleURLChange); window.addEventListener("hashchange", handleURLChange); const observer = new MutationObserver(handleURLChange); observer.observe(document, { subtree: true, childList: true }); } async function buildPanel() { const wrapper = document.createElement("div"); wrapper.id = "panelWrapper"; const box = document.createElement("div"); box.id = "panelBox"; const header = buildPanelHeader(); const sponsoredCount = await processSponsoredContent(); const body = determinePanelState(sponsoredCount, state.ui.hidingEnabled); const footer = buildPanelFooter(); const createDivider = () => { const hr = document.createElement("hr"); hr.className = "section-divider"; return hr; }; box.append(header, createDivider(), body, createDivider(), footer); wrapper.appendChild(box); document.body.appendChild(wrapper); const minimizePanelButton = document.getElementById("minimizePanelButton"); if (minimizePanelButton) { minimizePanelButton.addEventListener("click", () => { const panelBox = document.getElementById("panelBox"); const newState = !panelBox.classList.contains("minimized"); localStorage.setItem("panelMinimized", newState); minimizePanel(newState); }); } const isPanelMinimized = localStorage.getItem("panelMinimized") === "true"; minimizePanel(isPanelMinimized); const toggleInput = document.getElementById("toggleSponsoredContentSwitch"); if (!toggleInput) { updateLockIcon(); return; } toggleInput.addEventListener("change", (e) => { state.ui.hidingEnabled = e.target.checked; localStorage.setItem(APP_KEY_HIDE_SPONSORED_CONTENT, state.ui.hidingEnabled); updateLockIcon(); scheduleHighlightUpdate(); }); updateLockIcon(); } function buildPanelHeader() { const header = document.createElement("div"); header.id = "panelHeader"; header.innerHTML = `
${APP_ICONS.locked} ${APP_ICONS.unlocked}

${APP_NAME}

`; return header; } function buildPanelFooter() { const footer = Object.assign(document.createElement("div"), { className: "panel-footer", }); const creatorLink = Object.assign(document.createElement("a"), { id: "creatorPage", href: "https://github.com/OsborneLabs/Spotless", target: "_blank", rel: "noopener noreferrer", textContent: "Osborne", }); creatorLink.style.textDecoration = "none"; const separator = Object.assign(document.createElement("span"), { textContent: " · ", }); const donateLink = Object.assign(document.createElement("a"), { href: "https://ko-fi.com/osbornelabs", target: "_blank", rel: "noopener noreferrer", innerHTML: APP_ICONS.heart, }); Object.assign(donateLink.style, { display: "inline-flex", alignItems: "center", justifyContent: "center", }); footer.append(creatorLink, separator, donateLink); return footer; } function buildPanelRow(innerHTML = "") { const row = document.createElement("div"); row.className = "panel-body-row"; row.innerHTML = innerHTML; return row; } function buildSponsoredCountRow() { const row = buildPanelRow(` Sponsored listings 0 `); row.id = "countSponsoredContentRow"; return row; } function buildSponsoredToggleRow() { const row = buildPanelRow(` Hide all sponsored `); row.id = "toggleSponsoredContentRow"; return row; } function buildPanelHomePage() { const pageContainer = Object.assign(document.createElement("div"), { id: "panelPagecontainer", className: "panel-page-container", }); const homePage = Object.assign(document.createElement("div"), { id: "homePage", className: "panel-page", }); homePage.style.display = "block"; const countRow = buildSponsoredCountRow(); const toggleRow = buildSponsoredToggleRow(); homePage.append(countRow, toggleRow); pageContainer.append(homePage); return pageContainer; } function buildPanelErrorPage() { const errorPage = Object.assign(document.createElement("div"), { className: "error-page panel-page", }); const message = document.createElement("p"); message.textContent = "Nothing sponsored found"; const links = document.createElement("p"); const updateLink = Object.assign(document.createElement("a"), { textContent: "Update", href: "https://greasyfork.org/en/scripts/541981-spotless-for-ebay", target: "_blank", rel: "noopener noreferrer", className: "outbound-update-page", }); const statusLink = Object.assign(document.createElement("a"), { textContent: "check status", href: "https://github.com/OsborneLabs/Spotless#ebay", target: "_blank", rel: "noopener noreferrer", className: "outbound-status-page", }); links.append(updateLink, " or ", statusLink); errorPage.append(message, links); return errorPage; } function hideOrShowPanel() { const panelBox = document.getElementById("panelBox"); if (!panelBox) return; const isSearchPage = validateCurrentPage(); if (isSearchPage) { panelBox.classList.add("show"); } else { panelBox.classList.remove("show"); } } function minimizePanel(minimized) { const panelBox = document.getElementById("panelBox"); if (!panelBox) return; const panelPage = panelBox.querySelector(".panel-page"); const sectionDivider = panelBox.querySelectorAll(".section-divider"); const panelFooter = panelBox.querySelector(".panel-footer"); panelBox.classList.toggle("minimized", minimized); if (panelPage) panelPage.style.display = minimized ? "none" : "block"; sectionDivider.forEach(el => { el.style.display = minimized ? "none" : ""; }); if (panelFooter) panelFooter.style.display = minimized ? "none" : ""; } function updateLockIcon() { const locked = document.getElementById("lockedIcon"); const unlocked = document.getElementById("unlockedIcon"); locked.classList.toggle("active", state.ui.hidingEnabled); unlocked.classList.toggle("active", !state.ui.hidingEnabled); } function determinePanelState(sponsoredCount) { if (!validateSponsoredCount(sponsoredCount)) { return buildPanelErrorPage(); } return buildPanelHomePage(); } function determineCarouselDetection() { return isCheckoutPage() || isListingPage(); } async function processSponsoredContent() { if (state.ui.isContentProcessing) return 0; state.ui.isContentProcessing = true; try { observer.disconnect(); resetSponsoredContent(); const detectedSponsoredElements = new Set(); const separatorSizeMethod = detectSponsoredListingBySeparatorSize(); separatorSizeMethod.forEach(li => detectedSponsoredElements.add(li)); if (detectedSponsoredElements.size === 0) { const ariaGroupMethod = detectSponsoredListingByAriaGroup(); ariaGroupMethod.forEach(li => detectedSponsoredElements.add(li)); } if (detectedSponsoredElements.size === 0) { const fontFamilyMethod = detectSponsoredListingByFontFamily(); fontFamilyMethod.forEach(li => detectedSponsoredElements.add(li)); } if (detectedSponsoredElements.size === 0) { const translateYMethod = detectSponsoredListingByTranslateY(); translateYMethod.forEach(li => detectedSponsoredElements.add(li)); } if (detectedSponsoredElements.size === 0) { const invertMethod = detectSponsoredListingByInvertStyle(); if (invertMethod?.elements) { invertMethod.elements.forEach(li => detectedSponsoredElements.add(li)); } } if (detectedSponsoredElements.size === 0) { const svgMethod = await detectSponsoredListingBySVG(); svgMethod.forEach(el => { const li = el.closest("li"); if (li) detectedSponsoredElements.add(li); }); } requestAnimationFrame(() => { const count = detectedSponsoredElements.size; const sponsoredDetectionValid = validateSponsoredCount(count); if (sponsoredDetectionValid) { for (const el of detectedSponsoredElements) { if (!el.hasAttribute("data-sponsored-processed")) { designateSponsoredContent(el); highlightSponsoredContent(el); hideShowSponsoredContent(el, state.ui.hidingEnabled); } } } createSellerLink(); removeSponsoredRibbons(); cleanListingURLs(); cleanGeneralURLs(); cleanGeneralClutter(); hideOrShowPanel(); countSponsoredContent(count); initObserver(); state.ui.isContentProcessing = false; }); return detectedSponsoredElements.size; } catch (err) { console.error(`${APP_NAME_DEBUG_MODE}: UNABLE TO PROCESS SPONSORED CONTENT, SEE CONSOLE ERROR\n`, err); state.ui.isContentProcessing = false; initObserver(); return 0; } } function validateSponsoredCount(sponsoredCount) { const listings = getListingElements(); const listingCount = listings.length; if (listingCount === 0) return false; const MIN_SPONSORED = 2; const MAX_PERCENT = 0.5; const sponsoredPercent = sponsoredCount / listingCount; return sponsoredCount >= MIN_SPONSORED && sponsoredPercent <= MAX_PERCENT; } function countSponsoredContent(count) { const countBubble = document.getElementById("countBubble"); if (countBubble) countBubble.textContent = count; } function getListingElements() { return Array.from(document.querySelectorAll("li[class*='s-']")).filter( (el) => el.className.split(/\s+/).some((cls) => /^s-[\w-]+$/.test(cls)) ); } function detectSponsoredListingBySeparatorSize() { const listings = getListingElements(); const sponsoredListings = []; listings.forEach(listing => { const separatorSpan = listing.querySelector('span.s-item__sep'); if (!separatorSpan) return; const innerSpan = separatorSpan.querySelector('span'); const width = innerSpan?.offsetWidth || 0; const height = innerSpan?.offsetHeight || 0; const isSponsored = width > 0 && height > 0; if (isSponsored) { sponsoredListings.push(listing); } }); return sponsoredListings; } function detectSponsoredListingByAriaGroup() { function generateAriaGroupLabel(num) { let letters = ''; do { letters = String.fromCharCode(65 + (num % 26)) + letters; num = Math.floor(num / 26) - 1; } while (num >= 0); return letters; } const listings = getListingElements(); const groupMap = {}; const ariaLabelToGroup = {}; let groupCounter = 0; listings.forEach(listing => { const labelSpan = listing.querySelector('span[aria-labelledby]'); if (!labelSpan) return; const ariaLabel = labelSpan.getAttribute('aria-labelledby'); if (!ariaLabel || !ariaLabel.includes("s-")) return; if (!ariaLabelToGroup[ariaLabel]) { ariaLabelToGroup[ariaLabel] = `Group ${generateAriaGroupLabel(groupCounter)}`; groupCounter++; } const group = ariaLabelToGroup[ariaLabel]; if (!groupMap[group]) { groupMap[group] = []; } groupMap[group].push(listing); }); let sponsoredGroup = null; let minCount = Infinity; for (const [group, listings] of Object.entries(groupMap)) { if (listings.length < minCount) { sponsoredGroup = group; minCount = listings.length; } } return sponsoredGroup ? groupMap[sponsoredGroup] : []; } function detectSponsoredListingByFontFamily() { const listings = getListingElements(); if (!listings.length) return new Set(); const groups = new Map(); const fontCache = new WeakMap(); for (const li of listings) { const badgeSpan = li.querySelector('div[role="heading"] > span[aria-hidden="true"]'); if (!badgeSpan) continue; const wrapper = badgeSpan.parentElement; if (!wrapper) continue; let fontFamily = fontCache.get(wrapper); if (!fontFamily) { fontFamily = getComputedStyle(wrapper).fontFamily; if (!fontFamily) continue; fontFamily = fontFamily .replace(/["']/g, "") .trim() .toLowerCase(); fontCache.set(wrapper, fontFamily); } if (!groups.has(fontFamily)) groups.set(fontFamily, []); groups.get(fontFamily).push(li); } let sponsoredGroup = null; let minSize = Infinity; for (const arr of groups.values()) { if (arr.length < minSize) { minSize = arr.length; sponsoredGroup = arr; } } return new Set(sponsoredGroup || []); } function detectSponsoredListingByTranslateY() { const listings = getListingElements(); if (!listings.length) return []; const listingCache = new WeakMap(); const groups = new Map(); const translateYMap = new Map(); for (const sheet of document.styleSheets) { let rules; try { rules = sheet.cssRules; } catch { continue; } if (!rules) continue; for (const rule of rules) { const sel = rule.selectorText; if (!sel) continue; const m = rule.cssText.match(/translateY\((-?\d+)\s*px\)/i); if (m) translateYMap.set(sel, Number(m[1])); } } for (const listing of listings) { if (listingCache.has(listing)) { const v = listingCache.get(listing); (groups.get(v) || groups.set(v, []).get(v)).push(listing); continue; } const span = listing.querySelector("span[role='heading']"); if (!span) continue; const cls = Array.from(span.classList).find(c => c.startsWith("s-")); if (!cls) continue; const selector = `span.${cls} > div`; const val = translateYMap.get(selector); if (val == null) continue; listingCache.set(listing, val); (groups.get(val) || groups.set(val, []).get(val)).push(listing); } let sponsoredGroup = null; let minCount = Infinity; for (const [val, arr] of groups.entries()) { if (arr.length < minCount) { minCount = arr.length; sponsoredGroup = val; } } return sponsoredGroup != null ? groups.get(sponsoredGroup) : []; } function detectSponsoredListingByInvertStyle() { const invertStyleMatch = /div\.([a-zA-Z0-9_-]+)(?:\s+div)?\s*\{[^}]*color:\s*(black|white);[^}]*filter:\s*invert\(([-\d.]+)\)/g; const sponsoredGroups = {}; const classToInvertMap = {}; const styleTags = Array.from(document.querySelectorAll("style")); styleTags.forEach(styleTag => { const css = styleTag.textContent; let match; while ((match = invertStyleMatch.exec(css)) !== null) { const [_, className, color, invertValue] = match; if (!classToInvertMap[className]) { classToInvertMap[className] = []; } classToInvertMap[className].push({ color, invert: parseFloat(invertValue) }); } }); const containers = Array.from(document.querySelectorAll('div[role="text"]')).filter(container => { return container.querySelector('div[aria-hidden="true"]'); }); containers.forEach(container => { const targetDiv = container.querySelector('div[aria-hidden="true"]'); if (!targetDiv) return; const ancestorDiv = container.closest("div[class*='_']"); if (!ancestorDiv) return; const classList = Array.from(ancestorDiv.classList); const dynamicClass = classList.find(cls => classToInvertMap[cls]); if (!dynamicClass) return; const candidates = classToInvertMap[dynamicClass]; const invertEntry = candidates?.[0]; if (!invertEntry) return; const key = invertEntry.invert; if (!sponsoredGroups[key]) { sponsoredGroups[key] = []; } sponsoredGroups[key].push(container); }); const groupEntries = Object.entries(sponsoredGroups); if (groupEntries.length === 0) { return { invert: null, elements: [], allGroups: [] }; } const sortedGroups = groupEntries.sort((a, b) => a[1].length - b[1].length); const [sponsoredInvert, sponsoredList] = sortedGroups[0]; const sponsoredElements = sponsoredList .map(container => container.closest("li")) .filter(Boolean); return { invert: parseFloat(sponsoredInvert), elements: sponsoredElements, allGroups: groupEntries }; } function detectSponsoredListingBySVG(batchSize = 10) { return new Promise((resolve) => { const listings = getListingElements(); const sponsoredElements = []; let index = 0; function processBatch() { const end = Math.min(index + batchSize, listings.length); const batch = listings.slice(index, end); let processedInBatch = 0; if (batch.length === 0) { resolve(sponsoredElements); return; } batch.forEach((listing) => { let svgDivSpan = listing.querySelector(".s-item__sep span[aria-hidden='true']"); let backgroundImage; if (svgDivSpan) { backgroundImage = getComputedStyle(svgDivSpan.parentElement).backgroundImage; } else { const svgDivB = listing.querySelector(".s-card__sep b[style*='data:image/svg+xml']"); if (!svgDivB) return done(); backgroundImage = getComputedStyle(svgDivB).backgroundImage; } const match = backgroundImage.match(/url\("data:image\/svg\+xml;base64,([^"]+)"\)/); if (!match || !match[1]) return done(); const base64 = match[1]; const svgString = atob(base64); const img = new Image(); const canvas = document.createElement("canvas"); const ctx = canvas.getContext("2d"); img.src = "data:image/svg+xml;base64," + btoa(svgString); img.onload = () => { canvas.width = img.naturalWidth || 20; canvas.height = img.naturalHeight || 20; ctx.drawImage(img, 0, 0); const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height).data; const colors = new Set(); const sampleWidth = 15; const sampleHeight = 15; for (let y = 0; y < sampleHeight && y < canvas.height; y++) { for (let x = 0; x < sampleWidth && x < canvas.width; x++) { const i = (y * canvas.width + x) * 4; const r = imageData[i]; const g = imageData[i + 1]; const b = imageData[i + 2]; const a = imageData[i + 3]; if (a > 0) { colors.add(`${r},${g},${b}`); if (colors.size > 1) break; } } if (colors.size > 1) break; } if (colors.size > 1) { sponsoredElements.push(listing); } done(); }; img.onerror = done; function done() { processedInBatch++; if (processedInBatch === batch.length) { index += batchSize; setTimeout(processBatch, 0); } } }); } if (listings.length === 0) { resolve([]); } else { processBatch(); } }); } function createSellerLink() { const listings = getListingElements(); listings.forEach(listing => { const sellerElements = listing.querySelectorAll(` .su-card-container__attributes__secondary .s-card__attribute-row, .s-card__program-badge-container--sellerOrStoreInfo `); sellerElements.forEach(el => { let username, feedback; const spans = el.querySelectorAll("span.su-styled-text"); if (spans.length >= 2) { username = spans[0].textContent.trim(); feedback = spans[1].textContent.trim(); el = spans[0].parentElement; } else { const combinedSpan = el.querySelector("span.su-styled-text.default"); if (!combinedSpan) return; const parts = combinedSpan.textContent.trim().split(/\s{2,}/); if (parts.length < 2) return; username = parts[0]; feedback = parts[1]; const wrapper = document.createElement("span"); const usernameSpan = document.createElement("span"); usernameSpan.textContent = username; const feedbackSpan = document.createElement("span"); feedbackSpan.textContent = feedback; wrapper.appendChild(usernameSpan); wrapper.append(" "); wrapper.appendChild(feedbackSpan); combinedSpan.replaceWith(wrapper); el = wrapper; } if (!username) return; if (el.querySelector("a.seller-linked")) return; const link = document.createElement("a"); link.href = `https://${window.location.hostname.replace(/^www\./, "")}/usr/${encodeURIComponent(username)}`; link.target = "_blank"; link.rel = "noopener noreferrer"; link.textContent = username; link.style.color = "var(--color-foreground-accent)"; link.style.textDecoration = "none"; link.classList.add("seller-linked"); const firstTextNode = el.firstChild; if (firstTextNode) firstTextNode.replaceWith(link); const feedbackSpan = Array.from(el.children).find(c => c.textContent === feedback); if (feedbackSpan && el.contains(feedbackSpan)) { const bullet = document.createElement("span"); bullet.textContent = " · "; bullet.style.padding = "0 0.15em"; el.insertBefore(bullet, feedbackSpan); feedbackSpan.style.marginLeft = "0"; } }); }); } function removeSponsoredBanners(root = document) { if (!(root instanceof Element || root instanceof Document)) { return; } const banners = Array.from( root.querySelectorAll( ".s-answer-region-center-top.s-answer-region > :not(.sponsored-hidden-banner)" ) ).filter(el => el.offsetHeight >= 140); banners.forEach(banner => { banner.classList.add("sponsored-hidden-banner"); }); } function removeSponsoredRibbons() { document.querySelectorAll('.x-breadcrumb, .x-pda-placements') .forEach(el => el.remove()); const whitelisted = ['statusmessage', 'x-alert', 'x-sme-atf', 'x-vi-evo-cvip-container']; const ribbons = document.querySelectorAll('.x-evo-atf-top-river.vi-grid.vim > .d-vi-evo-region.vim > div'); ribbons.forEach(ribbon => { const isWhitelisted = whitelisted.some(pattern => ribbon.classList.value.includes(pattern) ); if (!isWhitelisted) ribbon.remove(); }); } function removeSponsoredCarousels() { if (!determineCarouselDetection()) return; const CAROUSEL_SPONSORED_KEYWORDS = [ 'sponsored', 'anzeige', 'gesponsord', 'patrocinado', 'sponsorisé', 'sponsorizzato', 'sponsorowane', '助贊' ]; const normalizeText = (text) => text.trim().normalize("NFKC").replace(/[\u200B-\u200D\u061C\uFEFF]/g, '').toLowerCase(); const labelSponsored = (carousel) => { carousel.classList.add('sponsored-hidden-carousel'); removeSiteTelemetry(carousel); }; document.querySelectorAll('[class*="x-atc-layer"][class*="--ads"]').forEach(el => el.remove()); let sponsoredCarouselDetected = false; const carousels = document.querySelectorAll('[data-viewport]'); carousels.forEach(carousel => { if (carousel.classList.contains('sponsored-hidden-carousel')) return; if (carousel.closest('.lightbox-dialog, .ux-overlay, [role="dialog"]')) return; const titleElements = carousel.querySelector('h2, h3, h4'); if (titleElements && CAROUSEL_SPONSORED_KEYWORDS.some(kw => normalizeText(titleElements.textContent).includes(kw))) { labelSponsored(carousel); sponsoredCarouselDetected = true; return; } const textElements = Array.from(carousel.querySelectorAll('div, span')); if (textElements.some(el => CAROUSEL_SPONSORED_KEYWORDS.some(kw => normalizeText(el.textContent).includes(kw)))) { labelSponsored(carousel); sponsoredCarouselDetected = true; return; } const characters = textElements .map(el => normalizeText(el.textContent)) .filter(t => t.length === 1 && /^\p{L}$/u.test(t)); if (CAROUSEL_SPONSORED_KEYWORDS.some(kw => { let matchIndex = 0; for (const char of characters) { if (char === kw[matchIndex]) { matchIndex++; if (matchIndex === kw.length) return true; } } return false; })) { labelSponsored(carousel); sponsoredCarouselDetected = true; } }); if (isCheckoutPage() && sponsoredCarouselDetected) { const selector = document.querySelectorAll('.merch-container'); selector.forEach(el => el.remove()); } if (isCheckoutPage()) { const selector = document.querySelectorAll('.dynamic-banner'); selector.forEach(el => el.remove()); } } function removeSiteTelemetry(context = document) { const selector = '[trackableid], [trackablemoduleid]'; const removeTrackingAttributes = (el) => { el.removeAttribute('trackableid'); el.removeAttribute('trackablemoduleid'); }; context.querySelectorAll('[data-viewport]').forEach((el) => { el.setAttribute('data-viewport', '{}'); const trackedElements = el.matches(selector) ? [el, ...el.querySelectorAll(selector)] : el.querySelectorAll(selector); trackedElements.forEach(removeTrackingAttributes); }); context.querySelectorAll('li[data-viewport]').forEach((li) => { li.setAttribute('data-viewport', '{}'); li.removeAttribute('data-listingid'); li.removeAttribute('data-view'); li.removeAttribute('id'); const trackedElements = li.matches(selector) ? [li, ...li.querySelectorAll(selector)] : li.querySelectorAll(selector); trackedElements.forEach(removeTrackingAttributes); }); const telemetryAttributesRegex = [ /^data-atf/i, /^data-gr\d$/i, /^data-s-[a-z0-9]+$/i ]; const TELEMETRY_ATTRIBUTES = [ '_sp', 'data-click', 'data-clientpresentationmetadata', 'data-config', 'data-defertimer', 'data-ebayui', 'data-testid', 'data-track', 'data-tracking', 'data-vi-scrolltracking', 'data-vi-tracking', 'modulemeta' ]; context.querySelectorAll('*').forEach((el) => { Array.from(el.attributes).forEach(({ name }) => { if (telemetryAttributesRegex.some((rx) => rx.test(name)) || TELEMETRY_ATTRIBUTES.includes(name)) { el.removeAttribute(name); } }); }); context.querySelectorAll('img[onerror]').forEach((img) => { img.removeAttribute('onerror'); }); } function designateSponsoredContent(el) { el.setAttribute("data-sponsored", "true"); el.setAttribute("data-sponsored-processed", "true"); state.ui.highlightedSponsoredContent.push(el); } function resetSponsoredContent() { const elements = state.ui.highlightedSponsoredContent; elements.forEach(el => { el.classList.remove("sponsored-hidden"); el.removeAttribute("data-sponsored"); el.removeAttribute("data-sponsored-processed"); el.style.border = ""; el.style.backgroundColor = ""; }); elements.length = 0; } function highlightSponsoredContent(element) { element.setAttribute("data-sponsored", "true"); element.classList.add("sponsored-highlight"); } function hideShowSponsoredContent(element, hide) { element.classList.toggle("sponsored-hidden", hide); } function scheduleHighlightUpdate() { if (state.ui.updateScheduled || state.ui.isContentProcessing) return; state.ui.updateScheduled = true; requestAnimationFrame(() => { processSponsoredContent().finally(() => { state.ui.updateScheduled = false; }); }); } function cleanListingURLs() { const url = /^https:\/\/((?:[a-z0-9-]+\.)*)ebay\.([a-z.]+)\/itm\/(\d+)(?:[/?#].*)?/i; const links = document.querySelectorAll("a[href*='/itm/']"); links.forEach((link) => { const match = link.href.match(url); if (match) { let subdomains = match[1] || ""; const tld = match[2]; const itemId = match[3]; const parts = subdomains.split(".").filter(Boolean).filter(p => p !== "www"); const subdomain = parts.length ? parts.join(".") + "." : ""; const cleanURL = `https://${subdomain}ebay.${tld}/itm/${itemId}`; if (link.href !== cleanURL) { link.href = cleanURL; } } link.removeAttribute("data-interactions"); }); removeSiteTelemetry(); } function cleanGeneralURLs() { const GENERAL_TRACKING_KEYWORDS = [ "_from", "_odkw", "_osacat", "_trksid", "campaign", "campid", "mkcid", "mkevt", "mkrid", "promoted_items", "siteid", "source", "sr", "templateId", "toolid" ]; document.querySelectorAll("a[href*='ebay.']").forEach(link => { try { const url = new URL(link.href); if (!/(^|\.)ebay\.[a-z.]+$/i.test(url.hostname)) return; if (/^\/sch\/[^/]+\/m\.html$/i.test(url.pathname)) { url.search = ""; } else { const params = url.searchParams; GENERAL_TRACKING_KEYWORDS.forEach(key => params.delete(key)); for (const key of Array.from(params.keys())) { if (key.startsWith("utm_") || key.startsWith("_trk")) { params.delete(key); } } } const cleanURL = `${url.origin}${url.pathname}${url.search}${url.hash}`; if (link.href !== cleanURL) link.href = cleanURL; } catch {} }); document.querySelectorAll("form[action*='ebay.'], form[action='/sch/i.html']").forEach(form => { const cleanFormInputs = () => { form.querySelectorAll(GENERAL_TRACKING_KEYWORDS.map(k => `[name='${k}']`).join(",")).forEach(input => input.remove()); }; cleanFormInputs(); form.addEventListener("submit", () => cleanFormInputs(), true); }); } function cleanGeneralClutter() { const selector = [ '.d-sell-now--filmstrip-margin', '.madrona-banner', '.s-faq-list', '.s-feedback', '[class*="BOS_PLACEHOLDER"]', '[class*="EBAY_LIVE_ENTRY"]', '[class*="FAQ_KW_SRP_MODULE"]', '[class*="START_LISTING_BANNER"]' ]; const elements = document.querySelectorAll(selector.join(',')); elements.forEach(el => el.remove()); } const observer = new MutationObserver(() => { hideOrShowPanel(); scheduleHighlightUpdate(); }); if (document.readyState === 'loading') { document.addEventListener('DOMContentLoaded', initCleanListingObserver); } else { initCleanListingObserver(); } window.addEventListener("storage", ({ key, newValue }) => { if (key === APP_KEY_HIDE_SPONSORED_CONTENT) { const isEnabled = newValue === "true"; if (isEnabled === state.ui.hidingEnabled) return; state.ui.hidingEnabled = isEnabled; const toggleInput = document.getElementById("toggleSponsoredContentSwitch"); if (toggleInput) toggleInput.checked = isEnabled; updateLockIcon(); scheduleHighlightUpdate(); return; } if (key === "panelMinimized") { minimizePanel(newValue === "true"); } }); const initAfterDOM = async () => { init(); }; if (["complete", "interactive"].includes(document.readyState)) { initAfterDOM(); } else { window.addEventListener("DOMContentLoaded", initAfterDOM); } })();