mirror of
https://github.com/RGBCube/serenity
synced 2025-05-15 03:24:58 +00:00

SPDX License Identifiers are a more compact / standardized way of representing file license information. See: https://spdx.dev/resources/use/#identifiers This was done with the `ambr` search and replace tool. ambr --no-parent-ignore --key-from-file --rep-from-file key.txt rep.txt *
167 lines
3.4 KiB
C++
167 lines
3.4 KiB
C++
/*
|
|
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include "PlaybackManager.h"
|
|
|
|
PlaybackManager::PlaybackManager(NonnullRefPtr<Audio::ClientConnection> connection)
|
|
: m_connection(connection)
|
|
{
|
|
m_timer = Core::Timer::construct(100, [&]() {
|
|
if (!m_loader)
|
|
return;
|
|
next_buffer();
|
|
});
|
|
m_timer->stop();
|
|
}
|
|
|
|
PlaybackManager::~PlaybackManager()
|
|
{
|
|
}
|
|
|
|
void PlaybackManager::set_loader(NonnullRefPtr<Audio::Loader>&& loader)
|
|
{
|
|
stop();
|
|
m_loader = loader;
|
|
if (m_loader) {
|
|
m_total_length = m_loader->total_samples() / static_cast<float>(m_loader->sample_rate());
|
|
m_timer->start();
|
|
load_next_buffer();
|
|
} else {
|
|
m_timer->stop();
|
|
}
|
|
}
|
|
|
|
void PlaybackManager::stop()
|
|
{
|
|
set_paused(true);
|
|
m_connection->clear_buffer(true);
|
|
m_buffers.clear();
|
|
m_last_seek = 0;
|
|
m_next_buffer = nullptr;
|
|
m_current_buffer = nullptr;
|
|
m_next_ptr = 0;
|
|
|
|
if (m_loader)
|
|
m_loader->reset();
|
|
}
|
|
|
|
void PlaybackManager::play()
|
|
{
|
|
set_paused(false);
|
|
}
|
|
|
|
void PlaybackManager::loop(bool loop)
|
|
{
|
|
m_loop = loop;
|
|
}
|
|
|
|
void PlaybackManager::seek(const int position)
|
|
{
|
|
if (!m_loader)
|
|
return;
|
|
|
|
m_last_seek = position;
|
|
bool paused_state = m_paused;
|
|
set_paused(true);
|
|
|
|
m_connection->clear_buffer(true);
|
|
m_next_buffer = nullptr;
|
|
m_current_buffer = nullptr;
|
|
m_next_ptr = 0;
|
|
m_buffers.clear();
|
|
m_loader->seek(position);
|
|
|
|
if (!paused_state)
|
|
set_paused(false);
|
|
}
|
|
|
|
void PlaybackManager::pause()
|
|
{
|
|
set_paused(true);
|
|
}
|
|
|
|
void PlaybackManager::remove_dead_buffers()
|
|
{
|
|
int id = m_connection->get_playing_buffer();
|
|
int current_id = -1;
|
|
if (m_current_buffer)
|
|
current_id = m_current_buffer->id();
|
|
|
|
if (id >= 0 && id != current_id) {
|
|
while (!m_buffers.is_empty()) {
|
|
--m_next_ptr;
|
|
auto buffer = m_buffers.take_first();
|
|
|
|
if (buffer->id() == id) {
|
|
m_current_buffer = buffer;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void PlaybackManager::load_next_buffer()
|
|
{
|
|
if (m_buffers.size() < 10) {
|
|
for (int i = 0; i < 20 && m_loader->loaded_samples() < m_loader->total_samples(); i++) {
|
|
auto buffer = m_loader->get_more_samples(PLAYBACK_MANAGER_BUFFER_SIZE);
|
|
if (buffer) {
|
|
m_buffers.append(buffer);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (m_next_ptr < m_buffers.size()) {
|
|
m_next_buffer = m_buffers.at(m_next_ptr++);
|
|
if (on_load_sample_buffer)
|
|
on_load_sample_buffer(*m_next_buffer);
|
|
} else {
|
|
m_next_buffer = nullptr;
|
|
}
|
|
}
|
|
|
|
void PlaybackManager::set_paused(bool paused)
|
|
{
|
|
if (!m_next_buffer && m_loader)
|
|
load_next_buffer();
|
|
|
|
m_paused = paused;
|
|
m_connection->set_paused(paused);
|
|
}
|
|
|
|
bool PlaybackManager::toggle_pause()
|
|
{
|
|
if (m_paused) {
|
|
play();
|
|
} else {
|
|
pause();
|
|
}
|
|
return m_paused;
|
|
}
|
|
|
|
void PlaybackManager::next_buffer()
|
|
{
|
|
if (on_update)
|
|
on_update();
|
|
if (m_paused)
|
|
return;
|
|
|
|
remove_dead_buffers();
|
|
if (!m_next_buffer) {
|
|
if (!m_connection->get_remaining_samples() && !m_paused) {
|
|
stop();
|
|
if (on_finished_playing)
|
|
on_finished_playing();
|
|
}
|
|
return;
|
|
}
|
|
|
|
bool enqueued = m_connection->try_enqueue(*m_next_buffer);
|
|
if (!enqueued)
|
|
return;
|
|
|
|
load_next_buffer();
|
|
}
|