| [bcf653] | 1 | /* | 
|---|
|  | 2 | * Project: MoleCuilder | 
|---|
|  | 3 | * Description: creates and alters molecular systems | 
|---|
| [0aa122] | 4 | * Copyright (C)  2010-2012 University of Bonn. All rights reserved. | 
|---|
| [5aaa43] | 5 | * Copyright (C)  2013 Frederik Heber. All rights reserved. | 
|---|
| [94d5ac6] | 6 | * | 
|---|
|  | 7 | * | 
|---|
|  | 8 | *   This file is part of MoleCuilder. | 
|---|
|  | 9 | * | 
|---|
|  | 10 | *    MoleCuilder is free software: you can redistribute it and/or modify | 
|---|
|  | 11 | *    it under the terms of the GNU General Public License as published by | 
|---|
|  | 12 | *    the Free Software Foundation, either version 2 of the License, or | 
|---|
|  | 13 | *    (at your option) any later version. | 
|---|
|  | 14 | * | 
|---|
|  | 15 | *    MoleCuilder is distributed in the hope that it will be useful, | 
|---|
|  | 16 | *    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|---|
|  | 17 | *    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|---|
|  | 18 | *    GNU General Public License for more details. | 
|---|
|  | 19 | * | 
|---|
|  | 20 | *    You should have received a copy of the GNU General Public License | 
|---|
|  | 21 | *    along with MoleCuilder.  If not, see <http://www.gnu.org/licenses/>. | 
|---|
| [bcf653] | 22 | */ | 
|---|
|  | 23 |  | 
|---|
| [f66195] | 24 | /** \file boundary.cpp | 
|---|
| [edb93c] | 25 | * | 
|---|
|  | 26 | * Implementations and super-function for envelopes | 
|---|
| [2319ed] | 27 | */ | 
|---|
|  | 28 |  | 
|---|
| [bf3817] | 29 | // include config.h | 
|---|
|  | 30 | #ifdef HAVE_CONFIG_H | 
|---|
|  | 31 | #include <config.h> | 
|---|
|  | 32 | #endif | 
|---|
|  | 33 |  | 
|---|
| [ad011c] | 34 | #include "CodePatterns/MemDebug.hpp" | 
|---|
| [112b09] | 35 |  | 
|---|
| [6f0841] | 36 | #include "Atom/atom.hpp" | 
|---|
| [129204] | 37 | #include "Bond/bond.hpp" | 
|---|
| [8eb17a] | 38 | #include "boundary.hpp" | 
|---|
| [34c43a] | 39 | #include "BoundaryLineSet.hpp" | 
|---|
|  | 40 | #include "BoundaryPointSet.hpp" | 
|---|
|  | 41 | #include "BoundaryTriangleSet.hpp" | 
|---|
|  | 42 | #include "Box.hpp" | 
|---|
|  | 43 | #include "CandidateForTesselation.hpp" | 
|---|
|  | 44 | #include "CodePatterns/Info.hpp" | 
|---|
|  | 45 | #include "CodePatterns/Log.hpp" | 
|---|
|  | 46 | #include "CodePatterns/Verbose.hpp" | 
|---|
| [f66195] | 47 | #include "config.hpp" | 
|---|
| [3bdb6d] | 48 | #include "Element/element.hpp" | 
|---|
| [34c43a] | 49 | #include "LinearAlgebra/Plane.hpp" | 
|---|
|  | 50 | #include "LinearAlgebra/RealSpaceMatrix.hpp" | 
|---|
| [53c7fc] | 51 | #include "LinkedCell/linkedcell.hpp" | 
|---|
|  | 52 | #include "LinkedCell/PointCloudAdaptor.hpp" | 
|---|
| [f66195] | 53 | #include "molecule.hpp" | 
|---|
| [34c43a] | 54 | #include "RandomNumbers/RandomNumberGeneratorFactory.hpp" | 
|---|
|  | 55 | #include "RandomNumbers/RandomNumberGenerator.hpp" | 
|---|
| [f66195] | 56 | #include "tesselation.hpp" | 
|---|
|  | 57 | #include "tesselationhelpers.hpp" | 
|---|
| [b34306] | 58 | #include "World.hpp" | 
|---|
| [2319ed] | 59 |  | 
|---|
| [986ed3] | 60 | #include <iostream> | 
|---|
| [36166d] | 61 | #include <iomanip> | 
|---|
|  | 62 |  | 
|---|
| [357fba] | 63 | #include<gsl/gsl_poly.h> | 
|---|
| [d6eb80] | 64 | #include<time.h> | 
|---|
| [2319ed] | 65 |  | 
|---|
| [357fba] | 66 | // ========================================== F U N C T I O N S ================================= | 
|---|
| [2319ed] | 67 |  | 
|---|
|  | 68 |  | 
|---|
| [357fba] | 69 | /** Determines greatest diameters of a cluster defined by its convex envelope. | 
|---|
|  | 70 | * Looks at lines parallel to one axis and where they intersect on the projected planes | 
|---|
| [2319ed] | 71 | * \param *out output stream for debugging | 
|---|
| [357fba] | 72 | * \param *BoundaryPoints NDIM set of boundary points defining the convex envelope on each projected plane | 
|---|
|  | 73 | * \param *mol molecule structure representing the cluster | 
|---|
| [776b64] | 74 | * \param *&TesselStruct Tesselation structure with triangles | 
|---|
| [357fba] | 75 | * \param IsAngstroem whether we have angstroem or atomic units | 
|---|
|  | 76 | * \return NDIM array of the diameters | 
|---|
| [e4ea46] | 77 | */ | 
|---|
| [e138de] | 78 | double *GetDiametersOfCluster(const Boundaries *BoundaryPtr, const molecule *mol, Tesselation *&TesselStruct, const bool IsAngstroem) | 
|---|
| [caf5d6] | 79 | { | 
|---|
| [ce7bfd] | 80 | //Info FunctionInfo(__func__); | 
|---|
| [357fba] | 81 | // get points on boundary of NULL was given as parameter | 
|---|
|  | 82 | bool BoundaryFreeFlag = false; | 
|---|
| [ad37ab] | 83 | double OldComponent = 0.; | 
|---|
|  | 84 | double tmp = 0.; | 
|---|
|  | 85 | double w1 = 0.; | 
|---|
|  | 86 | double w2 = 0.; | 
|---|
|  | 87 | Vector DistanceVector; | 
|---|
|  | 88 | Vector OtherVector; | 
|---|
|  | 89 | int component = 0; | 
|---|
|  | 90 | int Othercomponent = 0; | 
|---|
| [776b64] | 91 | Boundaries::const_iterator Neighbour; | 
|---|
|  | 92 | Boundaries::const_iterator OtherNeighbour; | 
|---|
| [ad37ab] | 93 | double *GreatestDiameter = new double[NDIM]; | 
|---|
|  | 94 |  | 
|---|
| [776b64] | 95 | const Boundaries *BoundaryPoints; | 
|---|
|  | 96 | if (BoundaryPtr == NULL) { | 
|---|
| [357fba] | 97 | BoundaryFreeFlag = true; | 
|---|
| [e138de] | 98 | BoundaryPoints = GetBoundaryPoints(mol, TesselStruct); | 
|---|
| [86234b] | 99 | } else { | 
|---|
| [776b64] | 100 | BoundaryPoints = BoundaryPtr; | 
|---|
| [47d041] | 101 | LOG(0, "Using given boundary points set."); | 
|---|
| [86234b] | 102 | } | 
|---|
| [357fba] | 103 | // determine biggest "diameter" of cluster for each axis | 
|---|
|  | 104 | for (int i = 0; i < NDIM; i++) | 
|---|
|  | 105 | GreatestDiameter[i] = 0.; | 
|---|
|  | 106 | for (int axis = 0; axis < NDIM; axis++) | 
|---|
|  | 107 | { // regard each projected plane | 
|---|
| [47d041] | 108 | //LOG(1, "Current axis is " << axis << "."); | 
|---|
| [357fba] | 109 | for (int j = 0; j < 2; j++) | 
|---|
|  | 110 | { // and for both axis on the current plane | 
|---|
|  | 111 | component = (axis + j + 1) % NDIM; | 
|---|
|  | 112 | Othercomponent = (axis + 1 + ((j + 1) & 1)) % NDIM; | 
|---|
| [47d041] | 113 | //LOG(1, "Current component is " << component << ", Othercomponent is " << Othercomponent << "."); | 
|---|
| [776b64] | 114 | for (Boundaries::const_iterator runner = BoundaryPoints[axis].begin(); runner != BoundaryPoints[axis].end(); runner++) { | 
|---|
| [47d041] | 115 | //LOG(1, "Current runner is " << *(runner->second.second) << "."); | 
|---|
| [357fba] | 116 | // seek for the neighbours pair where the Othercomponent sign flips | 
|---|
|  | 117 | Neighbour = runner; | 
|---|
|  | 118 | Neighbour++; | 
|---|
|  | 119 | if (Neighbour == BoundaryPoints[axis].end()) // make it wrap around | 
|---|
|  | 120 | Neighbour = BoundaryPoints[axis].begin(); | 
|---|
| [d74077] | 121 | DistanceVector = (runner->second.second->getPosition()) - (Neighbour->second.second->getPosition()); | 
|---|
| [776b64] | 122 | do { // seek for neighbour pair where it flips | 
|---|
| [0a4f7f] | 123 | OldComponent = DistanceVector[Othercomponent]; | 
|---|
| [357fba] | 124 | Neighbour++; | 
|---|
|  | 125 | if (Neighbour == BoundaryPoints[axis].end()) // make it wrap around | 
|---|
|  | 126 | Neighbour = BoundaryPoints[axis].begin(); | 
|---|
| [d74077] | 127 | DistanceVector = (runner->second.second->getPosition()) - (Neighbour->second.second->getPosition()); | 
|---|
| [47d041] | 128 | //LOG(2, "OldComponent is " << OldComponent << ", new one is " << DistanceVector.x[Othercomponent] << "."); | 
|---|
| [776b64] | 129 | } while ((runner != Neighbour) && (fabs(OldComponent / fabs( | 
|---|
| [0a4f7f] | 130 | OldComponent) - DistanceVector[Othercomponent] / fabs( | 
|---|
|  | 131 | DistanceVector[Othercomponent])) < MYEPSILON)); // as long as sign does not flip | 
|---|
| [776b64] | 132 | if (runner != Neighbour) { | 
|---|
| [357fba] | 133 | OtherNeighbour = Neighbour; | 
|---|
|  | 134 | if (OtherNeighbour == BoundaryPoints[axis].begin()) // make it wrap around | 
|---|
|  | 135 | OtherNeighbour = BoundaryPoints[axis].end(); | 
|---|
|  | 136 | OtherNeighbour--; | 
|---|
| [47d041] | 137 | //LOG(1, "The pair, where the sign of OtherComponent flips, is: " << *(Neighbour->second.second) << " and " << *(OtherNeighbour->second.second) << "."); | 
|---|
| [357fba] | 138 | // now we have found the pair: Neighbour and OtherNeighbour | 
|---|
| [d74077] | 139 | OtherVector = (runner->second.second->getPosition()) - (OtherNeighbour->second.second->getPosition()); | 
|---|
| [47d041] | 140 | //LOG(1, "Distances to Neighbour and OtherNeighbour are " << DistanceVector.x[component] << " and " << OtherVector.x[component] << "."); | 
|---|
|  | 141 | //LOG(1, "OtherComponents to Neighbour and OtherNeighbour are " << DistanceVector.x[Othercomponent] << " and " << OtherVector.x[Othercomponent] << "."); | 
|---|
| [357fba] | 142 | // do linear interpolation between points (is exact) to extract exact intersection between Neighbour and OtherNeighbour | 
|---|
| [0a4f7f] | 143 | w1 = fabs(OtherVector[Othercomponent]); | 
|---|
|  | 144 | w2 = fabs(DistanceVector[Othercomponent]); | 
|---|
|  | 145 | tmp = fabs((w1 * DistanceVector[component] + w2 | 
|---|
|  | 146 | * OtherVector[component]) / (w1 + w2)); | 
|---|
| [357fba] | 147 | // mark if it has greater diameter | 
|---|
| [47d041] | 148 | //LOG(1, "Comparing current greatest " << GreatestDiameter[component] << " to new " << tmp << "."); | 
|---|
| [357fba] | 149 | GreatestDiameter[component] = (GreatestDiameter[component] | 
|---|
|  | 150 | > tmp) ? GreatestDiameter[component] : tmp; | 
|---|
|  | 151 | } //else | 
|---|
| [47d041] | 152 | //LOG(1, "Saw no sign flip, probably top or bottom node."); | 
|---|
| [3d919e] | 153 | } | 
|---|
|  | 154 | } | 
|---|
|  | 155 | } | 
|---|
| [47d041] | 156 | LOG(0, "RESULT: The biggest diameters are " | 
|---|
| [357fba] | 157 | << GreatestDiameter[0] << " and " << GreatestDiameter[1] << " and " | 
|---|
|  | 158 | << GreatestDiameter[2] << " " << (IsAngstroem ? "angstrom" | 
|---|
| [47d041] | 159 | : "atomiclength") << "."); | 
|---|
| [03648b] | 160 |  | 
|---|
| [357fba] | 161 | // free reference lists | 
|---|
|  | 162 | if (BoundaryFreeFlag) | 
|---|
|  | 163 | delete[] (BoundaryPoints); | 
|---|
| [e4ea46] | 164 |  | 
|---|
| [357fba] | 165 | return GreatestDiameter; | 
|---|
| [e4ea46] | 166 | } | 
|---|
|  | 167 | ; | 
|---|
| [03648b] | 168 |  | 
|---|
| [042f82] | 169 |  | 
|---|
| [357fba] | 170 | /** Determines the boundary points of a cluster. | 
|---|
|  | 171 | * Does a projection per axis onto the orthogonal plane, transforms into spherical coordinates, sorts them by the angle | 
|---|
|  | 172 | * and looks at triples: if the middle has less a distance than the allowed maximum height of the triangle formed by the plane's | 
|---|
|  | 173 | * center and first and last point in the triple, it is thrown out. | 
|---|
| [f01769] | 174 | * | 
|---|
|  | 175 | * \todo When storing const ptrs in tesselation structures, remove const_cast | 
|---|
|  | 176 | * | 
|---|
| [357fba] | 177 | * \param *out output stream for debugging | 
|---|
|  | 178 | * \param *mol molecule structure representing the cluster | 
|---|
| [776b64] | 179 | * \param *&TesselStruct pointer to Tesselation structure | 
|---|
| [e4ea46] | 180 | */ | 
|---|
| [e138de] | 181 | Boundaries *GetBoundaryPoints(const molecule *mol, Tesselation *&TesselStruct) | 
|---|
| [caf5d6] | 182 | { | 
|---|
| [ce7bfd] | 183 | //Info FunctionInfo(__func__); | 
|---|
| [357fba] | 184 | PointMap PointsOnBoundary; | 
|---|
|  | 185 | LineMap LinesOnBoundary; | 
|---|
|  | 186 | TriangleMap TrianglesOnBoundary; | 
|---|
| [833b15] | 187 | Vector MolCenter = mol->DetermineCenterOfAll(); | 
|---|
| [357fba] | 188 | Vector helper; | 
|---|
| [ad37ab] | 189 | BoundariesTestPair BoundaryTestPair; | 
|---|
|  | 190 | Vector AxisVector; | 
|---|
|  | 191 | Vector AngleReferenceVector; | 
|---|
|  | 192 | Vector AngleReferenceNormalVector; | 
|---|
|  | 193 | Vector ProjectedVector; | 
|---|
| [5309ba] | 194 | Boundaries *BoundaryPoints = new Boundaries[NDIM]; // first is alpha, second is (r, Nr) | 
|---|
| [ad37ab] | 195 | double angle = 0.; | 
|---|
| [042f82] | 196 |  | 
|---|
| [357fba] | 197 | // 3a. Go through every axis | 
|---|
|  | 198 | for (int axis = 0; axis < NDIM; axis++) { | 
|---|
|  | 199 | AxisVector.Zero(); | 
|---|
|  | 200 | AngleReferenceVector.Zero(); | 
|---|
|  | 201 | AngleReferenceNormalVector.Zero(); | 
|---|
| [0a4f7f] | 202 | AxisVector[axis] = 1.; | 
|---|
|  | 203 | AngleReferenceVector[(axis + 1) % NDIM] = 1.; | 
|---|
|  | 204 | AngleReferenceNormalVector[(axis + 2) % NDIM] = 1.; | 
|---|
| [042f82] | 205 |  | 
|---|
| [47d041] | 206 | LOG(1, "Axisvector is " << AxisVector << " and AngleReferenceVector is " << AngleReferenceVector << ", and AngleReferenceNormalVector is " << AngleReferenceNormalVector << "."); | 
|---|
| [042f82] | 207 |  | 
|---|
| [357fba] | 208 | // 3b. construct set of all points, transformed into cylindrical system and with left and right neighbours | 
|---|
| [59fff1] | 209 | // Boundaries stores non-const TesselPoint ref, hence we need iterator here | 
|---|
| [f01769] | 210 | for (molecule::const_iterator iter = mol->begin(); iter != mol->end(); ++iter) { | 
|---|
| [833b15] | 211 | ProjectedVector = (*iter)->getPosition() - (MolCenter); | 
|---|
| [273382] | 212 | ProjectedVector.ProjectOntoPlane(AxisVector); | 
|---|
| [042f82] | 213 |  | 
|---|
| [357fba] | 214 | // correct for negative side | 
|---|
| [776b64] | 215 | const double radius = ProjectedVector.NormSquared(); | 
|---|
| [357fba] | 216 | if (fabs(radius) > MYEPSILON) | 
|---|
| [273382] | 217 | angle = ProjectedVector.Angle(AngleReferenceVector); | 
|---|
| [357fba] | 218 | else | 
|---|
|  | 219 | angle = 0.; // otherwise it's a vector in Axis Direction and unimportant for boundary issues | 
|---|
| [042f82] | 220 |  | 
|---|
| [47d041] | 221 | //LOG(1, "Checking sign in quadrant : " << ProjectedVector.Projection(&AngleReferenceNormalVector) << "."); | 
|---|
| [273382] | 222 | if (ProjectedVector.ScalarProduct(AngleReferenceNormalVector) > 0) { | 
|---|
| [357fba] | 223 | angle = 2. * M_PI - angle; | 
|---|
|  | 224 | } | 
|---|
| [47d041] | 225 | LOG(1, "Inserting " << **iter << ": (r, alpha) = (" << radius << "," << angle << "): " << ProjectedVector); | 
|---|
| [f01769] | 226 | BoundaryTestPair = BoundaryPoints[axis].insert( | 
|---|
|  | 227 | BoundariesPair(angle, TesselPointDistancePair (radius, const_cast<atom *>(*iter)))); | 
|---|
| [357fba] | 228 | if (!BoundaryTestPair.second) { // same point exists, check first r, then distance of original vectors to center of gravity | 
|---|
| [47d041] | 229 | LOG(2, "Encountered two vectors whose projection onto axis " << axis << " is equal: "); | 
|---|
|  | 230 | LOG(2, "Present vector: " << *BoundaryTestPair.first->second.second); | 
|---|
|  | 231 | LOG(2, "New vector: " << **iter); | 
|---|
| [776b64] | 232 | const double ProjectedVectorNorm = ProjectedVector.NormSquared(); | 
|---|
|  | 233 | if ((ProjectedVectorNorm - BoundaryTestPair.first->second.first) > MYEPSILON) { | 
|---|
|  | 234 | BoundaryTestPair.first->second.first = ProjectedVectorNorm; | 
|---|
| [f01769] | 235 | BoundaryTestPair.first->second.second = const_cast<atom *>(*iter); | 
|---|
| [47d041] | 236 | LOG(2, "Keeping new vector due to larger projected distance " << ProjectedVectorNorm << "."); | 
|---|
| [776b64] | 237 | } else if (fabs(ProjectedVectorNorm - BoundaryTestPair.first->second.first) < MYEPSILON) { | 
|---|
| [833b15] | 238 | helper = (*iter)->getPosition() - (MolCenter); | 
|---|
| [776b64] | 239 | const double oldhelperNorm = helper.NormSquared(); | 
|---|
| [833b15] | 240 | helper = BoundaryTestPair.first->second.second->getPosition() - (MolCenter); | 
|---|
| [776b64] | 241 | if (helper.NormSquared() < oldhelperNorm) { | 
|---|
| [f01769] | 242 | BoundaryTestPair.first->second.second = const_cast<atom *>(*iter); | 
|---|
| [47d041] | 243 | LOG(2, "Keeping new vector due to larger distance to molecule center " << helper.NormSquared() << "."); | 
|---|
| [357fba] | 244 | } else { | 
|---|
| [47d041] | 245 | LOG(2, "Keeping present vector due to larger distance to molecule center " << oldhelperNorm << "."); | 
|---|
| [357fba] | 246 | } | 
|---|
|  | 247 | } else { | 
|---|
| [47d041] | 248 | LOG(2, "Keeping present vector due to larger projected distance " << ProjectedVectorNorm << "."); | 
|---|
| [357fba] | 249 | } | 
|---|
| [018741] | 250 | } | 
|---|
| [3d919e] | 251 | } | 
|---|
| [357fba] | 252 | // printing all inserted for debugging | 
|---|
|  | 253 | //    { | 
|---|
| [47d041] | 254 | //      std::stringstream output; | 
|---|
|  | 255 | //      output << "Printing list of candidates for axis " << axis << " which we have inserted so far: "; | 
|---|
| [357fba] | 256 | //      int i=0; | 
|---|
|  | 257 | //      for(Boundaries::iterator runner = BoundaryPoints[axis].begin(); runner != BoundaryPoints[axis].end(); runner++) { | 
|---|
|  | 258 | //        if (runner != BoundaryPoints[axis].begin()) | 
|---|
| [47d041] | 259 | //          output << ", " << i << ": " << *runner->second.second; | 
|---|
| [357fba] | 260 | //        else | 
|---|
| [47d041] | 261 | //          output << i << ": " << *runner->second.second; | 
|---|
| [357fba] | 262 | //        i++; | 
|---|
|  | 263 | //      } | 
|---|
| [47d041] | 264 | //      LOG(1, output.str()); | 
|---|
| [357fba] | 265 | //    } | 
|---|
|  | 266 | // 3c. throw out points whose distance is less than the mean of left and right neighbours | 
|---|
|  | 267 | bool flag = false; | 
|---|
| [47d041] | 268 | LOG(1, "Looking for candidates to kick out by convex condition ... "); | 
|---|
| [357fba] | 269 | do { // do as long as we still throw one out per round | 
|---|
|  | 270 | flag = false; | 
|---|
| [accebe] | 271 | Boundaries::iterator left = BoundaryPoints[axis].begin(); | 
|---|
|  | 272 | Boundaries::iterator right = BoundaryPoints[axis].begin(); | 
|---|
|  | 273 | Boundaries::iterator runner = BoundaryPoints[axis].begin(); | 
|---|
|  | 274 | bool LoopOnceDone = false; | 
|---|
|  | 275 | while (!LoopOnceDone) { | 
|---|
|  | 276 | runner = right; | 
|---|
|  | 277 | right++; | 
|---|
| [357fba] | 278 | // set neighbours correctly | 
|---|
|  | 279 | if (runner == BoundaryPoints[axis].begin()) { | 
|---|
|  | 280 | left = BoundaryPoints[axis].end(); | 
|---|
|  | 281 | } else { | 
|---|
|  | 282 | left = runner; | 
|---|
|  | 283 | } | 
|---|
|  | 284 | left--; | 
|---|
|  | 285 | if (right == BoundaryPoints[axis].end()) { | 
|---|
|  | 286 | right = BoundaryPoints[axis].begin(); | 
|---|
| [accebe] | 287 | LoopOnceDone = true; | 
|---|
| [357fba] | 288 | } | 
|---|
|  | 289 | // check distance | 
|---|
| [3d919e] | 290 |  | 
|---|
| [357fba] | 291 | // construct the vector of each side of the triangle on the projected plane (defined by normal vector AxisVector) | 
|---|
|  | 292 | { | 
|---|
|  | 293 | Vector SideA, SideB, SideC, SideH; | 
|---|
| [833b15] | 294 | SideA = left->second.second->getPosition() - (MolCenter); | 
|---|
| [273382] | 295 | SideA.ProjectOntoPlane(AxisVector); | 
|---|
| [47d041] | 296 | //          LOG(1, "SideA: " << SideA); | 
|---|
| [3d919e] | 297 |  | 
|---|
| [833b15] | 298 | SideB = right->second.second->getPosition() -(MolCenter); | 
|---|
| [273382] | 299 | SideB.ProjectOntoPlane(AxisVector); | 
|---|
| [47d041] | 300 | //          LOG(1, "SideB: " << SideB); | 
|---|
| [3d919e] | 301 |  | 
|---|
| [d74077] | 302 | SideC = left->second.second->getPosition() - right->second.second->getPosition(); | 
|---|
| [273382] | 303 | SideC.ProjectOntoPlane(AxisVector); | 
|---|
| [47d041] | 304 | //          LOG(1, "SideC: " << SideC); | 
|---|
| [3d919e] | 305 |  | 
|---|
| [833b15] | 306 | SideH = runner->second.second->getPosition() -(MolCenter); | 
|---|
| [273382] | 307 | SideH.ProjectOntoPlane(AxisVector); | 
|---|
| [47d041] | 308 | //          LOG(1, "SideH: " << SideH); | 
|---|
| [3d919e] | 309 |  | 
|---|
| [357fba] | 310 | // calculate each length | 
|---|
| [ad37ab] | 311 | const double a = SideA.Norm(); | 
|---|
|  | 312 | //const double b = SideB.Norm(); | 
|---|
|  | 313 | //const double c = SideC.Norm(); | 
|---|
|  | 314 | const double h = SideH.Norm(); | 
|---|
| [357fba] | 315 | // calculate the angles | 
|---|
| [273382] | 316 | const double alpha = SideA.Angle(SideH); | 
|---|
|  | 317 | const double beta = SideA.Angle(SideC); | 
|---|
|  | 318 | const double gamma = SideB.Angle(SideH); | 
|---|
|  | 319 | const double delta = SideC.Angle(SideH); | 
|---|
| [ad37ab] | 320 | const double MinDistance = a * sin(beta) / (sin(delta)) * (((alpha < M_PI / 2.) || (gamma < M_PI / 2.)) ? 1. : -1.); | 
|---|
| [47d041] | 321 | //LOG(1, " I calculated: a = " << a << ", h = " << h << ", beta(" << left->second.second->Name << "," << left->second.second->Name << "-" << right->second.second->Name << ") = " << beta << ", delta(" << left->second.second->Name << "," << runner->second.second->Name << ") = " << delta << ", Min = " << MinDistance << "."); | 
|---|
|  | 322 | LOG(1, "Checking CoG distance of runner " << *runner->second.second << " " << h << " against triangle's side length spanned by (" << *left->second.second << "," << *right->second.second << ") of " << MinDistance << "."); | 
|---|
| [357fba] | 323 | if ((fabs(h / fabs(h) - MinDistance / fabs(MinDistance)) < MYEPSILON) && ((h - MinDistance)) < -MYEPSILON) { | 
|---|
|  | 324 | // throw out point | 
|---|
| [47d041] | 325 | LOG(1, "Throwing out " << *runner->second.second << "."); | 
|---|
| [357fba] | 326 | BoundaryPoints[axis].erase(runner); | 
|---|
| [accebe] | 327 | runner = right; | 
|---|
| [357fba] | 328 | flag = true; | 
|---|
| [3d919e] | 329 | } | 
|---|
|  | 330 | } | 
|---|
|  | 331 | } | 
|---|
| [357fba] | 332 | } while (flag); | 
|---|
| [3d919e] | 333 | } | 
|---|
| [357fba] | 334 | return BoundaryPoints; | 
|---|
| [6ac7ee] | 335 | }; | 
|---|
|  | 336 |  | 
|---|
| [357fba] | 337 | /** Tesselates the convex boundary by finding all boundary points. | 
|---|
|  | 338 | * \param *out output stream for debugging | 
|---|
| [776b64] | 339 | * \param *mol molecule structure with Atom's and Bond's. | 
|---|
| [bdc91e] | 340 | * \param *BoundaryPts set of boundary points to use or NULL | 
|---|
| [357fba] | 341 | * \param *TesselStruct Tesselation filled with points, lines and triangles on boundary on return | 
|---|
| [6bd7e0] | 342 | * \param *LCList atoms in LinkedCell_deprecated list | 
|---|
| [357fba] | 343 | * \param *filename filename prefix for output of vertex data | 
|---|
|  | 344 | * \return *TesselStruct is filled with convex boundary and tesselation is stored under \a *filename. | 
|---|
| [6ac7ee] | 345 | */ | 
|---|
| [6bd7e0] | 346 | void FindConvexBorder(const molecule* mol, Boundaries *BoundaryPts, Tesselation *&TesselStruct, const LinkedCell_deprecated *LCList, const char *filename) | 
|---|
| [6ac7ee] | 347 | { | 
|---|
| [ce7bfd] | 348 | //Info FunctionInfo(__func__); | 
|---|
| [357fba] | 349 | bool BoundaryFreeFlag = false; | 
|---|
|  | 350 | Boundaries *BoundaryPoints = NULL; | 
|---|
| [3d919e] | 351 |  | 
|---|
| [776b64] | 352 | if (TesselStruct != NULL) // free if allocated | 
|---|
|  | 353 | delete(TesselStruct); | 
|---|
|  | 354 | TesselStruct = new class Tesselation; | 
|---|
| [3d919e] | 355 |  | 
|---|
| [357fba] | 356 | // 1. Find all points on the boundary | 
|---|
| [bdc91e] | 357 | if (BoundaryPts == NULL) { | 
|---|
|  | 358 | BoundaryFreeFlag = true; | 
|---|
|  | 359 | BoundaryPoints = GetBoundaryPoints(mol, TesselStruct); | 
|---|
| [357fba] | 360 | } else { | 
|---|
| [bdc91e] | 361 | BoundaryPoints = BoundaryPts; | 
|---|
| [47d041] | 362 | LOG(0, "Using given boundary points set."); | 
|---|
| [3d919e] | 363 | } | 
|---|
|  | 364 |  | 
|---|
| [357fba] | 365 | // printing all inserted for debugging | 
|---|
| [47d041] | 366 | if (DoLog(1)) { | 
|---|
|  | 367 | for (int axis=0; axis < NDIM; axis++) { | 
|---|
|  | 368 | std::stringstream output; | 
|---|
|  | 369 | output << "Printing list of candidates for axis " << axis << " which we have inserted so far: "; | 
|---|
|  | 370 | int i=0; | 
|---|
|  | 371 | for(Boundaries::iterator runner = BoundaryPoints[axis].begin(); runner != BoundaryPoints[axis].end(); runner++) { | 
|---|
|  | 372 | if (runner != BoundaryPoints[axis].begin()) | 
|---|
|  | 373 | output << ", " << i << ": " << *runner->second.second; | 
|---|
|  | 374 | else | 
|---|
|  | 375 | output << i << ": " << *runner->second.second; | 
|---|
|  | 376 | i++; | 
|---|
|  | 377 | } | 
|---|
|  | 378 | LOG(1, output.str()); | 
|---|
| [a37350] | 379 | } | 
|---|
| [bdc91e] | 380 | } | 
|---|
| [3d919e] | 381 |  | 
|---|
| [357fba] | 382 | // 2. fill the boundary point list | 
|---|
|  | 383 | for (int axis = 0; axis < NDIM; axis++) | 
|---|
|  | 384 | for (Boundaries::iterator runner = BoundaryPoints[axis].begin(); runner != BoundaryPoints[axis].end(); runner++) | 
|---|
| [776b64] | 385 | if (!TesselStruct->AddBoundaryPoint(runner->second.second, 0)) | 
|---|
| [47d041] | 386 | LOG(2, "Point " << *(runner->second.second) << " is already present."); | 
|---|
| [e4ea46] | 387 |  | 
|---|
| [47d041] | 388 | LOG(0, "I found " << TesselStruct->PointsOnBoundaryCount << " points on the convex boundary."); | 
|---|
| [357fba] | 389 | // now we have the whole set of edge points in the BoundaryList | 
|---|
| [018741] | 390 |  | 
|---|
| [357fba] | 391 | // listing for debugging | 
|---|
| [47d041] | 392 | //if (DoLog(1)) { | 
|---|
|  | 393 | //  std::stringstream output; | 
|---|
|  | 394 | //  output << "Listing PointsOnBoundary:"; | 
|---|
| [357fba] | 395 | //  for(PointMap::iterator runner = PointsOnBoundary.begin(); runner != PointsOnBoundary.end(); runner++) { | 
|---|
| [47d041] | 396 | //    output << " " << *runner->second; | 
|---|
| [357fba] | 397 | //  } | 
|---|
| [47d041] | 398 | //  LOG(1, output.str()); | 
|---|
|  | 399 | //} | 
|---|
| [018741] | 400 |  | 
|---|
| [357fba] | 401 | // 3a. guess starting triangle | 
|---|
| [e138de] | 402 | TesselStruct->GuessStartingTriangle(); | 
|---|
| [018741] | 403 |  | 
|---|
| [357fba] | 404 | // 3b. go through all lines, that are not yet part of two triangles (only of one so far) | 
|---|
| [caa06ef] | 405 | PointCloudAdaptor< molecule > cloud(const_cast<molecule *>(mol), mol->name); | 
|---|
| [34c43a] | 406 | TesselStruct->TesselateOnBoundary(cloud); | 
|---|
| [3d919e] | 407 |  | 
|---|
| [357fba] | 408 | // 3c. check whether all atoms lay inside the boundary, if not, add to boundary points, segment triangle into three with the new point | 
|---|
| [34c43a] | 409 | if (!TesselStruct->InsertStraddlingPoints(cloud, LCList)) | 
|---|
| [47d041] | 410 | ELOG(1, "Insertion of straddling points failed!"); | 
|---|
| [3d919e] | 411 |  | 
|---|
| [47d041] | 412 | LOG(0, "I created " << TesselStruct->TrianglesOnBoundary.size() << " intermediate triangles with " << TesselStruct->LinesOnBoundary.size() << " lines and " << TesselStruct->PointsOnBoundary.size() << " points."); | 
|---|
| [ef0e6d] | 413 |  | 
|---|
|  | 414 | // 4. Store triangles in tecplot file | 
|---|
| [d51975] | 415 | StoreTrianglesinFile(mol, TesselStruct, filename, "_intermed"); | 
|---|
| [ef0e6d] | 416 |  | 
|---|
| [357fba] | 417 | // 3d. check all baselines whether the peaks of the two adjacent triangles with respect to center of baseline are convex, if not, make the baseline between the two peaks and baseline endpoints become the new peaks | 
|---|
| [ad37ab] | 418 | bool AllConvex = true; | 
|---|
| [093645] | 419 | class BoundaryLineSet *line = NULL; | 
|---|
|  | 420 | do { | 
|---|
|  | 421 | AllConvex = true; | 
|---|
| [776b64] | 422 | for (LineMap::iterator LineRunner = TesselStruct->LinesOnBoundary.begin(); LineRunner != TesselStruct->LinesOnBoundary.end(); LineRunner++) { | 
|---|
| [093645] | 423 | line = LineRunner->second; | 
|---|
| [47d041] | 424 | LOG(1, "INFO: Current line is " << *line << "."); | 
|---|
| [e138de] | 425 | if (!line->CheckConvexityCriterion()) { | 
|---|
| [47d041] | 426 | LOG(1, "... line " << *line << " is concave, flipping it."); | 
|---|
| [093645] | 427 |  | 
|---|
|  | 428 | // flip the line | 
|---|
| [e138de] | 429 | if (TesselStruct->PickFarthestofTwoBaselines(line) == 0.) | 
|---|
| [47d041] | 430 | ELOG(1, "Correction of concave baselines failed!"); | 
|---|
| [57066a] | 431 | else { | 
|---|
| [e138de] | 432 | TesselStruct->FlipBaseline(line); | 
|---|
| [47d041] | 433 | LOG(1, "INFO: Correction of concave baselines worked."); | 
|---|
| [accebe] | 434 | LineRunner = TesselStruct->LinesOnBoundary.begin(); // LineRunner may have been erase if line was deleted from LinesOnBoundary | 
|---|
| [57066a] | 435 | } | 
|---|
| [093645] | 436 | } | 
|---|
|  | 437 | } | 
|---|
|  | 438 | } while (!AllConvex); | 
|---|
| [3d919e] | 439 |  | 
|---|
| [ef0e6d] | 440 | // 3e. we need another correction here, for TesselPoints that are below the surface (i.e. have an odd number of concave triangles surrounding it) | 
|---|
| [776b64] | 441 | //  if (!TesselStruct->CorrectConcaveTesselPoints(out)) | 
|---|
| [47d041] | 442 | //    ELOG(1, "Correction of concave tesselpoints failed!"); | 
|---|
| [ef0e6d] | 443 |  | 
|---|
| [47d041] | 444 | LOG(0, "I created " << TesselStruct->TrianglesOnBoundary.size() << " triangles with " << TesselStruct->LinesOnBoundary.size() << " lines and " << TesselStruct->PointsOnBoundary.size() << " points."); | 
|---|
| [3d919e] | 445 |  | 
|---|
| [357fba] | 446 | // 4. Store triangles in tecplot file | 
|---|
| [d51975] | 447 | StoreTrianglesinFile(mol, TesselStruct, filename, ""); | 
|---|
| [ef0e6d] | 448 |  | 
|---|
| [357fba] | 449 | // free reference lists | 
|---|
|  | 450 | if (BoundaryFreeFlag) | 
|---|
|  | 451 | delete[] (BoundaryPoints); | 
|---|
| [3d919e] | 452 | }; | 
|---|
| [6ac7ee] | 453 |  | 
|---|
| [d4fa23] | 454 | /** For testing removes one boundary point after another to check for leaks. | 
|---|
|  | 455 | * \param *out output stream for debugging | 
|---|
|  | 456 | * \param *TesselStruct Tesselation containing envelope with boundary points | 
|---|
|  | 457 | * \param *mol molecule | 
|---|
|  | 458 | * \param *filename name of file | 
|---|
|  | 459 | * \return true - all removed, false - something went wrong | 
|---|
|  | 460 | */ | 
|---|
| [e138de] | 461 | bool RemoveAllBoundaryPoints(class Tesselation *&TesselStruct, const molecule * const mol, const char * const filename) | 
|---|
| [d4fa23] | 462 | { | 
|---|
| [ce7bfd] | 463 | //Info FunctionInfo(__func__); | 
|---|
| [d4fa23] | 464 | int i=0; | 
|---|
|  | 465 | char number[MAXSTRINGSIZE]; | 
|---|
|  | 466 |  | 
|---|
|  | 467 | if ((TesselStruct == NULL) || (TesselStruct->PointsOnBoundary.empty())) { | 
|---|
| [47d041] | 468 | ELOG(1, "TesselStruct is empty."); | 
|---|
| [d4fa23] | 469 | return false; | 
|---|
|  | 470 | } | 
|---|
|  | 471 |  | 
|---|
|  | 472 | PointMap::iterator PointRunner; | 
|---|
|  | 473 | while (!TesselStruct->PointsOnBoundary.empty()) { | 
|---|
| [47d041] | 474 | if (DoLog(1)) { | 
|---|
|  | 475 | std::stringstream output; | 
|---|
|  | 476 | output << "Remaining points are: "; | 
|---|
|  | 477 | for (PointMap::iterator PointSprinter = TesselStruct->PointsOnBoundary.begin(); PointSprinter != TesselStruct->PointsOnBoundary.end(); PointSprinter++) | 
|---|
|  | 478 | output << *(PointSprinter->second) << "\t"; | 
|---|
|  | 479 | LOG(1, output.str()); | 
|---|
|  | 480 | } | 
|---|
| [d4fa23] | 481 |  | 
|---|
|  | 482 | PointRunner = TesselStruct->PointsOnBoundary.begin(); | 
|---|
|  | 483 | // remove point | 
|---|
| [e138de] | 484 | TesselStruct->RemovePointFromTesselatedSurface(PointRunner->second); | 
|---|
| [d4fa23] | 485 |  | 
|---|
|  | 486 | // store envelope | 
|---|
|  | 487 | sprintf(number, "-%04d", i++); | 
|---|
| [e138de] | 488 | StoreTrianglesinFile(mol, (const Tesselation *&)TesselStruct, filename, number); | 
|---|
| [d4fa23] | 489 | } | 
|---|
|  | 490 |  | 
|---|
|  | 491 | return true; | 
|---|
|  | 492 | }; | 
|---|
|  | 493 |  | 
|---|
| [08ef35] | 494 | /** Creates a convex envelope from a given non-convex one. | 
|---|
| [093645] | 495 | * -# First step, remove concave spots, i.e. singular "dents" | 
|---|
|  | 496 | *   -# We go through all PointsOnBoundary. | 
|---|
|  | 497 | *   -# We CheckConvexityCriterion() for all its lines. | 
|---|
|  | 498 | *   -# If all its lines are concave, it cannot be on the convex envelope. | 
|---|
|  | 499 | *   -# Hence, we remove it and re-create all its triangles from its getCircleOfConnectedPoints() | 
|---|
|  | 500 | *   -# We calculate the additional volume. | 
|---|
|  | 501 | *   -# We go over all lines until none yields a concavity anymore. | 
|---|
|  | 502 | * -# Second step, remove concave lines, i.e. line-shape "dents" | 
|---|
|  | 503 | *   -# We go through all LinesOnBoundary | 
|---|
|  | 504 | *   -# We CheckConvexityCriterion() | 
|---|
|  | 505 | *   -# If it returns concave, we flip the line in this quadrupel of points (abusing the degeneracy of the tesselation) | 
|---|
|  | 506 | *   -# We CheckConvexityCriterion(), | 
|---|
|  | 507 | *   -# if it's concave, we continue | 
|---|
|  | 508 | *   -# if not, we mark an error and stop | 
|---|
| [08ef35] | 509 | * Note: This routine - for free - calculates the difference in volume between convex and | 
|---|
| [ee0032] | 510 | * non-convex envelope, as the former is easy to calculate - Tesselation::getVolumeOfConvexEnvelope() - it | 
|---|
| [08ef35] | 511 | * can be used to compute volumes of arbitrary shapes. | 
|---|
|  | 512 | * \param *out output stream for debugging | 
|---|
|  | 513 | * \param *TesselStruct non-convex envelope, is changed in return! | 
|---|
| [093645] | 514 | * \param *mol molecule | 
|---|
|  | 515 | * \param *filename name of file | 
|---|
| [08ef35] | 516 | * \return volume difference between the non- and the created convex envelope | 
|---|
|  | 517 | */ | 
|---|
| [5f7b95] | 518 | double ConvexizeNonconvexEnvelope( | 
|---|
|  | 519 | Tesselation *&TesselStruct, | 
|---|
|  | 520 | const molecule * const mol, | 
|---|
|  | 521 | const char * const filename, | 
|---|
|  | 522 | bool DebugOutputEveryStep) | 
|---|
| [08ef35] | 523 | { | 
|---|
| [ce7bfd] | 524 | //Info FunctionInfo(__func__); | 
|---|
| [08ef35] | 525 | double volume = 0; | 
|---|
|  | 526 | class BoundaryPointSet *point = NULL; | 
|---|
|  | 527 | class BoundaryLineSet *line = NULL; | 
|---|
| [ad37ab] | 528 | bool Concavity = false; | 
|---|
| [57066a] | 529 | char dummy[MAXSTRINGSIZE]; | 
|---|
| [ad37ab] | 530 | PointMap::iterator PointRunner; | 
|---|
|  | 531 | PointMap::iterator PointAdvance; | 
|---|
|  | 532 | LineMap::iterator LineRunner; | 
|---|
|  | 533 | LineMap::iterator LineAdvance; | 
|---|
|  | 534 | TriangleMap::iterator TriangleRunner; | 
|---|
|  | 535 | TriangleMap::iterator TriangleAdvance; | 
|---|
|  | 536 | int run = 0; | 
|---|
| [093645] | 537 |  | 
|---|
|  | 538 | // check whether there is something to work on | 
|---|
| [08ef35] | 539 | if (TesselStruct == NULL) { | 
|---|
| [47d041] | 540 | ELOG(1, "TesselStruct is empty!"); | 
|---|
| [08ef35] | 541 | return volume; | 
|---|
|  | 542 | } | 
|---|
|  | 543 |  | 
|---|
| [b8d215] | 544 | LOG(1, "INFO: Making tesselated surface with " << TesselStruct->TrianglesOnBoundaryCount | 
|---|
|  | 545 | << " convex ..."); | 
|---|
|  | 546 |  | 
|---|
| [2ef2cc] | 547 | // first purge all degenerate triangles | 
|---|
|  | 548 | TesselStruct->RemoveDegeneratedTriangles(); | 
|---|
|  | 549 |  | 
|---|
| [1d9b7aa] | 550 | do { | 
|---|
|  | 551 | Concavity = false; | 
|---|
| [d289c6] | 552 |  | 
|---|
| [5f7b95] | 553 | if (DebugOutputEveryStep) { | 
|---|
|  | 554 | sprintf(dummy, "-%d", run++); | 
|---|
|  | 555 | //CalculateConcavityPerBoundaryPoint(TesselStruct); | 
|---|
|  | 556 | LOG(1, "INFO: Writing " << run << "th tesselation file."); | 
|---|
|  | 557 | StoreTrianglesinFile(mol, (const Tesselation *&)TesselStruct, filename, dummy); | 
|---|
|  | 558 | } | 
|---|
| [57066a] | 559 |  | 
|---|
| [d289c6] | 560 | // first step: remove all full-concave point | 
|---|
| [1d9b7aa] | 561 | PointRunner = TesselStruct->PointsOnBoundary.begin(); | 
|---|
|  | 562 | PointAdvance = PointRunner; // we need an advanced point, as the PointRunner might get removed | 
|---|
|  | 563 | while (PointRunner != TesselStruct->PointsOnBoundary.end()) { | 
|---|
|  | 564 | PointAdvance++; | 
|---|
|  | 565 | point = PointRunner->second; | 
|---|
| [b8d215] | 566 | LOG(2, "INFO: Current point is " << *point << "."); | 
|---|
| [aac19f] | 567 | // check that at least a single line is concave | 
|---|
|  | 568 | LineMap::iterator LineRunner = point->lines.begin(); | 
|---|
|  | 569 | for (; LineRunner != point->lines.end(); LineRunner++) { | 
|---|
|  | 570 | const BoundaryLineSet * line = LineRunner->second; | 
|---|
| [b8d215] | 571 | LOG(3, "INFO: Current line of point " << *point << " is " << *line << "."); | 
|---|
| [aac19f] | 572 | if (!line->CheckConvexityCriterion()) | 
|---|
|  | 573 | break; | 
|---|
| [d289c6] | 574 | } | 
|---|
| [aac19f] | 575 | // remove the point if needed | 
|---|
|  | 576 | if (LineRunner != point->lines.end()) { | 
|---|
|  | 577 | const double tmp = TesselStruct->RemoveFullConcavePointFromTesselatedSurface(point); | 
|---|
|  | 578 | if (tmp > 0.) { | 
|---|
|  | 579 | volume += tmp; | 
|---|
|  | 580 | Concavity = true; | 
|---|
|  | 581 | } | 
|---|
| [1d9b7aa] | 582 | } | 
|---|
|  | 583 | PointRunner = PointAdvance; | 
|---|
| [093645] | 584 | } | 
|---|
|  | 585 |  | 
|---|
| [5f7b95] | 586 | if (DebugOutputEveryStep) { | 
|---|
|  | 587 | sprintf(dummy, "-%d", run++); | 
|---|
|  | 588 | //CalculateConcavityPerBoundaryPoint(TesselStruct); | 
|---|
|  | 589 | LOG(1, "INFO: Writing " << run << "th tesselation file."); | 
|---|
|  | 590 | StoreTrianglesinFile(mol, (const Tesselation *&)TesselStruct, filename, dummy); | 
|---|
|  | 591 | } | 
|---|
| [093645] | 592 |  | 
|---|
| [d289c6] | 593 | // second step: flip baselines, i.e. add general tetraeder at concave lines | 
|---|
|  | 594 | // when the tetraeder does not intersect with other already present triangles | 
|---|
| [1d9b7aa] | 595 | LineRunner = TesselStruct->LinesOnBoundary.begin(); | 
|---|
|  | 596 | LineAdvance = LineRunner;  // we need an advanced line, as the LineRunner might get removed | 
|---|
| [81ffd8] | 597 | std::map<double, std::pair<BoundaryLineSet *, double> > GainMap; | 
|---|
| [1d9b7aa] | 598 | while (LineRunner != TesselStruct->LinesOnBoundary.end()) { | 
|---|
|  | 599 | LineAdvance++; | 
|---|
|  | 600 | line = LineRunner->second; | 
|---|
| [d289c6] | 601 | if (!line->CheckConvexityCriterion()) { | 
|---|
|  | 602 | LOG(2, "INFO: concave line is " << *line << "."); | 
|---|
|  | 603 | // gather the other points | 
|---|
|  | 604 | BoundaryPointSet *BPS[4]; | 
|---|
|  | 605 | int m = 0; | 
|---|
|  | 606 | { | 
|---|
|  | 607 | for (TriangleMap::iterator runner = line->triangles.begin(); runner != line->triangles.end(); runner++) | 
|---|
|  | 608 | for (int j = 0; j < 3; j++) // all of their endpoints and baselines | 
|---|
|  | 609 | if (!line->ContainsBoundaryPoint(runner->second->endpoints[j])) // and neither of its endpoints | 
|---|
|  | 610 | BPS[m++] = runner->second->endpoints[j]; | 
|---|
|  | 611 | } | 
|---|
|  | 612 | BPS[2] = line->endpoints[0]; | 
|---|
|  | 613 | BPS[3] = line->endpoints[1]; | 
|---|
|  | 614 | LOG(3, "DEBUG: other line would consist of " << *BPS[0] << " and " | 
|---|
|  | 615 | << *BPS[1] << "."); | 
|---|
|  | 616 |  | 
|---|
|  | 617 | // check for already present (third) side of the tetraeder as we then | 
|---|
|  | 618 | // would create a degenerate triangle | 
|---|
|  | 619 | bool TetraederSidePresent = false; | 
|---|
|  | 620 | { | 
|---|
|  | 621 | class TesselPoint *TriangleCandidates[3]; | 
|---|
|  | 622 | TriangleCandidates[0] = BPS[0]->node; | 
|---|
|  | 623 | TriangleCandidates[1] = BPS[1]->node; | 
|---|
|  | 624 | TriangleCandidates[2] = BPS[2]->node; | 
|---|
|  | 625 | if ((TesselStruct->GetPresentTriangle(TriangleCandidates) != NULL)) { | 
|---|
|  | 626 | LOG(2, "REJECT: Triangle side " << *TriangleCandidates[0] << "," | 
|---|
|  | 627 | << *TriangleCandidates[1] << "," << *TriangleCandidates[2] << " present."); | 
|---|
|  | 628 | TetraederSidePresent = true; | 
|---|
|  | 629 | } | 
|---|
|  | 630 | TriangleCandidates[2] = BPS[3]->node; | 
|---|
|  | 631 | if ((TesselStruct->GetPresentTriangle(TriangleCandidates) != NULL)) { | 
|---|
|  | 632 | LOG(2, "REJECT: Triangle side " << *TriangleCandidates[0] << "," | 
|---|
|  | 633 | << *TriangleCandidates[1] << "," << *TriangleCandidates[2] << " present."); | 
|---|
|  | 634 | TetraederSidePresent = true; | 
|---|
|  | 635 | } | 
|---|
|  | 636 | } | 
|---|
|  | 637 |  | 
|---|
|  | 638 | if ((BPS[0] != BPS[1]) && (m == 2) && (!TetraederSidePresent)) { | 
|---|
|  | 639 | // check whether all adjacent triangles do not intersect with new line | 
|---|
|  | 640 | bool no_line_intersects = true; | 
|---|
|  | 641 | Vector Intersection; | 
|---|
|  | 642 | TriangleSet triangles; | 
|---|
|  | 643 | TriangleSet *firsttriangles = TesselStruct->GetAllTriangles(line->endpoints[0]); | 
|---|
|  | 644 | TriangleSet *secondtriangles = TesselStruct->GetAllTriangles(line->endpoints[1]); | 
|---|
|  | 645 | triangles.insert( firsttriangles->begin(), firsttriangles->end() ); | 
|---|
|  | 646 | triangles.insert( secondtriangles->begin(), secondtriangles->end() ); | 
|---|
|  | 647 | delete firsttriangles; | 
|---|
|  | 648 | delete secondtriangles; | 
|---|
|  | 649 | for (TriangleSet::const_iterator triangleiter = triangles.begin(); | 
|---|
|  | 650 | triangleiter != triangles.end(); ++triangleiter) { | 
|---|
|  | 651 | const BoundaryTriangleSet * triangle = *triangleiter; | 
|---|
|  | 652 | bool line_intersects = triangle->GetIntersectionInsideTriangle( | 
|---|
|  | 653 | BPS[0]->node->getPosition(), | 
|---|
|  | 654 | BPS[1]->node->getPosition(), | 
|---|
|  | 655 | Intersection); | 
|---|
|  | 656 | // switch result when coinciding with endpoint | 
|---|
|  | 657 | bool concave_adjacent_line = false; | 
|---|
|  | 658 | bool intersection_is_endnode = false; | 
|---|
|  | 659 | for (int j=0;j<2;++j) { | 
|---|
|  | 660 | if (Intersection.DistanceSquared(BPS[j]->node->getPosition()) < MYEPSILON) { | 
|---|
|  | 661 | intersection_is_endnode = true; | 
|---|
|  | 662 | // check whether its an adjacent triangle and if it's concavely connected | 
|---|
|  | 663 | // only then are we in danger of cutting through it and need to check | 
|---|
|  | 664 | // sign of normal vector and intersecting line | 
|---|
|  | 665 | for (int i=0;i<2;++i) | 
|---|
|  | 666 | for (int lineindex=0;lineindex < 3;++lineindex) | 
|---|
|  | 667 | if ((triangle->lines[lineindex]->ContainsBoundaryPoint(line->endpoints[i])) | 
|---|
|  | 668 | && (triangle->lines[lineindex]->ContainsBoundaryPoint(BPS[j]))) { | 
|---|
|  | 669 | concave_adjacent_line = !triangle->lines[lineindex]->CheckConvexityCriterion(); | 
|---|
|  | 670 | } | 
|---|
|  | 671 | if (concave_adjacent_line) { | 
|---|
|  | 672 | const Vector intersector = | 
|---|
|  | 673 | BPS[(j+1)%2]->node->getPosition() - Intersection; | 
|---|
|  | 674 | if (triangle->NormalVector.ScalarProduct(intersector) >= -MYEPSILON) { | 
|---|
|  | 675 | LOG(4, "ACCEPT: Intersection coincides with first endpoint " | 
|---|
|  | 676 | << *BPS[j] << "."); | 
|---|
|  | 677 | line_intersects = false; | 
|---|
|  | 678 | } else { | 
|---|
|  | 679 | LOG(4, "REJECT: Intersection ends on wrong side of triangle."); | 
|---|
|  | 680 | } | 
|---|
|  | 681 | } else { | 
|---|
|  | 682 | LOG(4, "ACCEPT: Intersection coincides with first endpoint " | 
|---|
|  | 683 | << *BPS[j] << "."); | 
|---|
|  | 684 | line_intersects = false; | 
|---|
|  | 685 | } | 
|---|
|  | 686 | } | 
|---|
|  | 687 | } | 
|---|
|  | 688 | // if we have an intersection, check that it is within either | 
|---|
|  | 689 | // endpoint, i.e. check that scalar product between vectors going | 
|---|
|  | 690 | // from intersction to either endpoint has negative sign (both | 
|---|
|  | 691 | // vectors point in opposite directions) | 
|---|
|  | 692 | if (!intersection_is_endnode && line_intersects) { | 
|---|
|  | 693 | const Vector firstvector = BPS[0]->node->getPosition() - Intersection; | 
|---|
|  | 694 | const Vector secondvector = BPS[1]->node->getPosition() - Intersection; | 
|---|
|  | 695 | if (firstvector.ScalarProduct(secondvector) >= 0) | 
|---|
|  | 696 | line_intersects = false; | 
|---|
|  | 697 | } | 
|---|
|  | 698 | no_line_intersects &= !line_intersects; | 
|---|
|  | 699 | } | 
|---|
|  | 700 |  | 
|---|
|  | 701 | if (no_line_intersects) { | 
|---|
|  | 702 | // calculate the volume | 
|---|
| [81ffd8] | 703 | const double tmp = line->CalculateConvexity(); | 
|---|
|  | 704 | const double gain = | 
|---|
|  | 705 | CalculateVolumeofGeneralTetraeder( | 
|---|
|  | 706 | BPS[0]->node->getPosition(), | 
|---|
|  | 707 | BPS[1]->node->getPosition(), | 
|---|
|  | 708 | BPS[2]->node->getPosition(), | 
|---|
|  | 709 | BPS[3]->node->getPosition()); | 
|---|
|  | 710 |  | 
|---|
|  | 711 | GainMap.insert(std::make_pair(tmp, std::make_pair(line,gain) )); | 
|---|
| [d289c6] | 712 | LOG(2, "DEBUG: Adding concave line " << *line << " with gain of " | 
|---|
| [81ffd8] | 713 | << gain << "."); | 
|---|
| [d289c6] | 714 | } else { | 
|---|
|  | 715 | // if 2 or 3 don't | 
|---|
|  | 716 | LOG(2, "DEBUG: We don't added concave line " << *line | 
|---|
|  | 717 | << " as other line intersects with adjacent triangles."); | 
|---|
|  | 718 | } | 
|---|
| [57066a] | 719 | } | 
|---|
| [1d9b7aa] | 720 | } | 
|---|
|  | 721 | LineRunner = LineAdvance; | 
|---|
|  | 722 | } | 
|---|
| [d289c6] | 723 | // flip line with most gain | 
|---|
|  | 724 | if (!GainMap.empty()) { | 
|---|
| [81ffd8] | 725 | line = GainMap.begin()->second.first; | 
|---|
|  | 726 | const double tmp = GainMap.begin()->second.second; | 
|---|
| [d289c6] | 727 | volume += tmp; | 
|---|
| [81ffd8] | 728 |  | 
|---|
| [d289c6] | 729 | //      GainMap.clear(); | 
|---|
|  | 730 |  | 
|---|
|  | 731 | // and flip the line | 
|---|
| [81ffd8] | 732 | LOG(1, "INFO: Flipping current most concave line " << *line << " with gain of " | 
|---|
| [d289c6] | 733 | << tmp << "."); | 
|---|
|  | 734 | TesselStruct->FlipBaseline(line); | 
|---|
|  | 735 | Concavity = true; | 
|---|
|  | 736 | } | 
|---|
|  | 737 | } while ((Concavity)); // && (run < 100) | 
|---|
| [093645] | 738 |  | 
|---|
| [e138de] | 739 | CalculateConcavityPerBoundaryPoint(TesselStruct); | 
|---|
|  | 740 | StoreTrianglesinFile(mol, (const Tesselation *&)TesselStruct, filename, ""); | 
|---|
| [0077b5] | 741 |  | 
|---|
|  | 742 | // end | 
|---|
| [d289c6] | 743 | LOG(0, "RESULT: Added volume in convexization is " << volume << "."); | 
|---|
| [0077b5] | 744 | return volume; | 
|---|
|  | 745 | }; | 
|---|
|  | 746 |  | 
|---|
| [6ac7ee] | 747 |  | 
|---|
| [7dea7c] | 748 | /** Stores triangles to file. | 
|---|
|  | 749 | * \param *out output stream for debugging | 
|---|
|  | 750 | * \param *mol molecule with atoms and bonds | 
|---|
| [71b20e] | 751 | * \param *TesselStruct Tesselation with boundary triangles | 
|---|
| [7dea7c] | 752 | * \param *filename prefix of filename | 
|---|
|  | 753 | * \param *extraSuffix intermediate suffix | 
|---|
|  | 754 | */ | 
|---|
| [71b20e] | 755 | void StoreTrianglesinFile(const molecule * const mol, const Tesselation * const TesselStruct, const char *filename, const char *extraSuffix) | 
|---|
| [7dea7c] | 756 | { | 
|---|
| [ce7bfd] | 757 | //Info FunctionInfo(__func__); | 
|---|
| [caa06ef] | 758 | PointCloudAdaptor< molecule > cloud(const_cast<molecule *>(mol), mol->name); | 
|---|
| [7dea7c] | 759 | // 4. Store triangles in tecplot file | 
|---|
|  | 760 | if (filename != NULL) { | 
|---|
|  | 761 | if (DoTecplotOutput) { | 
|---|
|  | 762 | string OutputName(filename); | 
|---|
|  | 763 | OutputName.append(extraSuffix); | 
|---|
|  | 764 | OutputName.append(TecplotSuffix); | 
|---|
|  | 765 | ofstream *tecplot = new ofstream(OutputName.c_str()); | 
|---|
| [34c43a] | 766 | WriteTecplotFile(tecplot, TesselStruct, cloud, -1); | 
|---|
| [7dea7c] | 767 | tecplot->close(); | 
|---|
|  | 768 | delete(tecplot); | 
|---|
|  | 769 | } | 
|---|
|  | 770 | if (DoRaster3DOutput) { | 
|---|
|  | 771 | string OutputName(filename); | 
|---|
|  | 772 | OutputName.append(extraSuffix); | 
|---|
|  | 773 | OutputName.append(Raster3DSuffix); | 
|---|
|  | 774 | ofstream *rasterplot = new ofstream(OutputName.c_str()); | 
|---|
| [34c43a] | 775 | WriteRaster3dFile(rasterplot, TesselStruct, cloud); | 
|---|
| [7dea7c] | 776 | rasterplot->close(); | 
|---|
|  | 777 | delete(rasterplot); | 
|---|
|  | 778 | } | 
|---|
|  | 779 | } | 
|---|
|  | 780 | }; | 
|---|
| [03648b] | 781 |  | 
|---|
| [6ac7ee] | 782 | /** Tesselates the non convex boundary by rolling a virtual sphere along the surface of the molecule. | 
|---|
|  | 783 | * \param *out output stream for debugging | 
|---|
|  | 784 | * \param *mol molecule structure with Atom's and Bond's | 
|---|
| [776b64] | 785 | * \param *&TesselStruct Tesselation filled with points, lines and triangles on boundary on return | 
|---|
| [6bd7e0] | 786 | * \param *&LCList atoms in LinkedCell_deprecated list | 
|---|
| [57066a] | 787 | * \param RADIUS radius of the virtual sphere | 
|---|
| [6ac7ee] | 788 | * \param *filename filename prefix for output of vertex data | 
|---|
| [4fc93f] | 789 | * \return true - tesselation successful, false - tesselation failed | 
|---|
| [6ac7ee] | 790 | */ | 
|---|
| [6bd7e0] | 791 | bool FindNonConvexBorder(molecule* const mol, Tesselation *&TesselStruct, const LinkedCell_deprecated *&LCList, const double RADIUS, const char *filename = NULL) | 
|---|
| [03648b] | 792 | { | 
|---|
| [ce7bfd] | 793 | //Info FunctionInfo(__func__); | 
|---|
| [3d919e] | 794 | bool freeLC = false; | 
|---|
| [4fc93f] | 795 | bool status = false; | 
|---|
| [6613ec] | 796 | CandidateForTesselation *baseline = NULL; | 
|---|
| [1e168b] | 797 | bool OneLoopWithoutSuccessFlag = true;  // marks whether we went once through all baselines without finding any without two triangles | 
|---|
| [a2a2f7] | 798 | //  bool TesselationFailFlag = false; | 
|---|
| [357fba] | 799 |  | 
|---|
| [a7b761b] | 800 | mol->getAtomCount(); | 
|---|
| [357fba] | 801 |  | 
|---|
| [776b64] | 802 | if (TesselStruct == NULL) { | 
|---|
| [47d041] | 803 | LOG(1, "Allocating Tesselation struct ..."); | 
|---|
| [776b64] | 804 | TesselStruct= new Tesselation; | 
|---|
| [ef0e6d] | 805 | } else { | 
|---|
| [776b64] | 806 | delete(TesselStruct); | 
|---|
| [47d041] | 807 | LOG(1, "Re-Allocating Tesselation struct ..."); | 
|---|
| [776b64] | 808 | TesselStruct = new Tesselation; | 
|---|
| [3d919e] | 809 | } | 
|---|
| [ad37ab] | 810 |  | 
|---|
| [57066a] | 811 | // initialise Linked Cell | 
|---|
| [caa06ef] | 812 | PointCloudAdaptor< molecule > cloud(mol, mol->name); | 
|---|
| [3d919e] | 813 | if (LCList == NULL) { | 
|---|
| [6bd7e0] | 814 | LCList = new LinkedCell_deprecated(cloud, 2.*RADIUS); | 
|---|
| [3d919e] | 815 | freeLC = true; | 
|---|
|  | 816 | } | 
|---|
|  | 817 |  | 
|---|
| [57066a] | 818 | // 1. get starting triangle | 
|---|
| [ce70970] | 819 | if (!TesselStruct->FindStartingTriangle(RADIUS, LCList)) { | 
|---|
| [47d041] | 820 | ELOG(0, "No valid starting triangle found."); | 
|---|
| [b5c2d7] | 821 | //performCriticalExit(); | 
|---|
| [ce70970] | 822 | } | 
|---|
| [f8bd7d] | 823 | if (filename != NULL) { | 
|---|
|  | 824 | if ((DoSingleStepOutput && ((TesselStruct->TrianglesOnBoundary.size() % SingleStepWidth == 0)))) { // if we have a new triangle and want to output each new triangle configuration | 
|---|
| [34c43a] | 825 | TesselStruct->Output(filename, cloud); | 
|---|
| [f8bd7d] | 826 | } | 
|---|
|  | 827 | } | 
|---|
| [3d919e] | 828 |  | 
|---|
| [57066a] | 829 | // 2. expand from there | 
|---|
| [1e168b] | 830 | while ((!TesselStruct->OpenLines.empty()) && (OneLoopWithoutSuccessFlag)) { | 
|---|
| [b40a93] | 831 | (cerr << "There are " <<  TesselStruct->TrianglesOnBoundary.size() << " triangles and " << TesselStruct->OpenLines.size() << " open lines to scan for candidates." << endl); | 
|---|
|  | 832 | // 2a. print OpenLines without candidates | 
|---|
| [47d041] | 833 | LOG(1, "There are the following open lines to scan for a candidates:"); | 
|---|
| [1e168b] | 834 | for (CandidateMap::iterator Runner = TesselStruct->OpenLines.begin(); Runner != TesselStruct->OpenLines.end(); Runner++) | 
|---|
| [b40a93] | 835 | if (Runner->second->pointlist.empty()) | 
|---|
| [47d041] | 836 | LOG(1, " " << *(Runner->second)); | 
|---|
| [1e168b] | 837 |  | 
|---|
| [734816] | 838 | // 2b. find best candidate for each OpenLine | 
|---|
| [a2a2f7] | 839 | const bool TesselationFailFlag = TesselStruct->FindCandidatesforOpenLines(RADIUS, LCList); | 
|---|
|  | 840 | ASSERT( TesselationFailFlag, | 
|---|
|  | 841 | "FindNonConvexBorder() - at least one open line without candidate exists."); | 
|---|
| [3d919e] | 842 |  | 
|---|
| [734816] | 843 | // 2c. print OpenLines with candidates again | 
|---|
| [47d041] | 844 | LOG(1, "There are " << TesselStruct->OpenLines.size() << " open lines to scan for the best candidates:"); | 
|---|
| [1e168b] | 845 | for (CandidateMap::iterator Runner = TesselStruct->OpenLines.begin(); Runner != TesselStruct->OpenLines.end(); Runner++) | 
|---|
| [47d041] | 846 | LOG(1, " " << *(Runner->second)); | 
|---|
| [1e168b] | 847 |  | 
|---|
| [734816] | 848 | // 2d. search for smallest ShortestAngle among all candidates | 
|---|
|  | 849 | double ShortestAngle = 4.*M_PI; | 
|---|
| [1e168b] | 850 | for (CandidateMap::iterator Runner = TesselStruct->OpenLines.begin(); Runner != TesselStruct->OpenLines.end(); Runner++) { | 
|---|
|  | 851 | if (Runner->second->ShortestAngle < ShortestAngle) { | 
|---|
|  | 852 | baseline = Runner->second; | 
|---|
|  | 853 | ShortestAngle = baseline->ShortestAngle; | 
|---|
| [47d041] | 854 | LOG(1, "New best candidate is " << *baseline->BaseLine << " with point " << *(*baseline->pointlist.begin()) << " and angle " << baseline->ShortestAngle); | 
|---|
| [1e168b] | 855 | } | 
|---|
|  | 856 | } | 
|---|
| [734816] | 857 | // 2e. if we found one, add candidate | 
|---|
| [f67b6e] | 858 | if ((ShortestAngle == 4.*M_PI) || (baseline->pointlist.empty())) | 
|---|
| [7dea7c] | 859 | OneLoopWithoutSuccessFlag = false; | 
|---|
| [1e168b] | 860 | else { | 
|---|
| [474961] | 861 | TesselStruct->AddCandidatePolygon(*baseline, RADIUS, LCList); | 
|---|
| [1e168b] | 862 | } | 
|---|
|  | 863 |  | 
|---|
| [734816] | 864 | // 2f. write temporary envelope | 
|---|
| [1e168b] | 865 | if (filename != NULL) { | 
|---|
|  | 866 | if ((DoSingleStepOutput && ((TesselStruct->TrianglesOnBoundary.size() % SingleStepWidth == 0)))) { // if we have a new triangle and want to output each new triangle configuration | 
|---|
| [34c43a] | 867 | TesselStruct->Output(filename, cloud); | 
|---|
| [1e168b] | 868 | } | 
|---|
| [3d919e] | 869 | } | 
|---|
|  | 870 | } | 
|---|
| [4fc93f] | 871 | //  // check envelope for consistency | 
|---|
|  | 872 | //  status = CheckListOfBaselines(TesselStruct); | 
|---|
|  | 873 | // | 
|---|
|  | 874 | //  // look whether all points are inside of the convex envelope, otherwise add them via degenerated triangles | 
|---|
|  | 875 | //  //->InsertStraddlingPoints(mol, LCList); | 
|---|
| [9879f6] | 876 | //  for (molecule::const_iterator iter = mol->begin(); iter != mol->end(); ++iter) { | 
|---|
| [57066a] | 877 | //  class TesselPoint *Runner = NULL; | 
|---|
| [9879f6] | 878 | //    Runner = *iter; | 
|---|
| [47d041] | 879 | //    LOG(1, "Checking on " << Runner->Name << " ... "); | 
|---|
| [e138de] | 880 | //    if (!->IsInnerPoint(Runner, LCList)) { | 
|---|
| [47d041] | 881 | //      LOG(2, Runner->Name << " is outside of envelope, adding via degenerated triangles."); | 
|---|
| [e138de] | 882 | //      ->AddBoundaryPointByDegeneratedTriangle(Runner, LCList); | 
|---|
| [57066a] | 883 | //    } else { | 
|---|
| [47d041] | 884 | //      LOG(2, Runner->Name << " is inside of or on envelope."); | 
|---|
| [57066a] | 885 | //    } | 
|---|
|  | 886 | //  } | 
|---|
| [357fba] | 887 |  | 
|---|
| [f67b6e] | 888 | //  // Purges surplus triangles. | 
|---|
|  | 889 | //  TesselStruct->RemoveDegeneratedTriangles(); | 
|---|
| [b32dbb] | 890 | // | 
|---|
|  | 891 | //  // check envelope for consistency | 
|---|
|  | 892 | //  status = CheckListOfBaselines(TesselStruct); | 
|---|
| [b998c3] | 893 |  | 
|---|
| [b8d215] | 894 | //  cout << "before correction" << endl; | 
|---|
| [c72112] | 895 |  | 
|---|
| [b998c3] | 896 | // store before correction | 
|---|
| [c72112] | 897 | StoreTrianglesinFile(mol, TesselStruct, filename, ""); | 
|---|
| [b998c3] | 898 |  | 
|---|
| [6613ec] | 899 | //  // correct degenerated polygons | 
|---|
|  | 900 | //  TesselStruct->CorrectAllDegeneratedPolygons(); | 
|---|
|  | 901 | // | 
|---|
| [e69c87] | 902 | // check envelope for consistency | 
|---|
|  | 903 | status = CheckListOfBaselines(TesselStruct); | 
|---|
| [ef0e6d] | 904 |  | 
|---|
| [57066a] | 905 | // write final envelope | 
|---|
| [e138de] | 906 | CalculateConcavityPerBoundaryPoint(TesselStruct); | 
|---|
| [b8d215] | 907 | //  cout << "after correction" << endl; | 
|---|
| [c72112] | 908 | StoreTrianglesinFile(mol, TesselStruct, filename, ""); | 
|---|
| [8c54a3] | 909 |  | 
|---|
| [3d919e] | 910 | if (freeLC) | 
|---|
|  | 911 | delete(LCList); | 
|---|
| [4fc93f] | 912 |  | 
|---|
|  | 913 | return status; | 
|---|
| [6ac7ee] | 914 | }; | 
|---|