您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Load a random reddit post from your subs
// ==UserScript== // @name Random reddit post // @description Load a random reddit post from your subs // @license MIT // @namespace http://tampermonkey.net/ // @version 0.1 // @description Random reddit post // @author John // @match https://old.reddit.com/* // @icon  // @run-at document-idle // @grant GM_registerMenuCommand // @grant GM_getValue // @grant GM_setValue // ==/UserScript== // a == random subreddit post // s == random post const verbose = true; const LOG_PREFIX = "[Random Reddit]:"; function navigate(currentDoc, path) { if (verbose) console.log(LOG_PREFIX, "Navigating to:", path); currentDoc.location = path; } function createButton(name) { let btn = document.createElement("a"); btn.innerHTML = name; btn.style = "text-decoration: none; font-size: 1.4em" btn.style.padding = "3px 3px 3px 3px"; // btn.style.border = "thick solid #FFFFFF"; // btn.style.margin = "3px 3px 3px 3px"; btn.href = "javascript:void(0)" let div = document.getElementById("header-bottom-right"); if (div) { div.insertBefore(btn, div.firstChild); } else { if (verbose) console.log(LOG_PREFIX, "header not ready"); } return btn; } function updateImages() { let maxHeight = window.innerHeight / 2.5; let width = '100%'; let linkElements = document.getElementsByClassName("link"); for (let linkElement of linkElements) { // linkElement.marginBottom = '10px'; // linkElement.width = width; // linkElement.style.maxWidth = width; // linkElement.style.width = width; } let thumbnailElements = document.getElementsByClassName("thumbnail"); // if (verbose) console.log(LOG_PREFIX, "thumbnails:", thumbnailElements) for (let thumbnailElement of thumbnailElements) { // thumbnailElement.style.maxWidth = width; thumbnailElement.style.width = "20%"; // thumbnailElement.style.width = 'auto'; // thumbnailElement.style.height = 'auto'; // thumbnailElement.style.height = '100px'; // thumbnailElement.width = '600px'; // thumbnailElement.marginBottom = '20%'; let img = thumbnailElement.childNodes.item(""); if (img) { var url = thumbnailElement.parentElement.getAttribute("data-url") // if (verbose) console.log(LOG_PREFIX, "url:", url) if (url != undefined && url != null && (url.includes(".jpg") || url.includes(".png") || url.includes(".jpeg")) ) { // if (verbose) console.log(LOG_PREFIX, "Setting url:", url) img.src = url; } // console.log(LOG_PREFIX, "img:", img) // img.style.maxHeight = '100%'; // img.maxWidth = width; // img.width = width; img.style.maxWidth = width; img.style.width = width; // img.style.width = '640px'; // img.style.width = '600px'; // img.style.maxHeight = '70%'; img.style.height = 'auto'; img.style.maxHeight = `${maxHeight}px`; img.style.objectFit = 'contain'; } else { // console.log(LOG_PREFIX, "No img"); } } } function shuffleArray(array) { for (var i = array.length - 1; i > 0; i--) { var j = Math.floor(Math.random() * (i + 1)); var temp = array[i]; array[i] = array[j]; array[j] = temp; } } // Blacklist, useful when overloaded with one subreddit const BLACKLIST_SR = [ "hentai" ]; function parsePosts(posts, filterMedia, startTime, getParams) { const prefix = window.location.protocol + "//" + window.location.host; if (filterMedia) { posts = posts.filter(function (post) { var isCrosspostVideo = false; var crosspostParent = post.data.crosspost_parent_list || undefined; // console.log("Post:", post) try { while (crosspostParent != undefined) { crosspostParent = crosspostParent[0]; // console.log("Crosspost parent:", crosspostParent); var postHint = crosspostParent.post_hint || ""; var url = crosspostParent.url || crosspostParent.link_url; if (postHint.includes("video") || url.includes(".mp4") || url.includes(".gif")) { isCrosspostVideo = true; console.log("isCrosspostVideo:", isCrosspostVideo) break; } crosspostParent = crosspostParent.crosspost_parent_list || undefined; } } catch (e) { } var postHint = post.data.post_hint || ""; var url = post.data.url || post.data.link_url; // console.log("Post:", postHint, url, post.data); // console.log("isCrosspostVideo 2:", isCrosspostVideo) // console.log("Post:", post.data.subreddit); var isValidPost = BLACKLIST_SR.some((sr) => window.location.href.includes(sr)) || !BLACKLIST_SR.includes(post.data.subreddit); return (postHint.includes("video") || url.includes(".mp4") || url.includes(".gif") || isCrosspostVideo || post.data.media) && isValidPost; }); if (verbose) console.log(LOG_PREFIX, 'Filter animated: ', posts); } if (verbose) console.log(LOG_PREFIX, 'Media posts: ', posts); if (posts.length > 0) { shuffleArray(posts); var randPost = posts[Math.floor(Math.random()*posts.length)]; // var randPostLink = prefix + randPost.data.permalink + `?${PREFIX_KEY}=` + (getParams || ""); var randPostLink = prefix + randPost.data.permalink + (getParams || ""); // console.log(LOG_PREFIX, randPost, randPostUrl); if (verbose) console.log(LOG_PREFIX, "Random post link:", randPostLink); navigate(document, randPostLink); } else { $("body").css("cursor", "default"); console.log("No media post") } var endTime = performance.now(); if (verbose) console.log(`Random post time taken: ${endTime - startTime} milliseconds`); } const PREFIX_KEY="prefix" const SEARCH_KEY="q" function randomPost(extraPath, filterMedia, getParams, onlyTop) { var startTime = performance.now() $("body").css("cursor", "progress"); if (verbose) console.log(LOG_PREFIX, "Random post:", extraPath, ", filter:", filterMedia, ", prefix:", getParams, ", only top:", onlyTop); const durationArray = ["all", "month", "week", "year"]; var duration = durationArray[Math.floor(Math.random()*durationArray.length)]; if (extraPath.startsWith("saved")) { duration = "all"; } // const sortArray = ["top", "new", "hot", "gilded"]; var sortArray = ["top", "new", "hot"]; var sortArrayWeights = [8, 2, 5]; if (onlyTop) { sortArray = ["top"]; sortArrayWeights = [1]; } // var sortType = sortArray[Math.floor(Math.random()*sortArray.length)]; var sortType = ""; const savedPath = extraPath.startsWith("saved") if (!savedPath) { sortType = weighted_random(sortArray, sortArrayWeights); } // const maxPostLimit = 100; // const minPostLimit = 25; // var postLimit = Math.floor(Math.random() * (maxPostLimit - minPostLimit + 1)) + minPostLimit; var postLimit = 100; if (verbose) console.log(LOG_PREFIX, "Sort type:", sortType, "post limit:", postLimit); var fetchUrl = `https://old.reddit.com/` + extraPath; fetchUrl += `${sortType}.json?`; if (getParams && getParams.includes("q")) { // Search fetchUrl = `https://old.reddit.com/` + `${extraPath}.json` + getParams + `&sort=${sortType}`; } fetchUrl += `&t=all&limit=${postLimit}`; if (verbose) console.log(LOG_PREFIX, "Fetch 1:", fetchUrl); fetch(fetchUrl) .then(result => result.json()) .then((output) => { if (verbose) console.log(LOG_PREFIX, 'Output: ', output); var posts = output.data.children; var lastPost = posts.at(-1); console.log("Last path:", lastPost.data.name); var fetchUrl = `https://old.reddit.com/` + extraPath; fetchUrl += `${sortType}.json?`; if (getParams && getParams.includes("q")) { // Search fetchUrl = `https://old.reddit.com/` + `${extraPath}.json` + getParams + `&sort=${sortType}`; } fetchUrl += `&t=${duration}&limit=${postLimit}&after=${lastPost.data.name}`; if (verbose) console.log(LOG_PREFIX, "Fetch 2:", fetchUrl); fetch(fetchUrl) .then(result => result.json()) .then((output) => { if (verbose) console.log(LOG_PREFIX, 'Output: ', output); // var posts = output.data.children; // console.log(LOG_PREFIX, 'All posts: ', posts); posts = posts.concat(output.data.children); // console.log(LOG_PREFIX, 'All posts: ', posts); // if (savedPath) { var lastPost = posts.at(-1); console.log("Last path:", lastPost.data.name); var fetchUrl = `https://old.reddit.com/` + extraPath; fetchUrl += `${sortType}.json?`; if (getParams && getParams.includes("q")) { // Search fetchUrl = `https://old.reddit.com/` + `${extraPath}.json` + getParams + `&sort=${sortType}`; } fetchUrl += `&t=${duration}&limit=${postLimit}&after=${lastPost.data.name}`; if (verbose) console.log(LOG_PREFIX, "Fetch 3:", fetchUrl); fetch(fetchUrl) .then(result => result.json()) .then((output) => { if (verbose) console.log(LOG_PREFIX, 'Output: ', output); // var posts = output.data.children; posts = posts.concat(output.data.children); // console.log(LOG_PREFIX, 'All posts: ', posts); parsePosts(posts, filterMedia, startTime, getParams); }).catch(err => console.error(err)) // } else { // parsePosts(posts, filterMedia, startTime); // } }).catch(err => console.error(err)) }).catch(err => console.error(err)) } function openInNewTab(url) { window.open(url, '_blank').focus(); } function openFirstVideo() { // Tries to open first video element var titleElements = document.getElementsByClassName("title"); for (let titleElement of titleElements) { try { if (titleElement.tagName == "A") { var videoLink = titleElement.href || titleElement.data-href-url; console.log("Opening video:", videoLink); // navigate(document, videoLink); openInNewTab(videoLink); break; } } catch (e) { console.error("Failed to open video:", e); } } } if (!window.location.pathname.includes("comments")) { var interval = setInterval(function(){ updateImages(); }, 1000); } else { var maxHeight = 550; var maxWidth = 300; try { var items = document.getElementsByClassName("expando"); // items[0].style.maxWidth = "100px"; // items[0].style.maxHeight = "1139px"; // items[0].style.maxHeight = "600px"; // // items[0].style.maxWidth = "811px"; // items[0].style.maxWidth = "300px"; items[0].style.maxHeight = `${maxHeight}px`; // items[0].style.maxWidth = `${maxWidth}px`; } catch (e) { // console.error("Failed to set video width:", e); } function allDescendants (node) { try { for (var i = 0; i < node.childNodes.length; i++) { var child = node.childNodes[i]; allDescendants(child); child.style.maxHeight = `${maxHeight}px`; // child.style.maxWidth = `${maxWidth}px`; // child.style.maxHeight = "600px"; // child.style.maxWidth = "300px"; } } catch (e) { // console.error("Failed to set video width:", e); } } allDescendants(items[0]); console.log("items:", items) var videoList = document.getElementsByTagName("video"); console.log("Videos:", videoList) for (var e of videoList) { // e.setAttribute('style', "width: 100px; height: 100px"); try { // var newHeight = document.body.clientHeight * 0.75; // console.log("New height:", newHeight); // e.parentElement.parentElement.parentElement.style = `height: ${newHeight}px` var videoWidth = e.parentElement.parentElement.parentElement.offsetWidth; e.setAttribute('style', `width: ${videoWidth}px`); console.log("New videoWidth:", videoWidth); } catch (e) { console.error("Failed to set video width:", e); } } } // var videoList = document.getElementsByTagName("video"); // console.log("Videos:", videoList) // for (var e of videoList) { // // e.setAttribute('style', "width: 100px; height: 100px"); // try { // var newHeight = document.body.clientHeight * 0.75; // console.log("New height:", newHeight); // e.parentElement.parentElement.parentElement.style = `height: ${newHeight}px` // var videoWidth = e.parentElement.parentElement.parentElement.offsetWidth; // e.setAttribute('style', `width: ${videoWidth}px`); // } catch (e) { // console.error("Failed to set video width:", e); // } // } // window.addEventListener('load', function() { // console.log("Window loaded"); // setTimeout(function(){}, 1000) // var videoList = document.getElementsByTagName("video"); // console.log("Videos:", videoList) // for (var e of videoList) { // // e.setAttribute('style', "width: 100px; height: 100px"); // try { // var newHeight = document.body.clientHeight * 0.75; // console.log("New height:", newHeight); // e.parentElement.parentElement.parentElement.style = `height: ${newHeight}px` // // var videoWidth = e.parentElement.parentElement.parentElement.offsetWidth; // // e.setAttribute('style', `width: ${videoWidth}px`); // } catch (e) { // console.error("Failed to set video width:", e); // } // } // }); // var observer = new MutationObserver(function(mutations) { // console.log("Mutations:", mutations) // // if (document.contains(myElement)) { // // console.log("It's in the DOM!"); // // observer.disconnect(); // // } // setTimeout(function(){}, 1000) // var videoList = document.getElementsByTagName("video"); // for (var e of videoList) { // // e.setAttribute('style', "width: 100px; height: 100px"); // try { // var newHeight = document.body.clientHeight * 0.75; // console.log("New height:", newHeight); // e.parentElement.parentElement.parentElement.style = `height: ${newHeight}px` // // var videoWidth = e.parentElement.parentElement.parentElement.offsetWidth; // // e.setAttribute('style', `width: ${videoWidth}px`); // } catch (e) { // console.error("Failed to set video width:", e); // } // } // }); // observer.observe(document, {attributes: false, childList: true, characterData: false, subtree:true}); // observer.observe(document, {attributes: true, childList: true, characterData: false, subtree:true}); // window.addEventListener('load', function() { const randSavedPostBtn = createButton("Saved (Q)"); randSavedPostBtn.onclick = () => { randomPost("saved/"); }; const randSavedMediaPostBtn = createButton("Media saved (W)"); randSavedMediaPostBtn.onclick = () => { randomPost("saved/", filterMedia = true); }; const randPostBtn = createButton("Rand post (S)"); randPostBtn.onclick = () => { randomPost(""); }; var randSrPostPath = ""; if (window.location.pathname != "/") { var pathArray = window.location.pathname.split('/'); const randSrPostBtn = createButton("Sr post (A)"); randSrPostBtn.onclick = () => { randomPost(`r/${pathArray[2]}/`); }; } const randMediaPostBtn = createButton("Media (B)"); randMediaPostBtn.onclick = () => { randomPost("", filterMedia = true); }; if (window.location.pathname != "/") { var pathArray = window.location.pathname.split('/'); const randSrMediaPostBtn = createButton("Media sr (V)"); randSrMediaPostBtn.onclick = () => { randomPost(`r/${pathArray[2]}/`, filterMedia = true); }; } const openVideoBtn = createButton("Open video"); openVideoBtn.onclick = () => { openFirstVideo(); }; document.onkeyup = function (e) { console.log("event:", e); e = e || window.event; var charCode = e.charCode || e.keyCode, character = String.fromCharCode(charCode); var tagName = e.target.nodeName.toLowerCase(); // if (tagName == "input" || tagName == "textarea" || e.shiftKey || e.ctrlKey) { if (tagName == "input" || tagName == "textarea" || e.ctrlKey) { if (verbose) console.log(LOG_PREFIX, "Ignoring input"); return; } if (e.code === 'Enter' && e.ctrlKey) { // if (e.code === 'Enter') { openFirstVideo(); return; } const onlyTop = !e.shiftKey; if (verbose) console.log(LOG_PREFIX, "Key press:", character); if (character.toUpperCase() == 'A') { if (window.location.pathname != "/") { var pathArray = window.location.pathname.split('/'); if (window.location.pathname.endsWith("search")) { // var path = parse_query_string(window.location.search.substring(1))[SEARCH_KEY] var prefix = window.location.search + `&${PREFIX_KEY}=${window.location.pathname}` randomPost(`${window.location.pathname}`, true, prefix, onlyTop); } else if (window.location.search.includes(PREFIX_KEY)) { var path = parse_query_string(window.location.search.substring(1))[PREFIX_KEY] randomPost(`${path}`, true, window.location.search, onlyTop); } else if (window.location.pathname.includes("user/") && window.location.pathname.includes("m/")) { // 1 to 4 var path = pathArray.slice(1,5).join("/") randomPost(`${path}/`, null, `?${PREFIX_KEY}=${window.location.pathname}`, onlyTop); } else { randomPost(`r/${pathArray[2]}/`, null, null, onlyTop); } } } if (character.toUpperCase() == 'S') { randomPost("", null, null, onlyTop); } if (character.toUpperCase() == 'V') { if (window.location.pathname != "/") { var pathArray = window.location.pathname.split('/'); if (window.location.pathname.endsWith("search")) { // var path = parse_query_string(window.location.search.substring(1))[SEARCH_KEY] var prefix = window.location.search + `&${PREFIX_KEY}=${window.location.pathname}` randomPost(`${window.location.pathname}`, filterMedia = true, prefix, onlyTop); } else if (window.location.search.includes(PREFIX_KEY)) { var path = parse_query_string(window.location.search.substring(1))[PREFIX_KEY] randomPost(`${path}`, true, window.location.search, onlyTop); } else if (window.location.pathname.includes("user/") && window.location.pathname.includes("m/")) { // 1 to 4 var path = pathArray.slice(1,5).join("/") randomPost(`${path}/`, true, `?${PREFIX_KEY}=${window.location.pathname}`, onlyTop); } else { randomPost(`r/${pathArray[2]}/`, filterMedia = true, undefined, onlyTop); } } } if (character.toUpperCase() == 'B') { randomPost("", filterMedia = true, undefined, onlyTop); } if (character.toUpperCase() == 'Q') { randomPost("saved/", null, null, onlyTop); } if (character.toUpperCase() == 'W') { randomPost("saved/", filterMedia = true, null, onlyTop); } }; // }, false); function parse_query_string(query) { var vars = query.split("&"); var query_string = {}; for (var i = 0; i < vars.length; i++) { var pair = vars[i].split("="); var key = decodeURIComponent(pair.shift()); var value = decodeURIComponent(pair.join("=")); // If first entry with this name if (typeof query_string[key] === "undefined") { query_string[key] = value; // If second entry with this name } else if (typeof query_string[key] === "string") { var arr = [query_string[key], value]; query_string[key] = arr; // If third or later entry with this name } else { query_string[key].push(value); } } return query_string; } // https://old.reddit.com/r/javascript/top.json?t=all // &sort=desc // &limit=250 function weighted_random(items, weights) { var i; for (i = 0; i < weights.length; i++) weights[i] += weights[i - 1] || 0; var random = Math.random() * weights[weights.length - 1]; for (i = 0; i < weights.length; i++) if (weights[i] > random) break; return items[i]; } function toggleHentai() { console.log("toggleHentai"); var enableHentai = GM_getValue("enable_hentai"); if (!enableHentai) { console.log("init enableHentai") GM_setValue("enable_hentai", true); enableHentai = true; } console.log("enableHentai:", enableHentai) // [].forEach.call( // ['contextmenu', 'copy', 'cut', 'paste', 'mouseup', 'mousedown', 'keyup', 'keydown', 'drag', 'dragstart', 'select', 'selectstart'], // function(event) { // document.addEventListener(event, function(e) { e.stopPropagation(); }, true); // } // ); } function enableCommandMenu() { var commandMenu = true; try { if (typeof(GM_registerMenuCommand) == undefined) { return; } else { if (commandMenu == true ) { GM_registerMenuCommand('Toggle Hentai', function() { toggleHentai(); }); } } } catch(err) { console.log(err); } } function init() { enableCommandMenu();} init(); // $("body").css("cursor", "default"); // var prefix = "https://old.reddit.com/"; // var sortArray = ["top", "new", "hot", "gilded"]; // var sortType = sortArray[Math.floor(Math.random()*sortArray.length)]; // // var sortArrayWeights = [8, 3, 4, 4]; // // var sortType = weighted_random(sortArray, sortArrayWeights); // console.log("Sort type:", sortType); // fetch(`https://old.reddit.com/r/javascript/${sortType}.json?t=all&limit=250`) // .then(result => result.json()) // .then((output) => { // console.log('Output: ', output); // var posts = output.data.children; // posts = posts.filter(function (post) { // var postHint = post.data.post_hint || ""; // var url = post.data.url || post.data.link_url; // console.log("Post:", postHint, url, post.data); // return postHint.includes("video") || url.includes(".mp4") || url.includes(".gif"); // }); // if (posts.length > 0) { // var randPost = posts[Math.floor(Math.random()*posts.length)]; // var randPostLink = prefix + randPost.data.permalink; // console.log(randPost, randPostLink); // } else { // console.log("No media post") // } // }).catch(err => console.error(err)); //https://old.reddit.com/r/javascript/top.json?t=all