Compare commits

..

6 Commits

Author SHA1 Message Date
8573b82515 more ts switches, use setcontext instead of stores 2026-03-15 17:20:16 +01:00
b47a433414 update packages 2026-03-15 17:18:43 +01:00
4d8f7c2b02 add types 2026-03-15 14:50:56 +01:00
347438928a use SvelteSet instead of Set for favoritedStreams 2025-07-06 22:31:55 +02:00
dc84db8c79 don't bind volume to audio element 2025-06-30 13:11:29 +02:00
7978ea5b37 update build command 2025-05-25 01:43:08 +02:00
22 changed files with 647 additions and 4998 deletions

716
bun.lock

File diff suppressed because it is too large Load Diff

4327
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@@ -4,7 +4,7 @@
"private": true, "private": true,
"scripts": { "scripts": {
"dev": "vite dev", "dev": "vite dev",
"build": "vite build && node ./scripts/create_media_symlink.js", "build": "vite build && bun --bun run ./scripts/create_media_symlink.js",
"preview": "vite preview", "preview": "vite preview",
"check": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json", "check": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json",
"check:watch": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json --watch", "check:watch": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json --watch",
@@ -15,33 +15,34 @@
"generate_iconfont_subset": "cd scripts && bash ./generate_iconfont_subset.sh" "generate_iconfont_subset": "cd scripts && bash ./generate_iconfont_subset.sh"
}, },
"dependencies": { "dependencies": {
"@sveltejs/kit": "^2.17.1", "@sveltejs/kit": "^2.55.0",
"@types/bun": "^1.2.2", "@types/bun": "^1.3.10",
"@types/node": "^20.17.17", "@types/node": "^25.5.0",
"@typescript-eslint/eslint-plugin": "^5.62.0", "@typescript-eslint/eslint-plugin": "^8.57.0",
"@typescript-eslint/parser": "^5.62.0", "@typescript-eslint/parser": "^8.57.0",
"carta-md": "^4.6.7", "carta-md": "^4.11.1",
"dotenv": "^16.4.7", "dotenv": "^16.6.1",
"eslint": "^8.57.1", "eslint": "^10.0.3",
"eslint-config-prettier": "^8.10.0", "eslint-config-prettier": "^10.1.8",
"eslint-plugin-svelte": "^2.46.1", "eslint-plugin-svelte": "^3.15.2",
"fontkit": "^2.0.4", "fontkit": "^2.0.4",
"isomorphic-dompurify": "^2.21.0", "isomorphic-dompurify": "^3.3.0",
"prettier": "^3.4.2", "prettier": "^3.8.1",
"prettier-plugin-svelte": "^3.3.3", "prettier-plugin-svelte": "^3.5.1",
"sqids": "0.3.0", "sqids": "0.3.0",
"svelte": "^5.19.9", "svelte": "^5.53.12",
"svelte-check": "^4.1.4", "svelte-check": "^4.4.5",
"svelte-select": "^5.8.3", "svelte-select": "^5.8.3",
"tslib": "^2.8.1", "tslib": "^2.8.1",
"typescript": "^5.7.3", "typescript": "^5.9.3",
"vite": "^5.4.14" "vite": "^8.0.0"
}, },
"type": "module", "type": "module",
"devDependencies": { "devDependencies": {
"@sveltejs/vite-plugin-svelte": "^4.0.4", "@sveltejs/adapter-node": "^5.5.4",
"svelte-adapter-bun": "https://github.com/gornostay25/svelte-adapter-bun", "@sveltejs/vite-plugin-svelte": "^7.0.0",
"typescript-svelte-plugin": "^0.3.45" "svelte-adapter-bun": "^1.0.1",
"typescript-svelte-plugin": "^0.3.50"
}, },
"trustedDependencies": [ "trustedDependencies": [
"svelte-adapter-bun" "svelte-adapter-bun"

View File

@@ -1,36 +0,0 @@
import { Database } from 'bun:sqlite';
const dbName = './db/strimserve.db';
// Create a new database object and initialize it with the schema
const db = new Database(dbName);
export function getStreams() {
const indexData = db
.prepare(
'SELECT id, stream_date, title, tags, length_seconds ' + 'FROM Stream ORDER BY id DESC'
)
.all();
indexData.forEach((stream) => {
stream['stream_date'] = Date.parse(stream['stream_date']);
stream['tags'] = JSON.parse(stream['tags']);
});
return indexData;
}
export function getStreamInfo(streamId) {
const streamData = db
.prepare(
'SELECT id, stream_date, filename, ' +
'format, title, description, tags, ' +
'length_seconds, tracks FROM Stream ' +
'WHERE id = ?'
)
.get(streamId);
if (streamData) {
streamData['stream_date'] = Date.parse(streamData['stream_date']);
streamData['tracks'] = JSON.parse(streamData['tracks']);
streamData['tags'] = JSON.parse(streamData['tags']);
}
return streamData;
}

45
src/lib/database.ts Normal file
View File

@@ -0,0 +1,45 @@
import { Database } from 'bun:sqlite';
import type { Stream, StreamSummary } from './types.ts';
const dbName = './db/strimserve.db';
// Create a new database object and initialize it with the schema
const db = new Database(dbName);
export function getStreams(): StreamSummary[] {
const indexData = db
.prepare(
'SELECT id, stream_date, title, tags, length_seconds ' + 'FROM Stream ORDER BY id DESC'
)
.all() as Record<string, unknown>[];
return indexData.map((stream) => ({
id: stream.id as string,
stream_date: Date.parse(stream.stream_date as string),
title: stream.title as string | null,
tags: JSON.parse(stream.tags as string) as string[],
length_seconds: stream.length_seconds as number
}));
}
export function getStreamInfo(streamId: string): Stream | null {
const streamData = db
.prepare(
'SELECT id, stream_date, filename, ' +
'format, title, description, tags, ' +
'length_seconds, tracks FROM Stream ' +
'WHERE id = ?'
)
.get(streamId) as Record<string, unknown> | null;
if (!streamData) return null;
return {
id: streamData.id as string,
stream_date: Date.parse(streamData.stream_date as string),
filename: streamData.filename as string,
format: streamData.format as string,
title: streamData.title as string | null,
description: streamData.description as string | null,
tags: JSON.parse(streamData.tags as string) as string[],
length_seconds: streamData.length_seconds as number,
tracks: JSON.parse(streamData.tracks as string)
};
}

View File

@@ -1,85 +0,0 @@
import { writable, get } from 'svelte/store';
import { browser } from '$app/environment';
export const currentStream = writable({});
export const currentSongIndex = writable(null);
export const favoritedStreams = writable(
new Set(JSON.parse((browser && localStorage.getItem('favoritedStreams')) || '[]'))
);
if (browser) {
favoritedStreams.subscribe((val) => {
localStorage.setItem('favoritedStreams', JSON.stringify(Array.from(val)));
});
}
export const tagList = [
'acoustic',
'electronic',
'orchestral',
'rock',
'pop',
'metal',
'aggressive',
'folk',
'jazzy',
'dance.music',
'untz',
'breakbeats',
'electronica',
'chiptune',
'left.field',
'denpa',
'vocaloid',
'funky',
'lush',
'noisy',
'psychedelic',
'dark',
'calm',
'moody',
'uplifting'
];
let timestampList;
// utility
function locationOf(element, array, start, end) {
start = start || 0;
end = end || array.length;
var pivot = parseInt(start + (end - start) / 2, 10);
if (end - start <= 1 || array[pivot] === element) return pivot;
if (array[pivot] < element) {
return locationOf(element, array, pivot, end);
} else {
return locationOf(element, array, start, pivot);
}
}
// exported methods
export function getSongAtTime(currentTime) {
return locationOf(currentTime, timestampList) - 1;
}
// less operations needed when doing regular lookups
export function updateCurrentSong(currentTime, songIndex) {
function updateCurrentSongRecurse(songIndex) {
if (currentTime >= timestampList[songIndex + 2]) {
return updateCurrentSongRecurse(songIndex + 1);
} else if (currentTime < timestampList[songIndex + 1]) {
return updateCurrentSongRecurse(songIndex - 1);
}
return songIndex;
}
currentSongIndex.set(updateCurrentSongRecurse(songIndex));
}
export function updateCurrentStream(stream) {
currentStream.set(stream);
timestampList = [-Infinity, ...stream.tracks.map((track) => track[0]), Infinity];
currentSongIndex.set(0);
}

39
src/lib/stores.svelte.ts Normal file
View File

@@ -0,0 +1,39 @@
import { browser } from '$app/environment';
import { SvelteSet } from 'svelte/reactivity';
export const favoritedStreams = new SvelteSet<string>(
JSON.parse((browser && localStorage.getItem('favoritedStreams')) || '[]')
);
$effect.root(() => {
$effect(() => {
localStorage.setItem('favoritedStreams', JSON.stringify(Array.from(favoritedStreams)));
});
});
export const tagList = [
'acoustic',
'electronic',
'orchestral',
'rock',
'pop',
'metal',
'aggressive',
'folk',
'jazzy',
'dance.music',
'untz',
'breakbeats',
'electronica',
'chiptune',
'left.field',
'denpa',
'vocaloid',
'funky',
'lush',
'noisy',
'psychedelic',
'dark',
'calm',
'moody',
'uplifting'
];

View File

@@ -0,0 +1,47 @@
import { createContext } from 'svelte';
import type { Stream, StreamSummary } from './types.ts';
export class StreamContext {
streams: StreamSummary[];
current = $state<Stream | null>(null);
songIndex = $state<number | null>(null);
#timestamps: number[] = [];
constructor(streams: StreamSummary[]) {
this.streams = streams;
}
setCurrent(stream: Stream) {
this.current = stream;
this.#timestamps = [-Infinity, ...stream.tracks.map((t) => t[0]), Infinity];
this.songIndex = 0;
}
clearCurrent() {
this.current = null;
this.songIndex = null;
}
getSongAtTime(time: number): number {
return this.#locationOf(time, this.#timestamps) - 1;
}
updateCurrentSong(currentTime: number, songIndex: number) {
const ts = this.#timestamps;
const recurse = (idx: number): number => {
if (currentTime >= ts[idx + 2]) return recurse(idx + 1);
if (currentTime < ts[idx + 1]) return recurse(idx - 1);
return idx;
};
this.songIndex = recurse(songIndex);
}
#locationOf(element: number, array: number[], start = 0, end = array.length): number {
const pivot = Math.floor(start + (end - start) / 2);
if (end - start <= 1 || array[pivot] === element) return pivot;
if (array[pivot] < element) return this.#locationOf(element, array, pivot, end);
return this.#locationOf(element, array, start, pivot);
}
}
export const [getStreamContext, setStreamContext] = createContext<StreamContext>();

19
src/lib/types.ts Normal file
View File

@@ -0,0 +1,19 @@
/** A track entry: [timestamp_seconds, artist, title] */
export type Track = [number, string, string];
/** The summary shape used in sidebar listings. */
export interface StreamSummary {
id: string;
stream_date: number;
title: string | null;
tags: string[];
length_seconds: number;
}
/** The full stream shape used on the detail/player page. */
export interface Stream extends StreamSummary {
filename: string;
format: string;
description: string | null;
tracks: Track[];
}

View File

@@ -1,37 +0,0 @@
import Sqids from 'sqids';
let sqids = new Sqids({ minLength: 6, alphabet: 'abcdefghijklmnopqrstuvwxyz0123456789' });
export function hashcode(str) {
for (var i = 0, h = 9; i < str.length; ) h = Math.imul(h ^ str.charCodeAt(i++), 9 ** 9);
return h ^ (h >>> 9);
}
// for mnemonic display
export function shorthandCode(str) {
return sqids.encode([hashcode(str) & 0xffff]);
}
// for tag display
export function hashColor(str) {
const hash = hashcode(str);
return `hsl(${hash % 360}, ${65 + (hash % 30) + 1}%, ${85 + (hash % 10) + 1}%)`;
}
export function formatSecondsToHms(s) {
s = Number(s);
var h = Math.floor(s / 3600);
var m = Math.ceil((s % 3600) / 60);
var hDisplay = h > 0 ? h + (h == 1 ? ' hr' : ' hrs') + (m > 0 ? ', ' : '') : '';
var mDisplay = m > 0 ? m + (m == 1 ? ' min' : ' mins') : '';
return hDisplay + mDisplay;
}
export function formatDate(unix_timestamp) {
return new Date(unix_timestamp).toISOString().split('T')[0];
}
export function formatTrackTime(s) {
return new Date(s * 1000).toISOString().slice(11, 19);
}

38
src/lib/utils.ts Normal file
View File

@@ -0,0 +1,38 @@
import Sqids from 'sqids';
const sqids = new Sqids({ minLength: 6, alphabet: 'abcdefghijklmnopqrstuvwxyz0123456789' });
export function hashcode(str: string): number {
let h = 9;
for (let i = 0; i < str.length; ) h = Math.imul(h ^ str.charCodeAt(i++), 9 ** 9);
return h ^ (h >>> 9);
}
// for mnemonic display
export function shorthandCode(str: string): string {
return sqids.encode([hashcode(str) & 0xffff]);
}
// for tag display
export function hashColor(str: string): string {
const hash = hashcode(str);
return `hsl(${hash % 360}, ${65 + (hash % 30) + 1}%, ${85 + (hash % 10) + 1}%)`;
}
export function formatSecondsToHms(s: number): string {
s = Number(s);
const h = Math.floor(s / 3600);
const m = Math.ceil((s % 3600) / 60);
const hDisplay = h > 0 ? h + (h == 1 ? ' hr' : ' hrs') + (m > 0 ? ', ' : '') : '';
const mDisplay = m > 0 ? m + (m == 1 ? ' min' : ' mins') : '';
return hDisplay + mDisplay;
}
export function formatDate(unix_timestamp: number): string {
return new Date(unix_timestamp).toISOString().split('T')[0];
}
export function formatTrackTime(s: number): string {
return new Date(s * 1000).toISOString().slice(11, 19);
}

View File

@@ -1,4 +1,4 @@
import { getStreams } from '$lib/database.js'; import { getStreams } from '$lib/database.ts';
export function load() { export function load() {
return { return {

View File

@@ -1,11 +1,16 @@
<script lang="ts"> <script lang="ts">
import Sidebar from './Sidebar.svelte'; import Sidebar from './Sidebar.svelte';
import Footer from './Footer.svelte'; import Footer from './Footer.svelte';
let { data, children } = $props(); import { setStreamContext, StreamContext } from '$lib/stream-context.svelte.ts';
// streams are grabbed from the server here, then accessed throughout the rest
// of the components through the svelte context api
const { data, children } = $props();
setStreamContext(new StreamContext(data.streams));
</script> </script>
<div id="mainContainer"> <div id="mainContainer">
<div id="sidebar" class="panel"><Sidebar streams={data.streams} /></div> <div id="sidebar" class="panel"><Sidebar /></div>
<div id="footer" class="panel-alt"><Footer /></div> <div id="footer" class="panel-alt"><Footer /></div>
<div id="content">{@render children?.()}</div> <div id="content">{@render children?.()}</div>
</div> </div>

View File

@@ -1,11 +1,13 @@
<script lang="ts"> <script lang="ts">
import { untrack } from 'svelte'; import { untrack } from 'svelte';
import { goto } from '$app/navigation'; import { goto } from '$app/navigation';
import { currentStream, favoritedStreams } from '$lib/stores.js'; import { favoritedStreams } from '$lib/stores.svelte.ts';
import { hashColor, shorthandCode, formatSecondsToHms, formatDate } from '$lib/utils.js'; import { getStreamContext } from '$lib/stream-context.svelte.ts';
import type { StreamSummary } from '$lib/types';
import { hashColor, shorthandCode, formatSecondsToHms, formatDate } from '$lib/utils.ts';
import TagSelect from './TagSelect.svelte'; import TagSelect from './TagSelect.svelte';
let { streams } = $props(); const ctx = getStreamContext();
let filteredTags = $state([]); let filteredTags = $state([]);
let favoritesOnly = $state(false); let favoritesOnly = $state(false);
let listOpen = $state(); let listOpen = $state();
@@ -19,15 +21,15 @@
}); });
function streamsToDisplay() { function streamsToDisplay() {
return streams.filter( return ctx.streams.filter(
(stream) => (stream) =>
!('tags' in stream) || filteredTags.every((tag) => stream['tags'].includes(tag)) !('tags' in stream) || filteredTags.every((tag) => stream.tags.includes(tag))
); );
} }
function getRemainingTags() { function getRemainingTags() {
let tagCounts = displayedStreams.reduce((tags, stream) => { let tagCounts = displayedStreams.reduce((tags: Record<string, number>, stream) => {
stream['tags'].forEach((tag) => (tag in tags ? (tags[tag] += 1) : (tags[tag] = 1))); stream.tags.forEach((tag) => (tags[tag] ? (tags[tag] += 1) : (tags[tag] = 1)));
return tags; return tags;
}, {}); }, {});
return Object.entries(tagCounts) return Object.entries(tagCounts)
@@ -35,15 +37,14 @@
.map((el) => el[0]); .map((el) => el[0]);
} }
function getDisplayedTagsInList(streamTags, remainingTags) { function getDisplayedTagsInList(streamTags: string[], remainingTags: string[]) {
return streamTags.filter((tag) => remainingTags.includes(tag)); return streamTags.filter((tag) => remainingTags.includes(tag));
} }
function updateFavorites(stream) { function updateFavorites(stream: StreamSummary) {
$favoritedStreams.has(stream['id']) favoritedStreams.has(stream.id)
? $favoritedStreams.delete(stream['id']) ? favoritedStreams.delete(stream.id)
: $favoritedStreams.add(stream['id']); : favoritedStreams.add(stream.id);
$favoritedStreams = $favoritedStreams; // for reactivity
} }
</script> </script>
@@ -56,29 +57,27 @@
</div> </div>
<ul class="stream-list"> <ul class="stream-list">
{#each streams as stream} {#each ctx.streams as stream}
{@const favorited = $favoritedStreams.has(stream['id'])} {@const favorited = favoritedStreams.has(stream.id)}
{@const current = $currentStream['id'] === stream['id']} {@const current = ctx.current?.id === stream.id}
<li <li
hidden={!displayedStreams.includes(stream) || (favoritesOnly && !favorited)} hidden={!displayedStreams.includes(stream) || (favoritesOnly && !favorited)}
class="stream-item {current ? 'current-stream' : ''}" class="stream-item {current ? 'current-stream' : ''}"
id="stream-{stream['id']}" id="stream-{stream.id}"
> >
<button class="stream-item-button" onclick={() => goto('/streams/' + stream['id'])}> <button class="stream-item-button" onclick={() => goto('/streams/' + stream.id)}>
<span class="stream-item-id"> <span class="stream-item-id">
ID: ID:
{shorthandCode(stream['id'])}</span {shorthandCode(stream.id)}</span
> >
<span class="stream-item-date">{formatDate(stream['stream_date'])}</span> <span class="stream-item-date">{formatDate(stream.stream_date)}</span>
<span class="stream-item-length" <span class="stream-item-length">{formatSecondsToHms(stream.length_seconds)}</span>
>{formatSecondsToHms(stream['length_seconds'])}</span
>
<p class="stream-item-tags" hidden={!remainingTags.length}> <p class="stream-item-tags" hidden={!remainingTags.length}>
Tags: <span hidden={!filteredTags.length}>[...] </span>{getDisplayedTagsInList( Tags: <span hidden={!filteredTags.length}>[...] </span>{getDisplayedTagsInList(
stream['tags'], stream.tags,
remainingTags remainingTags
).join(', ')} ).join(', ')}
</p> </p>

View File

@@ -1,6 +1,6 @@
<script lang="ts"> <script lang="ts">
import Select from 'svelte-select'; import Select from 'svelte-select';
import { tagList } from '$lib/stores.js'; import { tagList } from '$lib/stores.svelte.ts';
let { checked = $bindable([]), remainingTags = [], listOpen = $bindable() } = $props(); let { checked = $bindable([]), remainingTags = [], listOpen = $bindable() } = $props();
let items = $state(tagList.map((x) => ({ value: x, label: x }))); let items = $state(tagList.map((x) => ({ value: x, label: x })));
@@ -58,7 +58,7 @@
{#snippet item({ item })} {#snippet item({ item })}
<div class="item"> <div class="item">
<label for={item.value}> <label for={item.value}>
<input type="checkbox" id={item.value} bind:checked={isChecked[item.value]} /> <input type="checkbox" id={item.value} checked={isChecked[item.value]} />
{item.label} {item.label}
</label> </label>
</div> </div>

View File

@@ -1,7 +1,8 @@
<script> <script>
import { currentStream } from '$lib/stores.js'; import { getStreamContext } from '$lib/stream-context.svelte.ts';
currentStream.set({}); const ctx = getStreamContext();
ctx.clearCurrent();
</script> </script>
<div class="stream-information"> <div class="stream-information">

View File

@@ -1,7 +1,7 @@
import fs from 'fs'; import fs from 'fs';
import path from 'path'; import path from 'path';
import { error } from '@sveltejs/kit'; import { error } from '@sveltejs/kit';
import { getStreamInfo } from '$lib/database.js'; import { getStreamInfo } from '$lib/database.ts';
import { dev } from '$app/environment'; import { dev } from '$app/environment';
import { STREAM_JSON_LOCATION } from '$env/static/private'; import { STREAM_JSON_LOCATION } from '$env/static/private';

View File

@@ -5,12 +5,13 @@
import MetadataEditor from './MetadataEditor.svelte'; import MetadataEditor from './MetadataEditor.svelte';
import Player from './Player.svelte'; import Player from './Player.svelte';
import { dev } from '$app/environment'; import { dev } from '$app/environment';
import { currentStream, updateCurrentStream } from '$lib/stores.js'; import { getStreamContext } from '$lib/stream-context.svelte.ts';
let { data } = $props(); let { data } = $props();
const ctx = getStreamContext();
run(() => { run(() => {
updateCurrentStream(data.stream); ctx.setCurrent(data.stream);
}); });
</script> </script>
@@ -22,8 +23,8 @@
<StreamPage /> <StreamPage />
</div> </div>
<div id="player"> <div id="player">
{#key $currentStream} {#key ctx.current}
<Player display={true} src="/media/tracks/{$currentStream.filename}" /> <Player display={true} src="/media/tracks/{ctx.current?.filename}" />
{/key} {/key}
</div> </div>
</div> </div>

View File

@@ -1,6 +1,6 @@
<script lang="ts"> <script lang="ts">
import { page } from '$app/stores'; import { page } from '$app/stores';
import { tagList } from '$lib/stores.js'; import { tagList } from '$lib/stores.svelte.ts';
let { original = $bindable() } = $props(); let { original = $bindable() } = $props();

View File

@@ -19,12 +19,10 @@
const bubble = createBubbler(); const bubble = createBubbler();
import { onMount, onDestroy } from 'svelte'; import { onMount, onDestroy } from 'svelte';
import { fade } from 'svelte/transition'; import { fade } from 'svelte/transition';
import { import { getStreamContext } from '$lib/stream-context.svelte.ts';
currentSongIndex, import type { Track } from '$lib/types';
currentStream,
getSongAtTime, const ctx = getStreamContext();
updateCurrentSong
} from '$lib/stores.js';
interface Props { interface Props {
src: any; src: any;
@@ -75,6 +73,7 @@
let seekTrack = $state(''); let seekTrack = $state('');
let seeking = $state(false); let seeking = $state(false);
let volumeSeeking = $state(false); let volumeSeeking = $state(false);
let pendingSeekTime = $state<number | null>(null);
let songBar = $state(); let songBar = $state();
let volumeBar = $state(); let volumeBar = $state();
let innerWidth = $state(); let innerWidth = $state();
@@ -85,14 +84,12 @@
const volumeData = localStorage.getItem('volume'); const volumeData = localStorage.getItem('volume');
volume = volumeData ? parseFloat(volumeData) : 0.67; volume = volumeData ? parseFloat(volumeData) : 0.67;
setVolume(volume); setVolume(volume);
});
// update browser metadata on track changes // update browser metadata on track changes
if ('mediaSession' in navigator) { $effect(() => {
currentSongIndex.subscribe((val) => { if ('mediaSession' in navigator && ctx.songIndex != null && !paused && ctx.current) {
if (val != null && !paused) { setMediaMetadata(ctx.current.tracks[ctx.songIndex]);
setMediaMetadata($currentStream.tracks[val]);
}
});
} }
}); });
@@ -111,7 +108,7 @@
} }
} }
function setMediaMetadata(track) { function setMediaMetadata(track: Track) {
navigator.mediaSession.metadata = new MediaMetadata({ navigator.mediaSession.metadata = new MediaMetadata({
artist: track[1], artist: track[1],
title: track[2] title: track[2]
@@ -120,8 +117,8 @@
// browsers don't like if you update this while no media is playing // browsers don't like if you update this while no media is playing
function setMediaMetadataOnPlay() { function setMediaMetadataOnPlay() {
if ('mediaSession' in navigator) { if ('mediaSession' in navigator && ctx.current && ctx.songIndex != null) {
setMediaMetadata($currentStream.tracks[$currentSongIndex]); setMediaMetadata(ctx.current.tracks[ctx.songIndex]);
} }
} }
@@ -144,6 +141,11 @@
audio.currentTime = seek(event, songBar.getBoundingClientRect()) * duration; audio.currentTime = seek(event, songBar.getBoundingClientRect()) * duration;
} }
function updateSeekVisual(event) {
if (!songBar) return;
pendingSeekTime = seek(event, songBar.getBoundingClientRect()) * duration;
}
function seekVolume(event) { function seekVolume(event) {
if (!volumeBar) return; if (!volumeBar) return;
volume = seek(event, volumeBar.getBoundingClientRect()); volume = seek(event, volumeBar.getBoundingClientRect());
@@ -173,18 +175,21 @@
} }
let bounds = songBar.getBoundingClientRect(); let bounds = songBar.getBoundingClientRect();
let seekValue = ((event.pageX - bounds.left) * duration) / bounds.width; let seekValue = ((event.pageX - bounds.left) * duration) / bounds.width;
let trackArray = $currentStream.tracks[getSongAtTime(seekValue)]; if (!ctx.current) return;
let trackArray = ctx.current.tracks[ctx.getSongAtTime(seekValue)];
seekTrack = trackArray[1] + ' - ' + trackArray[2]; seekTrack = trackArray[1] + ' - ' + trackArray[2];
seekText = formatSeconds(seekValue); seekText = formatSeconds(seekValue);
} }
function trackMouse(event) { function trackMouse(event) {
if (seeking) seekAudio(event); if (seeking) updateSeekVisual(event);
if (showTooltip && !disableTooltip) seekTooltip(event); if (showTooltip && !disableTooltip) seekTooltip(event);
if (volumeSeeking) seekVolume(event); if (volumeSeeking) seekVolume(event);
} }
run(() => { run(() => {
updateCurrentSong(currentTime, $currentSongIndex); if (ctx.songIndex != null) {
ctx.updateCurrentSong(currentTime, ctx.songIndex);
}
}); });
run(() => { run(() => {
updateAudioAttributes(audio); updateAudioAttributes(audio);
@@ -212,7 +217,13 @@
<svelte:window <svelte:window
bind:innerWidth bind:innerWidth
bind:innerHeight bind:innerHeight
onmouseup={() => (seeking = volumeSeeking = false)} onmouseup={() => {
if (seeking && pendingSeekTime != null) {
audio.currentTime = pendingSeekTime;
pendingSeekTime = null;
}
seeking = volumeSeeking = false;
}}
onmousemove={trackMouse} onmousemove={trackMouse}
/> />
@@ -231,7 +242,7 @@
</button> </button>
<progress <progress
bind:this={songBar} bind:this={songBar}
value={currentTime ? currentTime : 0} value={seeking && pendingSeekTime != null ? pendingSeekTime : (currentTime || 0)}
max={duration} max={duration}
onmousedown={() => (seeking = true)} onmousedown={() => (seeking = true)}
onmouseenter={() => (showTooltip = true)} onmouseenter={() => (showTooltip = true)}
@@ -296,7 +307,6 @@
bind:duration={null, (d) => (duration = d || 0)} bind:duration={null, (d) => (duration = d || 0)}
bind:currentTime bind:currentTime
{muted} {muted}
{volume}
onplay={setMediaMetadataOnPlay} onplay={setMediaMetadataOnPlay}
onended={bubble('ended')} onended={bubble('ended')}
{preload} {preload}

View File

@@ -1,6 +1,6 @@
<script> <script>
import { currentStream, currentSongIndex } from '$lib/stores.js'; import { getStreamContext } from '$lib/stream-context.svelte.ts';
import { shorthandCode, formatTrackTime, formatDate } from '$lib/utils.js'; import { shorthandCode, formatTrackTime, formatDate } from '$lib/utils.ts';
import { jumpToTrack } from './Player.svelte'; import { jumpToTrack } from './Player.svelte';
import { Carta } from 'carta-md'; import { Carta } from 'carta-md';
import DOMPurify from 'isomorphic-dompurify'; import DOMPurify from 'isomorphic-dompurify';
@@ -9,31 +9,36 @@
sanitizer: DOMPurify.sanitize sanitizer: DOMPurify.sanitize
}); });
let formattedStreamDate = $derived(formatDate($currentStream.stream_date)); const ctx = getStreamContext();
let formattedStreamDate = $derived(
ctx.current ? formatDate(ctx.current.stream_date) : ''
);
</script> </script>
<svelte:head> <svelte:head>
<title>{formattedStreamDate} | apt-get's auditorium</title> <title>{formattedStreamDate} | apt-get's auditorium</title>
</svelte:head> </svelte:head>
{#if ctx.current}
<div class="stream-information"> <div class="stream-information">
<div class="stream-information-flexbox"> <div class="stream-information-flexbox">
<h3 class="stream-id">ID: {shorthandCode($currentStream.id)}</h3> <h3 class="stream-id">ID: {shorthandCode(ctx.current.id)}</h3>
<h1 class="stream-date">{formattedStreamDate}</h1> <h1 class="stream-date">{formattedStreamDate}</h1>
</div> </div>
<h5 class="stream-tags"><u>Tags</u>: {$currentStream.tags.join(', ')}</h5> <h5 class="stream-tags"><u>Tags</u>: {ctx.current.tags.join(', ')}</h5>
</div> </div>
<div class="description-bubble"> <div class="description-bubble">
{@html carta.renderSSR($currentStream.description || 'No description available.')} {@html carta.renderSSR(ctx.current.description || 'No description available.')}
</div> </div>
<div id="table-container"> <div id="table-container">
<table> <table>
<tbody> <tbody>
<tr><th>Timestamp</th><th>Artist</th><th>Title</th></tr> <tr><th>Timestamp</th><th>Artist</th><th>Title</th></tr>
{#each $currentStream.tracks as track, i} {#each ctx.current.tracks as track, i}
<tr class:current={i == $currentSongIndex}> <tr class:current={i == ctx.songIndex}>
<td class="timestamp-field" <td class="timestamp-field"
><div class="timestamp-field-flex"> ><div class="timestamp-field-flex">
{formatTrackTime(track[0])} {formatTrackTime(track[0])}
@@ -52,6 +57,7 @@
</tbody> </tbody>
</table> </table>
</div> </div>
{/if}
<style> <style>
.stream-information { .stream-information {

View File

@@ -1,4 +1,3 @@
//import adapter from 'svelte-adapter-bun';
import adapter from 'svelte-adapter-bun'; import adapter from 'svelte-adapter-bun';
import { vitePreprocess } from '@sveltejs/vite-plugin-svelte'; import { vitePreprocess } from '@sveltejs/vite-plugin-svelte';