Lemmy post utilities - Filter posts by title

Filters posts on any lemmy instance by text in the title. It can also auto-open image posts, unblur thumbnails, and other things.

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==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.10
// @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;
  scrollbar-width: none;
}

/* 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
  post.style.display = "none";
  addFilterStubToPost(postInfo, tag)

  // 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);
  }
}

// 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);
  }

  // Add theming and add it before the filtered post
  stub.classList.add("post-listing");
  postInfo.post.parentNode.insertBefore(stub, postInfo.post);

  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);
        postInfo.post.style.display = "none";
      }
    } 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]){
        let intervalCount = 0;
        const checkInterval = setInterval(() => {
          const postChildren = post.children;
          clickableThumbnail.click();

          // Check that the post opened within 5 tries
          if (postChildren.length > 2 || intervalCount > 5)
            clearInterval(checkInterval);

          intervalCount++;
        }, 1000);
      }

      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);
        checkVideoPreviews(postsContainer);
      }
      intervalCount++;
    }, 1000);

    return;
  }

  checkVideoPreviews(postsContainer);
}

// Updates the video previews when needed so that they work again
function checkVideoPreviews(postsContainer) {
  setTimeout(() => {
    const posts = postsContainer.getElementsByClassName(postContainer);
    updateVideoPreviews(posts);
  }, 500);
}

// Updates the video previews
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("i.imgur.com") && postSourceLink.includes("imgur.com"))
        newSrc = postSourceLink.replace("imgur.com", "i.imgur.com") + ".mp4";
      else if (imageLinkContainer.length >= 1 && (postSourceLink != imageLinkContainer[0].href) && imageLinkContainer[0].href.endsWith(".webm") || imageLinkContainer[0].href.endsWith(".mp4")) {
        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.setAttribute("loop", "");
            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;
}