// 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'); const songHistory = document.getElementById('song-history'); const historyList = document.getElementById('history-list'); // 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'); // Progress bar elements const progressBar = document.getElementById('progress-bar'); const currentTime = document.getElementById('current-time'); const totalTime = document.getElementById('total-time'); // Base64 encoded default cover art (simple music note icon) const DEFAULT_COVER_ART = ''; // Track if initial data has been received let initialDataReceived = false; // Track song progress state let songStartTime = null; let songDurationSeconds = 0; let progressInterval = null; // Show loading indicator const loadingOverlay = document.getElementById('loading-overlay'); const connectionError = document.getElementById('connection-error'); // 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); initialDataReceived = true; updateUI(data); // Hide loading indicator once we've received data if (loadingOverlay) { loadingOverlay.classList.add('hidden'); } // Hide connection error message if it was showing if (connectionError) { connectionError.classList.add('hidden'); } }; 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...'; // Show connection error message if (connectionError) { connectionError.classList.remove('hidden'); } setTimeout(setupEventSource, 5000); }; } function fetchStatus() { fetch('/status') .then(response => response.json()) .then(data => { initialDataReceived = true; updateUI(data); // Hide loading indicator once we've received data if (loadingOverlay) { loadingOverlay.classList.add('hidden'); } // Hide connection error message if it was showing if (connectionError) { connectionError.classList.add('hidden'); } }) .catch(error => { console.error('Error fetching status:', error); statusDot.className = 'status-dot offline'; statusText.textContent = 'Error connecting to server'; // Show connection error message if (connectionError) { connectionError.classList.remove('hidden'); } }); } function updateUI(data) { // If this is the first data we've received, update everything const isInitialUpdate = !document.getElementById('data-loaded'); if (isInitialUpdate) { // Mark the page as having received data const dataLoadedMark = document.createElement('div'); dataLoadedMark.id = 'data-loaded'; dataLoadedMark.style.display = 'none'; document.body.appendChild(dataLoadedMark); } // 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'); // Stop progress bar updates if not playing stopProgressUpdates(); } // Update song information if we have a current song if (data.currentSong) { updateSongInfo(data.currentSong, data.currentSongStarted); } // Update song history if available // Always show history if it exists, even when nothing is playing if (data.songHistory && data.songHistory.length > 0) { updateSongHistory(data.songHistory, data.currentSong); songHistory.classList.remove('hidden'); } else { // Hide history section only if there's no history songHistory.classList.add('hidden'); } // Update last update time if (data.updateTime) { const updateDate = new Date(data.updateTime); lastUpdate.textContent = formatDate(updateDate); } } function updateSongInfo(song, startTimeStr) { // 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')}`; songDurationSeconds = song.duration; // Update total time in progress bar totalTime.textContent = `${minutes}:${seconds.toString().padStart(2, '0')}`; } else { songDuration.textContent = '--'; songDurationSeconds = 0; totalTime.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'; } // Set up progress bar setupProgressBar(startTimeStr); } function formatDate(date) { return date.toLocaleTimeString([], { hour: '2-digit', minute: '2-digit', second: '2-digit' }); } // Format seconds to MM:SS format function formatTime(seconds) { const minutes = Math.floor(seconds / 60); const secs = Math.floor(seconds % 60); return `${minutes}:${secs.toString().padStart(2, '0')}`; } // Set up progress bar with song start time function setupProgressBar(startTimeStr) { // Stop any existing progress updates stopProgressUpdates(); // Reset progress bar progressBar.style.width = '0%'; currentTime.textContent = '0:00'; // If we don't have a start time or duration, exit if (!startTimeStr || songDurationSeconds <= 0) { return; } // Parse the start time songStartTime = new Date(startTimeStr); // Start progress updates updateProgress(); progressInterval = setInterval(updateProgress, 1000); } // Update the progress bar function updateProgress() { if (!songStartTime || songDurationSeconds <= 0) { return; } // Calculate elapsed time in seconds const now = new Date(); const elapsedSeconds = (now - songStartTime) / 1000; // Calculate progress percentage const progressPercent = Math.min(100, (elapsedSeconds / songDurationSeconds) * 100); // Update progress bar width progressBar.style.width = `${progressPercent}%`; // Update current time display const displaySeconds = Math.min(songDurationSeconds, elapsedSeconds); currentTime.textContent = formatTime(displaySeconds); // If we've reached the end of the song (with a small buffer) if (elapsedSeconds >= songDurationSeconds + 2) { stopProgressUpdates(); } } // Stop progress updates function stopProgressUpdates() { if (progressInterval) { clearInterval(progressInterval); progressInterval = null; } songStartTime = null; } // Update the song history list function updateSongHistory(songs, currentSong) { // Clear the current list historyList.innerHTML = ''; // If there's no song history yet, show a message if (!songs || songs.length === 0) { const li = document.createElement('li'); li.className = 'history-item history-empty'; li.textContent = 'No song history available yet'; historyList.appendChild(li); return; } // Create a set of songs for the history that excludes the current song const historySongs = []; const currentId = currentSong ? currentSong.id : null; // Filter out null entries and the current song for (const song of songs) { if (!song) continue; if (currentId && song.id === currentId) continue; historySongs.push(song); } // If after filtering, we don't have any songs to display if (historySongs.length === 0) { const li = document.createElement('li'); li.className = 'history-item history-empty'; li.textContent = 'No previous songs to display'; historyList.appendChild(li); return; } // Add each song to the history list for (const song of historySongs) { const li = document.createElement('li'); li.className = 'history-item'; // Create cover art const img = document.createElement('img'); if (song.coverArt) { img.src = `/cover/${song.coverArt}`; } else { img.src = DEFAULT_COVER_ART; } img.alt = 'Cover'; img.className = 'history-cover'; // Create song details const details = document.createElement('div'); details.className = 'history-details'; const title = document.createElement('div'); title.className = 'history-title'; title.textContent = song.title || 'Unknown Title'; const artist = document.createElement('div'); artist.className = 'history-artist'; artist.textContent = song.artist || 'Unknown Artist'; // Assemble the elements details.appendChild(title); details.appendChild(artist); li.appendChild(img); li.appendChild(details); historyList.appendChild(li); } }