Soundgasm Improvements

Restyles and adds new functionality to Soundgasm --- dark mode/keyboard shortcuts/quick download/and more

Fra og med 04.10.2022. Se den nyeste version.

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==UserScript==
// @name        Soundgasm Improvements
// @namespace   V.L
// @version     0.12
// @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) {
        console.log(tag);
		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.", new Date() );
	
	// 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);
			console.log(title);
			
			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);
				 }
				console.log('descriptions done');
				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() {
			// Prevent Default Spacebar Function
			window.addEventListener('keydown', function(e) {
				if(e.keyCode == 32) {
				e.preventDefault();
				}
			});

			window.onkeyup = function(e) {
				// keylsit http://gcctech.org/csc/javascript/javascript_keycodes.htm
				var k = e.which || e.keyCode;
				var ctrl = e.ctrlKey;
				// p || k || space
				if(k === 80 || k === 75 || k === 32) {
					//play.click();
					if(!audio.paused) {
					audio.pause();
					} else {
					audio.play();
					}
				}
				if(k === 83) {
					stop.click();
					//audio.stop();
				}
				if(k === 68) {
					// dl.click();
					document.querySelector('.dl').click();
				}
				if(k === 37) {
					audio.currentTime -= 5.0;
				}
				if(k === 39) {
					audio.currentTime += 5.0;
				}
				if(ctrl & k === 37) {
					audio.currentTime -= 15.0;
				}
				if(ctrl & k === 39) {
					audio.currentTime += 15.0;
				}
				if(k === 38) {
					newVol = audio.volume + 0.1;
					if(newVol > 1) {
					newVol = 1.0;
					}
					audio.volume = newVol;
				}
				if(k === 40) {
					newVol = audio.volume - 0.1;
					if(newVol < 0) {
					newVol = 0.0;
					}
					audio.volume = newVol;
				}
				if(k === 48) {
					audio.currentTime = 0.0;
				}
				if(k === 49) {
					audio.currentTime = audio.duration / 10;
				}
				if(k === 50) {
					audio.currentTime = audio.duration / 10 * 2;
				}
				if(k === 51) {
					audio.currentTime = audio.duration / 10 * 3;
				}
				if(k === 52) {
					audio.currentTime = audio.duration / 10 * 4;
				}
				if(k === 53) {
					audio.currentTime = audio.duration / 10 * 5;
				}
				if(k === 54) {
					audio.currentTime = audio.duration / 10 * 6;
				}
				if(k === 55) {
					audio.currentTime = audio.duration / 10 * 7;
				}
				if(k === 56) {
					audio.currentTime = audio.duration / 10 * 8;
				}
				if(k === 57) {
					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) {
					console.log('==> audio loaded');
					// 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.", new Date() );
}