Newer
Older
/*=========================================================================
Library: iMSTK
Copyright (c) Kitware, Inc. & Center for Modeling, Simulation,
& Imaging in Medicine, Rensselaer Polytechnic Institute.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0.txt
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
=========================================================================*/
#include "imstkVTKSurfaceMeshRenderDelegate.h"
#include "imstkGeometryUtilities.h"

Andrew Wilson
committed
#include "imstkRenderMaterial.h"
#include "imstkSurfaceMesh.h"

Andrew Wilson
committed
#include "imstkTextureDelegate.h"
#include "imstkTextureManager.h"

Andrew Wilson
committed
#include "imstkVisualModel.h"

Andrew Wilson
committed
#include <vtkActor.h>
#include <vtkCellData.h>
#include <vtkDoubleArray.h>
#include <vtkFloatArray.h>

Andrew Wilson
committed
#include <vtkOpenGLPolyDataMapper.h>
#include <vtkOpenGLVertexBufferObject.h>
#include <vtkPointData.h>

Andrew Wilson
committed
#include <vtkPolyData.h>

Andrew Wilson
committed
#include <vtkTexture.h>
#include <vtkTransform.h>
VTKSurfaceMeshRenderDelegate::VTKSurfaceMeshRenderDelegate(std::shared_ptr<VisualModel> visualModel) : VTKPolyDataRenderDelegate(visualModel),
m_isDynamicMesh(m_visualModel->getRenderMaterial()->getIsDynamicMesh()),
m_polydata(vtkSmartPointer<vtkPolyData>::New()),
m_mappedVertexArray(vtkSmartPointer<vtkDoubleArray>::New()),
m_mappedNormalArray(vtkSmartPointer<vtkDoubleArray>::New())
m_geometry = std::static_pointer_cast<SurfaceMesh>(m_visualModel->getGeometry());
m_geometry->computeVertexNeighborTriangles();
// Get our own handles to these in case the geometry changes them
m_vertices = m_isDynamicMesh ? m_geometry->getVertexPositions() : m_geometry->getInitialVertexPositions();
m_indices = m_geometry->getTriangleIndices();
// Map vertices to VTK point data
if (m_vertices != nullptr)
m_mappedVertexArray = vtkDoubleArray::SafeDownCast(GeometryUtils::coupleVtkDataArray(m_vertices));
auto points = vtkSmartPointer<vtkPoints>::New();
points->SetNumberOfPoints(m_vertices->size());
points->SetData(m_mappedVertexArray);
m_polydata->SetPoints(points);
}
// Map indices to VTK cell data (copied)
if (m_indices != nullptr)
{
m_cellArray = vtkSmartPointer<vtkCellArray>::New();
vtkIdType cell[3];
for (const auto& t : *m_indices)
{
cell[i] = t[i];
}
m_cellArray->InsertNextCell(3, cell);
m_polydata->SetPolys(m_cellArray);
// Map vertex scalars if it has them
if (m_geometry->getVertexScalars() != nullptr)
setVertexScalarBuffer(m_geometry->getVertexScalars());
// Map cell scalars if it has them
if (m_geometry->getCellScalars() != nullptr)
{
setCellScalarBuffer(m_geometry->getCellScalars());
}
// Map normals, if none provided compute per vertex normals
if (m_geometry->getVertexNormals() == nullptr)
{
}
m_mappedNormalArray = vtkDoubleArray::SafeDownCast(GeometryUtils::coupleVtkDataArray(m_geometry->getVertexNormals()));
m_polydata->GetPointData()->SetNormals(m_mappedNormalArray);
// Map TCoords
if (m_geometry->getVertexTCoords() != nullptr)
{
setTextureCoordinateBuffer(m_geometry->getVertexTCoords());
}
// When geometry is modified, update data source, mostly for when an entirely new array/buffer was set
queueConnect<Event>(m_geometry, &Geometry::modified, this, &VTKSurfaceMeshRenderDelegate::geometryModified);
// When the vertex buffer internals are modified, ie: a single or N elements
queueConnect<Event>(m_vertices, &VecDataArray<double, 3>::modified, this, &VTKSurfaceMeshRenderDelegate::vertexDataModified);
// When index buffer internals are modified
queueConnect<Event>(m_indices, &VecDataArray<int, 3>::modified, this, &VTKSurfaceMeshRenderDelegate::indexDataModified);
// When index buffer internals are modified
queueConnect<Event>(m_geometry->getVertexNormals(), &VecDataArray<double, 3>::modified, this, &VTKSurfaceMeshRenderDelegate::normalDataModified);

Andrew Wilson
committed
connect<Event>(m_material, &RenderMaterial::texturesModified, this, &VTKSurfaceMeshRenderDelegate::texturesModified);
// Setup mapper
vtkNew<vtkPolyDataMapper> mapper;
mapper->SetInputData(m_polydata);
vtkNew<vtkActor> actor;
actor->SetMapper(mapper);
m_mapper = mapper;
m_actor = actor;
if (!m_isDynamicMesh)
{
actor->SetUserTransform(m_transform);
}
if (auto glMapper = vtkOpenGLPolyDataMapper::SafeDownCast(m_mapper))
glMapper->SetVBOShiftScaleMethod(vtkOpenGLVertexBufferObject::DISABLE_SHIFT_SCALE);
}
update();
updateRenderProperties();
}
void
VTKSurfaceMeshRenderDelegate::processEvents()
{
if (!m_isDynamicMesh)
{
// Update the transform
const Mat4d& mImstk = m_geometry->getTransform();
vtkNew<vtkMatrix4x4> mVtk;
for (int y = 0; y < 4; y++)
{
for (int x = 0; x < 4; x++)
{
mVtk->SetElement(x, y, mImstk(x, y));
}
}
m_transform->SetMatrix(mVtk);
}
// Custom handling of events
std::shared_ptr<VecDataArray<double, 3>> verticesPtr =
m_isDynamicMesh ? m_geometry->getVertexPositions() : m_geometry->getInitialVertexPositions();
std::shared_ptr<VecDataArray<int, 3>> indicesPtr = m_geometry->getTriangleIndices();
std::shared_ptr<AbstractDataArray> cellScalarsPtr = m_geometry->getCellScalars();
std::shared_ptr<AbstractDataArray> vertexScalarsPtr = m_geometry->getVertexScalars();
std::shared_ptr<AbstractDataArray> textureCoordinatesPtr = m_geometry->getVertexTCoords();
// Only use the most recent event from respective sender
std::array<Command, 8> cmds;
std::array<bool, 8> contains = { false, false, false, false, false, false, false, false };
rforeachEvent([&](Command cmd)
{
if (cmd.m_event->m_sender == m_visualModel.get() && !contains[0])
{
contains[0] = true;
}
else if (cmd.m_event->m_sender == m_material.get() && !contains[1])
{
contains[1] = true;
}
else if (cmd.m_event->m_sender == m_geometry.get() && !contains[2])
{
contains[2] = true;
}
else if (cmd.m_event->m_sender == verticesPtr.get() && !contains[3])
{
contains[3] = true;
}
else if (cmd.m_event->m_sender == cellScalarsPtr.get() && !contains[4])
else if (cmd.m_event->m_sender == vertexScalarsPtr.get() && !contains[5])
else if (cmd.m_event->m_sender == indicesPtr.get() && !contains[6])
{
cmds[6] = cmd;
contains[6] = true;
}
else if (cmd.m_event->m_sender == textureCoordinatesPtr.get() && !contains[7])
{
cmds[7] = cmd;
contains[7] = true;
}
});
// Now do all the commands
cmds[0].invoke(); // Update VisualModel
cmds[1].invoke(); // Update RenderMaterial
cmds[3].invoke(); // Update vertices
cmds[4].invoke(); // Update cell scalars
cmds[5].invoke(); // Update vertex scalars
cmds[6].invoke(); // Update indices
cmds[7].invoke(); // Update texture coordinates
}
void
VTKSurfaceMeshRenderDelegate::vertexDataModified(Event* imstkNotUsed(e))
{
setVertexBuffer(m_isDynamicMesh ? m_geometry->getVertexPositions() :
m_geometry->getInitialVertexPositions());
if (m_isDynamicMesh)
{
// If the material says we should recompute normals
if (m_visualModel->getRenderMaterial()->getRecomputeVertexNormals())
{
m_geometry->computeVertexNormals();
setNormalBuffer(m_geometry->getVertexNormals());
}
}
void
VTKSurfaceMeshRenderDelegate::indexDataModified(Event* imstkNotUsed(e))
{
}
void
VTKSurfaceMeshRenderDelegate::normalDataModified(Event* imstkNotUsed(e))
{
}
void
VTKSurfaceMeshRenderDelegate::vertexScalarsModified(Event* imstkNotUsed(e))
{
}
void
VTKSurfaceMeshRenderDelegate::cellScalarsModified(Event* imstkNotUsed(e))
{
void
VTKSurfaceMeshRenderDelegate::textureCoordinatesModified(Event* imstkNotUsed(e))
{
setTextureCoordinateBuffer(m_geometry->getVertexTCoords());
}
VTKSurfaceMeshRenderDelegate::geometryModified(Event* imstkNotUsed(e))
// If the mesh is dynamic check for buffer size changes & consistently reupload
// the vertex buffer. Recompute normals dynamically.
if (m_isDynamicMesh)
// If the vertices were reallocated
if (m_vertices != m_geometry->getVertexPositions())
{
setVertexBuffer(m_geometry->getVertexPositions());
}
// Consistently reupload the vertex buffer
m_mappedVertexArray->Modified();
// Only update index buffer when reallocated
if (m_indices != m_geometry->getTriangleIndices())
{
setIndexBuffer(m_geometry->getTriangleIndices());
}
if (m_normals != m_geometry->getVertexNormals())
{
setNormalBuffer(m_geometry->getVertexNormals());
}
if (m_visualModel->getRenderMaterial()->getRecomputeVertexNormals())
{
m_geometry->computeVertexNormals();
setNormalBuffer(m_geometry->getVertexNormals());
}
}
// If the mesh is not dynamic, avoid reuploading & recomputing any buffers
// vertices & normals can be changed rigidly by a transform in the shader
else
{
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
// If the vertices were reallocated
bool normalsOutdated = false;
if (m_vertices != m_geometry->getInitialVertexPositions())
{
setVertexBuffer(m_geometry->getInitialVertexPositions());
normalsOutdated = true;
}
// Only update index buffer when reallocated
if (m_indices != m_geometry->getTriangleIndices())
{
setIndexBuffer(m_geometry->getTriangleIndices());
normalsOutdated = true;
}
if (m_normals != m_geometry->getVertexNormals())
{
setNormalBuffer(m_geometry->getVertexNormals());
normalsOutdated = false;
}
if (normalsOutdated && m_visualModel->getRenderMaterial()->getRecomputeVertexNormals())
{
m_geometry->computeVertexNormals();
setNormalBuffer(m_geometry->getVertexNormals());
}
}
if (m_vertexScalars != m_geometry->getVertexScalars())
if (m_cellScalars != m_geometry->getCellScalars())
if (m_textureCoordinates != m_geometry->getVertexTCoords())
{
setTextureCoordinateBuffer(m_geometry->getVertexTCoords());
}
}

Andrew Wilson
committed
void
VTKSurfaceMeshRenderDelegate::texturesModified(Event* e)
{

Andrew Wilson
committed
// If a texture is set/swapped reinit all textures

Andrew Wilson
committed
RenderMaterial* material = static_cast<RenderMaterial*>(e->m_sender);
if (material != nullptr)
{
// Reload all textures
// If texture already present, don't do anything unless name changed

Andrew Wilson
committed
initializeTextures();

Andrew Wilson
committed
}
}
void
VTKSurfaceMeshRenderDelegate::setVertexBuffer(std::shared_ptr<VecDataArray<double, 3>> vertices)
{
// If the buffer changed
if (m_vertices != vertices)
{
// If previous buffer exist
if (m_vertices != nullptr)
{
// stop observing its changes
disconnect(m_vertices, this, &VecDataArray<double, 3>::modified);
}
// Set new buffer and observe
m_vertices = vertices;
queueConnect<Event>(m_vertices, &VecDataArray<double, 3>::modified, this, &VTKSurfaceMeshRenderDelegate::vertexDataModified);
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
// Couple the buffer
m_mappedVertexArray->SetNumberOfComponents(3);
m_mappedVertexArray->SetArray(reinterpret_cast<double*>(m_vertices->getPointer()), m_vertices->size() * 3, 1);
m_mappedVertexArray->Modified();
m_polydata->GetPoints()->SetNumberOfPoints(m_vertices->size());
}
void
VTKSurfaceMeshRenderDelegate::setNormalBuffer(std::shared_ptr<VecDataArray<double, 3>> normals)
{
// If the buffer changed
if (m_normals != normals)
{
// If previous buffer exist
if (m_normals != nullptr)
{
// stop observing its changes
disconnect(m_normals, this, &VecDataArray<double, 3>::modified);
}
// Set new buffer and observe
m_normals = normals;
queueConnect<Event>(m_normals, &VecDataArray<double, 3>::modified, this, &VTKSurfaceMeshRenderDelegate::normalDataModified);
}
// Couple the buffer
m_mappedNormalArray->SetNumberOfComponents(3);
m_mappedNormalArray->SetArray(reinterpret_cast<double*>(m_normals->getPointer()), m_normals->size() * 3, 1);
m_mappedNormalArray->Modified();
}
void
VTKSurfaceMeshRenderDelegate::setIndexBuffer(std::shared_ptr<VecDataArray<int, 3>> indices)
{
// If the buffer changed
if (m_indices != indices)
{
// If previous buffer exist
if (m_indices != nullptr)
{
// stop observing its changes
disconnect(m_indices, this, &VecDataArray<int, 3>::modified);
}
// Set new buffer and observe
m_indices = indices;
queueConnect<Event>(m_indices, &VecDataArray<int, 3>::modified, this, &VTKSurfaceMeshRenderDelegate::indexDataModified);
}
// Copy the buffer
// Copy cells
m_cellArray->Reset();
vtkIdType cell[3];
for (const auto& t : *m_indices)
{
{
cell[i] = t[i];
}
m_cellArray->InsertNextCell(3, cell);
}
m_cellArray->Modified();
void
VTKSurfaceMeshRenderDelegate::setVertexScalarBuffer(std::shared_ptr<AbstractDataArray> scalars)
{
// If the buffer changed
if (m_vertexScalars != scalars)
{
// If previous buffer exist
if (m_vertexScalars != nullptr)
{
// stop observing its changes
disconnect(m_vertexScalars, this, &AbstractDataArray::modified);
}
// Set new buffer and observe
m_vertexScalars = scalars;
queueConnect<Event>(m_vertexScalars, &AbstractDataArray::modified, this, &VTKSurfaceMeshRenderDelegate::vertexScalarsModified);
m_mappedVertexScalarArray = GeometryUtils::coupleVtkDataArray(m_vertexScalars);
m_polydata->GetPointData()->SetScalars(m_mappedVertexScalarArray);
m_mappedVertexScalarArray->SetNumberOfComponents(m_vertexScalars->getNumberOfComponents());
m_mappedVertexScalarArray->SetVoidArray(m_vertexScalars->getVoidPointer(),
m_mappedVertexScalarArray->Modified();
}
void
VTKSurfaceMeshRenderDelegate::setCellScalarBuffer(std::shared_ptr<AbstractDataArray> scalars)
{
// If the buffer changed
if (m_cellScalars != scalars)
{
// If previous buffer exist
if (m_cellScalars != nullptr)
{
// stop observing its changes
disconnect(m_cellScalars, this, &AbstractDataArray::modified);
}
// Set new buffer and observe
m_cellScalars = scalars;
queueConnect<Event>(m_cellScalars, &AbstractDataArray::modified, this, &VTKSurfaceMeshRenderDelegate::cellScalarsModified);
m_mappedCellScalarArray = GeometryUtils::coupleVtkDataArray(m_cellScalars);
m_polydata->GetCellData()->SetScalars(m_mappedCellScalarArray);
m_mappedCellScalarArray->SetNumberOfComponents(m_cellScalars->getNumberOfComponents());
m_mappedCellScalarArray->SetVoidArray(m_cellScalars->getVoidPointer(),
m_mappedCellScalarArray->Modified();
}
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
void
VTKSurfaceMeshRenderDelegate::setTextureCoordinateBuffer(std::shared_ptr<AbstractDataArray> textureCoordinates)
{
// If the buffer changed
if (m_textureCoordinates != textureCoordinates)
{
// If previous buffer exist
if (m_textureCoordinates != nullptr)
{
// stop observing its changes
disconnect(m_textureCoordinates, this, &AbstractDataArray::modified);
}
// Set new buffer and observe
m_textureCoordinates = textureCoordinates;
queueConnect<Event>(m_textureCoordinates, &AbstractDataArray::modified, this, &VTKSurfaceMeshRenderDelegate::textureCoordinatesModified);
m_mappedTCoordsArray = vtkFloatArray::SafeDownCast(GeometryUtils::coupleVtkDataArray(textureCoordinates));
m_mappedTCoordsArray->SetName(m_geometry->getActiveVertexTCoords().c_str());
m_polydata->GetPointData()->SetTCoords(m_mappedTCoordsArray);
}
m_mappedTCoordsArray->SetNumberOfComponents(m_textureCoordinates->getNumberOfComponents());
m_mappedTCoordsArray->SetVoidArray(m_textureCoordinates->getVoidPointer(), static_cast<vtkIdType>(m_textureCoordinates->size()), 1);
m_mappedTCoordsArray->Modified();
// Map Tangents
if (m_geometry->getVertexTangents() == nullptr)
{
m_geometry->computeVertexTangents();
}
// These need to be float for PBR
m_mappedTangentArray = vtkFloatArray::SafeDownCast(GeometryUtils::coupleVtkDataArray(m_geometry->getVertexTangents()));
m_polydata->GetPointData()->SetTangents(m_mappedTangentArray);
m_mappedTangentArray->Modified();
}

Andrew Wilson
committed
VTKSurfaceMeshRenderDelegate::initializeTextures()
auto material = m_visualModel->getRenderMaterial();
if (material == nullptr)
{
return;
}
unsigned int currentUnit = 0;
// Go through all of the textures

Andrew Wilson
committed
vtkSmartPointer<vtkActor> actor = vtkActor::SafeDownCast(m_actor);
actor->GetProperty()->RemoveAllTextures();
for (int unit = 0; unit < (int)Texture::Type::None; unit++)
{
// Get imstk texture
auto texture = material->getTexture((Texture::Type)unit);

Andrew Wilson
committed
// If neither of these are provided, the texture is not filled out
if (texture->getImageData() == nullptr && texture->getPath() == "")
{
continue;
}
// Get vtk texture

Andrew Wilson
committed
std::shared_ptr<TextureManager<VTKTextureDelegate>> textureManager = m_textureManager.lock();
auto textureDelegate = textureManager->getTextureDelegate(texture);
/* /!\ VTKTextureWrapMode not yet supported in VTK 7
* See here for some work that needs to be imported back to upstream:
* https://gitlab.kitware.com/iMSTK/vtk/commit/62a7ecd8a5f54e243c26960de22d5d1d23ef932b
*
texture->SetWrapMode(vtkTexture::VTKTextureWrapMode::ClampToBorder);
* /!\ MultiTextureAttribute not yet supported in VTK 7
* See here for some work that needs to be imported back to upstream:
* https://gitlab.kitware.com/iMSTK/vtk/commit/ae373026755db42b6fdce5093109ef1a39a76340
*
// Link texture unit to texture attribute
m_mapper->MapDataArrayToMultiTextureAttribute(unit, tCoordsName.c_str(),
vtkDataObject::FIELD_ASSOCIATION_POINTS);
*/
// Set texture

Andrew Wilson
committed
vtkSmartPointer<vtkTexture> currentTexture = textureDelegate->getVtkTexture();
if (material->getShadingModel() == RenderMaterial::ShadingModel::PBR)
{
actor->GetProperty()->SetBaseColorTexture(currentTexture);
actor->GetProperty()->SetNormalTexture(currentTexture);
actor->GetProperty()->SetORMTexture(currentTexture);

Andrew Wilson
committed
case Texture::Type::Anistropy:
{
actor->GetProperty()->SetAnisotropyTexture(currentTexture);
break;
}
case Texture::Type::CoatNormal:
{
actor->GetProperty()->SetCoatNormalTexture(currentTexture);
break;
}
else if (texture->getType() == Texture::Type::Diffuse)
actor->SetTexture(currentTexture);
currentUnit++;