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 | |
19 | QT_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 | |
41 | class QImageCapturePrivate |
42 | { |
43 | Q_DECLARE_PUBLIC(QImageCapture) |
44 | public: |
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 | |
61 | void 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 | |
80 | QImageCapture::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 | */ |
117 | void QImageCapture::setCaptureSession(QMediaCaptureSession *session) |
118 | { |
119 | Q_D(QImageCapture); |
120 | d->captureSession = session; |
121 | } |
122 | |
123 | /*! |
124 | Destroys images capture object. |
125 | */ |
126 | |
127 | QImageCapture::~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 | */ |
137 | bool 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 | */ |
149 | QMediaCaptureSession *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 | |
162 | QImageCapture::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 | |
175 | QString 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 | */ |
187 | QMediaMetaData 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 | */ |
197 | void 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 | */ |
210 | void 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 | */ |
226 | bool 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 | */ |
265 | int 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 | */ |
295 | int 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 | |
370 | QImageCapture::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 | */ |
379 | void 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 | */ |
397 | QList<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 | */ |
405 | QString 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 | */ |
431 | QString 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 | |
458 | QSize 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 | */ |
476 | void 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 | */ |
494 | void 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 | */ |
515 | QImageCapture::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 | */ |
524 | void 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 | */ |
540 | QPlatformImageCapture *QImageCapture::platformImageCapture() |
541 | { |
542 | Q_D(QImageCapture); |
543 | return d->control; |
544 | } |
545 | |
546 | QT_END_NAMESPACE |
547 | |
548 | #include "moc_qimagecapture.cpp" |
549 |
Definitions
- QImageCapturePrivate
- unsetError
- _q_error
- QImageCapture
- setCaptureSession
- ~QImageCapture
- isAvailable
- captureSession
- error
- errorString
- metaData
- setMetaData
- addMetaData
- isReadyForCapture
- captureToFile
- capture
- fileFormat
- setFileFormat
- supportedFormats
- fileFormatName
- fileFormatDescription
- resolution
- setResolution
- setResolution
- quality
- setQuality
Learn Advanced QML with KDAB
Find out more