discord-jukebox-bot/pkg/web/static/app.js

125 lines
No EOL
12 KiB
JavaScript

// DOM Elements
const statusIndicator = document.getElementById('status-indicator');
const statusDot = document.querySelector('.status-dot');
const statusText = document.getElementById('status-text');
const nowPlaying = document.getElementById('now-playing');
const notPlaying = document.getElementById('not-playing');
const lastUpdate = document.getElementById('last-update');
// Song info elements
const coverArt = document.getElementById('cover-art');
const songTitle = document.getElementById('song-title');
const songArtist = document.getElementById('song-artist');
const songAlbum = document.getElementById('song-album');
const songDuration = document.getElementById('song-duration');
const songGenre = document.getElementById('song-genre');
const songYear = document.getElementById('song-year');
// Base64 encoded default cover art (simple music note icon)
const DEFAULT_COVER_ART = '';
// Initial status load
fetchStatus();
// Set up Server-Sent Events for live updates
setupEventSource();
// Functions
function setupEventSource() {
const eventSource = new EventSource('/events');
eventSource.onmessage = function(event) {
const data = JSON.parse(event.data);
updateUI(data);
};
eventSource.onerror = function() {
// If connection fails, close it and retry after a delay
eventSource.close();
statusDot.className = 'status-dot offline';
statusText.textContent = 'Connection lost. Retrying...';
setTimeout(setupEventSource, 5000);
};
}
function fetchStatus() {
fetch('/status')
.then(response => response.json())
.then(data => {
updateUI(data);
})
.catch(error => {
console.error('Error fetching status:', error);
statusDot.className = 'status-dot offline';
statusText.textContent = 'Error connecting to server';
});
}
function updateUI(data) {
// Update status indicator
if (data.isPlaying) {
statusDot.className = 'status-dot online';
statusText.textContent = 'Playing music';
nowPlaying.classList.remove('hidden');
notPlaying.classList.add('hidden');
} else {
statusDot.className = 'status-dot offline';
statusText.textContent = 'Not playing';
nowPlaying.classList.add('hidden');
notPlaying.classList.remove('hidden');
}
// Update song information if we have a current song
if (data.currentSong) {
updateSongInfo(data.currentSong);
}
// Update last update time
if (data.updateTime) {
const updateDate = new Date(data.updateTime);
lastUpdate.textContent = formatDate(updateDate);
}
}
function updateSongInfo(song) {
// Update the song title
songTitle.textContent = song.title || 'Unknown Title';
// Update the artist
songArtist.textContent = song.artist || 'Unknown Artist';
// Update the album
songAlbum.textContent = song.album || 'Unknown Album';
// Update the duration if available
if (song.duration) {
const minutes = Math.floor(song.duration / 60);
const seconds = song.duration % 60;
songDuration.textContent = `${minutes}:${seconds.toString().padStart(2, '0')}`;
} else {
songDuration.textContent = '--';
}
// Update the genre if available
songGenre.textContent = song.genre || '--';
// Update the year if available
songYear.textContent = song.year ? song.year.toString() : '--';
// Update the cover art if available
if (song.coverArt) {
// This assumes your backend has a route to get cover art
// You might need to adjust this to match your API
const coverArtUrl = `/cover/${song.coverArt}`;
coverArt.src = coverArtUrl;
coverArt.alt = `${song.album} cover`;
} else {
coverArt.src = DEFAULT_COVER_ART;
coverArt.alt = 'Default cover art';
}
}
function formatDate(date) {
return date.toLocaleTimeString([], { hour: '2-digit', minute: '2-digit', second: '2-digit' });
}