void
{
  std::cerr << "\n";
  std::cerr << "Usage: DRR <options> [input]\n";
  std::cerr << "  calculates the Digitally Reconstructed Radiograph from a "
               "volume. \n\n";
  std::cerr << " where <options> is one or more of the following:\n\n";
  std::cerr << "  <-h>                    Display (this) usage information\n";
  std::cerr << "  <-v>                    Verbose output [default: no]\n";
  std::cerr << "  <-res float float>      Pixel spacing of the output image "
               "[default: "
               "1x1mm]  \n";
  std::cerr << "  <-size int int>         Dimension of the output image "
               "[default: 501x501]  \n";
  std::cerr
    << "  <-sid float>            Distance of ray source (focal point) "
       "[default: 400mm]\n";
  std::cerr
    << "  <-t float float float>  Translation parameter of the camera \n";
  std::cerr
    << "  <-rx float>             Rotation around x,y,z axis in degrees \n";
  std::cerr << "  <-ry float>\n";
  std::cerr << "  <-rz float>\n";
  std::cerr << "  <-normal float float>   The 2D projection normal position "
               "[default: 0x0mm]\n";
  std::cerr
    << "  <-cor float float float> The centre of rotation relative to centre "
       "of volume\n";
  std::cerr << "  <-threshold float>      Threshold [default: 0]\n";
  std::cerr << "  <-o file>               Output image filename\n\n";
  std::cerr << "                          by  thomas@hartkens.de\n";
  std::cerr << "                          and john.hipwell@kcl.ac.uk (CISG "
               "London)\n\n";
  exit(1);
}
 
int
main(int argc, char * argv[])
{
  char * input_name = nullptr;
  char * output_name = nullptr;
 
  bool ok;
  bool verbose = false;
 
  float rx = 0.;
  float ry = 0.;
  float rz = 0.;
 
  float tx = 0.;
  float ty = 0.;
  float tz = 0.;
 
  float cx = 0.;
  float cy = 0.;
  float cz = 0.;
 
  float sid = 400.;
 
  float sx = 1.;
  float sy = 1.;
 
  int dx = 501;
  int dy = 501;
 
  float o2Dx = 0;
  float o2Dy = 0;
 
  double threshold = 0;
 
 
  
 
  while (argc > 1)
  {
    ok = false;
 
    if ((ok == false) && (strcmp(argv[1], "-h") == 0))
    {
      argc--;
      argv++;
      ok = true;
    }
 
    if ((ok == false) && (strcmp(argv[1], "-v") == 0))
    {
      argc--;
      argv++;
      ok = true;
      verbose = true;
    }
 
    if ((ok == false) && (strcmp(argv[1], "-rx") == 0))
    {
      argc--;
      argv++;
      ok = true;
      rx = std::stod(argv[1]);
      argc--;
      argv++;
    }
 
    if ((ok == false) && (strcmp(argv[1], "-ry") == 0))
    {
      argc--;
      argv++;
      ok = true;
      ry = std::stod(argv[1]);
      argc--;
      argv++;
    }
 
    if ((ok == false) && (strcmp(argv[1], "-rz") == 0))
    {
      argc--;
      argv++;
      ok = true;
      rz = std::stod(argv[1]);
      argc--;
      argv++;
    }
 
    if ((ok == false) && (strcmp(argv[1], "-threshold") == 0))
    {
      argc--;
      argv++;
      ok = true;
      threshold = std::stod(argv[1]);
      argc--;
      argv++;
    }
 
    if ((ok == false) && (strcmp(argv[1], "-t") == 0))
    {
      argc--;
      argv++;
      ok = true;
      tx = std::stod(argv[1]);
      argc--;
      argv++;
      ty = std::stod(argv[1]);
      argc--;
      argv++;
      tz = std::stod(argv[1]);
      argc--;
      argv++;
    }
 
    if ((ok == false) && (strcmp(argv[1], "-cor") == 0))
    {
      argc--;
      argv++;
      ok = true;
      cx = std::stod(argv[1]);
      argc--;
      argv++;
      cy = std::stod(argv[1]);
      argc--;
      argv++;
      cz = std::stod(argv[1]);
      argc--;
      argv++;
    }
 
    if ((ok == false) && (strcmp(argv[1], "-res") == 0))
    {
      argc--;
      argv++;
      ok = true;
      sx = std::stod(argv[1]);
      argc--;
      argv++;
      sy = std::stod(argv[1]);
      argc--;
      argv++;
    }
 
    if ((ok == false) && (strcmp(argv[1], "-size") == 0))
    {
      argc--;
      argv++;
      ok = true;
      dx = std::stoi(argv[1]);
      argc--;
      argv++;
      dy = std::stoi(argv[1]);
      argc--;
      argv++;
    }
 
    if ((ok == false) && (strcmp(argv[1], "-sid") == 0))
    {
      argc--;
      argv++;
      ok = true;
      sid = std::stod(argv[1]);
      argc--;
      argv++;
    }
 
    if ((ok == false) && (strcmp(argv[1], "-normal") == 0))
    {
      argc--;
      argv++;
      ok = true;
      o2Dx = std::stod(argv[1]);
      argc--;
      argv++;
      o2Dy = std::stod(argv[1]);
      argc--;
      argv++;
    }
 
    if ((ok == false) && (strcmp(argv[1], "-o") == 0))
    {
      argc--;
      argv++;
      ok = true;
      output_name = argv[1];
      argc--;
      argv++;
    }
 
    if (ok == false)
    {
 
      if (input_name == nullptr)
      {
        input_name = argv[1];
        argc--;
        argv++;
      }
 
      else
      {
        std::cerr << "ERROR: Can not parse argument " << argv[1] << std::endl;
      }
    }
  }
 
  if (verbose)
  {
    if (input_name)
      std::cout << "Input image: " << input_name << std::endl;
    if (output_name)
      std::cout << "Output image: " << output_name << std::endl;
  }
 
  
  
  
  
  
  
 
  
  using InputPixelType = short;
  using OutputPixelType = unsigned char;
 
 
  InputImageType::Pointer image;
  
 
  
  
  
  
  
  
 
  if (input_name)
  {
 
    auto reader = ReaderType::New();
    reader->SetFileName(input_name);
 
 
    try
    {
      reader->Update();
    }
    {
      std::cerr << "ERROR: ExceptionObject caught !" << std::endl;
      std::cerr << err << std::endl;
      return EXIT_FAILURE;
    }
 
    image = reader->GetOutput();
  }
  else
  { 
 
    image = InputImageType::New();
 
    InputImageType::SpacingType spacing;
    spacing[0] = 3.;
    spacing[1] = 3.;
    spacing[2] = 3.;
    image->SetSpacing(spacing);
 
    InputImageType::PointType origin;
    origin[0] = 0.;
    origin[1] = 0.;
    origin[2] = 0.;
    image->SetOrigin(origin);
 
    InputImageType::IndexType start;
 
    start[0] = 0; 
    start[1] = 0; 
    start[2] = 0; 
 
    InputImageType::SizeType size;
 
    size[0] = 61; 
    size[1] = 61; 
    size[2] = 61; 
 
    InputImageType::RegionType region;
 
    region.SetSize(size);
    region.SetIndex(start);
 
    image->SetRegions(region);
    image->Allocate(true); 
 
    image->Update();
 
 
    IteratorType iterate(image, image->GetLargestPossibleRegion());
 
    while (!iterate.IsAtEnd())
    {
 
      InputImageType::IndexType idx = iterate.GetIndex();
 
      if ((idx[0] >= 6) && (idx[0] <= 54) && (idx[1] >= 6) &&
          (idx[1] <= 54) && (idx[2] >= 6) && (idx[2] <= 54)
 
          && ((((idx[0] <= 11) || (idx[0] >= 49)) &&
               ((idx[1] <= 11) || (idx[1] >= 49)))
 
              || (((idx[0] <= 11) || (idx[0] >= 49)) &&
                  ((idx[2] <= 11) || (idx[2] >= 49)))
 
              || (((idx[1] <= 11) || (idx[1] >= 49)) &&
                  ((idx[2] <= 11) || (idx[2] >= 49)))))
      {
        iterate.Set(10);
      }
 
      else if ((idx[0] >= 18) && (idx[0] <= 42) && (idx[1] >= 18) &&
               (idx[1] <= 42) && (idx[2] >= 18) && (idx[2] <= 42)
 
               && ((((idx[0] <= 23) || (idx[0] >= 37)) &&
                    ((idx[1] <= 23) || (idx[1] >= 37)))
 
                   || (((idx[0] <= 23) || (idx[0] >= 37)) &&
                       ((idx[2] <= 23) || (idx[2] >= 37)))
 
                   || (((idx[1] <= 23) || (idx[1] >= 37)) &&
                       ((idx[2] <= 23) || (idx[2] >= 37)))))
      {
        iterate.Set(60);
      }
 
      else if ((idx[0] == 30) && (idx[1] == 30) && (idx[2] == 30))
      {
        iterate.Set(100);
      }
 
      ++iterate;
    }
 
 
#ifdef WRITE_CUBE_IMAGE_TO_FILE
    const char * filename = "cube.gipl";
    auto writer = WriterType::New();
 
    writer->SetFileName(filename);
    writer->SetInput(image);
 
    try
    {
      std::cout << "Writing image: " << filename << std::endl;
      writer->Update();
    }
    {
 
      std::cerr << "ERROR: ExceptionObject caught !" << std::endl;
      std::cerr << err << std::endl;
      return EXIT_FAILURE;
    }
#endif
  }
 
 
  
 
  if (verbose)
  {
    unsigned int                      i;
    const InputImageType::SpacingType spacing = image->GetSpacing();
    std::cout << std::endl << "Input ";
 
    const InputImageType::RegionType region = image->GetBufferedRegion();
    region.Print(std::cout);
 
    std::cout << "  Resolution: [";
    {
      std::cout << spacing[i];
      if (i < Dimension - 1)
        std::cout << ", ";
    }
    std::cout << "]" << std::endl;
 
    const InputImageType::PointType origin = image->GetOrigin();
    std::cout << "  Origin: [";
    {
      std::cout << origin[i];
      if (i < Dimension - 1)
        std::cout << ", ";
    }
    std::cout << "]" << std::endl << std::endl;
  }
 
  
  
  
  
  
  
  
  
  
 
  
 
  auto filter = FilterType::New();
 
  filter->SetInput(image);
  filter->SetDefaultPixelValue(0);
  
 
  
  
  
  
  
  
  
 
  
 
  auto transform = TransformType::New();
 
  transform->SetComputeZYX(true);
 
  TransformType::OutputVectorType translation;
 
  translation[0] = tx;
  translation[1] = ty;
  translation[2] = tz;
 
  
  const double dtr = (std::atan(1.0) * 4.0) / 180.0;
 
  transform->SetTranslation(translation);
  transform->SetRotation(dtr * rx, dtr * ry, dtr * rz);
 
  InputImageType::PointType   imOrigin = image->GetOrigin();
  InputImageType::SpacingType imRes = image->GetSpacing();
 
  using InputImageRegionType = InputImageType::RegionType;
  using InputImageSizeType = InputImageRegionType::SizeType;
 
  const InputImageRegionType imRegion = image->GetBufferedRegion();
  InputImageSizeType         imSize = imRegion.GetSize();
 
  imOrigin[0] += imRes[0] * static_cast<double>(imSize[0]) / 2.0;
  imOrigin[1] += imRes[1] * static_cast<double>(imSize[1]) / 2.0;
  imOrigin[2] += imRes[2] * static_cast<double>(imSize[2]) / 2.0;
 
  TransformType::InputPointType center;
  center[0] = cx + imOrigin[0];
  center[1] = cy + imOrigin[1];
  center[2] = cz + imOrigin[2];
 
  transform->SetCenter(center);
 
  if (verbose)
  {
    std::cout << "Image size: " << imSize[0] << ", " << imSize[1] << ", "
              << imSize[2] << std::endl
              << "   resolution: " << imRes[0] << ", " << imRes[1] << ", "
              << imRes[2] << std::endl
              << "   origin: " << imOrigin[0] << ", " << imOrigin[1] << ", "
              << imOrigin[2] << std::endl
              << "   center: " << center[0] << ", " << center[1] << ", "
              << center[2] << std::endl
              << "Transform: " << transform << std::endl;
  }
  
 
  
  
  
  
  
  
  
  
  
 
  
  using InterpolatorType =
  auto interpolator = InterpolatorType::New();
  interpolator->SetTransform(transform);
  
 
  
  
  
  
  
  
 
  
  interpolator->SetThreshold(threshold);
  
 
  
  
  
  
  
  
  
  
  
  
 
  
  InterpolatorType::InputPointType focalpoint;
 
  focalpoint[0] = imOrigin[0];
  focalpoint[1] = imOrigin[1];
  focalpoint[2] = imOrigin[2] - sid / 2.;
 
  interpolator->SetFocalPoint(focalpoint);
  
 
  if (verbose)
  {
    std::cout << "Focal Point: " << focalpoint[0] << ", " << focalpoint[1]
              << ", " << focalpoint[2] << std::endl;
  }
 
  
  
  
  
  
  
 
  
  interpolator->Print(std::cout);
 
  filter->SetInterpolator(interpolator);
  filter->SetTransform(transform);
  
 
  
  
  
  
  
  
 
  
 
  
  InputImageType::SizeType size;
 
  size[0] = dx; 
  size[1] = dy; 
  size[2] = 1;  
 
  filter->SetSize(size);
 
  InputImageType::SpacingType spacing;
 
  spacing[0] = sx;  
  spacing[1] = sy;  
  spacing[2] = 1.0; 
 
  filter->SetOutputSpacing(spacing);
 
  
 
  if (verbose)
  {
    std::cout << "Output image size: " << size[0] << ", " << size[1] << ", "
              << size[2] << std::endl;
 
    std::cout << "Output image spacing: " << spacing[0] << ", " << spacing[1]
              << ", " << spacing[2] << std::endl;
  }
 
  
  
  
  
  
  
  
  
  
 
  
 
 
  origin[0] = imOrigin[0] + o2Dx - sx * (static_cast<double>(dx) - 1.) / 2.;
  origin[1] = imOrigin[1] + o2Dy - sy * (static_cast<double>(dy) - 1.) / 2.;
  origin[2] = imOrigin[2] + sid / 2.;
 
  filter->SetOutputOrigin(origin);
  
 
  if (verbose)
  {
    std::cout << "Output image origin: " << origin[0] << ", " << origin[1]
              << ", " << origin[2] << std::endl;
  }
 
  
 
  if (output_name)
  {
 
    
    
    
    
    
    
 
    
    using RescaleFilterType =
    auto rescaler = RescaleFilterType::New();
    rescaler->SetOutputMinimum(0);
    rescaler->SetOutputMaximum(255);
    rescaler->SetInput(filter->GetOutput());
 
    auto writer = WriterType::New();
 
    writer->SetFileName(output_name);
    writer->SetInput(rescaler->GetOutput());
 
    try
    {
      std::cout << "Writing image: " << output_name << std::endl;
      writer->Update();
    }
    {
      std::cerr << "ERROR: ExceptionObject caught !" << std::endl;
      std::cerr << err << std::endl;
    }
    
  }
  else
  {
    filter->Update();
  }
 
  return EXIT_SUCCESS;
}
Standard exception handling object.
Data source that reads image data from a single file.
Writes image data to a single file.
A multi-dimensional iterator templated over image type that walks pixels within a region and is speci...
Templated n-dimensional image class.
Projective interpolation of an image at specified positions.
Resample an image via a coordinate transform.
Applies a linear transformation to the intensity levels of the input Image.
constexpr unsigned int Dimension