source: src/Fragmentation/Exporters/SaturatedFragment.cpp@ 6ef48c

Last change on this file since 6ef48c was 97dff0, checked in by Frederik Heber <heber@…>, 11 years ago

Reimplemented SaturatedFragment::saturateAtom() and added ::setHydrogenReplacement().

  • this is now based on an ideal distribution of electrons around an atom via the VSPER model which we match as closely as possible with the still bound neighboring atoms and fill all remaining vacant spots with hydrogens.
  • SaturatedFragment sets father simply to the _given_ atom.
  • TEST: dimethyl_bromomalonate marked as XFAIL.
  • Property mode set to 100644
File size: 23.2 KB
Line 
1/*
2 * Project: MoleCuilder
3 * Description: creates and alters molecular systems
4 * Copyright (C) 2013 University of Bonn. All rights reserved.
5 * Copyright (C) 2013 Frederik Heber. All rights reserved.
6 *
7 *
8 * This file is part of MoleCuilder.
9 *
10 * MoleCuilder is free software: you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation, either version 2 of the License, or
13 * (at your option) any later version.
14 *
15 * MoleCuilder is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
19 *
20 * You should have received a copy of the GNU General Public License
21 * along with MoleCuilder. If not, see <http://www.gnu.org/licenses/>.
22 */
23
24/*
25 * SaturatedFragment.cpp
26 *
27 * Created on: Mar 3, 2013
28 * Author: heber
29 */
30
31// include config.h
32#ifdef HAVE_CONFIG_H
33#include <config.h>
34#endif
35
36#include "CodePatterns/MemDebug.hpp"
37
38#include "SaturatedFragment.hpp"
39
40#include <cmath>
41#include <iostream>
42
43#include "CodePatterns/Assert.hpp"
44#include "CodePatterns/Log.hpp"
45
46#include "LinearAlgebra/Exceptions.hpp"
47#include "LinearAlgebra/Plane.hpp"
48#include "LinearAlgebra/RealSpaceMatrix.hpp"
49#include "LinearAlgebra/Vector.hpp"
50
51#include "Atom/atom.hpp"
52#include "Bond/bond.hpp"
53#include "config.hpp"
54#include "Descriptors/AtomIdDescriptor.hpp"
55#include "Fragmentation/Exporters/HydrogenPool.hpp"
56#include "Fragmentation/Exporters/SphericalPointDistribution.hpp"
57#include "Fragmentation/HydrogenSaturation_enum.hpp"
58#include "Graph/BondGraph.hpp"
59#include "World.hpp"
60
61SaturatedFragment::SaturatedFragment(
62 const KeySet &_set,
63 KeySetsInUse_t &_container,
64 HydrogenPool &_hydrogens,
65 const enum HydrogenTreatment _treatment,
66 const enum HydrogenSaturation _saturation) :
67 container(_container),
68 set(_set),
69 hydrogens(_hydrogens),
70 FullMolecule(set),
71 treatment(_treatment),
72 saturation(_saturation)
73{
74 // add to in-use container
75 ASSERT( container.find(set) == container.end(),
76 "SaturatedFragment::SaturatedFragment() - the set "
77 +toString(set)+" is already marked as in use.");
78 container.insert(set);
79
80 // prepare saturation hydrogens
81 saturate();
82}
83
84SaturatedFragment::~SaturatedFragment()
85{
86 // release all saturation hydrogens if present
87 for (KeySet::iterator iter = SaturationHydrogens.begin();
88 !SaturationHydrogens.empty();
89 iter = SaturationHydrogens.begin()) {
90 hydrogens.releaseHydrogen(*iter);
91 SaturationHydrogens.erase(iter);
92 }
93
94 // remove ourselves from in-use container
95 KeySetsInUse_t::iterator iter = container.find(set);
96 ASSERT( container.find(set) != container.end(),
97 "SaturatedFragment::SaturatedFragment() - the set "
98 +toString(set)+" is not marked as in use.");
99 container.erase(iter);
100}
101
102void SaturatedFragment::saturate()
103{
104 // so far, we just have a set of keys. Hence, convert to atom refs
105 // and gather all atoms in a vector
106 std::vector<atom *> atoms;
107 for (KeySet::const_iterator iter = FullMolecule.begin();
108 iter != FullMolecule.end();
109 ++iter) {
110 atom * const Walker = World::getInstance().getAtom(AtomById(*iter));
111 ASSERT( Walker != NULL,
112 "SaturatedFragment::OutputConfig() - id "
113 +toString(*iter)+" is unknown to World.");
114 atoms.push_back(Walker);
115 }
116
117// bool LonelyFlag = false;
118 // go through each atom of the fragment and gather all cut bonds in list
119 typedef std::map<atom *, BondList > CutBonds_t;
120 CutBonds_t CutBonds;
121 for (std::vector<atom *>::const_iterator iter = atoms.begin();
122 iter != atoms.end();
123 ++iter) {
124 atom * const Walker = *iter;
125
126 // go through all bonds
127 const BondList& ListOfBonds = Walker->getListOfBonds();
128 for (BondList::const_iterator BondRunner = ListOfBonds.begin();
129 BondRunner != ListOfBonds.end();
130 ++BondRunner) {
131 atom * const OtherWalker = (*BondRunner)->GetOtherAtom(Walker);
132 // if other atom is in key set
133 if (set.find(OtherWalker->getId()) != set.end()) {
134 LOG(4, "DEBUG: Walker " << *Walker << " is bound to " << *OtherWalker << ".");
135// if (OtherWalker->getId() > Walker->getId()) { // add bond (Nr check is for adding only one of both variants: ab, ba)
136//// std::stringstream output;
137//// output << "ACCEPT: Adding Bond: "
138// output << Leaf->AddBond((*iter), OtherWalker, (*BondRunner)->getDegree());
139//// LOG(3, output.str());
140// //NumBonds[(*iter)->getNr()]++;
141// } else {
142//// LOG(3, "REJECY: Not adding bond, labels in wrong order.");
143// }
144// LonelyFlag = false;
145 } else {
146 LOG(4, "DEBUG: Walker " << *Walker << " is bound to "
147 << *OtherWalker << ", who is not in this fragment molecule.");
148 if (saturation == DoSaturate) {
149// LOG(3, "ACCEPT: Adding Hydrogen to " << (*iter)->Name << " and a bond in between.");
150 if (CutBonds.count(Walker) == 0)
151 CutBonds.insert( std::make_pair(Walker, BondList() ));
152 CutBonds[Walker].push_back(*BondRunner);
153 }
154// } else if ((treatment == ExcludeHydrogen) && (OtherWalker->getElementNo() == (atomicNumber_t)1)) {
155// // just copy the atom if it's a hydrogen
156// atom * const OtherWalker = Leaf->AddCopyAtom(OtherWalker);
157// Leaf->AddBond((*iter), OtherWalker, (*BondRunner)->getDegree());
158// }
159 //NumBonds[(*iter)->getNr()] += Binder->getDegree();
160 }
161 }
162 }
163
164 // go through all cut bonds and replace with a hydrogen
165 for (CutBonds_t::const_iterator atomiter = CutBonds.begin();
166 atomiter != CutBonds.end(); ++atomiter) {
167 atom * const Walker = atomiter->first;
168 if (!saturateAtom(Walker, atomiter->second))
169 exit(1);
170 }
171}
172
173bool SaturatedFragment::saturateAtom(
174 atom * const _atom,
175 const BondList &_cutbonds)
176{
177 // OLD WAY: use AddHydrogenReplacementAtom() on each cut bond
178// // go through each bond and replace
179// for (BondList::const_iterator bonditer = _cutbonds.begin();
180// bonditer != _cutbonds.end(); ++bonditer) {
181// atom * const OtherWalker = (*bonditer)->GetOtherAtom(_atom);
182// if (!AddHydrogenReplacementAtom(
183// (*bonditer),
184// _atom,
185// OtherWalker,
186// World::getInstance().getConfig()->IsAngstroem == 1))
187// return false;
188// }
189
190 // gather the nodes of the shape defined by the current set of bonded atoms
191 SphericalPointDistribution::Polygon_t Polygon;
192 for (BondList::const_iterator bonditer = _cutbonds.begin();
193 bonditer != _cutbonds.end(); ++bonditer) {
194 Vector DistanceVector;
195 if ((*bonditer)->leftatom == _atom)
196 DistanceVector = (*bonditer)->rightatom->getPosition() - (*bonditer)->leftatom->getPosition();
197 else
198 DistanceVector = (*bonditer)->leftatom->getPosition() - (*bonditer)->rightatom->getPosition();
199 // always use unit distances
200 DistanceVector.Normalize();
201 Polygon.push_back( DistanceVector );
202 }
203
204 // get the new number of bonds (where all cut bonds are replaced by as
205 // many hydrogens as is their degree)
206 // unsigned int NumberOfPoints = _atom->getElement().NoValenceOrbitals;
207 unsigned int NumberOfPoints =
208 _atom->getListOfBonds().size() - _cutbonds.size();
209 const BondList& ListOfBonds = _atom->getListOfBonds();
210 for (BondList::const_iterator BondRunner = ListOfBonds.begin();
211 BondRunner != ListOfBonds.end();
212 ++BondRunner) {
213 NumberOfPoints += (*BondRunner)->getDegree();
214 // ALTERNATIVE starting from valence:
215 // // if not one of the cut bonds, reduce by its bond degree -1 (for the one bond itself)
216 // if (_cutbonds.count(*BondRunner))
217 // NumberOfPoints -= (*BondRunner)->getDegree() - 1;
218 }
219
220 // get perfect node distribution for the given remaining atoms with respect
221 // to valence of the atoms (for a saturated fragment, resembles number of bonds)
222 SphericalPointDistribution polygonizer;
223 SphericalPointDistribution::Polygon_t NewPolygon;
224 switch (NumberOfPoints)
225 {
226 case 0:
227 NewPolygon = polygonizer.get<0>();
228 break;
229 case 1:
230 NewPolygon = polygonizer.get<1>();
231 break;
232 case 2:
233 NewPolygon = polygonizer.get<2>();
234 break;
235 case 3:
236 NewPolygon = polygonizer.get<3>();
237 break;
238 case 4:
239 NewPolygon = polygonizer.get<4>();
240 break;
241 case 5:
242 NewPolygon = polygonizer.get<5>();
243 break;
244 case 6:
245 NewPolygon = polygonizer.get<6>();
246 break;
247 case 7:
248 NewPolygon = polygonizer.get<7>();
249 break;
250 case 8:
251 NewPolygon = polygonizer.get<8>();
252 break;
253 case 9:
254 NewPolygon = polygonizer.get<9>();
255 break;
256 case 10:
257 NewPolygon = polygonizer.get<10>();
258 break;
259 case 11:
260 NewPolygon = polygonizer.get<11>();
261 break;
262 case 12:
263 NewPolygon = polygonizer.get<12>();
264 break;
265 case 14:
266 NewPolygon = polygonizer.get<14>();
267 break;
268 default:
269 ASSERT(0, "SaturatedFragment::saturateAtom() - cannot deal with the case "
270 +toString(NumberOfPoints)+".");
271 }
272
273 // then we need to match the old with the new
274 SphericalPointDistribution::Polygon_t RemainingPoints =
275 SphericalPointDistribution::matchSphericalPointDistributions(Polygon, NewPolygon);
276
277 // and place hydrogen atoms at each vacant spot in the distance given by the table
278 for(SphericalPointDistribution::Polygon_t::const_iterator iter = RemainingPoints.begin();
279 iter != RemainingPoints.end(); ++iter) {
280 // find nearest atom as father to this point
281 atom * const _father = _atom;
282 setHydrogenReplacement(
283 _atom,
284 *iter,
285 1.,
286 _father);
287 }
288
289 return true;
290}
291
292
293bool SaturatedFragment::OutputConfig(
294 std::ostream &out,
295 const ParserTypes _type) const
296{
297 // gather all atoms in a vector
298 std::vector<atom *> atoms;
299 for (KeySet::const_iterator iter = FullMolecule.begin();
300 iter != FullMolecule.end();
301 ++iter) {
302 atom * const Walker = World::getInstance().getAtom(AtomById(*iter));
303 ASSERT( Walker != NULL,
304 "SaturatedFragment::OutputConfig() - id "
305 +toString(*iter)+" is unknown to World.");
306 atoms.push_back(Walker);
307 }
308
309 // TODO: ScanForPeriodicCorrection() is missing so far!
310 // note however that this is not straight-forward for the following reasons:
311 // - we do not copy all atoms anymore, hence we are forced to shift the real
312 // atoms hither and back again
313 // - we use a long-range potential that supports periodic boundary conditions.
314 // Hence, there we would like the original configuration (split through the
315 // the periodic boundaries). Otherwise, we would have to shift (and probably
316 // interpolate) the potential with OBCs applying.
317
318 // list atoms in fragment for debugging
319 {
320 std::stringstream output;
321 output << "INFO: Contained atoms: ";
322 for (std::vector<atom *>::const_iterator iter = atoms.begin();
323 iter != atoms.end(); ++iter) {
324 output << (*iter)->getName() << " ";
325 }
326 LOG(3, output.str());
327 }
328
329 // store to stream via FragmentParser
330 const bool intermediateResult =
331 FormatParserStorage::getInstance().save(
332 out,
333 FormatParserStorage::getInstance().getSuffixFromType(_type),
334 atoms);
335
336 return intermediateResult;
337}
338
339atom * const SaturatedFragment::getHydrogenReplacement(atom * const replacement)
340{
341 atom * const _atom = hydrogens.leaseHydrogen(); // new atom
342 _atom->setAtomicVelocity(replacement->getAtomicVelocity()); // copy velocity
343 _atom->setFixedIon(replacement->getFixedIon());
344 // if we replace hydrogen, we mark it as our father, otherwise we are just an added hydrogen with no father
345 _atom->father = replacement;
346 SaturationHydrogens.insert(_atom->getId());
347 return _atom;
348}
349
350void SaturatedFragment::setHydrogenReplacement(
351 const atom * const _OwnerAtom,
352 const Vector &_position,
353 const double _distance,
354 atom * const _father)
355{
356 atom * const _atom = hydrogens.leaseHydrogen(); // new atom
357 _atom->setPosition( _OwnerAtom->getPosition() + _distance * _position );
358 // always set as fixed ion (not moving during molecular dynamics simulation)
359 _atom->setFixedIon(true);
360 // if we replace hydrogen, we mark it as our father, otherwise we are just an added hydrogen with no father
361 _atom->father = _father;
362 SaturationHydrogens.insert(_atom->getId());
363}
364
365bool SaturatedFragment::AddHydrogenReplacementAtom(
366 bond::ptr TopBond,
367 atom *Origin,
368 atom *Replacement,
369 bool IsAngstroem)
370{
371// Info info(__func__);
372 bool AllWentWell = true; // flag gathering the boolean return value of molecule::AddAtom and other functions, as return value on exit
373 double bondlength; // bond length of the bond to be replaced/cut
374 double bondangle; // bond angle of the bond to be replaced/cut
375 double BondRescale; // rescale value for the hydrogen bond length
376 bond::ptr FirstBond;
377 bond::ptr SecondBond; // Other bonds in double bond case to determine "other" plane
378 atom *FirstOtherAtom = NULL, *SecondOtherAtom = NULL, *ThirdOtherAtom = NULL; // pointer to hydrogen atoms to be added
379 double b,l,d,f,g, alpha, factors[NDIM]; // hold temporary values in triple bond case for coordination determination
380 Vector Orthovector1, Orthovector2; // temporary vectors in coordination construction
381 Vector InBondvector; // vector in direction of *Bond
382 const RealSpaceMatrix &matrix = World::getInstance().getDomain().getM();
383 bond::ptr Binder;
384
385 // create vector in direction of bond
386 InBondvector = Replacement->getPosition() - Origin->getPosition();
387 bondlength = InBondvector.Norm();
388
389 // is greater than typical bond distance? Then we have to correct periodically
390 // the problem is not the H being out of the box, but InBondvector have the wrong direction
391 // due to Replacement or Origin being on the wrong side!
392 const BondGraph * const BG = World::getInstance().getBondGraph();
393 const range<double> MinMaxBondDistance(
394 BG->getMinMaxDistance(Origin,Replacement));
395 if (!MinMaxBondDistance.isInRange(bondlength)) {
396// LOG(4, "InBondvector is: " << InBondvector << ".");
397 Orthovector1.Zero();
398 for (int i=NDIM;i--;) {
399 l = Replacement->at(i) - Origin->at(i);
400 if (fabs(l) > MinMaxBondDistance.last) { // is component greater than bond distance (check against min not useful here)
401 Orthovector1[i] = (l < 0) ? -1. : +1.;
402 } // (signs are correct, was tested!)
403 }
404 Orthovector1 *= matrix;
405 InBondvector -= Orthovector1; // subtract just the additional translation
406 bondlength = InBondvector.Norm();
407// LOG(4, "INFO: Corrected InBondvector is now: " << InBondvector << ".");
408 } // periodic correction finished
409
410 InBondvector.Normalize();
411 // get typical bond length and store as scale factor for later
412 ASSERT(Origin->getType() != NULL,
413 "SaturatedFragment::AddHydrogenReplacementAtom() - element of Origin is not given.");
414 BondRescale = Origin->getType()->getHBondDistance(TopBond->getDegree()-1);
415 if (BondRescale == -1) {
416 ELOG(1, "There is no typical hydrogen bond distance in replacing bond (" << Origin->getName() << "<->" << Replacement->getName() << ") of degree " << TopBond->getDegree() << "!");
417 BondRescale = Origin->getType()->getHBondDistance(TopBond->getDegree());
418 if (BondRescale == -1) {
419 ELOG(1, "There is no typical hydrogen bond distance in replacing bond (" << Origin->getName() << "<->" << Replacement->getName() << ") of any degree!");
420 return false;
421 BondRescale = bondlength;
422 }
423 } else {
424 if (!IsAngstroem)
425 BondRescale /= (1.*AtomicLengthToAngstroem);
426 }
427
428 // discern single, double and triple bonds
429 switch(TopBond->getDegree()) {
430 case 1:
431 // check whether replacement has been an excluded hydrogen
432 if (Replacement->getType()->getAtomicNumber() == HydrogenPool::HYDROGEN) { // neither rescale nor replace if it's already hydrogen
433 FirstOtherAtom = Replacement;
434 BondRescale = bondlength;
435 } else {
436 FirstOtherAtom = getHydrogenReplacement(Replacement);
437 InBondvector *= BondRescale; // rescale the distance vector to Hydrogen bond length
438 FirstOtherAtom->setPosition(Origin->getPosition() + InBondvector); // set coordination to origin and add distance vector to replacement atom
439 }
440 FullMolecule.insert(FirstOtherAtom->getId());
441// LOG(4, "INFO: Added " << *FirstOtherAtom << " at: " << FirstOtherAtom->x << ".");
442 break;
443 case 2:
444 {
445 // determine two other bonds (warning if there are more than two other) plus valence sanity check
446 const BondList& ListOfBonds = Origin->getListOfBonds();
447 for (BondList::const_iterator Runner = ListOfBonds.begin();
448 Runner != ListOfBonds.end();
449 ++Runner) {
450 if ((*Runner) != TopBond) {
451 if (FirstBond == NULL) {
452 FirstBond = (*Runner);
453 FirstOtherAtom = (*Runner)->GetOtherAtom(Origin);
454 } else if (SecondBond == NULL) {
455 SecondBond = (*Runner);
456 SecondOtherAtom = (*Runner)->GetOtherAtom(Origin);
457 } else {
458 ELOG(2, "Detected more than four bonds for atom " << Origin->getName());
459 }
460 }
461 }
462 }
463 if (SecondOtherAtom == NULL) { // then we have an atom with valence four, but only 3 bonds: one to replace and one which is TopBond (third is FirstBond)
464 SecondBond = TopBond;
465 SecondOtherAtom = Replacement;
466 }
467 if (FirstOtherAtom != NULL) { // then we just have this double bond and the plane does not matter at all
468// LOG(3, "Regarding the double bond (" << Origin->Name << "<->" << Replacement->Name << ") to be constructed: Taking " << FirstOtherAtom->Name << " and " << SecondOtherAtom->Name << " along with " << Origin->Name << " to determine orthogonal plane.");
469
470 // determine the plane of these two with the *origin
471 try {
472 Orthovector1 = Plane(Origin->getPosition(), FirstOtherAtom->getPosition(), SecondOtherAtom->getPosition()).getNormal();
473 }
474 catch(LinearDependenceException &excp){
475 LOG(0, boost::diagnostic_information(excp));
476 // TODO: figure out what to do with the Orthovector in this case
477 AllWentWell = false;
478 }
479 } else {
480 Orthovector1.GetOneNormalVector(InBondvector);
481 }
482 //LOG(3, "INFO: Orthovector1: " << Orthovector1 << ".");
483 // orthogonal vector and bond vector between origin and replacement form the new plane
484 Orthovector1.MakeNormalTo(InBondvector);
485 Orthovector1.Normalize();
486 //LOG(3, "ReScaleCheck: " << Orthovector1.Norm() << " and " << InBondvector.Norm() << ".");
487
488 // create the two Hydrogens ...
489 FirstOtherAtom = getHydrogenReplacement(Replacement);
490 SecondOtherAtom = getHydrogenReplacement(Replacement);
491 FullMolecule.insert(FirstOtherAtom->getId());
492 FullMolecule.insert(SecondOtherAtom->getId());
493 bondangle = Origin->getType()->getHBondAngle(1);
494 if (bondangle == -1) {
495 ELOG(1, "There is no typical hydrogen bond angle in replacing bond (" << Origin->getName() << "<->" << Replacement->getName() << ") of degree " << TopBond->getDegree() << "!");
496 return false;
497 bondangle = 0;
498 }
499 bondangle *= M_PI/180./2.;
500// LOG(3, "INFO: ReScaleCheck: InBondvector " << InBondvector << ", " << Orthovector1 << ".");
501// LOG(3, "Half the bond angle is " << bondangle << ", sin and cos of it: " << sin(bondangle) << ", " << cos(bondangle));
502 FirstOtherAtom->Zero();
503 SecondOtherAtom->Zero();
504 for(int i=NDIM;i--;) { // rotate by half the bond angle in both directions (InBondvector is bondangle = 0 direction)
505 FirstOtherAtom->set(i, InBondvector[i] * cos(bondangle) + Orthovector1[i] * (sin(bondangle)));
506 SecondOtherAtom->set(i, InBondvector[i] * cos(bondangle) + Orthovector1[i] * (-sin(bondangle)));
507 }
508 FirstOtherAtom->Scale(BondRescale); // rescale by correct BondDistance
509 SecondOtherAtom->Scale(BondRescale);
510 //LOG(3, "ReScaleCheck: " << FirstOtherAtom->x.Norm() << " and " << SecondOtherAtom->x.Norm() << ".");
511 *FirstOtherAtom += Origin->getPosition();
512 *SecondOtherAtom += Origin->getPosition();
513 // ... and add to molecule
514// LOG(4, "INFO: Added " << *FirstOtherAtom << " at: " << FirstOtherAtom->x << ".");
515// LOG(4, "INFO: Added " << *SecondOtherAtom << " at: " << SecondOtherAtom->x << ".");
516 break;
517 case 3:
518 // take the "usual" tetraoidal angle and add the three Hydrogen in direction of the bond (height of the tetraoid)
519 FirstOtherAtom = getHydrogenReplacement(Replacement);
520 SecondOtherAtom = getHydrogenReplacement(Replacement);
521 ThirdOtherAtom = getHydrogenReplacement(Replacement);
522 FullMolecule.insert(FirstOtherAtom->getId());
523 FullMolecule.insert(SecondOtherAtom->getId());
524 FullMolecule.insert(ThirdOtherAtom->getId());
525
526 // we need to vectors orthonormal the InBondvector
527 AllWentWell = AllWentWell && Orthovector1.GetOneNormalVector(InBondvector);
528// LOG(3, "INFO: Orthovector1: " << Orthovector1 << ".");
529 try{
530 Orthovector2 = Plane(InBondvector, Orthovector1,0).getNormal();
531 }
532 catch(LinearDependenceException &excp) {
533 LOG(0, boost::diagnostic_information(excp));
534 AllWentWell = false;
535 }
536// LOG(3, "INFO: Orthovector2: " << Orthovector2 << ".")
537
538 // create correct coordination for the three atoms
539 alpha = (Origin->getType()->getHBondAngle(2))/180.*M_PI/2.; // retrieve triple bond angle from database
540 l = BondRescale; // desired bond length
541 b = 2.*l*sin(alpha); // base length of isosceles triangle
542 d = l*sqrt(cos(alpha)*cos(alpha) - sin(alpha)*sin(alpha)/3.); // length for InBondvector
543 f = b/sqrt(3.); // length for Orthvector1
544 g = b/2.; // length for Orthvector2
545// LOG(3, "Bond length and half-angle: " << l << ", " << alpha << "\t (b,d,f,g) = " << b << ", " << d << ", " << f << ", " << g << ", ");
546// LOG(3, "The three Bond lengths: " << sqrt(d*d+f*f) << ", " << sqrt(d*d+(-0.5*f)*(-0.5*f)+g*g) << ", " << sqrt(d*d+(-0.5*f)*(-0.5*f)+g*g));
547 factors[0] = d;
548 factors[1] = f;
549 factors[2] = 0.;
550 FirstOtherAtom->LinearCombinationOfVectors(InBondvector, Orthovector1, Orthovector2, factors);
551 factors[1] = -0.5*f;
552 factors[2] = g;
553 SecondOtherAtom->LinearCombinationOfVectors(InBondvector, Orthovector1, Orthovector2, factors);
554 factors[2] = -g;
555 ThirdOtherAtom->LinearCombinationOfVectors(InBondvector, Orthovector1, Orthovector2, factors);
556
557 // rescale each to correct BondDistance
558// FirstOtherAtom->x.Scale(&BondRescale);
559// SecondOtherAtom->x.Scale(&BondRescale);
560// ThirdOtherAtom->x.Scale(&BondRescale);
561
562 // and relative to *origin atom
563 *FirstOtherAtom += Origin->getPosition();
564 *SecondOtherAtom += Origin->getPosition();
565 *ThirdOtherAtom += Origin->getPosition();
566
567 // ... and add to molecule
568// LOG(4, "INFO: Added " << *FirstOtherAtom << " at: " << FirstOtherAtom->x << ".");
569// LOG(4, "INFO: Added " << *SecondOtherAtom << " at: " << SecondOtherAtom->x << ".");
570// LOG(4, "INFO: Added " << *ThirdOtherAtom << " at: " << ThirdOtherAtom->x << ".");
571 break;
572 default:
573 ELOG(1, "BondDegree does not state single, double or triple bond!");
574 AllWentWell = false;
575 break;
576 }
577
578 return AllWentWell;
579};
Note: See TracBrowser for help on using the repository browser.