Soundgasm Improvements

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

Tendrás que instalar una extensión para tu navegador como Tampermonkey, Greasemonkey o Violentmonkey si quieres utilizar este script.

Necesitarás instalar una extensión como Tampermonkey o Violentmonkey para instalar este script.

Necesitarás instalar una extensión como Tampermonkey o Violentmonkey para instalar este script.

Necesitarás instalar una extensión como Tampermonkey o Userscripts para instalar este script.

Necesitará instalar una extensión como Tampermonkey para instalar este script.

Necesitarás instalar una extensión para administrar scripts de usuario si quieres instalar este script.

(Ya tengo un administrador de scripts de usuario, déjame instalarlo)

Necesitará instalar una extensión como Stylus para instalar este estilo.

Necesitará instalar una extensión como Stylus para instalar este estilo.

Necesitará instalar una extensión como Stylus para instalar este estilo.

Necesitará instalar una extensión del gestor de estilos de usuario para instalar este estilo.

Necesitará instalar una extensión del gestor de estilos de usuario para instalar este estilo.

Necesitará instalar una extensión del gestor de estilos de usuario para instalar este estilo.

(Ya tengo un administrador de estilos de usuario, déjame instalarlo)

// ==UserScript==
// @name        Soundgasm Improvements
// @namespace   V.L
// @version     1.0
// @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==

'use strict';

document.addEventListener("DOMContentLoaded", domLoaded);

// Dark or Light mode

const theme = GM_getValue('theme', 'dark');
document.documentElement.classList.add(theme);

// CSS

const 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 {
		min-width: 424rem;
		max-width: 1200rem;
		padding: 40rem 20rem;
		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,
	.vl-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 Containers */

	body p.footer {
		border-color: var(--border);
	}

	body #container {
		max-width: 800rem;
	}

	.vl-container {
		max-width: 770rem;
		padding: 15rem;
		border-size: 1rem;
		margin: 30rem auto;
	}

	.vl-container-header {
		padding-bottom: 10rem;
		border-bottom: 1px solid var(--border);
		margin: 0 0 14rem;
		font-size: 16px;
		font-weight: normal;
	}

	.vl-column {
		display: flex;
		flex-flow: column nowrap;
		gap: 5rem 10rem;
		margin-bottom: 5rem;
		align-items: start;
	}

	.vl-link {
		color: var(--text-medium);
	}
	.vl-link:hover {
		color: var(--text-high);
		text-decoration: underline;
	}


	.vl-paragraph {
		margin: 0 0 5rem;
		font-size: 12rem;
		line-height: 1.35;
	}

	/* Home Page */

	.vl-user-list {
		display: grid;
		grid-template-columns: repeat(4, 1fr);
		gap: 5rem;
		justify-items: start;
		align-items: start;
	}

	/* User Page */

	body .sound-details {
		display: grid;
		width: calc(100% - 22rem);
		padding: 10rem;
		border-radius: 4rem;
		margin: 0 auto 12rem;
		grid-template-columns: 1fr 70rem;
		grid-template-rows: auto auto;
		grid-template-areas:
			"title plays"
			"description description";
		grid-auto-flow: column;
	}


	.sound-details > a {
		grid-area: title;
		justify-self: start;
		font-size: 16rem;
		font-weight: bold;
		white-space: normal;
	}

	.playCount {
		grid-area: plays;
		max-width: 70rem;
		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 {
		grid-area: description;
		order: 3;
		width: 100%;
		margin-top: 6rem;
	}

	/* Split Content + Sidebar */

	.vl-split {
		display: grid;
		max-width: calc(640rem + 20rem + 240rem);
		gap: 20rem;
		margin: 0 auto;
	}

	.vl-directory {
		display: grid;
		width: 100vw;
		max-width: 620rem;
		grid-auto-flow: row;
		margin: 0 auto;
	}

	.vl-sidebar {
		padding: 10rem;
		border-radius: 4rem;
		overflow-y: auto;
	}

	@media (max-width: 941px) {
		.vl-split {
			grid-auto-flow: row;
		}
		.vl-sidebar {
			max-width: 600rem;
			max-height: 80vh;
			border: 1px solid var(--border);
		}
	}
	@media (min-width: 940px) {
		.vl-split {
			grid-auto-flow: column;
			align-items: start;
		}
		.vl-sidebar {
			position: sticky;
			top: 20rem;
			width: 220rem;
			max-height: calc(100vh - 160rem);
			order: 1;
		}
	}

	/* Filters */

	.vl-hidden,
	.vl-hidden-by-search,
	.vl-hidden-by-tag {
		display: none !important;
	}

	.vl-filters {
		font-size: 12rem;
	}

	.vl-filter-section {
		margin-bottom: 15rem;
	}

	.vl-filter-header {
		border-bottom: 1rem solid var(--border);
		margin: 0 0 10rem;
		font-size: 14rem;
		line-height: 1.25em;
		font-weight: bold;
	}

	.vl-sort-btn::before {
		content: "• ";
	}

	.vl-sort-btn.is-active {
		font-weight: bold;
	}
	.vl-sort-btn.is-active::after {
		content: attr(data-direction);
		color: var(--text-low);
		font-size: 10rem;
		margin-left: 5rem;
	}

	.vl-search {
		padding: 4rem;
		border-radius: 3px;
		margin: 0 0 5rem;
	}

	.vl-tag-list {
		width: 100%;
		text-align: left;
	}
	.vl-tag-list th {
		font-size: 12rem;
	}
	.vl-tag-list tr > *:last-child {
		text-align: right;
	}

	.vl-tag-btn {
		display: inline;
		padding: 2rem 4rem;
		background: none;
		border: none;
		border-radius: 2.5rem;
		color: var(--text-medium);
		font-size: 11rem;
		text-align: left;
		text-transform: capitalize;
		cursor: pointer;
	}
	.vl-tag-btn.is-active,
	.vl-tag-btn:hover {
		background: var(--foreground-2);
		color: var(--text-high);
	}
	.vl-tag-btn.is-active {
		font-weight: bold;
	}

	.vl-tag-count {
		padding: 1rem;
		font-size: 12rem;
	}

	/* 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;
		}
	}

	.jp-gui .jp-volume-controls {
		width: 0;
	}

	.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 {
		margin: 12rem 0;
	}
	.sound-details .vl-desc-new, .sound-details .vl-desc-raw:not([style*="none"]) {
		display: inline !important;
		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;
	}


	/* fixes */
	
	.patreon-widget {
		width: 300px !important;
		height: 36px !important;
	}
`;

document.documentElement.appendChild(css);

// Functions & Classes

function paragraph( text ){
	let p = document.createElement('p');
	p.className = 'vl-paragraph';
	p.textContent = text;
	return p;
}

var users = new class UserDatabase {
	constructor( ){
		let storage = GM_getValue('knownUsernames', '[]');
		this.names = JSON.parse(storage);
	}

	generateUserList( ){
		
	}

	add( name ){
		let index = this.names.indexOf(name);
		if( index === -1 ){
			this.names.push(name);
			this.save();
		}
	}

	remove( name ){
		let index = this.names.indexOf(name);
		if( index > -1 ){
			this.names.splice(index, 1);
			this.save();
		}
	}

	save( ){
		console.log('saving', JSON.stringify(this.names));
		GM_setValue('knownUsernames', JSON.stringify(this.names));
	}
}

class AudioListing {
	constructor({ element, titleSelector, descriptionSelector, playCountSelector = false, order = 0 }){
		// Process description
		let titleDestination = element.querySelector(titleSelector);
		let descDestination = element.querySelector(descriptionSelector);
		let desc = descDestination.textContent;
		let title = titleDestination.textContent;
		let originalTitle = title;
		let originalDesc = desc;
		let processedTitleDiv = document.createElement('span');
		let rawTitleDiv = document.createElement('span');
		let processedDescDiv = document.createElement('div');
		let rawDescDiv = document.createElement('p');
		let tagsDiv = document.createElement('div');
		let descDiv = document.createElement('p');
		let tags = new Set();
		// match all words inside brackets [] {}. also matches parentheses () but only for one-word sections to try and avoid false positives
		const extractTagsRegex = /[\[\{](.*?)[\]\}]|\(([^\s]+)\)/g;
		// same as the extraction regex but with extra whitespace matching rules
		const removeTagsRegex = /\s*(?:[\[\{].*?[\]\}]|\([^\s]+\))\s*/g;

		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';

		let combined = title + desc;
		var tagMatches = combined.matchAll(extractTagsRegex);

		for( let match of tagMatches ){
			let tag = match[1] === undefined ? match[2] : match[1];
			
			if( tag.length > 0 ){
				tags.add(tag);
			}
		}

		// remove tags from text
		title = title.replaceAll(removeTagsRegex, '')
		desc = desc.replaceAll(removeTagsRegex, '')

		// sort the tags by length
		tags = Array.from(tags);
		tags.sort( (a, b) => { return a.length - b.length; } );

		// create the element
		for( let 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 = 'javascript:void(0);';
		viewRawBtn.classList.add('vl-showraw');
		viewRawBtn.textContent = 'Show raw.'
		viewRawBtn.onclick = ()=>{
			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();

		// Add everything back to DOM unless it is identical
		if( title !== originalTitle || desc !== originalDesc ){
			titleDestination.replaceChildren(processedTitleDiv, rawTitleDiv);
			descDestination.replaceChildren(processedDescDiv, rawDescDiv, viewRawBtn);
		}

		// parse play count and change html
		let plays = false;
		if( playCountSelector ){
			// this if else and element.append are place here due to weird HTML bugs on the default website
			let playElement = element.querySelector(playCountSelector);
			if( !playElement ){
				plays = 0;
				playElement = document.createElement('span');
				playElement.className = 'playCount';
				playElement.textContent = 'unknown';
			}
			else {
				plays = playElement.textContent.split(': ')[1];

				let playText = String(plays);
				if( plays.length > 3 ) {
					playElement.title = `played ${plays} times`;
					playText = playText.substring(0, plays.length - 3) + 'k';
				}
				playElement.textContent = playText;
			}
			element.append(playElement);
		}

		// Assign variables for use in AudioDirectory classes
		this.element = element;
		this.title = title;
		this.description = desc;
		this.plays = plays;
		this.order = order;
		this.tags = tags;
	}
}

class AudioDirectory {
	constructor({ elements, titleSelector, descriptionSelector, playCountSelector = false, filterElement = false }){
		this.audios = []; // used to iterate through audio info
		this.tags = {}; // used as a reference for which items have which tags
		this.selectedTags = []; // used to keep track of current filters
		this.availableElements = []; // used to update tag counts in the sidebar
		this.tagButtons = []; // used to update tag counts in the sidebar

		// process all audios
		for( let index = 0; index < elements.length; index++ ){
			let audio = new AudioListing({
				element: elements[index],
				titleSelector: titleSelector,
				descriptionSelector: descriptionSelector,
				playCountSelector: playCountSelector,
				order: index
			});
			this.audios.push(audio);
			for( let tag of audio.tags ){
				let tagName = tag.toLowerCase();
				if(!( tagName in this.tags )){ this.tags[tagName] = []; }
				this.tags[tagName].push(audio.element);
			}
		}

		// intialise filters and sorting
		if( filterElement ){
			// set up DOM
			this.filterElement = filterElement;
			this.filterElement.classList.add('vl-filters');

			this.sortElement = document.createElement('div');
			this.sortElement.className = 'vl-filter-section';
			this.searchElement = document.createElement('div');
			this.searchElement.className = 'vl-filter-section';
			this.tagElement = document.createElement('div');
			this.tagElement.className = 'vl-filter-section';

			let sortHeader = document.createElement('h6');
			sortHeader.className = 'vl-filter-header';
			sortHeader.textContent = 'Sort by...';
			this.sortElement.append(sortHeader);

			let searchHeader = document.createElement('h6');
			searchHeader.className = 'vl-filter-header';
			searchHeader.textContent = 'Search...';
			this.searchElement.append(searchHeader);

			let tagHeader = document.createElement('h6');
			tagHeader.className = 'vl-filter-header';
			tagHeader.textContent = 'Filter by tag...';
			this.tagElement.append(tagHeader);

			this.sortList = document.createElement('div');
			this.sortList.className = 'vl-column';
			this.sortElement.append(this.sortList);

			this.tagList = document.createElement('table');
			this.tagList.className = 'vl-tag-list';
			this.tagList.insertAdjacentHTML("afterbegin", `
				<tr><th>Name</th><th>Count</th></tr>
			`);
			this.tagElement.append(this.tagList);

			// set up sorting
			this.calculatedSorts = {};
			this.sortButtons = {};

			this.createSortButton('Title', 'title', 'ascending');
			this.createSortButton('Play Count', 'plays', 'descending');
			this.createSortButton('Date', 'order', 'descending');

			this.sort();

			// set up search

			this.searchBar = document.createElement('input');
			this.searchBar.type = 'search';
			this.searchBar.className = 'vl-search';
			this.searchBar.placeholder = 'Search here.';
			this.searchElement.append(this.searchBar);
			this.searchElement.append(paragraph(`Search supports some basic operators. For example: "phrase" to require an exact string or word and -word to excluse a word. Un-quoted words are treated as OR.`));

			this.searchBar.addEventListener('input', ()=>{ this.search() });
			this.searchTimeout = setTimeout(null, 0);

			// set up tag filters

			let sortedTags = Object.keys(this.tags).sort((first,second)=>{
				// sort primarily by number of elements that match the tag with a secondary alphabetical sort 
				let tagCount = this.tags[second].length - this.tags[first].length
				let tagName = first < second ? -1 : first > second ? 1 : 0;
				return tagCount === 0 ? tagName : tagCount;
			});
			for( let tag of sortedTags ){
				this.createTagButton(tag);
			}

			// Append all workspace items to DOM
			this.filterElement.append(this.sortElement, this.searchElement, this.tagElement);
		}

		if( Object.keys(this.tags).length === 0 ){
			this.tagElement.remove();
		}
	}

	createSortButton( title, column, direction ){
		let button = document.createElement('a');
		button.href = 'javascript:void(0);';
		button.textContent = title;
		button.className = 'vl-sort-btn';
		button.dataset.column = column;
		button.dataset.direction = direction;

		button.addEventListener('click', ()=>{
			this.sort(column, direction);
		});
		this.sortList.append(button);
		this.sortButtons[column] = button;
	}

	createTagButton( tag ){
		let row = document.createElement('tr');
		let cell1 = document.createElement('td');
		let cell2 = document.createElement('td');

		let button = document.createElement('button');
		button.type = 'button';
		button.textContent = tag;
		button.className = 'vl-tag-btn';
		cell1.append(button)

		let count = this.tags[tag].length;
		let countElement = document.createElement('span');
		countElement.className = 'vl-tag-count';
		countElement.textContent = count;
		cell2.append(countElement);
		
		row.append(cell1, cell2);
		this.tagList.append(row);

		button.addEventListener('click', ()=>{
			let selected = this.selectedTags.indexOf(tag);
			if( selected > -1 ){
				button.classList.remove('is-active');
				this.selectedTags.splice(selected, 1);
			}
			else {
				button.classList.add('is-active');
				this.selectedTags.push(tag);
			}
			this.applySelectedTags();
		});
		this.tagButtons.push({
			'element': row,
			'tag': tag,
			'countElement': countElement
		});
	}

	sort( column = 'order', direction = 'descending' ){
		// flip direction if already sorting this way
		if( this?.sorted?.column === column && this?.sorted?.direction === direction ){
			direction = direction === 'descending' ? 'ascending' : 'descending';
		}

		this.sorted = { column, direction };

		for( let button of Object.values(this.sortButtons) ){
			if( button.dataset.column === column ){
				button.classList.add('is-active');
				button.dataset.direction = direction;
			}
			else {
				button.classList.remove('is-active');
			}
		}

		// if list was already sorted once, just re-use the previous sort
		let array = [];
		if( `${column}-${direction}` in this.calculatedSorts ){
			array = this.calculatedSorts[`${column}-${direction}`];
		}

		// if not sorted yet, choose correct function and sort
		else {
			// 'order' column gets sorted in reverse due to being front-facingly labelled as date
			let sortFunction = () => { throw new Error('unknown sort'); };
			if( column === 'plays' && direction === 'ascending'
			|| column === 'order' && direction === 'descending' ){
				sortFunction = (first, second) => { return first['value'] - second['value']; };
			}
			else if( column === 'plays' && direction === 'descending'
			|| column === 'order' && direction === 'ascending' ){
				sortFunction = (first, second) => { return second['value'] - first['value']; };
			}
			else if( column === 'title' && direction === 'ascending' ){
				sortFunction = (first, second) => {
					let a = first['value'];
					let b = second['value'];
					return (a < b) ? -1 : (a > b) ? 1 : 0;
				};
			}
			else if( column === 'title' && direction === 'descending' ){
				sortFunction = (first, second) => {
					let a = first['value'].toLowerCase();
					let b = second['value'].toLowerCase();
					return (b < a) ? -1 : (b > a) ? 1 : 0;
				};
			}

			for( let audio of this.audios ){
				array.push({'element': audio.element, 'value': audio[column]});
			}
			array.sort(sortFunction);

			this.calculatedSorts[`${column}-${direction}`] = array;
		}

		// apply sort to items using CSS 'order' values
		for( let i = 0; i < array.length; i++ ){
			array[i]['element'].style.order = i;
		}
	}

	search( ){
		clearTimeout(this.searchTimeout);

		// parse query
		const exclusionRegex = /(?:^|\s)+-(\w+)/g;
		const phraseRegex = /"([^"]+)"/g;
		let query = this.searchBar.value.toLowerCase();

		let failMatches = query.matchAll(exclusionRegex);
		let exclusions = Array.from(failMatches).map( match => match[1] );
		query = query.replaceAll(exclusionRegex, '').trim();

		let phraseMatches = query.matchAll(phraseRegex);
		let phrases = Array.from(phraseMatches).map( match => match[1] );
		query = query.replaceAll(phraseRegex, '').trim();

		let words = query.split(' ');

		this.searchTimeout = setTimeout(()=>{
			for( let audio of this.audios ){
				if( this.passesSearch(audio, phrases, words, exclusions) ){
					audio.element.classList.remove('vl-hidden-by-search');
					this.updateAvailableElements(audio.element);
				}
				else {
					audio.element.classList.add('vl-hidden-by-search');
					this.updateAvailableElements(audio.element);
				}
			}
			this.updateTagCounts();
		}, 350);
	}

	passesSearch( audioListing, phrases, words, exclusions ){
		console.log(phrases, words, exclusions);
		const title = audioListing.title.toLowerCase();
		const tags = audioListing.tags.join(' ').toLowerCase();
		const any = title + tags;

		// cannot match any exclusions
		for( let str of exclusions ){
			if( any.includes(str) ){
				return false;
			}
		}

		// must match all phrases
		for( let phrase of phrases ){
			if(! any.includes(phrase) ){
				return false;
			}
		}

		// can match any word
		for( let word of words ){
			if( any.includes(word) ){
				return true;
			}
		}
	}

	applySelectedTags( ){
		if( this.selectedTags.length === 0 ){
			for( let audio of this.audios ){
				audio.element.classList.remove('vl-hidden-by-tag');
				this.updateAvailableElements( audio.element );
			}
			this.updateTagCounts();
			return;
		}

		for( let audio of this.audios ){
			let passesAllTags = true;
			for( let tag of this.selectedTags ){
				if(! this.tags[tag].includes(audio.element) ){
					passesAllTags = false;
					break;
				}
			}
			if( passesAllTags ){
				audio.element.classList.remove('vl-hidden-by-tag');
				this.updateAvailableElements( audio.element );
			}
			else {
				audio.element.classList.add('vl-hidden-by-tag');
				this.updateAvailableElements( audio.element );
			}
		}
		this.updateTagCounts();
	}

	updateAvailableElements( element ){
		let index = this.availableElements.indexOf(element);
		let hidden = element.classList.contains('vl-hidden-by-tag') | element.classList.contains('vl-hidden-by-search')
		if( hidden && index > -1 ){
			this.availableElements.splice(index, 1);
		}
		else if( !hidden && index === -1 ){
			this.availableElements.push(element);
		}
	}

	updateTagCounts( ){
		for( let data of this.tagButtons ){
			let tag = data.tag;
			let availableCount = 0;
			for( let element of this.availableElements ){
				if( this.tags[tag].includes(element) ){
					availableCount++;
				}
			}

			if( availableCount > 0 ){
				data.countElement.textContent = availableCount;
				data.element.classList.remove('vl-hidden');
			}
			else {
				data.element.classList.add('vl-hidden');
			}
		}
	}
}

// Begin modifying page

function domLoaded() {
	// If content is blank
	let content = document.querySelector('body > div');
	if( content === null ) {
		let 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
	let footer = document.createElement('footer');
	footer.classList.add('vl-footer');

	// Theme switcher
	let themeSwitcher = document.createElement('a');
	themeSwitcher.textContent = 'Theme';
	themeSwitcher.href = 'javascript:void(0);';
	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;

	// Per-page sections

	// Any page with a user as long as it has content
	if( content && path.startsWith('/u/') && path.split('/').length > 2 ){
		let username = path.split('/')[2];
		users.add(username);
	}

	// Homepage
	if( path === '/' ){
		document.querySelector('h1').textContent = 'Welcome to Soundgasm.net, improved!';

		let container = document.createElement('div');
		container.className = 'vl-container';
		container.style.marginBottom = '0';
		let header = document.createElement('h3');
		header.className = 'vl-container-header';
		header.textContent = 'Known user list.';
		container.append(header);

		if( users.names.length === 0 ){
			container.append(paragraph(`The script will remember usernames from the audio and user pages you visit. Once you've opened a few, you can always come back here to find them all!`));
		}
		else {
			let userList = document.createElement('div');
			userList.classList.add('vl-user-list');
			userList.style.fontSize = '14rem';

			for( let username of users.names.sort() ){
				let link = document.createElement('a');
				link.href = `/u/${username}`;
				link.textContent = `• ${username}`;
				link.className = 'vl-link';
				userList.append(link);
			}

			container.append(userList);
		}
		footer.insertAdjacentElement('beforebegin', container);
	}

	// User pages
	if( content && path.startsWith('/u/') && path.split('/').length < 4 ){
		// Prep DOM for filters
		let container = document.createElement('div');
		container.className = 'vl-split';

		let directory = document.createElement('main');
		directory.className = 'vl-directory';
		let sidebar = document.createElement('aside');
		sidebar.className = 'vl-sidebar';

		let frag = new DocumentFragment();
		let items = document.querySelectorAll('.sound-details');
		for( let item of items ){
			frag.append(item);
		}
		directory.append(frag);
		container.append(sidebar, directory);
		document.getElementsByTagName('footer')[0].insertAdjacentElement('beforebegin', container);
		
		// catch any oddities such as patron links and other things from descriptions
		let patreon = document.querySelector('.soundDescription .patreon-widget');
		if( patreon ){
			let container = document.createElement('div');
			container.className = 'vl-container';
			container.style.marginTop = '0';
			container.style.width = 'calc(100% - 30rem)';
			let header = document.createElement('h3');
			header.className = 'vl-container-header';
			header.textContent = 'Extra user info.';
			container.append(header, patreon);

			directory.style.order = '-1';
			directory.prepend(container);
		}

		// Process audio listings
		new AudioDirectory({
			elements: items,
			titleSelector: 'a',
			descriptionSelector: '.soundDescription',
			playCountSelector: '.playCount',
			filterElement: sidebar
		});
	}

	// Player page
	if( path.startsWith('/u/') && path.split('/').length > 3 ){
		// Add custom descriptions
		new AudioListing({
			element: document.querySelector('.jp-type-single'),
			titleSelector: '.jp-title',
			descriptionSelector: '.jp-description'
		});

		let stop = document.querySelector('.jp-stop');
		let title = document.querySelector('.jp-title');
		let author = document.querySelector('div[style="margin:10px 0"] a');
		let 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') {
					let newVol = audio.volume + 0.1;
					if(newVol > 1) {
						newVol = 1.0;
					}
					audio.volume = newVol;
				}
				else if(k === 'arrowdown') {
					let 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() {
			let audio = document.querySelector('audio');
			let src = audio.getAttribute('src');
			let ext = src.split('.').pop();
			let 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';
		}
		function audioLoaded() {
			audio = document.querySelector('audio');
			if(audio !== null && audio.getAttribute('src') !== null) {
				// observer.disconnect();
				addDownload();
				setKeybinds();
			} else {
				setTimeout(audioLoaded, 100);
			}
		}

		// Wait for audio to load
		if(audio !== null && audio.getAttribute('src') !== null) {
			addDownload();
		} else {
			audioLoaded();
		}
	}

	// signup page
	if(window.location.pathname.startsWith('/signup')) {
		let h1 = document.querySelector('h1');
		let form = document.querySelector('.signupform');
		form.prepend(h1);
	}
}