/*========================================================================= Program: GDCM (Grassroots DICOM). A DICOM library Copyright (c) 2006-2011 Mathieu Malaterre All rights reserved. See Copyright.txt or http://gdcm.sourceforge.net/Copyright.html for details. This software is distributed WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the above copyright notice for more information. =========================================================================*/ /*========================================================================= Portions of this file are subject to the VTK Toolkit Version 3 copyright. Program: Visualization Toolkit Module: $RCSfile: vtkImageRGBToYBR.cxx,v $ Copyright (c) Ken Martin, Will Schroeder, Bill Lorensen All rights reserved. See Copyright.txt or http://www.kitware.com/Copyright.htm for details. This software is distributed WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the above copyright notice for more information. =========================================================================*/ #include "vtkImageRGBToYBR.h" #include "vtkImageData.h" #include "vtkImageProgressIterator.h" #include "vtkMath.h" #include "vtkObjectFactory.h" //vtkCxxRevisionMacro(vtkImageRGBToYBR, "$Revision: 1.31 $") vtkStandardNewMacro(vtkImageRGBToYBR) //---------------------------------------------------------------------------- vtkImageRGBToYBR::vtkImageRGBToYBR() { this->SetNumberOfInputPorts(1); this->SetNumberOfOutputPorts(1); } //---------------------------------------------------------------------------- // This templated function executes the filter for any type of data. template void vtkImageRGBToYBRExecute(vtkImageRGBToYBR *self, vtkImageData *inData, vtkImageData *outData, int outExt[6], int id, T *) { vtkImageIterator inIt(inData, outExt); vtkImageProgressIterator outIt(outData, outExt, self, id); int idxC; // find the region to loop over int maxC = inData->GetNumberOfScalarComponents()-1; // Loop through ouput pixels while (!outIt.IsAtEnd()) { T* inSI = inIt.BeginSpan(); T* outSI = outIt.BeginSpan(); T* outSIEnd = outIt.EndSpan(); while (outSI != outSIEnd) { // http://lestourtereaux.free.fr/papers/data/yuvrgb.pdf // Pixel operation unsigned char r = (unsigned char)(*inSI); ++inSI; unsigned char g = (unsigned char)(*inSI); ++inSI; unsigned char b = (unsigned char)(*inSI); ++inSI; double y = 0.257 * r + 0.504 * g + 0.098 * b + 16; double u = -0.148* r - 0.291 * g + 0.439 * b + 128; double v = 0.439 * r - 0.368 * g - 0.071 * b + 128; // assign output. *outSI = (T)(y); ++outSI; *outSI = (T)(u); ++outSI; *outSI = (T)(v); ++outSI; for (idxC = 3; idxC <= maxC; idxC++) { *outSI++ = *inSI++; } } inIt.NextSpan(); outIt.NextSpan(); } } //---------------------------------------------------------------------------- void vtkImageRGBToYBR::ThreadedExecute (vtkImageData *inData, vtkImageData *outData, int outExt[6], int id) { vtkDebugMacro(<< "Execute: inData = " << inData << ", outData = " << outData); // this filter expects that input is the same type as output. if (inData->GetScalarType() != outData->GetScalarType()) { vtkErrorMacro(<< "Execute: input ScalarType, " << inData->GetScalarType() << ", must match out ScalarType " << outData->GetScalarType()); return; } if (inData->GetScalarType() != VTK_UNSIGNED_CHAR ) { return; } // need three components for input and output if (inData->GetNumberOfScalarComponents() < 3) { vtkErrorMacro("Input has too few components"); return; } if (outData->GetNumberOfScalarComponents() < 3) { vtkErrorMacro("Output has too few components"); return; } switch (inData->GetScalarType()) { vtkTemplateMacro( vtkImageRGBToYBRExecute(this, inData, outData, outExt, id, static_cast(0))); default: vtkErrorMacro(<< "Execute: Unknown ScalarType"); return; } } //---------------------------------------------------------------------------- void vtkImageRGBToYBR::PrintSelf(ostream& os, vtkIndent indent) { this->Superclass::PrintSelf(os,indent); }