Skip to content
Snippets Groups Projects
Commit 4f2ea4f5 authored by Lucas Gandel's avatar Lucas Gandel
Browse files

Add vtkOpenVRRay to vtkOpenVRModel

This commit adds a way to draw a ray for a particular model. This will
be useful when picking is performed along a ray using the
vtkOpenVRPropPicker introduced in 653fee20c0eab618074868666874e8a3a7c0c1c0 .
The length of the ray can be set from the model and corresponds to a
distance in meters. To do so, you can access the different models using
the accessor introduced in 0ea021ed34c7241fa9ac5f499d9cc3f858f422ce .
parent f5584317
Branches Add-vtkOpenVRModel-and-RayPicker
No related tags found
No related merge requests found
......@@ -26,8 +26,174 @@ PURPOSE. See the above copyright notice for more information.
#include "vtkTextureObject.h"
#include "vtkRendererCollection.h"
#include "vtkMatrix4x4.h"
#include "vtkRenderWindowInteractor.h"
#include "vtkInteractorObserver.h"
vtkStandardNewMacro( vtkOpenVRModel );
/*=========================================================================
vtkOpenVRRay
=========================================================================*/
class vtkOpenVRRay : public vtkObject
{
public:
static vtkOpenVRRay *New();
vtkTypeMacro(vtkOpenVRRay, vtkObject);
bool Build(vtkOpenVRRenderWindow *win);
void Render(vtkOpenVRRenderWindow *win,
vtkMatrix4x4 *poseMatrix);
// show the model
void SetShow(bool v) {
this->Show = v;
};
bool GetShow() {
return this->Show;
};
void SetLength(float s) {
this->Length = s;
};
void ReleaseGraphicsResources(vtkRenderWindow *win);
protected:
vtkOpenVRRay();
~vtkOpenVRRay();
bool Show;
bool Loaded;
vtkOpenGLHelper ModelHelper;
vtkOpenGLVertexBufferObject *ModelVBO;
vtkNew<vtkMatrix4x4> PoseMatrix;
float Length;
};
vtkStandardNewMacro(vtkOpenVRRay);
vtkOpenVRRay::vtkOpenVRRay()
{
this->Show = false;
this->Length = 1.0f;
this->Loaded = false;
this->ModelVBO = vtkOpenGLVertexBufferObject::New();
};
vtkOpenVRRay::~vtkOpenVRRay()
{
this->ModelVBO->Delete();
this->ModelVBO = 0;
}
void vtkOpenVRRay::ReleaseGraphicsResources(vtkRenderWindow *win)
{
this->ModelVBO->ReleaseGraphicsResources();
this->ModelHelper.ReleaseGraphicsResources(win);
}
bool vtkOpenVRRay::Build(vtkOpenVRRenderWindow *win)
{
//Ray geometry
float vert[] = {
0, 0, 0,
0, 0, -1};
unsigned short ind[] = {0, 1};
this->ModelVBO->Upload(
vert,
2 * 3,
vtkOpenGLBufferObject::ArrayBuffer);
this->ModelHelper.IBO->Upload(
ind,
1 * 2,
vtkOpenGLBufferObject::ElementArrayBuffer);
this->ModelHelper.IBO->IndexCount = 1 * 2;
this->ModelHelper.Program = win->GetShaderCache()->ReadyShaderProgram(
// vertex shader
"//VTK::System::Dec\n"
"uniform mat4 matrix;\n"
"uniform float scale;\n"
"attribute vec3 position;\n"
"void main()\n"
"{\n"
" gl_Position = matrix * vec4(scale * position, 1);\n"
"}\n",
//fragment shader
"//VTK::System::Dec\n"
"//VTK::Output::Dec\n"
"void main()\n"
"{\n"
" gl_FragData[0] = vec4(1,0,0,1);\n"
"}\n",
// geom shader
""
);
vtkShaderProgram *program = this->ModelHelper.Program;
this->ModelHelper.VAO->Bind();
if (!this->ModelHelper.VAO->AddAttributeArray(program,
this->ModelVBO,
"position",
0,
3 * sizeof(float),
VTK_FLOAT, 3, false))
{
vtkErrorMacro(<< "Error setting position in shader VAO.");
}
return true;
}
void vtkOpenVRRay::Render(
vtkOpenVRRenderWindow *win,
vtkMatrix4x4 *poseMatrix)
{
//Load ray
if (!this->Loaded)
{
if (!this->Build(win))
{
vtkErrorMacro("Unable to build controller ray ");
}
this->Loaded = true;
}
// Render ray
win->GetShaderCache()->ReadyShaderProgram(this->ModelHelper.Program);
this->ModelHelper.VAO->Bind();
this->ModelHelper.IBO->Bind();
vtkRenderer *ren = static_cast< vtkRenderer * >(
win->GetRenderers()->GetItemAsObject(0));
if (!ren)
{
vtkErrorMacro("Unable get renderer");
return;
}
double unitV[4] = {0, 0, 0, 1};
double scaleFactor =
vtkMath::Norm( poseMatrix->MultiplyDoublePoint(unitV));
this->ModelHelper.Program->SetUniformf("scale",
this->Length / scaleFactor);
this->ModelHelper.Program->SetUniformMatrix("matrix",
poseMatrix);
glDrawElements(GL_LINES,
static_cast<GLsizei>(this->ModelHelper.IBO->IndexCount),
GL_UNSIGNED_SHORT, 0);
}
/*=========================================================================
vtkOpenVRModel
=========================================================================*/
vtkStandardNewMacro(vtkOpenVRModel);
vtkOpenVRModel::vtkOpenVRModel()
{
......@@ -225,12 +391,43 @@ void vtkOpenVRModel::Render(
(double *)(tcdc->Element),
(double *)(this->PoseMatrix->Element));
this->ModelHelper.Program->SetUniformMatrix("matrix", this->PoseMatrix.Get());
this->ModelHelper.Program->SetUniformMatrix("matrix",
this->PoseMatrix.Get());
}
glDrawElements(GL_TRIANGLES,
static_cast<GLsizei>(this->ModelHelper.IBO->IndexCount),
GL_UNSIGNED_SHORT, 0);
this->TextureObject->Deactivate();
//Handle drawing of the ray associated to this model
if (!win->GetInteractor())
{
vtkErrorMacro("Unable to get interactor");
return;
}
if (!win->GetInteractor()->GetInteractorStyle())
{
vtkErrorMacro("Unable to get interactor style");
return;
}
//Update ray points and draw state
win->GetInteractor()->GetInteractorStyle()->InvokeEvent(
vtkCommand::RenderEvent);
//Draw ray
if (this->Ray->GetShow())
{
this->Ray->Render(win, this->PoseMatrix.Get());
}
}
}
void vtkOpenVRModel::SetShowRay(bool v)
{
this->Ray->SetShow(v);
}
void vtkOpenVRModel::SetRayLength(double length)
{
this->Ray->SetLength(length);
}
......@@ -34,6 +34,7 @@ class vtkRenderWindow;
class vtkOpenGLVertexBufferObject;
class vtkTextureObject;
class vtkMatrix4x4;
class vtkOpenVRRay;
class VTKRENDERINGOPENVR_EXPORT vtkOpenVRModel : public vtkObject
{
......@@ -60,6 +61,9 @@ public:
return this->Show;
};
//Set Ray parameters
void SetShowRay(bool v);
void SetRayLength(double length);
void ReleaseGraphicsResources(vtkRenderWindow *win);
......@@ -80,6 +84,9 @@ protected:
vtkOpenGLVertexBufferObject *ModelVBO;
vtkNew<vtkTextureObject> TextureObject;
vtkNew<vtkMatrix4x4> PoseMatrix;
//Controller ray
vtkNew<vtkOpenVRRay> Ray;
};
#endif
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment