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#include <qimagecapture.h>
4#include <private/qplatformimagecapture_p.h>
5#include <qmediametadata.h>
6#include <private/qplatformmediacapture_p.h>
7#include <private/qplatformmediaintegration_p.h>
8#include <private/qplatformmediaformatinfo_p.h>
9#include <qmediacapturesession.h>
10
11#include "private/qobject_p.h"
12#include <qcamera.h>
13#include <private/qplatformcamera_p.h>
14#include <QtCore/qdebug.h>
15#include <QtCore/qurl.h>
16#include <QtCore/qstringlist.h>
17#include <QtCore/qmetaobject.h>
18
19QT_BEGIN_NAMESPACE
20
21/*!
22 \class QImageCapture
23 \inmodule QtMultimedia
24 \ingroup multimedia
25 \ingroup multimedia_camera
26
27
28 \brief The QImageCapture class is used for the recording of media content.
29
30 The QImageCapture class is a high level images recording class.
31 It's not intended to be used alone but for accessing the media
32 recording functions of other media objects, like QCamera.
33
34 \snippet multimedia-snippets/camera.cpp Camera
35
36 \snippet multimedia-snippets/camera.cpp Camera keys
37
38 \sa QCamera
39*/
40
41class QImageCapturePrivate
42{
43 Q_DECLARE_PUBLIC(QImageCapture)
44public:
45 QCamera *camera = nullptr;
46
47 QMediaCaptureSession *captureSession = nullptr;
48 QPlatformImageCapture *control = nullptr;
49
50 QImageCapture::Error error = QImageCapture::NoError;
51 QString errorString;
52 QMediaMetaData metaData;
53
54 void _q_error(int id, int error, const QString &errorString);
55
56 void unsetError() { error = QImageCapture::NoError; errorString.clear(); }
57
58 QImageCapture *q_ptr;
59};
60
61void QImageCapturePrivate::_q_error(int id, int error, const QString &errorString)
62{
63 Q_Q(QImageCapture);
64
65 this->error = QImageCapture::Error(error);
66 this->errorString = errorString;
67
68 emit q->errorChanged();
69 emit q->errorOccurred(id, error: this->error, errorString);
70}
71
72/*!
73 Constructs a image capture object, from a \a parent, that can capture
74 individual still images produced by a camera.
75
76 You must connect both an image capture object and a QCamera to a capture
77 session to capture images.
78*/
79
80QImageCapture::QImageCapture(QObject *parent)
81 : QObject(parent), d_ptr(new QImageCapturePrivate)
82{
83 Q_D(QImageCapture);
84 d->q_ptr = this;
85
86 auto maybeControl = QPlatformMediaIntegration::instance()->createImageCapture(this);
87 if (!maybeControl) {
88 qWarning() << "Failed to initialize QImageCapture" << maybeControl.error();
89 d->errorString = maybeControl.error();
90 d->error = NotReadyError;
91 return;
92 }
93
94 d->control = maybeControl.value();
95 connect(sender: d->control, signal: &QPlatformImageCapture::imageExposed, context: this, slot: &QImageCapture::imageExposed);
96 connect(sender: d->control, signal: &QPlatformImageCapture::imageCaptured, context: this, slot: &QImageCapture::imageCaptured);
97 connect(sender: d->control, signal: &QPlatformImageCapture::imageMetadataAvailable, context: this,
98 slot: &QImageCapture::imageMetadataAvailable);
99 connect(sender: d->control, signal: &QPlatformImageCapture::imageAvailable, context: this,
100 slot: &QImageCapture::imageAvailable);
101 connect(sender: d->control, signal: &QPlatformImageCapture::imageSaved, context: this, slot: &QImageCapture::imageSaved);
102 connect(sender: d->control, signal: &QPlatformImageCapture::readyForCaptureChanged, context: this,
103 slot: &QImageCapture::readyForCaptureChanged);
104 connect(sender: d->control, SIGNAL(error(int,int,QString)),
105 receiver: this, SLOT(_q_error(int,int,QString)));
106}
107
108/*!
109 \fn void QImageCapture::imageMetadataAvailable(int id, const QMediaMetaData &metaData)
110
111 Signals that an image identified by \a id has \a metaData.
112*/
113
114/*!
115 \internal
116*/
117void QImageCapture::setCaptureSession(QMediaCaptureSession *session)
118{
119 Q_D(QImageCapture);
120 d->captureSession = session;
121}
122
123/*!
124 Destroys images capture object.
125*/
126
127QImageCapture::~QImageCapture()
128{
129 if (d_ptr->captureSession)
130 d_ptr->captureSession->setImageCapture(nullptr);
131 delete d_ptr;
132}
133
134/*!
135 Returns true if the images capture service ready to use.
136*/
137bool QImageCapture::isAvailable() const
138{
139 return d_func()->control && d_func()->captureSession && d_func()->captureSession->camera();
140}
141
142/*!
143 Returns the capture session this camera is connected to, or
144 a nullptr if the camera is not connected to a capture session.
145
146 Use QMediaCaptureSession::setImageCapture() to connect the image capture to
147 a session.
148*/
149QMediaCaptureSession *QImageCapture::captureSession() const
150{
151 return d_ptr->captureSession;
152}
153
154/*!
155 \property QImageCapture::error
156
157 Returns the current error state.
158
159 \sa errorString()
160*/
161
162QImageCapture::Error QImageCapture::error() const
163{
164 return d_func()->error;
165}
166
167/*!
168 \property QImageCapture::errorString
169
170 Returns a string describing the current error state.
171
172 \sa error()
173*/
174
175QString QImageCapture::errorString() const
176{
177 return d_func()->errorString;
178}
179
180/*!
181 \property QImageCapture::metaData
182 \brief The meta data that will get embedded into the image.
183
184 \note Additional fields such as a time stamp or location may get added by
185 the camera back end.
186*/
187QMediaMetaData QImageCapture::metaData() const
188{
189 Q_D(const QImageCapture);
190 return d->metaData;
191}
192
193/*!
194 Replaces any existing meta data, to be embedded into the captured image,
195 with a set of \a metaData.
196*/
197void QImageCapture::setMetaData(const QMediaMetaData &metaData)
198{
199 Q_D(QImageCapture);
200 d->metaData = metaData;
201 if (d->control)
202 d->control->setMetaData(d->metaData);
203 emit metaDataChanged();
204}
205
206/*!
207 Adds additional \a metaData to any existing meta data, that is embedded
208 into the captured image.
209*/
210void QImageCapture::addMetaData(const QMediaMetaData &metaData)
211{
212 Q_D(QImageCapture);
213 auto data = d->metaData;
214 for (auto &&[key, value] : metaData.asKeyValueRange())
215 data.insert(k: key, value);
216 setMetaData(data);
217}
218
219/*!
220 \property QImageCapture::readyForCapture
221
222 Holds \c true if the camera is ready to capture an image immediately.
223 Calling capture() while \c readyForCapture is \c false is not
224 permitted and results in an error.
225*/
226bool QImageCapture::isReadyForCapture() const
227{
228 Q_D(const QImageCapture);
229 if (!d->control || !d->captureSession || !d->control->isReadyForCapture())
230 return false;
231 auto *camera = d->captureSession->camera();
232 if (!camera || !camera->isActive())
233 return false;
234 return true;
235}
236
237/*!
238 \fn QImageCapture::readyForCaptureChanged(bool ready)
239
240 Signals that a camera's \a ready for capture state has changed.
241*/
242
243
244/*!
245 Capture the image and save it to \a file.
246 This operation is asynchronous in majority of cases,
247 followed by signals QImageCapture::imageExposed(),
248 QImageCapture::imageCaptured(), QImageCapture::imageSaved()
249 or QImageCapture::error().
250
251 If an empty \a file is passed, the camera back end chooses
252 the default location and naming scheme for photos on the system,
253 if only file name without full path is specified, the image will be saved to
254 the default directory, with a full path reported with imageCaptured() and imageSaved() signals.
255
256 QCamera saves all the capture parameters like exposure settings or
257 image processing parameters, so changes to camera parameters after
258 capture() is called do not affect previous capture requests.
259
260 QImageCapture::capture returns the capture Id parameter, used with
261 imageExposed(), imageCaptured() and imageSaved() signals.
262
263 \sa isReadyForCapture()
264*/
265int QImageCapture::captureToFile(const QString &file)
266{
267 Q_D(QImageCapture);
268 if (!d->control) {
269 d->_q_error(id: -1, error: d->error, errorString: d->errorString);
270 return -1;
271 }
272
273 d->unsetError();
274
275 if (!isReadyForCapture()) {
276 d->_q_error(id: -1, error: NotReadyError, errorString: tr(s: "Could not capture in stopped state"));
277 return -1;
278 }
279
280 return d->control->capture(fileName: file);
281}
282
283/*!
284 Capture the image and make it available as a QImage.
285 This operation is asynchronous in majority of cases,
286 followed by signals QImageCapture::imageExposed(),
287 QImageCapture::imageCaptured()
288 or QImageCapture::error().
289
290 QImageCapture::capture returns the capture Id parameter, used with
291 imageExposed(), imageCaptured() and imageSaved() signals.
292
293 \sa isReadyForCapture()
294*/
295int QImageCapture::capture()
296{
297 Q_D(QImageCapture);
298 if (!d->control) {
299 d->_q_error(id: -1, error: d->error, errorString: d->errorString);
300 return -1;
301 } else {
302 d->unsetError();
303 return d->control->captureToBuffer();
304 }
305}
306
307/*!
308 \enum QImageCapture::Error
309
310 \value NoError No Errors.
311 \value NotReadyError The service is not ready for capture yet.
312 \value ResourceError Device is not ready or not available.
313 \value OutOfSpaceError No space left on device.
314 \value NotSupportedFeatureError Device does not support stillimages capture.
315 \value FormatError Current format is not supported.
316*/
317
318/*!
319 \fn QImageCapture::errorOccurred(int id, QImageCapture::Error error, const QString &errorString);
320
321 Signals that the capture request \a id has failed with an \a error
322 and \a errorString description.
323*/
324
325/*!
326 \fn QImageCapture::imageExposed(int id)
327
328 Signal emitted when the frame with request \a id was exposed.
329*/
330
331/*!
332 \fn QImageCapture::imageCaptured(int id, const QImage &preview);
333
334 Signal emitted when the frame with request \a id was captured, but not
335 processed and saved yet. Frame \a preview can be displayed to user.
336*/
337
338/*!
339 \fn QImageCapture::imageAvailable(int id, const QVideoFrame &frame)
340
341 Signal emitted when the \a frame with request \a id is available.
342*/
343
344/*!
345 \fn QImageCapture::imageSaved(int id, const QString &fileName)
346
347 Signal emitted when QImageCapture::CaptureToFile is set and
348 the frame with request \a id was saved to \a fileName.
349*/
350
351/*!
352 \enum QImageCapture::FileFormat
353
354 Choose one of the following image formats:
355
356 \value UnspecifiedFormat No format specified
357 \value JPEG \c .jpg or \c .jpeg format
358 \value PNG \c .png format
359 \value WebP \c .webp format
360 \value Tiff \c .tiff format
361 \omitvalue LastFileFormat
362*/
363
364
365/*!
366 \property QImageCapture::fileFormat
367 \brief The image format.
368*/
369
370QImageCapture::FileFormat QImageCapture::fileFormat() const
371{
372 Q_D(const QImageCapture);
373 return d->control ? d->control->imageSettings().format() : UnspecifiedFormat;
374}
375
376/*!
377 Sets the image \a format.
378*/
379void QImageCapture::setFileFormat(QImageCapture::FileFormat format)
380{
381 Q_D(QImageCapture);
382 if (!d->control)
383 return;
384 auto fmt = d->control->imageSettings();
385 if (fmt.format() == format)
386 return;
387 fmt.setFormat(format);
388 d->control->setImageSettings(fmt);
389 emit fileFormatChanged();
390}
391
392/*!
393 Returns a list of supported file formats.
394
395 \sa {QImageCapture::}{FileFormat}
396*/
397QList<QImageCapture::FileFormat> QImageCapture::supportedFormats()
398{
399 return QPlatformMediaIntegration::instance()->formatInfo()->imageFormats;
400}
401
402/*!
403 Returns the name of the given format, \a f.
404*/
405QString QImageCapture::fileFormatName(QImageCapture::FileFormat f)
406{
407 const char *name = nullptr;
408 switch (f) {
409 case UnspecifiedFormat:
410 name = "Unspecified image format";
411 break;
412 case JPEG:
413 name = "JPEG";
414 break;
415 case PNG:
416 name = "PNG";
417 break;
418 case WebP:
419 name = "WebP";
420 break;
421 case Tiff:
422 name = "Tiff";
423 break;
424 }
425 return QString::fromUtf8(utf8: name);
426}
427
428/*!
429 Returns the description of the given file format, \a f.
430*/
431QString QImageCapture::fileFormatDescription(QImageCapture::FileFormat f)
432{
433 const char *name = nullptr;
434 switch (f) {
435 case UnspecifiedFormat:
436 name = "Unspecified image format";
437 break;
438 case JPEG:
439 name = "JPEG";
440 break;
441 case PNG:
442 name = "PNG";
443 break;
444 case WebP:
445 name = "WebP";
446 break;
447 case Tiff:
448 name = "Tiff";
449 break;
450 }
451 return QString::fromUtf8(utf8: name);
452}
453
454/*!
455 Returns the resolution of the encoded image.
456*/
457
458QSize QImageCapture::resolution() const
459{
460 Q_D(const QImageCapture);
461 return d->control ? d->control->imageSettings().resolution() : QSize{};
462}
463
464/*!
465 \fn void QImageCapture::resolutionChanged()
466
467 Signals when the image resolution changes.
468*/
469
470/*!
471 Sets the \a resolution of the encoded image.
472
473 An empty QSize indicates the encoder should make an optimal choice based on
474 what is available from the image source and the limitations of the codec.
475*/
476void QImageCapture::setResolution(const QSize &resolution)
477{
478 Q_D(QImageCapture);
479 if (!d->control)
480 return;
481 auto fmt = d->control->imageSettings();
482 if (fmt.resolution() == resolution)
483 return;
484 fmt.setResolution(resolution);
485 d->control->setImageSettings(fmt);
486 emit resolutionChanged();
487}
488
489/*!
490 Sets the \a width and \a height of the resolution of the encoded image.
491
492 \overload
493*/
494void QImageCapture::setResolution(int width, int height)
495{
496 setResolution(QSize(width, height));
497}
498
499/*!
500 \enum QImageCapture::Quality
501
502 Enumerates quality encoding levels.
503
504 \value VeryLowQuality
505 \value LowQuality
506 \value NormalQuality
507 \value HighQuality
508 \value VeryHighQuality
509*/
510
511/*!
512 \property QImageCapture::quality
513 \brief The image encoding quality.
514*/
515QImageCapture::Quality QImageCapture::quality() const
516{
517 Q_D(const QImageCapture);
518 return d->control ? d->control->imageSettings().quality() : NormalQuality;
519}
520
521/*!
522 Sets the image encoding \a quality.
523*/
524void QImageCapture::setQuality(Quality quality)
525{
526 Q_D(QImageCapture);
527 if (!d->control)
528 return;
529 auto fmt = d->control->imageSettings();
530 if (fmt.quality() == quality)
531 return;
532 fmt.setQuality(quality);
533 d->control->setImageSettings(fmt);
534 emit resolutionChanged();
535}
536
537/*!
538 \internal
539*/
540QPlatformImageCapture *QImageCapture::platformImageCapture()
541{
542 Q_D(QImageCapture);
543 return d->control;
544}
545
546QT_END_NAMESPACE
547
548#include "moc_qimagecapture.cpp"
549

Provided by KDAB

Privacy Policy
Learn Advanced QML with KDAB
Find out more

source code of qtmultimedia/src/multimedia/camera/qimagecapture.cpp