# HG changeset patch # User Alain Mazy # Date 1657286812 -7200 # Node ID 255b02c68908b1e85c9662feb5069d1b3d0b5491 # Parent 1e435be86887c0ec988c1b830bb2895c194721e8 Added support for RGBA64 images in tools/create-dicom and /preview (Contribution from James Manners - Pliny) diff -r 1e435be86887 -r 255b02c68908 NEWS --- a/NEWS Fri Jul 08 07:24:10 2022 +0200 +++ b/NEWS Fri Jul 08 15:26:52 2022 +0200 @@ -1,3 +1,12 @@ +Pending changes in the mainline +=============================== + +General +------- + +* Added support for RGBA64 images in tools/create-dicom and /preview + + Version 1.11.1 (2022-06-30) =========================== diff -r 1e435be86887 -r 255b02c68908 OrthancFramework/Sources/DicomFormat/DicomImageInformation.cpp --- a/OrthancFramework/Sources/DicomFormat/DicomImageInformation.cpp Fri Jul 08 07:24:10 2022 +0200 +++ b/OrthancFramework/Sources/DicomFormat/DicomImageInformation.cpp Fri Jul 08 15:26:52 2022 +0200 @@ -380,6 +380,15 @@ return true; } + if (GetBitsStored() == 16 && + GetChannelCount() == 3 && + !IsSigned() && + (ignorePhotometricInterpretation || photometric_ == PhotometricInterpretation_RGB)) + { + format = PixelFormat_RGB48; + return true; + } + return false; } diff -r 1e435be86887 -r 255b02c68908 OrthancFramework/Sources/DicomParsing/ParsedDicomFile.cpp --- a/OrthancFramework/Sources/DicomParsing/ParsedDicomFile.cpp Fri Jul 08 07:24:10 2022 +0200 +++ b/OrthancFramework/Sources/DicomParsing/ParsedDicomFile.cpp Fri Jul 08 15:26:52 2022 +0200 @@ -1276,14 +1276,16 @@ accessor.GetFormat() != PixelFormat_Grayscale16 && accessor.GetFormat() != PixelFormat_SignedGrayscale16 && accessor.GetFormat() != PixelFormat_RGB24 && - accessor.GetFormat() != PixelFormat_RGBA32) + accessor.GetFormat() != PixelFormat_RGBA32 && + accessor.GetFormat() != PixelFormat_RGBA64) { throw OrthancException(ErrorCode_NotImplemented); } InvalidateCache(); - if (accessor.GetFormat() == PixelFormat_RGBA32) + if (accessor.GetFormat() == PixelFormat_RGBA32 || + accessor.GetFormat() == PixelFormat_RGBA64) { LOG(WARNING) << "Getting rid of the alpha channel when embedding a RGBA image inside DICOM"; } @@ -1335,6 +1337,20 @@ ReplacePlainString(DICOM_TAG_PLANAR_CONFIGURATION, "0"); // Color channels are interleaved break; + + case PixelFormat_RGBA64: + ReplacePlainString(DICOM_TAG_PHOTOMETRIC_INTERPRETATION, "RGB"); + ReplacePlainString(DICOM_TAG_SAMPLES_PER_PIXEL, "3"); + ReplacePlainString(DICOM_TAG_BITS_ALLOCATED, "16"); + ReplacePlainString(DICOM_TAG_BITS_STORED, "16"); + ReplacePlainString(DICOM_TAG_HIGH_BIT, "15"); + bytesPerPixel = 6; + + // "Planar configuration" must only present if "Samples per + // Pixel" is greater than 1 + ReplacePlainString(DICOM_TAG_PLANAR_CONFIGURATION, "0"); // Color channels are interleaved + + break; case PixelFormat_Grayscale16: case PixelFormat_SignedGrayscale16: @@ -1394,6 +1410,23 @@ break; } + + case PixelFormat_RGBA64: + { + // The alpha channel is not supported by the DICOM standard + const Uint8* source = reinterpret_cast(accessor.GetConstRow(y)); + for (unsigned int x = 0; x < width; x++, q += 6, source += 8) + { + q[0] = source[0]; + q[1] = source[1]; + q[2] = source[2]; + q[3] = source[3]; + q[4] = source[4]; + q[5] = source[5]; + } + + break; + } default: throw OrthancException(ErrorCode_NotImplemented); diff -r 1e435be86887 -r 255b02c68908 OrthancFramework/Sources/Enumerations.cpp --- a/OrthancFramework/Sources/Enumerations.cpp Fri Jul 08 07:24:10 2022 +0200 +++ b/OrthancFramework/Sources/Enumerations.cpp Fri Jul 08 15:26:52 2022 +0200 @@ -1914,6 +1914,7 @@ return 6; case PixelFormat_Grayscale64: + case PixelFormat_RGBA64: return 8; default: diff -r 1e435be86887 -r 255b02c68908 OrthancFramework/Sources/Enumerations.h --- a/OrthancFramework/Sources/Enumerations.h Fri Jul 08 07:24:10 2022 +0200 +++ b/OrthancFramework/Sources/Enumerations.h Fri Jul 08 15:26:52 2022 +0200 @@ -317,7 +317,14 @@ * {summary}{Graylevel, unsigned 64bpp image.} * {description}{The image is graylevel. Each pixel is unsigned and stored in 8 bytes.} **/ - PixelFormat_Grayscale64 = 10 + PixelFormat_Grayscale64 = 10, + + /** + * {summary}{Color image in RGBA64 format.} + * {description}{This format describes a color image. The pixels are stored in 8 + * consecutive bytes. The memory layout is RGBA.} + **/ + PixelFormat_RGBA64 = 11 }; diff -r 1e435be86887 -r 255b02c68908 OrthancFramework/Sources/Images/PngReader.cpp --- a/OrthancFramework/Sources/Images/PngReader.cpp Fri Jul 08 07:24:10 2022 +0200 +++ b/OrthancFramework/Sources/Images/PngReader.cpp Fri Jul 08 15:26:52 2022 +0200 @@ -175,6 +175,16 @@ format = PixelFormat_RGBA32; pitch = 4 * width; } + else if (color_type == PNG_COLOR_TYPE_RGBA && bit_depth == 16) + { + format = PixelFormat_RGBA64; + pitch = 8 * width; + + if (Toolbox::DetectEndianness() == Endianness_Little) + { + png_set_swap(rabi.png_); + } + } else { throw OrthancException(ErrorCode_NotImplemented); diff -r 1e435be86887 -r 255b02c68908 OrthancFramework/Sources/Images/PngWriter.cpp --- a/OrthancFramework/Sources/Images/PngWriter.cpp Fri Jul 08 07:24:10 2022 +0200 +++ b/OrthancFramework/Sources/Images/PngWriter.cpp Fri Jul 08 15:26:52 2022 +0200 @@ -165,6 +165,11 @@ bitDepth_ = 16; colorType_ = PNG_COLOR_TYPE_GRAY; break; + + case PixelFormat_RGBA64: + bitDepth_ = 16; + colorType_ = PNG_COLOR_TYPE_RGBA; + break; default: throw OrthancException(ErrorCode_NotImplemented); @@ -189,6 +194,7 @@ { case PixelFormat_Grayscale16: case PixelFormat_SignedGrayscale16: + case PixelFormat_RGBA64: { int transforms = 0; if (Toolbox::DetectEndianness() == Endianness_Little) diff -r 1e435be86887 -r 255b02c68908 OrthancFramework/UnitTestsSources/FromDcmtkTests.cpp --- a/OrthancFramework/UnitTestsSources/FromDcmtkTests.cpp Fri Jul 08 07:24:10 2022 +0200 +++ b/OrthancFramework/UnitTestsSources/FromDcmtkTests.cpp Fri Jul 08 15:26:52 2022 +0200 @@ -242,6 +242,16 @@ #if ORTHANC_SANDBOXED != 1 o.SaveToFile("UnitTestsResults/png4.dcm"); #endif + + // From http://www.schaik.com/pngsuite/pngsuite_bas_png.html + // 16Bit RGBA PNG + // License http://www.schaik.com/pngsuite/PngSuite.LICENSE + s = ""; + o.EmbedContent(s); + +#if ORTHANC_SANDBOXED != 1 + o.SaveToFile("UnitTestsResults/png5.dcm"); +#endif } } diff -r 1e435be86887 -r 255b02c68908 OrthancFramework/UnitTestsSources/ImageTests.cpp --- a/OrthancFramework/UnitTestsSources/ImageTests.cpp Fri Jul 08 07:24:10 2022 +0200 +++ b/OrthancFramework/UnitTestsSources/ImageTests.cpp Fri Jul 08 15:26:52 2022 +0200 @@ -83,6 +83,47 @@ ASSERT_EQ("604e785f53c99cae6ea4584870b2c41d", md5); } +TEST(PngWriter, Color16Pattern) +{ + Orthanc::PngWriter w; + unsigned int width = 17; + unsigned int height = 61; + unsigned int pitch = width * 8; + + std::vector image(height * pitch); + for (unsigned int y = 0; y < height; y++) + { + uint8_t *p = &image[0] + y * pitch; + for (unsigned int x = 0; x < width; x++, p += 8) + { + p[0] = (y % 8 == 0) ? 255 : 0; + p[1] = (y % 8 == 1) ? 255 : 0; + p[2] = (y % 8 == 2) ? 255 : 0; + p[3] = (y % 8 == 3) ? 255 : 0; + p[4] = (y % 8 == 4) ? 255 : 0; + p[5] = (y % 8 == 5) ? 255 : 0; + p[6] = (y % 8 == 6) ? 255 : 0; + p[7] = (y % 8 == 7) ? 255 : 0; + } + } + + Orthanc::ImageAccessor accessor; + accessor.AssignReadOnly(Orthanc::PixelFormat_RGBA64, width, height, pitch, &image[0]); + + std::string f; + +#if ORTHANC_SANDBOXED == 1 + Orthanc::IImageWriter::WriteToMemory(w, f, accessor); +#else + Orthanc::IImageWriter::WriteToFile(w, "UnitTestsResults/Color16Pattern.png", accessor); + Orthanc::SystemToolbox::ReadFile(f, "UnitTestsResults/Color16Pattern.png"); +#endif + + std::string md5; + Orthanc::Toolbox::ComputeMD5(md5, f); + ASSERT_EQ("1cca552b6bd152b6fdab35c4a9f02c2a", md5); +} + TEST(PngWriter, Gray8Pattern) { diff -r 1e435be86887 -r 255b02c68908 OrthancServer/Sources/OrthancRestApi/OrthancRestResources.cpp --- a/OrthancServer/Sources/OrthancRestApi/OrthancRestResources.cpp Fri Jul 08 07:24:10 2022 +0200 +++ b/OrthancServer/Sources/OrthancRestApi/OrthancRestResources.cpp Fri Jul 08 15:26:52 2022 +0200 @@ -1064,7 +1064,7 @@ targetHeight = boost::math::iround(ratio * static_cast(decoded->GetHeight())); } - if (decoded->GetFormat() == PixelFormat_RGB24) + if (decoded->GetFormat() == PixelFormat_RGB24 || decoded->GetFormat() == PixelFormat_RGB48) { if (targetWidth == decoded->GetWidth() && targetHeight == decoded->GetHeight())