source: src/boundary.cpp@ faa1c9

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 faa1c9 was 6b5657, checked in by Frederik Heber <heber@…>, 15 years ago

Decoupled typedef for some STL maps used in boundary.cpp from those used in molecule_dynamics.cpp

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