tdesktop/Telegram/SourceFiles/audio.h

242 lines
4.4 KiB
C
Raw Normal View History

2014-09-04 07:33:44 +00:00
/*
This file is part of Telegram Desktop,
the official desktop version of Telegram messaging app, see https://telegram.org
2014-09-04 07:33:44 +00:00
Telegram Desktop 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.
It 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.
Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
Copyright (c) 2014 John Preston, https://desktop.telegram.org
2014-09-04 07:33:44 +00:00
*/
#pragma once
#include "types.h"
void audioInit();
bool audioWorks();
void audioPlayNotify();
void audioFinish();
2015-05-29 18:52:43 +00:00
enum AudioPlayerState {
AudioPlayerStopped,
AudioPlayerStoppedAtStart,
AudioPlayerStarting,
AudioPlayerPlaying,
AudioPlayerFinishing,
AudioPlayerPausing,
AudioPlayerPaused,
AudioPlayerResuming,
2014-09-04 07:33:44 +00:00
};
class AudioPlayerFader;
class AudioPlayerLoaders;
2015-05-29 18:52:43 +00:00
class AudioPlayer : public QObject {
2014-09-04 07:33:44 +00:00
Q_OBJECT
public:
2015-05-29 18:52:43 +00:00
AudioPlayer();
2014-09-04 07:33:44 +00:00
void play(AudioData *audio);
void pauseresume();
2015-05-29 18:52:43 +00:00
void currentState(AudioData **audio, AudioPlayerState *state = 0, int64 *position = 0, int64 *duration = 0, int32 *frequency = 0);
void clearStoppedAtStart(AudioData *audio);
2015-06-03 12:18:46 +00:00
void resumeDevice();
2014-09-04 07:33:44 +00:00
2015-05-29 18:52:43 +00:00
~AudioPlayer();
2014-09-04 07:33:44 +00:00
public slots:
void onError(AudioData *audio);
signals:
void updated(AudioData *audio);
void stopped(AudioData *audio);
void faderOnTimer();
void loaderOnStart(AudioData *audio);
void loaderOnCancel(AudioData *audio);
private:
bool updateCurrentStarted(int32 pos = -1);
struct Msg {
Msg() : audio(0), position(0), duration(0), frequency(AudioVoiceMsgFrequency), skipStart(0), skipEnd(0), loading(0), started(0),
2015-05-29 18:52:43 +00:00
state(AudioPlayerStopped), source(0), nextBuffer(0) {
2014-09-04 07:33:44 +00:00
memset(buffers, 0, sizeof(buffers));
memset(samplesCount, 0, sizeof(samplesCount));
}
AudioData *audio;
QString fname;
QByteArray data;
int64 position, duration;
int32 frequency;
2014-09-04 07:33:44 +00:00
int64 skipStart, skipEnd;
bool loading;
int64 started;
2015-05-29 18:52:43 +00:00
AudioPlayerState state;
2014-09-04 07:33:44 +00:00
uint32 source;
int32 nextBuffer;
uint32 buffers[3];
int64 samplesCount[3];
};
int32 _current;
Msg _data[AudioVoiceMsgSimultaneously];
QMutex _mutex;
2015-05-29 18:52:43 +00:00
friend class AudioPlayerFader;
friend class AudioPlayerLoaders;
2014-09-04 07:33:44 +00:00
2015-05-29 18:52:43 +00:00
QThread _faderThread, _loaderThread;
AudioPlayerFader *_fader;
AudioPlayerLoaders *_loader;
2014-09-04 07:33:44 +00:00
};
class AudioCaptureInner;
2015-05-29 18:52:43 +00:00
class AudioCapture : public QObject {
Q_OBJECT
public:
AudioCapture();
void start();
void stop(bool needResult);
bool check();
2015-05-29 18:52:43 +00:00
~AudioCapture();
signals:
void captureOnStart();
void captureOnStop(bool needResult);
2014-09-04 07:33:44 +00:00
2015-05-29 18:52:43 +00:00
void onDone(QByteArray data, qint32 samples);
void onUpdate(qint16 level, qint32 samples);
void onError();
private:
friend class AudioCaptureInner;
QThread _captureThread;
AudioCaptureInner *_capture;
};
AudioPlayer *audioPlayer();
AudioCapture *audioCapture();
class AudioPlayerFader : public QObject {
2014-09-04 07:33:44 +00:00
Q_OBJECT
public:
2015-05-29 18:52:43 +00:00
AudioPlayerFader(QThread *thread);
2015-06-03 12:18:46 +00:00
void resumeDevice();
2014-09-04 07:33:44 +00:00
signals:
void error(AudioData *audio);
void playPositionUpdated(AudioData *audio);
void audioStopped(AudioData *audio);
void needToPreload(AudioData *audio);
2015-06-03 12:18:46 +00:00
void stopPauseDevice();
2015-01-10 13:08:30 +00:00
2014-09-04 07:33:44 +00:00
public slots:
void onInit();
void onTimer();
2015-06-03 12:18:46 +00:00
void onPauseTimer();
void onPauseTimerStop();
2014-09-04 07:33:44 +00:00
private:
2015-06-03 12:18:46 +00:00
QTimer _timer, _pauseTimer;
QMutex _pauseMutex;
bool _pauseFlag, _paused;
2014-09-04 07:33:44 +00:00
};
2015-05-29 18:52:43 +00:00
class AudioPlayerLoader;
class AudioPlayerLoaders : public QObject {
2014-09-04 07:33:44 +00:00
Q_OBJECT
public:
2015-05-29 18:52:43 +00:00
AudioPlayerLoaders(QThread *thread);
~AudioPlayerLoaders();
2014-09-04 07:33:44 +00:00
signals:
void error(AudioData *audio);
void needToCheck();
public slots:
void onInit();
void onStart(AudioData *audio);
void onLoad(AudioData *audio);
void onCancel(AudioData *audio);
private:
2015-05-29 18:52:43 +00:00
typedef QMap<AudioData*, AudioPlayerLoader*> Loaders;
2014-09-04 07:33:44 +00:00
Loaders _loaders;
void loadError(Loaders::iterator i);
};
2015-05-29 18:52:43 +00:00
struct AudioCapturePrivate;
2015-05-29 18:52:43 +00:00
class AudioCaptureInner : public QObject {
Q_OBJECT
public:
AudioCaptureInner(QThread *thread);
~AudioCaptureInner();
signals:
void error();
void update(qint16 level, qint32 samples);
void done(QByteArray data, qint32 samples);
public slots:
void onInit();
void onStart();
void onStop(bool needResult);
void onTimeout();
private:
void writeFrame(int32 offset, int32 framesize);
AudioCapturePrivate *d;
QTimer _timer;
QByteArray _captured;
};