您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Filters posts on any lemmy instance by text in the title. It can also auto-open image posts, unblur thumbnails, and other things.
当前为
// ==UserScript== // @name Lemmy post utilities - Filter posts by title // @namespace Violentmonkey Scripts // @description Filters posts on any lemmy instance by text in the title. It can also auto-open image posts, unblur thumbnails, and other things. // @match https://*lemmy*.*/ // @include https://burggit.moe/* // @include https://lemmit.online/* // @include https://yiffit.net/* // @include https://reddthat.com/* // @include https://sh.itjust.works/* // @grant GM_getValue // @grant GM_setValue // @version 2.0.1 // @icon  // @author Xynoth // @license GPT-3 // @date 14/7/2023, 20:15:03 // ==/UserScript== //---------------------------------------- // CONSTANTS //---------------------------------------- // Key constants const BLOCKED_TAGS_KEY = "blockedTags"; const BLOCKED_CASE_SENSITIVE_TAGS_KEY = "blockedCaseSensitiveTags"; const AUTO_OPEN_MEDIA_POSTS_KEY = "autoOpenMediaPosts"; const UNBLUR_THUMBNAILS_KEY = "unblurThumbs"; const SHOW_FILTERED_STUBS_KEY = "showFilteredStubs"; const COMMAS_AS_SEPARATORS_KEY = "useCommasAsSeparators"; const CASE_SENSITIVE_KEY = "caseSensitiveTag"; const WIDGET_HEIGHT_KEY = "widgetHeight"; const FIX_BROKEN_VIDEO_PREVIEWS_KEY = "fixBrokenVideoPreviews"; const MARK_POSTS_AS_NSFW_KEY = "markNewPostsAsNSFW"; // Constant selectors const postsContainerClass = "post-listings"; const profileContainerClass = "person-details"; const searchContainerClass = "search" const postContainer = "post-listing"; const loadingSpinnerSelector = ".icon.spin"; // This is the "creators" select in the search page const postCommunityContainer = ".community-link"; const postCommunityNameContainer = `${postCommunityContainer} > span`; const postSrcLink = `div:nth-of-type(2) p a`; const fixedPreviewContainerClass = "fixed-preview"; const fixedPreviewVideoClass = "fixed-preview-video"; const postAsNSFW = "#post-nsfw[type='checkbox']"; const createPostContainerId = "createPostForm"; const editPostClass = "post-form"; // GUI main elements const settingsWidgetId = "settings-widget"; const blockedTagsDialogId = "tags-dialog"; const dialogBgId = "dialog-bg"; const settingsWidgetContainerId = "widget-container"; const blockTagListId = "blocked-insensitive-tag-list"; const csBlockTagListId = "blocked-sensitive-tags-list"; const openedPostChecker = "already-opened"; const filteredPostChecker = "filter-checked"; const processedPageChecker = "processed-page"; // CSS Color constants const caseInsensitiveTagColor = "#dd2222"; const caseSensitiveTagColor = "#2052b3"; const primaryBtnColor = "#0052cc"; const primaryBtnHoverColor = "#0066ff"; const primaryBtnActiveColor = "#0047b3"; const widgetBgColor = "#1a1a1b"; const errorToastColor = "#f94b4b"; const errorToastBgColor = "#2f0808"; // Image constants const logoImg = GM_info.script.icon; // HTML content constants const bottomWidgetContent = ` <div id="${settingsWidgetContainerId}"> <h1><span id="widget-logo"></span>Lemmy post utilities</h1> <div class="form-entry"> <label>Blocked tags: </label> <div class="btn-container"> <button id="block-tag-btn" type="button">🖊</button> </div> </div> <div class="form-entry"> <label> Show stubs on filter: </label> <div class="btn-container"> <span class="switch" id="show-stub-swt"> <input type="checkbox"> <span class="slider round"></span> </span> </div> </div> <div class="form-entry"> <label> Fix broken video previews: </label> <div class="btn-container"> <span class="switch" id="fix-video-previews-swt"> <input type="checkbox"> <span class="slider round"></span> </span> </div> </div> <div class="form-entry"> <label> Auto open media previews: </label> <div class="btn-container"> <span class="switch" id="auto-open-swt"> <input type="checkbox"> <span class="slider round"></span> </span> </div> </div> <div class="form-entry"> <label> Unblur NSFW thumbnails: </label> <div class="btn-container"> <span class="switch" id="unblur-swt"> <input type="checkbox"> <span class="slider round"></span> </span> </div> </div> <div class="form-entry"> <label> Mark new posts as NSFW: </label> <div class="btn-container"> <span class="switch" id="default-nsfw-posts-swt"> <input type="checkbox"> <span class="slider round"></span> </span> </div> </div> </div> `; const tagsDialogContent = ` <div id="blocked-tags-dialog-container"> <div id ="blocked-tags-dialog-head"> <h1>Blocked tags editor</h1> <button type="button" class="close-dialog-btn">⨯</button> </div> <div> <p>Any tag added here will hide any post that contains the word in its title.</p> <p>These are the blocked tags you have for this instance:</p> <div id="blocked-tags-field-container"> <div id="blocked-tags-field"> <p id="empty-blocked-tags">You haven't blocked anything yet.</p> <ul id="${blockTagListId}" class="blocked-tags-list" hidden> </ul> <hr id="blocked-tags-separator" hidden> <ul id="${csBlockTagListId}" class="blocked-tags-list" hidden> </ul> </div> <div id="blocked-tags-field-legend"> <span id="tag-case-insensitive-legend" class="tag-legend"> <span class="tag-color-legend"></span> <small class="tag-label-legend">Case insensitive tag</small> </span> <span id="tag-case-sensitive-legend" class="tag-legend"> <span class="tag-color-legend"></span> <small class="tag-label-legend">Case sensitive tag</small> </span> </div> </div> <div class="switch-container"> <label> Use commas as tag separators: </label> <div class="btn-container"> <span class="switch" id="use-commas-swt"> <input type="checkbox"> <span class="slider round"></span> </span> </div> </div> <div class="switch-container"> <label> Add tags as case sensitive: </label> <div class="btn-container"> <span class="switch" id="case-sensitive-swt"> <input type="checkbox"> <span class="slider round"></span> </span> </div> </div> <div id="tag-input-container"> <input type="text" id="tag-input" placeholder="Add your tags"> <button type="button" id="tag-save-btn">Save</button> </div> <label id="blocked-tags-input-error" hidden>Tag must be longer than 1 non-whitespace character!</label> </div> </div> `; const tagContent = ` <span title="TOOLTIP-CONTENT"> <label>TAG-NAME</label> <button type="button">⨯</button> </span> `; const filteredPostStubContent = ` <div class="hidden-post-stub-meta-container"> <p>This post was hidden because it contained the tag 'TAG'.</p> <span class="hidden-post-stub-btn-container"> <button class="show-hidden-post-title-btn" type="button">Show title</button> <button class="show-hidden-post-btn" type="button">Show post</button> </span> </div> <span class="stub-hidden-post-title" hidden> <br> <p>Post title was '<b>POST-TITLE</b>' from <a class="stub-hidden-post-community-link">POST-COMMUNITY</a> community.</p> </span> `; const videoSourceContent = ` <source src="VIDEO-SOURCE" type="video/VIDEO-TYPE"> `; // CSS to add to style elements const initialCSS = getInitialCSS(); const guiCSS = () => ` /* Fix for stubs leaving empty space at the bottom of the page for some reason */ html { overflow-y: auto; } body { overflow-y: clip; } /* The settings widget */ #${settingsWidgetId} { display: flex; float: right; bottom: 0; right: 0.8rem; max-width: 15rem; position: fixed; transform: translateY(${getData(WIDGET_HEIGHT_KEY) ?? storeData(WIDGET_HEIGHT_KEY, "21rem")}); transition: 200ms ease-in-out transform; } #${settingsWidgetId}:hover { transform: translateY(1px); } /* The container for the widget */ #widget-container { flex-direction: column; border: 1px solid #333; font-size: 0.9rem; background-color: ${widgetBgColor}; width: 100%; padding: 1rem; border-radius: .5rem .5rem 0 0; margin-top: 50px; /* This is the space that will allow showing the popup when hovering over it */ } #widget-container:hover { margin-top: 0; } /* Widget title */ #widget-container > h1 { font-size: 1rem; font-weight: bold; height: 1rem; } #widget-logo { display: inline-block; background-image: url('${logoImg}'); height: 1rem; width: 1rem; background-size: contain; background-repeat: no-repeat; margin-right: 0.5rem; } /* Widget block button */ #block-tag-btn { appearance: none; color: #ddd; background: rgba(255,255,255,0.1); border: none; border-radius: 2rem; padding: .3rem .5rem; } /* The layout for each label + form control */ .form-entry { display: inline-flex; } .form-entry:not(:first-child) { margin-top: 0.8rem; } .form-entry > label { display: flex; align-items: center; justify-content: left; width: 10rem; } .btn-container { display: grid; place-items: center; } /* The dialog to filter tags */ #${blockedTagsDialogId} { appearance: none; border: none; background-color: ${widgetBgColor}; padding: 1rem; border-radius: 0.5rem; box-shadow: 0 0 20px 5px rgba(255,255,255,0.2); position: fixed; top: 50%; bottom: 50%; z-index: 1001; } #dialog-bg { display: none; background-color: rgba(0,0,0,.5); /* For browsers that don't support backdrop-filter */ position: fixed; height: 100vh; width: 100vw; backdrop-filter: blur(2px); top: 0; left: 0; z-index: 1000; } #blocked-tags-separator { width: 100%; margin-left: auto; margin-right: auto; border-top: 2px solid #666; margin-top: .5rem; margin-bottom: .5rem; } #blocked-tags-dialog-head { display: flex; } #blocked-tags-dialog-head > h1 { font-size: 1.5rem; margin-bottom: 1rem; color: #fff; } #tag-input-container { margin-top: 1rem; margin-bottom: 1rem; display: flex; } #blocked-tags-input-error { color: ${errorToastColor}; font-size: 0.8rem; background-color: ${errorToastBgColor}; padding: .4rem; border-radius: .5rem; font-weight: bold; } #blocked-tags-field-container { margin-bottom: 1rem; } #blocked-tags-field { display: grid; padding: 1rem; max-height: 20rem; overflow-y: auto; overflow-x: hidden; scrollbar-width: thin; place-items: center; border: 1px solid #555; border-radius: 0.3rem; } #empty-blocked-tags { color: #666; margin: 0; padding: 1rem; } #tag-input { border: none; flex-grow: 100; background-color: #333; border-radius: .5rem; padding: .5rem 1rem; width: 80%; margin-right: 1rem; color: #ddd; } #tag-save-btn { appearance: none; border: none; background-color: ${primaryBtnColor}; color: #ddd; padding: .5rem 1rem; border-radius: 0.3rem; } #tag-save-btn:hover { background-color: ${primaryBtnHoverColor}; color: #fff; } #tag-save-btn:active { background-color: ${primaryBtnActiveColor}; color: #fff; } #tag-case-insensitive-legend > span { background-color: ${caseInsensitiveTagColor}; } #tag-case-sensitive-legend > span { background-color: ${caseSensitiveTagColor}; } #blocked-tags-field-legend { display: inline-flex; align-items: center; justify-content: center; width: 100%; } .close-dialog-btn { appearance: none; color: #fff; font-weight: bold; background-color: transparent; border: none; padding: 0.2rem 0.6rem; position: absolute; right: 0; top: 0; transition: 200ms all; } .close-dialog-btn:hover { border-radius: 6rem; background-color: rgba(255,255,255,0.2); } .switch-container { width: 100%; display: flex; } .switch-container > label { width: 16rem; margin-right: 1rem; } .blocked-tags-list { display: flex; flex-wrap: wrap; max-width: 35.5rem; margin-bottom: 0; padding: 0; } .blocked-tags-list > li { display: inline-flex; list-style-type: none; } .tag-legend { margin-right: 1rem; } .tag-color-legend { display: inline-block; width: 7px; height: 7px; } .tag-label-legend { font-size: 0.6rem; } .blocked-tag { border-radius: 0.4rem; font-size: 0.8rem; color: #fff; margin: .2rem; cursor: grab; } .blocked-tag[data-dragged-item] { cursor: grabbing; opacity: 0.7; } .blocked-tag label { padding-left: .5rem; white-space: pre-wrap; cursor: grab; } .blocked-tag button { appearance: none; color: #fff; background: transparent; border: none; border-radius: .4rem; } .blocked-tag button:hover { background-color: rgba(255,255,255,0.2); } .case-insensitive-tag { background-color: ${caseInsensitiveTagColor}; } .case-sensitive-tag { background-color: ${caseSensitiveTagColor}; } /* The filtered post stubs */ .filtered-post-stub p { margin-bottom: 0; } .hidden-post-stub-meta-container { display: flex; } .hidden-post-stub-btn-container { margin-left: auto; } .hidden-post-stub-btn-container > button { appearance: none; background-color: transparent; border: none; cursor: pointer; color: #3498db; margin-right: 1rem; } .hide-post-btn { appearance: none; color: #f22; font-weight: bold; font-size: 2rem; line-height: 1rem; width: 1.8rem; background-color: transparent; border: none; padding: 0.2rem 0.6rem; height: 0; float: right; transition: 200ms all; } /* The switch - the box around the slider */ .switch { position: relative; display: inline-block; width: 30px; height: 16px; } /* Hide default HTML checkbox */ .switch input { opacity: 0; width: 0; height: 0; } /* The slider */ .slider { position: absolute; cursor: pointer; top: 0; left: 0; right: 0; bottom: 0; background-color: #777; -webkit-transition: .4s; transition: .4s; } .slider:before { position: absolute; content: ""; height: 12px; width: 12px; left: 2px; bottom: 2px; background-color: white; -webkit-transition: .4s; transition: .4s; } input:checked + .slider { background-color: #2196F3; } input:focus + .slider { box-shadow: 0 0 1px #2196F3; } input:checked + .slider:before { -webkit-transform: translateX(12px); -ms-transform: translateX(12px); transform: translateX(12px); } /* Rounded sliders */ .slider.round { border-radius: 34px; } .slider.round:before { border-radius: 50%; } `; const unblurCSS = ` /* Unblurs thumbnails */ .img-blur { filter: none !important; } ` //---------------------------------------- // GUI AND INITIAL SETUP //---------------------------------------- // Load initial settings and store defaults if they didn't exist console.info(`Loading data for domain '${document.location.host}'.`); const blockedTitleTags = getData(BLOCKED_TAGS_KEY) ?? storeData(BLOCKED_TAGS_KEY, []); const csBlockedTitleTags = getData(BLOCKED_CASE_SENSITIVE_TAGS_KEY) ?? storeData(BLOCKED_CASE_SENSITIVE_TAGS_KEY, []); const expandMediaPosts = getData(AUTO_OPEN_MEDIA_POSTS_KEY) ?? storeData(AUTO_OPEN_MEDIA_POSTS_KEY, false); const unblurThumbnails = getData(UNBLUR_THUMBNAILS_KEY) ?? storeData(UNBLUR_THUMBNAILS_KEY, false); const useCommasAsSeparators = getData(COMMAS_AS_SEPARATORS_KEY) ?? storeData(COMMAS_AS_SEPARATORS_KEY, true); const showStubsForFilteredPosts = getData(SHOW_FILTERED_STUBS_KEY) ?? storeData(SHOW_FILTERED_STUBS_KEY, true); const fixPostVideoPreviews = getData(FIX_BROKEN_VIDEO_PREVIEWS_KEY) ?? storeData(FIX_BROKEN_VIDEO_PREVIEWS_KEY, true); const caseSensitiveTags = getData(CASE_SENSITIVE_KEY) ?? storeData(CASE_SENSITIVE_KEY, false); const markNewPostsAsNSFW = getData(MARK_POSTS_AS_NSFW_KEY) ?? storeData(MARK_POSTS_AS_NSFW_KEY, false); if (blockedTitleTags.length > 0 || csBlockedTitleTags.length > 0) console.info("Waiting for page to fully load to block tags: ", blockedTitleTags, csBlockedTitleTags); // Add the GUI to control the settings to the document const settingWidget = addElement(document.body, "DIV", settingsWidgetId, bottomWidgetContent); const tagsDialog = addElement(document.body, "DIALOG", blockedTagsDialogId, tagsDialogContent); const dialogBg = addElement(document.body, "DIV", dialogBgId); // Add initial CSS changes updateCSS(); // Load tags into the dialog window updateVisibleTags(); // Reflect boolean values of settings on GUI document.querySelector("#show-stub-swt > input").checked = showStubsForFilteredPosts; document.querySelector("#auto-open-swt > input").checked = expandMediaPosts; document.querySelector("#unblur-swt > input").checked = unblurThumbnails; document.querySelector("#use-commas-swt > input").checked = useCommasAsSeparators; document.querySelector("#case-sensitive-swt > input").checked = caseSensitiveTags; document.querySelector("#fix-video-previews-swt > input").checked = fixPostVideoPreviews; document.querySelector("#default-nsfw-posts-swt > input").checked = markNewPostsAsNSFW; // --------- Add event listeners --------- // Blocked list dialog button document.getElementById("block-tag-btn").onclick = () => { openDialog(tagsDialog); } // Close dialog button tagsDialog.getElementsByClassName("close-dialog-btn")[0].onclick = () => { closeDialog(tagsDialog); } // Close dialog on clicking outside the dialog dialogBg.onclick = (event) => { event.stopPropagation(); closeDialog(tagsDialog); } // Show stubs on filtering posts document.getElementById("show-stub-swt").onclick = (event) => { const showStubs = storeData(SHOW_FILTERED_STUBS_KEY, toggleCheckbox(event)); const postsContainer = getPostContainer(); if (postsContainer) postsContainer.getElementsByClassName(postContainer).forEach(post => { let siblingNode = post.nextElementSibling; if (showStubs && post.hasAttribute("hidden")) { // Show sibling separator as well if (siblingNode && siblingNode.tagName == "HR") siblingNode.removeAttribute("hidden"); // Show post with the stub post.removeAttribute("hidden"); } else if (post.getElementsByClassName("filtered-post-stub")[0]) { // Remove separator if it exists if (siblingNode && siblingNode.tagName == "HR") siblingNode.setAttribute("hidden", true); // Hide post completelly post.setAttribute("hidden", true); } }) } // Fix some broken video previews document.getElementById("fix-video-previews-swt").onclick = (event) => { const fixVideoPreviews = storeData(FIX_BROKEN_VIDEO_PREVIEWS_KEY, toggleCheckbox(event)); const fixedPreviewContainers = document.getElementsByClassName(fixedPreviewContainerClass); const fixedPreviewVideos = document.getElementsByClassName(fixedPreviewVideoClass); const postsContainer = getPostContainer(); const wasProcessed = postsContainer.id === processedPageChecker; if (!postsContainer) return; if (!wasProcessed) postsContainer.setAttribute("id", processedPageChecker); if (fixVideoPreviews) { if (fixedPreviewContainers.length > 0 || fixedPreviewVideos.length > 0) { for(let i = 0; i < fixedPreviewContainers.length; i++) { fixedPreviewContainers[i].querySelector("picture").setAttribute("hidden", true); fixedPreviewVideos[i].removeAttribute("hidden"); } } else { fixBrokenVideoPreviews(postsContainer); } } else if (fixedPreviewContainers.length > 0 || fixedPreviewVideos.length > 0) { for(let i = 0; i < fixedPreviewContainers.length; i++) { fixedPreviewContainers[i].querySelector("picture").removeAttribute("hidden"); fixedPreviewVideos[i].setAttribute("hidden", true); } } } // Auto-open tags on page reload document.getElementById("auto-open-swt").onclick = (event) => { const openMediaPosts = storeData(AUTO_OPEN_MEDIA_POSTS_KEY, toggleCheckbox(event)); const postsContainer = getPostContainer(); const wasProcessed = postsContainer.id === processedPageChecker; if (!postsContainer) return; if (!wasProcessed) postsContainer.setAttribute("id", processedPageChecker); // Open posts if they weren't already if (openMediaPosts) { openPosts(postsContainer); } } // Unblur setting document.getElementById("unblur-swt").onclick = (event) => { storeData(UNBLUR_THUMBNAILS_KEY, toggleCheckbox(event)); // Update CSS of site after having changed the setting updateCSS(); } // Mark new posts as NSFW by default document.getElementById("default-nsfw-posts-swt").onclick = (event) => { const markAsNSFW = storeData(MARK_POSTS_AS_NSFW_KEY, toggleCheckbox(event)); const NSFWCheckbox = document.querySelector(postAsNSFW); const createPostForm = document.getElementById(createPostContainerId); if (NSFWCheckbox && !NSFWCheckbox.checked && markAsNSFW && createPostForm) NSFWCheckbox.click(); } // Auto-open tags on page reload document.getElementById("use-commas-swt").onclick = (event) => { storeData(COMMAS_AS_SEPARATORS_KEY, toggleCheckbox(event)); } // Auto-open tags on page reload document.getElementById("case-sensitive-swt").onclick = (event) => { storeData(CASE_SENSITIVE_KEY, toggleCheckbox(event)); } // Accept pressing enter while in some input to send the data document.getElementById("tag-input").onkeydown = (event) => { if(event.keyCode === 13){ document.getElementById("tag-save-btn").click(); } } // Event for tag save on button click or enter on input of tag blocking document.getElementById("tag-save-btn").onclick = () => { const tagInput = document.getElementById("tag-input"); const errorEl = document.getElementById("blocked-tags-input-error"); let tagsToSubmit = document.getElementById("tag-input").value; if (tagsToSubmit.trim().length > 1) { const isCaseSensitive = getData(CASE_SENSITIVE_KEY); const splitOnCommas = getData(COMMAS_AS_SEPARATORS_KEY); let tagsAsArray = getTagsAsArray(tagsToSubmit, splitOnCommas); let tagsKey; // Hide the error message if it was visible if (errorEl.hasAttribute("hidden")) errorEl.setAttribute("hidden", true); if (isCaseSensitive) tagsKey = BLOCKED_CASE_SENSITIVE_TAGS_KEY; else tagsKey = BLOCKED_TAGS_KEY; const oldTags = getData(tagsKey); tagsAsArray = tagsAsArray.filter(tag => !oldTags.includes(tag)); if (tagsAsArray.length > 0) { // Get the old array of tags and concatenate the new one const allTags = oldTags.concat(tagsAsArray); // Store the new array of tags and update the tags to show storeData(tagsKey, allTags); addTagsToDialog(tagsAsArray, tagsKey); // Hide empty tags element if it was visible and show the tags const blockedTagsListContainer = document.getElementById(blockTagListId); const csBlockedTagsListContainer = document.getElementById(csBlockTagListId); // Update tag section visibility if required checkForEmptyTags(getData(BLOCKED_TAGS_KEY), getData(BLOCKED_CASE_SENSITIVE_TAGS_KEY)); } // Clear input tagInput.value = ""; } else { errorEl.removeAttribute("hidden"); } } //---------------------------------------- // MAIN METHODS //---------------------------------------- // Wait for page to fully load to start doing things window.onload = () => { const baseContainer = document.getElementById("app"); let NSFWCheckbox = document.querySelector(postAsNSFW); let createPostForm = document.getElementById(createPostContainerId); let searchContainer = document.getElementsByClassName(searchContainerClass)[0]; let postsContainer = getPostContainer(); if (NSFWCheckbox && !NSFWCheckbox.checked && markNewPostsAsNSFW && createPostForm) NSFWCheckbox.click(); // Make sure that the widget height is correct updateWidgetHeightCSS(); // If there is a posts container in the page if (postsContainer) { // Perform first filter if (blockedTitleTags.length > 0 || csBlockedTitleTags.length > 0) { console.info("Page loaded, filtering tags..."); filterPosts(postsContainer); } // Open remaining posts if enabled if (expandMediaPosts) openPosts(postsContainer); // Fix video previews if there was any post if (fixPostVideoPreviews) fixBrokenVideoPreviews(postsContainer); document.getElementsByClassName(postsContainer.className)[0].setAttribute("id", processedPageChecker); } // Observe the changes of the page to know when to rethrow the filter method when the user changes the page const observer = new MutationObserver((e) => { if (document.getElementById(processedPageChecker)) return; const blockedTags = getData(BLOCKED_TAGS_KEY); const csBlockedTags = getData(BLOCKED_CASE_SENSITIVE_TAGS_KEY); const autoOpenMedia = getData(AUTO_OPEN_MEDIA_POSTS_KEY); const fixVideoPreviews = getData(FIX_BROKEN_VIDEO_PREVIEWS_KEY); const markAsNSFW = getData(MARK_POSTS_AS_NSFW_KEY); searchContainer = document.getElementsByClassName(searchContainerClass)[0]; NSFWCheckbox = document.querySelector(postAsNSFW); createPostForm = document.getElementById(createPostContainerId); postsContainer = getPostContainer(); // If on the creation post page, mark as NSFW the post if (NSFWCheckbox && !NSFWCheckbox.checked && markAsNSFW && createPostForm) NSFWCheckbox.click(); for (let i = 0; i < e.length; i++) { if (e[i].target.getElementsByClassName(editPostClass)[0]) return; let postEdit = e[i].target.getElementsByClassName("form-control")[0]; let filteredPostStub = e[i].target.getElementsByClassName("filtered-post-stub")[0]; let filteredPostBtn = e[i].target.getElementsByClassName("hide-post-btn")[0]; // Prevent filtering to retrigger if editting a post if (postEdit && searchContainer) return; // If a post was already filtered don't add stubs again if (filteredPostBtn || filteredPostStub) continue; // Perform actions if on one of the pages if (postsContainer) { if (blockedTags.length > 0 || csBlockedTags.length > 0) { console.info("Page reloaded, filtering new posts..."); filterPosts(postsContainer); } if (autoOpenMedia) openPosts(postsContainer); if (fixVideoPreviews) fixBrokenVideoPreviews(postsContainer); // Mark page as already processed document.getElementsByClassName(postsContainer.className)[0].setAttribute("id", processedPageChecker); break; } } }); observer.observe(baseContainer, {subtree: true, childList: true}); } // Gets initial CSS content function getInitialCSS() { let style = document.head.getElementsByTagName("style")[0]; if (!style) { style = document.createElement('style'); document.head.appendChild(style); return ""; } return style.innerHTML; } // Gets the CSS from the script to be in effect function getEffectiveCSS() { let fullCSS = guiCSS(); // Check each setting that changes CSS apart from the GUI if (getData(UNBLUR_THUMBNAILS_KEY)) fullCSS += unblurCSS; return fullCSS; } // Splits tags on commas if necessary function getTagsAsArray(tagsString, splitOnCommas=true) { if (splitOnCommas) { return tagsString.split(","); } return [tagsString]; } // Gets the current page posts container function getPostContainer() { return document.getElementsByClassName(postsContainerClass)[0] || document.getElementsByClassName(profileContainerClass)[0] || document.getElementsByClassName(searchContainerClass)[0] || document.querySelector(".post"); } // Adds an element to the document function addElement(parentEl, elementTag, elementId, html="", idAsClass=false) { let p = parentEl; let newElement = document.createElement(elementTag); if (idAsClass) newElement.className = elementId; else newElement.setAttribute('id', elementId); newElement.innerHTML = html; p.appendChild(newElement); return newElement; } // Adds the new tags to the dialog function addTagsToDialog(tagArray, tagsType) { let tagsContainerSelector = csBlockTagListId; if (tagsType === BLOCKED_TAGS_KEY) tagsContainerSelector = blockTagListId; const blockedTagsListEl = document.getElementById(tagsContainerSelector); tagArray.forEach(tag => { addTagElement(tag, tagsType, blockedTagsListEl); }); } // Adds a tag element to the dialog function addTagElement(tag, tagType, tagListElement) { // Get the specific css class for tag type let tagSpecificClass = "case-sensitive-tag"; if (tagType === BLOCKED_TAGS_KEY) tagSpecificClass = "case-insensitive-tag"; // Add the element to the dialog and bind the button event const newTag = addElement(tagListElement, "LI", `blocked-tag ${tagSpecificClass}`, tagContent .replace("TAG-NAME", tag) .replace("TAG-TYPE", tagType) .replace("TOOLTIP-CONTENT", `'${tag}' (${tagSpecificClass})`), true); newTag.getElementsByTagName("button")[0].onclick = () => { onRemoveBlockedTag(tagType, tag, newTag); } // Mark element as draggable newTag.setAttribute("draggable", true); // Handle drag & drop events newTag.ondragstart = (e) => { newTag.setAttribute("data-dragged-item", true); } newTag.ondragover = (e) => { e.preventDefault(); } newTag.ondragend = () => { newTag.removeAttribute("data-dragged-item"); } newTag.ondrop = (e) => { const target = e.target; const targetTag = target.nodeName === "LABEL" || target.nodeName == "BUTTON" ? target.parentNode.parentNode : target.parentNode; const draggedItem = document.querySelector('li[data-dragged-item]'); const nextTagName = targetTag.getElementsByTagName("label")[0].innerHTML; const movedTagName = draggedItem.getElementsByTagName("label")[0].innerHTML; let movedTagParentId = targetTag.parentNode.id; let draggedTagType = BLOCKED_TAGS_KEY; if (draggedItem.className.includes("case-sensitive-tag")) draggedTagType = BLOCKED_CASE_SENSITIVE_TAGS_KEY; // Only accept dropping it in the same area if (draggedItem.parentNode.id === movedTagParentId) { // Update order of tags let tags = getData(draggedTagType); // We need to change a bit the logic depending on the position of the element if (tags.indexOf(nextTagName) > tags.indexOf(movedTagName)) { tags.splice(tags.indexOf(nextTagName) + 1, 0, null); targetTag.parentNode.insertBefore(draggedItem, targetTag.nextSibling); } else { tags.splice(tags.indexOf(nextTagName), 0, null); targetTag.parentNode.insertBefore(draggedItem, targetTag); } tags.splice(tags.indexOf(movedTagName), 1); tags[tags.indexOf(null)] = movedTagName; storeData(draggedTagType, tags); } newTag.removeAttribute('data-dragged-item'); } } // Updates the CSS of the site function updateCSS() { let style = document.head.getElementsByTagName("style")[0]; if (!style) { style = document.createElement('style'); style.innerHTML = css; document.head.appendChild(style); return; } // Update the CSS with the initial one + the settings CSS style.innerHTML = initialCSS + getEffectiveCSS(); } // Update the visible part of the settings widget function updateWidgetHeightCSS() { const widget = document.getElementById(settingsWidgetContainerId); const widgetHeight = widget.getBoundingClientRect().height; const visibleTopHeight = 10; if (widgetHeight > 0 && widgetHeight - visibleTopHeight > 0) { const oldHeight = getData(WIDGET_HEIGHT_KEY); let updatedHeight = (widgetHeight - visibleTopHeight) + "px"; if (oldHeight != updatedHeight) { storeData(WIDGET_HEIGHT_KEY, (widgetHeight - visibleTopHeight) + "px"); updateCSS(); } } } // Updates the visible tags in the blocked tags dialog function updateVisibleTags() { const csBlockedTags = getData(BLOCKED_CASE_SENSITIVE_TAGS_KEY); const blockedTags = getData(BLOCKED_TAGS_KEY); const blockedTagsListEl = document.getElementById(blockTagListId); const csBlockedTagsListEl = document.getElementById(csBlockTagListId); // Reset list blockedTagsListEl.innerHTML = ""; csBlockedTagsListEl.innerHTML = ""; blockedTags.forEach(tag => { addTagElement(tag, BLOCKED_TAGS_KEY, blockedTagsListEl); }); csBlockedTags.forEach(tag => { addTagElement(tag, BLOCKED_CASE_SENSITIVE_TAGS_KEY, csBlockedTagsListEl); }); checkForEmptyTags(blockedTags, csBlockedTags); } // Toggles checkboxes on sliders inside the onclick event function toggleCheckbox(event) { let checkbox = event.target.parentNode.getElementsByTagName("input")[0]; checkbox.checked = !checkbox.checked; return checkbox.checked; } // Opens a dialog box function openDialog(dialogSelector) { dialogSelector.show(); dialogBg.style.display = "block"; } // Closes the dialog box function closeDialog(dialogSelector) { dialogSelector.close(); dialogBg.style.display = "none"; document.getElementById("blocked-tags-input-error").setAttribute("hidden", true); } // Adds the notice for no blocked tag if necessary function checkForEmptyTags(blockedTags, csBlockedTags) { const emptyTagsEl = document.getElementById("empty-blocked-tags"); const blockedTagsListEl = document.getElementById(blockTagListId); const csBlockedTagsListEl = document.getElementById(csBlockTagListId); const tagsSeparatorEl = document.getElementById("blocked-tags-separator"); const hasAnyBlockedTag = blockedTags.length > 0; const hasAnyCsBlockedTag = csBlockedTags.length > 0; const hasAnyTag = hasAnyBlockedTag || hasAnyCsBlockedTag; const isBlockedTagsListHidden = blockedTagsListEl.hasAttribute("hidden"); const isCsBlockedTagsListHidden = csBlockedTagsListEl.hasAttribute("hidden"); // Check if any has tags if (hasAnyTag) { emptyTagsEl.setAttribute("hidden", true); if (hasAnyBlockedTag) blockedTagsListEl.removeAttribute("hidden"); else blockedTagsListEl.setAttribute("hidden", true); if (hasAnyCsBlockedTag) csBlockedTagsListEl.removeAttribute("hidden"); else csBlockedTagsListEl.setAttribute("hidden", true); // Either both have tags or at least one has tags if (hasAnyBlockedTag && hasAnyCsBlockedTag) tagsSeparatorEl.removeAttribute("hidden"); else tagsSeparatorEl.setAttribute("hidden", true); // If it has no tags } else if (!hasAnyTag) { emptyTagsEl.removeAttribute("hidden"); blockedTagsListEl.setAttribute("hidden", true); tagsSeparatorEl.setAttribute("hidden", true); csBlockedTagsListEl.setAttribute("hidden", true); } } // Checks if the page is loading function checkIfPageIsLoading(postsContainer, posts) { const searchContainer = document.getElementsByClassName(searchContainerClass)[0]; const emptyResultsContainer = document.querySelector(loadingSpinnerSelector); return posts.length === 0 && postsContainer === searchContainer && emptyResultsContainer; } // Removes a tag from the view and the storeData function onRemoveBlockedTag(tagType, tagToRemove, tagElement) { // Remove the actual tag element tagElement.parentNode.removeChild(tagElement); // Update the storage let allTags = getData(tagType); let tagIndex = allTags.indexOf(tagToRemove); if (tagIndex != -1) allTags.splice(tagIndex, 1); storeData(tagType, allTags); // Toggle visibility of the elements where the tag was removed if they are empty checkForEmptyTags(getData(BLOCKED_TAGS_KEY), getData(BLOCKED_CASE_SENSITIVE_TAGS_KEY)); } // Filters posts by words from the title as specified by the user function filterPosts(postsContainer) { let posts = postsContainer.getElementsByClassName(postContainer); if (checkIfPageIsLoading(postsContainer, posts)) { let intervalCount = 0; // Check for posts in 1 second intervals const checkInterval = setInterval(() => { posts = document.getElementsByClassName(postContainer); // Finish interval checking the search finished or 30 seconds passed if (posts.length > 0 || intervalCount > 30 || !document.querySelector(loadingSpinnerSelector)) { clearInterval(checkInterval); filterPostsInPage(posts); } intervalCount++; }, 1000); return; } filterPostsInPage(posts); } // Filters the posts in the current page function filterPostsInPage(postsList) { const showStubs = getData(SHOW_FILTERED_STUBS_KEY); const blockedTags = getData(BLOCKED_TAGS_KEY); const csBlockedTags = getData(BLOCKED_CASE_SENSITIVE_TAGS_KEY); Array.from(postsList).forEach(post => { if (post.getElementsByClassName(editPostClass)[0]) return; post.setAttribute(filteredPostChecker, true); const communityEl = post.querySelector(postCommunityNameContainer); const communityLinkEl = post.querySelector(postCommunityContainer); let titleEl = post.querySelector(".post-title h1 span"); let community; let communityLink; let title; // Make sure we get a valid title selector if (!titleEl) titleEl = post.querySelector(".post-title h1 a"); title = titleEl.innerHTML; // Make sure community info is in post, otherwise get from page assuming the user // is in the community view instead, where the community isn't in the posts but the page if (!communityEl) { community = document.querySelector(postCommunityNameContainer).innerHTML; communityLink = document.querySelector(postCommunityContainer).href; } else { community = communityEl.innerHTML; communityLink = communityLinkEl.href; } const postInfo = { post: post, title: title, communityName: community, communityLink: communityLink } // Filter posts using case insensitive tags if (blockedTags.length > 0) { for (let bt = 0; bt < blockedTitleTags.length; bt++) { let tag = blockedTitleTags[bt] if (title.toLowerCase().includes(tag.toLowerCase())) { removePost(postInfo, tag, showStubs); return; } } } // Filter posts using case sensitive tags if (csBlockedTags.length > 0) { for (let bt = 0; bt < blockedTitleTags.length; bt++) { let tag = blockedTitleTags[bt] if (title.includes(tag)) { removePost(postInfo, tag, showStubs); return; } } } }) } // Filters out a post function removePost(postInfo, tag, showStubs) { const post = postInfo.post; // Hide the content of the children and add a stub by default addFilterStubToPost(postInfo, tag) hidePostChildren(post); // If the setting to show stubs is disabled we hide the post completelly if (!showStubs) { // Removes the posts from the body completelly and logs it to the console console.info(`Removing post with title ${postInfo.title} from community ${postInfo.communityName} because it contains tag ${tag}.`); post.setAttribute("hidden", true); let siblingNode = post.nextElementSibling; // Remove separator if it exists if (siblingNode && siblingNode.tagName == "HR") siblingNode.setAttribute("hidden", true); } } // Hides all elements of the post children function hidePostChildren(post) { for (let i = 0; i < post.children.length; i++) { let child = post.children[i]; if (child.className === "filtered-post-stub" || child.className === "hide-post-btn") continue; child.style.height = "0"; child.style.width = "0"; child.style.visibility = "hidden"; } } // Adds a filter stub after filtering a post function addFilterStubToPost(postInfo, tag) { let stub = addElement(postInfo.post, "DIV", "filtered-post-stub", filteredPostStubContent.replace("TAG", tag) .replace("POST-TITLE", postInfo.title) .replace("POST-COMMUNITY", postInfo.communityName), true); stub.getElementsByClassName("stub-hidden-post-community-link")[0].href = postInfo.communityLink; // Add actions for the buttons stub.getElementsByClassName("show-hidden-post-title-btn")[0].onclick = () => { stub.getElementsByClassName("stub-hidden-post-title")[0].removeAttribute("hidden"); stub.getElementsByClassName("show-hidden-post-title-btn")[0].setAttribute("hidden", true); } stub.getElementsByClassName("show-hidden-post-btn")[0].onclick = () => { stub.setAttribute("hidden", true); // Make sure the title button element is hidden stub.getElementsByClassName("stub-hidden-post-title")[0].setAttribute("hidden", true); stub.getElementsByClassName("show-hidden-post-title-btn")[0].removeAttribute("hidden"); // Prepend the post re-remover if it didn't exist yet let postRemoveEl = postInfo.post.getElementsByClassName("hide-post-btn")[0]; if (!post.getElementsByClassName("hide-post-btn")[0]) { postRemoveEl = document.createElement("BUTTON"); postRemoveEl.className = "hide-post-btn"; postRemoveEl.innerHTML = "🗶"; postRemoveEl.setAttribute("type", "button"); post.prepend(postRemoveEl); postRemoveEl.onclick = () => { stub.removeAttribute("hidden"); postRemoveEl.setAttribute("hidden", true); hidePostChildren(postInfo.post); } } else postRemoveEl.removeAttribute("hidden"); for (let i = 0; i < postInfo.post.children.length; i++) { postInfo.post.children[i].style = ""; } } } // Opens media posts so that the image or video is shown for all posts in the current page of the timeline function openPosts(postsContainer) { let posts = postsContainer.getElementsByClassName(postContainer); if (checkIfPageIsLoading(postsContainer, posts)) { let intervalCount = 0; // Check for posts in 1 second intervals const checkInterval = setInterval(() => { posts = document.getElementsByClassName(postContainer); // Finish interval checking the search finished or 30 seconds passed if (posts.length > 0 || intervalCount > 30 || !document.querySelector(loadingSpinnerSelector)) { clearInterval(checkInterval); clickPostsThumbnail(posts); } intervalCount++; }, 1000); return; } clickPostsThumbnail(posts) } // Clicks the thumbnail of the post to open it function clickPostsThumbnail(posts) { Array.from(posts).forEach(post => { const postHasMedia = post.querySelector("picture, video"); let clickableThumbnail = post.querySelector("button.thumbnail"); if (postHasMedia) { if (!clickableThumbnail) clickableThumbnail = post.querySelector("a.text-body[href$='.mp4'] .thumbnail, a.text-body[href$='.webm'] .thumbnail, a.text-body[href^='https://www.redgifs.com/watch'] .thumbnail") if (clickableThumbnail && !post.getElementsByClassName("filtered-post-stub")[0]) clickableThumbnail.click(); post.setAttribute(openedPostChecker, true); } }) } // Fixes imgur previews showing as jpg instead of the actual video function fixBrokenVideoPreviews(postsContainer) { let posts = postsContainer.getElementsByClassName(postContainer); if (checkIfPageIsLoading(postsContainer, posts)) { let intervalCount = 0; // Check for posts in 1 second intervals const checkInterval = setInterval(() => { posts = document.getElementsByClassName(postContainer); // Finish interval checking the search finished or 30 seconds passed if (posts.length > 0 || intervalCount > 30 || !document.querySelector(loadingSpinnerSelector)) { clearInterval(checkInterval); updateVideoPreviews(posts); } intervalCount++; }, 1000); return; } updateVideoPreviews(posts); } // Updates the video previews when needed so that they work again function updateVideoPreviews(posts) { Array.from(posts).forEach(post => { const postSourceLinkEl = post.querySelector(postSrcLink); const imageLinkContainer = post.querySelectorAll("div:nth-of-type(3) > a:not(.btn)"); if (postSourceLinkEl && imageLinkContainer.length > 0) { const postSourceLink = postSourceLinkEl.href; let newSrc = postSourceLink; if (postSourceLink.includes("i.imgur.com") && postSourceLink.endsWith(".gifv")) newSrc = postSourceLink.replace(".gifv", ".mp4"); if (postSourceLink.includes("i.redd.it") && postSourceLink.endsWith(".gif")) newSrc = imageLinkContainer[0].href; // Only apply changes if the src was suposed to be different if (newSrc != postSourceLink) { imageLinkContainer.forEach(imageContainer => { let pictureContainer = imageContainer.querySelector("picture"); if (pictureContainer) { pictureContainer.setAttribute("hidden", true); let videoElement = addElement(post.querySelector("div:nth-child(3).my-2"), "VIDEO", `embed-responsive-item ${fixedPreviewVideoClass} col-12`, videoSourceContent.replace("VIDEO-SOURCE", newSrc) .replace("VIDEO-TYPE", newSrc.split(".").at(-1)), true) videoElement.setAttribute("controls", ""); videoElement.parentNode.classList.add("embed-responsive", fixedPreviewContainerClass); } }); } } }); } //---------------------------------------- // STORAGE METHODS //---------------------------------------- // Composes the key with the current instance name to store data per-instance function composeInstanceKey(key) { return document.location.host + "->" + key; } // Saves data to the storage of the userscript function storeData(key, value) { const instanceKey = composeInstanceKey(key); let treatedValue = value; if (typeof value === "array" || typeof value === "object") treatedValue = JSON.stringify(value); GM_setValue(instanceKey, treatedValue); return treatedValue; } // Gets data from the userscript storage function getData(key) { const instanceKey = composeInstanceKey(key); let value = GM_getValue(instanceKey); if (value === null || value === undefined) return null; if (typeof value === "string") { let isValueArray = value.startsWith("[") && value.endsWith("]"); let isValueObject = value.startsWith("{") && value.endsWith("}"); if (isValueArray || isValueObject) return JSON.parse(value); } return value; }
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址