//============================================================================= // // Copyright (c) Kitware, Inc. // All rights reserved. // See LICENSE.txt 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. // // Copyright 2012 Sandia Corporation. // Under the terms of Contract DE-AC04-94AL85000 with Sandia Corporation, // the U.S. Government retains certain rights in this software. // //============================================================================= #include "vtkmContour.h" #include "vtkCellData.h" #include "vtkDataSet.h" #include "vtkInformation.h" #include "vtkInformationVector.h" #include "vtkNew.h" #include "vtkObjectFactory.h" #include "vtkPointData.h" #include "vtkPolyData.h" #include "vtkUnstructuredGrid.h" #include "vtkmlib/ArrayConverters.h" #include "vtkmlib/DataSetConverters.h" #include "vtkmlib/PolyDataConverter.h" #include "vtkmlib/Storage.h" #include "vtkmCellSetExplicit.h" #include "vtkmCellSetSingleType.h" #include "vtkmFilterPolicy.h" #include vtkStandardNewMacro(vtkmContour) //------------------------------------------------------------------------------ vtkmContour::vtkmContour() { } //------------------------------------------------------------------------------ vtkmContour::~vtkmContour() { } //------------------------------------------------------------------------------ void vtkmContour::PrintSelf(ostream& os, vtkIndent indent) { this->Superclass::PrintSelf(os, indent); } //------------------------------------------------------------------------------ int vtkmContour::RequestData(vtkInformation* request, vtkInformationVector** inputVector, vtkInformationVector* outputVector) { vtkInformation* inInfo = inputVector[0]->GetInformationObject(0); vtkInformation* outInfo = outputVector->GetInformationObject(0); vtkDataSet* input = vtkDataSet::SafeDownCast(inInfo->Get(vtkDataObject::DATA_OBJECT())); vtkPolyData* output = vtkPolyData::SafeDownCast(outInfo->Get(vtkDataObject::DATA_OBJECT())); const int numContours = this->GetNumberOfContours(); if(numContours == 0) { return 1; } vtkm::filter::MarchingCubes filter; // set local variables filter.SetGenerateNormals(this->GetComputeNormals() != 0); filter.SetNumberOfIsoValues(numContours); for(int i = 0; i < numContours; ++i) { filter.SetIsoValue(i, this->GetValue(i)); } // convert the input dataset to a vtkm::cont::DataSet vtkm::cont::DataSet in; if (this->ComputeScalars) { in = tovtkm::Convert(input, tovtkm::FieldsFlag::PointsAndCells); } else { in = tovtkm::Convert(input, tovtkm::FieldsFlag::None); } // we need to map the given property to the data set int association = this->GetInputArrayAssociation(0, inputVector); vtkDataArray* inputArray = this->GetInputArrayToProcess(0, inputVector); vtkm::cont::Field inField = tovtkm::Convert(inputArray, association); const bool dataSetValid = in.GetNumberOfCoordinateSystems() > 0 && in.GetNumberOfCellSets() > 0; const bool fieldValid = (inField.GetAssociation() != vtkm::cont::Field::ASSOC_ANY) && (inField.GetName() != std::string()); if (!dataSetValid) { vtkWarningMacro(<< "Will not be able to use VTKm dataset type is unknown"); } if (!fieldValid) { vtkWarningMacro(<< "Will not be able to use VTKm field type is unknown"); } vtkm::filter::Result result; bool convertedDataSet = false; if (dataSetValid && fieldValid) { vtkmInputFilterPolicy policy; result = filter.Execute(in, inField, policy); if (!result.IsDataSetValid()) { vtkWarningMacro(<< "VTKm contour algorithm was failed to run. \n" << "Falling back to serial implementation."); return this->Superclass::RequestData(request, inputVector, outputVector); } vtkm::Id numFields = static_cast(in.GetNumberOfFields()); for (vtkm::Id fieldIdx = 0; fieldIdx < numFields; ++fieldIdx) { const vtkm::cont::Field &field = in.GetField(fieldIdx); try { filter.MapFieldOntoOutput(result, field, policy); } catch (vtkm::cont::Error &e) { vtkWarningMacro(<< "Unable to use VTKm to convert field( " << field.GetName() << " ) to the MarchingCubes" << " output: " << e.what()); } } // convert back the dataset to VTK convertedDataSet = fromvtkm::Convert(result.GetDataSet(), output, input); if (!convertedDataSet) { vtkWarningMacro(<< "Unable to convert VTKm DataSet back to VTK.\n" << "Falling back to serial implementation."); return this->Superclass::RequestData(request, inputVector, outputVector); } } if (this->ComputeScalars) { output->GetPointData()->SetActiveScalars(inputArray->GetName()); } if (this->ComputeNormals) { output->GetPointData()->SetActiveAttribute( filter.GetNormalArrayName().c_str(), vtkDataSetAttributes::NORMALS); } // we got this far, everything is good return 1; }