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