Cloned library of VTK-5.0.0 with extra build files for internal package management.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

735 lines
20 KiB

2 years ago
/*=========================================================================
Program: Visualization Toolkit
Module: $RCSfile: vtkSphereWidget.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 "vtkSphereWidget.h"
#include "vtkActor.h"
#include "vtkAssemblyNode.h"
#include "vtkAssemblyPath.h"
#include "vtkCallbackCommand.h"
#include "vtkCamera.h"
#include "vtkCellPicker.h"
#include "vtkDoubleArray.h"
#include "vtkMath.h"
#include "vtkObjectFactory.h"
#include "vtkPolyData.h"
#include "vtkPolyDataMapper.h"
#include "vtkProperty.h"
#include "vtkRenderWindowInteractor.h"
#include "vtkRenderer.h"
#include "vtkSphere.h"
#include "vtkSphereSource.h"
vtkCxxRevisionMacro(vtkSphereWidget, "$Revision: 1.1 $");
vtkStandardNewMacro(vtkSphereWidget);
vtkSphereWidget::vtkSphereWidget()
{
this->State = vtkSphereWidget::Start;
this->EventCallbackCommand->SetCallback(vtkSphereWidget::ProcessEvents);
this->Representation = VTK_SPHERE_WIREFRAME;
//Build the representation of the widget
// Represent the sphere
this->SphereSource = vtkSphereSource::New();
this->SphereSource->SetThetaResolution(16);
this->SphereSource->SetPhiResolution(8);
this->SphereSource->LatLongTessellationOn();
this->SphereMapper = vtkPolyDataMapper::New();
this->SphereMapper->SetInput(this->SphereSource->GetOutput());
this->SphereActor = vtkActor::New();
this->SphereActor->SetMapper(this->SphereMapper);
// controls
this->Translation = 1;
this->Scale = 1;
// handles
this->HandleVisibility = 0;
this->HandleDirection[0] = 1.0;
this->HandleDirection[1] = 0.0;
this->HandleDirection[2] = 0.0;
this->HandleSource = vtkSphereSource::New();
this->HandleSource->SetThetaResolution(16);
this->HandleSource->SetPhiResolution(8);
this->HandleMapper = vtkPolyDataMapper::New();
this->HandleMapper->SetInput(this->HandleSource->GetOutput());
this->HandleActor = vtkActor::New();
this->HandleActor->SetMapper(this->HandleMapper);
// Define the point coordinates
double bounds[6];
bounds[0] = -0.5;
bounds[1] = 0.5;
bounds[2] = -0.5;
bounds[3] = 0.5;
bounds[4] = -0.5;
bounds[5] = 0.5;
// Initial creation of the widget, serves to initialize it
this->PlaceWidget(bounds);
//Manage the picking stuff
this->Picker = vtkCellPicker::New();
this->Picker->SetTolerance(0.005); //need some fluff
this->Picker->AddPickList(this->SphereActor);
this->Picker->AddPickList(this->HandleActor);
this->Picker->PickFromListOn();
// Set up the initial properties
this->SphereProperty = NULL;
this->SelectedSphereProperty = NULL;
this->HandleProperty = NULL;
this->SelectedHandleProperty = NULL;
this->CreateDefaultProperties();
}
vtkSphereWidget::~vtkSphereWidget()
{
this->SphereActor->Delete();
this->SphereMapper->Delete();
this->SphereSource->Delete();
this->Picker->Delete();
this->HandleSource->Delete();
this->HandleMapper->Delete();
this->HandleActor->Delete();
if ( this->SphereProperty )
{
this->SphereProperty->Delete();
}
if ( this->SelectedSphereProperty )
{
this->SelectedSphereProperty->Delete();
}
if ( this->HandleProperty )
{
this->HandleProperty->Delete();
}
if ( this->SelectedHandleProperty )
{
this->SelectedHandleProperty->Delete();
}
}
void vtkSphereWidget::SetEnabled(int enabling)
{
if ( ! this->Interactor )
{
vtkErrorMacro(<<"The interactor must be set prior to enabling/disabling widget");
return;
}
if ( enabling ) //----------------------------------------------------------
{
vtkDebugMacro(<<"Enabling sphere widget");
if ( this->Enabled ) //already enabled, just return
{
return;
}
if ( ! this->CurrentRenderer )
{
this->SetCurrentRenderer(
this->Interactor->FindPokedRenderer(
this->Interactor->GetLastEventPosition()[0],
this->Interactor->GetLastEventPosition()[1]));
if (this->CurrentRenderer == NULL)
{
return;
}
}
this->Enabled = 1;
// listen for the following events
vtkRenderWindowInteractor *i = this->Interactor;
i->AddObserver(vtkCommand::MouseMoveEvent,
this->EventCallbackCommand, this->Priority);
i->AddObserver(vtkCommand::LeftButtonPressEvent,
this->EventCallbackCommand, this->Priority);
i->AddObserver(vtkCommand::LeftButtonReleaseEvent,
this->EventCallbackCommand, this->Priority);
i->AddObserver(vtkCommand::RightButtonPressEvent,
this->EventCallbackCommand, this->Priority);
i->AddObserver(vtkCommand::RightButtonReleaseEvent,
this->EventCallbackCommand, this->Priority);
// Add the sphere
this->CurrentRenderer->AddActor(this->SphereActor);
this->SphereActor->SetProperty(this->SphereProperty);
this->CurrentRenderer->AddActor(this->HandleActor);
this->HandleActor->SetProperty(this->HandleProperty);
this->SelectRepresentation();
this->SizeHandles();
this->InvokeEvent(vtkCommand::EnableEvent,NULL);
}
else //disabling----------------------------------------------------------
{
vtkDebugMacro(<<"Disabling sphere widget");
if ( ! this->Enabled ) //already disabled, just return
{
return;
}
this->Enabled = 0;
// don't listen for events any more
this->Interactor->RemoveObserver(this->EventCallbackCommand);
// turn off the sphere
this->CurrentRenderer->RemoveActor(this->SphereActor);
this->CurrentRenderer->RemoveActor(this->HandleActor);
this->InvokeEvent(vtkCommand::DisableEvent,NULL);
this->SetCurrentRenderer(NULL);
}
this->Interactor->Render();
}
void vtkSphereWidget::ProcessEvents(vtkObject* vtkNotUsed(object),
unsigned long event,
void* clientdata,
void* vtkNotUsed(calldata))
{
vtkSphereWidget* self = reinterpret_cast<vtkSphereWidget *>( clientdata );
//okay, let's do the right thing
switch(event)
{
case vtkCommand::LeftButtonPressEvent:
self->OnLeftButtonDown();
break;
case vtkCommand::LeftButtonReleaseEvent:
self->OnLeftButtonUp();
break;
case vtkCommand::RightButtonPressEvent:
self->OnRightButtonDown();
break;
case vtkCommand::RightButtonReleaseEvent:
self->OnRightButtonUp();
break;
case vtkCommand::MouseMoveEvent:
self->OnMouseMove();
break;
}
}
void vtkSphereWidget::PrintSelf(ostream& os, vtkIndent indent)
{
this->Superclass::PrintSelf(os,indent);
os << indent << "Sphere Representation: ";
if ( this->Representation == VTK_SPHERE_OFF )
{
os << "Off\n";
}
else if ( this->Representation == VTK_SPHERE_WIREFRAME )
{
os << "Wireframe\n";
}
else //if ( this->Representation == VTK_SPHERE_SURFACE )
{
os << "Surface\n";
}
if ( this->SphereProperty )
{
os << indent << "Sphere Property: " << this->SphereProperty << "\n";
}
else
{
os << indent << "Sphere Property: (none)\n";
}
if ( this->SelectedSphereProperty )
{
os << indent << "Selected Sphere Property: "
<< this->SelectedSphereProperty << "\n";
}
else
{
os << indent << "Selected Sphere Property: (none)\n";
}
if ( this->HandleProperty )
{
os << indent << "Handle Property: " << this->HandleProperty << "\n";
}
else
{
os << indent << "Handle Property: (none)\n";
}
if ( this->SelectedHandleProperty )
{
os << indent << "Selected Handle Property: "
<< this->SelectedHandleProperty << "\n";
}
else
{
os << indent << "Selected Handle Property: (none)\n";
}
os << indent << "Translation: " << (this->Translation ? "On\n" : "Off\n");
os << indent << "Scale: " << (this->Scale ? "On\n" : "Off\n");
os << indent << "Handle Visibility: "
<< (this->HandleVisibility ? "On\n" : "Off\n");
os << indent << "Handle Direction: (" << this->HandleDirection[0] << ", "
<< this->HandleDirection[1] << ", "
<< this->HandleDirection[2] << ")\n";
os << indent << "Handle Position: (" << this->HandlePosition[0] << ", "
<< this->HandlePosition[1] << ", "
<< this->HandlePosition[2] << ")\n";
int thetaRes = this->SphereSource->GetThetaResolution();
int phiRes = this->SphereSource->GetPhiResolution();
double *center = this->SphereSource->GetCenter();
double r = this->SphereSource->GetRadius();
os << indent << "Theta Resolution: " << thetaRes << "\n";
os << indent << "Phi Resolution: " << phiRes << "\n";
os << indent << "Center: (" << center[0] << ", "
<< center[1] << ", " << center[2] << ")\n";
os << indent << "Radius: " << r << "\n";
}
void vtkSphereWidget::SelectRepresentation()
{
if ( ! this->HandleVisibility )
{
this->CurrentRenderer->RemoveActor(this->HandleActor);
}
if ( this->Representation == VTK_SPHERE_OFF )
{
this->CurrentRenderer->RemoveActor(this->SphereActor);
}
else if ( this->Representation == VTK_SPHERE_WIREFRAME )
{
this->CurrentRenderer->RemoveActor(this->SphereActor);
this->CurrentRenderer->AddActor(this->SphereActor);
this->SphereProperty->SetRepresentationToWireframe();
this->SelectedSphereProperty->SetRepresentationToWireframe();
}
else //if ( this->Representation == VTK_SPHERE_SURFACE )
{
this->CurrentRenderer->RemoveActor(this->SphereActor);
this->CurrentRenderer->AddActor(this->SphereActor);
this->SphereProperty->SetRepresentationToSurface();
this->SelectedSphereProperty->SetRepresentationToSurface();
}
}
void vtkSphereWidget::GetSphere(vtkSphere *sphere)
{
sphere->SetRadius(this->SphereSource->GetRadius());
sphere->SetCenter(this->SphereSource->GetCenter());
}
void vtkSphereWidget::HighlightSphere(int highlight)
{
if ( highlight )
{
this->ValidPick = 1;
this->Picker->GetPickPosition(this->LastPickPosition);
this->SphereActor->SetProperty(this->SelectedSphereProperty);
}
else
{
this->SphereActor->SetProperty(this->SphereProperty);
}
}
void vtkSphereWidget::HighlightHandle(int highlight)
{
if ( highlight )
{
this->ValidPick = 1;
this->Picker->GetPickPosition(this->LastPickPosition);
this->HandleActor->SetProperty(this->SelectedHandleProperty);
}
else
{
this->HandleActor->SetProperty(this->HandleProperty);
}
}
void vtkSphereWidget::OnLeftButtonDown()
{
if (!this->Interactor)
{
return;
}
int X = this->Interactor->GetEventPosition()[0];
int Y = this->Interactor->GetEventPosition()[1];
// Okay, make sure that the pick is in the current renderer
if (!this->CurrentRenderer || !this->CurrentRenderer->IsInViewport(X, Y))
{
this->State = vtkSphereWidget::Outside;
return;
}
// Okay, we can process this. Try to pick handles first;
// if no handles picked, then try to pick the sphere.
vtkAssemblyPath *path;
this->Picker->Pick(X,Y,0.0,this->CurrentRenderer);
path = this->Picker->GetPath();
if ( path == NULL )
{
this->State = vtkSphereWidget::Outside;
return;
}
else if (path->GetFirstNode()->GetViewProp() == this->SphereActor )
{
this->State = vtkSphereWidget::Moving;
this->HighlightSphere(1);
}
else if (path->GetFirstNode()->GetViewProp() == this->HandleActor )
{
this->State = vtkSphereWidget::Positioning;
this->HighlightHandle(1);
}
this->EventCallbackCommand->SetAbortFlag(1);
this->StartInteraction();
this->InvokeEvent(vtkCommand::StartInteractionEvent,NULL);
this->Interactor->Render();
}
void vtkSphereWidget::OnMouseMove()
{
// See whether we're active
if ( this->State == vtkSphereWidget::Outside ||
this->State == vtkSphereWidget::Start )
{
return;
}
if (!this->Interactor)
{
return;
}
int X = this->Interactor->GetEventPosition()[0];
int Y = this->Interactor->GetEventPosition()[1];
// Do different things depending on state
// Calculations everybody does
double focalPoint[4], pickPoint[4], prevPickPoint[4];
double z;
vtkCamera *camera = this->CurrentRenderer->GetActiveCamera();
if ( !camera )
{
return;
}
// Compute the two points defining the motion vector
camera->GetFocalPoint(focalPoint);
this->ComputeWorldToDisplay(focalPoint[0], focalPoint[1],
focalPoint[2], focalPoint);
z = focalPoint[2];
this->ComputeDisplayToWorld(double(this->Interactor->GetLastEventPosition()[0]),
double(this->Interactor->GetLastEventPosition()[1]),
z,
prevPickPoint);
this->ComputeDisplayToWorld(double(X), double(Y), z, pickPoint);
// Process the motion
if ( this->State == vtkSphereWidget::Moving )
{
this->Translate(prevPickPoint, pickPoint);
}
else if ( this->State == vtkSphereWidget::Scaling )
{
this->ScaleSphere(prevPickPoint, pickPoint, X, Y);
}
else if ( this->State == vtkSphereWidget::Positioning )
{
this->MoveHandle(prevPickPoint, pickPoint, X, Y);
}
// Interact, if desired
this->EventCallbackCommand->SetAbortFlag(1);
this->InvokeEvent(vtkCommand::InteractionEvent,NULL);
this->Interactor->Render();
}
void vtkSphereWidget::OnLeftButtonUp()
{
if ( this->State == vtkSphereWidget::Outside )
{
return;
}
this->State = vtkSphereWidget::Start;
this->HighlightSphere(0);
this->HighlightHandle(0);
this->SizeHandles();
this->EventCallbackCommand->SetAbortFlag(1);
this->EndInteraction();
this->InvokeEvent(vtkCommand::EndInteractionEvent,NULL);
if (this->Interactor)
{
this->Interactor->Render();
}
}
void vtkSphereWidget::OnRightButtonDown()
{
if (!this->Interactor)
{
return;
}
this->State = vtkSphereWidget::Scaling;
int X = this->Interactor->GetEventPosition()[0];
int Y = this->Interactor->GetEventPosition()[1];
// Okay, make sure that the pick is in the current renderer
if (!this->CurrentRenderer || !this->CurrentRenderer->IsInViewport(X, Y))
{
this->State = vtkSphereWidget::Outside;
return;
}
// Okay, we can process this. Try to pick handles first;
// if no handles picked, then pick the bounding box.
vtkAssemblyPath *path;
this->Picker->Pick(X,Y,0.0,this->CurrentRenderer);
path = this->Picker->GetPath();
if ( path == NULL )
{
this->State = vtkSphereWidget::Outside;
this->HighlightSphere(0);
return;
}
else
{
this->HighlightSphere(1);
}
this->EventCallbackCommand->SetAbortFlag(1);
this->StartInteraction();
this->InvokeEvent(vtkCommand::StartInteractionEvent,NULL);
this->Interactor->Render();
}
void vtkSphereWidget::OnRightButtonUp()
{
if ( this->State == vtkSphereWidget::Outside )
{
return;
}
this->State = vtkSphereWidget::Start;
this->HighlightSphere(0);
this->HighlightHandle(0);
this->SizeHandles();
this->EventCallbackCommand->SetAbortFlag(1);
this->EndInteraction();
this->InvokeEvent(vtkCommand::EndInteractionEvent,NULL);
if (this->Interactor)
{
this->Interactor->Render();
}
}
// Loop through all points and translate them
void vtkSphereWidget::Translate(double *p1, double *p2)
{
if ( !this->Translation )
{
return;
}
//Get the motion vector
double v[3];
v[0] = p2[0] - p1[0];
v[1] = p2[1] - p1[1];
v[2] = p2[2] - p1[2];
//int res = this->SphereSource->GetResolution();
double *center = this->SphereSource->GetCenter();
double center1[3];
for (int i=0; i<3; i++)
{
center1[i] = center[i] + v[i];
this->HandlePosition[i] += v[i];
}
this->SphereSource->SetCenter(center1);
this->HandleSource->SetCenter(HandlePosition);
this->SelectRepresentation();
}
void vtkSphereWidget::ScaleSphere(double *p1, double *p2,
int vtkNotUsed(X), int Y)
{
if ( !this->Scale )
{
return;
}
//Get the motion vector
double v[3];
v[0] = p2[0] - p1[0];
v[1] = p2[1] - p1[1];
v[2] = p2[2] - p1[2];
double radius = this->SphereSource->GetRadius();
double *c = this->SphereSource->GetCenter();
// Compute the scale factor
double sf = vtkMath::Norm(v) / radius;
if ( Y > this->Interactor->GetLastEventPosition()[1] )
{
sf = 1.0 + sf;
}
else
{
sf = 1.0 - sf;
}
this->SphereSource->SetRadius(sf*radius);
this->HandlePosition[0] = c[0]+sf*(this->HandlePosition[0]-c[0]);
this->HandlePosition[1] = c[1]+sf*(this->HandlePosition[1]-c[1]);
this->HandlePosition[2] = c[2]+sf*(this->HandlePosition[2]-c[2]);
this->HandleSource->SetCenter(this->HandlePosition);
this->SelectRepresentation();
}
void vtkSphereWidget::MoveHandle(double *p1, double *p2,
int vtkNotUsed(X), int vtkNotUsed(Y))
{
//Get the motion vector
double v[3];
v[0] = p2[0] - p1[0];
v[1] = p2[1] - p1[1];
v[2] = p2[2] - p1[2];
// Compute the new location of the sphere
double *center = this->SphereSource->GetCenter();
double radius = this->SphereSource->GetRadius();
// set the position of the sphere
double p[3];
for (int i=0; i<3; i++)
{
p[i] = this->HandlePosition[i] + v[i];
this->HandleDirection[i] = p[i] - center[i];
}
this->PlaceHandle(center,radius);
this->SelectRepresentation();
}
void vtkSphereWidget::CreateDefaultProperties()
{
if ( ! this->SphereProperty )
{
this->SphereProperty = vtkProperty::New();
}
if ( ! this->SelectedSphereProperty )
{
this->SelectedSphereProperty = vtkProperty::New();
}
if ( ! this->HandleProperty )
{
this->HandleProperty = vtkProperty::New();
this->HandleProperty->SetColor(1,1,1);
}
if ( ! this->SelectedHandleProperty )
{
this->SelectedHandleProperty = vtkProperty::New();
this->SelectedHandleProperty->SetColor(1,0,0);
}
}
void vtkSphereWidget::PlaceWidget(double bds[6])
{
double bounds[6], center[3], radius;
this->AdjustBounds(bds, bounds, center);
radius = (bounds[1]-bounds[0]) / 2.0;
if ( radius > ((bounds[3]-bounds[2])/2.0) )
{
radius = (bounds[3]-bounds[2])/2.0;
}
radius = (bounds[1]-bounds[0]) / 2.0;
if ( radius > ((bounds[5]-bounds[4])/2.0) )
{
radius = (bounds[5]-bounds[4])/2.0;
}
this->SphereSource->SetCenter(center);
this->SphereSource->SetRadius(radius);
this->SphereSource->Update();
// place the handle
this->PlaceHandle(center,radius);
for (int i=0; i<6; i++)
{
this->InitialBounds[i] = bounds[i];
}
this->InitialLength = sqrt((bounds[1]-bounds[0])*(bounds[1]-bounds[0]) +
(bounds[3]-bounds[2])*(bounds[3]-bounds[2]) +
(bounds[5]-bounds[4])*(bounds[5]-bounds[4]));
this->SizeHandles();
}
void vtkSphereWidget::PlaceHandle(double *center, double radius)
{
double sf = radius / vtkMath::Norm(this->HandleDirection);
this->HandlePosition[0] = center[0] + sf*this->HandleDirection[0];
this->HandlePosition[1] = center[1] + sf*this->HandleDirection[1];
this->HandlePosition[2] = center[2] + sf*this->HandleDirection[2];
this->HandleSource->SetCenter(this->HandlePosition);
}
void vtkSphereWidget::SizeHandles()
{
double radius = this->vtk3DWidget::SizeHandles(1.25);
this->HandleSource->SetRadius(radius);
}
void vtkSphereWidget::GetPolyData(vtkPolyData *pd)
{
pd->ShallowCopy(this->SphereSource->GetOutput());
}