mirror of
https://github.com/OpenCMT/uLib.git
synced 2025-12-06 07:21:31 +01:00
341 lines
10 KiB
C++
341 lines
10 KiB
C++
/*//////////////////////////////////////////////////////////////////////////////
|
|
// CMT Cosmic Muon Tomography project //////////////////////////////////////////
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
Copyright (c) 2014, Universita' degli Studi di Padova, INFN sez. di Padova
|
|
All rights reserved
|
|
|
|
Authors: Andrea Rigoni Garola < andrea.rigoni@pd.infn.it >
|
|
|
|
------------------------------------------------------------------
|
|
This library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Lesser General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 3.0 of the License, or (at your option) any later version.
|
|
|
|
This library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
License along with this library.
|
|
|
|
//////////////////////////////////////////////////////////////////////////////*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
|
|
#include <vtkSmartPointer.h>
|
|
#include <vtkImageImport.h>
|
|
#include <vtkImageExport.h>
|
|
|
|
#include <vtkFloatArray.h>
|
|
#include <vtkPointData.h>
|
|
|
|
#include <vtkGenericDataObjectReader.h>
|
|
#include <vtkImageShiftScale.h>
|
|
|
|
#include <vtkSmartVolumeMapper.h>
|
|
#include <vtkColorTransferFunction.h>
|
|
#include <vtkPiecewiseFunction.h>
|
|
#include <vtkVolumeProperty.h>
|
|
|
|
#include <vtkActor.h>
|
|
#include <vtkProperty.h>
|
|
#include <vtkPolyDataMapper.h>
|
|
|
|
|
|
#include <Math/VoxImage.h>
|
|
|
|
#include "vtkVoxImage.h"
|
|
|
|
|
|
namespace uLib {
|
|
namespace Vtk {
|
|
|
|
|
|
void vtkVoxImage::GetContent()
|
|
{
|
|
// ULIB -> VTK //
|
|
const int *dims = static_cast<const int *>(m_Content.GetDims().data());
|
|
m_Image->SetDimensions(dims);
|
|
|
|
float *spacing = m_Content.GetSpacing().data();
|
|
m_Image->SetSpacing(spacing[0],spacing[1],spacing[2]);
|
|
|
|
// Warning .. TODO: Read complete Transform matrix //
|
|
float *pos = m_Content.GetPosition().data();
|
|
m_Image->SetOrigin(pos[0],pos[1],pos[2]);
|
|
|
|
vtkFloatArray *array = vtkFloatArray::SafeDownCast
|
|
(m_Image->GetPointData()->GetScalars());
|
|
array->SetNumberOfTuples(m_Content.GetDims().prod());
|
|
Vector3i index(0,0,0);
|
|
int i=0;
|
|
for (int zv = 0; zv < dims[2]; ++zv) {
|
|
for (int yv = 0; yv < dims[1]; ++yv) {
|
|
for (int xv = 0; xv < dims[0]; ++xv) {
|
|
index << xv,yv,zv;
|
|
array->SetValue(i++,m_Content.GetValue(index));
|
|
}
|
|
}
|
|
}
|
|
m_Image->GetPointData()->SetScalars(array);
|
|
// m_Image->Update();
|
|
}
|
|
|
|
void vtkVoxImage::SetContent()
|
|
{
|
|
// VTK -> ULIB //
|
|
int *ext = m_Image->GetExtent();
|
|
int dims[3] = { ext[1]-ext[0] + 1, ext[3]-ext[2] + 1, ext[5]-ext[4] + 1 };
|
|
m_Content.SetDims(Vector3i(dims[0],dims[1],dims[2]));
|
|
|
|
double *spacing = m_Image->GetSpacing();
|
|
m_Content.SetSpacing(Vector3f(spacing[0],spacing[1],spacing[2]));
|
|
|
|
// Warning .. TODO: Read complete Transform matrix //
|
|
double *pos = m_Image->GetOrigin();
|
|
m_Content.SetPosition(Vector3f(pos[0],pos[1],pos[2]));
|
|
|
|
vtkFloatArray *array = vtkFloatArray::SafeDownCast
|
|
(m_Image->GetPointData()->GetScalars());
|
|
if(array) {
|
|
Vector3i index(0,0,0);
|
|
int i=0;
|
|
for (int zv = 0; zv < dims[2]; ++zv) {
|
|
for (int yv = 0; yv < dims[1]; ++yv) {
|
|
for (int xv = 0; xv < dims[0]; ++xv) {
|
|
index << xv,yv,zv;
|
|
m_Content.SetValue(index,array->GetValue(i++));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
std::cerr << "Error reading array Value Data\n";
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// VTK VOXIMAGE
|
|
|
|
vtkVoxImage::vtkVoxImage(Content &content) :
|
|
m_Content(content),
|
|
m_Actor(vtkVolume::New()),
|
|
m_Image(vtkImageData::New()),
|
|
m_Outline(vtkCubeSource::New()),
|
|
m_Reader(NULL),
|
|
m_Writer(NULL),
|
|
writer_factor(1.E6)
|
|
{
|
|
GetContent();
|
|
InstallPipe();
|
|
}
|
|
|
|
vtkVoxImage::~vtkVoxImage()
|
|
{
|
|
m_Image->Delete();
|
|
m_Actor->Delete();
|
|
m_Outline->Delete();
|
|
}
|
|
|
|
vtkImageData *vtkVoxImage::GetImageData()
|
|
{
|
|
GetContent();
|
|
return m_Image;
|
|
}
|
|
|
|
void vtkVoxImage::SaveToXMLFile(const char *fname)
|
|
{
|
|
vtkSmartPointer<vtkXMLImageDataWriter> writer =
|
|
vtkSmartPointer<vtkXMLImageDataWriter>::New();
|
|
writer->SetFileName(fname);
|
|
GetContent();
|
|
vtkSmartPointer<vtkImageShiftScale> vtkscale =
|
|
vtkSmartPointer<vtkImageShiftScale>::New();
|
|
|
|
# if VTK_MAJOR_VERSION <= 5
|
|
vtkscale->SetInputConnection(m_Image->GetProducerPort());
|
|
# else
|
|
vtkscale->SetInputData(m_Image);
|
|
# endif
|
|
vtkscale->SetScale(writer_factor);
|
|
vtkscale->Update();
|
|
writer->SetInputConnection(vtkscale->GetOutputPort());
|
|
writer->Update();
|
|
writer->Write();
|
|
}
|
|
|
|
void vtkVoxImage::ReadFromVKTFile(const char *fname)
|
|
{
|
|
vtkSmartPointer<vtkGenericDataObjectReader> reader =
|
|
vtkSmartPointer<vtkGenericDataObjectReader>::New();
|
|
reader->SetFileName(fname);
|
|
reader->Update();
|
|
if(reader->IsFileStructuredPoints())
|
|
{
|
|
vtkSmartPointer<vtkImageShiftScale> vtkscale =
|
|
vtkSmartPointer<vtkImageShiftScale>::New();
|
|
vtkscale->SetInputConnection(reader->GetOutputPort());
|
|
vtkscale->SetScale(1/writer_factor);
|
|
vtkscale->Update();
|
|
|
|
m_Image->DeepCopy(vtkscale->GetOutput()); // FIX! (pipe connection)
|
|
SetContent();
|
|
}
|
|
else {
|
|
std::cerr << "Error: file does not contain structured points\n";
|
|
}
|
|
m_Actor->Update();
|
|
}
|
|
|
|
void vtkVoxImage::ReadFromXMLFile(const char *fname)
|
|
{
|
|
vtkSmartPointer<vtkXMLImageDataReader> reader =
|
|
vtkSmartPointer<vtkXMLImageDataReader>::New();
|
|
reader->SetFileName(fname);
|
|
reader->Update();
|
|
vtkSmartPointer<vtkImageShiftScale> vtkscale =
|
|
vtkSmartPointer<vtkImageShiftScale>::New();
|
|
vtkscale->SetInputConnection(reader->GetOutputPort());
|
|
vtkscale->SetScale(1/writer_factor);
|
|
vtkscale->Update();
|
|
|
|
m_Image->DeepCopy(vtkscale->GetOutput());
|
|
SetContent();
|
|
}
|
|
|
|
|
|
void vtkVoxImage::setShadingPreset(int blendType)
|
|
{
|
|
vtkSmartVolumeMapper *mapper =
|
|
(vtkSmartVolumeMapper *)m_Actor->GetMapper();
|
|
vtkVolumeProperty *property = m_Actor->GetProperty();
|
|
|
|
static vtkColorTransferFunction *colorFun = vtkColorTransferFunction::New();
|
|
static vtkPiecewiseFunction *opacityFun = vtkPiecewiseFunction::New();
|
|
|
|
float window = 40 / writer_factor;
|
|
float level = 20 / writer_factor;
|
|
|
|
property->SetColor(colorFun);
|
|
property->SetScalarOpacity(opacityFun);
|
|
property->SetInterpolationTypeToLinear();
|
|
|
|
if(blendType != 6) {
|
|
colorFun->RemoveAllPoints();
|
|
opacityFun->RemoveAllPoints();
|
|
}
|
|
|
|
switch (blendType) {
|
|
// MIP
|
|
// Create an opacity ramp from the window and level values.
|
|
// Color is white. Blending is MIP.
|
|
case 0:
|
|
colorFun->AddRGBSegment(0.0, 1.0, 1.0, 1.0, 255.0, 1.0, 1.0, 1.0);
|
|
opacityFun->AddSegment(level - 0.5 * window, 0.0,
|
|
level + 0.5 * window, 1.0);
|
|
mapper->SetBlendModeToMaximumIntensity();
|
|
break;
|
|
|
|
// CompositeRamp
|
|
// Create a ramp from the window and level values. Use compositing
|
|
// without shading. Color is a ramp from black to white.
|
|
case 1:
|
|
colorFun->AddRGBSegment(level - 0.5 * window, 0.0, 0.0, 0.0,
|
|
level + 0.5 * window, 1.0, 1.0, 1.0);
|
|
opacityFun->AddSegment(level - 0.5 * window, 0.0,
|
|
level + 0.5 * window, 1.0);
|
|
mapper->SetBlendModeToComposite();
|
|
property->ShadeOff();
|
|
break;
|
|
|
|
// CompositeShadeRamp
|
|
// Create a ramp from the window and level values. Use compositing
|
|
// with shading. Color is white.
|
|
case 2:
|
|
colorFun->AddRGBSegment(0.0, 1.0, 1.0, 1.0, 255.0, 1.0, 1.0, 1.0);
|
|
opacityFun->AddSegment(level - 0.5 * window, 0.0,
|
|
level + 0.5 * window, 1.0);
|
|
mapper->SetBlendModeToComposite();
|
|
property->ShadeOn();
|
|
break;
|
|
|
|
// CT_Skin
|
|
// Use compositing and functions set to highlight skin in CT data
|
|
// Not for use on RGB data
|
|
case 3:
|
|
colorFun->AddRGBPoint(-3024, 0, 0, 0, 0.5, 0.0);
|
|
colorFun->AddRGBPoint(-1000, .62, .36, .18, 0.5, 0.0);
|
|
colorFun->AddRGBPoint(-500, .88, .60, .29, 0.33, 0.45);
|
|
colorFun->AddRGBPoint(3071, .83, .66, 1, 0.5, 0.0);
|
|
|
|
opacityFun->AddPoint(-3024, 0, 0.5, 0.0);
|
|
opacityFun->AddPoint(-1000, 0, 0.5, 0.0);
|
|
opacityFun->AddPoint(-500, 1.0, 0.33, 0.45);
|
|
opacityFun->AddPoint(3071, 1.0, 0.5, 0.0);
|
|
|
|
mapper->SetBlendModeToComposite();
|
|
property->ShadeOn();
|
|
property->SetAmbient(0.1);
|
|
property->SetDiffuse(0.9);
|
|
property->SetSpecular(0.2);
|
|
property->SetSpecularPower(10.0);
|
|
property->SetScalarOpacityUnitDistance(0.8919);
|
|
break;
|
|
|
|
|
|
default:
|
|
vtkGenericWarningMacro("Unknown blend type.");
|
|
break;
|
|
}
|
|
}
|
|
|
|
void vtkVoxImage::Update() {
|
|
m_Actor->Update();
|
|
m_Outline->SetBounds(m_Image->GetBounds());
|
|
m_Outline->Update();
|
|
}
|
|
|
|
void vtkVoxImage::InstallPipe()
|
|
{
|
|
vtkSmartPointer<vtkSmartVolumeMapper> mapper =
|
|
vtkSmartPointer<vtkSmartVolumeMapper>::New();
|
|
# if VTK_MAJOR_VERSION <= 5
|
|
mapper->SetInputConnection(m_Image->GetProducerPort());
|
|
# else
|
|
mapper->SetInputData(m_Image);
|
|
# endif
|
|
mapper->Update();
|
|
|
|
m_Actor->SetMapper(mapper);
|
|
this->setShadingPreset(0);
|
|
mapper->Update();
|
|
|
|
m_Outline->SetBounds(m_Image->GetBounds());
|
|
vtkSmartPointer<vtkPolyDataMapper> mmapper = vtkSmartPointer<vtkPolyDataMapper>::New();
|
|
mmapper->SetInputConnection(m_Outline->GetOutputPort());
|
|
vtkSmartPointer<vtkActor> actor = vtkSmartPointer<vtkActor>::New();
|
|
actor->SetMapper(mmapper);
|
|
actor->GetProperty()->SetRepresentationToWireframe();
|
|
actor->GetProperty()->SetAmbient(0.7);
|
|
|
|
// this->SetProp(actor);
|
|
this->SetProp(m_Actor);
|
|
}
|
|
|
|
|
|
} // vtk
|
|
} // uLib
|