Files
libopenshot/src/Qt/AudioPlaybackThread.cpp

148 lines
3.8 KiB
C++
Raw Normal View History

2014-01-27 19:03:46 +08:00
/**
* @file
* @brief Source file for AudioPlaybackThread class
* @author Duzy Chan <code@duzy.info>
* @author Jonathan Thomas <jonathan@openshot.org> *
2014-01-27 19:03:46 +08:00
*
* @section LICENSE
*
* Copyright (c) 2008-2013 OpenShot Studios, LLC
* (http://www.openshotstudios.com). This file is part of
* OpenShot Library (http://www.openshot.org), an open-source project
* dedicated to delivering high quality video editing and animation solutions
* to the world.
*
* OpenShot Library 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 3 of the License, or
* (at your option) any later version.
*
* OpenShot Library 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 OpenShot Library. If not, see <http://www.gnu.org/licenses/>.
*/
#include "../include/ReaderBase.h"
#include "../include/RendererBase.h"
#include "../include/AudioReaderSource.h"
2014-01-27 19:03:46 +08:00
#include "AudioPlaybackThread.h"
namespace openshot
{
2014-02-02 02:20:30 +08:00
struct SafeTimeSliceThread : TimeSliceThread
{
SafeTimeSliceThread(const String & s) : TimeSliceThread(s) {}
void run()
{
try {
TimeSliceThread::run();
} catch (const TooManySeeks & e) {
// ...
}
}
};
// Construtor
2014-01-27 19:03:46 +08:00
AudioPlaybackThread::AudioPlaybackThread()
: Thread("audio-playback")
, audioDeviceManager()
, player()
, transport()
, mixer()
, source(NULL)
, sampleRate(0.0)
, numChannels(0)
, buffer_size(10000)
{
}
// Destructor
AudioPlaybackThread::~AudioPlaybackThread()
{
}
// Set the reader object
void AudioPlaybackThread::Reader(ReaderBase *reader)
{
if (!source) {
sampleRate = reader->info.sample_rate;
numChannels = reader->info.channels;
source = new AudioReaderSource(reader, 1, buffer_size);
}
}
// Get the current frame object (which is filling the buffer)
tr1::shared_ptr<Frame> AudioPlaybackThread::getFrame()
{
if (source) return source->getFrame();
return tr1::shared_ptr<Frame>();
}
// Get the currently playing frame number
int AudioPlaybackThread::getCurrentFramePosition()
{
return source ? source->getEstimatedFrame() : 0;
}
// Seek the audio thread
void AudioPlaybackThread::Seek(int new_position)
{
source->Seek(new_position);
}
// Start audio thread
void AudioPlaybackThread::run()
2014-01-27 19:03:46 +08:00
{
2014-02-11 15:53:19 +08:00
// Init audio device
2014-01-27 19:03:46 +08:00
audioDeviceManager.initialise (
0, /* number of input channels */
numChannels, /* number of output channels */
2014-01-27 19:03:46 +08:00
0, /* no XML settings.. */
true /* select default device on failure */);
// Add callback
2014-01-27 19:03:46 +08:00
audioDeviceManager.addAudioCallback(&player);
// Create TimeSliceThread for audio buffering
2014-02-02 02:20:30 +08:00
SafeTimeSliceThread thread("audio-buffer");
thread.startThread();
// Connect source to transport
transport.setSource(
source,
buffer_size, // tells it to buffer this many samples ahead
2014-02-02 02:20:30 +08:00
&thread,
sampleRate,
numChannels);
transport.setPosition(0);
transport.setGain(1.0);
// Connect transport to mixer and player
mixer.addInputSource(&transport, false);
player.setSource(&mixer);
cout << "starting transport" << endl;
transport.start();
2014-02-11 15:53:19 +08:00
while (!threadShouldExit() && transport.isPlaying()) {
sleep(100);
}
transport.stop();
transport.setSource(0);
2014-01-27 19:03:46 +08:00
player.setSource(0);
audioDeviceManager.removeAudioCallback(&player);
audioDeviceManager.closeAudioDevice();
audioDeviceManager.removeAllChangeListeners();
audioDeviceManager.dispatchPendingMessages();
// Remove source
delete source;
source = NULL;
2014-01-27 19:03:46 +08:00
}
}