Changeset fe238c


Ignore:
Timestamp:
Apr 1, 2010, 6:52:09 PM (15 years ago)
Author:
Frederik Heber <heber@…>
Branches:
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, 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
Children:
734816
Parents:
388049
Message:

Enhanced CountHydrogenBridgeBonds() a bit.

Location:
src
Files:
2 edited

Legend:

Unmodified
Added
Removed
  • src/analysis_bonds.cpp

    r388049 rfe238c  
    8282};
    8383
     84/** Calculate the angle between \a *first and \a *origin and \a *second and \a *origin.
     85 * \param *first first Vector
     86 * \param *origin origin of angle taking
     87 * \param *second second Vector
     88 * \return angle between \a *first and \a *second, both relative to origin at \a *origin.
     89 */
     90double CalculateAngle(Vector *first, Vector *central, Vector *second)
     91{
     92  Vector OHBond;
     93  Vector OOBond;
     94
     95  OHBond.CopyVector(first);
     96  OHBond.SubtractVector(central);
     97  OOBond.CopyVector(second);
     98  OOBond.SubtractVector(central);
     99  const double angle = OHBond.Angle(&OOBond);
     100  return angle;
     101};
     102
     103/** Checks whether the angle between \a *Oxygen and \a *Hydrogen and \a *Oxygen and \a *OtherOxygen is less than 30 degrees.
     104 * Note that distance criterion is not checked.
     105 * \param *Oxygen first oxygen atom, bonded to \a *Hydrogen
     106 * \param *Hydrogen hydrogen bonded to \a *Oxygen
     107 * \param *OtherOxygen other oxygen atom
     108 * \return true - angle criteria fulfilled, false - criteria not fulfilled, angle greater than 30 degrees.
     109 */
     110bool CheckHydrogenBridgeBondAngle(atom *Oxygen, atom *Hydrogen, atom *OtherOxygen)
     111{
     112  Info FunctionInfo(__func__);
     113
     114  // check angle
     115  if (CalculateAngle(&Hydrogen->x, &Oxygen->x, &OtherOxygen->x) < M_PI*(30./180.)) {
     116    return true;
     117  } else {
     118    return false;
     119  }
     120};
    84121
    85122/** Counts the number of hydrogen bridge bonds.
     
    91128int CountHydrogenBridgeBonds(MoleculeListClass *molecules, element * InterfaceElement = NULL)
    92129{
    93   Info FunctionInfo(__func__);
    94130  atom *Walker = NULL;
    95131  atom *Runner = NULL;
    96   Vector OHBond;
    97   Vector OOBond;
    98132  int count = 0;
     133  int OtherHydrogens = 0;
     134  double Otherangle = 0.;
    99135  bool InterfaceFlag = false;
     136  bool OtherHydrogenFlag = true;
    100137  for (MoleculeList::const_iterator MolWalker = molecules->ListOfMolecules.begin();MolWalker != molecules->ListOfMolecules.end(); MolWalker++) {
    101138    Walker = (*MolWalker)->start;
    102139    while (Walker->next != (*MolWalker)->end) {
    103140      Walker = Walker->next;
    104       for (MoleculeList::const_iterator MolRunner = molecules->ListOfMolecules.begin();MolRunner != MolWalker; MolRunner++) {
     141      for (MoleculeList::const_iterator MolRunner = molecules->ListOfMolecules.begin();MolRunner != molecules->ListOfMolecules.end(); MolRunner++) {
    105142        Runner = (*MolRunner)->start;
    106143        while (Runner->next != (*MolRunner)->end) {
    107144          Runner = Runner->next;
    108           if ((Runner != Walker) && (Walker->type->Z  == 8) && (Runner->type->Z  == 8)) {
     145          if ((Walker->type->Z  == 8) && (Runner->type->Z  == 8)) {
    109146            // check distance
    110147            const double distance = Runner->x.DistanceSquared(&Walker->x);
    111148            if ((distance > MYEPSILON) && (distance < HBRIDGEDISTANCE*HBRIDGEDISTANCE)) { // distance >0 means  different atoms
     149              // on other atom(Runner) we check for bond to interface element and
     150              // check that O-O line is not in between the shanks of the two connected hydrogens (Otherangle > 104.5)
     151              OtherHydrogenFlag = true;
     152              Otherangle = 0.;
     153              OtherHydrogens = 0;
    112154              InterfaceFlag = (InterfaceElement == NULL);
    113               // on other atom(Runner) we check for bond to interface element
    114               for (BondList::const_iterator BondRunner = Runner->ListOfBonds.begin(); ((!InterfaceFlag) && (BondRunner != Runner->ListOfBonds.end())); BondRunner++) {
     155              for (BondList::const_iterator BondRunner = Runner->ListOfBonds.begin(); BondRunner != Runner->ListOfBonds.end(); BondRunner++) {
    115156                atom * const OtherAtom = (*BondRunner)->GetOtherAtom(Runner);
     157                // if hydrogen, check angle to be greater(!) than 30 degrees
     158                if (OtherAtom->type->Z == 1) {
     159                  const double angle = CalculateAngle(&OtherAtom->x, &Runner->x, &Walker->x);
     160                  OtherHydrogenFlag = OtherHydrogenFlag && (angle > M_PI*(30./180.) + MYEPSILON);
     161                  Otherangle += angle;
     162                  OtherHydrogens++;
     163                }
    116164                InterfaceFlag = InterfaceFlag || (OtherAtom->type == InterfaceElement);
    117165              }
    118               if (InterfaceFlag) {
     166              DoLog(1) && (Log() << Verbose(1) << "Otherangle is " << Otherangle << " for " << OtherHydrogens << " hydrogens." << endl);
     167              switch (OtherHydrogens) {
     168                case 0:
     169                case 1:
     170                  break;
     171                case 2:
     172                  OtherHydrogenFlag = OtherHydrogenFlag && (Otherangle > M_PI*(104.5/180.) + MYEPSILON);
     173                  break;
     174                default: // 3 or more hydrogens ...
     175                  OtherHydrogenFlag = false;
     176                  break;
     177              }
     178              if (InterfaceFlag && OtherHydrogenFlag) {
    119179                // on this element (Walker) we check for bond to hydrogen, i.e. part of water molecule
    120180                for (BondList::const_iterator BondRunner = Walker->ListOfBonds.begin(); BondRunner != Walker->ListOfBonds.end(); BondRunner++) {
     
    122182                  if (OtherAtom->type->Z == 1) {
    123183                    // check angle
    124                     OHBond.CopyVector(&OtherAtom->x);
    125                     OHBond.SubtractVector(&Walker->x);
    126                     OOBond.CopyVector(&Runner->x);
    127                     OOBond.SubtractVector(&Walker->x);
    128                     const double angle = OHBond.Angle(&OOBond);
    129                     if (angle < M_PI*(30./180.)) {
    130                       DoLog(1) && (Log() << Verbose(1) << Walker->Name << ", " << OtherAtom->Name << " and " << Runner->Name << " have a hydrogen bridge bond with " << sqrt(distance) << " and at angle " << (180./M_PI)*angle << " degrees." << endl);
     184                    if (CheckHydrogenBridgeBondAngle(Walker, OtherAtom, Runner)) {
     185                      DoLog(1) && (Log() << Verbose(1) << Walker->Name << ", " << OtherAtom->Name << " and " << Runner->Name << " has a hydrogen bridge bond with distance " << sqrt(distance) << " and angle " << CalculateAngle(&OtherAtom->x, &Walker->x, &Runner->x)*(180./M_PI) << "." << endl);
    131186                      count++;
    132187                      break;
  • src/unittests/CountBondsUnitTest.cpp

    r388049 rfe238c  
    159159  //OutputTestMolecule(TestMolecule1, "testmolecule1.xyz");
    160160
    161   // offset of (3,0,0), hence angles are (104.5, 0, 75.5, 180) < 30
     161  cout << "Case 1: offset of (3,0,0), hence angles are (104.5, 0, 75.5, 180) < 30." << endl;
    162162  Translator.Init(3,0,0);
    163163  TestMolecule2->Translate(&Translator);
     
    168168  TestMolecule2->Translate(&Translator);
    169169
    170   // offset of (0,3,0), hence angle are (14.5, 165.5, 90) < 30 (only three, because other 90 is missing due to first H01 only fulfilling H-bond criteria)
     170  cout << "Case 2: offset of (0,3,0), hence angle are (14.5, 165.5, 90) < 30 (only three, because other 90 is missing due to first H01 only fulfilling H-bond criteria)." << endl;
    171171  Translator.Init(0,3,0);
    172172  TestMolecule2->Translate(&Translator);
     
    176176  TestMolecule2->Translate(&Translator);
    177177
    178   // offset of (0,-3,0) and mirror, hence angle are (165.5, 90, 165.5, 90) > 30
     178  cout << "Case 3: offset of (0,-3,0) and mirror, hence angle are (165.5, 90, 165.5, 90) > 30." << endl;
    179179  Translator.Init(0,-3,0);
    180180  TestMolecule2->Scale((const double ** const)&mirror);
     
    186186  TestMolecule2->Scale((const double ** const)&mirror);
    187187
    188   // offset of (2,1,0), hence angle are (78, 26.6, 102, 153.4) < 30
     188  cout << "Case 4: offset of (2,1,0), hence angle are (78, 26.6, 102, 153.4) < 30." << endl;
    189189  Translator.Init(2,1,0);
    190190  TestMolecule2->Translate(&Translator);
     
    194194  TestMolecule2->Translate(&Translator);
    195195
    196   // offset of (0,0,3), hence angle are (90, 90, 90, 90) > 30
     196  cout << "Case 5: offset of (0,0,3), hence angle are (90, 90, 90, 90) > 30." << endl;
    197197  Translator.Init(0,0,3);
    198198  TestMolecule2->Translate(&Translator);
     
    202202  TestMolecule2->Translate(&Translator);
    203203
    204   // offset of (-3,0,0) and mirror, hence angle are (75.5, 180, 104.5, 180) > 30
     204  cout << "Case 6: offset of (-3,0,0) and mirror, hence angle are (75.5, 180, 104.5, 180) > 30." << endl;
    205205  Translator.Init(-3,0,0);
    206206  TestMolecule2->Scale((const double ** const)&mirror);
     
    211211  TestMolecule2->Translate(&Translator);
    212212  TestMolecule2->Scale((const double ** const)&mirror);
     213
     214  cout << "Case 7: offset of (3,0,0) and mirror, hence angles are (104.5, 0, 104.5, 0) < 30, but interfering hydrogens." << endl;
     215  Translator.Init(3,0,0);
     216  TestMolecule2->Scale((const double ** const)&mirror);
     217  TestMolecule2->Translate(&Translator);
     218  CPPUNIT_ASSERT_EQUAL( 0 , CountHydrogenBridgeBonds(molecules, NULL) );
     219  //OutputTestMolecule(TestMolecule2, "testmolecule2-7.xyz");
     220  Translator.Init(-3,0,0);
     221  TestMolecule2->Translate(&Translator);
     222  TestMolecule2->Scale((const double ** const)&mirror);
     223
     224  cout << "Case 8: offset of (0,3,0), hence angle are (14.5, 90, 14.5, 90) < 30, but interfering hydrogens." << endl;
     225  Translator.Init(0,3,0);
     226  TestMolecule2->Scale((const double ** const)&mirror);
     227  TestMolecule2->Translate(&Translator);
     228  //OutputTestMolecule(TestMolecule2, "testmolecule2-8.xyz");
     229  CPPUNIT_ASSERT_EQUAL( 0 , CountHydrogenBridgeBonds(molecules, NULL) );
     230  Translator.Init(0,-3,0);
     231  TestMolecule2->Translate(&Translator);
     232  TestMolecule2->Scale((const double ** const)&mirror);
     233
    213234  delete(mirror);
    214235};
Note: See TracChangeset for help on using the changeset viewer.