예제 #1
0
void QmitkDenoisingView::StartDenoising()
{
  if (!m_ThreadIsRunning)
  {
    if (m_ImageNode.IsNotNull())
    {
      m_LastProgressCount = 0;
      switch (m_SelectedFilter)
      {
        case NOFILTERSELECTED:
        {
          break;
        }
        case NLM:
        {
          // initialize NLM
          m_InputImage = dynamic_cast<DiffusionImageType*> (m_ImageNode->GetData());
          m_NonLocalMeansFilter = NonLocalMeansDenoisingFilterType::New();

          if (m_BrainMaskNode.IsNotNull())
          {
            // use brainmask if set
            m_ImageMask = dynamic_cast<mitk::Image*>(m_BrainMaskNode->GetData());
            itk::Image<DiffusionPixelType, 3>::Pointer itkMask;
            mitk::CastToItkImage(m_ImageMask, itkMask);
            m_NonLocalMeansFilter->SetInputMask(itkMask);

            itk::ImageRegionIterator< itk::Image<DiffusionPixelType, 3> > mit(itkMask, itkMask->GetLargestPossibleRegion());
            mit.GoToBegin();
            itk::Image<DiffusionPixelType, 3>::IndexType minIndex;
            itk::Image<DiffusionPixelType, 3>::IndexType maxIndex;
            minIndex.Fill(10000);
            maxIndex.Fill(0);
            while (!mit.IsAtEnd())
            {
              if (mit.Get())
              {
                // calculation of the start & end index of the smallest masked region
                minIndex[0] = minIndex[0] < mit.GetIndex()[0] ? minIndex[0] : mit.GetIndex()[0];
                minIndex[1] = minIndex[1] < mit.GetIndex()[1] ? minIndex[1] : mit.GetIndex()[1];
                minIndex[2] = minIndex[2] < mit.GetIndex()[2] ? minIndex[2] : mit.GetIndex()[2];

                maxIndex[0] = maxIndex[0] > mit.GetIndex()[0] ? maxIndex[0] : mit.GetIndex()[0];
                maxIndex[1] = maxIndex[1] > mit.GetIndex()[1] ? maxIndex[1] : mit.GetIndex()[1];
                maxIndex[2] = maxIndex[2] > mit.GetIndex()[2] ? maxIndex[2] : mit.GetIndex()[2];
              }
              ++mit;
            }
            itk::Image<DiffusionPixelType, 3>::SizeType size;
            size[0] = maxIndex[0] - minIndex[0] + 1;
            size[1] = maxIndex[1] - minIndex[1] + 1;
            size[2] = maxIndex[2] - minIndex[2] + 1;

            m_MaxProgressCount = size[0] * size[1] * size[2];
          }
          else
          {
            // initialize the progressbar
            m_MaxProgressCount = m_InputImage->GetDimension(0) * m_InputImage->GetDimension(1) * m_InputImage->GetDimension(2);
          }


          mitk::ProgressBar::GetInstance()->AddStepsToDo(m_MaxProgressCount);


          m_NonLocalMeansFilter->SetInputImage(m_InputImage->GetVectorImage());
          m_NonLocalMeansFilter->SetUseRicianAdaption(m_Controls->m_RicianCheckbox->isChecked());
          m_NonLocalMeansFilter->SetUseJointInformation(m_Controls->m_JointInformationCheckbox->isChecked());
          m_NonLocalMeansFilter->SetSearchRadius(m_Controls->m_SpinBoxParameter1->value());
          m_NonLocalMeansFilter->SetComparisonRadius(m_Controls->m_SpinBoxParameter2->value());
          m_NonLocalMeansFilter->SetVariance(m_Controls->m_DoubleSpinBoxParameter3->value());





          // start denoising in detached thread
          m_DenoisingThread.start(QThread::HighestPriority);

          break;
        }
        case GAUSS:
        {
          // initialize GAUSS and run
          m_InputImage = dynamic_cast<DiffusionImageType*> (m_ImageNode->GetData());

          ExtractFilterType::Pointer extractor = ExtractFilterType::New();
          extractor->SetInput(m_InputImage->GetVectorImage());
          ComposeFilterType::Pointer composer = ComposeFilterType::New();

          for (unsigned int i = 0; i < m_InputImage->GetVectorImage()->GetVectorLength(); ++i)
          {
            extractor->SetIndex(i);
            extractor->Update();

            m_GaussianFilter = GaussianFilterType::New();
            m_GaussianFilter->SetVariance(m_Controls->m_SpinBoxParameter1->value());

            if (m_BrainMaskNode.IsNotNull())
            {
              m_ImageMask = dynamic_cast<mitk::Image*>(m_BrainMaskNode->GetData());
              itk::Image<DiffusionPixelType, 3>::Pointer itkMask = itk::Image<DiffusionPixelType, 3>::New();
              mitk::CastToItkImage(m_ImageMask, itkMask);

              itk::MaskImageFilter<itk::Image<DiffusionPixelType, 3> , itk::Image<DiffusionPixelType, 3> >::Pointer maskImageFilter = itk::MaskImageFilter<itk::Image<DiffusionPixelType, 3> , itk::Image<DiffusionPixelType, 3> >::New();
              maskImageFilter->SetInput(extractor->GetOutput());
              maskImageFilter->SetMaskImage(itkMask);
              maskImageFilter->Update();
              m_GaussianFilter->SetInput(maskImageFilter->GetOutput());
            }
            else
            {
              m_GaussianFilter->SetInput(extractor->GetOutput());
            }
            m_GaussianFilter->Update();

            composer->SetInput(i, m_GaussianFilter->GetOutput());
          }
          composer->Update();


          DiffusionImageType::Pointer image = DiffusionImageType::New();
          image->SetVectorImage(composer->GetOutput());
          image->SetReferenceBValue(m_InputImage->GetReferenceBValue());
          image->SetDirections(m_InputImage->GetDirections());
          image->InitializeFromVectorImage();
          mitk::DataNode::Pointer imageNode = mitk::DataNode::New();
          imageNode->SetData( image );
          QString name = m_ImageNode->GetName().c_str();

          imageNode->SetName((name+"_gauss_"+QString::number(m_Controls->m_SpinBoxParameter1->value())).toStdString().c_str());
          GetDefaultDataStorage()->Add(imageNode);

          break;
        }
      }
    }
  }

  else
  {
    m_NonLocalMeansFilter->AbortGenerateDataOn();
    m_CompletedCalculation = false;
  }
}
예제 #2
0
int main(int argc, char* argv[])
{
  mitkCommandLineParser parser;

  parser.setTitle("DmriDenoising");
  parser.setCategory("Preprocessing Tools");
  parser.setDescription("dMRI denoising tool");
  parser.setContributor("MIC");

  parser.setArgumentPrefix("--", "-");

  parser.beginGroup("1. Mandatory arguments:");
  parser.addArgument("", "i", mitkCommandLineParser::String, "Input:", "input image", us::Any(), false, false, false, mitkCommandLineParser::Input);
  parser.addArgument("", "o", mitkCommandLineParser::String, "Output:", "output image", us::Any(), false, false, false, mitkCommandLineParser::Output);
  parser.addArgument("type", "", mitkCommandLineParser::Int, "Type:", "0 (TotalVariation), 1 (Gauss), 2 (NLM)", 0);
  parser.endGroup();

  parser.beginGroup("2. Total variation parameters:");
  parser.addArgument("tv_iterations", "", mitkCommandLineParser::Int, "Iterations:", "", 1);
  parser.addArgument("lambda", "", mitkCommandLineParser::Float, "Lambda:", "", 0.1);
  parser.endGroup();

  parser.beginGroup("3. Gauss parameters:");
  parser.addArgument("variance", "", mitkCommandLineParser::Float, "Variance:", "", 1.0);
  parser.endGroup();

  parser.beginGroup("4. NLM parameters:");
  parser.addArgument("nlm_iterations", "", mitkCommandLineParser::Int, "Iterations:", "", 4);
  parser.addArgument("sampling_radius", "", mitkCommandLineParser::Int, "Sampling radius:", "", 4);
  parser.addArgument("patch_radius", "", mitkCommandLineParser::Int, "Patch radius:", "", 1);
  parser.addArgument("num_patches", "", mitkCommandLineParser::Int, "Num. patches:", "", 10);
  parser.endGroup();

  std::map<std::string, us::Any> parsedArgs = parser.parseArguments(argc, argv);
  if (parsedArgs.size()==0)
    return EXIT_FAILURE;

  // mandatory arguments
  std::string imageName = us::any_cast<std::string>(parsedArgs["i"]);
  std::string outImage = us::any_cast<std::string>(parsedArgs["o"]);

  int type = 0;
  if (parsedArgs.count("type"))
    type = us::any_cast<int>(parsedArgs["type"]);

  int tv_iterations = 1;
  if (parsedArgs.count("tv_iterations"))
    tv_iterations = us::any_cast<int>(parsedArgs["tv_iterations"]);

  float lambda = 0.1;
  if (parsedArgs.count("lambda"))
    lambda = us::any_cast<float>(parsedArgs["lambda"]);

  float variance = 1.0;
  if (parsedArgs.count("variance"))
    variance = us::any_cast<float>(parsedArgs["variance"]);

  int nlm_iterations = 4;
  if (parsedArgs.count("nlm_iterations"))
    nlm_iterations = us::any_cast<int>(parsedArgs["nlm_iterations"]);

  int sampling_radius = 4;
  if (parsedArgs.count("sampling_radius"))
    sampling_radius = us::any_cast<int>(parsedArgs["sampling_radius"]);

  int patch_radius = 1;
  if (parsedArgs.count("patch_radius"))
    patch_radius = us::any_cast<int>(parsedArgs["patch_radius"]);

  int num_patches = 10;
  if (parsedArgs.count("num_patches"))
    num_patches = us::any_cast<int>(parsedArgs["num_patches"]);

  try
  {
    mitk::PreferenceListReaderOptionsFunctor functor = mitk::PreferenceListReaderOptionsFunctor({"Diffusion Weighted Images"}, {});
    mitk::Image::Pointer input_image = mitk::IOUtil::Load<mitk::Image>(imageName, &functor);

    typedef short                                                                         DiffusionPixelType;
    typedef itk::VectorImage<DiffusionPixelType, 3>                                       DwiImageType;
    typedef itk::Image<DiffusionPixelType, 3>                                             DwiVolumeType;
    typedef itk::DiscreteGaussianImageFilter < DwiVolumeType, DwiVolumeType >             GaussianFilterType;
    typedef itk::PatchBasedDenoisingImageFilter < DwiVolumeType, DwiVolumeType >          NlmFilterType;
    typedef itk::VectorImageToImageFilter < DiffusionPixelType >                          ExtractFilterType;
    typedef itk::ComposeImageFilter < itk::Image<DiffusionPixelType, 3> >                 ComposeFilterType;

    if (!mitk::DiffusionPropertyHelper::IsDiffusionWeightedImage(input_image))
      mitkThrow() << "Input is not a diffusion-weighted image!";

    DwiImageType::Pointer itkVectorImagePointer = mitk::DiffusionPropertyHelper::GetItkVectorImage(input_image);

    mitk::Image::Pointer denoised_image = nullptr;

    switch (type)
    {
    case 0:
    {
      ComposeFilterType::Pointer composer = ComposeFilterType::New();
      for (unsigned int i=0; i<itkVectorImagePointer->GetVectorLength(); ++i)
      {
        ExtractFilterType::Pointer extractor = ExtractFilterType::New();
        extractor->SetInput( itkVectorImagePointer );
        extractor->SetIndex( i );
        extractor->Update();
        DwiVolumeType::Pointer gradient_volume = extractor->GetOutput();
        itk::TotalVariationDenoisingImageFilter< DwiVolumeType, DwiVolumeType >::Pointer filter = itk::TotalVariationDenoisingImageFilter< DwiVolumeType, DwiVolumeType >::New();
        filter->SetInput(gradient_volume);
        filter->SetLambda(lambda);
        filter->SetNumberIterations(tv_iterations);
        filter->Update();
        composer->SetInput(i, filter->GetOutput());
      }
      composer->Update();
      denoised_image = mitk::GrabItkImageMemory(composer->GetOutput());
      break;
    }
    case 1:
    {
      ExtractFilterType::Pointer extractor = ExtractFilterType::New();
      extractor->SetInput( itkVectorImagePointer );
      ComposeFilterType::Pointer composer = ComposeFilterType::New();
      for (unsigned int i = 0; i < itkVectorImagePointer->GetVectorLength(); ++i)
      {
        extractor->SetIndex(i);
        extractor->Update();
        GaussianFilterType::Pointer filter = GaussianFilterType::New();
        filter->SetVariance(variance);
        filter->SetInput(extractor->GetOutput());
        filter->Update();
        composer->SetInput(i, filter->GetOutput());
      }
      composer->Update();
      denoised_image = mitk::GrabItkImageMemory(composer->GetOutput());
      break;
    }
    case 2:
    {
      typedef itk::Statistics::GaussianRandomSpatialNeighborSubsampler< NlmFilterType::PatchSampleType, DwiVolumeType::RegionType > SamplerType;
      // sampling the image to find similar patches
      SamplerType::Pointer sampler = SamplerType::New();
      sampler->SetRadius( sampling_radius );
      sampler->SetVariance( sampling_radius*sampling_radius );
      sampler->SetNumberOfResultsRequested( num_patches );

      MITK_INFO << "Starting NLM denoising";
      ExtractFilterType::Pointer extractor = ExtractFilterType::New();
      extractor->SetInput( itkVectorImagePointer );
      ComposeFilterType::Pointer composer = ComposeFilterType::New();
      for (unsigned int i = 0; i < itkVectorImagePointer->GetVectorLength(); ++i)
      {
        extractor->SetIndex(i);
        extractor->Update();
        NlmFilterType::Pointer filter = NlmFilterType::New();
        filter->SetInput(extractor->GetOutput());
        filter->SetPatchRadius(patch_radius);
        filter->SetNoiseModel(NlmFilterType::RICIAN);
        filter->UseSmoothDiscPatchWeightsOn();
        filter->UseFastTensorComputationsOn();

        filter->SetNumberOfIterations(nlm_iterations);
        filter->SetSmoothingWeight( 1 );
        filter->SetKernelBandwidthEstimation(true);

        filter->SetSampler( sampler );
        filter->Update();
        composer->SetInput(i, filter->GetOutput());
        MITK_INFO << "Gradient " << i << " finished";
      }
      composer->Update();
      denoised_image = mitk::GrabItkImageMemory(composer->GetOutput());
      break;
    }
    }

    mitk::DiffusionPropertyHelper::SetGradientContainer(denoised_image, mitk::DiffusionPropertyHelper::GetGradientContainer(input_image));
    mitk::DiffusionPropertyHelper::SetReferenceBValue(denoised_image, mitk::DiffusionPropertyHelper::GetReferenceBValue(input_image));
    mitk::DiffusionPropertyHelper::InitializeImage( denoised_image );

    std::string ext = itksys::SystemTools::GetFilenameExtension(outImage);
    if (ext==".nii" || ext==".nii.gz")
      mitk::IOUtil::Save(denoised_image, "DWI_NIFTI", outImage);
    else
      mitk::IOUtil::Save(denoised_image, outImage);
  }
  catch (itk::ExceptionObject e)
  {
    std::cout << e;
    return EXIT_FAILURE;
  }
  catch (std::exception e)
  {
    std::cout << e.what();
    return EXIT_FAILURE;
  }
  catch (...)
  {
    std::cout << "ERROR!?!";
    return EXIT_FAILURE;
  }
  return EXIT_SUCCESS;
}
예제 #3
0
void QmitkSimpleRegistrationView::OnRegResultIsAvailable(mitk::MAPRegistrationWrapper::Pointer spResultRegistration, const QmitkRegistrationJob* job)
{
  mitk::Image::Pointer movingImage = dynamic_cast<mitk::Image*>(m_MovingImageNode->GetData());
  mitk::Image::Pointer image;

  if (m_RegistrationType==0 && !m_Controls->m_ResampleBox->isChecked())
  {
    image = mitk::ImageMappingHelper::refineGeometry(movingImage, spResultRegistration, true);

    mitk::DiffusionPropertyHelper::CopyProperties(movingImage, image, true);
    auto reg = spResultRegistration->GetRegistration();
    typedef mitk::DiffusionImageCorrectionFilter CorrectionFilterType;
    CorrectionFilterType::Pointer corrector = CorrectionFilterType::New();
    corrector->SetImage( image );
    corrector->CorrectDirections( mitk::MITKRegistrationHelper::getAffineMatrix(reg, false)->GetMatrix().GetVnlMatrix() );
  }
  else
  {
    if (!mitk::DiffusionPropertyHelper::IsDiffusionWeightedImage(movingImage))
    {
      image = mitk::ImageMappingHelper::map(movingImage, spResultRegistration, false, 0, job->m_spTargetData->GetGeometry(), false, 0, mitk::ImageMappingInterpolator::BSpline_3);
    }
    else
    {
      typedef itk::ComposeImageFilter < ITKDiffusionVolumeType > ComposeFilterType;
      ComposeFilterType::Pointer composer = ComposeFilterType::New();

      ItkDwiType::Pointer itkVectorImagePointer = mitk::DiffusionPropertyHelper::GetItkVectorImage(movingImage);
      for (unsigned int i=0; i<itkVectorImagePointer->GetVectorLength(); ++i)
      {
        itk::ExtractDwiChannelFilter< short >::Pointer filter = itk::ExtractDwiChannelFilter< short >::New();
        filter->SetInput( itkVectorImagePointer);
        filter->SetChannelIndex(i);
        filter->Update();

        mitk::Image::Pointer gradientVolume = mitk::Image::New();
        gradientVolume->InitializeByItk( filter->GetOutput() );
        gradientVolume->SetImportChannel( filter->GetOutput()->GetBufferPointer() );

        mitk::Image::Pointer registered_mitk_image = mitk::ImageMappingHelper::map(gradientVolume, spResultRegistration, false, 0, job->m_spTargetData->GetGeometry(), false, 0, mitk::ImageMappingInterpolator::BSpline_3);

        ITKDiffusionVolumeType::Pointer registered_itk_image = ITKDiffusionVolumeType::New();
        mitk::CastToItkImage(registered_mitk_image, registered_itk_image);
        composer->SetInput(i, registered_itk_image);
      }

      composer->Update();

      image = mitk::GrabItkImageMemory( composer->GetOutput() );
      mitk::DiffusionPropertyHelper::CopyProperties(movingImage, image, true);

      auto reg = spResultRegistration->GetRegistration();
      typedef mitk::DiffusionImageCorrectionFilter CorrectionFilterType;
      CorrectionFilterType::Pointer corrector = CorrectionFilterType::New();
      corrector->SetImage( image );
      corrector->CorrectDirections( mitk::MITKRegistrationHelper::getAffineMatrix(reg, false)->GetMatrix().GetVnlMatrix() );
    }
  }

  if (mitk::DiffusionPropertyHelper::IsDiffusionWeightedImage(image))
    mitk::DiffusionPropertyHelper::InitializeImage( image );

  mitk::DataNode::Pointer resultNode = mitk::DataNode::New();
  resultNode->SetData(image);

  if (m_MovingImageNode.IsNotNull())
  {
    m_MovingImageNode->SetVisibility(false);
    QString name = m_MovingImageNode->GetName().c_str();
    if (m_RegistrationType==0)
      resultNode->SetName((name+"_registered (rigid)").toStdString().c_str());
    else
      resultNode->SetName((name+"_registered (affine)").toStdString().c_str());
  }
  else
  {
    if (m_RegistrationType==0)
      resultNode->SetName("Registered (rigid)");
    else
      resultNode->SetName("Registered (affine)");
  }
//  resultNode->SetOpacity(0.6);
//  resultNode->SetColor(0.0, 0.0, 1.0);
  GetDataStorage()->Add(resultNode);

  mitk::RenderingManager::GetInstance()->InitializeViews( resultNode->GetData()->GetTimeGeometry(),
                                                          mitk::RenderingManager::REQUEST_UPDATE_ALL,
                                                          true);

  if (m_Controls->m_RegOutputBox->isChecked())
  {
    mitk::DataNode::Pointer registration_node = mitk::DataNode::New();
    registration_node->SetData(spResultRegistration);
    if (m_RegistrationType==0)
      registration_node->SetName("Registration Object (rigid)");
    else
      registration_node->SetName("Registration Object (affine)");
    GetDataStorage()->Add(registration_node, resultNode);
  }

  this->GetRenderWindowPart()->RequestUpdate();

  m_Controls->m_RegistrationStartButton->setEnabled(true);
  m_Controls->m_RegistrationStartButton->setText("Start Registration");

  m_MovingImageNode = nullptr;

  TractoChanged();
}