// $Id$ // // Copyright (C) 2005-2006 Rational Discovery LLC // // @@ All Rights Reserved @@ // #include "UniformGrid3D.h" #include #include #include "point.h" #include #define OFFSET_TOL 1.e-8 #define SPACING_TOL 1.e-8 using namespace RDKit; namespace RDGeom { unsigned int ci_GRIDPICKLE_VERSION=0x1; UniformGrid3D::UniformGrid3D(const UniformGrid3D &other){ PRECONDITION(other.dp_storage,"cannot copy an unintialized grid"); RDKit::DiscreteValueVect *data=new RDKit::DiscreteValueVect(*other.dp_storage); initGrid(other.d_numX*other.d_spacing, other.d_numY*other.d_spacing, other.d_numZ*other.d_spacing, other.d_spacing, other.dp_storage->getValueType(), other.d_offSet, data); } void UniformGrid3D::initGrid(double dimX, double dimY, double dimZ, double spacing, RDKit::DiscreteValueVect::DiscreteValueType valType, const RDGeom::Point3D &offSet, RDKit::DiscreteValueVect *data) { PRECONDITION(dimX > 0.0, "Invalid x-dimension for grid"); PRECONDITION(dimY > 0.0, "Invalid y-dimension for grid"); PRECONDITION(dimZ > 0.0, "Invalid z-dimension for grid"); PRECONDITION(spacing > 0.0, "Invalid spacing for grid"); d_numX = static_cast(floor(dimX/spacing + 0.5)); d_numY = static_cast(floor(dimY/spacing + 0.5)); d_numZ = static_cast(floor(dimZ/spacing + 0.5)); PRECONDITION((!data)||data->getValueType()==valType,"grid data type mismatch"); PRECONDITION((!data)||data->getLength()==d_numX*d_numY*d_numZ,"grid data size mismatch"); d_spacing = spacing; d_offSet = offSet; if(!data){ dp_storage = new RDKit::DiscreteValueVect(valType, d_numX*d_numY*d_numZ); } else { dp_storage = data; } } UniformGrid3D::UniformGrid3D(const std::string pkl) { dp_storage=0; this->initFromText(pkl.c_str(),pkl.size()); } UniformGrid3D::UniformGrid3D(const char *pkl, const unsigned int len){ dp_storage=0; this->initFromText(pkl,len); } UniformGrid3D::~UniformGrid3D() { if (dp_storage) { delete dp_storage; } } int UniformGrid3D::getGridIndex(unsigned int xi, unsigned int yi, unsigned int zi) const { if (xi >= d_numX) { return -1; } if (yi >= d_numY) { return -1; } if (zi >= d_numZ) { return -1; } return (zi*d_numX*d_numY + yi*d_numX + xi); } int UniformGrid3D::getGridPointIndex(const Point3D &point) const { Point3D tPt(point); tPt -= d_offSet;//d_origin; tPt /= d_spacing; int xi, yi, zi; double move = 0.5; xi = static_cast(floor(tPt.x + move)); yi = static_cast(floor(tPt.y + move)); zi = static_cast(floor(tPt.z + move)); if ((xi < 0) || (xi >= static_cast(d_numX))) { return -1; } if ((yi < 0) || (yi >= static_cast(d_numY))) { return -1; } if ((zi < 0) || (zi >= static_cast(d_numZ))) { return -1; } return (zi*d_numX*d_numY + yi*d_numX + xi); } int UniformGrid3D::getVal(const Point3D &point) const { int id = getGridPointIndex(point); if (id < 0) { return -1; } return dp_storage->getVal(static_cast(id)); } unsigned int UniformGrid3D::getVal(unsigned int pointId) const { return dp_storage->getVal(pointId); } void UniformGrid3D::setVal(const Point3D &point, unsigned int val) { int id = getGridPointIndex(point); if (id < 0) { return; } dp_storage->setVal(static_cast(id), val); } Point3D UniformGrid3D::getGridPointLoc(unsigned int pointId) const { RANGE_CHECK(0, pointId, d_numX*d_numY*d_numZ-1); Point3D res; res.x = (pointId%d_numX)*d_spacing; res.y = ((pointId%(d_numX*d_numY))/d_numX)*d_spacing; res.z = (pointId/(d_numX*d_numY))*d_spacing; res += d_offSet; //d_origin; return res; } void UniformGrid3D::setVal(unsigned int pointId, unsigned int val) { dp_storage->setVal(pointId, val); } bool UniformGrid3D::compareParams(const UniformGrid3D &other) const { if (d_numX != other.getNumX()) return false; if (d_numY != other.getNumY()) return false; if (d_numZ != other.getNumZ()) return false; if (fabs(d_spacing - other.getSpacing()) > SPACING_TOL) return false; Point3D dOffset = d_offSet; dOffset -= other.getOffset(); if (dOffset.lengthSq() > OFFSET_TOL) { return false; } return true; } void UniformGrid3D::setSphereOccupancy(const Point3D & center, double radius, double stepSize, int maxNumLayers, bool ignoreOutOfBound) { unsigned int ptIndex = this->getGridPointIndex(center); if (ptIndex == -1) { if (ignoreOutOfBound) { return; } else { throw GridException("Center outside the grdi boundary"); } } Point3D gPt(center); // point on the grid gPt -= d_offSet; gPt /= d_spacing; unsigned int z = ptIndex/(d_numX*d_numY); unsigned int y = (ptIndex%(d_numX*d_numY))/d_numX; unsigned int x = ptIndex%d_numX; unsigned int bPerVal = dp_storage->getNumBitsPerVal(); unsigned int maxVal = (1 << bPerVal) - 1; unsigned int nLayers = maxVal; unsigned int valStep = 1; if ((maxNumLayers > 0) && (maxNumLayers <= static_cast(nLayers))) { nLayers = maxNumLayers; valStep = (maxVal+1)/nLayers; } double bgRad = radius/d_spacing; // base radius in grid coords double gStepSize = stepSize/d_spacing; //step size in grid coords double gRadius = bgRad + nLayers*gStepSize; // largest radius in grid coords double gRad2 = gRadius*gRadius; double bgRad2 = bgRad*bgRad; int i, j, k; double dx, dy, dz, d, d2, dy2z2, dz2; int xmin, xmax, ymin, ymax, zmin, zmax; xmax = (int)floor(gPt.x + gRadius); xmin = (int)ceil(gPt.x - gRadius); ymax = (int)floor(gPt.y + gRadius); ymin = (int)ceil(gPt.y - gRadius); zmax = (int)floor(gPt.z + gRadius); zmin = (int)ceil(gPt.z - gRadius); unsigned int oval, val, valChange; int ptId1, ptId2; for (k = zmin; k <= zmax; ++k) { if ((k >= 0) && (k < (int)d_numZ)) { // we are inside the grid in the z-direction dz = static_cast(k) - gPt.z; dz2 = dz*dz; ptId1 = k*d_numX*d_numY; for (j = ymin; j <= ymax; ++j) { if ((j >= 0) && (j < (int)d_numY)) { // inside the grid in the y-direction dy = static_cast(j) - gPt.y; dy2z2 = dy*dy + dz2; if (dy2z2 < gRad2) { // we are within the radius at least from the y,z coordinates ptId2 = ptId1 + j*d_numX; for (i = xmin; i <= xmax; ++i) { if ((i >= 0) && (i < (int)d_numX)) { oval = dp_storage->getVal((unsigned int)(ptId2+i)); if (oval < maxVal) { // if we are already at maxVal we will not change that dx = static_cast(i) - gPt.x; d2 = dx*dx + dy2z2; if (d2 < gRad2) { //we are inside the sphere if (d2 < bgRad2) { val = maxVal; } else { d = sqrt(d2); valChange = (static_cast((d - bgRad)/gStepSize + 1))*(valStep); if (valChange < maxVal) { val = maxVal - valChange; } else { val = 0; } } if (val > oval) { dp_storage->setVal(ptId2+i, val); } } // we are inside the sphere } // grid point does not already have maxVal } // inside the grid in x-direction } // loop over points in x-direction } // inside the sphere based on only z and y coords } // we are inside the grid in the y-direction } // loop over points in y-direction } // inside grid in z-direction } // loop over points in z-direction } UniformGrid3D &UniformGrid3D::operator|=(const UniformGrid3D &other) { PRECONDITION(dp_storage,"unintialized grid"); PRECONDITION(other.dp_storage,"unintialized grid"); PRECONDITION(compareParams(other),"incompatible grids"); // EFF: we're probably doing too much copying here: RDKit::DiscreteValueVect *newData=new RDKit::DiscreteValueVect((*dp_storage) | (*other.dp_storage)); delete dp_storage; dp_storage = newData; return *this; } UniformGrid3D &UniformGrid3D::operator&=(const UniformGrid3D &other) { PRECONDITION(dp_storage,"unintialized grid"); PRECONDITION(other.dp_storage,"unintialized grid"); PRECONDITION(compareParams(other),"incompatible grids"); // EFF: we're probably doing too much copying here: RDKit::DiscreteValueVect *newData=new RDKit::DiscreteValueVect((*dp_storage) & (*other.dp_storage)); delete dp_storage; dp_storage = newData; return *this; } UniformGrid3D &UniformGrid3D::operator+=(const UniformGrid3D &other) { PRECONDITION(dp_storage,"unintialized grid"); PRECONDITION(other.dp_storage,"unintialized grid"); PRECONDITION(compareParams(other),"incompatible grids"); // EFF: we're probably doing too much copying here: *dp_storage += *other.dp_storage; return *this; } UniformGrid3D &UniformGrid3D::operator-=(const UniformGrid3D &other) { PRECONDITION(dp_storage,"unintialized grid"); PRECONDITION(other.dp_storage,"unintialized grid"); PRECONDITION(compareParams(other),"incompatible grids"); // EFF: we're probably doing too much copying here: *dp_storage -= *other.dp_storage; return *this; } std::string UniformGrid3D::toString() const { std::stringstream ss(std::ios_base::binary|std::ios_base::out|std::ios_base::in); int tVers = ci_GRIDPICKLE_VERSION*-1; streamWrite(ss,tVers); streamWrite(ss,d_numX); streamWrite(ss,d_numY); streamWrite(ss,d_numZ); streamWrite(ss,d_spacing); streamWrite(ss,d_offSet.x); streamWrite(ss,d_offSet.y); streamWrite(ss,d_offSet.z); std::string storePkl=dp_storage->toString(); unsigned int pklSz=storePkl.size(); streamWrite(ss,pklSz); ss.write(storePkl.c_str(),pklSz*sizeof(char)); std::string res(ss.str()); return(res); } void UniformGrid3D::initFromText(const char *pkl,const unsigned int length){ std::stringstream ss(std::ios_base::binary|std::ios_base::in|std::ios_base::out); ss.write(pkl,length); int tVers; streamRead(ss,tVers); tVers *= -1; if(tVers==0x1){ } else { throw ValueErrorException("bad version in UniformGrid3D pickle"); } streamRead(ss,d_numX); streamRead(ss,d_numY); streamRead(ss,d_numZ); streamRead(ss,d_spacing); double oX,oY,oZ; streamRead(ss,oX); streamRead(ss,oY); streamRead(ss,oZ); d_offSet = Point3D(oX,oY,oZ); unsigned int pklSz; streamRead(ss,pklSz); char *buff = new char[pklSz]; ss.read(buff,pklSz*sizeof(char)); if(dp_storage) delete dp_storage; dp_storage = new RDKit::DiscreteValueVect(buff,pklSz); delete [] buff; } void writeGridToStream(const UniformGrid3D &grid, std::ostream &outStrm) { int dimX = (int)grid.getNumX();//+2; int dimY = (int)grid.getNumY();//+2; int dimZ = (int)grid.getNumZ();//+2; double spacing = grid.getSpacing(); double lenX = dimX*spacing; double lenY = dimY*spacing; double lenZ = dimZ*spacing; Point3D offSet = grid.getOffset(); offSet /= spacing; outStrm << "Grid file representing a Shape \n\n"; outStrm << lenX << " " << lenY << " " << lenZ << " 90.0 90.0 90.0" << std::endl; outStrm << dimX-1 << " " << dimY-1 << " " << dimZ-1 << std::endl; int outX1 = (int)(floor(offSet.x+0.5)); int outX2 = (int)(floor(offSet.x+0.5))+ (int)(dimX-1); // REVIEW: ok - here is a fix to try and make the grid closer to the molecule when displayed // (atleast in PyMol). The difference between the pair of values (outX1, outX2) is (dimX-1), // that is not the case with pairs the (outY1, outY2) a (outZ1, outZ2). In these cases, the difference is // dimY and dimZ respectively. Not sure why this should be the case, but almost always we get a // better display in PyMol. int outY1 = (int)(floor(offSet.y+0.5)); int outY2 = (int)(floor(offSet.y+0.5))+ (int)(dimY); int outZ1 = (int)(floor(offSet.z+0.5)); int outZ2 = (int)(floor(offSet.z+0.5))+ (int)(dimZ); outStrm << "1" << " " << outX1 << " " << outX2 << " " << outY1 << " " << outY2 << " " << outZ1 << " " << outZ2 << "\n"; unsigned int i, nPts = grid.getSize(); for (i = 0; i < nPts; i++) { outStrm << static_cast(grid.getVal(i)) << std::endl; } } void writeGridToFile(const UniformGrid3D &grid, std::string filename) { //std::ofstream ofStrm(filename.c_str()); std::ofstream *ofStrm = new std::ofstream(filename.c_str()); std::ostream *oStrm = static_cast(ofStrm); writeGridToStream(grid, *oStrm); delete ofStrm; } }