Qt
Internal/Contributor docs for the Qt SDK. <b>Note:</b> These are NOT official API docs; those are found <a href='https://doc.qt.io/'>here</a>.
Loading...
Searching...
No Matches
qffmpegmediarecorder.cpp
Go to the documentation of this file.
1// Copyright (C) 2016 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
5#include "qaudiodevice.h"
6#include <private/qmediastoragelocation_p.h>
7#include <private/qplatformcamera_p.h>
8#include <private/qplatformsurfacecapture_p.h>
9#include "qaudiosource.h"
10#include "qffmpegaudioinput_p.h"
11#include "qaudiobuffer.h"
14
15#include <qdebug.h>
16#include <qloggingcategory.h>
17
18Q_STATIC_LOGGING_CATEGORY(qLcMediaEncoder, "qt.multimedia.ffmpeg.encoder");
19
21
25
27
29{
30 return true;
31}
32
33void QFFmpegMediaRecorder::handleSessionError(QMediaRecorder::Error code, const QString &description)
34{
35 updateError(code, description);
36 stop();
37}
38
40{
41 if (!m_session || state() != QMediaRecorder::StoppedState)
42 return;
43
44 auto videoSources = m_session->activeVideoSources();
45 auto audioInputs = m_session->activeAudioInputs();
46 const auto hasVideo = !videoSources.empty();
47 const auto hasAudio = !audioInputs.empty();
48
49 if (!hasVideo && !hasAudio) {
50 updateError(QMediaRecorder::ResourceError, QMediaRecorder::tr("No video or audio input"));
51 return;
52 }
53
54 if (outputDevice() && !outputLocation().isEmpty())
55 qCWarning(qLcMediaEncoder)
56 << "Both outputDevice and outputLocation has been set to QMediaRecorder";
57
58 if (outputDevice() && !outputDevice()->isWritable())
59 qCWarning(qLcMediaEncoder) << "Output device has been set but not it's not writable";
60
62 auto formatContext = std::make_unique<QFFmpeg::EncodingFormatContext>(settings.fileFormat());
63
64 if (outputDevice() && outputDevice()->isWritable()) {
65 formatContext->openAVIO(outputDevice());
66 } else {
68 qCDebug(qLcMediaEncoder) << "recording new media to" << actualLocation;
69 formatContext->openAVIO(actualLocation);
70 }
71
72 qCDebug(qLcMediaEncoder) << "requested format:" << settings.fileFormat()
73 << settings.audioCodec();
74
75 if (!formatContext->isAVIOOpen()) {
77 QMediaRecorder::tr("Cannot open the output location for writing"));
78 return;
79 }
80
81 m_recordingEngine.reset(new RecordingEngine(settings, std::move(formatContext)));
82 m_recordingEngine->setMetaData(m_metaData);
83
84 connect(m_recordingEngine.get(), &QFFmpeg::RecordingEngine::durationChanged, this,
85 &QFFmpegMediaRecorder::newDuration);
86 connect(m_recordingEngine.get(), &QFFmpeg::RecordingEngine::finalizationDone, this,
87 &QFFmpegMediaRecorder::finalizationDone);
88 connect(m_recordingEngine.get(), &QFFmpeg::RecordingEngine::sessionError, this,
89 &QFFmpegMediaRecorder::handleSessionError);
90
92
93 auto handleStreamInitializationError = [this](QMediaRecorder::Error code,
94 const QString &description) {
95 qCWarning(qLcMediaEncoder) << "Stream initialization error:" << description;
96 updateError(code, description);
97 };
98
100 handleStreamInitializationError);
101
105
106 m_recordingEngine->initialize(audioInputs, videoSources);
107}
108
110{
111 if (!m_session || state() != QMediaRecorder::RecordingState)
112 return;
113
114 Q_ASSERT(m_recordingEngine);
115 m_recordingEngine->setPaused(true);
116
118}
119
121{
122 if (!m_session || state() != QMediaRecorder::PausedState)
123 return;
124
125 Q_ASSERT(m_recordingEngine);
126 m_recordingEngine->setPaused(false);
127
129}
130
132{
133 if (!m_session || state() == QMediaRecorder::StoppedState)
134 return;
135 auto * input = m_session ? m_session->audioInput() : nullptr;
136 if (input)
137 static_cast<QFFmpegAudioInput *>(input)->setRunning(false);
138 qCDebug(qLcMediaEncoder) << "stop";
139
140 m_recordingEngine.reset();
141}
142
143void QFFmpegMediaRecorder::finalizationDone()
144{
146}
147
149{
150 if (!m_session)
151 return;
152 m_metaData = metaData;
153}
154
156{
157 return m_metaData;
158}
159
161{
162 auto *captureSession = session;
163 if (m_session == captureSession)
164 return;
165
166 if (m_session)
167 stop();
168
169 m_session = captureSession;
170 if (!m_session)
171 return;
172}
173
175{
176 const bool autoStop = mediaRecorder()->autoStop();
177 if (!m_recordingEngine || m_recordingEngine->autoStop() == autoStop)
178 return;
179
180 if (autoStop)
181 connect(m_recordingEngine.get(), &QFFmpeg::RecordingEngine::autoStopped, this,
183 else
184 disconnect(m_recordingEngine.get(), &QFFmpeg::RecordingEngine::autoStopped, this,
186
187 m_recordingEngine->setAutoStop(autoStop);
188}
189
190void QFFmpegMediaRecorder::RecordingEngineDeleter::operator()(
191 RecordingEngine *recordingEngine) const
192{
193 // ### all of the below should be done asynchronous. finalize() should do it's work in a thread
194 // to avoid blocking the UI in case of slow codecs
195 recordingEngine->finalize();
196}
197
199
200#include "moc_qffmpegmediarecorder_p.cpp"
QPlatformAudioInput * audioInput() const
std::vector< QPlatformAudioBufferInputBase * > activeAudioInputs() const
void setCaptureSession(QFFmpegMediaCaptureSession *session)
void record(QMediaEncoderSettings &settings) override
bool isLocationWritable(const QUrl &sink) const override
virtual ~QFFmpegMediaRecorder()
QMediaMetaData metaData() const override
void setMetaData(const QMediaMetaData &) override
QFFmpegMediaRecorder(QMediaRecorder *parent)
void sessionError(QMediaRecorder::Error code, const QString &description)
void durationChanged(qint64 duration)
void streamInitializationError(QMediaRecorder::Error code, const QString &description)
\inmodule QtMultimedia
\inmodule QtMultimedia
bool autoStop
This property controls whether the media recorder stops automatically when all media inputs have repo...
Error
\qmlproperty enumeration QtMultimedia::MediaRecorder::error
static QMetaObject::Connection connect(const QObject *sender, const char *signal, const QObject *receiver, const char *member, Qt::ConnectionType=Qt::AutoConnection)
\threadsafe
Definition qobject.cpp:2960
std::vector< QPlatformVideoSource * > activeVideoSources()
void actualLocationChanged(const QUrl &location)
void updateError(QMediaRecorder::Error error, const QString &errorString)
void stateChanged(QMediaRecorder::RecorderState state)
void durationChanged(qint64 position)
virtual QMediaRecorder::RecorderState state() const
QString findActualLocation(const QMediaEncoderSettings &settings) const
\macro QT_RESTRICTED_CAST_FROM_ASCII
Definition qstring.h:129
\inmodule QtCore
Definition qurl.h:94
static QUrl fromLocalFile(const QString &localfile)
Returns a QUrl representation of localFile, interpreted as a local file.
Definition qurl.cpp:3368
Combined button and popup list for selecting options.
#define qCWarning(category,...)
#define qCDebug(category,...)
#define Q_STATIC_LOGGING_CATEGORY(name,...)
GLenum GLenum GLenum input
#define Q_ASSERT(cond)
Definition qrandom.cpp:47
QSettings settings("MySoft", "Star Runner")
[0]
myObject disconnect()
[26]