11#include "microexif_p.h"
13#include <libheif/heif.h>
22#ifndef HEIF_MAX_METADATA_SIZE
26#define HEIF_MAX_METADATA_SIZE (4 * 1024 * 1024)
29size_t HEIFHandler::m_initialized_count = 0;
30bool HEIFHandler::m_plugins_queried =
false;
31bool HEIFHandler::m_heif_decoder_available =
false;
32bool HEIFHandler::m_heif_encoder_available =
false;
33bool HEIFHandler::m_hej2_decoder_available =
false;
34bool HEIFHandler::m_hej2_encoder_available =
false;
35bool HEIFHandler::m_avci_decoder_available =
false;
38static struct heif_error heifhandler_write_callback(struct heif_context * ,
const void *data,
size_t size,
void *userdata)
41 error.code = heif_error_Ok;
42 error.subcode = heif_suberror_Unspecified;
43 error.message =
"Success";
45 if (!userdata || !data || size == 0) {
46 error.code = heif_error_Usage_error;
47 error.subcode = heif_suberror_Null_pointer_argument;
48 error.message =
"Wrong parameters!";
53 qint64 bytesWritten = ioDevice->
write(
static_cast<const char *
>(data), size);
55 if (bytesWritten <
static_cast<qint64
>(size)) {
56 error.code = heif_error_Encoding_error;
57 error.message =
"Bytes written to QIODevice are smaller than input data size";
58 error.subcode = heif_suberror_Cannot_write_output_data;
65HEIFHandler::HEIFHandler()
66 : m_parseState(ParseHeicNotParsed)
71bool HEIFHandler::canRead()
const
73 if (m_parseState == ParseHeicNotParsed) {
78 if (HEIFHandler::isSupportedBMFFType(header)) {
83 if (HEIFHandler::isSupportedHEJ2(header)) {
88 if (HEIFHandler::isSupportedAVCI(header)) {
96 if (m_parseState != ParseHeicError) {
102bool HEIFHandler::read(
QImage *outImage)
104 if (!ensureParsed()) {
108 *outImage = m_current_image;
112bool HEIFHandler::write(
const QImage &image)
115 qWarning(
"No image data to save");
119#if LIBHEIF_HAVE_VERSION(1, 13, 0)
123 bool success = write_helper(image);
125#if LIBHEIF_HAVE_VERSION(1, 13, 0)
132bool HEIFHandler::write_helper(
const QImage &image)
150 if (image.
depth() > 32) {
158 heif_compression_format encoder_codec = heif_compression_HEVC;
159#if LIBHEIF_HAVE_VERSION(1, 13, 0)
160 if (format() ==
"hej2") {
161 encoder_codec = heif_compression_JPEG2000;
167 if (save_depth > 8) {
178 chroma = heif_chroma_interleaved_RGBA;
181 chroma = heif_chroma_interleaved_RGB;
185#if QT_VERSION >= QT_VERSION_CHECK(6, 8, 0)
188 if (cs.isValid() && cs.colorModel() == QColorSpace::ColorModel::Cmyk && image.
format() == QImage::Format_CMYK8888) {
190 }
else if (cs.isValid() && cs.colorModel() == QColorSpace::ColorModel::Gray
193 float gamma_new = cs.gamma();
194 if (trc_new == QColorSpace::TransferFunction::Custom) {
195 trc_new = QColorSpace::TransferFunction::SRgb;
205 struct heif_context *context = heif_context_alloc();
206 struct heif_error err;
207 struct heif_image *h_image =
nullptr;
209 err = heif_image_create(tmpimage.
width(), tmpimage.
height(), heif_colorspace_RGB, chroma, &h_image);
211 qWarning() <<
"heif_image_create error:" << err.message;
212 heif_context_free(context);
217 if (iccprofile.
size() > 0) {
218 heif_image_set_raw_color_profile(h_image,
"prof", iccprofile.
constData(), iccprofile.
size());
221 heif_image_add_plane(h_image, heif_channel_interleaved, image.
width(), image.
height(), save_depth);
223 uint8_t *
const dst = heif_image_get_plane(h_image, heif_channel_interleaved, &stride);
226 switch (save_depth) {
229 for (
int y = 0; y < tmpimage.
height(); y++) {
230 const uint16_t *src_word =
reinterpret_cast<const uint16_t *
>(tmpimage.
constScanLine(y));
231 uint16_t *dest_word =
reinterpret_cast<uint16_t *
>(dst + (y * stride));
232 for (
int x = 0; x < tmpimage.
width(); x++) {
235 tmp_pixelval = (int)(((
float)(*src_word) / 65535.0f) * 1023.0f + 0.5f);
236 *dest_word = qBound(0, tmp_pixelval, 1023);
240 tmp_pixelval = (int)(((
float)(*src_word) / 65535.0f) * 1023.0f + 0.5f);
241 *dest_word = qBound(0, tmp_pixelval, 1023);
245 tmp_pixelval = (int)(((
float)(*src_word) / 65535.0f) * 1023.0f + 0.5f);
246 *dest_word = qBound(0, tmp_pixelval, 1023);
250 tmp_pixelval = (int)(((
float)(*src_word) / 65535.0f) * 1023.0f + 0.5f);
251 *dest_word = qBound(0, tmp_pixelval, 1023);
257 for (
int y = 0; y < tmpimage.
height(); y++) {
258 const uint16_t *src_word =
reinterpret_cast<const uint16_t *
>(tmpimage.
constScanLine(y));
259 uint16_t *dest_word =
reinterpret_cast<uint16_t *
>(dst + (y * stride));
260 for (
int x = 0; x < tmpimage.
width(); x++) {
263 tmp_pixelval = (int)(((
float)(*src_word) / 65535.0f) * 1023.0f + 0.5f);
264 *dest_word = qBound(0, tmp_pixelval, 1023);
268 tmp_pixelval = (int)(((
float)(*src_word) / 65535.0f) * 1023.0f + 0.5f);
269 *dest_word = qBound(0, tmp_pixelval, 1023);
273 tmp_pixelval = (int)(((
float)(*src_word) / 65535.0f) * 1023.0f + 0.5f);
274 *dest_word = qBound(0, tmp_pixelval, 1023);
284 rowbytes = save_alpha ? (tmpimage.
width() * 4) : (tmpimage.width() * 3);
285 for (
int y = 0; y < tmpimage.
height(); y++) {
286 memcpy(dst + (y * stride), tmpimage.
constScanLine(y), rowbytes);
290 qWarning() <<
"Unsupported depth:" << save_depth;
291 heif_image_release(h_image);
292 heif_context_free(context);
297 struct heif_encoder *encoder =
nullptr;
298 err = heif_context_get_encoder_for_format(context, encoder_codec, &encoder);
300 qWarning() <<
"Unable to get an encoder instance:" << err.message;
301 heif_image_release(h_image);
302 heif_context_free(context);
306 heif_encoder_set_lossy_quality(encoder, m_quality);
307 if (m_quality > 90) {
308 if (m_quality == 100) {
309 heif_encoder_set_lossless(encoder,
true);
311 heif_encoder_set_parameter_string(encoder,
"chroma",
"444");
314 struct heif_encoding_options *encoder_options = heif_encoding_options_alloc();
315 encoder_options->save_alpha_channel = save_alpha;
317 if ((tmpimage.
width() % 2 == 1) || (tmpimage.
height() % 2 == 1)) {
318 qWarning() <<
"Image has odd dimension!\nUse even-numbered dimension(s) for better compatibility with other HEIF implementations.";
321 encoder_options->macOS_compatibility_workaround = 0;
325 struct heif_image_handle *handle;
326 err = heif_context_encode_image(context, h_image, encoder, encoder_options, &handle);
329 if (err.code == heif_error_Ok) {
330 auto exif = MicroExif::fromImage(tmpimage);
331 if (!exif.isEmpty()) {
332 auto ba = exif.toByteArray();
333 err = heif_context_add_exif_metadata(context, handle, ba.constData(), ba.size());
337 if (err.code == heif_error_Ok) {
338 auto xmp = image.
text(QStringLiteral(META_KEY_XMP_ADOBE));
339 if (!xmp.isEmpty()) {
340 auto ba = xmp.toUtf8();
341 err = heif_context_add_XMP_metadata(context, handle, ba.constData(), ba.size());
345 if (encoder_options) {
346 heif_encoding_options_free(encoder_options);
350 qWarning() <<
"heif_context_encode_image failed:" << err.message;
351 heif_encoder_release(encoder);
352 heif_image_release(h_image);
353 heif_context_free(context);
357 struct heif_writer writer;
358 writer.writer_api_version = 1;
359 writer.write = heifhandler_write_callback;
361 err = heif_context_write(context, &writer, device());
363 heif_encoder_release(encoder);
364 heif_image_release(h_image);
367 qWarning() <<
"Writing HEIF image failed:" << err.message;
368 heif_context_free(context);
372 heif_context_free(context);
376bool HEIFHandler::isSupportedBMFFType(
const QByteArray &header)
378 if (header.
size() < 28) {
383 if (memcmp(buffer + 4,
"ftyp", 4) == 0) {
384 if (memcmp(buffer + 8,
"heic", 4) == 0) {
387 if (memcmp(buffer + 8,
"heis", 4) == 0) {
390 if (memcmp(buffer + 8,
"heix", 4) == 0) {
395 if (memcmp(buffer + 8,
"mif1", 4) == 0) {
396 for (
int offset = 16; offset <= 24; offset += 4) {
397 if (memcmp(buffer + offset,
"avif", 4) == 0) {
404 if (memcmp(buffer + 8,
"mif2", 4) == 0) {
407 if (memcmp(buffer + 8,
"msf1", 4) == 0) {
415bool HEIFHandler::isSupportedHEJ2(
const QByteArray &header)
417 if (header.
size() < 28) {
422 if (memcmp(buffer + 4,
"ftypj2ki", 8) == 0) {
429bool HEIFHandler::isSupportedAVCI(
const QByteArray &header)
431 if (header.
size() < 28) {
436 if (memcmp(buffer + 4,
"ftypavci", 8) == 0) {
443QVariant HEIFHandler::option(ImageOption option)
const
445 if (option == Quality) {
449 if (!supportsOption(option) || !ensureParsed()) {
455 return m_current_image.size();
463void HEIFHandler::setOption(ImageOption option,
const QVariant &value)
467 m_quality = value.
toInt();
468 if (m_quality > 100) {
470 }
else if (m_quality < 0) {
480bool HEIFHandler::supportsOption(ImageOption option)
const
482 return option == Quality || option == Size;
485bool HEIFHandler::ensureParsed()
const
487 if (m_parseState == ParseHeicSuccess) {
490 if (m_parseState == ParseHeicError) {
494 HEIFHandler *that =
const_cast<HEIFHandler *
>(
this);
496#if LIBHEIF_HAVE_VERSION(1, 13, 0)
500 bool success = that->ensureDecoder();
502#if LIBHEIF_HAVE_VERSION(1, 13, 0)
508bool HEIFHandler::ensureDecoder()
510 if (m_parseState != ParseHeicNotParsed) {
511 if (m_parseState == ParseHeicSuccess) {
517 const QByteArray buffer = device()->readAll();
518 if (!HEIFHandler::isSupportedBMFFType(buffer) && !HEIFHandler::isSupportedHEJ2(buffer) && !HEIFHandler::isSupportedAVCI(buffer)) {
519 m_parseState = ParseHeicError;
523 struct heif_context *ctx = heif_context_alloc();
524 struct heif_error err = heif_context_read_from_memory(ctx,
static_cast<const void *
>(buffer.
constData()), buffer.
size(),
nullptr);
527 qWarning() <<
"heif_context_read_from_memory error:" << err.message;
528 heif_context_free(ctx);
529 m_parseState = ParseHeicError;
533 struct heif_image_handle *handle =
nullptr;
534 err = heif_context_get_primary_image_handle(ctx, &handle);
536 qWarning() <<
"heif_context_get_primary_image_handle error:" << err.message;
537 heif_context_free(ctx);
538 m_parseState = ParseHeicError;
542 if ((heif_image_handle_get_width(handle) == 0) || (heif_image_handle_get_height(handle) == 0)) {
543 m_parseState = ParseHeicError;
544 heif_image_handle_release(handle);
545 heif_context_free(ctx);
546 qWarning() <<
"HEIC image has zero dimension";
550 const int bit_depth = heif_image_handle_get_luma_bits_per_pixel(handle);
553 m_parseState = ParseHeicError;
554 heif_image_handle_release(handle);
555 heif_context_free(ctx);
556 qWarning() <<
"HEIF image with undefined or unsupported bit depth.";
560 const bool hasAlphaChannel = heif_image_handle_has_alpha_channel(handle);
565 if (bit_depth == 10 || bit_depth == 12 || bit_depth == 16) {
566 if (hasAlphaChannel) {
573 }
else if (bit_depth == 8) {
574 if (hasAlphaChannel) {
575 chroma = heif_chroma_interleaved_RGBA;
578 chroma = heif_chroma_interleaved_RGB;
582 m_parseState = ParseHeicError;
583 heif_image_handle_release(handle);
584 heif_context_free(ctx);
585 qWarning() <<
"Unsupported bit depth:" << bit_depth;
589 struct heif_decoding_options *decoder_option = heif_decoding_options_alloc();
591#if LIBHEIF_HAVE_VERSION(1, 13, 0)
592 decoder_option->strict_decoding = 1;
595 struct heif_image *img =
nullptr;
596 err = heif_decode_image(handle, &img, heif_colorspace_RGB, chroma, decoder_option);
598#if LIBHEIF_HAVE_VERSION(1, 13, 0)
599 if (err.code == heif_error_Invalid_input && err.subcode == heif_suberror_Unknown_NCLX_matrix_coefficients && img ==
nullptr && buffer.
contains(
"Xiaomi")) {
600 qWarning() <<
"Non-standard HEIF image with invalid matrix_coefficients, probably made by a Xiaomi device!";
603 decoder_option->strict_decoding = 0;
604 err = heif_decode_image(handle, &img, heif_colorspace_RGB, chroma, decoder_option);
608 if (decoder_option) {
609 heif_decoding_options_free(decoder_option);
613 qWarning() <<
"heif_decode_image error:" << err.message;
614 heif_image_handle_release(handle);
615 heif_context_free(ctx);
616 m_parseState = ParseHeicError;
620 const int imageWidth = heif_image_get_width(img, heif_channel_interleaved);
621 const int imageHeight = heif_image_get_height(img, heif_channel_interleaved);
623 QSize imageSize(imageWidth, imageHeight);
625 if (!imageSize.isValid()) {
626 heif_image_release(img);
627 heif_image_handle_release(handle);
628 heif_context_free(ctx);
629 m_parseState = ParseHeicError;
630 qWarning() <<
"HEIC image size invalid:" << imageSize;
635 const uint8_t *
const src = heif_image_get_plane_readonly(img, heif_channel_interleaved, &stride);
637 if (!src || stride <= 0) {
638 heif_image_release(img);
639 heif_image_handle_release(handle);
640 heif_context_free(ctx);
641 m_parseState = ParseHeicError;
642 qWarning() <<
"HEIC data pixels information not valid!";
646 m_current_image = imageAlloc(imageSize, target_image_format);
647 if (m_current_image.isNull()) {
648 heif_image_release(img);
649 heif_image_handle_release(handle);
650 heif_context_free(ctx);
651 m_parseState = ParseHeicError;
652 qWarning() <<
"Unable to allocate memory!";
658 if (hasAlphaChannel) {
659 for (
int y = 0; y < imageHeight; y++) {
660 memcpy(m_current_image.scanLine(y), src + (y * stride), 8 *
size_t(imageWidth));
663 for (
int y = 0; y < imageHeight; y++) {
664 const uint16_t *src_word =
reinterpret_cast<const uint16_t *
>(src + (y * stride));
665 uint16_t *dest_data =
reinterpret_cast<uint16_t *
>(m_current_image.scanLine(y));
666 for (
int x = 0; x < imageWidth; x++) {
668 *dest_data = *src_word;
672 *dest_data = *src_word;
676 *dest_data = *src_word;
687 if (hasAlphaChannel) {
688 for (
int y = 0; y < imageHeight; y++) {
689 const uint16_t *src_word =
reinterpret_cast<const uint16_t *
>(src + (y * stride));
690 uint16_t *dest_data =
reinterpret_cast<uint16_t *
>(m_current_image.scanLine(y));
691 for (
int x = 0; x < imageWidth; x++) {
694 tmpvalue = (int)(((
float)(0x0fff & (*src_word)) / 4095.0f) * 65535.0f + 0.5f);
695 tmpvalue = qBound(0, tmpvalue, 65535);
696 *dest_data = (uint16_t)tmpvalue;
700 tmpvalue = (int)(((
float)(0x0fff & (*src_word)) / 4095.0f) * 65535.0f + 0.5f);
701 tmpvalue = qBound(0, tmpvalue, 65535);
702 *dest_data = (uint16_t)tmpvalue;
706 tmpvalue = (int)(((
float)(0x0fff & (*src_word)) / 4095.0f) * 65535.0f + 0.5f);
707 tmpvalue = qBound(0, tmpvalue, 65535);
708 *dest_data = (uint16_t)tmpvalue;
712 tmpvalue = (int)(((
float)(0x0fff & (*src_word)) / 4095.0f) * 65535.0f + 0.5f);
713 tmpvalue = qBound(0, tmpvalue, 65535);
714 *dest_data = (uint16_t)tmpvalue;
720 for (
int y = 0; y < imageHeight; y++) {
721 const uint16_t *src_word =
reinterpret_cast<const uint16_t *
>(src + (y * stride));
722 uint16_t *dest_data =
reinterpret_cast<uint16_t *
>(m_current_image.scanLine(y));
723 for (
int x = 0; x < imageWidth; x++) {
726 tmpvalue = (int)(((
float)(0x0fff & (*src_word)) / 4095.0f) * 65535.0f + 0.5f);
727 tmpvalue = qBound(0, tmpvalue, 65535);
728 *dest_data = (uint16_t)tmpvalue;
732 tmpvalue = (int)(((
float)(0x0fff & (*src_word)) / 4095.0f) * 65535.0f + 0.5f);
733 tmpvalue = qBound(0, tmpvalue, 65535);
734 *dest_data = (uint16_t)tmpvalue;
738 tmpvalue = (int)(((
float)(0x0fff & (*src_word)) / 4095.0f) * 65535.0f + 0.5f);
739 tmpvalue = qBound(0, tmpvalue, 65535);
740 *dest_data = (uint16_t)tmpvalue;
751 if (hasAlphaChannel) {
752 for (
int y = 0; y < imageHeight; y++) {
753 const uint16_t *src_word =
reinterpret_cast<const uint16_t *
>(src + (y * stride));
754 uint16_t *dest_data =
reinterpret_cast<uint16_t *
>(m_current_image.scanLine(y));
755 for (
int x = 0; x < imageWidth; x++) {
758 tmpvalue = (int)(((
float)(0x03ff & (*src_word)) / 1023.0f) * 65535.0f + 0.5f);
759 tmpvalue = qBound(0, tmpvalue, 65535);
760 *dest_data = (uint16_t)tmpvalue;
764 tmpvalue = (int)(((
float)(0x03ff & (*src_word)) / 1023.0f) * 65535.0f + 0.5f);
765 tmpvalue = qBound(0, tmpvalue, 65535);
766 *dest_data = (uint16_t)tmpvalue;
770 tmpvalue = (int)(((
float)(0x03ff & (*src_word)) / 1023.0f) * 65535.0f + 0.5f);
771 tmpvalue = qBound(0, tmpvalue, 65535);
772 *dest_data = (uint16_t)tmpvalue;
776 tmpvalue = (int)(((
float)(0x03ff & (*src_word)) / 1023.0f) * 65535.0f + 0.5f);
777 tmpvalue = qBound(0, tmpvalue, 65535);
778 *dest_data = (uint16_t)tmpvalue;
784 for (
int y = 0; y < imageHeight; y++) {
785 const uint16_t *src_word =
reinterpret_cast<const uint16_t *
>(src + (y * stride));
786 uint16_t *dest_data =
reinterpret_cast<uint16_t *
>(m_current_image.scanLine(y));
787 for (
int x = 0; x < imageWidth; x++) {
790 tmpvalue = (int)(((
float)(0x03ff & (*src_word)) / 1023.0f) * 65535.0f + 0.5f);
791 tmpvalue = qBound(0, tmpvalue, 65535);
792 *dest_data = (uint16_t)tmpvalue;
796 tmpvalue = (int)(((
float)(0x03ff & (*src_word)) / 1023.0f) * 65535.0f + 0.5f);
797 tmpvalue = qBound(0, tmpvalue, 65535);
798 *dest_data = (uint16_t)tmpvalue;
802 tmpvalue = (int)(((
float)(0x03ff & (*src_word)) / 1023.0f) * 65535.0f + 0.5f);
803 tmpvalue = qBound(0, tmpvalue, 65535);
804 *dest_data = (uint16_t)tmpvalue;
815 if (hasAlphaChannel) {
816 for (
int y = 0; y < imageHeight; y++) {
817 const uint8_t *src_byte = src + (y * stride);
818 uint32_t *dest_pixel =
reinterpret_cast<uint32_t *
>(m_current_image.scanLine(y));
819 for (
int x = 0; x < imageWidth; x++) {
820 int red = *src_byte++;
821 int green = *src_byte++;
822 int blue = *src_byte++;
823 int alpha = *src_byte++;
824 *dest_pixel = qRgba(red, green, blue, alpha);
829 for (
int y = 0; y < imageHeight; y++) {
830 const uint8_t *src_byte = src + (y * stride);
831 uint32_t *dest_pixel =
reinterpret_cast<uint32_t *
>(m_current_image.scanLine(y));
832 for (
int x = 0; x < imageWidth; x++) {
833 int red = *src_byte++;
834 int green = *src_byte++;
835 int blue = *src_byte++;
836 *dest_pixel = qRgb(red, green, blue);
843 heif_image_release(img);
844 heif_image_handle_release(handle);
845 heif_context_free(ctx);
846 m_parseState = ParseHeicError;
847 qWarning() <<
"Unsupported bit depth:" << bit_depth;
852 heif_color_profile_type profileType = heif_image_handle_get_color_profile_type(handle);
853 if (profileType == heif_color_profile_type_prof || profileType == heif_color_profile_type_rICC) {
854 size_t rawProfileSize = heif_image_handle_get_raw_color_profile_size(handle);
855 if (rawProfileSize > 0 && rawProfileSize < std::numeric_limits<int>::max()) {
857 err = heif_image_handle_get_raw_color_profile(handle, ba.data());
859 qWarning() <<
"icc profile loading failed";
863 qWarning() <<
"HEIC image has Qt-unsupported or invalid ICC profile!";
865#if (QT_VERSION >= QT_VERSION_CHECK(6, 8, 0))
866 else if (colorspace.colorModel() == QColorSpace::ColorModel::Cmyk) {
867 qWarning(
"CMYK ICC profile is not expected for HEIF, discarding the ICCprofile!");
869 }
else if (colorspace.colorModel() == QColorSpace::ColorModel::Gray) {
870 if (hasAlphaChannel) {
871 QPointF gray_whitePoint = colorspace.whitePoint();
872 if (gray_whitePoint.
isNull()) {
873 gray_whitePoint =
QPointF(0.3127f, 0.329f);
876 const QPointF redP(0.64f, 0.33f);
877 const QPointF greenP(0.3f, 0.6f);
878 const QPointF blueP(0.15f, 0.06f);
881 float gamma_new = colorspace.
gamma();
882 if (trc_new == QColorSpace::TransferFunction::Custom) {
883 trc_new = QColorSpace::TransferFunction::SRgb;
885 colorspace =
QColorSpace(gray_whitePoint, redP, greenP, blueP, trc_new, gamma_new);
887 qWarning(
"HEIF plugin created invalid QColorSpace!");
894 m_current_image.setColorSpace(colorspace);
897 qWarning() <<
"icc profile is empty or above limits";
900 }
else if (profileType == heif_color_profile_type_nclx) {
901 struct heif_color_profile_nclx *nclx =
nullptr;
902 err = heif_image_handle_get_nclx_color_profile(handle, &nclx);
903 if (err.code || !nclx) {
904 qWarning() <<
"nclx profile loading failed";
906 const QPointF redPoint(nclx->color_primary_red_x, nclx->color_primary_red_y);
907 const QPointF greenPoint(nclx->color_primary_green_x, nclx->color_primary_green_y);
908 const QPointF bluePoint(nclx->color_primary_blue_x, nclx->color_primary_blue_y);
909 const QPointF whitePoint(nclx->color_primary_white_x, nclx->color_primary_white_y);
912 float q_trc_gamma = 0.0f;
914 switch (nclx->transfer_characteristics) {
916 q_trc = QColorSpace::TransferFunction::Gamma;
920 q_trc = QColorSpace::TransferFunction::Gamma;
924 q_trc = QColorSpace::TransferFunction::Linear;
928 q_trc = QColorSpace::TransferFunction::SRgb;
930#if (QT_VERSION >= QT_VERSION_CHECK(6, 8, 0))
932 q_trc = QColorSpace::TransferFunction::St2084;
935 q_trc = QColorSpace::TransferFunction::Hlg;
939 qWarning(
"CICP color_primaries: %d, transfer_characteristics: %d\nThe colorspace is unsupported by this plug-in yet.",
940 nclx->color_primaries,
941 nclx->transfer_characteristics);
942 q_trc = QColorSpace::TransferFunction::SRgb;
946 if (q_trc != QColorSpace::TransferFunction::Custom) {
947 switch (nclx->color_primaries) {
950 m_current_image.setColorSpace(
QColorSpace(QColorSpace::Primaries::SRgb, q_trc, q_trc_gamma));
953 m_current_image.setColorSpace(
QColorSpace(QColorSpace::Primaries::DciP3D65, q_trc, q_trc_gamma));
956 m_current_image.setColorSpace(
QColorSpace(whitePoint, redPoint, greenPoint, bluePoint, q_trc, q_trc_gamma));
960 heif_nclx_color_profile_free(nclx);
962 if (!m_current_image.colorSpace().isValid()) {
963 qWarning() <<
"HEIC plugin created invalid QColorSpace from NCLX!";
972 if (
auto numMetadata = heif_image_handle_get_number_of_metadata_blocks(handle,
nullptr)) {
974 heif_image_handle_get_list_of_metadata_block_IDs(handle,
nullptr, ids.data(), numMetadata);
975 for (
int n = 0; n < numMetadata; ++n) {
976 auto itemtype = heif_image_handle_get_metadata_type(handle, ids[n]);
977 auto contenttype = heif_image_handle_get_metadata_content_type(handle, ids[n]);
978 auto isExif = !std::strcmp(itemtype,
"Exif");
979 auto isXmp = !std::strcmp(contenttype,
"application/rdf+xml");
980 if (isExif || isXmp) {
981 auto sz = heif_image_handle_get_metadata_size(handle, ids[n]);
982 if (sz == 0 || sz >= HEIF_MAX_METADATA_SIZE)
985 auto err = heif_image_handle_get_metadata(handle, ids[n], ba.data());
986 if (err.code != heif_error_Ok) {
987 qWarning() <<
"Error while reading metadata" << err.message;
991 m_current_image.setText(QStringLiteral(META_KEY_XMP_ADOBE),
QString::fromUtf8(ba));
993 auto exif = MicroExif::fromByteArray(ba,
true);
994 if (!exif.isEmpty()) {
995 exif.updateImageResolution(m_current_image);
996 exif.updateImageMetadata(m_current_image);
1003 heif_image_release(img);
1004 heif_image_handle_release(handle);
1005 heif_context_free(ctx);
1006 m_parseState = ParseHeicSuccess;
1010bool HEIFHandler::isHeifDecoderAvailable()
1012 HEIFHandler::queryHeifLib();
1014 return m_heif_decoder_available;
1017bool HEIFHandler::isHeifEncoderAvailable()
1019 HEIFHandler::queryHeifLib();
1021 return m_heif_encoder_available;
1024bool HEIFHandler::isHej2DecoderAvailable()
1026 HEIFHandler::queryHeifLib();
1028 return m_hej2_decoder_available;
1031bool HEIFHandler::isHej2EncoderAvailable()
1033 HEIFHandler::queryHeifLib();
1035 return m_hej2_encoder_available;
1038bool HEIFHandler::isAVCIDecoderAvailable()
1040 HEIFHandler::queryHeifLib();
1042 return m_avci_decoder_available;
1045void HEIFHandler::queryHeifLib()
1049 if (!m_plugins_queried) {
1050#if LIBHEIF_HAVE_VERSION(1, 13, 0)
1051 if (m_initialized_count == 0) {
1056 m_heif_encoder_available = heif_have_encoder_for_format(heif_compression_HEVC);
1057 m_heif_decoder_available = heif_have_decoder_for_format(heif_compression_HEVC);
1058#if LIBHEIF_HAVE_VERSION(1, 13, 0)
1059 m_hej2_decoder_available = heif_have_decoder_for_format(heif_compression_JPEG2000);
1060 m_hej2_encoder_available = heif_have_encoder_for_format(heif_compression_JPEG2000);
1062#if LIBHEIF_HAVE_VERSION(1, 19, 6)
1063 m_avci_decoder_available = heif_have_decoder_for_format(heif_compression_AVC);
1065 m_plugins_queried =
true;
1067#if LIBHEIF_HAVE_VERSION(1, 13, 0)
1068 if (m_initialized_count == 0) {
1075void HEIFHandler::startHeifLib()
1077#if LIBHEIF_HAVE_VERSION(1, 13, 0)
1080 if (m_initialized_count == 0) {
1084 m_initialized_count++;
1088void HEIFHandler::finishHeifLib()
1090#if LIBHEIF_HAVE_VERSION(1, 13, 0)
1093 if (m_initialized_count == 0) {
1097 m_initialized_count--;
1098 if (m_initialized_count == 0) {
1105QMutex &HEIFHandler::getHEIFHandlerMutex()
1107 static QMutex heif_handler_mutex;
1108 return heif_handler_mutex;
1113 if (format ==
"heif" || format ==
"heic") {
1115 if (HEIFHandler::isHeifDecoderAvailable()) {
1116 format_cap |= CanRead;
1118 if (HEIFHandler::isHeifEncoderAvailable()) {
1119 format_cap |= CanWrite;
1124 if (format ==
"hej2") {
1126 if (HEIFHandler::isHej2DecoderAvailable()) {
1127 format_cap |= CanRead;
1129 if (HEIFHandler::isHej2EncoderAvailable()) {
1130 format_cap |= CanWrite;
1135 if (format ==
"avci") {
1137 if (HEIFHandler::isAVCIDecoderAvailable()) {
1138 format_cap |= CanRead;
1154 if ((HEIFHandler::isSupportedBMFFType(header) && HEIFHandler::isHeifDecoderAvailable())
1155 || (HEIFHandler::isSupportedHEJ2(header) && HEIFHandler::isHej2DecoderAvailable())
1156 || (HEIFHandler::isSupportedAVCI(header) && HEIFHandler::isAVCIDecoderAvailable())) {
1161 if (device->
isWritable() && (HEIFHandler::isHeifEncoderAvailable() || HEIFHandler::isHej2EncoderAvailable())) {
1175#include "moc_heif_p.cpp"
KGUIADDONS_EXPORT qreal chroma(const QColor &)
QFlags< Capability > Capabilities
void error(QWidget *parent, const QString &text, const QString &title, const KGuiItem &buttonOk, Options options=Notify)
const char * constData() const const
bool contains(QByteArrayView bv) const const
bool isEmpty() const const
qsizetype size() const const
QColorSpace fromIccProfile(const QByteArray &iccProfile)
float gamma() const const
QByteArray iccProfile() const const
bool isValid() const const
TransferFunction transferFunction() const const
QColorSpace colorSpace() const const
const uchar * constScanLine(int i) const const
QImage convertedToColorSpace(const QColorSpace &colorSpace) const const
bool hasAlphaChannel() const const
bool isNull() const const
QString text(const QString &key) const const
void setDevice(QIODevice *device)
virtual void setOption(ImageOption option, const QVariant &value)
bool isOpen() const const
bool isReadable() const const
bool isWritable() const const
QByteArray peek(qint64 maxSize)
qint64 write(const QByteArray &data)
bool isNull() const const
QString fromUtf8(QByteArrayView str)
int toInt(bool *ok) const const