add types
This commit is contained in:
5
bun.lock
5
bun.lock
@@ -1,10 +1,12 @@
|
|||||||
{
|
{
|
||||||
"lockfileVersion": 1,
|
"lockfileVersion": 1,
|
||||||
|
"configVersion": 0,
|
||||||
"workspaces": {
|
"workspaces": {
|
||||||
"": {
|
"": {
|
||||||
"name": "strimserve",
|
"name": "strimserve",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@sveltejs/kit": "^2.17.1",
|
"@sveltejs/kit": "^2.17.1",
|
||||||
|
"@types/bun": "^1.2.2",
|
||||||
"@types/node": "^20.17.17",
|
"@types/node": "^20.17.17",
|
||||||
"@typescript-eslint/eslint-plugin": "^5.62.0",
|
"@typescript-eslint/eslint-plugin": "^5.62.0",
|
||||||
"@typescript-eslint/parser": "^5.62.0",
|
"@typescript-eslint/parser": "^5.62.0",
|
||||||
@@ -27,7 +29,6 @@
|
|||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@sveltejs/vite-plugin-svelte": "^4.0.4",
|
"@sveltejs/vite-plugin-svelte": "^4.0.4",
|
||||||
"@types/bun": "^1.2.2",
|
|
||||||
"svelte-adapter-bun": "https://github.com/gornostay25/svelte-adapter-bun",
|
"svelte-adapter-bun": "https://github.com/gornostay25/svelte-adapter-bun",
|
||||||
"typescript-svelte-plugin": "^0.3.45",
|
"typescript-svelte-plugin": "^0.3.45",
|
||||||
},
|
},
|
||||||
@@ -733,7 +734,7 @@
|
|||||||
|
|
||||||
"svelte": ["svelte@5.19.9", "", { "dependencies": { "@ampproject/remapping": "^2.3.0", "@jridgewell/sourcemap-codec": "^1.5.0", "@types/estree": "^1.0.5", "acorn": "^8.12.1", "acorn-typescript": "^1.4.13", "aria-query": "^5.3.1", "axobject-query": "^4.1.0", "clsx": "^2.1.1", "esm-env": "^1.2.1", "esrap": "^1.4.3", "is-reference": "^3.0.3", "locate-character": "^3.0.0", "magic-string": "^0.30.11", "zimmerframe": "^1.1.2" } }, "sha512-860s752/ZZxHIsii31ELkdKBOCeAuDsfb/AGUXJyQyzUVLRSt4oqEw/BV5+2+mNg8mbqmD3OK+vMvwWMPM6f8A=="],
|
"svelte": ["svelte@5.19.9", "", { "dependencies": { "@ampproject/remapping": "^2.3.0", "@jridgewell/sourcemap-codec": "^1.5.0", "@types/estree": "^1.0.5", "acorn": "^8.12.1", "acorn-typescript": "^1.4.13", "aria-query": "^5.3.1", "axobject-query": "^4.1.0", "clsx": "^2.1.1", "esm-env": "^1.2.1", "esrap": "^1.4.3", "is-reference": "^3.0.3", "locate-character": "^3.0.0", "magic-string": "^0.30.11", "zimmerframe": "^1.1.2" } }, "sha512-860s752/ZZxHIsii31ELkdKBOCeAuDsfb/AGUXJyQyzUVLRSt4oqEw/BV5+2+mNg8mbqmD3OK+vMvwWMPM6f8A=="],
|
||||||
|
|
||||||
"svelte-adapter-bun": ["svelte-adapter-bun@github:gornostay25/svelte-adapter-bun#ade8792", { "dependencies": { "tiny-glob": "^0.2.9" } }, "gornostay25-svelte-adapter-bun-ade8792"],
|
"svelte-adapter-bun": ["svelte-adapter-bun@github:gornostay25/svelte-adapter-bun#ade8792", { "dependencies": { "tiny-glob": "^0.2.9" } }, "gornostay25-svelte-adapter-bun-ade8792", "sha512-eZOxFn47qOyyqeFa8Qtd5eUhkhsT2yy+BWV6vxnsTs0Dv/Q1dO19ekQ7MrgR2lc6m9GWF15qHxtgyVzFyb040w=="],
|
||||||
|
|
||||||
"svelte-check": ["svelte-check@4.1.4", "", { "dependencies": { "@jridgewell/trace-mapping": "^0.3.25", "chokidar": "^4.0.1", "fdir": "^6.2.0", "picocolors": "^1.0.0", "sade": "^1.7.4" }, "peerDependencies": { "svelte": "^4.0.0 || ^5.0.0-next.0", "typescript": ">=5.0.0" }, "bin": { "svelte-check": "bin/svelte-check" } }, "sha512-v0j7yLbT29MezzaQJPEDwksybTE2Ups9rUxEXy92T06TiA0cbqcO8wAOwNUVkFW6B0hsYHA+oAX3BS8b/2oHtw=="],
|
"svelte-check": ["svelte-check@4.1.4", "", { "dependencies": { "@jridgewell/trace-mapping": "^0.3.25", "chokidar": "^4.0.1", "fdir": "^6.2.0", "picocolors": "^1.0.0", "sade": "^1.7.4" }, "peerDependencies": { "svelte": "^4.0.0 || ^5.0.0-next.0", "typescript": ">=5.0.0" }, "bin": { "svelte-check": "bin/svelte-check" } }, "sha512-v0j7yLbT29MezzaQJPEDwksybTE2Ups9rUxEXy92T06TiA0cbqcO8wAOwNUVkFW6B0hsYHA+oAX3BS8b/2oHtw=="],
|
||||||
|
|
||||||
|
|||||||
@@ -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
45
src/lib/database.ts
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
import { Database } from 'bun:sqlite';
|
||||||
|
import type { Stream, StreamSummary } from './types.js';
|
||||||
|
|
||||||
|
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)
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -1,11 +1,12 @@
|
|||||||
import { writable } from 'svelte/store';
|
import { writable } from 'svelte/store';
|
||||||
import { browser } from '$app/environment';
|
import { browser } from '$app/environment';
|
||||||
import { SvelteSet } from 'svelte/reactivity';
|
import { SvelteSet } from 'svelte/reactivity';
|
||||||
|
import type { Stream } from '$lib/types';
|
||||||
|
|
||||||
export const currentStream = writable({});
|
export const currentStream = writable<Stream | null>(null);
|
||||||
export const currentSongIndex = writable(null);
|
export const currentSongIndex = writable<number | null>(null);
|
||||||
|
|
||||||
export const favoritedStreams = new SvelteSet(
|
export const favoritedStreams = new SvelteSet<string>(
|
||||||
JSON.parse((browser && localStorage.getItem('favoritedStreams')) || '[]')
|
JSON.parse((browser && localStorage.getItem('favoritedStreams')) || '[]')
|
||||||
);
|
);
|
||||||
$effect.root(() => {
|
$effect.root(() => {
|
||||||
@@ -42,14 +43,14 @@ export const tagList = [
|
|||||||
'uplifting'
|
'uplifting'
|
||||||
];
|
];
|
||||||
|
|
||||||
let timestampList;
|
let timestampList: number[];
|
||||||
|
|
||||||
// utility
|
// utility
|
||||||
|
|
||||||
function locationOf(element, array, start, end) {
|
function locationOf(element: number, array: number[], start?: number, end?: number): number {
|
||||||
start = start || 0;
|
start = start || 0;
|
||||||
end = end || array.length;
|
end = end || array.length;
|
||||||
var pivot = parseInt(start + (end - start) / 2, 10);
|
const pivot = parseInt(String(start + (end - start) / 2), 10);
|
||||||
if (end - start <= 1 || array[pivot] === element) return pivot;
|
if (end - start <= 1 || array[pivot] === element) return pivot;
|
||||||
if (array[pivot] < element) {
|
if (array[pivot] < element) {
|
||||||
return locationOf(element, array, pivot, end);
|
return locationOf(element, array, pivot, end);
|
||||||
@@ -60,13 +61,13 @@ function locationOf(element, array, start, end) {
|
|||||||
|
|
||||||
// exported methods
|
// exported methods
|
||||||
|
|
||||||
export function getSongAtTime(currentTime) {
|
export function getSongAtTime(currentTime: number): number {
|
||||||
return locationOf(currentTime, timestampList) - 1;
|
return locationOf(currentTime, timestampList) - 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// less operations needed when doing regular lookups
|
// less operations needed when doing regular lookups
|
||||||
export function updateCurrentSong(currentTime, songIndex) {
|
export function updateCurrentSong(currentTime: number, songIndex: number): void {
|
||||||
function updateCurrentSongRecurse(songIndex) {
|
function updateCurrentSongRecurse(songIndex: number): number {
|
||||||
if (currentTime >= timestampList[songIndex + 2]) {
|
if (currentTime >= timestampList[songIndex + 2]) {
|
||||||
return updateCurrentSongRecurse(songIndex + 1);
|
return updateCurrentSongRecurse(songIndex + 1);
|
||||||
} else if (currentTime < timestampList[songIndex + 1]) {
|
} else if (currentTime < timestampList[songIndex + 1]) {
|
||||||
@@ -78,7 +79,7 @@ export function updateCurrentSong(currentTime, songIndex) {
|
|||||||
currentSongIndex.set(updateCurrentSongRecurse(songIndex));
|
currentSongIndex.set(updateCurrentSongRecurse(songIndex));
|
||||||
}
|
}
|
||||||
|
|
||||||
export function updateCurrentStream(stream) {
|
export function updateCurrentStream(stream: Stream): void {
|
||||||
currentStream.set(stream);
|
currentStream.set(stream);
|
||||||
timestampList = [-Infinity, ...stream.tracks.map((track) => track[0]), Infinity];
|
timestampList = [-Infinity, ...stream.tracks.map((track) => track[0]), Infinity];
|
||||||
currentSongIndex.set(0);
|
currentSongIndex.set(0);
|
||||||
19
src/lib/types.ts
Normal file
19
src/lib/types.ts
Normal 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[];
|
||||||
|
}
|
||||||
@@ -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
38
src/lib/utils.ts
Normal 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);
|
||||||
|
}
|
||||||
@@ -2,10 +2,11 @@
|
|||||||
import { untrack } from 'svelte';
|
import { untrack } from 'svelte';
|
||||||
import { goto } from '$app/navigation';
|
import { goto } from '$app/navigation';
|
||||||
import { currentStream, favoritedStreams } from '$lib/stores.svelte.js';
|
import { currentStream, favoritedStreams } from '$lib/stores.svelte.js';
|
||||||
|
import type { StreamSummary } from '$lib/types';
|
||||||
import { hashColor, shorthandCode, formatSecondsToHms, formatDate } from '$lib/utils.js';
|
import { hashColor, shorthandCode, formatSecondsToHms, formatDate } from '$lib/utils.js';
|
||||||
import TagSelect from './TagSelect.svelte';
|
import TagSelect from './TagSelect.svelte';
|
||||||
|
|
||||||
let { streams } = $props();
|
let { streams }: { streams: StreamSummary[] } = $props();
|
||||||
let filteredTags = $state([]);
|
let filteredTags = $state([]);
|
||||||
let favoritesOnly = $state(false);
|
let favoritesOnly = $state(false);
|
||||||
let listOpen = $state();
|
let listOpen = $state();
|
||||||
@@ -21,13 +22,13 @@
|
|||||||
function streamsToDisplay() {
|
function streamsToDisplay() {
|
||||||
return streams.filter(
|
return 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,14 +36,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);
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
@@ -56,28 +57,26 @@
|
|||||||
|
|
||||||
<ul class="stream-list">
|
<ul class="stream-list">
|
||||||
{#each streams as stream}
|
{#each streams as stream}
|
||||||
{@const favorited = favoritedStreams.has(stream['id'])}
|
{@const favorited = favoritedStreams.has(stream.id)}
|
||||||
{@const current = $currentStream['id'] === stream['id']}
|
{@const current = $currentStream?.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>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<script>
|
<script>
|
||||||
import { currentStream } from '$lib/stores.svelte.js';
|
import { currentStream } from '$lib/stores.svelte.js';
|
||||||
|
|
||||||
currentStream.set({});
|
currentStream.set(null);
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="stream-information">
|
<div class="stream-information">
|
||||||
|
|||||||
@@ -6,6 +6,7 @@
|
|||||||
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.svelte.js';
|
import { currentStream, updateCurrentStream } from '$lib/stores.svelte.js';
|
||||||
|
import type { Stream } from '$lib/types';
|
||||||
|
|
||||||
let { data } = $props();
|
let { data } = $props();
|
||||||
|
|
||||||
@@ -23,7 +24,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div id="player">
|
<div id="player">
|
||||||
{#key $currentStream}
|
{#key $currentStream}
|
||||||
<Player display={true} src="/media/tracks/{$currentStream.filename}" />
|
<Player display={true} src="/media/tracks/{$currentStream?.filename}" />
|
||||||
{/key}
|
{/key}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -25,6 +25,7 @@
|
|||||||
getSongAtTime,
|
getSongAtTime,
|
||||||
updateCurrentSong
|
updateCurrentSong
|
||||||
} from '$lib/stores.svelte.js';
|
} from '$lib/stores.svelte.js';
|
||||||
|
import type { Track } from '$lib/types';
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
src: any;
|
src: any;
|
||||||
@@ -89,7 +90,7 @@
|
|||||||
// update browser metadata on track changes
|
// update browser metadata on track changes
|
||||||
if ('mediaSession' in navigator) {
|
if ('mediaSession' in navigator) {
|
||||||
currentSongIndex.subscribe((val) => {
|
currentSongIndex.subscribe((val) => {
|
||||||
if (val != null && !paused) {
|
if (val != null && !paused && $currentStream) {
|
||||||
setMediaMetadata($currentStream.tracks[val]);
|
setMediaMetadata($currentStream.tracks[val]);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -111,7 +112,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,7 +121,7 @@
|
|||||||
|
|
||||||
// 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 && $currentStream && $currentSongIndex != null) {
|
||||||
setMediaMetadata($currentStream.tracks[$currentSongIndex]);
|
setMediaMetadata($currentStream.tracks[$currentSongIndex]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -173,6 +174,7 @@
|
|||||||
}
|
}
|
||||||
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;
|
||||||
|
if (!$currentStream) return;
|
||||||
let trackArray = $currentStream.tracks[getSongAtTime(seekValue)];
|
let trackArray = $currentStream.tracks[getSongAtTime(seekValue)];
|
||||||
seekTrack = trackArray[1] + ' - ' + trackArray[2];
|
seekTrack = trackArray[1] + ' - ' + trackArray[2];
|
||||||
seekText = formatSeconds(seekValue);
|
seekText = formatSeconds(seekValue);
|
||||||
@@ -184,7 +186,9 @@
|
|||||||
if (volumeSeeking) seekVolume(event);
|
if (volumeSeeking) seekVolume(event);
|
||||||
}
|
}
|
||||||
run(() => {
|
run(() => {
|
||||||
updateCurrentSong(currentTime, $currentSongIndex);
|
if ($currentSongIndex != null) {
|
||||||
|
updateCurrentSong(currentTime, $currentSongIndex);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
run(() => {
|
run(() => {
|
||||||
updateAudioAttributes(audio);
|
updateAudioAttributes(audio);
|
||||||
|
|||||||
@@ -9,49 +9,53 @@
|
|||||||
sanitizer: DOMPurify.sanitize
|
sanitizer: DOMPurify.sanitize
|
||||||
});
|
});
|
||||||
|
|
||||||
let formattedStreamDate = $derived(formatDate($currentStream.stream_date));
|
let formattedStreamDate = $derived(
|
||||||
|
$currentStream ? formatDate($currentStream.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>
|
||||||
|
|
||||||
<div class="stream-information">
|
{#if $currentStream}
|
||||||
<div class="stream-information-flexbox">
|
<div class="stream-information">
|
||||||
<h3 class="stream-id">ID: {shorthandCode($currentStream.id)}</h3>
|
<div class="stream-information-flexbox">
|
||||||
<h1 class="stream-date">{formattedStreamDate}</h1>
|
<h3 class="stream-id">ID: {shorthandCode($currentStream.id)}</h3>
|
||||||
|
<h1 class="stream-date">{formattedStreamDate}</h1>
|
||||||
|
</div>
|
||||||
|
<h5 class="stream-tags"><u>Tags</u>: {$currentStream.tags.join(', ')}</h5>
|
||||||
</div>
|
</div>
|
||||||
<h5 class="stream-tags"><u>Tags</u>: {$currentStream.tags.join(', ')}</h5>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="description-bubble">
|
<div class="description-bubble">
|
||||||
{@html carta.renderSSR($currentStream.description || 'No description available.')}
|
{@html carta.renderSSR($currentStream.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 $currentStream.tracks as track, i}
|
||||||
<tr class:current={i == $currentSongIndex}>
|
<tr class:current={i == $currentSongIndex}>
|
||||||
<td class="timestamp-field"
|
<td class="timestamp-field"
|
||||||
><div class="timestamp-field-flex">
|
><div class="timestamp-field-flex">
|
||||||
{formatTrackTime(track[0])}
|
{formatTrackTime(track[0])}
|
||||||
<button
|
<button
|
||||||
onclick={() => jumpToTrack(track[0])}
|
onclick={() => jumpToTrack(track[0])}
|
||||||
class="material-icons"
|
class="material-icons"
|
||||||
style="padding: 4px 6px; margin-top: -4px; margin-bottom: -4px; margin-right: -6px"
|
style="padding: 4px 6px; margin-top: -4px; margin-bottom: -4px; margin-right: -6px"
|
||||||
>fast_forward</button
|
>fast_forward</button
|
||||||
>
|
>
|
||||||
</div>
|
</div>
|
||||||
</td>
|
</td>
|
||||||
<td class="artist-field">{track[1]}</td>
|
<td class="artist-field">{track[1]}</td>
|
||||||
<td class="track-field">{track[2]}</td>
|
<td class="track-field">{track[2]}</td>
|
||||||
</tr>
|
</tr>
|
||||||
{/each}
|
{/each}
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
|
{/if}
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
.stream-information {
|
.stream-information {
|
||||||
|
|||||||
Reference in New Issue
Block a user