1b6666fa39
The TAG_MODIFIED handler (i.e. playlist::TagModified()) works only if the modified song is the current song - something that is not updated until SYNC_WITH_PLAYER is finished. This fixes tag updates right after a new song is started.
150 lines
3.0 KiB
C++
150 lines
3.0 KiB
C++
/*
|
|
* Copyright 2003-2017 The Music Player Daemon Project
|
|
* http://www.musicpd.org
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include "Partition.hxx"
|
|
#include "Instance.hxx"
|
|
#include "DetachedSong.hxx"
|
|
#include "mixer/Volume.hxx"
|
|
#include "IdleFlags.hxx"
|
|
|
|
Partition::Partition(Instance &_instance,
|
|
unsigned max_length,
|
|
unsigned buffer_chunks,
|
|
unsigned buffered_before_play,
|
|
AudioFormat configured_audio_format,
|
|
const ReplayGainConfig &replay_gain_config)
|
|
:instance(_instance),
|
|
global_events(instance.event_loop, BIND_THIS_METHOD(OnGlobalEvent)),
|
|
playlist(max_length, *this),
|
|
outputs(*this),
|
|
pc(*this, outputs, buffer_chunks, buffered_before_play,
|
|
configured_audio_format, replay_gain_config)
|
|
{
|
|
UpdateEffectiveReplayGainMode();
|
|
}
|
|
|
|
void
|
|
Partition::EmitIdle(unsigned mask)
|
|
{
|
|
instance.EmitIdle(mask);
|
|
}
|
|
|
|
void
|
|
Partition::UpdateEffectiveReplayGainMode()
|
|
{
|
|
auto mode = replay_gain_mode;
|
|
if (mode == ReplayGainMode::AUTO)
|
|
mode = playlist.queue.random
|
|
? ReplayGainMode::TRACK
|
|
: ReplayGainMode::ALBUM;
|
|
|
|
pc.LockSetReplayGainMode(mode);
|
|
|
|
outputs.SetReplayGainMode(mode);
|
|
}
|
|
|
|
#ifdef ENABLE_DATABASE
|
|
|
|
const Database *
|
|
Partition::GetDatabase() const
|
|
{
|
|
return instance.GetDatabase();
|
|
}
|
|
|
|
const Database &
|
|
Partition::GetDatabaseOrThrow() const
|
|
{
|
|
return instance.GetDatabaseOrThrow();
|
|
}
|
|
|
|
void
|
|
Partition::DatabaseModified(const Database &db)
|
|
{
|
|
playlist.DatabaseModified(db);
|
|
EmitIdle(IDLE_DATABASE);
|
|
}
|
|
|
|
#endif
|
|
|
|
void
|
|
Partition::TagModified()
|
|
{
|
|
DetachedSong *song = pc.LockReadTaggedSong();
|
|
if (song != nullptr) {
|
|
playlist.TagModified(std::move(*song));
|
|
delete song;
|
|
}
|
|
}
|
|
|
|
void
|
|
Partition::SyncWithPlayer()
|
|
{
|
|
playlist.SyncWithPlayer(pc);
|
|
}
|
|
|
|
void
|
|
Partition::OnQueueModified()
|
|
{
|
|
EmitIdle(IDLE_PLAYLIST);
|
|
}
|
|
|
|
void
|
|
Partition::OnQueueOptionsChanged()
|
|
{
|
|
EmitIdle(IDLE_OPTIONS);
|
|
}
|
|
|
|
void
|
|
Partition::OnQueueSongStarted()
|
|
{
|
|
EmitIdle(IDLE_PLAYER);
|
|
}
|
|
|
|
void
|
|
Partition::OnPlayerSync()
|
|
{
|
|
EmitGlobalEvent(SYNC_WITH_PLAYER);
|
|
}
|
|
|
|
void
|
|
Partition::OnPlayerTagModified()
|
|
{
|
|
EmitGlobalEvent(TAG_MODIFIED);
|
|
}
|
|
|
|
void
|
|
Partition::OnMixerVolumeChanged(gcc_unused Mixer &mixer, gcc_unused int volume)
|
|
{
|
|
InvalidateHardwareVolume();
|
|
|
|
/* notify clients */
|
|
EmitIdle(IDLE_MIXER);
|
|
}
|
|
|
|
void
|
|
Partition::OnGlobalEvent(unsigned mask)
|
|
{
|
|
if ((mask & SYNC_WITH_PLAYER) != 0)
|
|
SyncWithPlayer();
|
|
|
|
if ((mask & TAG_MODIFIED) != 0)
|
|
TagModified();
|
|
}
|