// ==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
// @homepageURL https://github.com/ValerioLyndon/Soundgasm-Improvements
// @supportURL https://github.com/ValerioLyndon/Soundgasm-Improvements/issues
// @license GPL-3.0-only
// @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." );
}