source: src/boundary.cpp@ e0b6fd

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 e0b6fd was ea7176, checked in by Tillmann Crueger <crueger@…>, 16 years ago

FIX: Made AtomCount a Cacheable so that the number of atoms in a molecule will always be correct

All unittests working.
All Complete testcases fail.

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