- // ==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://lemy.nl/*
- // @include https://burggit.moe/*
- // @include https://lemmit.online/*
- // @include https://yiffit.net/*
- // @include https://reddthat.com/*
- // @include https://sh.itjust.works/*
- // @exclude https://lemmyverse.net/*
- // @exclude https://lemmy-status.org/*
- // @exclude https://search-lemmy.com/*
- // @exclude https://join-lemmy.org/*
- // @grant GM_getValue
- // @grant GM_setValue
- // @grant GM_setClipboard
- // @version 2.2.3
- // @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";
-
- // Other constants
- const logoImg = GM_info.script.icon;
- const namedRegex = "regex\\((.*?)\\):";
-
- // 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>You can also use some advanced filtering options like the following:</p>
- <ul>
- <li>Use <a href="https://regex101.com/">regex</a> starting a tag with <code>regex:</code></li>
- <li>Create a named regex starting a tag with <code>regex(your-regex-name):</code></li>
- <li>Filter by linked source instead of title starting a tag with <code>source:</code></li>
- <li>Combine <code>source:</code> with any variant of <code>regex:</code></li>
- <li>You can click any tag to copy it's raw value.</li>
- </ul>
- <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>
- <p id="clipboard-notice" hidden>Tag value copied to the clipboard!</p>
- <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;
- max-width: 36.5rem;
- }
-
- #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;
- }
-
- #clipboard-notice {
- color: #2f2;
- font-size: 0.8rem;
- width: 100%;
- text-align: center;
- }
-
- #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");
- const clipboardNoticeEl = document.getElementById("clipboard-notice");
- const tagsToSubmit = document.getElementById("tag-input").value;
- clipboardNoticeEl.setAttribute("hidden", true);
-
- 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;
-
- if (isCaseSensitive)
- tagsKey = BLOCKED_CASE_SENSITIVE_TAGS_KEY;
- else
- tagsKey = BLOCKED_TAGS_KEY;
-
- const oldTags = getData(tagsKey);
-
- tagsAsArray = tagsAsArray.filter(tag => !oldTags.includes(tag));
-
- for(let i = 0; i < tagsAsArray.length; i++) {
- const keywordStart = /^(regex:|source:(regex(\((.*?)\):|:))*|regex\((.*?)\):)/ig;
- console.log(tagsAsArray[i].replace(keywordStart, ""));
- if (tagsAsArray[i].replace(keywordStart, "").trim().length === 0) {
- errorEl.removeAttribute("hidden");
- return;
- }
- }
-
- // Hide the error message if it was visible
- if (errorEl.hasAttribute("hidden"))
- errorEl.setAttribute("hidden", true);
-
- 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", parseTagName(tag))
- .replace("TAG-TYPE", tagType)
- .replace("TOOLTIP-CONTENT", `'${tag}' (${tagSpecificClass})`),
- true);
-
- newTag.getElementsByTagName("button")[0].onclick = () => {
- onRemoveBlockedTag(tagType, tag, newTag);
- }
-
- newTag.getElementsByTagName("label")[0].onclick = () => {
- GM_setClipboard(tag);
- document.getElementById("clipboard-notice").removeAttribute("hidden");
- }
-
- // 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);
- }
-
- // Parses the tag name if it's a named regex
- function parseTagName(tag) {
- if ((tag.toLowerCase().startsWith("regex(") || tag.toLowerCase().startsWith("source:regex(")) && new RegExp(namedRegex, "i").test(tag)) {
- let regexName = getRegexNameFromTag(tag);
- if (regexName != null) {
- return tag.replace(new RegExp(namedRegex + ".*", "i"), `regex: ${regexName}`);
- }
- // If the named regex wasn't correctly formatted we just return the regular tag however it was typed
- return tag;
- }
- return tag;
- }
-
- // Gets the regex name from a named regex tab
- function getRegexNameFromTag(tag) {
- let regexMatch = new RegExp(namedRegex).exec(tag);
- if (regexMatch != null) {
- return regexMatch[1];
- }
- return null;
- }
-
- // 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);
- document.getElementById("clipboard-notice").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);
-
- const blockedTagsByType = {
- byTitle: [],
- bySource: []
- }
-
- const csBlockedTagsByType = {
- byTitle: [],
- bySource: []
- }
-
- // Get type of filter for each tag first to avoid unnecesary iterations
- blockedTags.forEach(tag => {
- if (tag.startsWith("source:"))
- blockedTagsByType.bySource.push(tag);
- else
- blockedTagsByType.byTitle.push(tag);
- })
-
- csBlockedTags.forEach(tag => {
- if (tag.startsWith("source:"))
- csBlockedTagsByType.bySource.push(tag);
- else
- csBlockedTagsByType.byTitle.push(tag);
- })
-
- // Filter every post
- 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);
- const sourceEl = post.querySelector(postSrcLink)
- let titleEl = post.querySelector(".post-title h1 span");
- let source = null;
- 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;
- }
-
- if (sourceEl)
- source = sourceEl.href;
-
- const postInfo = {
- post: post,
- title: title,
- communityName: community,
- communityLink: communityLink,
- source: source
- }
-
- // Filter posts by title first
- const wasFiltered = filterPost(blockedTagsByType.byTitle, csBlockedTagsByType.byTitle, showStubs, postInfo);
-
- // Filter posts by source after if it wasn't already filtered by title
- if (!wasFiltered && source)
- filterPost(blockedTagsByType.bySource, csBlockedTagsByType.bySource, showStubs, postInfo, true);
- })
- }
-
- function filterPost(blockedTags, csBlockedTags, showStubs, postInfo, filterBySource=false) {
- // Filter posts using case insensitive tags
- if (blockedTags.length > 0) {
- for(let i = 0; i < blockedTags.length; i++) {
- let tag = blockedTags[i];
-
- if (filterBySource)
- tag = tag.substring(7);
-
- const regex = new RegExp(escapeRegex(tag), "i");
- if (regex.test(filterBySource ? postInfo.source : postInfo.title)) {
- removePost(postInfo, blockedTags[i], showStubs);
-
- return true;
- }
- }
- }
-
- // Filter posts using case sensitive tags
- if (csBlockedTags.length > 0) {
- for(let i = 0; i < csBlockedTags.length; i++) {
- const tag = csBlockedTags[i];
-
- if (filterBySource)
- tag = tag.substring(7);
-
- const regex = new RegExp(escapeRegex(tag));
- if (regex.test(filterBySource ? postInfo.source : postInfo.title)) {
- removePost(postInfo, csBlockedTags[i], showStubs);
- return true;
- }
- }
- }
- }
-
- // Escapes the special characters entered by a user
- function escapeRegex(regex) {
- // Escape \'s in regex with double \\ if it's a regex to be used in "new RegExp()" method
- if (regex.toLowerCase().startsWith("regex:"))
- return regex.substring(6).replace("\\", "\\");
-
- // If it's a named regex
- else if (regex.toLowerCase().startsWith("regex(") && new RegExp(namedRegex, "i").test(regex)) {
- const regStart = `regex(${getRegexNameFromTag(regex)}):`;
- return regex.substring(regStart.length).replace("\\", "\\");
- }
- return regex.replace(/([()[{*+.$^\\|?])/g, '\\$1');
-
- }
-
- // 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 the post contained the tag ${parseTagName(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", parseTagName(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);
- postInfo.post.style= "";
-
- // 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 (!postInfo.post.getElementsByClassName("hide-post-btn")[0]) {
- postRemoveEl = document.createElement("BUTTON");
- postRemoveEl.className = "hide-post-btn";
- postRemoveEl.innerHTML = "🗶";
- postRemoveEl.setAttribute("type", "button");
- postInfo.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");
- else if (postSourceLink.includes("imgur.com"))
- newSrc = postSourceLink.replace("imgur.com", "i.imgur.com") + ".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;
- }