source: src/boundary.cpp@ 753f02

Action_Thermostats Add_AtomRandomPerturbation Add_FitFragmentPartialChargesAction Add_RotateAroundBondAction Add_SelectAtomByNameAction Added_ParseSaveFragmentResults AddingActions_SaveParseParticleParameters Adding_Graph_to_ChangeBondActions Adding_MD_integration_tests Adding_ParticleName_to_Atom Adding_StructOpt_integration_tests AtomFragments Automaking_mpqc_open AutomationFragmentation_failures Candidate_v1.5.4 Candidate_v1.6.0 Candidate_v1.6.1 Candidate_v1.7.0 ChangeBugEmailaddress ChangingTestPorts ChemicalSpaceEvaluator CombiningParticlePotentialParsing Combining_Subpackages Debian_Package_split Debian_package_split_molecuildergui_only Disabling_MemDebug Docu_Python_wait EmpiricalPotential_contain_HomologyGraph EmpiricalPotential_contain_HomologyGraph_documentation Enable_parallel_make_install Enhance_userguide Enhanced_StructuralOptimization Enhanced_StructuralOptimization_continued Example_ManyWaysToTranslateAtom Exclude_Hydrogens_annealWithBondGraph FitPartialCharges_GlobalError Fix_BoundInBox_CenterInBox_MoleculeActions Fix_ChargeSampling_PBC Fix_ChronosMutex Fix_FitPartialCharges Fix_FitPotential_needs_atomicnumbers Fix_ForceAnnealing Fix_IndependentFragmentGrids Fix_ParseParticles Fix_ParseParticles_split_forward_backward_Actions Fix_PopActions Fix_QtFragmentList_sorted_selection Fix_Restrictedkeyset_FragmentMolecule Fix_StatusMsg Fix_StepWorldTime_single_argument Fix_Verbose_Codepatterns Fix_fitting_potentials Fixes ForceAnnealing_goodresults ForceAnnealing_oldresults ForceAnnealing_tocheck ForceAnnealing_with_BondGraph ForceAnnealing_with_BondGraph_continued ForceAnnealing_with_BondGraph_continued_betteresults ForceAnnealing_with_BondGraph_contraction-expansion FragmentAction_writes_AtomFragments FragmentMolecule_checks_bonddegrees GeometryObjects Gui_Fixes Gui_displays_atomic_force_velocity ImplicitCharges IndependentFragmentGrids IndependentFragmentGrids_IndividualZeroInstances IndependentFragmentGrids_IntegrationTest IndependentFragmentGrids_Sole_NN_Calculation JobMarket_RobustOnKillsSegFaults JobMarket_StableWorkerPool JobMarket_unresolvable_hostname_fix MoreRobust_FragmentAutomation ODR_violation_mpqc_open PartialCharges_OrthogonalSummation PdbParser_setsAtomName PythonUI_with_named_parameters QtGui_reactivate_TimeChanged_changes Recreated_GuiChecks Rewrite_FitPartialCharges RotateToPrincipalAxisSystem_UndoRedo SaturateAtoms_findBestMatching SaturateAtoms_singleDegree StoppableMakroAction Subpackage_CodePatterns Subpackage_JobMarket Subpackage_LinearAlgebra Subpackage_levmar Subpackage_mpqc_open Subpackage_vmg Switchable_LogView ThirdParty_MPQC_rebuilt_buildsystem TrajectoryDependenant_MaxOrder TremoloParser_IncreasedPrecision TremoloParser_MultipleTimesteps TremoloParser_setsAtomName Ubuntu_1604_changes stable
Last change on this file since 753f02 was 1bd79e, checked in by Tillmann Crueger <crueger@…>, 16 years ago

Changed implementation of Vector to forward operations to contained objects

  • Property mode set to 100755
File size: 49.6 KB
RevLine 
[f66195]1/** \file boundary.cpp
[edb93c]2 *
3 * Implementations and super-function for envelopes
[2319ed]4 */
5
[cbc5fb]6#include "World.hpp"
[f66195]7#include "atom.hpp"
8#include "bond.hpp"
[8eb17a]9#include "boundary.hpp"
[f66195]10#include "config.hpp"
11#include "element.hpp"
12#include "helpers.hpp"
[f67b6e]13#include "info.hpp"
[f66195]14#include "linkedcell.hpp"
[e138de]15#include "log.hpp"
[29812d]16#include "memoryallocator.hpp"
[f66195]17#include "molecule.hpp"
18#include "tesselation.hpp"
19#include "tesselationhelpers.hpp"
[0a4f7f]20#include "Plane.hpp"
[2319ed]21
[357fba]22#include<gsl/gsl_poly.h>
[d6eb80]23#include<time.h>
[2319ed]24
[357fba]25// ========================================== F U N C T I O N S =================================
[2319ed]26
27
[357fba]28/** Determines greatest diameters of a cluster defined by its convex envelope.
29 * Looks at lines parallel to one axis and where they intersect on the projected planes
[2319ed]30 * \param *out output stream for debugging
[357fba]31 * \param *BoundaryPoints NDIM set of boundary points defining the convex envelope on each projected plane
32 * \param *mol molecule structure representing the cluster
[776b64]33 * \param *&TesselStruct Tesselation structure with triangles
[357fba]34 * \param IsAngstroem whether we have angstroem or atomic units
35 * \return NDIM array of the diameters
[e4ea46]36 */
[e138de]37double *GetDiametersOfCluster(const Boundaries *BoundaryPtr, const molecule *mol, Tesselation *&TesselStruct, const bool IsAngstroem)
[caf5d6]38{
[f67b6e]39 Info FunctionInfo(__func__);
[357fba]40 // get points on boundary of NULL was given as parameter
41 bool BoundaryFreeFlag = false;
[ad37ab]42 double OldComponent = 0.;
43 double tmp = 0.;
44 double w1 = 0.;
45 double w2 = 0.;
46 Vector DistanceVector;
47 Vector OtherVector;
48 int component = 0;
49 int Othercomponent = 0;
[776b64]50 Boundaries::const_iterator Neighbour;
51 Boundaries::const_iterator OtherNeighbour;
[ad37ab]52 double *GreatestDiameter = new double[NDIM];
53
[776b64]54 const Boundaries *BoundaryPoints;
55 if (BoundaryPtr == NULL) {
[357fba]56 BoundaryFreeFlag = true;
[e138de]57 BoundaryPoints = GetBoundaryPoints(mol, TesselStruct);
[86234b]58 } else {
[776b64]59 BoundaryPoints = BoundaryPtr;
[f67b6e]60 Log() << Verbose(0) << "Using given boundary points set." << endl;
[86234b]61 }
[357fba]62 // determine biggest "diameter" of cluster for each axis
63 for (int i = 0; i < NDIM; i++)
64 GreatestDiameter[i] = 0.;
65 for (int axis = 0; axis < NDIM; axis++)
66 { // regard each projected plane
[e138de]67 //Log() << Verbose(1) << "Current axis is " << axis << "." << endl;
[357fba]68 for (int j = 0; j < 2; j++)
69 { // and for both axis on the current plane
70 component = (axis + j + 1) % NDIM;
71 Othercomponent = (axis + 1 + ((j + 1) & 1)) % NDIM;
[e138de]72 //Log() << Verbose(1) << "Current component is " << component << ", Othercomponent is " << Othercomponent << "." << endl;
[776b64]73 for (Boundaries::const_iterator runner = BoundaryPoints[axis].begin(); runner != BoundaryPoints[axis].end(); runner++) {
[f67b6e]74 //Log() << Verbose(1) << "Current runner is " << *(runner->second.second) << "." << endl;
[357fba]75 // seek for the neighbours pair where the Othercomponent sign flips
76 Neighbour = runner;
77 Neighbour++;
78 if (Neighbour == BoundaryPoints[axis].end()) // make it wrap around
79 Neighbour = BoundaryPoints[axis].begin();
[273382]80 DistanceVector = runner->second.second->x - Neighbour->second.second->x;
[776b64]81 do { // seek for neighbour pair where it flips
[0a4f7f]82 OldComponent = DistanceVector[Othercomponent];
[357fba]83 Neighbour++;
84 if (Neighbour == BoundaryPoints[axis].end()) // make it wrap around
85 Neighbour = BoundaryPoints[axis].begin();
[273382]86 DistanceVector = runner->second.second->x - Neighbour->second.second->x;
[f67b6e]87 //Log() << Verbose(2) << "OldComponent is " << OldComponent << ", new one is " << DistanceVector.x[Othercomponent] << "." << endl;
[776b64]88 } while ((runner != Neighbour) && (fabs(OldComponent / fabs(
[0a4f7f]89 OldComponent) - DistanceVector[Othercomponent] / fabs(
90 DistanceVector[Othercomponent])) < MYEPSILON)); // as long as sign does not flip
[776b64]91 if (runner != Neighbour) {
[357fba]92 OtherNeighbour = Neighbour;
93 if (OtherNeighbour == BoundaryPoints[axis].begin()) // make it wrap around
94 OtherNeighbour = BoundaryPoints[axis].end();
95 OtherNeighbour--;
[f67b6e]96 //Log() << Verbose(1) << "The pair, where the sign of OtherComponent flips, is: " << *(Neighbour->second.second) << " and " << *(OtherNeighbour->second.second) << "." << endl;
[357fba]97 // now we have found the pair: Neighbour and OtherNeighbour
[273382]98 OtherVector = runner->second.second->x - OtherNeighbour->second.second->x;
[f67b6e]99 //Log() << Verbose(1) << "Distances to Neighbour and OtherNeighbour are " << DistanceVector.x[component] << " and " << OtherVector.x[component] << "." << endl;
100 //Log() << Verbose(1) << "OtherComponents to Neighbour and OtherNeighbour are " << DistanceVector.x[Othercomponent] << " and " << OtherVector.x[Othercomponent] << "." << endl;
[357fba]101 // do linear interpolation between points (is exact) to extract exact intersection between Neighbour and OtherNeighbour
[0a4f7f]102 w1 = fabs(OtherVector[Othercomponent]);
103 w2 = fabs(DistanceVector[Othercomponent]);
104 tmp = fabs((w1 * DistanceVector[component] + w2
105 * OtherVector[component]) / (w1 + w2));
[357fba]106 // mark if it has greater diameter
[f67b6e]107 //Log() << Verbose(1) << "Comparing current greatest " << GreatestDiameter[component] << " to new " << tmp << "." << endl;
[357fba]108 GreatestDiameter[component] = (GreatestDiameter[component]
109 > tmp) ? GreatestDiameter[component] : tmp;
110 } //else
[f67b6e]111 //Log() << Verbose(1) << "Saw no sign flip, probably top or bottom node." << endl;
[3d919e]112 }
113 }
114 }
[e138de]115 Log() << Verbose(0) << "RESULT: The biggest diameters are "
[357fba]116 << GreatestDiameter[0] << " and " << GreatestDiameter[1] << " and "
117 << GreatestDiameter[2] << " " << (IsAngstroem ? "angstrom"
118 : "atomiclength") << "." << endl;
[03648b]119
[357fba]120 // free reference lists
121 if (BoundaryFreeFlag)
122 delete[] (BoundaryPoints);
[e4ea46]123
[357fba]124 return GreatestDiameter;
[e4ea46]125}
126;
[03648b]127
[042f82]128
[357fba]129/** Determines the boundary points of a cluster.
130 * Does a projection per axis onto the orthogonal plane, transforms into spherical coordinates, sorts them by the angle
131 * and looks at triples: if the middle has less a distance than the allowed maximum height of the triangle formed by the plane's
132 * center and first and last point in the triple, it is thrown out.
133 * \param *out output stream for debugging
134 * \param *mol molecule structure representing the cluster
[776b64]135 * \param *&TesselStruct pointer to Tesselation structure
[e4ea46]136 */
[e138de]137Boundaries *GetBoundaryPoints(const molecule *mol, Tesselation *&TesselStruct)
[caf5d6]138{
[f67b6e]139 Info FunctionInfo(__func__);
[357fba]140 atom *Walker = NULL;
141 PointMap PointsOnBoundary;
142 LineMap LinesOnBoundary;
143 TriangleMap TrianglesOnBoundary;
[e138de]144 Vector *MolCenter = mol->DetermineCenterOfAll();
[357fba]145 Vector helper;
[ad37ab]146 BoundariesTestPair BoundaryTestPair;
147 Vector AxisVector;
148 Vector AngleReferenceVector;
149 Vector AngleReferenceNormalVector;
150 Vector ProjectedVector;
151 Boundaries *BoundaryPoints = new Boundaries[NDIM]; // first is alpha, second is (r, nr)
152 double angle = 0.;
[042f82]153
[357fba]154 // 3a. Go through every axis
155 for (int axis = 0; axis < NDIM; axis++) {
156 AxisVector.Zero();
157 AngleReferenceVector.Zero();
158 AngleReferenceNormalVector.Zero();
[0a4f7f]159 AxisVector[axis] = 1.;
160 AngleReferenceVector[(axis + 1) % NDIM] = 1.;
161 AngleReferenceNormalVector[(axis + 2) % NDIM] = 1.;
[042f82]162
[e138de]163 Log() << Verbose(1) << "Axisvector is " << AxisVector << " and AngleReferenceVector is " << AngleReferenceVector << ", and AngleReferenceNormalVector is " << AngleReferenceNormalVector << "." << endl;
[042f82]164
[357fba]165 // 3b. construct set of all points, transformed into cylindrical system and with left and right neighbours
166 Walker = mol->start;
167 while (Walker->next != mol->end) {
168 Walker = Walker->next;
[273382]169 ProjectedVector = Walker->x - (*MolCenter);
170 ProjectedVector.ProjectOntoPlane(AxisVector);
[042f82]171
[357fba]172 // correct for negative side
[776b64]173 const double radius = ProjectedVector.NormSquared();
[357fba]174 if (fabs(radius) > MYEPSILON)
[273382]175 angle = ProjectedVector.Angle(AngleReferenceVector);
[357fba]176 else
177 angle = 0.; // otherwise it's a vector in Axis Direction and unimportant for boundary issues
[042f82]178
[f67b6e]179 //Log() << Verbose(1) << "Checking sign in quadrant : " << ProjectedVector.Projection(&AngleReferenceNormalVector) << "." << endl;
[273382]180 if (ProjectedVector.ScalarProduct(AngleReferenceNormalVector) > 0) {
[357fba]181 angle = 2. * M_PI - angle;
182 }
[f67b6e]183 Log() << Verbose(1) << "Inserting " << *Walker << ": (r, alpha) = (" << radius << "," << angle << "): " << ProjectedVector << endl;
[357fba]184 BoundaryTestPair = BoundaryPoints[axis].insert(BoundariesPair(angle, DistancePair (radius, Walker)));
185 if (!BoundaryTestPair.second) { // same point exists, check first r, then distance of original vectors to center of gravity
[e138de]186 Log() << Verbose(2) << "Encountered two vectors whose projection onto axis " << axis << " is equal: " << endl;
187 Log() << Verbose(2) << "Present vector: " << *BoundaryTestPair.first->second.second << endl;
188 Log() << Verbose(2) << "New vector: " << *Walker << endl;
[776b64]189 const double ProjectedVectorNorm = ProjectedVector.NormSquared();
190 if ((ProjectedVectorNorm - BoundaryTestPair.first->second.first) > MYEPSILON) {
191 BoundaryTestPair.first->second.first = ProjectedVectorNorm;
[357fba]192 BoundaryTestPair.first->second.second = Walker;
[e138de]193 Log() << Verbose(2) << "Keeping new vector due to larger projected distance " << ProjectedVectorNorm << "." << endl;
[776b64]194 } else if (fabs(ProjectedVectorNorm - BoundaryTestPair.first->second.first) < MYEPSILON) {
[273382]195 helper = Walker->x - (*MolCenter);
[776b64]196 const double oldhelperNorm = helper.NormSquared();
[273382]197 helper = BoundaryTestPair.first->second.second->x - (*MolCenter);
[776b64]198 if (helper.NormSquared() < oldhelperNorm) {
[357fba]199 BoundaryTestPair.first->second.second = Walker;
[e138de]200 Log() << Verbose(2) << "Keeping new vector due to larger distance to molecule center " << helper.NormSquared() << "." << endl;
[357fba]201 } else {
[e138de]202 Log() << Verbose(2) << "Keeping present vector due to larger distance to molecule center " << oldhelperNorm << "." << endl;
[357fba]203 }
204 } else {
[e138de]205 Log() << Verbose(2) << "Keeping present vector due to larger projected distance " << ProjectedVectorNorm << "." << endl;
[357fba]206 }
[018741]207 }
[3d919e]208 }
[357fba]209 // printing all inserted for debugging
210 // {
[f67b6e]211 // Log() << Verbose(1) << "Printing list of candidates for axis " << axis << " which we have inserted so far." << endl;
[357fba]212 // int i=0;
213 // for(Boundaries::iterator runner = BoundaryPoints[axis].begin(); runner != BoundaryPoints[axis].end(); runner++) {
214 // if (runner != BoundaryPoints[axis].begin())
[f67b6e]215 // Log() << Verbose(0) << ", " << i << ": " << *runner->second.second;
[357fba]216 // else
[f67b6e]217 // Log() << Verbose(0) << i << ": " << *runner->second.second;
[357fba]218 // i++;
219 // }
[f67b6e]220 // Log() << Verbose(0) << endl;
[357fba]221 // }
222 // 3c. throw out points whose distance is less than the mean of left and right neighbours
223 bool flag = false;
[e138de]224 Log() << Verbose(1) << "Looking for candidates to kick out by convex condition ... " << endl;
[357fba]225 do { // do as long as we still throw one out per round
226 flag = false;
227 Boundaries::iterator left = BoundaryPoints[axis].end();
228 Boundaries::iterator right = BoundaryPoints[axis].end();
229 for (Boundaries::iterator runner = BoundaryPoints[axis].begin(); runner != BoundaryPoints[axis].end(); runner++) {
230 // set neighbours correctly
231 if (runner == BoundaryPoints[axis].begin()) {
232 left = BoundaryPoints[axis].end();
233 } else {
234 left = runner;
235 }
236 left--;
237 right = runner;
238 right++;
239 if (right == BoundaryPoints[axis].end()) {
240 right = BoundaryPoints[axis].begin();
241 }
242 // check distance
[3d919e]243
[357fba]244 // construct the vector of each side of the triangle on the projected plane (defined by normal vector AxisVector)
245 {
246 Vector SideA, SideB, SideC, SideH;
[273382]247 SideA = left->second.second->x - (*MolCenter);
248 SideA.ProjectOntoPlane(AxisVector);
[f67b6e]249 // Log() << Verbose(1) << "SideA: " << SideA << endl;
[3d919e]250
[273382]251 SideB = right->second.second->x -(*MolCenter);
252 SideB.ProjectOntoPlane(AxisVector);
[f67b6e]253 // Log() << Verbose(1) << "SideB: " << SideB << endl;
[3d919e]254
[273382]255 SideC = left->second.second->x - right->second.second->x;
256 SideC.ProjectOntoPlane(AxisVector);
[f67b6e]257 // Log() << Verbose(1) << "SideC: " << SideC << endl;
[3d919e]258
[273382]259 SideH = runner->second.second->x -(*MolCenter);
260 SideH.ProjectOntoPlane(AxisVector);
[f67b6e]261 // Log() << Verbose(1) << "SideH: " << SideH << endl;
[3d919e]262
[357fba]263 // calculate each length
[ad37ab]264 const double a = SideA.Norm();
265 //const double b = SideB.Norm();
266 //const double c = SideC.Norm();
267 const double h = SideH.Norm();
[357fba]268 // calculate the angles
[273382]269 const double alpha = SideA.Angle(SideH);
270 const double beta = SideA.Angle(SideC);
271 const double gamma = SideB.Angle(SideH);
272 const double delta = SideC.Angle(SideH);
[ad37ab]273 const double MinDistance = a * sin(beta) / (sin(delta)) * (((alpha < M_PI / 2.) || (gamma < M_PI / 2.)) ? 1. : -1.);
[f67b6e]274 //Log() << Verbose(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 << "." << endl;
[e138de]275 Log() << Verbose(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 << "." << endl;
[357fba]276 if ((fabs(h / fabs(h) - MinDistance / fabs(MinDistance)) < MYEPSILON) && ((h - MinDistance)) < -MYEPSILON) {
277 // throw out point
[e138de]278 Log() << Verbose(1) << "Throwing out " << *runner->second.second << "." << endl;
[357fba]279 BoundaryPoints[axis].erase(runner);
280 flag = true;
[3d919e]281 }
282 }
283 }
[357fba]284 } while (flag);
[3d919e]285 }
[357fba]286 delete(MolCenter);
287 return BoundaryPoints;
[6ac7ee]288};
289
[357fba]290/** Tesselates the convex boundary by finding all boundary points.
291 * \param *out output stream for debugging
[776b64]292 * \param *mol molecule structure with Atom's and Bond's.
[357fba]293 * \param *TesselStruct Tesselation filled with points, lines and triangles on boundary on return
294 * \param *LCList atoms in LinkedCell list
295 * \param *filename filename prefix for output of vertex data
296 * \return *TesselStruct is filled with convex boundary and tesselation is stored under \a *filename.
[6ac7ee]297 */
[e138de]298void FindConvexBorder(const molecule* mol, Tesselation *&TesselStruct, const LinkedCell *LCList, const char *filename)
[6ac7ee]299{
[f67b6e]300 Info FunctionInfo(__func__);
[357fba]301 bool BoundaryFreeFlag = false;
302 Boundaries *BoundaryPoints = NULL;
[3d919e]303
[776b64]304 if (TesselStruct != NULL) // free if allocated
305 delete(TesselStruct);
306 TesselStruct = new class Tesselation;
[3d919e]307
[357fba]308 // 1. Find all points on the boundary
309 if (BoundaryPoints == NULL) {
310 BoundaryFreeFlag = true;
[e138de]311 BoundaryPoints = GetBoundaryPoints(mol, TesselStruct);
[357fba]312 } else {
[f67b6e]313 Log() << Verbose(0) << "Using given boundary points set." << endl;
[3d919e]314 }
315
[357fba]316// printing all inserted for debugging
317 for (int axis=0; axis < NDIM; axis++)
318 {
[f67b6e]319 Log() << Verbose(1) << "Printing list of candidates for axis " << axis << " which we have inserted so far." << endl;
[357fba]320 int i=0;
321 for(Boundaries::iterator runner = BoundaryPoints[axis].begin(); runner != BoundaryPoints[axis].end(); runner++) {
322 if (runner != BoundaryPoints[axis].begin())
[f67b6e]323 Log() << Verbose(0) << ", " << i << ": " << *runner->second.second;
[357fba]324 else
[f67b6e]325 Log() << Verbose(0) << i << ": " << *runner->second.second;
[357fba]326 i++;
[a37350]327 }
[f67b6e]328 Log() << Verbose(0) << endl;
[a37350]329 }
[3d919e]330
[357fba]331 // 2. fill the boundary point list
332 for (int axis = 0; axis < NDIM; axis++)
333 for (Boundaries::iterator runner = BoundaryPoints[axis].begin(); runner != BoundaryPoints[axis].end(); runner++)
[776b64]334 if (!TesselStruct->AddBoundaryPoint(runner->second.second, 0))
[717e0c]335 eLog() << Verbose(2) << "Point " << *(runner->second.second) << " is already present!" << endl;
[e4ea46]336
[f67b6e]337 Log() << Verbose(0) << "I found " << TesselStruct->PointsOnBoundaryCount << " points on the convex boundary." << endl;
[357fba]338 // now we have the whole set of edge points in the BoundaryList
[018741]339
[357fba]340 // listing for debugging
[e138de]341 // Log() << Verbose(1) << "Listing PointsOnBoundary:";
[357fba]342 // for(PointMap::iterator runner = PointsOnBoundary.begin(); runner != PointsOnBoundary.end(); runner++) {
[f67b6e]343 // Log() << Verbose(0) << " " << *runner->second;
[357fba]344 // }
[f67b6e]345 // Log() << Verbose(0) << endl;
[018741]346
[357fba]347 // 3a. guess starting triangle
[e138de]348 TesselStruct->GuessStartingTriangle();
[018741]349
[357fba]350 // 3b. go through all lines, that are not yet part of two triangles (only of one so far)
[e138de]351 TesselStruct->TesselateOnBoundary(mol);
[3d919e]352
[357fba]353 // 3c. check whether all atoms lay inside the boundary, if not, add to boundary points, segment triangle into three with the new point
[e138de]354 if (!TesselStruct->InsertStraddlingPoints(mol, LCList))
[f67b6e]355 eLog() << Verbose(1) << "Insertion of straddling points failed!" << endl;
[3d919e]356
[f67b6e]357 Log() << Verbose(0) << "I created " << TesselStruct->TrianglesOnBoundary.size() << " intermediate triangles with " << TesselStruct->LinesOnBoundary.size() << " lines and " << TesselStruct->PointsOnBoundary.size() << " points." << endl;
[ef0e6d]358
359 // 4. Store triangles in tecplot file
360 if (filename != NULL) {
361 if (DoTecplotOutput) {
362 string OutputName(filename);
363 OutputName.append("_intermed");
364 OutputName.append(TecplotSuffix);
365 ofstream *tecplot = new ofstream(OutputName.c_str());
[e138de]366 WriteTecplotFile(tecplot, TesselStruct, mol, 0);
[ef0e6d]367 tecplot->close();
368 delete(tecplot);
369 }
370 if (DoRaster3DOutput) {
371 string OutputName(filename);
372 OutputName.append("_intermed");
373 OutputName.append(Raster3DSuffix);
374 ofstream *rasterplot = new ofstream(OutputName.c_str());
[e138de]375 WriteRaster3dFile(rasterplot, TesselStruct, mol);
[ef0e6d]376 rasterplot->close();
377 delete(rasterplot);
378 }
379 }
380
[357fba]381 // 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]382 bool AllConvex = true;
[093645]383 class BoundaryLineSet *line = NULL;
384 do {
385 AllConvex = true;
[776b64]386 for (LineMap::iterator LineRunner = TesselStruct->LinesOnBoundary.begin(); LineRunner != TesselStruct->LinesOnBoundary.end(); LineRunner++) {
[093645]387 line = LineRunner->second;
[e138de]388 Log() << Verbose(1) << "INFO: Current line is " << *line << "." << endl;
389 if (!line->CheckConvexityCriterion()) {
390 Log() << Verbose(1) << "... line " << *line << " is concave, flipping it." << endl;
[093645]391
392 // flip the line
[e138de]393 if (TesselStruct->PickFarthestofTwoBaselines(line) == 0.)
[717e0c]394 eLog() << Verbose(1) << "Correction of concave baselines failed!" << endl;
[57066a]395 else {
[e138de]396 TesselStruct->FlipBaseline(line);
397 Log() << Verbose(1) << "INFO: Correction of concave baselines worked." << endl;
[57066a]398 }
[093645]399 }
400 }
401 } while (!AllConvex);
[3d919e]402
[ef0e6d]403 // 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]404// if (!TesselStruct->CorrectConcaveTesselPoints(out))
[e138de]405// Log() << Verbose(1) << "Correction of concave tesselpoints failed!" << endl;
[ef0e6d]406
[f67b6e]407 Log() << Verbose(0) << "I created " << TesselStruct->TrianglesOnBoundary.size() << " triangles with " << TesselStruct->LinesOnBoundary.size() << " lines and " << TesselStruct->PointsOnBoundary.size() << " points." << endl;
[3d919e]408
[357fba]409 // 4. Store triangles in tecplot file
410 if (filename != NULL) {
411 if (DoTecplotOutput) {
412 string OutputName(filename);
413 OutputName.append(TecplotSuffix);
414 ofstream *tecplot = new ofstream(OutputName.c_str());
[e138de]415 WriteTecplotFile(tecplot, TesselStruct, mol, 0);
[357fba]416 tecplot->close();
417 delete(tecplot);
[3d919e]418 }
[357fba]419 if (DoRaster3DOutput) {
420 string OutputName(filename);
421 OutputName.append(Raster3DSuffix);
422 ofstream *rasterplot = new ofstream(OutputName.c_str());
[e138de]423 WriteRaster3dFile(rasterplot, TesselStruct, mol);
[357fba]424 rasterplot->close();
425 delete(rasterplot);
[042f82]426 }
[3d919e]427 }
428
[ef0e6d]429
[357fba]430 // free reference lists
431 if (BoundaryFreeFlag)
432 delete[] (BoundaryPoints);
[3d919e]433};
[6ac7ee]434
[d4fa23]435/** For testing removes one boundary point after another to check for leaks.
436 * \param *out output stream for debugging
437 * \param *TesselStruct Tesselation containing envelope with boundary points
438 * \param *mol molecule
439 * \param *filename name of file
440 * \return true - all removed, false - something went wrong
441 */
[e138de]442bool RemoveAllBoundaryPoints(class Tesselation *&TesselStruct, const molecule * const mol, const char * const filename)
[d4fa23]443{
[f67b6e]444 Info FunctionInfo(__func__);
[d4fa23]445 int i=0;
446 char number[MAXSTRINGSIZE];
447
448 if ((TesselStruct == NULL) || (TesselStruct->PointsOnBoundary.empty())) {
[717e0c]449 eLog() << Verbose(1) << "TesselStruct is empty." << endl;
[d4fa23]450 return false;
451 }
452
453 PointMap::iterator PointRunner;
454 while (!TesselStruct->PointsOnBoundary.empty()) {
[f67b6e]455 Log() << Verbose(1) << "Remaining points are: ";
[d4fa23]456 for (PointMap::iterator PointSprinter = TesselStruct->PointsOnBoundary.begin(); PointSprinter != TesselStruct->PointsOnBoundary.end(); PointSprinter++)
[f67b6e]457 Log() << Verbose(0) << *(PointSprinter->second) << "\t";
458 Log() << Verbose(0) << endl;
[d4fa23]459
460 PointRunner = TesselStruct->PointsOnBoundary.begin();
461 // remove point
[e138de]462 TesselStruct->RemovePointFromTesselatedSurface(PointRunner->second);
[d4fa23]463
464 // store envelope
465 sprintf(number, "-%04d", i++);
[e138de]466 StoreTrianglesinFile(mol, (const Tesselation *&)TesselStruct, filename, number);
[d4fa23]467 }
468
469 return true;
470};
471
[08ef35]472/** Creates a convex envelope from a given non-convex one.
[093645]473 * -# First step, remove concave spots, i.e. singular "dents"
474 * -# We go through all PointsOnBoundary.
475 * -# We CheckConvexityCriterion() for all its lines.
476 * -# If all its lines are concave, it cannot be on the convex envelope.
477 * -# Hence, we remove it and re-create all its triangles from its getCircleOfConnectedPoints()
478 * -# We calculate the additional volume.
479 * -# We go over all lines until none yields a concavity anymore.
480 * -# Second step, remove concave lines, i.e. line-shape "dents"
481 * -# We go through all LinesOnBoundary
482 * -# We CheckConvexityCriterion()
483 * -# If it returns concave, we flip the line in this quadrupel of points (abusing the degeneracy of the tesselation)
484 * -# We CheckConvexityCriterion(),
485 * -# if it's concave, we continue
486 * -# if not, we mark an error and stop
[08ef35]487 * Note: This routine - for free - calculates the difference in volume between convex and
488 * non-convex envelope, as the former is easy to calculate - VolumeOfConvexEnvelope() - it
489 * can be used to compute volumes of arbitrary shapes.
490 * \param *out output stream for debugging
491 * \param *TesselStruct non-convex envelope, is changed in return!
[093645]492 * \param *mol molecule
493 * \param *filename name of file
[08ef35]494 * \return volume difference between the non- and the created convex envelope
495 */
[e138de]496double ConvexizeNonconvexEnvelope(class Tesselation *&TesselStruct, const molecule * const mol, const char * const filename)
[08ef35]497{
[f67b6e]498 Info FunctionInfo(__func__);
[08ef35]499 double volume = 0;
500 class BoundaryPointSet *point = NULL;
501 class BoundaryLineSet *line = NULL;
[ad37ab]502 bool Concavity = false;
[57066a]503 char dummy[MAXSTRINGSIZE];
[ad37ab]504 PointMap::iterator PointRunner;
505 PointMap::iterator PointAdvance;
506 LineMap::iterator LineRunner;
507 LineMap::iterator LineAdvance;
508 TriangleMap::iterator TriangleRunner;
509 TriangleMap::iterator TriangleAdvance;
510 int run = 0;
[093645]511
512 // check whether there is something to work on
[08ef35]513 if (TesselStruct == NULL) {
[717e0c]514 eLog() << Verbose(1) << "TesselStruct is empty!" << endl;
[08ef35]515 return volume;
516 }
517
[093645]518 // First step: RemovePointFromTesselatedSurface
[1d9b7aa]519 do {
520 Concavity = false;
[57066a]521 sprintf(dummy, "-first-%d", run);
[e138de]522 //CalculateConcavityPerBoundaryPoint(TesselStruct);
523 StoreTrianglesinFile(mol, (const Tesselation *&)TesselStruct, filename, dummy);
[57066a]524
[1d9b7aa]525 PointRunner = TesselStruct->PointsOnBoundary.begin();
526 PointAdvance = PointRunner; // we need an advanced point, as the PointRunner might get removed
527 while (PointRunner != TesselStruct->PointsOnBoundary.end()) {
528 PointAdvance++;
529 point = PointRunner->second;
[e138de]530 Log() << Verbose(1) << "INFO: Current point is " << *point << "." << endl;
[1d9b7aa]531 for (LineMap::iterator LineRunner = point->lines.begin(); LineRunner != point->lines.end(); LineRunner++) {
532 line = LineRunner->second;
[f67b6e]533 Log() << Verbose(1) << "INFO: Current line of point " << *point << " is " << *line << "." << endl;
[e138de]534 if (!line->CheckConvexityCriterion()) {
[57066a]535 // remove the point if needed
[e138de]536 Log() << Verbose(1) << "... point " << *point << " cannot be on convex envelope." << endl;
537 volume += TesselStruct->RemovePointFromTesselatedSurface(point);
[57066a]538 sprintf(dummy, "-first-%d", ++run);
[e138de]539 StoreTrianglesinFile(mol, (const Tesselation *&)TesselStruct, filename, dummy);
[57066a]540 Concavity = true;
541 break;
542 }
[1d9b7aa]543 }
544 PointRunner = PointAdvance;
[093645]545 }
546
[57066a]547 sprintf(dummy, "-second-%d", run);
[e138de]548 //CalculateConcavityPerBoundaryPoint(TesselStruct);
549 StoreTrianglesinFile(mol, (const Tesselation *&)TesselStruct, filename, dummy);
[093645]550
[1d9b7aa]551 // second step: PickFarthestofTwoBaselines
552 LineRunner = TesselStruct->LinesOnBoundary.begin();
553 LineAdvance = LineRunner; // we need an advanced line, as the LineRunner might get removed
554 while (LineRunner != TesselStruct->LinesOnBoundary.end()) {
555 LineAdvance++;
556 line = LineRunner->second;
[e138de]557 Log() << Verbose(1) << "INFO: Picking farthest baseline for line is " << *line << "." << endl;
[1d9b7aa]558 // take highest of both lines
[e138de]559 if (TesselStruct->IsConvexRectangle(line) == NULL) {
560 const double tmp = TesselStruct->PickFarthestofTwoBaselines(line);
[57066a]561 volume += tmp;
[ad37ab]562 if (tmp != 0.) {
[e138de]563 TesselStruct->FlipBaseline(line);
[57066a]564 Concavity = true;
565 }
[1d9b7aa]566 }
567 LineRunner = LineAdvance;
568 }
[57066a]569 run++;
[1d9b7aa]570 } while (Concavity);
[e138de]571 //CalculateConcavityPerBoundaryPoint(TesselStruct);
572 //StoreTrianglesinFile(mol, filename, "-third");
[093645]573
574 // third step: IsConvexRectangle
[7dea7c]575// LineRunner = TesselStruct->LinesOnBoundary.begin();
576// LineAdvance = LineRunner; // we need an advanced line, as the LineRunner might get removed
577// while (LineRunner != TesselStruct->LinesOnBoundary.end()) {
578// LineAdvance++;
579// line = LineRunner->second;
[e138de]580// Log() << Verbose(1) << "INFO: Current line is " << *line << "." << endl;
[7dea7c]581// //if (LineAdvance != TesselStruct->LinesOnBoundary.end())
[e138de]582// //Log() << Verbose(1) << "INFO: Next line will be " << *(LineAdvance->second) << "." << endl;
[7dea7c]583// if (!line->CheckConvexityCriterion(out)) {
[e138de]584// Log() << Verbose(1) << "... line " << *line << " is concave, flipping it." << endl;
[7dea7c]585//
586// // take highest of both lines
[e138de]587// point = TesselStruct->IsConvexRectangle(line);
[7dea7c]588// if (point != NULL)
[e138de]589// volume += TesselStruct->RemovePointFromTesselatedSurface(point);
[7dea7c]590// }
591// LineRunner = LineAdvance;
592// }
[093645]593
[e138de]594 CalculateConcavityPerBoundaryPoint(TesselStruct);
595 StoreTrianglesinFile(mol, (const Tesselation *&)TesselStruct, filename, "");
[0077b5]596
597 // end
[f67b6e]598 Log() << Verbose(0) << "Volume is " << volume << "." << endl;
[0077b5]599 return volume;
600};
601
[6ac7ee]602
[357fba]603/** Determines the volume of a cluster.
604 * Determines first the convex envelope, then tesselates it and calculates its volume.
[6ac7ee]605 * \param *out output stream for debugging
[357fba]606 * \param *TesselStruct Tesselation filled with points, lines and triangles on boundary on return
607 * \param *configuration needed for path to store convex envelope file
608 * \return determined volume of the cluster in cubed config:GetIsAngstroem()
[3d919e]609 */
[e138de]610double VolumeOfConvexEnvelope(class Tesselation *TesselStruct, class config *configuration)
[357fba]611{
[f67b6e]612 Info FunctionInfo(__func__);
[357fba]613 bool IsAngstroem = configuration->GetIsAngstroem();
614 double volume = 0.;
[ad37ab]615 Vector x;
616 Vector y;
[6ac7ee]617
[357fba]618 // 6a. Every triangle forms a pyramid with the center of gravity as its peak, sum up the volumes
619 for (TriangleMap::iterator runner = TesselStruct->TrianglesOnBoundary.begin(); runner != TesselStruct->TrianglesOnBoundary.end(); runner++)
620 { // go through every triangle, calculate volume of its pyramid with CoG as peak
[273382]621 x = (*runner->second->endpoints[0]->node->node) - (*runner->second->endpoints[1]->node->node);
622 y = (*runner->second->endpoints[0]->node->node) - (*runner->second->endpoints[2]->node->node);
623 const double a = sqrt(runner->second->endpoints[0]->node->node->DistanceSquared(*runner->second->endpoints[1]->node->node));
624 const double b = sqrt(runner->second->endpoints[0]->node->node->DistanceSquared(*runner->second->endpoints[2]->node->node));
625 const double c = sqrt(runner->second->endpoints[2]->node->node->DistanceSquared(*runner->second->endpoints[1]->node->node));
[ad37ab]626 const double G = sqrt(((a + b + c) * (a + b + c) - 2 * (a * a + b * b + c * c)) / 16.); // area of tesselated triangle
[0a4f7f]627 x = Plane(*(runner->second->endpoints[0]->node->node),
628 *(runner->second->endpoints[1]->node->node),
629 *(runner->second->endpoints[2]->node->node)).getNormal();
[273382]630 x.Scale(runner->second->endpoints[1]->node->node->ScalarProduct(x));
[ad37ab]631 const double h = x.Norm(); // distance of CoG to triangle
632 const double PyramidVolume = (1. / 3.) * G * h; // this formula holds for _all_ pyramids (independent of n-edge base or (not) centered peak)
[f67b6e]633 Log() << Verbose(1) << "Area of triangle is " << setprecision(10) << G << " "
[357fba]634 << (IsAngstroem ? "angstrom" : "atomiclength") << "^2, height is "
635 << h << " and the volume is " << PyramidVolume << " "
636 << (IsAngstroem ? "angstrom" : "atomiclength") << "^3." << endl;
637 volume += PyramidVolume;
[3d919e]638 }
[7f4bee]639 Log() << Verbose(0) << "RESULT: The summed volume is " << setprecision(6)
[357fba]640 << volume << " " << (IsAngstroem ? "angstrom" : "atomiclength") << "^3."
641 << endl;
[6ac7ee]642
[357fba]643 return volume;
[7dea7c]644};
645
646/** Stores triangles to file.
647 * \param *out output stream for debugging
648 * \param *mol molecule with atoms and bonds
[71b20e]649 * \param *TesselStruct Tesselation with boundary triangles
[7dea7c]650 * \param *filename prefix of filename
651 * \param *extraSuffix intermediate suffix
652 */
[71b20e]653void StoreTrianglesinFile(const molecule * const mol, const Tesselation * const TesselStruct, const char *filename, const char *extraSuffix)
[7dea7c]654{
[f67b6e]655 Info FunctionInfo(__func__);
[7dea7c]656 // 4. Store triangles in tecplot file
657 if (filename != NULL) {
658 if (DoTecplotOutput) {
659 string OutputName(filename);
660 OutputName.append(extraSuffix);
661 OutputName.append(TecplotSuffix);
662 ofstream *tecplot = new ofstream(OutputName.c_str());
[6a7f78c]663 WriteTecplotFile(tecplot, TesselStruct, mol, -1);
[7dea7c]664 tecplot->close();
665 delete(tecplot);
666 }
667 if (DoRaster3DOutput) {
668 string OutputName(filename);
669 OutputName.append(extraSuffix);
670 OutputName.append(Raster3DSuffix);
671 ofstream *rasterplot = new ofstream(OutputName.c_str());
[e138de]672 WriteRaster3dFile(rasterplot, TesselStruct, mol);
[7dea7c]673 rasterplot->close();
674 delete(rasterplot);
675 }
676 }
677};
[03648b]678
[357fba]679/** Creates multiples of the by \a *mol given cluster and suspends them in water with a given final density.
680 * We get cluster volume by VolumeOfConvexEnvelope() and its diameters by GetDiametersOfCluster()
681 * \param *out output stream for debugging
682 * \param *configuration needed for path to store convex envelope file
683 * \param *mol molecule structure representing the cluster
[776b64]684 * \param *&TesselStruct Tesselation structure with triangles on return
[357fba]685 * \param ClusterVolume guesstimated cluster volume, if equal 0 we used VolumeOfConvexEnvelope() instead.
686 * \param celldensity desired average density in final cell
[8c54a3]687 */
[e138de]688void PrepareClustersinWater(config *configuration, molecule *mol, double ClusterVolume, double celldensity)
[357fba]689{
[f67b6e]690 Info FunctionInfo(__func__);
[fa649a]691 bool IsAngstroem = true;
[ad37ab]692 double *GreatestDiameter = NULL;
693 Boundaries *BoundaryPoints = NULL;
694 class Tesselation *TesselStruct = NULL;
695 Vector BoxLengths;
696 int repetition[NDIM] = { 1, 1, 1 };
697 int TotalNoClusters = 1;
698 atom *Walker = NULL;
699 double totalmass = 0.;
700 double clustervolume = 0.;
701 double cellvolume = 0.;
702
[357fba]703 // transform to PAS
[e138de]704 mol->PrincipalAxisSystem(true);
[3d919e]705
[ad37ab]706 IsAngstroem = configuration->GetIsAngstroem();
[e138de]707 GreatestDiameter = GetDiametersOfCluster(BoundaryPoints, mol, TesselStruct, IsAngstroem);
708 BoundaryPoints = GetBoundaryPoints(mol, TesselStruct);
[357fba]709 LinkedCell LCList(mol, 10.);
[e138de]710 FindConvexBorder(mol, TesselStruct, &LCList, NULL);
[ad37ab]711
712 // some preparations beforehand
[357fba]713 if (ClusterVolume == 0)
[e138de]714 clustervolume = VolumeOfConvexEnvelope(TesselStruct, configuration);
[357fba]715 else
716 clustervolume = ClusterVolume;
[ad37ab]717
[357fba]718 for (int i = 0; i < NDIM; i++)
719 TotalNoClusters *= repetition[i];
[8c54a3]720
[357fba]721 // sum up the atomic masses
[ad37ab]722 Walker = mol->start;
723 while (Walker->next != mol->end) {
[357fba]724 Walker = Walker->next;
725 totalmass += Walker->type->mass;
[ad37ab]726 }
[e138de]727 Log() << Verbose(0) << "RESULT: The summed mass is " << setprecision(10) << totalmass << " atomicmassunit." << endl;
728 Log() << Verbose(0) << "RESULT: The average density is " << setprecision(10) << totalmass / clustervolume << " atomicmassunit/" << (IsAngstroem ? "angstrom" : "atomiclength") << "^3." << endl;
[8c54a3]729
[357fba]730 // solve cubic polynomial
[e138de]731 Log() << Verbose(1) << "Solving equidistant suspension in water problem ..." << endl;
[357fba]732 if (IsAngstroem)
[ad37ab]733 cellvolume = (TotalNoClusters * totalmass / SOLVENTDENSITY_A - (totalmass / clustervolume)) / (celldensity - 1);
[357fba]734 else
[ad37ab]735 cellvolume = (TotalNoClusters * totalmass / SOLVENTDENSITY_a0 - (totalmass / clustervolume)) / (celldensity - 1);
[e138de]736 Log() << Verbose(1) << "Cellvolume needed for a density of " << celldensity << " g/cm^3 is " << cellvolume << " " << (IsAngstroem ? "angstrom" : "atomiclength") << "^3." << endl;
[ad37ab]737
738 double minimumvolume = TotalNoClusters * (GreatestDiameter[0] * GreatestDiameter[1] * GreatestDiameter[2]);
[e138de]739 Log() << Verbose(1) << "Minimum volume of the convex envelope contained in a rectangular box is " << minimumvolume << " atomicmassunit/" << (IsAngstroem ? "angstrom" : "atomiclength") << "^3." << endl;
[ad37ab]740 if (minimumvolume > cellvolume) {
[717e0c]741 eLog() << Verbose(1) << "the containing box already has a greater volume than the envisaged cell volume!" << endl;
[e138de]742 Log() << Verbose(0) << "Setting Box dimensions to minimum possible, the greatest diameters." << endl;
[ad37ab]743 for (int i = 0; i < NDIM; i++)
[0a4f7f]744 BoxLengths[i] = GreatestDiameter[i];
[e138de]745 mol->CenterEdge(&BoxLengths);
[ad37ab]746 } else {
[0a4f7f]747 BoxLengths[0] = (repetition[0] * GreatestDiameter[0] + repetition[1] * GreatestDiameter[1] + repetition[2] * GreatestDiameter[2]);
748 BoxLengths[1] = (repetition[0] * repetition[1] * GreatestDiameter[0] * GreatestDiameter[1] + repetition[0] * repetition[2] * GreatestDiameter[0] * GreatestDiameter[2] + repetition[1] * repetition[2] * GreatestDiameter[1] * GreatestDiameter[2]);
749 BoxLengths[2] = minimumvolume - cellvolume;
[ad37ab]750 double x0 = 0.;
751 double x1 = 0.;
752 double x2 = 0.;
[0a4f7f]753 if (gsl_poly_solve_cubic(BoxLengths[0], BoxLengths[1], BoxLengths[2], &x0, &x1, &x2) == 1) // either 1 or 3 on return
[e138de]754 Log() << Verbose(0) << "RESULT: The resulting spacing is: " << x0 << " ." << endl;
[ad37ab]755 else {
[e138de]756 Log() << Verbose(0) << "RESULT: The resulting spacings are: " << x0 << " and " << x1 << " and " << x2 << " ." << endl;
[ad37ab]757 x0 = x2; // sorted in ascending order
[357fba]758 }
[8c54a3]759
[ad37ab]760 cellvolume = 1.;
761 for (int i = 0; i < NDIM; i++) {
[0a4f7f]762 BoxLengths[i] = repetition[i] * (x0 + GreatestDiameter[i]);
763 cellvolume *= BoxLengths[i];
[8c54a3]764 }
[ad37ab]765
766 // set new box dimensions
[e138de]767 Log() << Verbose(0) << "Translating to box with these boundaries." << endl;
[ad37ab]768 mol->SetBoxDimension(&BoxLengths);
[e138de]769 mol->CenterInBox();
[ad37ab]770 }
[357fba]771 // update Box of atoms by boundary
772 mol->SetBoxDimension(&BoxLengths);
[0a4f7f]773 Log() << Verbose(0) << "RESULT: The resulting cell dimensions are: " << BoxLengths[0] << " and " << BoxLengths[1] << " and " << BoxLengths[2] << " with total volume of " << cellvolume << " " << (IsAngstroem ? "angstrom" : "atomiclength") << "^3." << endl;
[ad37ab]774};
[8c54a3]775
776
[357fba]777/** Fills the empty space of the simulation box with water/
778 * \param *out output stream for debugging
779 * \param *List list of molecules already present in box
780 * \param *TesselStruct contains tesselated surface
781 * \param *filler molecule which the box is to be filled with
782 * \param configuration contains box dimensions
783 * \param distance[NDIM] distance between filling molecules in each direction
[9473f6]784 * \param boundary length of boundary zone between molecule and filling mollecules
[71b20e]785 * \param epsilon distance to surface which is not filled
[357fba]786 * \param RandAtomDisplacement maximum distance for random displacement per atom
787 * \param RandMolDisplacement maximum distance for random displacement per filler molecule
788 * \param DoRandomRotation true - do random rotiations, false - don't
789 * \return *mol pointer to new molecule with filled atoms
[6ac7ee]790 */
[9473f6]791molecule * FillBoxWithMolecule(MoleculeListClass *List, molecule *filler, config &configuration, const double distance[NDIM], const double boundary, const double RandomAtomDisplacement, const double RandomMolDisplacement, const bool DoRandomRotation)
[6ac7ee]792{
[f67b6e]793 Info FunctionInfo(__func__);
[23b547]794 molecule *Filling = World::getInstance().createMolecule();
[357fba]795 Vector CurrentPosition;
796 int N[NDIM];
797 int n[NDIM];
[f66195]798 double *M = ReturnFullMatrixforSymmetric(filler->cell_size);
[357fba]799 double Rotations[NDIM*NDIM];
800 Vector AtomTranslations;
801 Vector FillerTranslations;
802 Vector FillerDistance;
803 double FillIt = false;
[ef0e6d]804 atom *Walker = NULL;
[357fba]805 bond *Binder = NULL;
[ad37ab]806 int i = 0;
[ef0e6d]807 LinkedCell *LCList[List->ListOfMolecules.size()];
[ad37ab]808 double phi[NDIM];
[776b64]809 class Tesselation *TesselStruct[List->ListOfMolecules.size()];
[ef0e6d]810
811 i=0;
812 for (MoleculeList::iterator ListRunner = List->ListOfMolecules.begin(); ListRunner != List->ListOfMolecules.end(); ListRunner++) {
[e138de]813 Log() << Verbose(1) << "Pre-creating linked cell lists for molecule " << *ListRunner << "." << endl;
[71b20e]814 LCList[i] = new LinkedCell((*ListRunner), 10.); // get linked cell list
815 Log() << Verbose(1) << "Pre-creating tesselation for molecule " << *ListRunner << "." << endl;
816 TesselStruct[i] = NULL;
817 FindNonConvexBorder((*ListRunner), TesselStruct[i], (const LinkedCell *&)LCList[i], 5., NULL);
[ef0e6d]818 i++;
819 }
[8c54a3]820
[357fba]821 // Center filler at origin
[e138de]822 filler->CenterOrigin();
[357fba]823 filler->Center.Zero();
[8c54a3]824
[e138de]825 filler->CountAtoms();
[ef0e6d]826 atom * CopyAtoms[filler->AtomCount];
827
[357fba]828 // calculate filler grid in [0,1]^3
[1bd79e]829 FillerDistance = Vector(distance[0], distance[1], distance[2]);
[357fba]830 FillerDistance.InverseMatrixMultiplication(M);
[71b20e]831 for(int i=0;i<NDIM;i++)
[0a4f7f]832 N[i] = static_cast<int>(ceil(1./FillerDistance[i]));
[71b20e]833 Log() << Verbose(1) << "INFO: Grid steps are " << N[0] << ", " << N[1] << ", " << N[2] << "." << endl;
[8c54a3]834
[d6eb80]835 // initialize seed of random number generator to current time
836 srand ( time(NULL) );
837
[357fba]838 // go over [0,1]^3 filler grid
839 for (n[0] = 0; n[0] < N[0]; n[0]++)
840 for (n[1] = 0; n[1] < N[1]; n[1]++)
841 for (n[2] = 0; n[2] < N[2]; n[2]++) {
842 // calculate position of current grid vector in untransformed box
[1bd79e]843 CurrentPosition = Vector((double)n[0]/(double)N[0], (double)n[1]/(double)N[1], (double)n[2]/(double)N[2]);
[357fba]844 CurrentPosition.MatrixMultiplication(M);
[e138de]845 Log() << Verbose(2) << "INFO: Current Position is " << CurrentPosition << "." << endl;
[357fba]846 // Check whether point is in- or outside
847 FillIt = true;
[ef0e6d]848 i=0;
[357fba]849 for (MoleculeList::iterator ListRunner = List->ListOfMolecules.begin(); ListRunner != List->ListOfMolecules.end(); ListRunner++) {
[ef0e6d]850 // get linked cell list
[776b64]851 if (TesselStruct[i] == NULL) {
[8468cb]852 eLog() << Verbose(0) << "TesselStruct of " << (*ListRunner) << " is NULL. Didn't we pre-create it?" << endl;
[ef0e6d]853 FillIt = false;
[776b64]854 } else {
[9473f6]855 const double distance = (TesselStruct[i]->GetDistanceSquaredToSurface(CurrentPosition, LCList[i]));
856 FillIt = FillIt && (distance > boundary*boundary);
857 if (FillIt) {
[71b20e]858 Log() << Verbose(1) << "INFO: Position at " << CurrentPosition << " is outer point." << endl;
859 } else {
[9473f6]860 Log() << Verbose(1) << "INFO: Position at " << CurrentPosition << " is inner point or within boundary." << endl;
861 break;
[71b20e]862 }
[776b64]863 i++;
864 }
[3d919e]865 }
[8c54a3]866
[8468cb]867 if (FillIt) {
[357fba]868 // fill in Filler
[e138de]869 Log() << Verbose(2) << "Space at " << CurrentPosition << " is unoccupied by any molecule, filling in." << endl;
[8c54a3]870
[357fba]871 // create molecule random translation vector ...
872 for (int i=0;i<NDIM;i++)
[0a4f7f]873 FillerTranslations[i] = RandomMolDisplacement*(rand()/(RAND_MAX/2.) - 1.);
[f67b6e]874 Log() << Verbose(2) << "INFO: Translating this filler by " << FillerTranslations << "." << endl;
[8c54a3]875
[357fba]876 // go through all atoms
877 Walker = filler->start;
[ef0e6d]878 while (Walker->next != filler->end) {
[357fba]879 Walker = Walker->next;
880 // copy atom ...
[46d958]881 CopyAtoms[Walker->nr] = Walker->clone();
[8c54a3]882
[357fba]883 // create atomic random translation vector ...
884 for (int i=0;i<NDIM;i++)
[0a4f7f]885 AtomTranslations[i] = RandomAtomDisplacement*(rand()/(RAND_MAX/2.) - 1.);
[8c54a3]886
[357fba]887 // ... and rotation matrix
888 if (DoRandomRotation) {
889 for (int i=0;i<NDIM;i++) {
890 phi[i] = rand()/(RAND_MAX/(2.*M_PI));
[8c54a3]891 }
[3d919e]892
[357fba]893 Rotations[0] = cos(phi[0]) *cos(phi[2]) + (sin(phi[0])*sin(phi[1])*sin(phi[2]));
894 Rotations[3] = sin(phi[0]) *cos(phi[2]) - (cos(phi[0])*sin(phi[1])*sin(phi[2]));
895 Rotations[6] = cos(phi[1])*sin(phi[2]) ;
896 Rotations[1] = - sin(phi[0])*cos(phi[1]) ;
897 Rotations[4] = cos(phi[0])*cos(phi[1]) ;
898 Rotations[7] = sin(phi[1]) ;
899 Rotations[3] = - cos(phi[0]) *sin(phi[2]) + (sin(phi[0])*sin(phi[1])*cos(phi[2]));
900 Rotations[5] = - sin(phi[0]) *sin(phi[2]) - (cos(phi[0])*sin(phi[1])*cos(phi[2]));
901 Rotations[8] = cos(phi[1])*cos(phi[2]) ;
[8c54a3]902 }
903
[357fba]904 // ... and put at new position
905 if (DoRandomRotation)
[ef0e6d]906 CopyAtoms[Walker->nr]->x.MatrixMultiplication(Rotations);
[273382]907 CopyAtoms[Walker->nr]->x += AtomTranslations;
908 CopyAtoms[Walker->nr]->x += FillerTranslations;
909 CopyAtoms[Walker->nr]->x += CurrentPosition;
[ef0e6d]910
[357fba]911 // insert into Filling
[f3278b]912
913 // FIXME: gives completely different results if CopyAtoms[..] used instead of Walker, why???
[e138de]914 Log() << Verbose(4) << "Filling atom " << *Walker << ", translated to " << AtomTranslations << ", at final position is " << (CopyAtoms[Walker->nr]->x) << "." << endl;
[ef0e6d]915 Filling->AddAtom(CopyAtoms[Walker->nr]);
[357fba]916 }
[3d919e]917
[357fba]918 // go through all bonds and add as well
919 Binder = filler->first;
[ef0e6d]920 while(Binder->next != filler->last) {
[357fba]921 Binder = Binder->next;
[e138de]922 Log() << Verbose(3) << "Adding Bond between " << *CopyAtoms[Binder->leftatom->nr] << " and " << *CopyAtoms[Binder->rightatom->nr]<< "." << endl;
[ef0e6d]923 Filling->AddBond(CopyAtoms[Binder->leftatom->nr], CopyAtoms[Binder->rightatom->nr], Binder->BondDegree);
[8c54a3]924 }
[018741]925 } else {
[357fba]926 // leave empty
[e138de]927 Log() << Verbose(2) << "Space at " << CurrentPosition << " is occupied." << endl;
[8c54a3]928 }
929 }
[1614174]930 Free(&M);
[71b20e]931
932 // output to file
933 TesselStruct[0]->LastTriangle = NULL;
934 StoreTrianglesinFile(Filling, TesselStruct[0], "Tesselated", ".dat");
935
[7f4bee]936 for (size_t i=0;i<List->ListOfMolecules.size();i++) {
937 delete(LCList[i]);
938 delete(TesselStruct[i]);
939 }
[357fba]940 return Filling;
[3d919e]941};
[8c54a3]942
943
[6ac7ee]944/** Tesselates the non convex boundary by rolling a virtual sphere along the surface of the molecule.
945 * \param *out output stream for debugging
946 * \param *mol molecule structure with Atom's and Bond's
[776b64]947 * \param *&TesselStruct Tesselation filled with points, lines and triangles on boundary on return
948 * \param *&LCList atoms in LinkedCell list
[57066a]949 * \param RADIUS radius of the virtual sphere
[6ac7ee]950 * \param *filename filename prefix for output of vertex data
[4fc93f]951 * \return true - tesselation successful, false - tesselation failed
[6ac7ee]952 */
[4fc93f]953bool FindNonConvexBorder(const molecule* const mol, Tesselation *&TesselStruct, const LinkedCell *&LCList, const double RADIUS, const char *filename = NULL)
[03648b]954{
[f67b6e]955 Info FunctionInfo(__func__);
[3d919e]956 bool freeLC = false;
[4fc93f]957 bool status = false;
[24a5e0]958 CandidateForTesselation *baseline=0;
[ad37ab]959 LineMap::iterator testline;
[1e168b]960 bool OneLoopWithoutSuccessFlag = true; // marks whether we went once through all baselines without finding any without two triangles
[ad37ab]961 bool TesselationFailFlag = false;
[f67b6e]962 BoundaryTriangleSet *T = NULL;
[357fba]963
[776b64]964 if (TesselStruct == NULL) {
[e138de]965 Log() << Verbose(1) << "Allocating Tesselation struct ..." << endl;
[776b64]966 TesselStruct= new Tesselation;
[ef0e6d]967 } else {
[776b64]968 delete(TesselStruct);
[e138de]969 Log() << Verbose(1) << "Re-Allocating Tesselation struct ..." << endl;
[776b64]970 TesselStruct = new Tesselation;
[3d919e]971 }
[ad37ab]972
[57066a]973 // initialise Linked Cell
[3d919e]974 if (LCList == NULL) {
975 LCList = new LinkedCell(mol, 2.*RADIUS);
976 freeLC = true;
977 }
978
[57066a]979 // 1. get starting triangle
[e138de]980 TesselStruct->FindStartingTriangle(RADIUS, LCList);
[3d919e]981
[57066a]982 // 2. expand from there
[1e168b]983 while ((!TesselStruct->OpenLines.empty()) && (OneLoopWithoutSuccessFlag)) {
984 // 2a. fill all new OpenLines
[f67b6e]985 Log() << Verbose(1) << "There are " << TesselStruct->OpenLines.size() << " open lines to scan for candidates:" << endl;
[1e168b]986 for (CandidateMap::iterator Runner = TesselStruct->OpenLines.begin(); Runner != TesselStruct->OpenLines.end(); Runner++)
[f67b6e]987 Log() << Verbose(2) << *(Runner->second) << endl;
[1e168b]988
989 for (CandidateMap::iterator Runner = TesselStruct->OpenLines.begin(); Runner != TesselStruct->OpenLines.end(); Runner++) {
990 baseline = Runner->second;
[f67b6e]991 if (baseline->pointlist.empty()) {
992 T = (((baseline->BaseLine->triangles.begin()))->second);
993 Log() << Verbose(1) << "Finding best candidate for open line " << *baseline->BaseLine << " of triangle " << *T << endl;
994 TesselationFailFlag = TesselStruct->FindNextSuitableTriangle(*baseline, *T, RADIUS, LCList); //the line is there, so there is a triangle, but only one.
[e359a8]995 }
[3d919e]996 }
997
[1e168b]998 // 2b. search for smallest ShortestAngle among all candidates
999 double ShortestAngle = 4.*M_PI;
[f67b6e]1000 Log() << Verbose(1) << "There are " << TesselStruct->OpenLines.size() << " open lines to scan for the best candidates:" << endl;
[1e168b]1001 for (CandidateMap::iterator Runner = TesselStruct->OpenLines.begin(); Runner != TesselStruct->OpenLines.end(); Runner++)
[f67b6e]1002 Log() << Verbose(2) << *(Runner->second) << endl;
[1e168b]1003
1004 for (CandidateMap::iterator Runner = TesselStruct->OpenLines.begin(); Runner != TesselStruct->OpenLines.end(); Runner++) {
1005 if (Runner->second->ShortestAngle < ShortestAngle) {
1006 baseline = Runner->second;
1007 ShortestAngle = baseline->ShortestAngle;
[f67b6e]1008 //Log() << Verbose(1) << "New best candidate is " << *baseline->BaseLine << " with point " << *baseline->point << " and angle " << baseline->ShortestAngle << endl;
[1e168b]1009 }
1010 }
[f67b6e]1011 if ((ShortestAngle == 4.*M_PI) || (baseline->pointlist.empty()))
[7dea7c]1012 OneLoopWithoutSuccessFlag = false;
[1e168b]1013 else {
1014 TesselStruct->AddCandidateTriangle(*baseline);
1015 }
1016
1017 // write temporary envelope
1018 if (filename != NULL) {
1019 if ((DoSingleStepOutput && ((TesselStruct->TrianglesOnBoundary.size() % SingleStepWidth == 0)))) { // if we have a new triangle and want to output each new triangle configuration
1020 TesselStruct->Output(filename, mol);
1021 }
[3d919e]1022 }
1023 }
[4fc93f]1024// // check envelope for consistency
1025// status = CheckListOfBaselines(TesselStruct);
1026//
1027// // look whether all points are inside of the convex envelope, otherwise add them via degenerated triangles
1028// //->InsertStraddlingPoints(mol, LCList);
[57066a]1029// mol->GoToFirst();
1030// class TesselPoint *Runner = NULL;
1031// while (!mol->IsEnd()) {
1032// Runner = mol->GetPoint();
[e138de]1033// Log() << Verbose(1) << "Checking on " << Runner->Name << " ... " << endl;
1034// if (!->IsInnerPoint(Runner, LCList)) {
1035// Log() << Verbose(2) << Runner->Name << " is outside of envelope, adding via degenerated triangles." << endl;
1036// ->AddBoundaryPointByDegeneratedTriangle(Runner, LCList);
[57066a]1037// } else {
[e138de]1038// Log() << Verbose(2) << Runner->Name << " is inside of or on envelope." << endl;
[57066a]1039// }
1040// mol->GoToNext();
1041// }
[357fba]1042
[f67b6e]1043// // Purges surplus triangles.
1044// TesselStruct->RemoveDegeneratedTriangles();
[4fc93f]1045
[b998c3]1046 // check envelope for consistency
1047 status = CheckListOfBaselines(TesselStruct);
1048
1049 // store before correction
1050 StoreTrianglesinFile(mol, (const Tesselation *&)TesselStruct, filename, "");
1051
[856098]1052 // correct degenerated polygons
1053 TesselStruct->CorrectAllDegeneratedPolygons();
1054
[4fc93f]1055 // check envelope for consistency
1056 status = CheckListOfBaselines(TesselStruct);
[ef0e6d]1057
[57066a]1058 // write final envelope
[e138de]1059 CalculateConcavityPerBoundaryPoint(TesselStruct);
1060 StoreTrianglesinFile(mol, (const Tesselation *&)TesselStruct, filename, "");
[8c54a3]1061
[3d919e]1062 if (freeLC)
1063 delete(LCList);
[4fc93f]1064
1065 return status;
[6ac7ee]1066};
[03648b]1067
[57066a]1068
[ad37ab]1069/** Finds a hole of sufficient size in \a *mols to embed \a *srcmol into it.
[ca2587]1070 * \param *out output stream for debugging
[ad37ab]1071 * \param *mols molecules in the domain to embed in between
1072 * \param *srcmol embedding molecule
[ca2587]1073 * \return *Vector new center of \a *srcmol for embedding relative to \a this
1074 */
[e138de]1075Vector* FindEmbeddingHole(MoleculeListClass *mols, molecule *srcmol)
[ca2587]1076{
[f67b6e]1077 Info FunctionInfo(__func__);
[ca2587]1078 Vector *Center = new Vector;
1079 Center->Zero();
1080 // calculate volume/shape of \a *srcmol
1081
1082 // find embedding holes
1083
1084 // if more than one, let user choose
1085
1086 // return embedding center
1087 return Center;
1088};
1089
Note: See TracBrowser for help on using the repository browser.