您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Restyles and adds new functionality to Soundgasm --- dark mode/keyboard shortcuts/quick download/and more
当前为
// ==UserScript== // @name Soundgasm Improvements // @namespace V.L // @version 0.14 // @description Restyles and adds new functionality to Soundgasm --- dark mode/keyboard shortcuts/quick download/and more // @author Valerio Lyndon // @match https://soundgasm.net/* // @run-at document-start // @grant GM_getValue // @grant GM_setValue // ==/UserScript== // document.onreadystatechange = function () { // if (document.readyState === 'interactive') { // preload(); // } // } document.addEventListener ("DOMContentLoaded", loaded); window.addEventListener ("load", fullyloaded); // Dark or Light mode theme = GM_getValue('theme', 'dark'); document.documentElement.classList.add(theme); // CSS var css = document.createElement('style'); css.textContent = ` html { font-size: 1px; --icons: url(); } html.dark { --background: hsl(0, 0%, 6.5%); --foreground-1: hsl(0, 0%, 12%); --foreground-bar-2: hsl(0, 0%, 15%); --foreground-bar: hsl(0, 0%, 27%); --foreground-2: hsl(0, 0%, 17.6%); --border: var(--foreground-bar-2); --text-low: hsl(0, 0%, 65%); --text-medium: hsl(0, 0%, 80%); --text-high: hsl(0, 0%, 98%); --accent: hsl(310, 30%, 30%); } html.light { --background: hsl(0, 0%, 96%); --foreground-1: hsl(0, 0%, 100%); --foreground-bar-2: hsl(0, 0%, 13.3%); --foreground-bar: hsl(0, 0%, 9%); --foreground-2: hsl(0, 0%, 94%); --border: hsl(0, 0%, 94%); --text-low: hsl(0, 0%, 25%); --text-medium: hsl(0, 0%, 7%); --text-high: hsl(0, 0%, 0%); --accent: hsl(310, 30%, 70%); } html body { max-width: 800rem; padding: 40rem; margin: 0 auto; background: var(--background); font-size: 12rem; color: var(--text-low); } a { color: var(--text-medium) !important; text-decoration: none; } a:hover { color: var(--text-high) !important; } html *::selection { background-color: var(--accent); } body input, body textarea { background: var(--foreground-2); border: 1px solid var(--border); color: var(--text-medium); resize: vertical; } body input[type="submit"]:hover, body input[type="submit"]:active { cursor: pointer; border-color: var(--accent); } /* Header */ body header { min-height: 20rem; padding-bottom: 40rem; text-align: center; } nav a { display: inline-block; } body .logo { display: none; } nav a[href="https://soundgasm.net/logout"] { font-size: 0; } nav a[href="https://soundgasm.net/logout"]::before { content: "Logout"; font-size: 16px; } /* Multiple-page rules */ body #container, body .sound-details, #jp_container_1, body .uploadform, body .contactform, body .loginform, body .signupform, body .passwordresetform, .vl-sidebar { background: var(--foreground-1); box-shadow: 0 2rem 4rem var(--background), 0 4rem 10rem hsla(0,0%,0%,10%); border-color: var(--border); margin: 0 auto; } #container h1, body h1 { border-color: var(--border); color: var(--text-low); } /* Generic Container */ body p.footer { border-color: var(--border); } /* User Page */ body .sound-details { display: flex; width: 620rem; border-radius: 4rem; //margin: 0 0 12rem; margin: 0 auto 12rem; flex-flow: row wrap; } .sound-details > a { max-width: calc(100% - 70px); font-size: 16rem; font-weight: bold; white-space: normal; } .playCount { max-width: 70px; margin-left: auto; text-align: right; } .playCount::before { content: ""; display: inline-block; border-color: transparent; border-left-color: var(--text-low); border-style: solid; border-width: .45em .65em; margin-right: -0.4em; vertical-align: middle; } .soundDescription { order: 3; width: 100%; margin-top: 6rem; } /* Sort Header */ .vl-sortheader { height: 20rem; margin-bottom: 25rem; text-align: center; } .vl-sortheader a { display: inline-block; padding: 0 15rem; vertical-align: top; } .vl-sortheader a.active { font-weight: bold; } .vl-sortheader a.active::after { content: attr(data-direction); display: block; color: var(--text-low); font-size: 10px; } .vl-clearbtn { display: none; } .active ~ .vl-clearbtn { display: inline-block; } /* Sidebar */ .vl-sidebar { position: absolute; right: 0; top: 0; width: 120rem; padding: 10rem; border-radius: 4rem; } /* Player Page */ div[style="margin:10px 0"] { margin: 0 0 25rem !important; font-size: 18rem; text-align: center; } #jp_container_1, .jp-audio .jp-audio-stream, .jp-audio .jp-video { border: 2rem solid var(--border); color: var(--text-low); } #jp_container_1 { width: 420rem; } .jp-interface { background: var(--foreground-bar); } .jp-audio .jp-details { background: var(--foreground-bar-2); } .jp-details .jp-title { font-size: 12rem; } .light .jp-details .jp-title { color: var(--background); } .jp-description { padding: 0 10rem; font-size: 12rem; } /* Player */ .jp-state-muted .jp-unmute { background: url("../image/jplayer.blue.monday.jpg") -60px -170px no-repeat; } .jp-state-muted .jp-unmute:focus { background: url("../image/jplayer.blue.monday.jpg") -79px -170px no-repeat; } #jp_container_1 button, .jp-gui .jp-seek-bar, .jp-gui .jp-play-bar, .jp-gui .jp-volume-bar, .jp-gui .jp-volume-bar-value { background-image: var(--icons); } .jp-gui .jp-progress { background: none; border-radius: 2.5rem; } .jp-progress .jp-seeking-bg { background: var(--icons) 0 -202px repeat-x; animation: seeking .8s ease-in-out infinite alternate; } @keyframes seeking { 0% { opacity: 1; } 100% { opacity: 0.3; } } .dark .jp-current-time, .dark .jp-duration { color: var(--text-medium); } .light .jp-current-time, .light .jp-duration { color: var(--background); } /* Description */ .vl-desc-container { margin: 12rem 0 0; } .sound-details .vl-desc-container { display: inline; margin: 0; } .vl-desc-new, .vl-desc-raw { white-space: pre-wrap; margin: 12rem 0; } .sound-details .vl-desc-new, .sound-details .vl-desc-raw { display: inline; white-space: normal; } .vl-tag { display: inline-block; padding: 2rem 4rem; background: var(--foreground-2); border-radius: 2.5rem; margin: 0 4rem 4rem 0; color: var(--text-medium); font-size: 11rem; text-transform: capitalize; } .vl-showraw { display: inline-block; opacity: 0.5; } .vl-showraw:hover { opacity: 1; } .jp-audio .vl-showraw { margin-bottom: 12rem; } .sound-details .vl-showraw { float: right; } /* Contact page */ header + ul { width: 414rem; padding-left: 16rem; margin: 12rem auto; word-break: break-word; } /* Footer */ .vl-footer { width: 420rem; margin: 0 auto; text-align: center; padding-top: 30rem; } .vl-footer a { padding: 0 15rem; } /* Loading Spinner */ .vl-loader-parent { position: fixed; bottom: 0; left: 0; width: 100%; padding: 6px; box-sizing: border-box; } .vl-loader { display: flex; height: 40px; padding: 0 12px; border-radius: 10px; background: var(--foreground-1); box-shadow: 0 1rem 4rem var(--background), 0 2rem 10rem hsla(0,0%,0%,10%); justify-content: center; align-items: center; float: right; } .vl-loader-icon { width: 16px; height: 16px; border: 3px solid transparent; border-left-color: var(--text-low); border-top-color: var(--text-low); border-radius: 50%; animation: 1s cubic-bezier(.54,.39,.45,.63) 0s infinite spin; } .vl-loader-text { white-space: nowrap; margin: 0 6px; } @keyframes spin { from { transform: rotate(30deg); } to { transform: rotate(390deg); } } `; document.documentElement.appendChild(css); // Functions & Classes function processDescription(desc, descDest, title, titleDest) { var originalTitle = title, originalDesc = desc, processedTitleDiv = document.createElement('span'), rawTitleDiv = document.createElement('span'), processedDescDiv = document.createElement('div'), rawDescDiv = document.createElement('p'), tagsDiv = document.createElement('div'), descDiv = document.createElement('p'); rawTitleDiv.textContent = title; rawTitleDiv.style.display = 'none'; processedDescDiv.classList.add('vl-desc-container'); tagsDiv.classList.add('vl-tags'); descDiv.classList.add('vl-desc-new'); processedDescDiv.appendChild(tagsDiv); processedDescDiv.appendChild(descDiv); rawDescDiv.classList.add('vl-desc-raw'); rawDescDiv.textContent = desc; rawDescDiv.style.display = 'none'; // match regex and iterate matches into an array combined = title + desc; var tagIterator = combined.matchAll(/(?:[\[\{](.*?)[\]\}]|\(([^\s]+)\))/g); /* todo: comment this regex because it's an abomination */ // todo: remove duplicates tags = []; for(tag of tagIterator) { if(typeof tag[1] !== 'undefined') { tags.push(tag[1]); } else if(typeof tag[2] !== 'undefined') { tags.push(tag[2]); } } // remove tags from text title = title.replace(/(?:\s|^|)*(?:[\[\{].*?[\]\}]|\([^\s]+\))(?:\s|$|)*/g, '') desc = desc.replace(/(?:\s|^|)*(?:[\[\{].*?[\]\}]|\([^\s]+\))(?:\s|$|)*/g, '') // sort by length tags.sort( (a, b) => { return a.length - b.length; } ); // create the element for(i = 0; i < tags.length; i++) { var tagSpan = document.createElement('span'); tagSpan.classList.add('vl-tag'); tagSpan.textContent = tags[i]; tagsDiv.appendChild(tagSpan); } // Create "view raw" button var viewRawBtn = document.createElement('a'); viewRawBtn.href = '#'; viewRawBtn.classList.add('vl-showraw'); viewRawBtn.textContent = 'Show raw.' viewRawBtn.onclick = function() { if(processedDescDiv.style.display === 'none') { processedTitleDiv.style.display = 'inline'; rawTitleDiv.style.display = 'none'; processedDescDiv.style.display = 'block'; rawDescDiv.style.display = 'none'; viewRawBtn.textContent = 'Show raw.'; } else { processedTitleDiv.style.display = 'none'; rawTitleDiv.style.display = 'inline'; processedDescDiv.style.display = 'none'; rawDescDiv.style.display = 'block'; viewRawBtn.textContent = 'Show processed.'; } } // finish up with tags & description descDiv.textContent = desc.trim(); processedTitleDiv.textContent = title.trim(); if(title === originalTitle && desc === originalDesc) { return false; } // Add everything back to DOM //destination.innerHMTL = ""; <-- this doesn't work for some reason so instead we use a while loop while(descDest.firstChild){ descDest.removeChild(descDest.firstChild); } while(titleDest.firstChild){ titleDest.removeChild(titleDest.firstChild); } titleDest.appendChild(processedTitleDiv); titleDest.appendChild(rawTitleDiv); descDest.appendChild(processedDescDiv); descDest.appendChild(rawDescDiv); descDest.appendChild(viewRawBtn); } class Loader { constructor(description = "Loading...", total = null) { this.desc = description; this.count = 0; this.total = total; this.container = document.querySelector('.vl-loader-parent'); if(this.container === null) { this.container = document.createElement('div'); this.container.classList.add('vl-loader-parent'); } document.body.appendChild(this.container); this.element = document.createElement('div'); this.element.classList.add('vl-loader'); this.element.innerHTML = `<div class="vl-loader-icon"></div> <span class="vl-loader-text">${this.desc}</span>`; this.container.appendChild(this.element); } enableCounter() { this.counter = document.createElement('span'); this.counter.classList.add('vl-loader-count'); this.element.appendChild(this.counter); this.refreshCounter(); } refreshCounter() { this.counter.innerText = `${this.count} of ${this.total}`; } setCount(count) { this.count = count; this.refreshCounter(); } show() { this.element.style.display = 'block'; } hide() { this.element.style.display = 'none'; } } // Begin modifying page function loaded() { console.log ("==> DOM is loaded."); // If content is blank var content = document.querySelector('body > div'); if(content === null) { var blank = document.createElement('div'); blank.id = 'container'; blank.innerHTML = `<div id="body"><p>There's nothing here.</p></div>`; document.body.appendChild(blank); } // Add footer var footer = document.createElement('footer'); footer.classList.add('vl-footer'); // Theme switcher var themeSwitcher = document.createElement('a'); themeSwitcher.textContent = 'Theme'; themeSwitcher.href = '#'; themeSwitcher.onclick = function() { if(GM_getValue('theme', 'dark') === 'dark') { GM_setValue('theme', 'light'); document.documentElement.classList.add('light'); document.documentElement.classList.remove('dark'); } else { GM_setValue('theme', 'dark'); document.documentElement.classList.add('dark'); document.documentElement.classList.remove('light'); } }; footer.appendChild(themeSwitcher); document.body.appendChild(footer); var path = window.location.pathname; if(path.slice(-1) === '/') { path = path.substr(0, path.length - 1); } // user page if(path.startsWith('/u/') && path.split('/').length < 4) { var items = document.querySelectorAll('.sound-details'); // Add loading spinner var spin = new Loader('Processing descriptions...', items.length); spin.enableCounter(); // Add custom descriptions // var descriptions = document.querySelectorAll('.sound-details'); // for(i = 0; i < descriptions.length; i++) { // var descDest = descriptions[i].querySelector('.soundDescription'), // desc = descDest.textContent, // titleDest = descriptions[i].querySelector('a'), // title = titleDest.textContent; // processDescription(desc, descDest, title, titleDest); // } var k = 0; // slowly feeds descriptions into process because otherwise the browser likes to crash. I have a feeling I coded this wrong. function feedDesc() { var descDest = items[k].querySelector('.soundDescription'), desc = descDest.textContent, titleDest = items[k].querySelector('a'), title = titleDest.textContent; processDescription(desc, descDest, title, titleDest); k++; spin.setCount(k); if(k < items.length) { timer = k * 3 < 200 ? k * 7 : 200; setTimeout(feedDesc, timer); } else { for(i = 0; i < items.length; i++) { items[i].setAttribute('data-title', items[i].querySelector('a').textContent); } spin.hide(); } } feedDesc(); // Modify playcounts & add sort data for(i = 0; i < items.length; i++) { var item = items[i], title = item.querySelector('a').textContent, countEle = item.querySelector('.playCount'); item.setAttribute('data-order', i); item.setAttribute('data-title', title); var count = countEle.textContent.split(': ')[1]; item.setAttribute('data-count', count); if(count.length > 3) { //count = count.substr(0, count.length - 3) + ',' + count.substr(count.length - 3, 3); count = count.substr(0, count.length - 3) + 'k'; } countEle.textContent = count; } // Prep for sort columns document.body.style.display = "flex"; document.body.style.flexDirection = "column"; document.querySelector('header').style.order = '-1'; document.querySelector('footer').style.order = '99999'; // Add sort columns var sortHeader = document.createElement('div'); sortHeader.classList.add('vl-sortheader'); sortHeader.textContent = 'Sort by: '; function addSortBtn(title, defaultDirection = 'desc') { var ele = document.createElement('a'); ele.href = '#'; ele.setAttribute('data-direction', defaultDirection); ele.textContent = title; sortHeader.appendChild(ele); return ele; } function sortByTitle() { var btns = document.querySelectorAll('.vl-sortheader a'); for(i = 0; i < btns.length; i++) { btns[i].classList.remove('active'); } titleBtn.classList.add('active'); direction = titleBtn.getAttribute('data-direction'); if(direction === 'desc') { titleBtn.setAttribute('data-direction', 'asc'); } else { titleBtn.setAttribute('data-direction', 'desc'); } direction = titleBtn.getAttribute('data-direction'); var array = []; for(i = 0; i < items.length; i++) { var order = items[i].getAttribute('data-order'), title = items[i].getAttribute('data-title'); array.push([order, title]); } array.sort( (first, second) => { var a = first[1].toUpperCase(), b = second[1].toUpperCase(); return (a < b) ? -1 : (a > b) ? 1 : 0; } ); if(direction === 'asc') { for(i = 0; i < array.length; i++) { var item = document.querySelector('[data-order="'+array[i][0]+'"]'); item.style.order = i; } } else { for(i = 0; i < array.length; i++) { var item = document.querySelector('[data-order="'+array[i][0]+'"]'); item.style.order = array.length - i; } } } titleBtn = addSortBtn('Title', 'desc'); titleBtn.onclick = sortByTitle; function sortByCount() { var btns = document.querySelectorAll('.vl-sortheader a'); for(i = 0; i < btns.length; i++) { btns[i].classList.remove('active'); } countBtn.classList.add('active'); direction = countBtn.getAttribute('data-direction'); if(direction === 'desc') { countBtn.setAttribute('data-direction', 'asc'); } else { countBtn.setAttribute('data-direction', 'desc'); } direction = countBtn.getAttribute('data-direction'); var array = []; for(i = 0; i < items.length; i++) { var order = items[i].getAttribute('data-order'), count = items[i].getAttribute('data-count'); array.push([order, count]); } array.sort( (first, second) => { return first[1] - second[1] } ); if(direction === 'asc') { for(i = 0; i < array.length; i++) { var item = document.querySelector('[data-order="'+array[i][0]+'"]'); item.style.order = i; } } else { for(i = 0; i < array.length; i++) { var item = document.querySelector('[data-order="'+array[i][0]+'"]'); item.style.order = array.length - i; } } } countBtn = addSortBtn('Play Count', 'asc'); countBtn.onclick = sortByCount; function sortByDate() { var btns = document.querySelectorAll('.vl-sortheader a'); for(i = 0; i < btns.length; i++) { btns[i].classList.remove('active'); } dateBtn.classList.add('active'); direction = dateBtn.getAttribute('data-direction'); if(direction === 'desc') { dateBtn.setAttribute('data-direction', 'asc'); } else { dateBtn.setAttribute('data-direction', 'desc'); } direction = dateBtn.getAttribute('data-direction'); if(direction === 'asc') { for(i = 0; i < items.length; i++) { items[i].style.order = items.length - items[i].getAttribute('data-order'); } } else { for(i = 0; i < items.length; i++) { items[i].style.order = items[i].getAttribute('data-order'); } } } dateBtn = addSortBtn('Date Uploaded', 'desc'); dateBtn.onclick = sortByDate; function clearSort() { var btns = document.querySelectorAll('.vl-sortheader a'); for(i = 0; i < btns.length; i++) { btns[i].classList.remove('active'); } for(i = 0; i < items.length; i++) { items[i].style.order = ""; } } clearBtn = addSortBtn('clear'); clearBtn.onclick = clearSort; clearBtn.classList.add('vl-clearbtn'); document.body.insertBefore(sortHeader, document.querySelector('.sound-details')); // Add filters // var sidebarAnchor = document.createElement('div'), // sidebar = document.createElement('div'); // sidebarAnchor.id = 'sidebar-anchor'; // sidebarAnchor.style.position = 'relative'; // sidebarAnchor.appendChild(sidebar); // sidebar.classList.add('vl-sidebar'); // sidebar.textContent = 'Filter by tag'; // document.body.insertBefore(sidebarAnchor, document.querySelector('.sound-details')); } // player page if(path.startsWith('/u/') && path.split('/').length > 3) { // Add custom descriptions var desc = document.querySelector('.jp-description p').textContent, descDest = document.querySelector('.jp-description'), titleDest = document.querySelector('.jp-title'), title = titleDest.textContent; processDescription(desc, descDest, title, titleDest); // basic variables var play = document.querySelector('.jp-play'), stop = document.querySelector('.jp-stop'), title = document.querySelector('.jp-title'), author = document.querySelector('div[style="margin:10px 0"] a'), audio = document.querySelector('audio'); // Keypress handler function setKeybinds() { window.addEventListener('keydown', (e) => { let k = e.key.toLowerCase(); if(e.key === ' ') { e.preventDefault(); } }); window.addEventListener('keyup', (e) => { let k = e.key.toLowerCase(); let ctrl = e.ctrlKey; let time = 5.0; if(ctrl){ time = 15.0; } if(k === 'p' || k === 'k' || k === ' ') { if(!audio.paused) { audio.pause(); } else { audio.play(); } } else if(k === 's') { stop.click(); } else if(k === 'd') { document.querySelector('.dl').click(); } else if(k === 'arrowleft') { audio.currentTime -= time; } else if(k === 'arrowright') { audio.currentTime += time; } else if(k === 'arrowup') { newVol = audio.volume + 0.1; if(newVol > 1) { newVol = 1.0; } audio.volume = newVol; } else if(k === 'arrowdown') { newVol = audio.volume - 0.1; if(newVol < 0) { newVol = 0.0; } audio.volume = newVol; } else if(k === '0') { audio.currentTime = 0.0; } else if(k === '1') { audio.currentTime = audio.duration / 10; } else if(k === '2') { audio.currentTime = audio.duration / 10 * 2; } else if(k === '3') { audio.currentTime = audio.duration / 10 * 3; } else if(k === '4') { audio.currentTime = audio.duration / 10 * 4; } else if(k === '5') { audio.currentTime = audio.duration / 10 * 5; } else if(k === '6') { audio.currentTime = audio.duration / 10 * 6; } else if(k === '7') { audio.currentTime = audio.duration / 10 * 7; } else if(k === '8') { audio.currentTime = audio.duration / 10 * 8; } else if(k === '9') { audio.currentTime = audio.duration / 10 * 9; } }); } // Download button function addDownload() { var audio = document.querySelector('audio'), src = audio.getAttribute('src'), ext = src.split('.').pop(), dl = document.createElement('a'); dl.classList.add('dl'); footer.appendChild(dl); dl.href = src; dl.setAttribute("download", title.innerText + ' by ' + author.innerText + '.' + ext); dl.setAttribute("target", "_blank"); dl.textContent = 'Download this audio'; } // Wait for audio to load if(audio !== null && audio.getAttribute('src') !== null) { addDownload(); } else { function audioLoaded() { audio = document.querySelector('audio'); if(audio !== null && audio.getAttribute('src') !== null) { // observer.disconnect(); addDownload(); setKeybinds(); } else { setTimeout(audioLoaded, 100); } } audioLoaded(); } } // signup page if(window.location.pathname.startsWith('/signup')) { var h1 = document.querySelector('h1'), form = document.querySelector('.signupform'); form.prepend(h1); } } function fullyloaded () { console.log ("==> Page is fully loaded, including images." ); }
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址