您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Changes 'Continue this thread' links to insert the linked comments into the current page
当前为
// ==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)