// ==UserScript==
// @name redditmod
// @namespace derv82
// @description Subset of RES features I like.
// @include https://*.reddit.com/*
// @version 1.6.0
// @grant GM_xmlhttpRequest
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_addStyle
// ==/UserScript==
/*
TODO Features:
-> UX Overhaul: Hovering a post has /large/ buttons to 1. Expand the content, 2. Expand the comments, 3. Hide this post, 4. Filter the subreddit
-> Media overhaul: Don't rely on max-height, allow resizing of images
-> Customizable UI colors. Or at least different themes.
*/
var Redditmod = {};
Redditmod.CSS = (function() {
var self = this;
this.colors = {
bg: {
base: "#121234",
input: "#232356",
filterHover: "#f00",
flair: "#18f",
thingHover: "#18f"
},
fg: {
text: "#aaa",
title: "#18f",
visited: "#88f",
filter: "#f00",
filterHover: "#fff",
flair: "#000",
spoiler: "#f80"
},
border: {
input: "#555",
flair: "#000"
},
shadow: { thingHover: "#18f" }
};
GM_addStyle("#chat,.eddit-duplicate,.eddit-filtered-post,.footer,.footer-parent,.goldvertisement,.hohoho{display:none}:root{--fg:#E0E2E4;--bg:#293134;--bg-odd:#303a3d;--fg-2:#678CB1;--bg-2:#3F4B4E;--fg-muted:#999;--fg-input:#A7CCF1;--bg-input:#293134;--fg-input-hover:#E0E2E4;--bg-input-hover:#536364;--border:solid 0.5px #678CB1;--border-color:#678CB1;--fg-link:#EC7600;--fg-link-hover:#FC8610;--fg-link-visited:#CA5400;--fg-disabled:#ddd;--bg-disabled:#777;--fg-error:#fff;--fg-warning:#000;--fg-info:#fff;--fg-success:#fff;--bg-error:#600;--bg-warning:#fa4;--bg-info:#14a;--bg-success:#151;--syntax-fg:#F1F2F3;--syntax-fg-2:#E8E2B7;--syntax-bg:#293134;--syntax-bg-2:#3F4B4E;--syntax-comment:#66747B;--syntax-builtin:#93C763;--syntax-function:#678CB1;--syntax-keyword:#7CCADD;--syntax-number:#FFCD22;--syntax-string:#EC7600;--syntax-font-family:\"Menlo\",\"Liberation Mono\",\"Consolas\",\"DejaVu Sans Mono\",\"Ubuntu Mono\",\"Courier New\",\"andale mono\",\"lucida console\",monospace;--selection-fg:#000}.ProfileSidebar__titleContainer,.ProfileTemplate.m-updated,.content,.link,.listing-chooser li.selected,.morelink .nub,.side,.sidebox .spacer,.tabmenu li.selected a,.user-subreddit-banner,body,body.with-listing-chooser .listing-chooser .grippy,html{background-color:var(--bg)}#header,#sr-header-area,.BlueBar,.Media.m-profile.m-image,.Post,.Post__pinInfo,.Post__stickiedDivider,.ProfileBarDropdown__list,.SubscriptionBar,.author-tooltip__head,.authorized-app,.drop-choices,.gold-accent,.hover-bubble,.infobar,.link .usertext-body .md,.linkinfo,.listing-chooser li,.liveupdate-listing li.liveupdate time::before,.liveupdate-listing li.separator time,.modal-content,.nextprev a:hover,.raisedbox,.roundfield,.searchpane,.server-seconds,.sr-interest-bar,.sr-interest-bar .bubble,.subreddit .usertext .md,.tabmenu li a,.trophy-area .content,body.with-listing-chooser .listing-chooser,div.link.self:hover{background-color:var(--bg-2)}#header-bottom-right,#sr-more-link,.BlueBar__account,.goldvertisement a,.link .md pre,.md code,.md pre,.nextprev a,.usertable tr:hover,.usertext.border .usertext-body,oddrow{background-color:var(--bg-odd)}.form-control,button.report-button,input,input[type=text],input[type=password],select,textarea{background-color:var(--bg-input);color:var(--fg-input)}.form-control:focus,.goldvertisement a:hover,.listing-chooser li:hover,body.with-listing-chooser .listing-chooser .grippy:hover::after,body.with-listing-chooser.listing-chooser-collapsed .listing-chooser .grippy::before,input:focus,input[type=text]:focus,input[type=password]:focus,select:focus,textarea:focus{background-color:var(--bg-input-hover);color:var(--fg-input-hover)}.form-control,.infobar.onboardingbar,.listing-chooser li.selected,.navbar,.sidebox.hohoho .morelink,body,body.with-listing-chooser .listing-chooser .grippy:hover::after,body.with-listing-chooser.listing-chooser-collapsed .listing-chooser .grippy::before,input[type=text],select,textarea{background-image:none}.icon-menu a,.titlebox form.toggle,aside.sidebar #discussions li,div.link.self{background-color:transparent}.ProfileBarDropdown__listItem:hover,.drop-choices a.choice:hover,.flair,.linkflairlabel,.wiki-page .pageactions .wikiaction:hover,a.message-count{background-color:var(--fg-link);color:var(--bg)!important}.md hr{background-color:var(--border-color)}.Post__pinInfo,.TabMenu__tab.m-active,.card a,.gold-accent,.goldvertisement,.hover-bubble,.liveupdate-listing li.separator time,.md,.titlebox,.titlebox h1 a,.trophy-name,.user-subreddit-public-description,body,html{color:var(--fg)}.Post__score,.TrendingProfilesSidebar__header,.TrophyCaseSidebar__header,.UserModeratedSubredditsSidebar__header,.UserSpecialsListSidebar__header,.author-tooltip__credentials-list strong,.brand,.dropdown.srdrop .selected,.gold-accent th,.goldvertisement .progress p,.hover-bubble.multi-selector strong,.link .rank,.listing-chooser h3,.server-seconds em,.side .titlebox .md ul li,.tagline a.subreddit,.wiki-page .wikititle,.wiki-page-content .md h1,.wiki-page-content .md h2,.wiki-page-content .md h3,.wiki-page-content .md h4,.wiki-page-content .md h5,aside.sidebar .md h2,aside.sidebar section,h1,h2,h3,h4,h5{color:var(--fg-2)}#sr-header-area .selected a,#sr-more-link,.Post__authorLink,.Post__authorLink:visited,.Post__flatListItem:link,.Post__source .Post__sourceLink,.Post__tagline,.ProfileSidebar__counters,.ProfileSidebar__description,.ProfileSidebar__displayName,.SubredditListItem__subscribers,.TrophyCaseSidebar__trophyDescription,.TrophyCaseSidebar__trophyTitle,.UserSpecialsListSidebar__date,.UserSpecialsListSidebar__title,.bottommenu,.entry .buttons li a,.flat-list li.selected a,.flat-vert.title,.footer,.gray,.listing-chooser li a .description,.liveupdate-listing li.liveupdate time,.md blockquote,.md del,.muted,.nextprev,.preftable .details,.separator,.sidebox .subtitle,.sidecontentbox .more a,.sidecontentbox .title h1,.sr-list a:link,.sr-list a:link:hover,.sr-list a:visited,.subscription-box .title:visited,.tagline,.tagline .expand,.tagline>a.author,.titlebox .bottom,.titlebox form.toggle,.user,a.title.visited,aside.sidebar #discussions li,span.domain>a{color:var(--fg-muted)}.BlueBar__logout,.ProfileBarDropdown__title,.listing-chooser .create button,.liveupdate-listing li.liveupdate a.author,.liveupdate-listing li.liveupdate:hover time,.subscription-box .title,.thing .title,a,a:link{color:var(--fg-link)}#sr-header-area .selected a:hover,.BlueBar__logout:hover,.Post__authorLink:hover,.Post__stickied-title:hover,.Post__subredditLink:hover,.ProfileBarDropdown__title.m-opened,.author-tooltip__head a:hover,.author-tooltip__link-list a:hover,.flat-list li.selected a:hover,.liveupdate-listing li.liveupdate:hover a.author,.subscription-box .title:hover,.thing .title:hover,a:hover,a:visited:hover{color:var(--fg-link-hover)}.thing .title:visited,a:visited{color:var(--fg-link-visited)}.morelink>a,.spoiler-stamp,a.TrendingProfilesSidebar__follow,a.c-btn-primary{color:#fff}.tagline .submitter{color:#4af}.Post__moderatorFlair,.tagline .moderator,.tagline .pinned-tagline,.tagline .stickied-tagline,.thing.stickied.link a.title,.thing.sticky-pinned.link a.title{color:#3a3}input[style*=\"color:black\"]{color:var(--fg-input)!important}hr{color:var(--border-color)}#liveupdate-statusbar.complete,.Post.m-profile,.Post__scoreDisplay,.content,.flair,.footer,.link,.linkflairlabel,.linkinfo,.navbar,.permission-summary,.reddit-infobar,div.link.self,noborder{border:none}button.report-button,custom-border,input,select{border:var(--border)}#compose-message .roundfield select,#header,#header hr,#sr-header-area,.BlueBar,.author-tooltip__credentials-list li,.author-tooltip__head,.author-tooltip_self .author-tooltip__head,.authorized-app,.drop-choices,.filtered-details form.add-sr .sr-name,.footer .col,.hover-bubble,.link .md pre,.link .usertext-body .md,.listing-chooser li,.listing-chooser li.selected,.liveupdate-listing li.separator::before,.md code,.md pre,.md td,.md th,.menuarea,.morelink,.nextprev a,.raisedbox,.roundfield input[type=text],.roundfield textarea,.sidebox.hohoho .morelink,.sidecontentbox .content,.subreddit .usertext .md,.subscription-box .box-separator,.titlebox .bottom,.wiki-page .pageactions,.wiki-page .wiki-page-content .wiki>.toc>ul,body.with-listing-chooser .listing-chooser .grippy,hr,input,select,textarea{border-color:var(--border-color)}.tabmenu li.selected a{border-bottom-color:transparent}.goldvertisement a,.listing-chooser .create button{border-color:var(--fg-link)}.nextprev .separator{border-color:var(--fg-muted)}.author-tooltip.hover-bubble.anchor-bottom-left::before,.author-tooltip.hover-bubble.anchor-bottom-right::before{border-top-color:var(--bg-2)}.ProfileBarDropdown__title.m-arrow.m-opened::after,.ProfileBarDropdown__title.m-arrow::after,.TabMenu__tab.m-active::after{border-top-color:var(--fg-muted)}#liveupdate-statusbar.complete,success,success a{background-color:var(--bg-success);color:var(--fg-success)}.reddit-infobar,.reddit-infobar .md,info,info a{background-color:var(--bg-info);color:var(--fg-info)}.app-scope,warning{background-color:var(--bg-warning);color:var(--fg-warning)}error,error a{background-color:var(--bg-error);color:var(--fg-error)}disabled{background-color:var(--bg-disabled);color:var(--fg-disabled)}pre{background-color:var(--syntax-bg);color:var(--syntax-fg)}body>.content .link .rank{width:auto}.side{position:absolute;right:0;transition:width .2s ease-in-out;box-shadow:-5px 0 3px 0 rgba(0,0,0,.5);padding-left:10px;width:0;overflow-x:hidden}.side:hover{width:300px}.side .nub{opacity:0}.side:hover .nub{opacity:1}.eddit-filter-subreddit-link{border:.5px solid #f33;background-color:#f33;color:#fff!important;font-weight:900;border-radius:10px;padding:0 1px;margin-left:5px}.thing{padding:3px 10px;border-radius:10px}.thing.gilded:hover,.thing:hover{background-color:var(--bg-2);transition:background-color .2s cubic-bezier}.eddit-content-other{width:50%;margin-left:60px;border:var(--border)}.eddit-content-other p{margin:10px 0}.eddit-content-other img{max-width:50%}.eddit-content-other *,.eddit-content-other .content{background-color:transparent}.arrow.up{background-position:-84px -1618px}.arrow.down{background-position:-42px -1618px}#header-img.default-header,#mail.havemail,#mail.nohavemail,#modmail.havemail,#modmail.nohavemail,.arrow.down,.arrow.up,.filtered-details button.add.add,.icon-menu .moderator-mail::before,.icon-menu .reddit-edited::before,.icon-menu .reddit-moderationlog::before,.icon-menu .reddit-modqueue::before,.morelink,.morelink .nub,.thumbnail.default,.thumbnail.nsfw,.thumbnail.self,.thumbnail.spoiler{background-image:url()}.expando-button.collapsed.selftext,.expando-button.collapsed.video,.expando-button.expanded.selftext{background-image:url()}");
return this;
})();
Redditmod.Error = function(message, url) {
var div = document.createElement("div");
// TODO: Move style to stylesheet.
div.style = "background-color: #800; border: solid 0.5px #c00; color: #fff; font-weight: bold; font-size: 1.2rem; padding: 5px;";
div.classList.add("eddit-content-error");
div.textContent = message;
if (url) {
// TODO: Move style to stylesheet.
div.innerHTML += ' from <a href="' + url + '" target="_BLANK" style="color: #fff">' + url + '</a>';
}
return div;
};
Redditmod.ImagePromise = function(sourceURLs) {
if (!(this instanceof Redditmod.ImagePromise)) return new Redditmod.ImagePromise(sourceURLs);
var self = this;
this.sourceURLs = (sourceURLs instanceof String || typeof(sourceURLs) === "string") ? [sourceURLs] : sourceURLs;
this.currentIndex = 0;
this.img = null;
this.createAlbumNav = function() {
var albumStatus = document.createElement("span");
var albumPrevButton = document.createElement("a");
albumPrevButton.textContent = "<";
albumPrevButton.style = "cursor: pointer; font-size: 1.4rem;";
albumPrevButton.addEventListener("click", function(e) {
e.stopPropagation();
if (self.currentIndex === 0) {
self.currentIndex = index = self.sourceURLs.length;
}
self.currentIndex--;
albumStatus.textContent = (self.currentIndex + 1) + "/" + self.sourceURLs.length;
self.img.src = self.sourceURLs[self.currentIndex];
}, true);
var albumNextButton = document.createElement("a");
albumNextButton.textContent = ">";
albumNextButton.style = "cursor: pointer; font-size: 1.4rem;";
albumNextButton.addEventListener("click", function(e) {
e.stopPropagation();
if (self.currentIndex === self.sourceURLs.length - 1) {
self.currentIndex = -1;
}
self.currentIndex++;
albumStatus.textContent = (self.currentIndex + 1) + "/" + self.sourceURLs.length;
self.img.src = self.sourceURLs[self.currentIndex];
}, true);
albumStatus.textContent = "1/" + self.sourceURLs.length;
albumStatus.style = "cursor: default; font-size: 1.4rem;";
var albumNav = document.createElement("div");
albumNav.appendChild(albumPrevButton);
albumNav.appendChild(albumStatus);
albumNav.appendChild(albumNextButton);
return albumNav;
};
return new Promise(function(resolve, reject) {
var imageContainer = document.createElement("div");
if (self.sourceURLs.length > 1) {
imageContainer.appendChild(self.createAlbumNav());
}
self.img = document.createElement("img");
self.img.src = self.sourceURLs[0];
self.img.style["max-height"] = window.innerHeight + "px";
imageContainer.appendChild(self.img);
resolve(imageContainer);
});
};
Redditmod.VideoPromise = function(sourceURLs) {
return new Promise(function(resolve, reject) {
var video = document.createElement("video");
video.controls = false;
video.autoplay = true;
video.loop = true;
video.classList.add("eddit-content-video");
video.style.display = "block";
video.style.width = "auto";
video.style.height = "auto";
sourceURLs.forEach(function(sourceURL) {
var source = document.createElement("source");
source.src = sourceURL;
video.appendChild(source);
});
resolve(video);
});
};
Redditmod.GiphyPromise = function(url) {
// https://giphy.com/gifs/xUPGctxgaSqOpZx9zW
// https://media.giphy.com/media/xUPGctxgaSqOpZx9zW/giphy.gif
// https://media.giphy.com/media/xUPGctxgaSqOpZx9zW/giphy.mp4
var matches = url.href.match(/giphy\.com\/(?:gifs|media)\/(?:[a-z0-9\-]*-)?([a-z0-9]+)/i);
if (!matches) return null;
var shortcode = matches[1];
return Redditmod.VideoPromise([
"https://media.giphy.com/media/" + shortcode + "/giphy.mp4"
]);
};
Redditmod.GfycatPromise = function(url) {
var gfycatUrl, shortCode = url.href.match(/gfycat\.com\/(?:.*\/)?([a-z0-9]*)/i);
if (!shortCode) {
gfycatUrl = url.href;
} else {
gfycatUrl = "https://gfycat.com/" + shortCode[1];
}
return new Promise(function(resolve, reject) {
GM_xmlhttpRequest({
method: "GET",
url: gfycatUrl,
onabort: reject,
onerror: reject,
onload: function(response) {
try {
var html = document.createElement("html");
html.innerHTML = response.responseText;
var videoSource = html.querySelector("#webmSource").src;
Redditmod.VideoPromise([videoSource]).then(resolve, reject);
} catch (error) {
reject("Error (" + error + "): Failed to read " + gfycatUrl);
}
}
});
});
};
Redditmod.StreamablePromise = function(url) {
var matches = url.href.match(/streamable\.com\/([a-zA-Z0-9]*)/);
if (!matches) return;
var shortcode = matches[1];
var apiUrl = "https://api.streamable.com/videos/" + shortcode;
return new Promise(function(resolve, reject) {
GM_xmlhttpRequest({
method: "GET",
url: apiUrl,
onabort: reject,
onerror: reject,
onload: function(response) {
try {
var json = JSON.parse(response.responseText);
Redditmod.VideoPromise([json.files.mp4.url]).then(resolve, reject);
} catch (error) {
reject("Error (" + error + "): Failed to read " + apiUrl);
}
}
});
});
};
Redditmod.TwitchClipPromise = function(url) {
var twitchUrl = url.href;
return new Promise(function(resolve, reject) {
GM_xmlhttpRequest({
method: "GET",
url: twitchUrl,
onabort: reject,
onerror: reject,
onload: function(response) {
try {
var matches = response.responseText.match(/quality_options: (\[.*\]),/);
if (!matches) {
reject("Error: No 'quality_options' at " + twitchUrl);
} else {
var json = JSON.parse(matches[1]);
Redditmod.VideoPromise([json[0].source]).then(resolve, reject);
}
} catch (error) {
reject("Error (" + error + "): Failed to read " + twitchUrl);
}
}
});
});
};
Redditmod.XkcdPromise = function(url) {
var matches = url.href.match(/xkcd\.com\/([0-9]+)/);
if (!matches) return;
var shortcode = matches[1];
var apiUrl = "https://xkcd.com/" + shortcode + "/info.0.json";
return new Promise(function(resolve, reject) {
GM_xmlhttpRequest({
method: "GET",
url: apiUrl,
onabort: reject,
onerror: reject,
onload: function(response) {
try {
var json = JSON.parse(response.responseText);
var xkcdDiv = document.createElement("div");
var h3 = document.createElement("h3");
h3.textContent = json.title;
var img = document.createElement("img");
img.src = json.img;
img.title = json.alt;
var h5 = document.createElement("h5");
h5.textContent = json.alt;
xkcdDiv.appendChild(h3);
xkcdDiv.appendChild(img);
xkcdDiv.appendChild(h5);
resolve(xkcdDiv);
} catch (error) {
reject("Error (" + error + "): Failed to read " + apiUrl);
}
}
});
});
};
Redditmod.DeviantartPromise = function(url) {
var theUrl = url.href;
return new Promise(function(resolve, reject) {
GM_xmlhttpRequest({
method: "GET",
url: theUrl,
onabort: reject,
onerror: reject,
onload: function(response) {
try {
var html = document.createElement("html");
html.innerHTML = response.responseText;
var fullImg = html.querySelector('img[dev-content-full]');
var smallImg = html.querySelector('meta[property="og:image"]');
if (fullImg) {
Redditmod.ImagePromise([fullImg.getAttribute("src")]).then(resolve, reject);
} else if (smallImg) {
Redditmod.ImagePromise([smallImg.getAttribute("content")]).then(resolve, reject);
} else {
reject("No images found ", theUrl);
}
} catch (error) {
reject("Error (" + error + "): Failed to read ", theUrl);
}
}
});
});
};
Redditmod.TwitterPromise = function(url) {
var theUrl = url.href.replace(/[^\/]*\.twitter.com/, "twitter.com");
return new Promise(function(resolve, reject) {
GM_xmlhttpRequest({
method: "GET",
url: theUrl,
onabort: reject,
onerror: reject,
onload: function(response) {
try {
var html = document.createElement("html");
html.innerHTML = response.responseText;
var result = document.createElement("div");
var domTweet = html.querySelector("div.tweet");
if (domTweet) {
var resultUser = document.createElement("div");
resultUser.textContent = domTweet.dataset.screenName + " (" + domTweet.dataset.name + ")";
result.appendChild(resultUser);
}
var domDescription = html.querySelector('meta[property="og:description"]');
if (domDescription) {
var resultDescription = document.createElement("div");
resultDescription.textContent = domDescription.getAttribute("content");
result.appendChild(resultDescription);
}
var domVideo = html.querySelector('meta[property="og:video:url"]');
var domPicture = html.querySelector('meta[property="og:image:user_generated"]');
if (domVideo) {
var resultVideo = document.createElement("iframe");
resultVideo.width = html.querySelector('meta[property="og:video:width"]').getAttribute("content");
resultVideo.height = html.querySelector('meta[property="og:video:height"]').getAttribute("content");
resultVideo.src = domVideo.getAttribute("content");
resultVideo.cssText = "border: none";
result.appendChild(resultVideo);
} else if (domPicture) {
var resultPicture = document.createElement("img");
resultPicture.src = html.querySelector('meta[property="og:image"]').getAttribute("content");
result.appendChild(resultPicture);
}
resolve(result);
} catch (error) {
reject("Error (" + error + "): Failed to read ", theUrl);
}
}
});
});
};
Redditmod.LightshotPromise = function(url) {
var lightshotUrl = url.href;
return new Promise(function(resolve, reject) {
GM_xmlhttpRequest({
method: "GET",
url: lightshotUrl,
onabort: reject,
onerror: reject,
onload: function(response) {
try {
var html = document.createElement("html");
html.innerHTML = response.responseText;
var imageMeta = html.querySelector('meta[property="og:image"]');
if (imageMeta) {
Redditmod.ImagePromise([imageMeta.getAttribute("content")]).then(resolve, reject);
} else {
reject("No images found ", lightshotUrl);
}
} catch (error) {
reject("Error (" + error + "): Failed to read " + lightshotUrl);
}
}
});
});
};
Redditmod.ImgflipPromise = function(url) {
var imgflipUrl = url.href;
if (/\.(jpg|jpeg|png)$/i.test(imgflipUrl)) {
return Redditmod.ImagePromise([imgflipUrl]);
}
return new Promise(function(resolve, reject) {
GM_xmlhttpRequest({
method: "GET",
url: imgflipUrl,
onabort: reject,
onerror: reject,
onload: function(response) {
try {
var html = document.createElement("html");
html.innerHTML = response.responseText;
var imageMeta = html.querySelector('img#im');
if (imageMeta) {
Redditmod.ImagePromise([imageMeta.getAttribute("src")]).then(resolve, reject);
} else {
reject("No images found ", imgflipUrl);
}
} catch (error) {
reject("Error (" + error + "): Failed to read " + imgflipUrl);
}
}
});
});
};
Redditmod.QuestionablecontentPromise = function(url) {
var qcUrl = url.href;
if (/\.(jpg|jpeg|png)$/i.test(qcUrl)) {
return Redditmod.ImagePromise([qcUrl]);
}
return new Promise(function(resolve, reject) {
GM_xmlhttpRequest({
method: "GET",
url: qcUrl,
onabort: reject,
onerror: reject,
onload: function(response) {
try {
var html = document.createElement("html");
html.innerHTML = response.responseText;
var imageMeta = html.querySelector('img#strip');
if (imageMeta) {
Redditmod.ImagePromise(["http://questionablecontent.net/" + imageMeta.getAttribute("src")]).then(resolve, reject);
} else {
reject("No images found ", qcUrl);
}
} catch (error) {
reject("Error (" + error + "): Failed to read " + qcUrl);
}
}
});
});
};
Redditmod.InstagramPromise = function(url) {
var theUrl = url.href;
var matches = theUrl.match(/instagram\.com\/p\/([a-zA-Z0-9_\-]*)/);
if (!matches) reject("No images found", theUrl);
var shortcode = matches[1];
var apiUrl = "https://instagram.com/p/" + shortcode + "/";
return new Promise(function(resolve, reject) {
GM_xmlhttpRequest({
method: "GET",
url: apiUrl,
onabort: reject,
onerror: reject,
onload: function(response) {
try {
var html = document.createElement("html");
html.innerHTML = response.responseText;
var videoMeta = html.querySelector('meta[property="og:video"]');
var imageMeta = html.querySelector('meta[property="og:image"]');
if (videoMeta) {
Redditmod.VideoPromise([videoMeta.getAttribute("content")]).then(resolve, reject);
} else if (imageMeta) {
Redditmod.ImagePromise([imageMeta.getAttribute("content")]).then(resolve, reject);
} else {
reject("No images found ", apiUrl);
}
} catch (error) {
reject("Error (" + error + "): Failed to read " + apiUrl);
}
}
});
});
};
Redditmod.ImgurPromise = function(url) {
var href = url.href.replace(/\?.*/, "");
if (href.indexOf("/a/") >= 0 || href.indexOf("/gallery/") >= 0) {
return Redditmod.ImgurAlbumPromise(href);
} else if (/\.gifv$/.test(href) || /\.gif$/.test(href) || /\.mp4$/.test(href)) {
// it's a GIF/video.
href = href.replace(/\.(gifv|gif|mp4)$/, ".mp4");
return Redditmod.VideoPromise([href]);
} else {
href = href.replace(/[^/]*\.imgur\.com/, "i.imgur.com");
href = href.replace(/_[a-z]./, ".");
href = href.replace(/\.(gif|jpg|jpeg|png)$/i, "");
href = href + ".jpg";
return Redditmod.ImagePromise([href]);
}
};
Redditmod.ImgurAlbumPromise = function(url) {
var theUrlForReal = url;
return new Promise(function(resolve, reject) {
GM_xmlhttpRequest({
method: "GET",
url: theUrlForReal,
onabort: reject,
onerror: reject,
onload: function(response) {
// Parsing imgur album HTML for Javascript via Regex. Lord'avmercy
try {
var jsonChunks = response.response.match(/\s*image\s*:\s*(.*),\s*/);
var json = JSON.parse(jsonChunks[1] || "{}");
var album_images = json.album_images || {};
var images = album_images.images || [];
if (images.length === 0) {
// No images, it might be a "gallery" link.
if (/imgur\.com\/gallery/.test(theUrlForReal)) {
var imgurHtml = document.createElement("html");
imgurHtml.innerHTML = response.responseText;
var imgurImage = imgurHtml.querySelector('link[rel="image_src"]');
if (imgurImage) {
Redditmod.ImagePromise([imgurImage.getAttribute("href")]).then(resolve, reject);
} else {
reject("No images found ", theUrlForReal);
}
} else {
reject("No images found ", theUrlForReal);
}
} else {
var urls = images.map(function(image) {
return "https://i.imgur.com/" + image.hash + image.ext;
});
Redditmod.ImagePromise(urls).then(resolve, reject);
}
} catch (error) {
reject("Error (" + error + "): Failed to load imgur album ");
}
}
});
});
};
Redditmod.FlickrPromise = function(url) {
var theUrlForReal = url.href;
return new Promise(function(resolve, reject) {
GM_xmlhttpRequest({
method: "GET",
url: theUrlForReal,
onabort: reject,
onerror: reject,
onload: function(response) {
// Parsing flickr HTML for Javascript via Regex. Lord'avmercy
try {
var jsonChunks = response.response.match(/modelExport: (\{.*})/);
var json = JSON.parse(jsonChunks[1] || "{}");
var photo_models = json["photo-models"] || [];
var images = photo_models.map(function(model) {
var imageObjs = [];
for (var key in model.sizes) {
imageObjs.push(model.sizes[key]);
}
imageObjs = imageObjs.sort(function(a,b) {
return a.width < b.width;
});
if (imageObjs.length > 0) {
return window.location.protocol + imageObjs[0].url;
} else {
return null;
}
}).filter(function(imageUrl) {
return imageUrl !== null;
});
if (images.length === 0) {
reject("No images found ", theUrlForReal);
} else {
Redditmod.ImagePromise(images).then(resolve, reject);
}
} catch (error) {
reject("Error (" + error + "): Failed to load Flickr page ");
}
}
});
});
};
Redditmod.RedditCommentsPromise = function(url) {
var theUrlForReal = url.href;
return new Promise(function(resolve, reject) {
GM_xmlhttpRequest({
method: "GET",
url: theUrlForReal,
onabort: reject,
onerror: reject,
onload: function(response) {
try {
var html = document.createElement("html");
html.innerHTML = response.responseText;
var commentContainer = html.querySelector(".commentarea > .sitetable");
if (commentContainer) {
// Process incoming comments
commentContainer.querySelectorAll(".thing.comment").forEach(Redditmod.Comments.add);
resolve(commentContainer);
} else {
reject("Failed to find commentarea at ", theUrlForReal);
}
} catch (error) {
reject("Error (" + error + "): Failed to load page ", theUrlForReal);
}
}
});
});
};
Redditmod.CustomPromise = function(url) {
var theUrl = url.href;
return new Promise(function(resolve, reject) {
GM_xmlhttpRequest({
method: "GET",
url: theUrl,
onabort: reject,
onerror: reject,
onload: function(response) {
try {
var html = document.createElement("html");
html.innerHTML = response.responseText;
var domTitle = html.querySelector('meta[property="og:title"]');
var domBody = html.querySelector('meta[property="og:description"]');
var domDate = html.querySelector('meta[property="article:published_time"]');
if (!domTitle && !domBody) {
//Redditmod.OtherPromise(url).then(resolve, reject);
reject("Page data could not be loaded", theUrl);
return;
}
var result = document.createElement("div");
result.cssText = "max-width: 33%";
var resultTitleBox = document.createElement("h3");
var resultTitle = document.createElement("a");
resultTitle.textContent = domTitle.getAttribute("content");
resultTitle.href = theUrl;
resultTitle.target = "_BLANK";
resultTitle.cssText = "font-style: italic";
resultTitleBox.appendChild(resultTitle);
result.appendChild(resultTitleBox);
if (domDate) {
var resultDate = document.createElement("div");
resultDate.textContent = domDate.getAttribute("content");
result.appendChild(resultDate);
}
var resultBody = document.createElement("div");
resultBody.textContent = domBody.getAttribute("content");
result.appendChild(resultBody);
resolve(result);
} catch (error) {
reject("Error (" + error + "): Failed to load page ", theUrl);
}
}
});
});
}
Redditmod.OtherPromise = function(url) {
var theUrl = url.href;
return new Promise(function(resolve, reject) {
GM_xmlhttpRequest({
method: "GET",
headers: {"X-api-key": "NtFdFjTYzQXF4WUWBivfsnTj0zXZyvwCKbSQeuAB"},
url: "https://mercury.postlight.com/parser?url=" + encodeURIComponent(theUrl),
onload: function(response) {
try {
var json = JSON.parse(response.response);
if (json.content === "<body></body>") {
throw Error("Empty content");
}
var otherContent = document.createElement("div");
otherContent.innerHTML = json.content;
otherContent.classList.add("eddit-content-other");
resolve(otherContent);
} catch (error) {
//reject("Error (" + error + "): Failed to load page ");
Redditmod.CustomPromise(url).then(resolve, reject);
return;
}
},
onerror: function(xhr) {
//reject("Error (status:" + xhr.status + " " + xhr.statusText + ") ");
Redditmod.CustomPromise(url).then(resolve, reject);
return;
},
onabort: function(xhr) {
//reject("Error (status:" + xhr.status + " " + xhr.statusText + ") ");
Redditmod.CustomPromise(url).then(resolve, reject);
return;
}
});
});
};
Redditmod.VisitedLinks = (function() {
var self = this;
this._visitedLinks = GM_getValue("eddit-visited-links", {});
this.resetVisitedLink = document.createElement("a");
this.resetVisitedLink.textContent = "Reset Visited Links";
this.resetVisitedLink.href = "#";
this.resetVisitedLink.classList.add("choice");
this.resetVisitedLink.addEventListener("click", function(e) {
e.stopPropagation();
e.preventDefault();
GM_setValue("eddit-visited-links", {});
});
var dropdown = document.querySelector("#sr-header-area .drop-choices");
if (dropdown) {
dropdown.appendChild(document.createElement("hr"));
dropdown.appendChild(this.resetVisitedLink);
}
this.contains = function(link) {
return (self._visitedLinks[link] === true);
};
this.add = function(link) {
if (!self._visitedLinks[link]) {
self._visitedLinks[link] = true;
GM_setValue("eddit-visited-links", self._visitedLinks);
}
};
return {
contains: this.contains,
add: this.add
};
})();
Redditmod.MediaHandler = function(domPost) {
if (!(this instanceof Redditmod.MediaHandler)) return new Redditmod.MediaHandler(domPost);
var self = this;
this._domPost = domPost;
this._domCommentsLink = domPost.querySelector(".buttons a.comments");
this._loadedMedia = false;
this._loadedComments = false;
this._expandedMedia = false;
this._expandedComments = false;
this._mediaObj = null;
this._commentsObj = null;
this._shouldUseExpando = self._domPost.classList.contains("self");
this.url = (function() {
var thisUrl = self._domPost.getAttribute("data-url");
if (!thisUrl) {
return null;
}
if (thisUrl && thisUrl.indexOf("/") === 0) {
thisUrl = window.location.protocol + "//" + window.location.host + thisUrl;
}
try {
return new URL(thisUrl);
} catch (e) {
return thisUrl;
}
})();
this._loadMedia = function() {
if (self._loadedMedia) return;
self._loadedMedia = true;
self._expandedMedia = true;
if (self._expandedComments) self._hideComments();
if (self._domPost.classList.contains("self")) {
self._shouldUseExpando = true;
return;
}
var mediaPromise = Redditmod.MediaPromise(self.url);
if (mediaPromise instanceof Promise) {
mediaPromise.then(function(mediaDiv) {
mediaDiv.style["max-width"] = self._domPost.clientWidth + "px";
mediaDiv.style["max-height"] = (screen.availHeight - self._domPost.clientHeight - 20) + "px";
self._mediaObj = mediaDiv;
self._domPost.appendChild(mediaDiv);
}).catch(function(message, url) {
self._mediaObj = new Redditmod.Error(message, self.url)
self._domPost.appendChild(self._mediaObj);
});
} else {
self._shouldUseExpando = true;
}
};
this._showMedia = function() {
self._loadMedia();
self._expandedMedia = true;
if (self._shouldUseExpando) {
self._clickExpando();
} else if (self._mediaObj) {
self._mediaObj.style.display = "block";
if (self._mediaObj.tagName === "VIDEO") self._mediaObj.load();
}
if (self._expandoButton && self._expandoButton.classList.contains("collapsed")) {
self._expandoButton.classList.add("expanded");
self._expandoButton.classList.remove("collapsed");
}
};
this._hideMedia = function() {
if (!self._loadedMedia) return;
self._loadMedia();
self._expandedMedia = false;
if (self._shouldUseExpando) {
self._clickExpando();
} else if (self._mediaObj) {
self._mediaObj.style.display = "none";
if (self._mediaObj.tagName === "VIDEO") self._mediaObj.pause();
}
if (self._expandoButton && self._expandoButton.classList.contains("expanded")) {
self._expandoButton.classList.add("collapsed");
self._expandoButton.classList.remove("expanded");
}
};
this._loadComments = function() {
if (self._loadedComments) return;
if (!self._domCommentsLink) return;
self._loadedComments = true;
self._expandedComments = true;
var commentsUrl = new URL("https://reddit.com" + self._domCommentsLink.getAttribute("href"));
var commentsPromise = Redditmod.RedditCommentsPromise(commentsUrl);
commentsPromise.then(function(commentsDiv) {
commentsDiv.style["max-width"] = self._domPost.clientWidth + "px";
self._commentsObj = commentsDiv;
self._domPost.appendChild(commentsDiv);
}).catch(function(reason) {
self._domPost.appendChild(Redditmod.Error(reason, commentsUrl));
});
};
this._showComments = function() {
self._loadComments();
self._expandedComments = true;
if (self._commentsObj) {
self._commentsObj.style.display = "block";
}
};
this._hideComments = function() {
if (!self._loadedComments) return;
self._loadComments();
self._expandedComments = false;
if (self._commentsObj) {
self._commentsObj.style.display = "none";
}
};
this.markVisited = function() {
var linkTitle = self._domPost.querySelector("a.title");
if (!linkTitle) return;
linkTitle.classList.add("visited");
};
this._clickExpando = function() {
if (self._expandoButton) {
self._expandoButton.click();
}
};
this._postClick = function(event) {
var target = Redditmod.Utils.findThing(event);
if (!target) return;
target.scrollIntoView({behavior: "smooth"});
self._toggleMedia(event);
};
this._toggleComments = function(e) {
e.stopPropagation();
e.preventDefault();
if (self._expandedMedia) self._hideMedia();
if (self._expandedComments) {
self._hideComments();
} else {
self._showComments();
}
};
this._toggleMedia = function(e) {
e.stopPropagation();
e.preventDefault();
if (self._expandedComments) self._hideComments();
if (!self.url) return;
Redditmod.VisitedLinks.add(self.url.href);
self.markVisited();
if (self._expandedMedia) {
self._hideMedia();
} else {
self._showMedia();
}
};
this._expandoButton = (function() {
var button = self._domPost.querySelector(".expando-button");
if (!button) {
button = document.createElement("a");
button.classList.add("expando-button");
button.classList.add("collapsed");
button.classList.add("video");
button.onclick = self._toggleMedia;
var entry = self._domPost.querySelector(".entry");
var tagline = self._domPost.querySelector(".tagline");
// TODO: Apparently this doesn't work because button isn't a child of entry, or tagline, or something.
//entry.insertBefore(button, tagline);
}
return button;
})();
if (self._domPost) {
self._domPost.addEventListener("click", self._postClick);
}
if (self._domCommentsLink) {
self._domCommentsLink.addEventListener("click", self._toggleComments);
}
};
// top-level domain name (no subdomains)
var DOMAIN_NAME_REGEX = RegExp(/([a-z0-9\-]+\.[a-z]{2,}$)/);
/**
* @returns Promise for a <div> holding the content found at URL.
* Returns null if reddit's built-in expando should be used.
*/
Redditmod.MediaPromise = function(url) {
if (!(this instanceof Redditmod.MediaPromise)) return new Redditmod.MediaPromise(url);
var host, hostMatches = DOMAIN_NAME_REGEX.exec(url.host);
host = hostMatches ? hostMatches[1] : url.host;
if (host === "youtube.com" || host === "youtu.be" || host === "vimeo.com") {
return null; // Should use expando
} else if (url.host === "v.redd.it") {
return null;
}
// Custom media Promises
var hostToPromise = {
"gfycat.com": Redditmod.GfycatPromise,
"imgur.com": Redditmod.ImgurPromise,
"xkcd.com": Redditmod.XkcdPromise,
"instagram.com": Redditmod.InstagramPromise,
"flickr.com": Redditmod.FlickrPromise,
"streamable.com": Redditmod.StreamablePromise,
"twitch.tv": Redditmod.TwitchClipPromise,
"reddit.com": Redditmod.RedditCommentsPromise,
"giphy.com": Redditmod.GiphyPromise,
"deviantart.com": Redditmod.DeviantartPromise,
"twitter.com": Redditmod.TwitterPromise,
"prnt.sc": Redditmod.LightshotPromise,
"imgflip.com": Redditmod.ImgflipPromise,
"questionablecontent.net": Redditmod.QuestionablecontentPromise
};
if (host in hostToPromise) {
return hostToPromise[host](url);
}
var isImage = (/\.(gif|jpg|jpeg|png)/i.test(url.href) || host === "reddituploads.com");
if (isImage) {
return Redditmod.ImagePromise([url.href]);
}
return Redditmod.OtherPromise(url);
};
Redditmod.SubredditFilter = function(subreddit, enabled) {
if (!(this instanceof Redditmod.SubredditFilter)) return new Redditmod.SubredditFilter(subreddit, enabled);
var self = this;
self.subreddit = subreddit;
self.enabled = enabled;
self.filterLink = null;
this.init = function() {
self.filterLink = document.createElement("a");
self.filterLink.href = "#";
self.filterLink.classList.add("choice");
self.filterLink.textContent = self.subreddit;
self.filterLink.addEventListener("click", function(e) {
e.stopPropagation();
e.preventDefault();
self.toggle();
Redditmod.SubredditFilters.save();
});
if (self.enabled) {
self.enable();
} else {
self.disable();
}
var dropdown = document.querySelector("#sr-header-area .drop-choices");
dropdown.appendChild(self.filterLink);
};
this.disable = function() {
self.filterLink.classList.add("eddit-subreddit-disabled");
self.filterLink.classList.remove("eddit-subreddit-enabled");
self.filterLink.innerHTML = "☐ " + self.subreddit;
self.enabled = false;
};
this.enable = function() {
self.filterLink.classList.add("eddit-subreddit-enabled");
self.filterLink.classList.remove("eddit-subreddit-disabled");
self.filterLink.innerHTML = "☑ " + self.subreddit;
self.enabled = true;
};
this.toggle = function() {
if (self.enabled) {
self.disable();
} else {
self.enable();
}
};
this.init();
};
Redditmod.NsfwFilter = (function() {
var self = this;
this.enabled = GM_getValue("eddit-nsfw-filter", false);
this.filter = document.createElement("a");
this.filter.href = "#";
this.filter.classList.add("choice");
this.filter.addEventListener("click", function(e) {
e.stopPropagation();
e.preventDefault();
self.enabled = !self.enabled;
GM_setValue("eddit-nsfw-filter", self.enabled);
self.refreshNsfwFilter();
});
this.refreshNsfwFilter = function() {
if (self.enabled) {
self.filter.innerHTML = "☑ NSFW Filter";
} else {
self.filter.innerHTML = "☐ NSFW Filter";
}
if (Redditmod.Posts) {
Redditmod.Posts.refresh();
}
};
var dropdown = document.querySelector("#sr-header-area .drop-choices");
if (dropdown) {
dropdown.appendChild(document.createElement("hr"));
dropdown.appendChild(this.filter);
}
this.refreshNsfwFilter();
return this;
})();
/**
* Wrapper around filtered-subreddits config.
* Usage:
* if (!Redditmod.SubredditFilters.isFiltered("wtf")) {
* Redditmod.SubredditFilters.add("wtf");
* }
*/
Redditmod.SubredditFilters = (function() {
var self = this;
this._filters = {};
this._load = function() {
var dropdown = document.querySelector("#sr-header-area .drop-choices");
if (!dropdown) return;
dropdown.appendChild(document.createElement("hr"));
var filterHeader = document.createElement("h4");
filterHeader.textContent = "Filtered Subreddits";
dropdown.appendChild(filterHeader);
var selectAll = document.createElement("a");
selectAll.textContent = "Filter All";
selectAll.href = "#";
selectAll.style["padding-left"] = "10px";
selectAll.style["font-size"] = "0.8em";
selectAll.addEventListener("click", function(e) {
e.stopPropagation();
e.preventDefault();
Object.keys(self._filters).forEach(function(key) {
self._filters[key].enable();
});
Redditmod.Posts.refresh();
});
dropdown.appendChild(selectAll);
var selectNone = document.createElement("a");
selectNone.textContent = "Filter None";
selectNone.href = "#";
selectNone.style["padding-left"] = "10px";
selectNone.style["font-size"] = "0.8em";
selectNone.addEventListener("click", function(e) {
e.stopPropagation();
e.preventDefault();
Object.keys(self._filters).forEach(function(key) {
self._filters[key].disable();
});
Redditmod.Posts.refresh();
});
dropdown.appendChild(selectNone);
var subData = GM_getValue("eddit-filtered-subreddits", {});
for (var subreddit in subData) {
self._filters[subreddit] = Redditmod.SubredditFilter(subreddit, subData[subreddit]);
}
if (Redditmod.Posts) {
Redditmod.Posts.refresh();
}
};
this._stripAndLower = function(sub) {
sub = sub || "";
return sub.replace(/(^ +| +$)/, "").toLowerCase();
};
this._shouldFilterPage = function() {
var path = window.location.pathname;
if (/\/r\//.test(path)) {
var sub = path.match(/\/r\/([^?#\/]*)/)[1];
return sub === "all" || sub === "popular";
} else {
return false;
}
};
this.isFiltered = function(sub) {
var strippedSub = self._stripAndLower(sub);
if (self._shouldFilterPage() && strippedSub in self._filters) {
return self._filters[strippedSub].enabled === true;
} else {
return false;
}
};
this.add = function(sub) {
var strippedSub = self._stripAndLower(sub);
if (!(strippedSub in self._filters)) {
self._filters[strippedSub] = new Redditmod.SubredditFilter(strippedSub, true);
}
self._filters[strippedSub].enable();
self.save();
};
this.save = function() {
var toSave = {};
for (var sub in self._filters) {
toSave[sub] = self._filters[sub].enabled;
}
GM_setValue("eddit-filtered-subreddits", toSave);
if (Redditmod.Posts) {
Redditmod.Posts.refresh();
}
};
this._load();
return {
add: this.add,
save: this.save,
isFiltered: this.isFiltered
};
})();
/**
* Represents a post ("thing" in reddit-terms).
* @param thingElement - Thing DOM element on the page.
* Usage:
* var thing = Redditmod.Post(document.querySelector(".thing"));
*/
Redditmod.Post = function(domPost) {
if (!(this instanceof Redditmod.Post)) return new Redditmod.Post(domPost);
var self = this;
this.element = domPost;
this.subreddit = this.element.getAttribute("data-subreddit");
this.mediaHandler = new Redditmod.MediaHandler(this.element);
this.init = function() {
self._addFilterLink();
self.refresh();
if (document.querySelectorAll('#siteTable .thing.link[id="' + self.element.id + '"]').length > 1) {
self.element.classList.add("eddit-duplicate");
}
document.querySelectorAll('#siteTable .thing').forEach(function(thing) {
var thumbnail = thing.querySelector("a.thumbnail");
thumbnail.setAttribute("data-outbound-url", thumbnail.getAttribute("href"));
var titleLink = thing.querySelector("a.title");
titleLink.setAttribute("data-outbound-url", titleLink.getAttribute("href"));
});
};
this._addFilterLink = function() {
var filterLink = document.createElement("a");
filterLink.innerHTML = "×";
filterLink.href = "#";
filterLink.title = "Filter /r/" + self.subreddit + " from appearing";
filterLink.classList.add("eddit-filter-subreddit-link");
filterLink.addEventListener("click", self._filterLinkClick);
var tagLine = self.element.querySelector(".tagline");
if (tagLine) {
tagLine.appendChild(filterLink);
}
};
this._filterLinkClick = function(e) {
e.stopPropagation();
e.preventDefault();
Redditmod.SubredditFilters.add(self.subreddit);
};
this.hide = function() { self.element.classList.add("eddit-filtered-post"); };
this.show = function() { self.element.classList.remove("eddit-filtered-post"); };
this.refresh = function() {
if (Redditmod.SubredditFilters.isFiltered(self.subreddit)) {
self.hide();
} else if (Redditmod.NsfwFilter.enabled && self.element.classList.contains("over18")) {
self.hide();
} else {
if (Redditmod.VisitedLinks.contains(self.mediaHandler.url)) {
self.mediaHandler.markVisited();
}
self.show();
}
};
this.clickExpando = function() {
var button = self.element.querySelector(".expando-button");
if (button) {
button.click();
return true;
} else {
return false;
}
};
this.markVisited = function() {
var linkTitle = self.element.querySelector("a.title");
if (linkTitle) {
linkTitle.classList.add("visited");
}
};
this.init();
};
Redditmod.Posts = (function() {
var self = this;
this._things = [];
this.init = function() {
var postContainer = document.querySelector("#siteTable");
if (!postContainer) return;
var domPosts = postContainer.querySelectorAll(".thing.link");
domPosts.forEach(function(domPost) {
self._add(domPost);
});
self._postListener.observe(postContainer, {childList: true});
};
this._postListener = new MutationObserver(function(mutations) {
mutations.forEach(function(mutation) {
mutation.addedNodes.forEach(function(addedNode) {
if (addedNode.classList.contains("thing")) {
self._add(addedNode);
}
});
});
});
this._add = function(thingElement) {
self._things.push(new Redditmod.Post(thingElement));
};
this.refresh = function() {
self._things.forEach(function(thing) {
thing.refresh();
});
};
this.init();
return this;
})();
Redditmod.Nav = (function() {
var self = this;
// Flag when we are already loading the next page.
this.loading = false;
this.init = function() {
self.addScrollListener();
self._scrollListener();
self.overrideNextButton();
};
// Load more posts when user scrolls near bottom of the page.
this.addScrollListener = function() {
window.addEventListener("scroll", self._scrollListener);
};
this.removeScrollListener = function() {
window.removeEventListener("scroll", self._scrollListener);
};
this._scrollListener = function(event) {
var evt = event || {pageY:0};
if (document.body.clientHeight - (window.scrollY + window.innerHeight) < 200) {
self.loadMorePosts();
}
};
// Instead of navigating to the next page, use AJAX to load the posts.
this.overrideNextButton = function() {
var nextButton = document.querySelector(".next-button a");
if (!nextButton) return;
nextButton.addEventListener("click", function(e) {
e.stopPropagation();
e.preventDefault();
self.loadMorePosts();
});
};
// Inserts reddit posts from the AJAX response onto the current page.
this._injectPosts = function(response) {
var nav = document.querySelector(".nav-buttons");
// Convert AJAX response to DOM, add just the posts to the current page.
var nextPage = document.createElement("html");
nextPage.innerHTML = response.responseText;
nextPage.querySelectorAll("#siteTable > *").forEach(function(otherElement) {
nav.parentNode.insertBefore(otherElement, nav);
});
nav.parentNode.removeChild(nav);
// Re-enable features on the "new page".
self.overrideNextButton();
self.addScrollListener();
self.loading = false;
setTimeout(self._scrollListener, 250);
};
// Fetches posts from the current page's "next" button.
this.loadMorePosts = function() {
var nextButton = document.querySelector(".next-button a");
if (!self.loading && nextButton) {
self.loading = true;
self.removeScrollListener();
GM_xmlhttpRequest({
method: "GET",
url: nextButton.href,
onload: self._injectPosts
});
var parentNode = nextButton.parentNode.parentNode;
parentNode.style["background-color"] = "#aaa";
parentNode.opacity = "0.5";
parentNode.cursor = "not-allowed";
parentNode.childNodes.forEach(function(child) {
if (child.style) {
child.style.display = "none";
}
});
}
};
this.init();
})();
Redditmod.Utils = (function() {
var self = this;
/** Looks at the parents of the event's target until it hits a ".thing" */
this.findThing = function(event) {
var IGNORED_CLASSES = ["expando-button", "midcol"];
var UNIGNORED_CLASSES = ["thumbnail"];
var IGNORED_TAGS = ["A", "INPUT", "TEXTAREA", "BUTTON"];
var target = event.target, ignoredClass, ignoredTag, doNotIgnore, shouldIgnore;
while (!target.classList.contains("thing")) {
ignoredClass = IGNORED_CLASSES.find(function(c) { return target.classList.contains(c); }) !== undefined;
ignoredTag = IGNORED_TAGS.indexOf(target.tagName.toUpperCase()) >= 0;
doNotIgnore = UNIGNORED_CLASSES.find(function(c) { return target.classList.contains(c); }) !== undefined;
shouldIgnore = (ignoredClass || ignoredTag) && !doNotIgnore;
if (shouldIgnore) return null;
target = target.parentElement;
if (!target) return null;
}
return target;
};
return this;
})();
Redditmod.Comment = function(domComment) {
if (!(this instanceof Redditmod.Comment)) return new Redditmod.Comment(domComment);
var self = this;
this.element = domComment;
this.toggleCollapse = function(e) {
var target = Redditmod.Utils.findThing(e);
if (!target) return;
e.stopPropagation();
e.preventDefault();
if (target.classList.contains("noncollapsed")) {
target.classList.remove("noncollapsed");
target.classList.add("collapsed");
} else {
target.classList.remove("collapsed");
target.classList.add("noncollapsed");
}
};
domComment.querySelector(".entry").addEventListener("mouseenter", function(e) {
document.querySelectorAll(".eddit-comment-hover").forEach(function(element) {
element.classList.remove("eddit-comment-hover");
});
self.element.classList.add("eddit-comment-hover");
});
domComment.querySelector(".entry").addEventListener("mouseleave", function(e) {
self.element.classList.remove("eddit-comment-hover");
});
domComment.addEventListener("click", this.toggleCollapse);
};
Redditmod.Comments = (function() {
var self = this;
this._comments = [];
this.add = function(domComment) {
self._comments.push(Redditmod.Comment(domComment));
};
document.querySelectorAll(".thing.comment").forEach(self.add);
return {
add: this.add
};
})();