changeset 5054:255b02c68908

Added support for RGBA64 images in tools/create-dicom and /preview (Contribution from James Manners - Pliny)
author Alain Mazy <am@osimis.io>
date Fri, 08 Jul 2022 15:26:52 +0200
parents 1e435be86887
children 942eb3849d8d
files NEWS OrthancFramework/Sources/DicomFormat/DicomImageInformation.cpp OrthancFramework/Sources/DicomParsing/ParsedDicomFile.cpp OrthancFramework/Sources/Enumerations.cpp OrthancFramework/Sources/Enumerations.h OrthancFramework/Sources/Images/PngReader.cpp OrthancFramework/Sources/Images/PngWriter.cpp OrthancFramework/UnitTestsSources/FromDcmtkTests.cpp OrthancFramework/UnitTestsSources/ImageTests.cpp OrthancServer/Sources/OrthancRestApi/OrthancRestResources.cpp
diffstat 10 files changed, 130 insertions(+), 4 deletions(-) [+]
line wrap: on
line diff
--- 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)
 ===========================
 
--- 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;
   }
 
--- 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<const Uint8*>(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);
--- 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:
--- 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
   };
 
 
--- 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);
--- 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)
--- 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
   }
 }
 
--- 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<uint8_t> 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)
 {
--- 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<float>(decoded->GetHeight()));
         }
         
-        if (decoded->GetFormat() == PixelFormat_RGB24)
+        if (decoded->GetFormat() == PixelFormat_RGB24 || decoded->GetFormat() == PixelFormat_RGB48)
         {
           if (targetWidth == decoded->GetWidth() &&
               targetHeight == decoded->GetHeight())