// ==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 <[email protected]> (http://github.com/spiralx)
// @namespace http://spiralx.org/
// @version 1.7.0
// @icon 
// @icon64 
// @match *://*.reddit.com/r/*/comments/*
// @grant none
// @run-at document-end
// @require https://ajax.googleapis.com/ajax/libs/jquery/2.2.0/jquery.js
// @require https://gf.zukizuki.org/scripts/7602-mutation-observer/code/mutation-observer.js
// ==/UserScript==
/* jshint asi: true, esnext: true */
/* global jQuery, MutationSummary */
; (function userScript($) {
'use strict'
const NORMAL = 'font-weight: normal; text-decoration: none; color: black'
const ERROR = 'font-weight: bold; color: #f4f'
const LINK = 'color: #05f; text-decoration: underline'
const BOLD = 'font-weight: bold'
const BLUE = 'color: #05f'
const EXPAND_ICON = ''
// --------------------------------------------------------------------
const units = (v, s) => `${v}${s}`
const pluralise = (w, n) => w + (n !== 1 ? 's' : '')
function* flatten (arr) {
for (let x of arr) {
if (Array.isArray(x)) {
yield* (flatten(x))
}
else {
yield x
}
}
}
// --------------------------------------------------------------------
$.fn.extend({
spinner (options) {
options = Object.assign({}, $.fn.spinner.defaults, options)
const $spinner = $('<div class="pulsar-horizontal"></div>')
.css({
padding: units(options.size * 0.25, 'px'),
height: units(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: units(options.size, 'px'),
height: units(options.size, 'px'),
backgroundColor: options.colour,
animationDuration: units(total_duration, 's'),
animationDelay: units(delay, 's')
})
.appendTo($spinner)
}
if (options.replace) {
this.empty()
}
return this.append($spinner)
},
log (name, ...extras) {
const title = [ `%c${name || '$'}%c : %c${this.length}%c ${pluralise('item', this.length)}`, BOLD, NORMAL, BLUE, NORMAL ]
if (this.length > 0 || extras.length > 0) {
console.group.apply(console, title)
if (this.length > 0) {
console.info(this)
}
extras.forEach(extra => {
console.log(extra)
})
console.groupEnd()
}
else {
console.info.apply(console, title)
}
return this
}
})
$.fn.spinner.defaults = {
replace: false,
steps: 3,
size: 16,
colour: 'white',
step_duration: 0.4
}
// --------------------------------------------------------------------
function loadComments ($span, $target, ...ids) {
const urls = ids.map(id => postUrl + id)
/*
console.group(`%cloadComments%c(${ids.join(', ')})`, BOLD, NORMAL)
console.info($span[0].outerHTML)
console.log(`%c${urls.join('\n')}%c`, LINK, NORMAL)
console.groupEnd()
*/
$span.spinner({
colour: '#28f',
size: 24,
step_duration: 0.25,
replace: true
})
const pageRequests = urls.map(url => {
return $.get(url)
.then(
data => $('.nestedlisting > .thing', data).next().andSelf().get(),
console.warn.bind(console)
)
})
$.when(...pageRequests).then((...$children) => {
const $all = $([...flatten($children)])
if ($all.length === 2) {
$all
.find('> .entry > .usertext.border')
.log('.border')
.removeClass('border')
}
$all
.replaceAll($target)
.find('.usertext.border .usertext-body')
.css('animation', 'fadenewpost 4s ease-out 4s both')
})
}
// --------------------------------------------------------------------
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),
$a = $span.children('a'),
cid = $a[0].pathname.split('/').pop()
const $target = $span.parents(`.thing[data-fullname=t1_${cid}]`)
$span
.attr('data-comment-ids', cid)
.addClass('expand-inline')
$a.one('click', event => {
loadComments($span, $target, cid)
return false
})
})
}
// --------------------------------------------------------------------
function processMoreCommentsSpans(moreCommentsSpans) {
const $moreCommentsSpans = $(moreCommentsSpans)
.filter(':not([data-comment-ids])')
// console.info(`processMoreCommentsSpans: processing ${$moreCommentsSpans.length}/${moreCommentsSpans.length} more comment spans`)
$moreCommentsSpans.each(function() {
const $span = $(this),
$target = $span.closest('.thing'),
$a = $span.children('a'),
cids = $a.attr('onclick').split(', ')[3].slice(1, -1).split(',')
$span
.attr('data-comment-ids', cids.join(','))
.addClass('expand-inline')
$a
.removeAttr('onclick')
.one('click', event => {
loadComments($span, $target, ...cids)
return false
})
})
}
// --------------------------------------------------------------------
const rootUrl = `https://${location.hostname}/`
const postUrl = $('.thing.link > .entry a.comments').prop('href')
// console.info(`%cSite:%c %c${rootUrl}%c\n%cPost:%c %c${postUrl}%c`, BOLD, NORMAL, LINK, NORMAL, BOLD, NORMAL, LINK, NORMAL)
// --------------------------------------------------------------------
const observer = new MutationSummary({
callback(summaries) {
const deepThreadSpans = summaries.shift().added,
moreCommentsSpans = summaries.shift().added
// console.log(`Added ${deepThreadSpans.length} deep thread spans and ${moreCommentsSpans.length} more comment spans`)
processDeepThreadSpans(deepThreadSpans)
processMoreCommentsSpans(moreCommentsSpans)
},
rootNode: document.body,
queries: [
{ element: 'span.deepthread' },
{ element: 'span.morecomments' }
]
})
// To process spans in the HTML source
processDeepThreadSpans($('span.deepthread'))
processMoreCommentsSpans($('span.morecomments'))
// --------------------------------------------------------------------
$(document.body).append(`<style type="text/css">
.expand-inline {
display: block;
padding: 0;
}
.expand-inline:after {
display: none !important;
}
.expand-inline a {
display: block;
background: transparent url(${EXPAND_ICON}) no-repeat center left;
padding-left: 40px;
height: 40px;
line-height: 40px;
font-size: 1.4rem !important;
font-weight: normal !important;
vertical-align: middle;
text-align: left;
}
.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)
jQuery.noConflict(true)