// ==UserScript== // @name FB Mobile - Clean my feeds // @namespace Violentmonkey Scripts // @version 1.00 // @description Removes Sponsored and Suggested posts from Facebook mobile chromium/react version // @license GNU General Public License v3.0 // @author https://github.com/webdevsk // @match https://m.facebook.com/* // @match https://www.facebook.com/* // @match https://touch.facebook.com/* // @icon  // @grant GM_getValue // @grant GM_setValue // @grant GM_addStyle // @grant GM_addValueChangeListener // @grant GM_removeValueChangeListener // @run-at document-end // @downloadURL https://update.greasyfork.icu/scripts/479868/FB%20Mobile%20-%20Clean%20my%20feeds.user.js // @updateURL https://update.greasyfork.icu/scripts/479868/FB%20Mobile%20-%20Clean%20my%20feeds.meta.js // ==/UserScript== // src/config.ts var devMode = false; var pathnameMatches = ["/"]; var bodyId = "app-body"; var screenRootSelector = "#screen-root"; var routeNodeSelector = "[data-screen-id]:first-child"; var postContainerSelector = "[data-pull-to-refresh-action-id]"; var possibleTargetsSelectorInPost = "span.f2:not(.a), span.f5, [style^='margin-top:9px; height:21px'] > .native-text"; var navBarSelector = "[role='tablist']"; var showPlaceholder = true; var theme = { textColor: "ffffff", iconColor: "e4e6eb", bgClassName: "bg-fallback", iconBgClassName: "icon-bg-fallback" }; // src/lib/block-counter.ts class BlockCounter { static instance = null; elm = null; whitelisted = 0; blacklisted = 0; constructor() {} static getInstance() { if (!BlockCounter.instance) { BlockCounter.instance = new BlockCounter; } return BlockCounter.instance; } register() { if (!this.elm) { this.elm = document.createElement("div"); this.elm.id = "block-counter"; document.body.appendChild(this.elm); if (devMode) console.log("block counter register successful"); } else if (!document.body.contains(this.elm)) { document.body.appendChild(this.elm); if (devMode) console.log("block counter register successful"); } this.render(); return () => this.destroy(); } render() { if (this.elm) { this.elm.innerHTML = `

Whitelisted: ${this.whitelisted}

Blacklisted: ${this.blacklisted}

`; } } destroy() { if (this.elm && document.body.contains(this.elm)) { this.elm.remove(); } } increaseWhite() { this.whitelisted += 1; this.render(); } increaseBlack() { this.blacklisted += 1; this.render(); } } // src/lib/make-navbar-sticky.ts var makeNavbarSticky = () => { const navbar = document.querySelector(navBarSelector); const screenRoot = document.querySelector(screenRootSelector); navbar.classList.add(theme.bgClassName); Object.assign(navbar.style, { position: "sticky", top: "-1px", zIndex: "1", insetInline: "0" }); Object.assign(screenRoot.style, { overflow: "visible" }); }; // src/lib/menu-buttons-injector.ts class MenuButtonsInjector { static instance = null; buttonsInjected = false; buttonElements = []; constructor() {} static getInstance() { if (!MenuButtonsInjector.instance) { MenuButtonsInjector.instance = new MenuButtonsInjector; } return MenuButtonsInjector.instance; } inject() { if (this.buttonsInjected) { console.warn("Menu buttons already injected"); return () => { this.destroy(); }; } this.injectButtons(); this.buttonsInjected = true; if (devMode) console.log("Menu buttons injected successfully"); return () => { this.destroy(); }; } setupTabBarStyles() { const tabBarEle = document.querySelector('[role="tablist"]'); if (tabBarEle) { tabBarEle.style.position = "sticky"; tabBarEle.style.zIndex = "1"; tabBarEle.style.top = "0"; } } createButton(id, imgSrc) { const button = document.createElement("div"); button.id = id; button.className = "customBtns"; button.innerHTML = `
`; return button; } injectButtons() { const titleBarEle = document.querySelector(".filler")?.nextElementSibling; if (!titleBarEle || !(titleBarEle instanceof HTMLElement)) { if (devMode) console.error("Title bar element not found"); return; } const innerScreenText = document.querySelector("#screen-root .fixed-container.top .f2")?.textContent || ""; if (innerScreenText) return; this.destroy(); this.buttonElements = []; if (!document.getElementById("settingsBtn")) { const settingsBtn = this.createButton("settingsBtn", "https://static.xx.fbcdn.net/rsrc.php/v4/yC/r/FgGUIEUfnev.png"); titleBarEle.after(settingsBtn); this.buttonElements.push(settingsBtn); } if (!document.getElementById("feedsBtn")) { const feedsBtn = this.createButton("feedsBtn", "https://static.xx.fbcdn.net/rsrc.php/v4/yB/r/Bc4BAjXDBat.png"); titleBarEle.after(feedsBtn); this.buttonElements.push(feedsBtn); } } destroy() { if (!this.buttonsInjected) return; this.buttonElements.forEach((button) => { if (devMode) console.log("Removing button:", button); button.remove(); }); this.buttonElements = []; this.buttonsInjected = false; } } // src/utils/watch-for-selectors.ts function watchForSelectors(selectors, callback, options = {}) { if (!Array.isArray(selectors)) throw new Error("watchForSelectors: Selectors must be an array"); if (!callback || typeof callback !== "function") throw new Error("watchForSelectors: Callback must be a function"); if (typeof options !== "object") throw new Error("watchForSelectors: Options must be an object"); if ("resolver" in options && typeof options.resolver !== "function") throw new Error("watchForSelectors: Resolver must be a function that resolves to boolean"); const elements = selectors.map((selector) => document.querySelector(selector)); if (options.resolver?.(elements) ?? elements.every((elm) => elm !== null)) { callback(); return () => null; } const observer = new MutationObserver((_, observer2) => { const elements2 = selectors.map((selector) => document.querySelector(selector)); if (options.resolver?.(elements2) ?? elements2.every((elm) => elm !== null)) { observer2.disconnect(); callback(); } }); observer.observe(options.target ?? document, options.observerOptions ?? { childList: true, subtree: true }); options.signal?.addEventListener("abort", () => { observer.disconnect(); }, { once: true }); return observer.disconnect.bind(observer); } function watchForSelectorsPromise(selectors, options = {}) { return new Promise((resolve, reject) => { watchForSelectors(selectors, resolve, options); options.signal?.addEventListener("abort", () => reject(new Error("Aborted by signal"))); }); } // src/lib/on-ready-for-scripting.ts var onReadyForScripting = async (cb) => { let cleanupFn = null; const main = () => { if (devMode) console.log("Main node found. Running cleanup function and restarting..."); cleanupFn?.(); cleanupFn = cb(); }; await watchForSelectorsPromise([screenRootSelector]); onNavigation((routeNode) => { if (devMode) console.log("onNavigation callback called"); if (!pathnameMatches.some((pathname) => pathname === location.pathname)) { if (devMode) console.log("Not on main page. Terminating..."); return () => null; } if (document.querySelector(postContainerSelector)) main(); const observer = new MutationObserver((mutationList) => { if (devMode) console.log("Main node detector mutation detected"); for (const mutation of mutationList) { for (const node of mutation.addedNodes) { if (node instanceof HTMLElement && node.nodeType === Node.ELEMENT_NODE && node.matches(postContainerSelector)) { main(); } } } }); observer.observe(routeNode, { childList: true }); return () => { cleanupFn?.(); cleanupFn = null; observer.disconnect(); }; }); }; var onNavigation = async (cb) => { let cleanupFn = null; const screenRoot = document.querySelector(screenRootSelector); const initialRouteNode = screenRoot.querySelector(routeNodeSelector); if (initialRouteNode && initialRouteNode instanceof HTMLElement) { cleanupFn = cb(initialRouteNode); } new MutationObserver((mutationList) => { if (devMode) console.log("Running navigation mutation"); for (const mutation of mutationList) { for (const node of mutation.addedNodes) { if (node instanceof HTMLElement && node.nodeType === Node.ELEMENT_NODE && node.matches(routeNodeSelector)) { if (devMode) console.log("Navigation detected, ", node); cleanupFn?.(); cleanupFn = cb(node); } } } }).observe(screenRoot, { childList: true }); }; // src/lib/register-auto-reload-after-idle.ts function registerAutoReloadAfterIdle(minutes = 15) { let leaveTime = null; let ctrl = new AbortController; if (devMode) console.log("Auto reload after idle registered"); document.addEventListener("visibilitychange", () => { if (document.hidden) { leaveTime = new Date; } else { if (!leaveTime) return; const currentTime = new Date; const timeDiff = (currentTime.getTime() - leaveTime.getTime()) / 60000; if (timeDiff > minutes) location.reload(); } }, { signal: ctrl.signal }); return () => { ctrl.abort(); ctrl = new AbortController; if (devMode) console.log("Auto reload after idle unregistered", ctrl.signal); ctrl.signal.throwIfAborted(); }; } // src/data/keywords-per-language.ts var keywordsPerLanguage = { placeholderMessage: { "en-US": "Removed", en: "Removed", bn: "বাতিল" }, suggested: { "en-US": "Suggested", en: "Suggested", bn: "আপনার জন্য প্রস্তাবিত" }, sponsored: { "en-US": "Sponsored", en: "Sponsored", bn: "স্পনসর্ড" }, uncategorized: { "en-US": ["Join", "Follow"], en: ["Join", "Follow"], bn: ["ফলো করুন", "যোগ দিন"] }, peopleYouMayKnow: { "en-US": "People You May Know", en: "People You May Know" }, reels: { "en-US": "Reels", en: "Reels" } }; // src/data/filters-database.ts var filtersDatabase = { suggested: { title: "Suggested", description: "Removes un-needed algorithm suggested posts", icon: "\uDB86\uDD01", keywordsDB: keywordsPerLanguage.suggested }, sponsored: { title: "Sponsored", description: "Removes annoying ads", icon: "\uDB86\uDC11", keywordsDB: keywordsPerLanguage.sponsored }, reels: { title: "Reels", description: "Removes annoying short videos", icon: "\uDB80\uDF83", keywordsDB: keywordsPerLanguage.reels }, peopleYouMayKnow: { title: "People You May Know", description: "Removes suggested friends", icon: "\uDB80\uDF8D", keywordsDB: keywordsPerLanguage.peopleYouMayKnow }, uncategorized: { title: "Uncategorized", description: "Removes suggested pages with join/follow link", icon: "\uDB86\uDC02", keywordsDB: keywordsPerLanguage.uncategorized } }; // src/lib/get-own-language-filters.ts var getOwnLangFilters = (obj) => navigator.languages.flatMap((lang) => obj[lang]); // src/lib/purge-element.ts var purgeElement = ({ element, reason, author, placeHolderMessage, sponsoredFilters }) => { element.tabIndex = -1; element.dataset.purged = "true"; if (showPlaceholder && !sponsoredFilters.includes(reason)) { element.dataset.actualHeight = "32"; element.classList.add(theme.bgClassName); element.style.height = "2rem"; const overlay = document.createElement("article"); overlay.className = "placeholder"; overlay.innerHTML = `

${placeHolderMessage}: ${author} (${reason})

`; element.appendChild(overlay); } else { element.dataset.actualHeight = "0"; element.dataset.forceHide = "true"; element.style.height = "0rem"; const { previousElementSibling: prevElm } = element; if (!(prevElm && prevElm instanceof HTMLElement) || prevElm.dataset.actualHeight !== "1") return; prevElm.style.marginTop = "0px"; prevElm.style.height = "0px"; prevElm.dataset.actualHeight = "0"; } for (const image of element.querySelectorAll("img")) { image.dataset.src = image.src; image.removeAttribute("src"); image.dataset.nulled = "true"; } }; // src/lib/spinner.ts class Spinner { static instance = null; elm = null; isVisible = false; constructor() {} static getInstance() { if (!Spinner.instance) { Spinner.instance = new Spinner; } return Spinner.instance; } register() { if (!this.elm) { this.elm = document.createElement("div"); this.elm.id = "spinner"; this.elm.innerHTML = `
`; document.body.appendChild(this.elm); if (devMode) console.log("Spinner register successful"); } else if (!document.body.contains(this.elm)) { document.body.appendChild(this.elm); if (devMode) console.log("Spinner register successful"); } return () => this.destroy(); } destroy() { if (this.elm && document.body.contains(this.elm)) { this.elm.remove(); } } show() { this.isVisible = true; if (this.elm) { this.elm.style.display = "block"; } } hide() { this.isVisible = false; if (this.elm) { this.elm.style.display = "none"; } } } // src/lib/whitelisted-filters-storage.ts class WhitelistedFiltersStorage { storageKey = "whitelisted-filters"; listenerId = null; defaultValue = []; cache = []; static instance = null; static getInstance() { if (!WhitelistedFiltersStorage.instance) { WhitelistedFiltersStorage.instance = new WhitelistedFiltersStorage; } return WhitelistedFiltersStorage.instance; } register() { const initialValue = GM_getValue(this.storageKey, this.defaultValue); this.update(initialValue); if (devMode) console.log("WhitelistedFiltersStorage register successful"); return this.onChange(); } update(value) { if (Array.isArray(value) && value.every((val) => typeof val === "string")) { this.cache = value; } else { this.cache = this.defaultValue; this.set(this.defaultValue); } } get() { return this.cache; } set(filters) { if (devMode) console.log("Set new filters", filters); GM_setValue(this.storageKey, filters); } onChange(cb) { this.listenerId = GM_addValueChangeListener(this.storageKey, (_name, _oldValue, newValue, _remote) => { if (devMode) console.log("New filter value", newValue); this.update(newValue); cb?.(this.get()); }); return () => { if (this.listenerId) GM_removeValueChangeListener(this.listenerId); }; } } // src/lib/run-feeds-cleaner.ts var runFeedsCleaner = () => { if (devMode) console.log("navigator.languages", navigator.languages); const root = document.querySelector(postContainerSelector); const whitelistedStorageInstance = WhitelistedFiltersStorage.getInstance(); const allFilters = Object.keys(filtersDatabase); const whitelistedFilters = whitelistedStorageInstance.get(); const activeFilters = []; const setActiveFilters = (whitelistedFilters2) => { activeFilters.length = 0; activeFilters.push(...new Set(allFilters.flatMap((filter) => whitelistedFilters2.includes(filter) ? [] : getOwnLangFilters(filtersDatabase[filter].keywordsDB)).filter((d) => d))); }; setActiveFilters(whitelistedFilters); whitelistedStorageInstance.onChange(setActiveFilters); const sponsoredFilters = getOwnLangFilters(filtersDatabase.sponsored.keywordsDB); const placeHolderMessage = getOwnLangFilters(keywordsPerLanguage.placeholderMessage)[0]; const checkElement = (element) => { if (element.dataset.purged === "true") return; let suspect = false; let reason = null; let raw = null; for (const span of element.querySelectorAll(possibleTargetsSelectorInPost)) { if (!activeFilters.some((str) => span.textContent?.includes(str))) continue; suspect = true; reason = span.innerHTML.split("\uDB81\uDF8B")[0]; raw = span.innerHTML; break; } if (!suspect) { BlockCounter.getInstance().increaseWhite(); return; } BlockCounter.getInstance().increaseBlack(); purgeElement({ element, reason: reason ?? raw ?? "", author: element.querySelector("span.f2")?.innerHTML ?? "", placeHolderMessage, sponsoredFilters }); }; if (devMode) console.log("Initial checking"); for (const element of root.querySelectorAll("[data-tracking-duration-id]")) { checkElement(element); } if (devMode) console.log("Initial checking done"); if (devMode) console.log("Mutation observer setup for new posts"); const observer = new MutationObserver((mutations) => { for (const mutation of mutations) { if (mutation.addedNodes.length === 0) continue; if (devMode) console.log("Checking posts count:", mutation.addedNodes.length); Spinner.getInstance().show(); for (const element of mutation.addedNodes) { if (!(element instanceof HTMLElement) || element.nodeType !== Node.ELEMENT_NODE) continue; if (!element.hasAttribute("data-tracking-duration-id")) continue; checkElement(element); } Spinner.getInstance().hide(); } }); observer.observe(root, { childList: true }); if (devMode) console.log("Mutation observer setup for new posts done on", root); return () => { observer.disconnect(); if (devMode) console.log("Mutation observer for posts disconnected"); }; }; // src/lib/settings-menu-injector.ts var closeMenuIcon = "\uDB85\uDE73"; class SettingsMenuInjector { static instance; ctrl; overlayId = "settingsOverlay"; constructor() { this.ctrl = new AbortController; } static getInstance() { if (!SettingsMenuInjector.instance) { SettingsMenuInjector.instance = new SettingsMenuInjector; } return SettingsMenuInjector.instance; } generateSettingsOverlay() { return `
FB Mobile - Clean my feeds
Mark filters to hide posts
${Object.entries(filtersDatabase).map(([filterType, item]) => { return ` `; }).join(` `).concat(`
${closeMenuIcon}
Close Menu
`)}
`; } handleCheckboxChange = (event) => { const target = event.target; if (!target.matches(`#${this.overlayId} input[type="checkbox"]`)) return; const { name, checked } = target; if (!Object.keys(filtersDatabase).includes(name)) return; const whiteListedFilters = WhitelistedFiltersStorage.getInstance().get(); const isWhiteListed = whiteListedFilters.includes(name); if (!checked === isWhiteListed) return; WhitelistedFiltersStorage.getInstance().set(isWhiteListed ? whiteListedFilters.filter((filter) => filter !== name) : [...whiteListedFilters, name]); }; handleDocumentClick = (event) => { const target = event.target; if (target.matches("#settingsBtn")) { this.show(); } else if (target.matches("#closeMenuTile")) { this.hide(); } else if (target.matches("#feedsBtn")) { document.querySelector('[aria-label="Facebook Menu"]')?.click(); watchForSelectors(['[aria-label="Feeds"]'], () => { document.querySelector('[aria-label="Feeds"]').click(); }, { signal: this.ctrl.signal, target: document.querySelector(screenRootSelector) }); } }; setupEventListeners() { document.addEventListener("click", this.handleDocumentClick, { signal: this.ctrl.signal }); document.addEventListener("change", this.handleCheckboxChange, { signal: this.ctrl.signal }); } destroyEventListeners() { this.ctrl.signal.throwIfAborted(); this.ctrl.abort(); } inject() { if (devMode) console.log("SettingsMenuInjector inject called"); this.ctrl = new AbortController; this.setupEventListeners(); if (devMode) console.log("SettingsMenuInjector inject successful"); return () => { if (devMode) console.log("SettingsMenuInjector cleanup called"); this.hide(); this.destroyEventListeners(); }; } show() { if (devMode) console.log("SettingsMenuInjector show called"); if (document.getElementById(this.overlayId)) return; document.body.insertAdjacentHTML("beforeend", this.generateSettingsOverlay()); } hide() { if (devMode) console.log("SettingsMenuInjector hide called"); document.getElementById(this.overlayId)?.remove(); } } // src/lib/updateThemeConfigWhenPossible.ts var updateThemeConfigWhenPossible = () => watchForSelectors([ ".native-text:last-child", '[role="tablist"]>*:last-child .native-text', '[aria-label="Search Facebook"] [class*="bg-"]' ], () => { const bgClassName = document.querySelector('[role="tablist"]>*:last-child').classList.values().find((v) => v.startsWith("bg-")); const iconBgClassName = document.querySelector('[aria-label="Search Facebook"] [class*="bg-"]').classList.values().find((v) => v.startsWith("bg-")); if (!bgClassName || !iconBgClassName) return; theme.bgClassName = bgClassName; theme.iconBgClassName = iconBgClassName; theme.textColor = getComputedStyle(document.querySelector(".native-text:last-child")).color; theme.iconColor = getComputedStyle(document.querySelector('[role="tablist"]>*:last-child .native-text')).color; if (devMode) console.log("Theme assignment successful"); }, { target: document.querySelector(postContainerSelector) }); // src/styles/style.css var style_default = '.dialog-screen{background-color:#0000007f;flex-direction:column;justify-content:flex-end;display:flex;position:fixed;inset:0;overflow-y:auto}.settings-container{padding-block:2rem;padding-inline:.5rem;position:relative;& .settings-header{flex-direction:column;gap:.5rem;margin-bottom:1rem;padding:.5rem 1rem;display:flex;& .settings-title{font-size:1.5rem;font-weight:600}}& .settingsItem{grid-template-columns:max-content minmax(0,1fr) max-content;align-items:center;gap:.75rem;min-height:2.5rem;padding:.5rem;display:grid;& *{pointer-events:none}& .settingsIcon{font-size:1.5rem}& .settingsLabel{font-size:1rem;font-weight:600;display:block}& .settingsDescription{white-space:nowrap;text-overflow:ellipsis;width:100%;display:block;overflow:hidden}}}.bg-fallback:before{content:"";z-index:-1;background-color:#242526;width:100%;height:100%;position:absolute;top:0;left:0}.icon-bg-fallback:before{content:"";left:calc((100% - var(--diameter))/2);top:calc((100% - var(--diameter))/2);width:var(--diameter);height:var(--diameter);z-index:-1;background-color:#ffffff1a;border-radius:50%;position:absolute}.fb-check{cursor:pointer;user-select:none;vertical-align:middle;align-items:center;font-family:sans-serif;display:inline-flex;& input{display:none}& .checkmark{border:2px solid;border-radius:3px;width:18px;height:18px;transition:background-color .2s;position:relative}& input:checked+.checkmark{background-color:#1877f2}& input:checked+.checkmark:after{content:"";border:2px solid #fff;border-width:0 2px 2px 0;width:5px;height:10px;position:absolute;top:0;left:4px;transform:rotate(45deg)}}.customBtns{z-index:2;width:45px;height:43px;position:absolute;pointer-events:all!important;&#settingsBtn{left:calc(100dvw - 138px)}&#feedsBtn{left:calc(100dvw - 180px)}&>div{z-index:0;--diameter:35px;box-sizing:border-box;flex-direction:column;flex-shrink:0;width:35px;height:35px;margin-top:4px;margin-left:5px;display:flex;position:relative}& img{filter:grayscale();object-fit:contain;width:100%;height:100%;position:absolute}}div[data-purged=true]{pointer-events:none;height:32px;position:relative;overflow-y:hidden;&[data-force-hide=true]{pointer-events:none;height:0;overflow-y:hidden}&>div{display:none!important}& .placeholder{pointer-events:auto;place-items:center;padding-inline:.5rem;display:grid;position:absolute;inset:0;& p{text-overflow:ellipsis;white-space:nowrap;text-align:center;width:100%;overflow:hidden}}}[data-screen-id]:first-child [data-comp-id~="22222"]{display:none!important}#block-counter{z-index:99;color:#ddd;pointer-events:none;background:#323436;border-radius:.2rem;flex-wrap:wrap;gap:.5rem;padding:.5rem 1rem;font-size:.8rem;display:flex;position:fixed;top:0;left:0}#spinner{pointer-events:none;z-index:100;position:fixed;top:20px;left:16px}'; // src/index.ts (() => { if (document.body.id !== bodyId) { console.error("ID 'app-body' not found."); return; } if (!document.head.contains(GM_addStyle(style_default))) console.error("Failed to add style node"); onReadyForScripting(() => { console.log("Ready for scripting"); const aborts = [ updateThemeConfigWhenPossible(), ...devMode ? [BlockCounter.getInstance().register()] : [], WhitelistedFiltersStorage.getInstance().register(), MenuButtonsInjector.getInstance().inject(), SettingsMenuInjector.getInstance().inject(), runFeedsCleaner(), registerAutoReloadAfterIdle() ]; makeNavbarSticky(); return () => { console.log("Not Ready for scripting"); aborts.forEach((abort) => abort?.()); aborts.length = 0; }; }); })();