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.
183 lines
4.5 KiB
183 lines
4.5 KiB
/*=========================================================================
|
|
|
|
Program: Visualization Toolkit
|
|
Module: $RCSfile: vtkImplicitSum.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 "vtkImplicitSum.h"
|
|
|
|
#include "vtkDoubleArray.h"
|
|
#include "vtkImplicitFunctionCollection.h"
|
|
#include "vtkObjectFactory.h"
|
|
|
|
#include <math.h>
|
|
|
|
vtkCxxRevisionMacro(vtkImplicitSum, "$Revision: 1.6 $");
|
|
vtkStandardNewMacro(vtkImplicitSum);
|
|
|
|
// Constructor.
|
|
vtkImplicitSum::vtkImplicitSum()
|
|
{
|
|
this->FunctionList = vtkImplicitFunctionCollection::New();
|
|
this->Weights = vtkDoubleArray::New();
|
|
this->Weights->SetNumberOfComponents(1);
|
|
this->TotalWeight = 0.0;
|
|
this->NormalizeByWeight = 0;
|
|
}
|
|
|
|
vtkImplicitSum::~vtkImplicitSum()
|
|
{
|
|
this->FunctionList->Delete();
|
|
this->Weights->Delete();
|
|
}
|
|
|
|
unsigned long int vtkImplicitSum::GetMTime()
|
|
{
|
|
unsigned long int fMtime;
|
|
unsigned long int mtime = this->vtkImplicitFunction::GetMTime();
|
|
vtkImplicitFunction *f;
|
|
|
|
fMtime = this->Weights->GetMTime();
|
|
if ( fMtime > mtime )
|
|
{
|
|
mtime = fMtime;
|
|
}
|
|
|
|
vtkCollectionSimpleIterator sit;
|
|
for (this->FunctionList->InitTraversal(sit);
|
|
(f=this->FunctionList->GetNextImplicitFunction(sit)); )
|
|
{
|
|
fMtime = f->GetMTime();
|
|
if ( fMtime > mtime )
|
|
{
|
|
mtime = fMtime;
|
|
}
|
|
}
|
|
return mtime;
|
|
}
|
|
|
|
// Add another implicit function to the list of functions.
|
|
void vtkImplicitSum::AddFunction(vtkImplicitFunction *f, double scale)
|
|
{
|
|
this->Modified();
|
|
this->FunctionList->AddItem(f);
|
|
this->Weights->InsertNextValue(scale);
|
|
this->CalculateTotalWeight();
|
|
}
|
|
|
|
void vtkImplicitSum::SetFunctionWeight(vtkImplicitFunction *f, double scale)
|
|
{
|
|
int loc = this->FunctionList->IsItemPresent(f);
|
|
if (! loc)
|
|
{
|
|
vtkWarningMacro("Function not found in function list");
|
|
return;
|
|
}
|
|
loc--; // IsItemPresent returns index+1.
|
|
|
|
if ( this->Weights->GetValue(loc) != scale )
|
|
{
|
|
this->Modified();
|
|
this->Weights->SetValue(loc, scale);
|
|
this->CalculateTotalWeight();
|
|
}
|
|
}
|
|
|
|
void vtkImplicitSum::RemoveAllFunctions()
|
|
{
|
|
this->Modified();
|
|
this->FunctionList->RemoveAllItems();
|
|
this->Weights->Initialize();
|
|
this->TotalWeight = 0.0;
|
|
}
|
|
|
|
void vtkImplicitSum::CalculateTotalWeight(void)
|
|
{
|
|
this->TotalWeight = 0.0;
|
|
|
|
for(int i = this->Weights->GetNumberOfTuples() - 1; i >= 0; i--)
|
|
{
|
|
this->TotalWeight += this->Weights->GetValue(i);
|
|
}
|
|
}
|
|
|
|
|
|
// Evaluate sum of implicit functions.
|
|
double vtkImplicitSum::EvaluateFunction(double x[3])
|
|
{
|
|
double sum = 0;
|
|
double c;
|
|
int i;
|
|
vtkImplicitFunction *f;
|
|
double *weights = this->Weights->GetPointer(0);
|
|
|
|
vtkCollectionSimpleIterator sit;
|
|
for (i = 0, this->FunctionList->InitTraversal(sit);
|
|
(f=this->FunctionList->GetNextImplicitFunction(sit)); i++)
|
|
{
|
|
c = weights[i];
|
|
if (c != 0.0)
|
|
{
|
|
sum += f->FunctionValue(x)*c;
|
|
}
|
|
}
|
|
if (this->NormalizeByWeight && this->TotalWeight != 0.0)
|
|
{
|
|
sum /= this->TotalWeight;
|
|
}
|
|
return sum;
|
|
}
|
|
|
|
// Evaluate gradient of sum of functions (valid only if linear)
|
|
void vtkImplicitSum::EvaluateGradient(double x[3], double g[3])
|
|
{
|
|
double c;
|
|
int i;
|
|
double gtmp[3];
|
|
vtkImplicitFunction *f;
|
|
double *weights = this->Weights->GetPointer(0);
|
|
|
|
g[0] = g[1] = g[2] = 0.0;
|
|
vtkCollectionSimpleIterator sit;
|
|
for (i = 0, this->FunctionList->InitTraversal(sit);
|
|
(f=this->FunctionList->GetNextImplicitFunction(sit)); i++)
|
|
{
|
|
c = weights[i];
|
|
if ( c != 0.0 )
|
|
{
|
|
f->FunctionGradient(x, gtmp);
|
|
g[0] += gtmp[0]*c;
|
|
g[1] += gtmp[1]*c;
|
|
g[2] += gtmp[2]*c;
|
|
}
|
|
}
|
|
|
|
if (this->NormalizeByWeight && this->TotalWeight != 0.0)
|
|
{
|
|
g[0] /= this->TotalWeight;
|
|
g[1] /= this->TotalWeight;
|
|
g[2] /= this->TotalWeight;
|
|
}
|
|
}
|
|
|
|
void vtkImplicitSum::PrintSelf(ostream& os, vtkIndent indent)
|
|
{
|
|
this->Superclass::PrintSelf(os,indent);
|
|
|
|
os << indent << "NormalizeByWeight: "
|
|
<< (this->NormalizeByWeight ? "On\n" : "Off\n");
|
|
|
|
os << indent << "Function List:\n";
|
|
this->FunctionList->PrintSelf(os,indent.GetNextIndent());
|
|
|
|
os << indent << "Weights:\n";
|
|
this->Weights->PrintSelf(os,indent.GetNextIndent());
|
|
}
|
|
|