diff --git a/pkg/discord/jukebox.go b/pkg/discord/jukebox.go index 10120c1..a896f76 100644 --- a/pkg/discord/jukebox.go +++ b/pkg/discord/jukebox.go @@ -22,6 +22,7 @@ import ( type JukeboxPlayer struct { bot *Bot currentSong *subsonic.Song + currentSongStarted time.Time playlist []subsonic.Song playlistMutex sync.Mutex playingMutex sync.Mutex @@ -29,6 +30,7 @@ type JukeboxPlayer struct { songHistory []*subsonic.Song historyMutex sync.RWMutex maxHistorySize int + initialized bool } // NewJukeboxPlayer creates a new jukebox player @@ -38,6 +40,7 @@ func NewJukeboxPlayer(bot *Bot) *JukeboxPlayer { playlist: make([]subsonic.Song, 0), songHistory: make([]*subsonic.Song, 0), maxHistorySize: 20, // Store the last 20 songs + initialized: true, } // Register command handlers @@ -236,6 +239,7 @@ func (j *JukeboxPlayer) startPlaying() { // Clear current song when stopping j.playingMutex.Lock() j.currentSong = nil + j.currentSongStarted = time.Time{} // Reset time to zero value j.playingMutex.Unlock() return @@ -273,6 +277,7 @@ func (j *JukeboxPlayer) startPlaying() { j.playingMutex.Lock() j.currentSong = song + j.currentSongStarted = time.Now() j.playingMutex.Unlock() // Add song to history when it starts playing @@ -781,6 +786,13 @@ func (j *JukeboxPlayer) GetCurrentSong() *subsonic.Song { return j.currentSong } +// GetCurrentSongInfo returns the current song and its start time +func (j *JukeboxPlayer) GetCurrentSongInfo() (*subsonic.Song, time.Time) { + j.playingMutex.Lock() + defer j.playingMutex.Unlock() + return j.currentSong, j.currentSongStarted +} + // addToHistory adds a song to the playback history func (j *JukeboxPlayer) addToHistory(song *subsonic.Song) { if song == nil { @@ -790,6 +802,12 @@ func (j *JukeboxPlayer) addToHistory(song *subsonic.Song) { j.historyMutex.Lock() defer j.historyMutex.Unlock() + // Avoid duplicate entries at the beginning of history + if len(j.songHistory) > 0 && j.songHistory[0] != nil && + j.songHistory[0].ID == song.ID { + return + } + // Add the song to the beginning of the history list j.songHistory = append([]*subsonic.Song{song}, j.songHistory...) @@ -804,6 +822,11 @@ func (j *JukeboxPlayer) GetSongHistory() []*subsonic.Song { j.historyMutex.RLock() defer j.historyMutex.RUnlock() + // Ensure we don't return a nil slice even if no history yet + if j.songHistory == nil { + return make([]*subsonic.Song, 0) + } + // Create a copy of the history to return history := make([]*subsonic.Song, len(j.songHistory)) copy(history, j.songHistory) diff --git a/pkg/web/server.go b/pkg/web/server.go index 3cbd17c..8cfbc3e 100644 --- a/pkg/web/server.go +++ b/pkg/web/server.go @@ -149,10 +149,11 @@ func (s *Server) handleStatic(w http.ResponseWriter, r *http.Request) { // StatusResponse contains the jukebox status type StatusResponse struct { - IsPlaying bool `json:"isPlaying"` - CurrentSong *subsonic.Song `json:"currentSong"` - SongHistory []*subsonic.Song `json:"songHistory"` - UpdateTime string `json:"updateTime"` + IsPlaying bool `json:"isPlaying"` + CurrentSong *subsonic.Song `json:"currentSong"` + CurrentSongStarted string `json:"currentSongStarted"` + SongHistory []*subsonic.Song `json:"songHistory"` + UpdateTime string `json:"updateTime"` } // handleStatus returns the current jukebox status as JSON @@ -165,15 +166,22 @@ func (s *Server) handleStatus(w http.ResponseWriter, r *http.Request) { // getStatus returns the current status func (s *Server) getStatus() StatusResponse { - currentSong := s.jukebox.GetCurrentSong() + currentSong, songStartTime := s.jukebox.GetCurrentSongInfo() songHistory := s.jukebox.GetSongHistory() isPlaying := s.bot.IsPlaying() + + // Format the start time if it's not zero + startTimeStr := "" + if !songStartTime.IsZero() { + startTimeStr = songStartTime.Format(time.RFC3339) + } return StatusResponse{ - IsPlaying: isPlaying, - CurrentSong: currentSong, - SongHistory: songHistory, - UpdateTime: time.Now().Format(time.RFC3339), + IsPlaying: isPlaying, + CurrentSong: currentSong, + CurrentSongStarted: startTimeStr, + SongHistory: songHistory, + UpdateTime: time.Now().Format(time.RFC3339), } } @@ -208,7 +216,8 @@ func (s *Server) handleEvents(w http.ResponseWriter, r *http.Request) { notifyChan <- true }() - // Send initial status update + // Send initial status update as soon as connection is established + // This ensures the client receives current data immediately without waiting for changes status := s.getStatus() data, err := json.Marshal(status) if err == nil { @@ -287,6 +296,7 @@ func (s *Server) broadcastUpdates() { // Keep track of the previous status to only send updates when there are changes var prevIsPlaying bool var prevSongID string + var prevHistoryHash string ticker := time.NewTicker(1 * time.Second) defer ticker.Stop() @@ -309,12 +319,23 @@ func (s *Server) broadcastUpdates() { currentSongID = status.CurrentSong.ID } - hasChanged := (prevIsPlaying != status.IsPlaying) || (prevSongID != currentSongID) + // Create a simple hash of the history to detect changes + historyHash := "" + for _, song := range status.SongHistory { + if song != nil { + historyHash += song.ID + ":" + } + } + + hasChanged := (prevIsPlaying != status.IsPlaying) || + (prevSongID != currentSongID) || + (prevHistoryHash != historyHash) if hasChanged { // Update previous state prevIsPlaying = status.IsPlaying prevSongID = currentSongID + prevHistoryHash = historyHash // Broadcast to all clients data, err := json.Marshal(status) diff --git a/pkg/web/static/app.js b/pkg/web/static/app.js index d923e89..d7e7021 100644 --- a/pkg/web/static/app.js +++ b/pkg/web/static/app.js @@ -17,9 +17,26 @@ 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(); @@ -32,7 +49,18 @@ function setupEventSource() { 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() { @@ -41,6 +69,11 @@ function setupEventSource() { statusDot.className = 'status-dot offline'; statusText.textContent = 'Connection lost. Retrying...'; + // Show connection error message + if (connectionError) { + connectionError.classList.remove('hidden'); + } + setTimeout(setupEventSource, 5000); }; } @@ -49,16 +82,43 @@ 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'; @@ -70,19 +130,23 @@ function updateUI(data) { 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); + 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 { - // No history available + // Hide history section only if there's no history songHistory.classList.add('hidden'); } @@ -93,7 +157,7 @@ function updateUI(data) { } } -function updateSongInfo(song) { +function updateSongInfo(song, startTimeStr) { // Update the song title songTitle.textContent = song.title || 'Unknown Title'; @@ -108,8 +172,14 @@ function updateSongInfo(song) { 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 @@ -129,24 +199,86 @@ function updateSongInfo(song) { 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 = ''; - // Show history even when nothing is playing - - // Start from the first song - const startIndex = 0; - - // If no songs are available after filtering current one, show a message - if (songs.length === 0) { + // 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'; @@ -154,14 +286,28 @@ function updateSongHistory(songs, currentSong) { return; } - // Add each song to the history list - for (let i = startIndex; i < songs.length; i++) { - const song = songs[i]; + // 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; - - // Skip if this song is the same as the currently playing song - if (currentSong && song.id === currentSong.id) 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'; diff --git a/pkg/web/static/index.html b/pkg/web/static/index.html index 821f626..0998a08 100644 --- a/pkg/web/static/index.html +++ b/pkg/web/static/index.html @@ -8,6 +8,10 @@
+Loading jukebox data...
+