- // ==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)