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.
 
 
 
 
 
 

446 lines
10 KiB

/*=========================================================================
Program: Visualization Toolkit
Module: $RCSfile: vtkStructuredData.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 "vtkStructuredData.h"
#include "vtkIdList.h"
#include "vtkObjectFactory.h"
vtkCxxRevisionMacro(vtkStructuredData, "$Revision: 1.60 $");
// Return the topological dimension of the data (e.g., 0, 1, 2, or 3D).
int vtkStructuredData::GetDataDimension(int dataDescription)
{
switch (dataDescription)
{
case VTK_EMPTY: return 0; // Should I put -1?
case VTK_SINGLE_POINT: return 0;
case VTK_X_LINE: case VTK_Y_LINE: case VTK_Z_LINE: return 1;
case VTK_XY_PLANE: case VTK_YZ_PLANE: case VTK_XZ_PLANE: return 2;
case VTK_XYZ_GRID: return 3;
default:
return -1;
}
}
// Specify the dimensions of a regular, rectangular dataset. The input is
// the new dimensions (inDim) and the current dimensions (dim). The function
// returns the dimension of the dataset (0-3D). If the dimensions are
// improperly specified a -1 is returned. If the dimensions are unchanged, a
// value of 100 is returned.
int vtkStructuredData::SetDimensions(int inDim[3], int dim[3])
{
int dataDim, i;
int dataDescription=VTK_UNCHANGED;
if ( inDim[0] != dim[0] || inDim[1] != dim[1] || inDim[2] != dim[2] )
{
for (dataDim=0, i=0; i<3 ; i++)
{
dim[i] = inDim[i];
if (inDim[i] > 1)
{
dataDim++;
}
}
if ( inDim[0]<1 || inDim[1]<1 || inDim[2]<1 )
{
return VTK_EMPTY;
}
if ( dataDim == 3 )
{
dataDescription = VTK_XYZ_GRID;
}
else if ( dataDim == 2)
{
if ( inDim[0] == 1 )
{
dataDescription = VTK_YZ_PLANE;
}
else if ( inDim[1] == 1 )
{
dataDescription = VTK_XZ_PLANE;
}
else
{
dataDescription = VTK_XY_PLANE;
}
}
else if ( dataDim == 1 )
{
if ( inDim[0] != 1 )
{
dataDescription = VTK_X_LINE;
}
else if ( inDim[1] != 1 )
{
dataDescription = VTK_Y_LINE;
}
else
{
dataDescription = VTK_Z_LINE;
}
}
else
{
dataDescription = VTK_SINGLE_POINT;
}
}
return dataDescription;
}
// Specify the extent of a regular, rectangular dataset. The input is
// the new extent (inExt) and the current extent (ext). The function
// returns the dimension of the dataset (0-3D). If the extents are
// improperly specified a -1 is returned. If the dimensions are unchanged, a
// value of 100 is returned.
int vtkStructuredData::SetExtent(int inExt[6], int ext[6])
{
int dataDim, i;
int dataDescription;
if ( inExt[0] == ext[0] && inExt[1] == ext[1] &&
inExt[2] == ext[2] && inExt[3] == ext[3] &&
inExt[4] == ext[4] && inExt[5] == ext[5])
{
return VTK_UNCHANGED;
}
dataDim = 0;
for (i=0; i<3 ; ++i)
{
ext[i*2] = inExt[i*2];
ext[i*2+1] = inExt[i*2+1];
if (inExt[i*2] < inExt[i*2+1])
{
dataDim++;
}
}
if ( inExt[0]>inExt[1] || inExt[2]>inExt[3] || inExt[4]>inExt[5] )
{
return VTK_EMPTY;
}
if ( dataDim == 3 )
{
dataDescription = VTK_XYZ_GRID;
}
else if ( dataDim == 2)
{
if ( inExt[0] == inExt[1] )
{
dataDescription = VTK_YZ_PLANE;
}
else if ( inExt[2] == inExt[3] )
{
dataDescription = VTK_XZ_PLANE;
}
else
{
dataDescription = VTK_XY_PLANE;
}
}
else if ( dataDim == 1 )
{
if ( inExt[0] < inExt[1] )
{
dataDescription = VTK_X_LINE;
}
else if ( inExt[2] < inExt[3] )
{
dataDescription = VTK_Y_LINE;
}
else
{
dataDescription = VTK_Z_LINE;
}
}
else
{
dataDescription = VTK_SINGLE_POINT;
}
return dataDescription;
}
// Get the points defining a cell. (See vtkDataSet for more info.)
void vtkStructuredData::GetCellPoints(vtkIdType cellId, vtkIdList *ptIds,
int dataDescription, int dim[3])
{
int loc[3];
vtkIdType idx, npts;
int iMin, iMax, jMin, jMax, kMin, kMax;
int d01 = dim[0]*dim[1];
ptIds->Reset();
iMin = iMax = jMin = jMax = kMin = kMax = 0;
switch (dataDescription)
{
case VTK_EMPTY:
return;
case VTK_SINGLE_POINT: // cellId can only be = 0
break;
case VTK_X_LINE:
iMin = cellId;
iMax = cellId + 1;
break;
case VTK_Y_LINE:
jMin = cellId;
jMax = cellId + 1;
break;
case VTK_Z_LINE:
kMin = cellId;
kMax = cellId + 1;
break;
case VTK_XY_PLANE:
iMin = cellId % (dim[0]-1);
iMax = iMin + 1;
jMin = cellId / (dim[0]-1);
jMax = jMin + 1;
break;
case VTK_YZ_PLANE:
jMin = cellId % (dim[1]-1);
jMax = jMin + 1;
kMin = cellId / (dim[1]-1);
kMax = kMin + 1;
break;
case VTK_XZ_PLANE:
iMin = cellId % (dim[0]-1);
iMax = iMin + 1;
kMin = cellId / (dim[0]-1);
kMax = kMin + 1;
break;
case VTK_XYZ_GRID:
iMin = cellId % (dim[0] - 1);
iMax = iMin + 1;
jMin = (cellId / (dim[0] - 1)) % (dim[1] - 1);
jMax = jMin + 1;
kMin = cellId / ((dim[0] - 1) * (dim[1] - 1));
kMax = kMin + 1;
break;
}
// Extract point ids
for (npts=0,loc[2]=kMin; loc[2]<=kMax; loc[2]++)
{
for (loc[1]=jMin; loc[1]<=jMax; loc[1]++)
{
for (loc[0]=iMin; loc[0]<=iMax; loc[0]++)
{
idx = loc[0] + loc[1]*dim[0] + loc[2]*d01;
ptIds->InsertId(npts++,idx);
}
}
}
}
// Get the cells using a point. (See vtkDataSet for more info.)
void vtkStructuredData::GetPointCells(vtkIdType ptId, vtkIdList *cellIds,
int dim[3])
{
int cellDim[3];
int ptLoc[3], cellLoc[3];
int i, j;
vtkIdType cellId;
static int offset[8][3] = {{-1,0,0}, {-1,-1,0}, {-1,-1,-1}, {-1,0,-1},
{0,0,0}, {0,-1,0}, {0,-1,-1}, {0,0,-1}};
for (i=0; i<3; i++)
{
cellDim[i] = dim[i] - 1;
if (cellDim[i] == 0)
{
cellDim[i] = 1;
}
}
// Get the location of the point
//
ptLoc[0] = ptId % dim[0];
ptLoc[1] = (ptId / dim[0]) % dim[1];
ptLoc[2] = ptId / (dim[0]*dim[1]);
// From the point location, compute the cell locations. There are at
// most eight possible.
//
cellIds->Reset();
for (j=0; j<8; j++)
{
for (i=0; i<3; i++)
{
cellLoc[i] = ptLoc[i] + offset[j][i];
if ( cellLoc[i] < 0 || cellLoc[i] >= cellDim[i] )
{
break;
}
}
if ( i >= 3 ) //add cell
{
cellId = cellLoc[0] + cellLoc[1]*cellDim[0] +
cellLoc[2]*cellDim[0]*cellDim[1];
cellIds->InsertNextId(cellId);
}
}
return;
}
void vtkStructuredData::GetCellNeighbors(vtkIdType cellId, vtkIdList *ptIds,
vtkIdList *cellIds, int dim[3])
{
int j, seedLoc[3], ptLoc[3], cellLoc[3], cellDim[3];
int offset[8][3];
vtkIdType numPts=ptIds->GetNumberOfIds(), id, i;
cellIds->Reset();
// Start by finding the "space" of the points in i-j-k space.
// The points define a point, line, or plane in topological space,
// which results in degrees of freedom in three, two or one direction.
// The numbers of DOF determines which neighbors to select.
// Start by finding a seed point
id = ptIds->GetId(0);
seedLoc[0] = id % dim[0];
seedLoc[1] = (id / dim[0]) % dim[1];
seedLoc[2] = id / (dim[0]*dim[1]);
// This defines the space around the seed
offset[0][0] = -1; offset[0][1] = -1; offset[0][2] = -1;
offset[1][0] = 0; offset[1][1] = -1; offset[1][2] = -1;
offset[2][0] = -1; offset[2][1] = 0; offset[2][2] = -1;
offset[3][0] = 0; offset[3][1] = 0; offset[3][2] = -1;
offset[4][0] = -1; offset[4][1] = -1; offset[4][2] = 0;
offset[5][0] = 0; offset[5][1] = -1; offset[5][2] = 0;
offset[6][0] = -1; offset[6][1] = 0; offset[6][2] = 0;
offset[7][0] = 0; offset[7][1] = 0; offset[7][2] = 0;
// For the rest of the points, trim the seed region
// This is essentially an intersection of edge neighbors.
for (i=1; i<numPts; i++)
{
// Get the location of the point
id = ptIds->GetId(i);
ptLoc[0] = id % dim[0];
ptLoc[1] = (id / dim[0]) % dim[1];
ptLoc[2] = id / (dim[0]*dim[1]);
if ( (ptLoc[0]-1) == seedLoc[0] )
{
offset[0][0] = -10;
offset[2][0] = -10;
offset[4][0] = -10;
offset[6][0] = -10;
}
else if ( (ptLoc[0]+1) == seedLoc[0] )
{
offset[1][0] = -10;
offset[3][0] = -10;
offset[5][0] = -10;
offset[7][0] = -10;
}
else if ( (ptLoc[1]-1) == seedLoc[1] )
{
offset[0][1] = -10;
offset[1][1] = -10;
offset[4][1] = -10;
offset[5][1] = -10;
}
else if ( (ptLoc[1]+1) == seedLoc[1] )
{
offset[2][1] = -10;
offset[3][1] = -10;
offset[6][1] = -10;
offset[7][1] = -10;
}
else if ( (ptLoc[2]-1) == seedLoc[2] )
{
offset[0][2] = -10;
offset[1][2] = -10;
offset[2][2] = -10;
offset[3][2] = -10;
}
else if ( (ptLoc[2]+1) == seedLoc[2] )
{
offset[4][2] = -10;
offset[5][2] = -10;
offset[6][2] = -10;
offset[7][2] = -10;
}
}
// Load the non-trimmed cells
cellDim[0] = dim[0] - 1;
cellDim[1] = dim[1] - 1;
cellDim[2] = dim[2] - 1;
for(i=0; i<3; i++)
{
if ( cellDim[i] < 1 )
{
cellDim[i] = 1;
}
}
for (j=0; j<8; j++)
{
for (i=0; i<3; i++)
{
if ( offset[j][i] != -10 )
{
cellLoc[i] = seedLoc[i] + offset[j][i];
if ( cellLoc[i] < 0 || cellLoc[i] >= cellDim[i] )
{
break;
}
}
else
{
break;
}
}
if ( i >= 3 ) //add cell
{
id = cellLoc[0] + cellLoc[1]*cellDim[0] +
cellLoc[2]*cellDim[0]*cellDim[1];
if (id != cellId )
{
cellIds->InsertNextId(id);
}
}
}
}