Soundgasm Improvements

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

// ==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);
	}
}