- Timestamp:
- Apr 1, 2010, 6:52:09 PM (16 years ago)
- 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, 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
- Children:
- 734816
- Parents:
- 388049
- Location:
- src
- Files:
- 
      - 2 edited
 
 - 
          
  analysis_bonds.cpp (modified) (3 diffs)
- 
          
  unittests/CountBondsUnitTest.cpp (modified) (7 diffs)
 
Legend:
- Unmodified
- Added
- Removed
- 
      src/analysis_bonds.cppr388049 rfe238c 82 82 }; 83 83 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 */ 90 double 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 */ 110 bool 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 }; 84 121 85 122 /** Counts the number of hydrogen bridge bonds. … … 91 128 int CountHydrogenBridgeBonds(MoleculeListClass *molecules, element * InterfaceElement = NULL) 92 129 { 93 Info FunctionInfo(__func__);94 130 atom *Walker = NULL; 95 131 atom *Runner = NULL; 96 Vector OHBond;97 Vector OOBond;98 132 int count = 0; 133 int OtherHydrogens = 0; 134 double Otherangle = 0.; 99 135 bool InterfaceFlag = false; 136 bool OtherHydrogenFlag = true; 100 137 for (MoleculeList::const_iterator MolWalker = molecules->ListOfMolecules.begin();MolWalker != molecules->ListOfMolecules.end(); MolWalker++) { 101 138 Walker = (*MolWalker)->start; 102 139 while (Walker->next != (*MolWalker)->end) { 103 140 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++) { 105 142 Runner = (*MolRunner)->start; 106 143 while (Runner->next != (*MolRunner)->end) { 107 144 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)) { 109 146 // check distance 110 147 const double distance = Runner->x.DistanceSquared(&Walker->x); 111 148 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; 112 154 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++) { 115 156 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 } 116 164 InterfaceFlag = InterfaceFlag || (OtherAtom->type == InterfaceElement); 117 165 } 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) { 119 179 // on this element (Walker) we check for bond to hydrogen, i.e. part of water molecule 120 180 for (BondList::const_iterator BondRunner = Walker->ListOfBonds.begin(); BondRunner != Walker->ListOfBonds.end(); BondRunner++) { … … 122 182 if (OtherAtom->type->Z == 1) { 123 183 // 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); 131 186 count++; 132 187 break; 
- 
      src/unittests/CountBondsUnitTest.cppr388049 rfe238c 159 159 //OutputTestMolecule(TestMolecule1, "testmolecule1.xyz"); 160 160 161 // offset of (3,0,0), hence angles are (104.5, 0, 75.5, 180) < 30161 cout << "Case 1: offset of (3,0,0), hence angles are (104.5, 0, 75.5, 180) < 30." << endl; 162 162 Translator.Init(3,0,0); 163 163 TestMolecule2->Translate(&Translator); … … 168 168 TestMolecule2->Translate(&Translator); 169 169 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; 171 171 Translator.Init(0,3,0); 172 172 TestMolecule2->Translate(&Translator); … … 176 176 TestMolecule2->Translate(&Translator); 177 177 178 // offset of (0,-3,0) and mirror, hence angle are (165.5, 90, 165.5, 90) > 30178 cout << "Case 3: offset of (0,-3,0) and mirror, hence angle are (165.5, 90, 165.5, 90) > 30." << endl; 179 179 Translator.Init(0,-3,0); 180 180 TestMolecule2->Scale((const double ** const)&mirror); … … 186 186 TestMolecule2->Scale((const double ** const)&mirror); 187 187 188 // offset of (2,1,0), hence angle are (78, 26.6, 102, 153.4) < 30188 cout << "Case 4: offset of (2,1,0), hence angle are (78, 26.6, 102, 153.4) < 30." << endl; 189 189 Translator.Init(2,1,0); 190 190 TestMolecule2->Translate(&Translator); … … 194 194 TestMolecule2->Translate(&Translator); 195 195 196 // offset of (0,0,3), hence angle are (90, 90, 90, 90) > 30196 cout << "Case 5: offset of (0,0,3), hence angle are (90, 90, 90, 90) > 30." << endl; 197 197 Translator.Init(0,0,3); 198 198 TestMolecule2->Translate(&Translator); … … 202 202 TestMolecule2->Translate(&Translator); 203 203 204 // offset of (-3,0,0) and mirror, hence angle are (75.5, 180, 104.5, 180) > 30204 cout << "Case 6: offset of (-3,0,0) and mirror, hence angle are (75.5, 180, 104.5, 180) > 30." << endl; 205 205 Translator.Init(-3,0,0); 206 206 TestMolecule2->Scale((const double ** const)&mirror); … … 211 211 TestMolecule2->Translate(&Translator); 212 212 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 213 234 delete(mirror); 214 235 }; 
  Note:
 See   TracChangeset
 for help on using the changeset viewer.
  
