Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qffmpegtimecontroller.cpp
Go to the documentation of this file.
1// Copyright (C) 2021 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3
4#include "playbackengine/qffmpegtimecontroller_p.h"
5
6#include "qglobal.h"
7#include "qdebug.h"
8
9#include <algorithm>
10
11QT_BEGIN_NAMESPACE
12
13namespace QFFmpeg {
14
16{
17 sync();
18}
19
21{
22 return m_playbackRate;
23}
24
26{
27 if (playbackRate == m_playbackRate)
28 return;
29
30 Q_ASSERT(playbackRate > 0.f);
31
32 scrollTimeTillNow();
33 m_playbackRate = playbackRate;
34
35 if (m_softSyncData)
36 m_softSyncData = makeSoftSyncData(m_timePoint, m_position, m_softSyncData->dstTimePoint);
37}
38
39void TimeController::sync(TrackPosition trackPos)
40{
41 sync(SteadyClock::now(), trackPos);
42}
43
44void TimeController::sync(TimePoint tp, TrackPosition pos)
45{
46 m_softSyncData.reset();
47 m_position = pos;
48 m_timePoint = tp;
49}
50
51void TimeController::syncSoft(TimePoint tp, TrackPosition pos, SteadyClock::duration fixingTime)
52{
53 const auto srcTime = SteadyClock::now();
54 const auto srcPos = positionFromTime(srcTime, true);
55 const auto dstTime = srcTime + fixingTime;
56
57 m_position = pos;
58 m_timePoint = tp;
59
60 m_softSyncData = makeSoftSyncData(srcTime, srcPos, dstTime);
61}
62
63TrackPosition TimeController::currentPosition(SteadyClock::duration offset) const
64{
65 return positionFromTime(SteadyClock::now() + offset);
66}
67
68void TimeController::setPaused(bool paused)
69{
70 if (m_paused == paused)
71 return;
72
73 scrollTimeTillNow();
74 m_paused = paused;
75}
76
77TrackPosition TimeController::positionFromTime(TimePoint tp, bool ignorePause) const
78{
79 tp = m_paused && !ignorePause ? m_timePoint : tp;
80
81 if (m_softSyncData && tp < m_softSyncData->dstTimePoint) {
82 const PlaybackRate rate =
83 tp > m_softSyncData->srcTimePoint ? m_softSyncData->internalRate : m_playbackRate;
84
85 return m_softSyncData->srcPosition
86 + toTrackDuration(tp - m_softSyncData->srcTimePoint, rate);
87 }
88
89 return positionFromTimeInternal(tp);
90}
91
93 bool ignorePause) const
94{
95 auto position = m_paused && !ignorePause ? m_position : TrackPosition(pos);
96
97 if (m_softSyncData && position < m_softSyncData->dstPosition) {
98 const auto rate = position > m_softSyncData->srcPosition ? m_softSyncData->internalRate
99 : m_playbackRate;
100 return m_softSyncData->srcTimePoint
101 + toClockDuration(position - m_softSyncData->srcPosition, rate);
102 }
103
104 return timeFromPositionInternal(position);
105}
106
107TimeController::SoftSyncData TimeController::makeSoftSyncData(const TimePoint &srcTp,
108 const TrackPosition &srcPos,
109 const TimePoint &dstTp) const
110{
111 SoftSyncData result;
112 result.srcTimePoint = srcTp;
113 result.srcPosition = srcPos;
114 result.dstTimePoint = dstTp;
115 result.srcPosOffest = srcPos - positionFromTimeInternal(srcTp);
116 result.dstPosition = positionFromTimeInternal(dstTp);
117 result.internalRate =
118 static_cast<PlaybackRate>(toClockDuration(result.dstPosition - srcPos).count())
119 / (dstTp - srcTp).count();
120
121 return result;
122}
123
124TrackPosition TimeController::positionFromTimeInternal(const TimePoint &tp) const
125{
126 return m_position + toTrackDuration(tp - m_timePoint, m_playbackRate);
127}
128
129TimeController::TimePoint TimeController::timeFromPositionInternal(const TrackPosition &pos) const
130{
131 return m_timePoint + toClockDuration(pos - m_position, m_playbackRate);
132}
133
134void TimeController::scrollTimeTillNow()
135{
136 const auto now = SteadyClock::now();
137 if (!m_paused) {
138 m_position = positionFromTimeInternal(now);
139
140 // let's forget outdated syncronizations
141 if (m_softSyncData && m_softSyncData->dstTimePoint <= now)
142 m_softSyncData.reset();
143 } else if (m_softSyncData) {
144 m_softSyncData->dstTimePoint += now - m_timePoint;
145 m_softSyncData->srcTimePoint += now - m_timePoint;
146 }
147
148 m_timePoint = now;
149}
150
151SteadyClock::duration TimeController::toClockDuration(TrackDuration trackDuration, PlaybackRate rate)
152{
153 return std::chrono::duration_cast<SteadyClock::duration>(
154 std::chrono::microseconds(trackDuration.get()) / rate);
155}
156
157TrackDuration TimeController::toTrackDuration(SteadyClock::duration clockDuration, PlaybackRate rate)
158{
159 return TrackDuration(
160 std::chrono::duration_cast<std::chrono::microseconds>(clockDuration * rate).count());
161}
162
163} // namespace QFFmpeg
164
165QT_END_NAMESPACE
void syncSoft(TimePoint tp, TrackPosition pos, SteadyClock::duration fixingTime=std::chrono::seconds(4))
TimePoint timeFromPosition(TrackPosition pos, bool ignorePause=false) const
PlaybackRate playbackRate() const
TrackPosition currentPosition(SteadyClock::duration offset=SteadyClock::duration{ 0 }) const
TrackPosition positionFromTime(TimePoint tp, bool ignorePause=false) const
void sync(TimePoint tp, TrackPosition pos)
void setPlaybackRate(PlaybackRate playbackRate)
void sync(TrackPosition trackPos=TrackPosition(0))
std::conditional_t< QT_FFMPEG_AVIO_WRITE_CONST, const uint8_t *, uint8_t * > AvioWriteBufferType