/* playback-controller.cpp - Implementation of the playback controller object Copyright (C) Lumiera.org 2008, Joel Holdsworth 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., 675 Mass Ave, Cambridge, MA 02139, USA. * *****************************************************/ #include "gui/controller/playback-controller.hpp" #include "lib/error.hpp" #include namespace gui { namespace controller { PlaybackController::PlaybackController() : finish_playback_thread(false), playing(false), playHandle(0) { } PlaybackController::~PlaybackController() { mutex.lock(); finish_playback_thread = true; mutex.unlock(); if (thread) thread->join(); } void PlaybackController::play() { Glib::Mutex::Lock lock(mutex); try { playHandle = & (proc::DummyPlayer::facade().start()); start_playback_thread(); playing = true; } catch (lumiera::error::State& err) { WARN (operate, "failed to start playback: %s" ,err.what()); lumiera_error(); playing = false; } } void PlaybackController::pause() { Glib::Mutex::Lock lock(mutex); playing = false; if (playHandle) playHandle->pause(true); } void PlaybackController::stop() { Glib::Mutex::Lock lock(mutex); playing = false; // TODO: stop player somehow? } bool PlaybackController::is_playing() { Glib::Mutex::Lock lock(mutex); return playing; } void PlaybackController::start_playback_thread() { dispatcher.connect(sigc::mem_fun(this, &PlaybackController::on_frame)); thread = Glib::Thread::create (sigc::mem_fun( this, &PlaybackController::playback_thread), true); } void PlaybackController::attach_viewer( const sigc::slot& on_frame) { frame_signal.connect(on_frame); } void PlaybackController::playback_thread() { for(;;) { { Glib::Mutex::Lock lock(mutex); if(finish_playback_thread) return; } if(is_playing()) pull_frame(); ////////////////////////////////TODO: usleep Glib::Thread::yield(); } } void PlaybackController::pull_frame() { REQUIRE (is_playing()); REQUIRE (playHandle); unsigned char * newBuffer = reinterpret_cast (playHandle->getFrame()); if (newBuffer != currentBuffer) { currentBuffer = newBuffer; dispatcher.emit(); } else { TRACE (render, "frame dropped?"); } } void PlaybackController::on_frame() { frame_signal.emit(currentBuffer); } } // namespace controller } // namespace gui