AlignFrames
Repository source: AlignFrames
Other languages
See (CSharp)
Question
If you have a question about this example, please use the VTK Discourse Forum
Code¶
AlignFrames.cxx
#include <vtkLandmarkTransform.h>
#include <vtkMath.h>
#include <vtkNew.h>
#include <vtkPoints.h>
#include <vtkPolyData.h>
#include <vtkTransform.h>
#include <vtkTransformFilter.h>
#include <vtkVertexGlyphFilter.h>
#include <vtkXMLPolyDataWriter.h>
namespace {
struct Frame
{
Frame(float o[3], float x[3], float y[3], float z[3])
{
this->SetOrigin(o);
this->SetXDirection(x);
this->SetYDirection(y);
this->SetZDirection(z);
std::cout << "Origin: " << this->origin[0] << " " << this->origin[1] << " "
<< this->origin[2] << std::endl;
std::cout << "xDirection: " << this->xDirection[0] << " "
<< this->xDirection[1] << " " << this->xDirection[2] << std::endl;
std::cout << "yDirection: " << this->yDirection[0] << " "
<< this->yDirection[1] << " " << this->yDirection[2] << std::endl;
std::cout << "zDirection: " << this->zDirection[0] << " "
<< this->zDirection[1] << " " << this->zDirection[2] << std::endl;
}
void ApplyTransform(vtkTransform* transform, std::string const& filename)
{
vtkNew<vtkPolyData> polydata;
CreatePolydata(polydata);
vtkNew<vtkTransformFilter> transformFilter;
transformFilter->SetInputData(polydata);
transformFilter->SetTransform(transform);
transformFilter->Update();
vtkNew<vtkXMLPolyDataWriter> writer;
writer->SetFileName(filename.c_str());
writer->SetInputConnection(transformFilter->GetOutputPort());
writer->Write();
}
void CreatePolydata(vtkPolyData* polydata)
{
vtkNew<vtkPoints> points;
points->InsertNextPoint(this->origin);
float x[3];
vtkMath::Add(this->origin, this->xDirection, x);
points->InsertNextPoint(x);
float y[3];
vtkMath::Add(this->origin, this->yDirection, y);
points->InsertNextPoint(y);
float z[3];
vtkMath::Add(this->origin, this->zDirection, z);
points->InsertNextPoint(z);
polydata->SetPoints(points);
vtkNew<vtkVertexGlyphFilter> vertexGlyphFilter;
vertexGlyphFilter->AddInputData(polydata);
vertexGlyphFilter->Update();
polydata->ShallowCopy(vertexGlyphFilter->GetOutput());
}
void Write(std::string const& filename)
{
vtkNew<vtkPolyData> polydata;
CreatePolydata(polydata);
vtkNew<vtkXMLPolyDataWriter> writer;
writer->SetFileName(filename.c_str());
writer->SetInputData(polydata);
writer->Write();
}
float origin[3];
float xDirection[3];
float yDirection[3];
float zDirection[3];
void SetOrigin(const float o[3])
{
this->origin[0] = o[0];
this->origin[1] = o[1];
this->origin[2] = o[2];
}
void SetXDirection(float direction[3])
{
vtkMath::Normalize(direction);
this->xDirection[0] = direction[0];
this->xDirection[1] = direction[1];
this->xDirection[2] = direction[2];
}
void SetYDirection(float direction[3])
{
vtkMath::Normalize(direction);
this->yDirection[0] = direction[0];
this->yDirection[1] = direction[1];
this->yDirection[2] = direction[2];
}
void SetZDirection(float direction[3])
{
vtkMath::Normalize(direction);
this->zDirection[0] = direction[0];
this->zDirection[1] = direction[1];
this->zDirection[2] = direction[2];
}
};
void AlignFrames(Frame sourceFrame, Frame destinationFrame,
vtkTransform* transform);
} // namespace
int main(int, char*[])
{
float frame1origin[3] = {0, 0, 0};
float frame1XDirection[3] = {1, 0, 0};
float frame1YDirection[3] = {0, 1, 0};
std::cout << frame1YDirection[0] << " " << frame1YDirection[1] << " "
<< frame1YDirection[2] << std::endl;
float frame1ZDirection[3] = {0, 0, 1};
Frame frame1(frame1origin, frame1XDirection, frame1YDirection,
frame1ZDirection);
frame1.Write("frame1.vtp");
float frame2origin[3] = {0, 0, 0};
float frame2XDirection[3] = {.707f, .707f, 0};
float frame2YDirection[3] = {-.707f, .707f, 0};
float frame2ZDirection[3] = {0, 0, 1};
Frame frame2(frame2origin, frame2XDirection, frame2YDirection,
frame2ZDirection);
frame2.Write("frame2.vtp");
vtkNew<vtkTransform> transform;
AlignFrames(frame2, frame1, transform); // Brings frame2 to frame1
// std::cout << *transform << std::endl;
frame2.ApplyTransform(transform, "transformed.vtp");
return EXIT_SUCCESS;
}
namespace {
void AlignFrames(Frame sourceFrame, Frame targetFrame, vtkTransform* transform)
{
// This function takes two frames and finds the matrix M between them.
vtkNew<vtkLandmarkTransform> landmarkTransform;
// Setup source points
vtkNew<vtkPoints> sourcePoints;
sourcePoints->InsertNextPoint(sourceFrame.origin);
float sourceX[3];
vtkMath::Add(sourceFrame.origin, sourceFrame.xDirection, sourceX);
sourcePoints->InsertNextPoint(sourceX);
float sourceY[3];
vtkMath::Add(sourceFrame.origin, sourceFrame.yDirection, sourceY);
sourcePoints->InsertNextPoint(sourceY);
float sourceZ[3];
vtkMath::Add(sourceFrame.origin, sourceFrame.zDirection, sourceZ);
sourcePoints->InsertNextPoint(sourceZ);
// Setup target points
vtkNew<vtkPoints> targetPoints;
targetPoints->InsertNextPoint(targetFrame.origin);
float targetX[3];
vtkMath::Add(targetFrame.origin, targetFrame.xDirection, targetX);
targetPoints->InsertNextPoint(targetX);
float targetY[3];
vtkMath::Add(targetFrame.origin, targetFrame.yDirection, targetY);
targetPoints->InsertNextPoint(targetY);
float targetZ[3];
vtkMath::Add(targetFrame.origin, targetFrame.zDirection, targetZ);
targetPoints->InsertNextPoint(targetZ);
landmarkTransform->SetSourceLandmarks(sourcePoints);
landmarkTransform->SetTargetLandmarks(targetPoints);
landmarkTransform->SetModeToRigidBody();
landmarkTransform->Update();
vtkMatrix4x4* M = landmarkTransform->GetMatrix();
transform->SetMatrix(M);
}
} // namespace
CMakeLists.txt¶
cmake_minimum_required(VERSION 3.12 FATAL_ERROR)
project(AlignFrames)
find_package(VTK COMPONENTS
CommonCore
CommonDataModel
CommonTransforms
FiltersGeneral
IOXML
)
if (NOT VTK_FOUND)
message(FATAL_ERROR "AlignFrames: Unable to find the VTK build folder.")
endif()
# Prevent a "command line is too long" failure in Windows.
set(CMAKE_NINJA_FORCE_RESPONSE_FILE "ON" CACHE BOOL "Force Ninja to use response files.")
add_executable(AlignFrames MACOSX_BUNDLE AlignFrames.cxx )
target_link_libraries(AlignFrames PRIVATE ${VTK_LIBRARIES}
)
# vtk_module_autoinit is needed
vtk_module_autoinit(
TARGETS AlignFrames
MODULES ${VTK_LIBRARIES}
)
Download and Build AlignFrames¶
Click here to download AlignFrames and its CMakeLists.txt file. Once the tarball AlignFrames.tar has been downloaded and extracted,
cd AlignFrames/build
If VTK is installed:
cmake ..
If VTK is not installed but compiled on your system, you will need to specify the path to your VTK build:
cmake -DVTK_DIR:PATH=/home/me/vtk_build ..
Build the project:
make
and run it:
./AlignFrames
WINDOWS USERS
Be sure to add the VTK bin directory to your path. This will resolve the VTK dll's at run time.