/*========================================================================= Program: Visualization Toolkit Module: vtkAMRBox.h 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. =========================================================================*/ /** * @class vtkAMRBox * @brief Encloses a rectangular region of voxel like cells. * * * vtkAMRBox stores information for an AMR block * * @sa * vtkAMRInformation */ #ifndef vtkAMRBox_h #define vtkAMRBox_h #include "vtkCommonDataModelModule.h" // For export macro #include "vtkObject.h" #include "vtkStructuredData.h" // For VTK_XYZ_GRID definition class VTKCOMMONDATAMODEL_EXPORT vtkAMRBox { public: /** * Construct the empty box. */ vtkAMRBox(); /** * Copy construct this box from another. */ vtkAMRBox(const vtkAMRBox &other); /** * Construct a specific 3D box. */ vtkAMRBox( int ilo,int jlo,int klo, int ihi,int jhi,int khi); /** * Construct an AMR box from the description a vtkUniformGrid * Note that the dimensions specify the node dimensions, rather than the cell dimensions */ vtkAMRBox(const double* origin, const int* dimensions, const double* spacing, const double* globalOrigin, int gridDescription=VTK_XYZ_GRID); /** * Construct a specific box. (ilo,jlo,klo,)(ihi,jhi,khi) */ vtkAMRBox(const int lo[3], const int hi[3]); vtkAMRBox(const int dims[6]); /** * Copy the other box to this box. */ vtkAMRBox &operator=(const vtkAMRBox &other); virtual ~vtkAMRBox(){} //@{ /** * Set the box to be invalid; */ void Invalidate() { this->LoCorner[0]=this->LoCorner[1]=this->LoCorner[2]=0; this->HiCorner[0]=this->HiCorner[1]=this->HiCorner[2]=-2; } //@} /** * Whether dimension i is empty, e.g. if the data set is type VTK_XY_PLANE */ bool EmptyDimension(int i) const{ return HiCorner[i]<=LoCorner[i]-1;} /** * Set the dimensions of the box. ilo,jlo,klo,ihi,jhi,khi */ void SetDimensions( int ilo, int jlo, int klo, int ihi, int jhi, int khi, int desc=VTK_XYZ_GRID); /** * Set the dimensions of the box. (ilo,jlo,klo),(ihi,jhi,khi) */ void SetDimensions(const int lo[3], const int hi[3], int desc=VTK_XYZ_GRID); /** * Set the dimensions of the box. (ilo,ihi,jlo,jhi,klo,khi) */ void SetDimensions(const int dims[6], int desc = VTK_XYZ_GRID); /** * Get the dimensions of this box. (ilo,jlo,jhi),(ihi,jhi,khi) */ void GetDimensions(int lo[3], int hi[3]) const; /** * Get the dimensions of this box. (ilo,ihi, jlo,jhi, klo,khi) */ void GetDimensions(int dims[6]) const; //@{ /** * Gets the number of cells enclosed by the box. */ vtkIdType GetNumberOfCells() const; void GetNumberOfCells(int num[3]) const; //@} //@{ /** * Gets the number of nodes required to construct * a physical representation of the box. */ void GetNumberOfNodes(int ext[3]) const; vtkIdType GetNumberOfNodes() const; //@} /** * Determines the dimension of the AMR box given the * box indices. Note, the AMR box can be on an arbitrary * axis-aligned plane, i.e., XZ or YZ. */ int ComputeDimension() const; /** * Get the low corner index. */ const int *GetLoCorner() const { return this->LoCorner; } const int *GetHiCorner() const { return this->HiCorner; } /** * Return a high corner. If dimension j is empty, * then hi[j] is set from lo[j]. This is convenient * For algorithm that must iterate over all cells */ void GetValidHiCorner(int hi[3]) const; bool Empty() const {return this->IsInvalid();} /** * Check to see if the AMR box instance is invalid. */ bool IsInvalid() const { return ((this->HiCorner[0] < this->LoCorner[0]-1) || (this->HiCorner[1] < this->LoCorner[1]-1) || (this->HiCorner[2] < this->LoCorner[2]-1)); } /** * Test if this box is equal with the box instance on the rhs. * Note: Two AMR boxes are equal if: (a) they have the same dimensionality * (b) they are at the same level and (c) they occupy the same index space. */ bool operator==(const vtkAMRBox &other) const; /** * Test if this box is NOT equal with the box instance on the rhs. * Note: Two AMR boxes are equal if: (a) they have the same dimensionality * (b) they are at the same level and (c) they occupy the same index space. */ bool operator!=(const vtkAMRBox &other) const { return( !(*this == other) ); } /** * Send the box to a stream. "(ilo,jlo,jhi),(ihi,jhi,khi)" */ ostream &Print(ostream &os) const; //@{ /** * Serializes this object instance into a byte-stream. * buffer -- user-supplied pointer where the serialized object is stored. * bytesize -- number of bytes, i.e., the size of the buffer. * NOTE: buffer is allocated internally by this method. * Pre-conditions: * buffer == nullptr * Post-conditions: * buffer != nullptr * bytesize != 0 */ void Serialize( unsigned char*& buffer, vtkIdType &bytesize ); void Serialize(int* buffer) const; //@} /** * Deserializes this object instance from the given byte-stream. * Pre-conditions: * buffer != nullptr * bytesize != 0 */ void Deserialize( unsigned char* buffer, const vtkIdType &bytesize ); /** * Checks if this instance of vtkAMRBox intersects with the box passed through * the argument list along the given dimension q. True is returned iff the box * intersects successfully. Otherwise, there is no intersection along the * given dimension and false is returned. */ bool DoesBoxIntersectAlongDimension(const vtkAMRBox &other, const int q) const; bool DoesIntersect(const vtkAMRBox &other) const; /** * Coarsen the box. */ void Coarsen(int r); /** * Refine the box. */ void Refine(int r); //@{ /** * Grows the box in all directions. */ void Grow(int byN); void Shrink(int byN); //@} //@{ /** * Shifts the box in index space */ void Shift(int i, int j, int k); void Shift(const int I[3]); //@} /** * Intersect this box with another box in place. Returns * true if the boxes do intersect. Note that the box is * modified to be the intersection or is made invalid. */ bool Intersect(const vtkAMRBox &other); //@{ /** * Test to see if a given cell index is inside this box. */ bool Contains(int i,int j,int k) const; bool Contains(const int I[3]) const; //@} /** * Test to see if a given box is inside this box. */ bool Contains(const vtkAMRBox&) const; /** * Given an AMR box and the refinement ratio, r, this method computes the * number of ghost layers in each of the 6 directions, i.e., * [imin,imax,jmin,jmax,kmin,kmax] */ void GetGhostVector(int r, int nghost[6]) const; /** * Given an AMR box and the refinement ratio, r, this shrinks * the AMRBox */ void RemoveGhosts(int r); public: /** * Returns the number of bytes allocated by this instance. In addition, * this number of bytes corresponds to the buffer size required to serialize * any vtkAMRBox instance. */ static vtkIdType GetBytesize(){return 6*sizeof(int); }; /** * Returns the linear index of the given cell structured coordinates */ static int GetCellLinearIndex(const vtkAMRBox& box, const int i, const int j, const int k, int imageDimension[3] ); /** * Get the bounds of this box. */ static void GetBounds(const vtkAMRBox& box, const double origin[3], const double spacing[3],double bounds[6]) ; /** * Get the world space origin of this box. The origin is the * location of the lower corner cell's lower corner node, */ static void GetBoxOrigin(const vtkAMRBox& box, const double X0[3], const double spacing[3],double x0[3]); /** * Checks if the point is inside this AMRBox instance. * x,y,z the world point */ static bool HasPoint(const vtkAMRBox& box, const double origin[3], const double spacing[3], double x, double y, double z ); /** * Compute structured coordinates */ static int ComputeStructuredCoordinates(const vtkAMRBox& box, const double dataOrigin[3], const double h[3], const double x[3], int ijk[3], double pcoords[3]); protected: /** * Initializes this box instance. */ void Initialize( ); /** * Intersects this instance of vtkAMRbox with box passed through the argument * list along the given dimension q. True is returned iff the box intersects * successfully. Otherwise, false is returned if there is no intersection at * the given dimension. */ bool IntersectBoxAlongDimension(const vtkAMRBox &other, const int q); private: int LoCorner[3]; // lo corner cell id. int HiCorner[3]; // hi corner cell id. //@{ /** * This method builds the AMR box with the given dimensions. * Note: the dimension of the AMR box is automatically detected * within this method. */ void BuildAMRBox( const int ilo, const int jlo, const int klo, const int ihi, const int jhi, const int khi ); }; //@} //***************************************************************************** //@{ /** * Fill the region of "pArray" enclosed by "destRegion" with "fillValue" * "pArray" is defined on "arrayRegion". */ template void FillRegion( T *pArray, const vtkAMRBox &arrayRegion, const vtkAMRBox &destRegion, T fillValue) { // Convert regions to array index space. VTK arrays // always start with 0,0,0. int ofs[3]; ofs[0]=-arrayRegion.GetLoCorner()[0]; ofs[1]=-arrayRegion.GetLoCorner()[1]; ofs[2]=-arrayRegion.GetLoCorner()[2]; vtkAMRBox arrayDims(arrayRegion); arrayDims.Shift(ofs); vtkAMRBox destDims(destRegion); destDims.Shift(ofs); // Quick sanity check. if (!arrayRegion.Contains(destRegion)) { vtkGenericWarningMacro( << "ERROR: Array must enclose the destination region. " << "Aborting the fill."); } // Get the bounds of the indices we fill. const int* destLo = destDims.GetLoCorner(); int destHi[3]; destDims.GetValidHiCorner(destHi); // Get the array dimensions. int arrayHi[3]; arrayDims.GetNumberOfCells(arrayHi); // Fill. for (int k=destLo[2]; k<=destHi[2]; ++k) { vtkIdType kOfs=k*arrayHi[0]*arrayHi[1]; for (int j=destLo[1]; j<=destHi[1]; ++j) { vtkIdType idx=kOfs+j*arrayHi[0]+destLo[0]; for (int i=destLo[0]; i<=destHi[0]; ++i) { pArray[idx]=fillValue; ++idx; } } } //@} } #endif // VTK-HeaderTest-Exclude: vtkAMRBox.h