您需要先安装一个扩展,例如 篡改猴、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 <spiralx@gmail.com> (http://github.com/spiralx)
- // @namespace http://spiralx.org/
- // @version 1.9.6
- // @icon 
- // @match *://*.reddit.com/r/*/comments/*
- // @grant none
- // @run-at document-end
- // @require https://unpkg.com/jquery@3/dist/jquery.min.js
- // @require https://greasyfork.org/scripts/7602-mutation-observer/code/mutation-observer.js
- // ==/UserScript==
- /* jshint asi: true, es6: true, laxbreak: true */
- /* global jQuery, MutationSummary */
- /*
- ==== 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
- */
- ; ($ => {
- 'use strict'
- const EXPAND_ICON = ''
- // --------------------------------------------------------------------
- const units = (v, s) => `${v}${s}`
- const pluralise = (w, n) => w + (n !== 1 ? 's' : '')
- const capitalise = s => typeof s === 'string' && s && s.split(/\s+/g).map(w => w[0].toUpperCase() + w.substr(1).toLowerCase()).join(' ')
- 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 options.mode === 'prepend'
- ? this.prepend($spinner)
- : this.append($spinner)
- },
- log (name, ...extras) {
- const title = [ `%c${name || '$'}%c : %c${this.length}%c ${pluralise('item', this.length)}`, 'font-weight: bold', '', 'color: #05f', '' ]
- 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: true,
- mode: 'append',
- steps: 3,
- size: 24,
- colour: '#28f',
- step_duration: 0.25
- }
- // --------------------------------------------------------------------
- async function getCommentPage (id) {
- const data = await $.get(postUrl + id)
- return $('.nestedlisting', data)
- }
- // --------------------------------------------------------------------
- function addComments ($target, $comments) {
- $target
- .empty()
- .append($comments)
- .find('.usertext.border .usertext-body')
- .css('animation', 'fadenewpost 4s ease-out 4s both')
- }
- // --------------------------------------------------------------------
- function loadComments ($span, $target, ids) {
- let insertChildren = false
- if (!Array.isArray(ids)) {
- ids = [ ids ]
- insertChildren = true
- }
- const urls = ids.map(id => postUrl + id)
- const original = $span.parent().html()
- $span.spinner()
- const pageRequests = urls.map(url => {
- return $.get(url)
- .then(
- data => $('.nestedlisting', data).get(),
- (xhr, textStatus, errorThrown) => {
- console.warn(`%c${capitalise(textStatus)}: ${xhr.status} ${xhr.statusText}%c ${url}`, 'font-weight: bold; color: #f4f', '')
- }
- )
- })
- $.when(...pageRequests)
- .then((...children) => {
- let $children = $([...flatten(children)])
- if (insertChildren) {
- $children = $children.find('> .thing > .child > .sitetable')
- $children
- .find('> .entry > .usertext.border')
- .removeClass('border')
- }
- $target
- .empty()
- .append($children)
- .find('.usertext.border .usertext-body')
- .css('animation', 'fadenewpost 4s ease-out 4s both')
- })
- .fail((xhr, textStatus, errorThrown) => {
- $span.parent().html(original)
- })
- }
- // --------------------------------------------------------------------
- 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 ])
- $span
- .attr('data-comment-ids', cid)
- .addClass('expand-inline')
- async function load () {
- $span.spinner()
- const $listing = await getCommentPage(cid)
- const $children = $listing.find('> .thing > .child > .sitetable')
- addComments($target, $children)
- }
- $a.one('click', event => {
- load()
- 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)
- const $target = $span.closest('.child')
- const $a = $span.children('a')
- const onclick = $a.attr('onclick')
- const cids = onclick.split(', ')[3].slice(1, -1).split(',')
- $span
- .attr('data-comment-ids', cids.join(','))
- .addClass('expand-inline')
- async function load () {
- $span.spinner()
- const $listings = $(await Promise.all(cids.map(getCommentPage)))
- addComments($target, $listings)
- }
- $a
- .removeAttr('onclick')
- .attr('data-onclick', onclick)
- .one('click', event => {
- load()
- return false
- })
- })
- }
- function processMoreCommentsSpans2 (moreCommentsSpans) {
- $(moreCommentsSpans).addClass('expand-inline')
- }
- // --------------------------------------------------------------------
- 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', '')
- // --------------------------------------------------------------------
- const observer = new MutationSummary({
- callback([ deepThreadSpans, moreCommentsSpans ]) {
- // console.log(`Added ${deepThreadSpans.added.length} deep thread spans and ${moreCommentsSpans.added.length} more comment spans`)
- processDeepThreadSpans(deepThreadSpans.added)
- processMoreCommentsSpans2(moreCommentsSpans.added)
- },
- rootNode: document.body,
- queries: [
- { element: 'span.deepthread' },
- { element: 'span.morecomments' }
- ]
- })
- // To process spans in the HTML source
- processDeepThreadSpans($('span.deepthread'))
- processMoreCommentsSpans2($('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: 32px;
- line-height: 32px;
- font-size: 1.2rem !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)