// ==UserScript==
// @name Reddit - Load 'Continue this thread' inline
// @description Changes 'Continue this thread' links to insert the linked comments into the current page
// @author James Skinner <spiralx@gmail.com> (http://github.com/spiralx)
// @namespace http://spiralx.org/
// @version 2.3.3
// @license MIT
// @icon 
// @match *://*.reddit.com/r/*/comments/*
// @match *://*.reddit.com/user/*/comments/*
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_setValue
// @grant GM_registerMenuCommand
// @grant GM_addStyle
// @grant GM_addValueChangeListener
// @grant GM.getValue
// @grant GM.setValue
// @grant GM.deleteValue
// @grant GM.registerMenuCommand
// @grant GM.addStyle
// @grant GM.addValueChangeListener
// @run-at document-end
// @require https://unpkg.com/jquery@3/dist/jquery.min.js
// @require https://unpkg.com/mutation-summary@1/dist/umd/mutation-summary.js
// @require https://greasyfork.org/scripts/371339-gm-webextpref/code/GM_webextPref.js?version=961539
// ==/UserScript==
/* jshint asi: true, esnext: true, laxbreak: true */
/* global jQuery, MutationSummary, GM_webextPref */
/*
==== 2.3.3 (2022.10.13) ====
* Actually fix bug supposedly fixed by previous version...
==== 2.3.2 (2022.08.28) ====
* Fix bug where clicking on "Continue this thread" after hover loading was triggered would open the comment's page
==== 2.3.1 (2022.06.26) ====
* Use GM_webextPref library to support Greasemonkey 4 users
==== 2.3.0 (2022.05.03) ====
* Fix centred text in expand links
* Add configuration for expanding links by moving the mouse over the text "Continue this thread" or "Load more comments"
==== 2.2.1 (2022.05.02) ====
* Make expand links a block again so they stretch across whole width
==== 2.2.0 (2022.05.01) ====
* Use MonkeyConfig library to provide settings for intersection observer behaviour
* CHanged styling of expandos and replaced icon with emoji ↘️
==== 2.1.0 (2022.04.17) ====
* Use IntersectionObserver to automatically open "Load more comments" when they scroll into view
* Put above behaviour behind USE_INTERSECTION_OBSERVER feature flag
==== 2.0.0 (2022.04.02) ====
* Added MIT license
* Expand non-top level collapsed comments on load
* Expand collapsed comments inserted from clicking "Load more comments" or "Continue this thread"
* Script now also runs on posts made to a user's homepage
* Remove old code handling "Load more comments" links
* Tidied up old code and updated to use current JS features
==== 1.9.7 (2021.11.05) ====
* Use MutationSummary from unpkg.com instead of Greasyfork
==== 1.9.6 (2020.08.08) ====
* Reduced size of load more links compared to comment text
* Fixed script icon
* Removed some unnecessary code
==== 1.9.5 (2018.07.11) ====
* Updated jQuery to v3 and source from unpkg.com
* Add downloadURL to update from Gist
==== 1.9.4 (2018.02.11) ====
* Added @icon field in metadata as SVG wasn't displaying on the installed userscript page
==== 1.9.3 (2017.12.03) ====
* Changed base-64 encoded PNG icons to an SVG icon
==== 1.9.2 (2017.10.11) ====
* Gets correct comment ID for links
* Changed location in comment HTML to use as its root
* Get children of first comment when it is already on the page
==== 1.9.1 (2017.10.11) ====
* Fix broken $target selector
==== 1.9.0 ====
* Catch failed loads, log them to the console and then restore original load link
*/
; (async ($, MutationSummary) => {
const config = GM_webextPref({
navbar: false,
default: {
autoExpandWhenVisible: false,
expandOnMouseOver: false,
expandOnMouseOverDelay: 500,
},
body: [
{
key: 'autoExpandWhenVisible',
label: 'Automatically expand any links when they come into view?',
type: 'checkbox',
},
{
key: 'expandOnMouseOver',
label: 'Expand links when you move the mouse over them?',
type: 'checkbox',
},
{
key: 'expandOnMouseOverDelay',
label: 'Delay between when you move the mouse over a link and it expands (ms)',
type: 'number',
},
],
onSave(newSettings) {
settings = newSettings
createOrDestroyIntersectionObserver()
addOrRemoveMouseoverHandler()
},
})
await config.ready()
config.on('change', changedSettings => {
settings = { ...settings, ...changedSettings }
createOrDestroyIntersectionObserver()
addOrRemoveMouseoverHandler()
})
let settings = config.getAll()
// --------------------------------------------------------------------
$.fn.extend({
spinner(options) {
options = {
replace: true,
mode: 'append',
steps: 3,
size: 24,
colour: '#28f',
step_duration: 0.25,
...options
}
const $spinner = $('<div class="pulsar-horizontal"></div>')
.css({
padding: `${options.size * 0.25}px`,
height: `${options.size}px`
})
const total_duration = (options.steps + 1) * options.step_duration
for (let i = 0; i < options.steps; i++) {
const delay = i * options.step_duration
$('<div></div>')
.css({
width: `${options.size}px`,
height: `${options.size}px`,
backgroundColor: options.colour,
animationDuration: `${total_duration}s`,
animationDelay: `${delay}s`
})
.appendTo($spinner)
}
if (options.replace) {
this.empty()
}
return options.mode === 'prepend'
? this.prepend($spinner)
: this.append($spinner)
},
log(name = '$') {
const title = [ `%c${name}%c : %c${this.length}%c ${this.length > 1 ? 'items' : 'item'}`, 'font-weight: bold', '', 'color: #05f', '' ]
if (this.length > 0) {
console.group(...title)
console.info(this)
console.groupEnd()
} else {
console.info(...title)
}
return this
}
})
// --------------------------------------------------------------------
async function loadAndInsertComments(cid, $span, $target) {
$target.data('loading', 'true')
$span.spinner()
const data = await $.get(postUrl + cid)
const $comments = $('.nestedlisting > .thing > .child > .sitetable', data)
$target
.empty()
.append($comments)
.find('.usertext.border .usertext-body')
.css('animation', 'fadenewpost 4s ease-out 4s both')
}
// --------------------------------------------------------------------
function getCommentId(linkElem) {
const m = linkElem.pathname.match(/\/([a-z0-9]+)\/?$/)
if (!m) {
throw new Error(`No comment ID parsed from link URL "${linkElem.href}"`)
}
return m[1]
}
// --------------------------------------------------------------------
function processDeepThreadSpans(deepThreadSpans) {
const $deepThreadSpans = $(deepThreadSpans)
.filter(':not([data-comment-ids])')
// console.info(`processDeepThreadSpans: processing ${$deepThreadSpans.length}/${deepThreadSpans.length} deep thread spans`)
$deepThreadSpans.each(function () {
const $span = $(this)
const $target = $span.closest('.child')
const $a = $span.children('a')
const cid = getCommentId($a[ 0 ])
let first = true
$span
.attr('data-comment-ids', cid)
.addClass('expand-inline')
$a
.wrapInner('<span class="expand-text"></span>')
.on('click', event => {
const loading = $target.data('loading')
if (first && !loading) {
first = false
loadAndInsertComments(cid, $span, $target)
}
return false
})
})
}
// --------------------------------------------------------------------
function uncollapseComments($collapsedComments) {
$collapsedComments
.removeClass('collapsed')
.addClass('noncollapsed')
.find('> .entry .tagline .expand')
.text('[-]')
}
function uncollapseAllComments($collapsedComments, depth = 3) {
// console.log($collapsedComments, depth)
if ($collapsedComments.length > 0 && depth > 0) {
uncollapseComments($collapsedComments)
requestAnimationFrame(() => {
uncollapseAllComments($collapsedComments.find('.thing.comment.collapsed'), depth - 1)
})
}
}
// --------------------------------------------------------------------
const rootUrl = `https://${location.hostname}/`
const postUrl = $('.thing.link > .entry a.comments').prop('href')
// console.info(`%cSite:%c ${rootUrl}\n%cPost:%c ${postUrl}`, 'font-weight: bold', '', 'font-weight: bold', '')
// --------------------------------------------------------------------
let intersectionObserver = null
function createOrDestroyIntersectionObserver() {
if (settings.autoExpandWhenVisible && !intersectionObserver) {
intersectionObserver = new IntersectionObserver(
(entries, observer) => {
for (const entry of entries) {
if (entry.isIntersecting) {
entry.target.click()
observer.unobserve(entry.target)
}
}
},
{
threshold: 0.5
}
)
$('span.morecomments, span.deepthread').each(function() {
intersectionObserver.observe(this.firstElementChild)
})
console.log('IntersectionObserver created')
} else if (!settings.autoExpandWhenVisible && intersectionObserver) {
intersectionObserver.disconnect()
intersectionObserver = null
console.log('IntersectionObserver destroyed')
}
}
createOrDestroyIntersectionObserver()
// --------------------------------------------------------------------
function addOrRemoveMouseoverHandler() {
$('.commentarea').off('mouseenter.spiralx')
if (settings.expandOnMouseOver) {
const hoveredElems = new WeakMap()
$('.commentarea')
.on('mouseenter.spiralx', '.expand-text', function() {
const elem = this
const timeoutId = setTimeout(() => {
hoveredElems.delete(elem)
elem.click()
}, settings.expandOnMouseOverDelay)
hoveredElems.set(elem, timeoutId)
})
.on('mouseleave.spiralx', '.expand-text', function() {
const timeoutId = hoveredElems.get(this)
if (timeoutId) {
clearTimeout(timeoutId)
hoveredElems.delete(this)
}
})
}
}
addOrRemoveMouseoverHandler()
// --------------------------------------------------------------------
function markAsExpand(selectorOrElements, observe = true) {
const $elems = $(selectorOrElements)
.addClass('expand-inline')
.children('a')
.wrapInner('<span class="expand-text"></span>')
if (intersectionObserver) {
$elems.each(function() {
intersectionObserver.observe(this.firstElementChild)
})
}
}
// --------------------------------------------------------------------
// Uncollapse non-top level comments on page load
uncollapseAllComments($('.thing.comment .thing.comment.collapsed'))
const observer = new MutationSummary({
callback([ deepThreadSpans, moreCommentsSpans, comments ]) {
// console.log(`Added ${deepThreadSpans.added.length} deep thread spans and ${moreCommentsSpans.added.length} more comment spans`)
markAsExpand(moreCommentsSpans.added)
processDeepThreadSpans(deepThreadSpans.added)
const $collapsedComments = $(comments.added).filter('.collapsed')
uncollapseAllComments($collapsedComments)
},
rootNode: document.body,
queries: [
{ element: 'span.deepthread' },
{ element: 'span.morecomments' },
{ element: '.thing.comment' },
]
})
// To process spans in the HTML source
markAsExpand('span.morecomments', false)
processDeepThreadSpans($('span.deepthread'))
// --------------------------------------------------------------------
$(document.body).append(`<style type="text/css">
.expand-inline {
display: block;
padding: 0;
}
.expand-inline:after {
display: none !important;
}
.expand-inline a {
display: block;
text-align: left;
}
.expand-inline a:before {
content: "↘️";
padding-right: 0.4em;
}
.expand-inline a:hover {
background-color: rgba(0, 105, 255, 0.05);
text-decoration: none;
}
.pulsar-horizontal {
display: inline-block;
}
.pulsar-horizontal > div {
display: inline-block;
border-radius: 100%;
animation-name: pulsing;
animation-timing-function: ease-in-out;
animation-iteration-count: infinite;
animation-fill-mode: both;
}
@keyframes pulsing {
0%, 100% {
transform: scale(0);
opacity: 0.5;
}
50% {
transform: scale(1);
opacity: 1;
}
}
@keyframes fadenewpost {
0% {
background-color: #ffc;
padding-left: 5px;
}
100% {
background-color: transparent;
padding-left: 0;
}
}
</style>`)
})(jQuery, MutationSummary?.MutationSummary)
jQuery.noConflict(true)