lumiera_/src/gui/controller/playback-controller.cpp

167 lines
3.4 KiB
C++
Raw Normal View History

2009-01-17 17:12:11 +01:00
/*
playback-controller.cpp - Implementation of the playback controller object
2009-01-17 17:12:11 +01:00
Copyright (C) Lumiera.org
2008, Joel Holdsworth <joel@airwebreathe.org.uk>
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 <nobug.h>
2009-01-17 17:12:11 +01:00
namespace gui {
namespace controller {
PlaybackController::PlaybackController() :
thread(0),
finish_playback_thread(false),
playing(false),
playHandle(0)
{ }
PlaybackController::~PlaybackController()
{
end_playback_thread();
}
2009-01-17 17:12:11 +01:00
void
PlaybackController::play()
{
Lock sync(this);
try
{
if (playing && thread && playHandle) playHandle->pause(false);
else
{
playHandle = & (proc::DummyPlayer::facade().start());
if (thread)
end_playback_thread();
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()
{
Lock sync(this);
playing = false;
if (playHandle)
playHandle->pause(true);
}
void
PlaybackController::stop()
{
Lock sync(this);
playing = false;
end_playback_thread();
playHandle = 0;
// TODO: stop player somehow?
}
bool
PlaybackController::is_playing()
{
Lock sync(this);
return playing;
2009-01-17 17:12:11 +01:00
}
void
PlaybackController::start_playback_thread()
2009-01-17 17:53:01 +01:00
{
dispatcher.connect(sigc::mem_fun(this, &PlaybackController::on_frame));
finish_playback_thread = false;
thread = Glib::Thread::create (sigc::mem_fun(
this, &PlaybackController::playback_thread), true);
2009-01-17 17:53:01 +01:00
}
void
PlaybackController::end_playback_thread()
{
Lock sync(this);
finish_playback_thread = true;
if (thread)
thread->join();
thread = 0;
}
void
PlaybackController::attach_viewer(
const sigc::slot<void, void*>& on_frame)
2009-01-17 17:53:01 +01:00
{
frame_signal.connect(on_frame);
}
void
PlaybackController::playback_thread()
{
for(;;)
{
{
Lock sync(this);
if(finish_playback_thread)
return;
}
if(is_playing())
pull_frame();
usleep(40000); // ca 25 frames pre second
}
}
void
PlaybackController::pull_frame()
{
REQUIRE (is_playing());
REQUIRE (playHandle);
unsigned char * newBuffer = reinterpret_cast<unsigned char*> (playHandle->getFrame());
if (newBuffer != currentBuffer)
{
currentBuffer = newBuffer;
dispatcher.emit();
}
else
{
TRACE (render, "frame dropped?");
}
}
void
PlaybackController::on_frame()
{
frame_signal.emit(currentBuffer);
2009-01-17 17:53:01 +01:00
}
2009-01-17 17:12:11 +01:00
} // namespace controller
} // namespace gui