feat: websockets & realtime updates

This commit is contained in:
Mat12143 2025-08-02 05:04:30 +02:00
parent 2a167ba8ad
commit 0bf67061f8
4 changed files with 41 additions and 46 deletions

View file

@ -1210,18 +1210,6 @@
"dev": true, "dev": true,
"license": "MIT" "license": "MIT"
}, },
"node_modules/@types/node": {
"version": "24.1.0",
"resolved": "https://registry.npmjs.org/@types/node/-/node-24.1.0.tgz",
"integrity": "sha512-ut5FthK5moxFKH2T1CUOC6ctR67rQRvvHdFLCD2Ql6KXmMuCrjsSsRI9UsLCm9M18BMwClv4pn327UvB7eeO1w==",
"dev": true,
"license": "MIT",
"optional": true,
"peer": true,
"dependencies": {
"undici-types": "~7.8.0"
}
},
"node_modules/acorn": { "node_modules/acorn": {
"version": "8.15.0", "version": "8.15.0",
"resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz", "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz",
@ -2315,15 +2303,6 @@
"node": ">=14.17" "node": ">=14.17"
} }
}, },
"node_modules/undici-types": {
"version": "7.8.0",
"resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.8.0.tgz",
"integrity": "sha512-9UJ2xGDvQ43tYyVMpuHlsgApydB8ZKfVYTsLDhXkFL/6gfkp+U8xTGdh8pMJv1SpZna0zxG1DwsKZsreLbXBxw==",
"dev": true,
"license": "MIT",
"optional": true,
"peer": true
},
"node_modules/vite": { "node_modules/vite": {
"version": "7.0.6", "version": "7.0.6",
"resolved": "https://registry.npmjs.org/vite/-/vite-7.0.6.tgz", "resolved": "https://registry.npmjs.org/vite/-/vite-7.0.6.tgz",

View file

@ -60,10 +60,8 @@ export const triggerPlayNext = async function (roomId: string): Promise<[FetchEr
let songs: Song[] = [] let songs: Song[] = []
if (json["ended"]) { json["queue"].forEach(async (i: any) => {
json["queue"].forEach(async (i: any) => { songs.push(await parseSong(i))
songs.push(await parseSong(i)) })
})
}
return [null, songs, json["index"]] return [null, songs, json["index"]]
} }

View file

@ -4,6 +4,7 @@
import { onMount } from "svelte" import { onMount } from "svelte"
import type { FetchError } from "$lib/types" import type { FetchError } from "$lib/types"
import { getQueueSongs, triggerPlayNext } from "$lib/utils.js" import { getQueueSongs, triggerPlayNext } from "$lib/utils.js"
import Error from "$lib/components/Error.svelte"
let { data } = $props() let { data } = $props()
@ -20,24 +21,22 @@
playingIndex = index playingIndex = index
}) })
$effect(() => {
$inspect(queueSongs)
})
async function playNext() { async function playNext() {
let songs, index let songs, index
;[returnError, songs, index] = await triggerPlayNext(data.roomId) ;[returnError, songs, index] = await triggerPlayNext(data.roomId)
if (returnError) return if (returnError) return
if (songs.length != 0) queueSongs = songs queueSongs = songs
playingIndex = index playingIndex = index
} }
</script> </script>
{returnError} {#if returnError}
<Error {returnError} />
<div class="flex w-full flex-col items-center justify-center p-4 lg:p-10"> {:else}
<QueueSlider {queueSongs} {playingIndex} /> <div class="flex w-full flex-col items-center justify-center p-4 lg:p-10">
<button onclick={playNext}>Next</button> <QueueSlider {queueSongs} {playingIndex} />
</div> <button onclick={playNext}>Next</button>
</div>
{/if}

View file

@ -2,12 +2,12 @@
import QueueSlider from "$lib/components/QueueSlider.svelte" import QueueSlider from "$lib/components/QueueSlider.svelte"
import SuggestionInput from "$lib/components/SuggestionInput.svelte" import SuggestionInput from "$lib/components/SuggestionInput.svelte"
import Error from "$lib/components/Error.svelte" import Error from "$lib/components/Error.svelte"
import { type Suggestion, type Song } from "$lib/types.js" import { type Suggestion, type Song, parseSong, parseSuggestion } from "$lib/types.js"
import { onMount } from "svelte" import { onDestroy, onMount } from "svelte"
import SuggestionList from "$lib/components/SuggestionList.svelte" import SuggestionList from "$lib/components/SuggestionList.svelte"
import { getQueueSongs, getSuggestions, joinRoom } from "$lib/utils.js" import { getQueueSongs, getSuggestions, joinRoom } from "$lib/utils.js"
import type { FetchError } from "$lib/types.js" import type { FetchError } from "$lib/types.js"
import { io } from "socket.io-client" import { io, Socket } from "socket.io-client"
let { data } = $props() let { data } = $props()
@ -18,23 +18,20 @@
let returnError = $state<FetchError | null>() let returnError = $state<FetchError | null>()
let wsUrl = "" let socket: Socket
onMount(async () => { onMount(async () => {
// Join the room // Join the room
socket = io("/", { path: "/ws" })
let sugg, queue, index let sugg, queue, index
;[returnError, wsUrl] = await joinRoom(data.roomId) ;[returnError] = await joinRoom(data.roomId)
if (returnError) { if (returnError) {
return return
} }
;[returnError, sugg] = await getSuggestions(data.roomId) ;[returnError, sugg] = await getSuggestions(data.roomId)
if (returnError) return if (returnError) return
// Setup websocket connection
let socket = io("/", { path: "/ws" })
await socket.emitWithAck("join_room", { id: data.roomId })
;[returnError, queue, index] = await getQueueSongs(data.roomId) ;[returnError, queue, index] = await getQueueSongs(data.roomId)
if (returnError) { if (returnError) {
return return
@ -43,6 +40,28 @@
queueSongs = queue queueSongs = queue
suggestions = sugg suggestions = sugg
playingIndex = index playingIndex = index
// Setup websocket connection
await socket.emitWithAck("join_room", { id: data.roomId })
socket.on("queue_update", async (d) => {
const songs = await Promise.all(d.queue.map(parseSong))
queueSongs = songs
playingIndex = d.index
})
socket.on("new_vote", async (d) => {
const updated = await parseSuggestion(d.song)
suggestions = suggestions.map((s) => (s.uuid === updated.uuid ? updated : s))
})
socket.on("new_song", async (d) => {
const song = await parseSuggestion(d.song)
suggestions = [...suggestions, song]
})
})
onDestroy(() => {
if (socket) socket.disconnect()
}) })
</script> </script>