mirror of https://github.com/procxx/kepka.git
Support streaming playback speed 0.5 - 2.
This commit is contained in:
parent
26ea6c4e63
commit
ec9512899e
|
@ -324,9 +324,9 @@ void StartStreaming(
|
||||||
base::take(video) = nullptr;
|
base::take(video) = nullptr;
|
||||||
});
|
});
|
||||||
|
|
||||||
player->init(
|
auto options = Media::Streaming::PlaybackOptions();
|
||||||
(document->isAudioFile() ? Mode::Audio : Mode::Both),
|
options.speed = 1.;
|
||||||
0);
|
player->init(options);
|
||||||
player->updates(
|
player->updates(
|
||||||
) | rpl::start_with_next_error_done([=](Update &&update) {
|
) | rpl::start_with_next_error_done([=](Update &&update) {
|
||||||
update.data.match([&](Information &update) {
|
update.data.match([&](Information &update) {
|
||||||
|
|
|
@ -35,11 +35,19 @@ ALCcontext *AudioContext = nullptr;
|
||||||
constexpr auto kSuppressRatioAll = 0.2;
|
constexpr auto kSuppressRatioAll = 0.2;
|
||||||
constexpr auto kSuppressRatioSong = 0.05;
|
constexpr auto kSuppressRatioSong = 0.05;
|
||||||
constexpr auto kPlaybackSpeedMultiplier = 1.7;
|
constexpr auto kPlaybackSpeedMultiplier = 1.7;
|
||||||
constexpr auto kPlaybackSpeedTune = -9;
|
|
||||||
|
|
||||||
auto VolumeMultiplierAll = 1.;
|
auto VolumeMultiplierAll = 1.;
|
||||||
auto VolumeMultiplierSong = 1.;
|
auto VolumeMultiplierSong = 1.;
|
||||||
|
|
||||||
|
// Value for AL_PITCH_SHIFTER_COARSE_TUNE effect, 0.5 <= speed <= 2.
|
||||||
|
int CoarseTuneForSpeed(float64 speed) {
|
||||||
|
Expects(speed >= 0.5 && speed <= 2.);
|
||||||
|
|
||||||
|
constexpr auto kTuneSteps = 12;
|
||||||
|
const auto tuneRatio = std::log(speed) / std::log(2.);
|
||||||
|
return -int(std::round(kTuneSteps * tuneRatio));
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
namespace Media {
|
namespace Media {
|
||||||
|
@ -165,7 +173,7 @@ bool CreatePlaybackDevice() {
|
||||||
// initialize the pitch shifter effect
|
// initialize the pitch shifter effect
|
||||||
alEffecti(_playbackSpeedData.uiEffect, AL_EFFECT_TYPE, AL_EFFECT_PITCH_SHIFTER);
|
alEffecti(_playbackSpeedData.uiEffect, AL_EFFECT_TYPE, AL_EFFECT_PITCH_SHIFTER);
|
||||||
// 12 semitones = 1 octave
|
// 12 semitones = 1 octave
|
||||||
alEffecti(_playbackSpeedData.uiEffect, AL_PITCH_SHIFTER_COARSE_TUNE, kPlaybackSpeedTune);
|
alEffecti(_playbackSpeedData.uiEffect, AL_PITCH_SHIFTER_COARSE_TUNE, CoarseTuneForSpeed(kPlaybackSpeedMultiplier));
|
||||||
// connect the effect with the effect slot
|
// connect the effect with the effect slot
|
||||||
alAuxiliaryEffectSloti(_playbackSpeedData.uiEffectSlot, AL_EFFECTSLOT_EFFECT, _playbackSpeedData.uiEffect);
|
alAuxiliaryEffectSloti(_playbackSpeedData.uiEffectSlot, AL_EFFECTSLOT_EFFECT, _playbackSpeedData.uiEffect);
|
||||||
// initialize a filter to disable the direct (dry) path
|
// initialize a filter to disable the direct (dry) path
|
||||||
|
@ -337,6 +345,22 @@ void Mixer::Track::createStream(AudioMsgId::Type type) {
|
||||||
alGenBuffers(3, stream.buffers);
|
alGenBuffers(3, stream.buffers);
|
||||||
if (type == AudioMsgId::Type::Voice) {
|
if (type == AudioMsgId::Type::Voice) {
|
||||||
mixer()->updatePlaybackSpeed(this);
|
mixer()->updatePlaybackSpeed(this);
|
||||||
|
} else if (speedEffect) {
|
||||||
|
alGenAuxiliaryEffectSlots(1, &speedEffect->effectSlot);
|
||||||
|
alGenEffects(1, &speedEffect->effect);
|
||||||
|
alGenFilters(1, &speedEffect->filter);
|
||||||
|
alEffecti(speedEffect->effect, AL_EFFECT_TYPE, AL_EFFECT_PITCH_SHIFTER);
|
||||||
|
alEffecti(speedEffect->effect, AL_PITCH_SHIFTER_COARSE_TUNE, speedEffect->coarseTune);
|
||||||
|
alAuxiliaryEffectSloti(speedEffect->effectSlot, AL_EFFECTSLOT_EFFECT, speedEffect->effect);
|
||||||
|
alFilteri(speedEffect->filter, AL_FILTER_TYPE, AL_FILTER_LOWPASS);
|
||||||
|
alFilterf(speedEffect->filter, AL_LOWPASS_GAIN, 0.f);
|
||||||
|
|
||||||
|
alSourcef(stream.source, AL_PITCH, speedEffect->speed);
|
||||||
|
alSource3i(stream.source, AL_AUXILIARY_SEND_FILTER, speedEffect->effectSlot, 0, 0);
|
||||||
|
alSourcei(stream.source, AL_DIRECT_FILTER, speedEffect->filter);
|
||||||
|
} else {
|
||||||
|
alSource3i(stream.source, AL_AUXILIARY_SEND_FILTER, AL_EFFECTSLOT_NULL, 0, 0);
|
||||||
|
alSourcei(stream.source, AL_DIRECT_FILTER, AL_FILTER_NULL);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -349,6 +373,18 @@ void Mixer::Track::destroyStream() {
|
||||||
for (auto i = 0; i != 3; ++i) {
|
for (auto i = 0; i != 3; ++i) {
|
||||||
stream.buffers[i] = 0;
|
stream.buffers[i] = 0;
|
||||||
}
|
}
|
||||||
|
destroySpeedEffect();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Mixer::Track::destroySpeedEffect() {
|
||||||
|
if (!speedEffect) {
|
||||||
|
return;
|
||||||
|
} else if (alIsEffect(speedEffect->effect)) {
|
||||||
|
alDeleteEffects(1, &speedEffect->effect);
|
||||||
|
alDeleteAuxiliaryEffectSlots(1, &speedEffect->effectSlot);
|
||||||
|
alDeleteFilters(1, &speedEffect->filter);
|
||||||
|
}
|
||||||
|
speedEffect = nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mixer::Track::reattach(AudioMsgId::Type type) {
|
void Mixer::Track::reattach(AudioMsgId::Type type) {
|
||||||
|
@ -381,6 +417,7 @@ void Mixer::Track::reattach(AudioMsgId::Type type) {
|
||||||
void Mixer::Track::detach() {
|
void Mixer::Track::detach() {
|
||||||
resetStream();
|
resetStream();
|
||||||
destroyStream();
|
destroyStream();
|
||||||
|
destroySpeedEffect();
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mixer::Track::clear() {
|
void Mixer::Track::clear() {
|
||||||
|
@ -402,7 +439,7 @@ void Mixer::Track::clear() {
|
||||||
bufferSamples[i] = QByteArray();
|
bufferSamples[i] = QByteArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
videoData = nullptr;
|
setVideoData(nullptr);
|
||||||
lastUpdateWhen = 0;
|
lastUpdateWhen = 0;
|
||||||
lastUpdateCorrectedMs = 0;
|
lastUpdateCorrectedMs = 0;
|
||||||
}
|
}
|
||||||
|
@ -480,6 +517,16 @@ int Mixer::Track::getNotQueuedBufferIndex() {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Mixer::Track::setVideoData(std::unique_ptr<VideoSoundData> data) {
|
||||||
|
destroySpeedEffect();
|
||||||
|
if (data && data->speed != 1.) {
|
||||||
|
speedEffect = std::make_unique<SpeedEffect>();
|
||||||
|
speedEffect->speed = data->speed;
|
||||||
|
speedEffect->coarseTune = CoarseTuneForSpeed(data->speed);
|
||||||
|
}
|
||||||
|
videoData = std::move(data);
|
||||||
|
}
|
||||||
|
|
||||||
void Mixer::Track::resetStream() {
|
void Mixer::Track::resetStream() {
|
||||||
if (isStreamCreated()) {
|
if (isStreamCreated()) {
|
||||||
alSourceStop(stream.source);
|
alSourceStop(stream.source);
|
||||||
|
@ -737,7 +784,7 @@ void Mixer::play(
|
||||||
current->lastUpdateWhen = 0;
|
current->lastUpdateWhen = 0;
|
||||||
current->lastUpdateCorrectedMs = 0;
|
current->lastUpdateCorrectedMs = 0;
|
||||||
if (videoData) {
|
if (videoData) {
|
||||||
current->videoData = std::move(videoData);
|
current->setVideoData(std::move(videoData));
|
||||||
} else {
|
} else {
|
||||||
current->file = audio.audio()->location(true);
|
current->file = audio.audio()->location(true);
|
||||||
current->data = audio.audio()->data();
|
current->data = audio.audio()->data();
|
||||||
|
|
|
@ -193,6 +193,8 @@ private:
|
||||||
|
|
||||||
int getNotQueuedBufferIndex();
|
int getNotQueuedBufferIndex();
|
||||||
|
|
||||||
|
void setVideoData(std::unique_ptr<VideoSoundData> data);
|
||||||
|
|
||||||
~Track();
|
~Track();
|
||||||
|
|
||||||
TrackState state;
|
TrackState state;
|
||||||
|
@ -217,12 +219,21 @@ private:
|
||||||
Stream stream;
|
Stream stream;
|
||||||
std::unique_ptr<VideoSoundData> videoData;
|
std::unique_ptr<VideoSoundData> videoData;
|
||||||
|
|
||||||
|
struct SpeedEffect {
|
||||||
|
uint32 effect = 0;
|
||||||
|
uint32 effectSlot = 0;
|
||||||
|
uint32 filter = 0;
|
||||||
|
int coarseTune = 0;
|
||||||
|
float64 speed = 1.;
|
||||||
|
};
|
||||||
|
std::unique_ptr<SpeedEffect> speedEffect;
|
||||||
crl::time lastUpdateWhen = 0;
|
crl::time lastUpdateWhen = 0;
|
||||||
crl::time lastUpdateCorrectedMs = 0;
|
crl::time lastUpdateCorrectedMs = 0;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void createStream(AudioMsgId::Type type);
|
void createStream(AudioMsgId::Type type);
|
||||||
void destroyStream();
|
void destroyStream();
|
||||||
|
void destroySpeedEffect();
|
||||||
void resetStream();
|
void resetStream();
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
|
@ -14,6 +14,7 @@ struct VideoSoundData {
|
||||||
AVFrame *frame = nullptr;
|
AVFrame *frame = nullptr;
|
||||||
int32 frequency = Media::Player::kDefaultFrequency;
|
int32 frequency = Media::Player::kDefaultFrequency;
|
||||||
int64 length = 0;
|
int64 length = 0;
|
||||||
|
float64 speed = 1.; // 0.5 <= speed <= 2.
|
||||||
~VideoSoundData();
|
~VideoSoundData();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -16,10 +16,12 @@ namespace Media {
|
||||||
namespace Streaming {
|
namespace Streaming {
|
||||||
|
|
||||||
AudioTrack::AudioTrack(
|
AudioTrack::AudioTrack(
|
||||||
|
const PlaybackOptions &options,
|
||||||
Stream &&stream,
|
Stream &&stream,
|
||||||
FnMut<void(const Information &)> ready,
|
FnMut<void(const Information &)> ready,
|
||||||
Fn<void()> error)
|
Fn<void()> error)
|
||||||
: _stream(std::move(stream))
|
: _options(options)
|
||||||
|
, _stream(std::move(stream))
|
||||||
, _ready(std::move(ready))
|
, _ready(std::move(ready))
|
||||||
, _error(std::move(error)) {
|
, _error(std::move(error)) {
|
||||||
Expects(_ready != nullptr);
|
Expects(_ready != nullptr);
|
||||||
|
@ -79,6 +81,7 @@ void AudioTrack::mixerInit() {
|
||||||
data->context = _stream.codec.release();
|
data->context = _stream.codec.release();
|
||||||
data->frequency = _stream.frequency;
|
data->frequency = _stream.frequency;
|
||||||
data->length = (_stream.duration * data->frequency) / 1000LL;
|
data->length = (_stream.duration * data->frequency) / 1000LL;
|
||||||
|
data->speed = _options.speed;
|
||||||
Media::Player::mixer()->play(
|
Media::Player::mixer()->play(
|
||||||
_audioMsgId,
|
_audioMsgId,
|
||||||
std::move(data),
|
std::move(data),
|
||||||
|
@ -105,7 +108,7 @@ void AudioTrack::mixerEnqueue(Packet &&packet) {
|
||||||
packet.release();
|
packet.release();
|
||||||
}
|
}
|
||||||
|
|
||||||
void AudioTrack::start() {
|
void AudioTrack::start(crl::time startTime) {
|
||||||
Expects(_ready == nullptr);
|
Expects(_ready == nullptr);
|
||||||
Expects(_audioMsgId.playId() != 0);
|
Expects(_audioMsgId.playId() != 0);
|
||||||
|
|
||||||
|
|
|
@ -18,13 +18,14 @@ public:
|
||||||
// Called from some unspecified thread.
|
// Called from some unspecified thread.
|
||||||
// Callbacks are assumed to be thread-safe.
|
// Callbacks are assumed to be thread-safe.
|
||||||
AudioTrack(
|
AudioTrack(
|
||||||
|
const PlaybackOptions &options,
|
||||||
Stream &&stream,
|
Stream &&stream,
|
||||||
FnMut<void(const Information &)> ready,
|
FnMut<void(const Information &)> ready,
|
||||||
Fn<void()> error);
|
Fn<void()> error);
|
||||||
|
|
||||||
// Called from the main thread.
|
// Called from the main thread.
|
||||||
// Must be called after 'ready' was invoked.
|
// Must be called after 'ready' was invoked.
|
||||||
void start();
|
void start(crl::time startTime);
|
||||||
|
|
||||||
// Called from the main thread.
|
// Called from the main thread.
|
||||||
// Non-const, because we subscribe to changes on the first call.
|
// Non-const, because we subscribe to changes on the first call.
|
||||||
|
@ -49,6 +50,8 @@ private:
|
||||||
void mixerEnqueue(Packet &&packet);
|
void mixerEnqueue(Packet &&packet);
|
||||||
void callReady();
|
void callReady();
|
||||||
|
|
||||||
|
const PlaybackOptions _options;
|
||||||
|
|
||||||
// Accessed from the same unspecified thread.
|
// Accessed from the same unspecified thread.
|
||||||
Stream _stream;
|
Stream _stream;
|
||||||
bool _noMoreData = false;
|
bool _noMoreData = false;
|
||||||
|
|
|
@ -22,6 +22,12 @@ enum class Mode {
|
||||||
Inspection,
|
Inspection,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct PlaybackOptions {
|
||||||
|
Mode mode = Mode::Both;
|
||||||
|
crl::time position = 0;
|
||||||
|
float64 speed = 1.; // Valid values between 0.5 and 2.
|
||||||
|
};
|
||||||
|
|
||||||
struct TrackState {
|
struct TrackState {
|
||||||
crl::time position = kTimeUnknown;
|
crl::time position = kTimeUnknown;
|
||||||
crl::time receivedTill = kTimeUnknown;
|
crl::time receivedTill = kTimeUnknown;
|
||||||
|
|
|
@ -92,11 +92,13 @@ void Player::start() {
|
||||||
// video finished
|
// video finished
|
||||||
}, _lifetime);
|
}, _lifetime);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_startedTime = crl::now();
|
||||||
if (_audio) {
|
if (_audio) {
|
||||||
_audio->start();
|
_audio->start(_startedTime);
|
||||||
}
|
}
|
||||||
if (_video) {
|
if (_video) {
|
||||||
_video->start();
|
_video->start(_startedTime);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -192,23 +194,26 @@ void Player::fileReady(Stream &&video, Stream &&audio) {
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
if (audio.codec && (_mode == Mode::Audio || _mode == Mode::Both)) {
|
const auto mode = _options.mode;
|
||||||
|
if (audio.codec && (mode == Mode::Audio || mode == Mode::Both)) {
|
||||||
_audio = std::make_unique<AudioTrack>(
|
_audio = std::make_unique<AudioTrack>(
|
||||||
|
_options,
|
||||||
std::move(audio),
|
std::move(audio),
|
||||||
ready,
|
ready,
|
||||||
error(_audio));
|
error(_audio));
|
||||||
}
|
}
|
||||||
if (video.codec && (_mode == Mode::Video || _mode == Mode::Both)) {
|
if (video.codec && (mode == Mode::Video || mode == Mode::Both)) {
|
||||||
_video = std::make_unique<VideoTrack>(
|
_video = std::make_unique<VideoTrack>(
|
||||||
|
_options,
|
||||||
std::move(video),
|
std::move(video),
|
||||||
ready,
|
ready,
|
||||||
error(_video));
|
error(_video));
|
||||||
}
|
}
|
||||||
if ((_mode == Mode::Audio && !_audio)
|
if ((mode == Mode::Audio && !_audio)
|
||||||
|| (_mode == Mode::Video && !_video)
|
|| (mode == Mode::Video && !_video)
|
||||||
|| (!_audio && !_video)) {
|
|| (!_audio && !_video)) {
|
||||||
LOG(("Streaming Error: Required stream not found for mode %1."
|
LOG(("Streaming Error: Required stream not found for mode %1."
|
||||||
).arg(int(_mode)));
|
).arg(int(mode)));
|
||||||
fileError();
|
fileError();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -277,8 +282,8 @@ void Player::provideStartInformation() {
|
||||||
|| (_video && _information.video.state.duration == kTimeUnknown)) {
|
|| (_video && _information.video.state.duration == kTimeUnknown)) {
|
||||||
return; // Not ready yet.
|
return; // Not ready yet.
|
||||||
} else if ((!_audio && !_video)
|
} else if ((!_audio && !_video)
|
||||||
|| (!_audio && _mode == Mode::Audio)
|
|| (!_audio && _options.mode == Mode::Audio)
|
||||||
|| (!_video && _mode == Mode::Video)) {
|
|| (!_video && _options.mode == Mode::Video)) {
|
||||||
fail();
|
fail();
|
||||||
} else {
|
} else {
|
||||||
_stage = Stage::Ready;
|
_stage = Stage::Ready;
|
||||||
|
@ -298,12 +303,14 @@ void Player::fail() {
|
||||||
stopGuarded();
|
stopGuarded();
|
||||||
}
|
}
|
||||||
|
|
||||||
void Player::init(Mode mode, crl::time position) {
|
void Player::init(const PlaybackOptions &options) {
|
||||||
|
Expects(options.speed >= 0.5 && options.speed <= 2.);
|
||||||
|
|
||||||
stop();
|
stop();
|
||||||
|
|
||||||
_mode = mode;
|
_options = options;
|
||||||
_stage = Stage::Initializing;
|
_stage = Stage::Initializing;
|
||||||
_file->start(delegate(), position);
|
_file->start(delegate(), _options.position);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Player::pause() {
|
void Player::pause() {
|
||||||
|
|
|
@ -33,7 +33,7 @@ public:
|
||||||
Player(const Player &other) = delete;
|
Player(const Player &other) = delete;
|
||||||
Player &operator=(const Player &other) = delete;
|
Player &operator=(const Player &other) = delete;
|
||||||
|
|
||||||
void init(Mode mode, crl::time position);
|
void init(const PlaybackOptions &options);
|
||||||
void start();
|
void start();
|
||||||
void pause();
|
void pause();
|
||||||
void resume();
|
void resume();
|
||||||
|
@ -102,7 +102,7 @@ private:
|
||||||
std::unique_ptr<AudioTrack> _audio;
|
std::unique_ptr<AudioTrack> _audio;
|
||||||
std::unique_ptr<VideoTrack> _video;
|
std::unique_ptr<VideoTrack> _video;
|
||||||
base::has_weak_ptr _sessionGuard;
|
base::has_weak_ptr _sessionGuard;
|
||||||
Mode _mode = Mode::Both;
|
PlaybackOptions _options;
|
||||||
|
|
||||||
// Belongs to the File thread while File is active.
|
// Belongs to the File thread while File is active.
|
||||||
bool _readTillEnd = false;
|
bool _readTillEnd = false;
|
||||||
|
@ -112,6 +112,7 @@ private:
|
||||||
Stage _stage = Stage::Uninitialized;
|
Stage _stage = Stage::Uninitialized;
|
||||||
bool _paused = false;
|
bool _paused = false;
|
||||||
|
|
||||||
|
crl::time _startedTime = kTimeUnknown;
|
||||||
crl::time _nextFrameTime = kTimeUnknown;
|
crl::time _nextFrameTime = kTimeUnknown;
|
||||||
base::Timer _renderFrameTimer;
|
base::Timer _renderFrameTimer;
|
||||||
rpl::event_stream<Update, Error> _updates;
|
rpl::event_stream<Update, Error> _updates;
|
||||||
|
|
|
@ -25,6 +25,7 @@ public:
|
||||||
|
|
||||||
VideoTrackObject(
|
VideoTrackObject(
|
||||||
crl::weak_on_queue<VideoTrackObject> weak,
|
crl::weak_on_queue<VideoTrackObject> weak,
|
||||||
|
const PlaybackOptions &options,
|
||||||
not_null<Shared*> shared,
|
not_null<Shared*> shared,
|
||||||
Stream &&stream,
|
Stream &&stream,
|
||||||
FnMut<void(const Information &)> ready,
|
FnMut<void(const Information &)> ready,
|
||||||
|
@ -34,7 +35,7 @@ public:
|
||||||
|
|
||||||
[[nodisacrd]] rpl::producer<crl::time> displayFrameAt() const;
|
[[nodisacrd]] rpl::producer<crl::time> displayFrameAt() const;
|
||||||
|
|
||||||
void start();
|
void start(crl::time startTime);
|
||||||
void interrupt();
|
void interrupt();
|
||||||
void frameDisplayed();
|
void frameDisplayed();
|
||||||
|
|
||||||
|
@ -54,6 +55,7 @@ private:
|
||||||
[[nodiscard]] crl::time trackTime() const;
|
[[nodiscard]] crl::time trackTime() const;
|
||||||
|
|
||||||
const crl::weak_on_queue<VideoTrackObject> _weak;
|
const crl::weak_on_queue<VideoTrackObject> _weak;
|
||||||
|
const PlaybackOptions _options;
|
||||||
|
|
||||||
// Main thread wrapper destructor will set _shared back to nullptr.
|
// Main thread wrapper destructor will set _shared back to nullptr.
|
||||||
// All queued method calls after that should be discarded.
|
// All queued method calls after that should be discarded.
|
||||||
|
@ -75,11 +77,13 @@ private:
|
||||||
|
|
||||||
VideoTrackObject::VideoTrackObject(
|
VideoTrackObject::VideoTrackObject(
|
||||||
crl::weak_on_queue<VideoTrackObject> weak,
|
crl::weak_on_queue<VideoTrackObject> weak,
|
||||||
|
const PlaybackOptions &options,
|
||||||
not_null<Shared*> shared,
|
not_null<Shared*> shared,
|
||||||
Stream &&stream,
|
Stream &&stream,
|
||||||
FnMut<void(const Information &)> ready,
|
FnMut<void(const Information &)> ready,
|
||||||
Fn<void()> error)
|
Fn<void()> error)
|
||||||
: _weak(std::move(weak))
|
: _weak(std::move(weak))
|
||||||
|
, _options(options)
|
||||||
, _shared(shared)
|
, _shared(shared)
|
||||||
, _stream(std::move(stream))
|
, _stream(std::move(stream))
|
||||||
, _ready(std::move(ready))
|
, _ready(std::move(ready))
|
||||||
|
@ -92,7 +96,9 @@ VideoTrackObject::VideoTrackObject(
|
||||||
rpl::producer<crl::time> VideoTrackObject::displayFrameAt() const {
|
rpl::producer<crl::time> VideoTrackObject::displayFrameAt() const {
|
||||||
return _nextFrameDisplayPosition.value(
|
return _nextFrameDisplayPosition.value(
|
||||||
) | rpl::map([=](crl::time displayPosition) {
|
) | rpl::map([=](crl::time displayPosition) {
|
||||||
return _startedTime + (displayPosition - _startedPosition);
|
return _startedTime
|
||||||
|
+ crl::time(std::round((displayPosition - _startedPosition)
|
||||||
|
/ _options.speed));
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -176,8 +182,8 @@ void VideoTrackObject::presentFrameIfNeeded() {
|
||||||
queueReadFrames(presented.nextCheckDelay);
|
queueReadFrames(presented.nextCheckDelay);
|
||||||
}
|
}
|
||||||
|
|
||||||
void VideoTrackObject::start() {
|
void VideoTrackObject::start(crl::time startTime) {
|
||||||
_startedTime = crl::now();
|
_startedTime = startTime;
|
||||||
queueReadFrames();
|
queueReadFrames();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -257,7 +263,9 @@ void VideoTrackObject::callReady() {
|
||||||
|
|
||||||
crl::time VideoTrackObject::trackTime() const {
|
crl::time VideoTrackObject::trackTime() const {
|
||||||
return _startedPosition
|
return _startedPosition
|
||||||
+ (_startedTime != kTimeUnknown ? (crl::now() - _startedTime) : 0);
|
+ crl::time((_startedTime != kTimeUnknown
|
||||||
|
? std::round((crl::now() - _startedTime) * _options.speed)
|
||||||
|
: 0.));
|
||||||
}
|
}
|
||||||
|
|
||||||
void VideoTrackObject::interrupt() {
|
void VideoTrackObject::interrupt() {
|
||||||
|
@ -409,6 +417,7 @@ not_null<VideoTrack::Frame*> VideoTrack::Shared::frameForPaint() {
|
||||||
}
|
}
|
||||||
|
|
||||||
VideoTrack::VideoTrack(
|
VideoTrack::VideoTrack(
|
||||||
|
const PlaybackOptions &options,
|
||||||
Stream &&stream,
|
Stream &&stream,
|
||||||
FnMut<void(const Information &)> ready,
|
FnMut<void(const Information &)> ready,
|
||||||
Fn<void()> error)
|
Fn<void()> error)
|
||||||
|
@ -417,6 +426,7 @@ VideoTrack::VideoTrack(
|
||||||
//, _streamRotation(stream.rotation)
|
//, _streamRotation(stream.rotation)
|
||||||
, _shared(std::make_unique<Shared>())
|
, _shared(std::make_unique<Shared>())
|
||||||
, _wrapped(
|
, _wrapped(
|
||||||
|
options,
|
||||||
_shared.get(),
|
_shared.get(),
|
||||||
std::move(stream),
|
std::move(stream),
|
||||||
std::move(ready),
|
std::move(ready),
|
||||||
|
@ -439,9 +449,9 @@ void VideoTrack::process(Packet &&packet) {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
void VideoTrack::start() {
|
void VideoTrack::start(crl::time startTime) {
|
||||||
_wrapped.with([](Implementation &unwrapped) {
|
_wrapped.with([=](Implementation &unwrapped) {
|
||||||
unwrapped.start();
|
unwrapped.start(startTime);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -21,6 +21,7 @@ public:
|
||||||
// Called from some unspecified thread.
|
// Called from some unspecified thread.
|
||||||
// Callbacks are assumed to be thread-safe.
|
// Callbacks are assumed to be thread-safe.
|
||||||
VideoTrack(
|
VideoTrack(
|
||||||
|
const PlaybackOptions &options,
|
||||||
Stream &&stream,
|
Stream &&stream,
|
||||||
FnMut<void(const Information &)> ready,
|
FnMut<void(const Information &)> ready,
|
||||||
Fn<void()> error);
|
Fn<void()> error);
|
||||||
|
@ -33,7 +34,7 @@ public:
|
||||||
void process(Packet &&packet);
|
void process(Packet &&packet);
|
||||||
|
|
||||||
// Called from the main thread.
|
// Called from the main thread.
|
||||||
void start();
|
void start(crl::time startTime);
|
||||||
// Returns the position of the displayed frame.
|
// Returns the position of the displayed frame.
|
||||||
[[nodiscard]] crl::time markFrameDisplayed(crl::time now);
|
[[nodiscard]] crl::time markFrameDisplayed(crl::time now);
|
||||||
[[nodiscard]] QImage frame(const FrameRequest &request) const;
|
[[nodiscard]] QImage frame(const FrameRequest &request) const;
|
||||||
|
|
Loading…
Reference in New Issue