
org.openscience.cdk.tools.SaturationChecker Maven / Gradle / Ivy
/* Copyright (C) 2001-2007 The Chemistry Development Kit (CDK) project
*
* Contact: [email protected]
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public License
* as published by the Free Software Foundation; either version 2.1
* of the License, or (at your option) any later version.
* All we ask is that proper credit is given for our work, which includes
* - but is not limited to - adding the above copyright notice to the beginning
* of your source code files, and to any copyright notice that you may distribute
* with programs based on this work.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
*/
package org.openscience.cdk.tools;
import java.util.ArrayList;
import java.util.List;
import org.openscience.cdk.CDKConstants;
import org.openscience.cdk.config.AtomTypeFactory;
import org.openscience.cdk.exception.CDKException;
import org.openscience.cdk.graph.Cycles;
import org.openscience.cdk.interfaces.IAtom;
import org.openscience.cdk.interfaces.IAtomContainer;
import org.openscience.cdk.interfaces.IAtomType;
import org.openscience.cdk.interfaces.IBond;
import org.openscience.cdk.interfaces.IBond.Order;
import org.openscience.cdk.interfaces.IChemObjectBuilder;
import org.openscience.cdk.interfaces.IPseudoAtom;
import org.openscience.cdk.interfaces.IRingSet;
import org.openscience.cdk.ringsearch.RingPartitioner;
import org.openscience.cdk.tools.manipulator.BondManipulator;
import org.openscience.cdk.tools.manipulator.RingSetManipulator;
/**
* Provides methods for checking whether an atoms valences are saturated with
* respect to a particular atom type.
*
* Important: this class does not deal with hybridization states, which makes
* it fail, for example, for situations where bonds are marked as aromatic (either
* 1.5 or single an AROMATIC).
*
* @author steinbeck
* @author Egon Willighagen
* @cdk.created 2001-09-04
*
* @cdk.keyword saturation
* @cdk.keyword atom, valency
*
* @cdk.module valencycheck
* @cdk.githash
*/
public class SaturationChecker implements IValencyChecker, IDeduceBondOrderTool {
AtomTypeFactory structgenATF;
private static ILoggingTool logger = LoggingToolFactory.createLoggingTool(SaturationChecker.class);
/**
* @param builder the ChemObjectBuilder implementation used to construct the AtomType's.
*/
protected AtomTypeFactory getAtomTypeFactory(IChemObjectBuilder builder) throws CDKException {
if (structgenATF == null) {
try {
structgenATF = AtomTypeFactory.getInstance("org/openscience/cdk/config/data/structgen_atomtypes.xml",
builder);
} catch (Exception exception) {
logger.debug(exception);
throw new CDKException("Could not instantiate AtomTypeFactory!", exception);
}
}
return structgenATF;
}
public boolean hasPerfectConfiguration(IAtom atom, IAtomContainer ac) throws CDKException {
double bondOrderSum = ac.getBondOrderSum(atom);
IBond.Order maxBondOrder = ac.getMaximumBondOrder(atom);
IAtomType[] atomTypes = getAtomTypeFactory(atom.getBuilder()).getAtomTypes(atom.getSymbol());
if (atomTypes.length == 0) return true;
logger.debug("*** Checking for perfect configuration ***");
try {
logger.debug("Checking configuration of atom " + ac.getAtomNumber(atom));
logger.debug("Atom has bondOrderSum = " + bondOrderSum);
logger.debug("Atom has max = " + bondOrderSum);
} catch (Exception exc) {
}
for (int f = 0; f < atomTypes.length; f++) {
if (bondOrderSum == atomTypes[f].getBondOrderSum() && maxBondOrder == atomTypes[f].getMaxBondOrder()) {
try {
logger.debug("Atom " + ac.getAtomNumber(atom) + " has perfect configuration");
} catch (Exception exc) {
}
return true;
}
}
try {
logger.debug("*** Atom " + ac.getAtomNumber(atom) + " has imperfect configuration ***");
} catch (Exception exc) {
}
return false;
}
/**
* Determines of all atoms on the AtomContainer are saturated.
*/
@Override
public boolean isSaturated(IAtomContainer container) throws CDKException {
return allSaturated(container);
}
public boolean allSaturated(IAtomContainer ac) throws CDKException {
logger.debug("Are all atoms saturated?");
for (int f = 0; f < ac.getAtomCount(); f++) {
if (!isSaturated(ac.getAtom(f), ac)) {
return false;
}
}
return true;
}
/**
* Returns whether a bond is unsaturated. A bond is unsaturated if
* both Atoms in the bond are unsaturated.
*/
public boolean isUnsaturated(IBond bond, IAtomContainer atomContainer) throws CDKException {
IAtom[] atoms = BondManipulator.getAtomArray(bond);
boolean isUnsaturated = true;
for (int i = 0; i < atoms.length; i++) {
isUnsaturated = isUnsaturated && !isSaturated(atoms[i], atomContainer);
}
return isUnsaturated;
}
/**
* Returns whether a bond is saturated. A bond is saturated if
* both Atoms in the bond are saturated.
*/
public boolean isSaturated(IBond bond, IAtomContainer atomContainer) throws CDKException {
IAtom[] atoms = BondManipulator.getAtomArray(bond);
boolean isSaturated = true;
for (int i = 0; i < atoms.length; i++) {
isSaturated = isSaturated && isSaturated(atoms[i], atomContainer);
}
return isSaturated;
}
/**
* Checks whether an Atom is saturated by comparing it with known AtomTypes.
*/
@Override
public boolean isSaturated(IAtom atom, IAtomContainer ac) throws CDKException {
IAtomType[] atomTypes = getAtomTypeFactory(atom.getBuilder()).getAtomTypes(atom.getSymbol());
if (atomTypes.length == 0) return true;
double bondOrderSum = ac.getBondOrderSum(atom);
IBond.Order maxBondOrder = ac.getMaximumBondOrder(atom);
Integer hcount = atom.getImplicitHydrogenCount() == CDKConstants.UNSET ? 0 : atom.getImplicitHydrogenCount();
Integer charge = atom.getFormalCharge() == CDKConstants.UNSET ? 0 : atom.getFormalCharge();
try {
logger.debug("*** Checking saturation of atom ", atom.getSymbol(), "" + ac.getAtomNumber(atom) + " ***");
logger.debug("bondOrderSum: " + bondOrderSum);
logger.debug("maxBondOrder: " + maxBondOrder);
logger.debug("hcount: " + hcount);
} catch (Exception exc) {
logger.debug(exc);
}
for (int f = 0; f < atomTypes.length; f++) {
if (bondOrderSum - charge + hcount == atomTypes[f].getBondOrderSum()
&& !BondManipulator.isHigherOrder(maxBondOrder, atomTypes[f].getMaxBondOrder())) {
logger.debug("*** Good ! ***");
return true;
}
}
logger.debug("*** Bad ! ***");
return false;
}
/**
* Checks if the current atom has exceeded its bond order sum value.
*
* @param atom The Atom to check
* @param ac The atomcontainer context
* @return oversaturated or not
*/
public boolean isOverSaturated(IAtom atom, IAtomContainer ac) throws CDKException {
IAtomType[] atomTypes = getAtomTypeFactory(atom.getBuilder()).getAtomTypes(atom.getSymbol());
if (atomTypes.length == 0) return false;
double bondOrderSum = ac.getBondOrderSum(atom);
IBond.Order maxBondOrder = ac.getMaximumBondOrder(atom);
Integer hcount = atom.getImplicitHydrogenCount() == CDKConstants.UNSET ? 0 : atom.getImplicitHydrogenCount();
Integer charge = atom.getFormalCharge() == CDKConstants.UNSET ? 0 : atom.getFormalCharge();
try {
logger.debug("*** Checking saturation of atom " + ac.getAtomNumber(atom) + " ***");
logger.debug("bondOrderSum: " + bondOrderSum);
logger.debug("maxBondOrder: " + maxBondOrder);
logger.debug("hcount: " + hcount);
} catch (Exception exc) {
}
for (int f = 0; f < atomTypes.length; f++) {
if (bondOrderSum - charge + hcount > atomTypes[f].getBondOrderSum()) {
logger.debug("*** Good ! ***");
return true;
}
}
logger.debug("*** Bad ! ***");
return false;
}
/**
* Returns the currently maximum formable bond order for this atom.
*
* @param atom The atom to be checked
* @param ac The AtomContainer that provides the context
* @return the currently maximum formable bond order for this atom
*/
public double getCurrentMaxBondOrder(IAtom atom, IAtomContainer ac) throws CDKException {
IAtomType[] atomTypes = getAtomTypeFactory(atom.getBuilder()).getAtomTypes(atom.getSymbol());
if (atomTypes.length == 0) return 0;
double bondOrderSum = ac.getBondOrderSum(atom);
Integer hcount = atom.getImplicitHydrogenCount() == CDKConstants.UNSET ? 0 : atom.getImplicitHydrogenCount();
double max = 0;
double current = 0;
for (int f = 0; f < atomTypes.length; f++) {
current = hcount + bondOrderSum;
if (atomTypes[f].getBondOrderSum() - current > max) {
max = atomTypes[f].getBondOrderSum() - current;
}
}
return max;
}
/**
* Resets the bond orders of all atoms to 1.0.
*/
public void unsaturate(IAtomContainer atomContainer) {
for (IBond bond : atomContainer.bonds())
bond.setOrder(Order.SINGLE);
}
/**
* Resets the bond order of the Bond to 1.0.
*/
public void unsaturateBonds(IAtomContainer container) {
for (IBond bond : container.bonds())
bond.setOrder(Order.SINGLE);
}
/**
* Saturates a molecule by setting appropriate bond orders.
* This method is known to fail, especially on pyrolle-like compounds.
* Consider using import org.openscience.cdk.smiles.DeduceBondSystemTool, which should work better
*
* @cdk.keyword bond order, calculation
* @cdk.created 2003-10-03
*/
public void newSaturate(IAtomContainer atomContainer) throws CDKException {
logger.info("Saturating atomContainer by adjusting bond orders...");
boolean allSaturated = allSaturated(atomContainer);
if (!allSaturated) {
IBond[] bonds = new IBond[atomContainer.getBondCount()];
for (int i = 0; i < bonds.length; i++)
bonds[i] = atomContainer.getBond(i);
boolean succeeded = newSaturate(bonds, atomContainer);
for (int i = 0; i < bonds.length; i++) {
if (bonds[i].getOrder() == IBond.Order.DOUBLE && bonds[i].getFlag(CDKConstants.ISAROMATIC)
&& (bonds[i].getAtom(0).getSymbol().equals("N") && bonds[i].getAtom(1).getSymbol().equals("N"))) {
int atomtohandle = 0;
if (bonds[i].getAtom(0).getSymbol().equals("N")) atomtohandle = 1;
List bondstohandle = atomContainer.getConnectedBondsList(bonds[i].getAtom(atomtohandle));
for (int k = 0; k < bondstohandle.size(); k++) {
IBond bond = bondstohandle.get(k);
if (bond.getOrder() == IBond.Order.SINGLE && bond.getFlag(CDKConstants.ISAROMATIC)) {
bond.setOrder(IBond.Order.DOUBLE);
bonds[i].setOrder(IBond.Order.SINGLE);
break;
}
}
}
}
if (!succeeded) {
throw new CDKException("Could not saturate this atomContainer!");
}
}
}
/**
* Saturates a set of Bonds in an AtomContainer.
* This method is known to fail, especially on pyrolle-like compounds.
* Consider using import org.openscience.cdk.smiles.DeduceBondSystemTool, which should work better
*/
public boolean newSaturate(IBond[] bonds, IAtomContainer atomContainer) throws CDKException {
logger.debug("Saturating bond set of size: " + bonds.length);
boolean bondsAreFullySaturated = true;
if (bonds.length > 0) {
IBond bond = bonds[0];
// determine bonds left
int leftBondCount = bonds.length - 1;
IBond[] leftBonds = new IBond[leftBondCount];
System.arraycopy(bonds, 1, leftBonds, 0, leftBondCount);
// examine this bond
if (isUnsaturated(bond, atomContainer)) {
// either this bonds should be saturated or not
// try to leave this bond unsaturated and saturate the left bondssaturate this bond
if (leftBondCount > 0) {
logger.debug("Recursing with unsaturated bond with #bonds: " + leftBondCount);
bondsAreFullySaturated = newSaturate(leftBonds, atomContainer)
&& !isUnsaturated(bond, atomContainer);
} else {
bondsAreFullySaturated = false;
}
// ok, did it work? if not, saturate this bond, and recurse
if (!bondsAreFullySaturated) {
logger.debug("First try did not work...");
// ok, revert saturating this bond, and recurse again
boolean couldSaturate = newSaturate(bond, atomContainer);
if (couldSaturate) {
if (leftBondCount > 0) {
logger.debug("Recursing with saturated bond with #bonds: " + leftBondCount);
bondsAreFullySaturated = newSaturate(leftBonds, atomContainer);
} else {
bondsAreFullySaturated = true;
}
} else {
bondsAreFullySaturated = false;
// no need to recurse, because we already know that this bond
// unsaturated does not work
}
}
} else if (isSaturated(bond, atomContainer)) {
logger.debug("This bond is already saturated.");
if (leftBondCount > 0) {
logger.debug("Recursing with #bonds: " + leftBondCount);
bondsAreFullySaturated = newSaturate(leftBonds, atomContainer);
} else {
bondsAreFullySaturated = true;
}
} else {
logger.debug("Cannot saturate this bond");
// but, still recurse (if possible)
if (leftBondCount > 0) {
logger.debug("Recursing with saturated bond with #bonds: " + leftBondCount);
bondsAreFullySaturated = newSaturate(leftBonds, atomContainer)
&& !isUnsaturated(bond, atomContainer);
} else {
bondsAreFullySaturated = !isUnsaturated(bond, atomContainer);
}
}
}
logger.debug("Is bond set fully saturated?: " + bondsAreFullySaturated);
logger.debug("Returning to level: " + (bonds.length + 1));
return bondsAreFullySaturated;
}
/**
* Saturate atom by adjusting its bond orders.
* This method is known to fail, especially on pyrolle-like compounds.
* Consider using import org.openscience.cdk.smiles.DeduceBondSystemTool, which should work better
*/
public boolean newSaturate(IBond bond, IAtomContainer atomContainer) throws CDKException {
IAtom[] atoms = BondManipulator.getAtomArray(bond);
IAtom atom = atoms[0];
IAtom partner = atoms[1];
logger.debug(" saturating bond: ", atom.getSymbol(), "-", partner.getSymbol());
IAtomType[] atomTypes1 = getAtomTypeFactory(bond.getBuilder()).getAtomTypes(atom.getSymbol());
IAtomType[] atomTypes2 = getAtomTypeFactory(bond.getBuilder()).getAtomTypes(partner.getSymbol());
boolean bondOrderIncreased = true;
while (bondOrderIncreased && !isSaturated(bond, atomContainer)) {
logger.debug("Can increase bond order");
bondOrderIncreased = false;
for (int atCounter1 = 0; atCounter1 < atomTypes1.length && !bondOrderIncreased; atCounter1++) {
IAtomType aType1 = atomTypes1[atCounter1];
logger.debug(" condidering atom type: ", aType1);
if (couldMatchAtomType(atomContainer, atom, aType1)) {
logger.debug(" trying atom type: ", aType1);
for (int atCounter2 = 0; atCounter2 < atomTypes2.length && !bondOrderIncreased; atCounter2++) {
IAtomType aType2 = atomTypes2[atCounter2];
logger.debug(" condidering partner type: ", aType1);
if (couldMatchAtomType(atomContainer, partner, atomTypes2[atCounter2])) {
logger.debug(" with atom type: ", aType2);
if (!BondManipulator.isLowerOrder(bond.getOrder(), aType2.getMaxBondOrder())
|| !BondManipulator.isLowerOrder(bond.getOrder(), aType1.getMaxBondOrder())) {
logger.debug("Bond order not increased: atoms has reached (or exceeded) maximum bond order for this atom type");
} else if (BondManipulator.isLowerOrder(bond.getOrder(), aType2.getMaxBondOrder())
&& BondManipulator.isLowerOrder(bond.getOrder(), aType1.getMaxBondOrder())) {
BondManipulator.increaseBondOrder(bond);
logger.debug("Bond order now " + bond.getOrder());
bondOrderIncreased = true;
}
}
}
}
}
}
return isSaturated(bond, atomContainer);
}
/**
* Determines if the atom can be of type AtomType.
*/
public boolean couldMatchAtomType(IAtomContainer atomContainer, IAtom atom, IAtomType atomType) {
logger.debug(" ... matching atom ", atom.getSymbol(), " vs ", atomType);
if (atomContainer.getBondOrderSum(atom) + atom.getImplicitHydrogenCount() < atomType.getBondOrderSum()) {
logger.debug(" Match!");
return true;
}
logger.debug(" No Match");
return false;
}
/**
* The method is known to fail for certain compounds. For more information, see
* cdk.test.limitations package.
* This method is known to fail, especially on pyrolle-like compounds.
* Consider using import org.openscience.cdk.smiles.DeduceBondSystemTool, which should work better
*
*/
@Override
public void saturate(IAtomContainer atomContainer) throws CDKException {
/*
* newSaturate(atomContainer); } public void oldSaturate(AtomContainer
* atomContainer) throws CDKException {
*/
IAtom partner = null;
IAtom atom = null;
List partners = null;
IAtomType[] atomTypes1 = null;
IAtomType[] atomTypes2 = null;
IBond bond = null;
for (int i = 1; i < 4; i++) {
// handle atoms with degree 1 first and then proceed to higher order
for (int f = 0; f < atomContainer.getAtomCount(); f++) {
atom = atomContainer.getAtom(f);
logger.debug("symbol: ", atom.getSymbol());
atomTypes1 = getAtomTypeFactory(atom.getBuilder()).getAtomTypes(atom.getSymbol());
if (atomTypes1.length > 0) {
logger.debug("first atom type: ", atomTypes1[0]);
if (atomContainer.getConnectedBondsCount(atom) == i) {
Integer hcount = atom.getImplicitHydrogenCount() == CDKConstants.UNSET ? 0 : atom
.getImplicitHydrogenCount();
if (atom.getFlag(CDKConstants.ISAROMATIC)
&& atomContainer.getBondOrderSum(atom) < atomTypes1[0].getBondOrderSum() - hcount) {
partners = atomContainer.getConnectedAtomsList(atom);
for (int g = 0; g < partners.size(); g++) {
partner = partners.get(g);
logger.debug("Atom has " + partners.size() + " partners");
atomTypes2 = getAtomTypeFactory(atom.getBuilder()).getAtomTypes(partner.getSymbol());
if (atomTypes2.length == 0) return;
hcount = partner.getImplicitHydrogenCount() == CDKConstants.UNSET ? 0 : partner
.getImplicitHydrogenCount();
if (atomContainer.getBond(partner, atom).getFlag(CDKConstants.ISAROMATIC)
&& atomContainer.getBondOrderSum(partner) < atomTypes2[0].getBondOrderSum()
- hcount) {
logger.debug("Partner has " + atomContainer.getBondOrderSum(partner)
+ ", may have: " + atomTypes2[0].getBondOrderSum());
bond = atomContainer.getBond(atom, partner);
logger.debug("Bond order was " + bond.getOrder());
BondManipulator.increaseBondOrder(bond);
logger.debug("Bond order now " + bond.getOrder());
break;
}
}
}
Double bondOrderSum = atomTypes1[0].getBondOrderSum() == CDKConstants.UNSET ? 0.0
: atomTypes1[0].getBondOrderSum();
Integer hydrogenCount = atom.getImplicitHydrogenCount() == CDKConstants.UNSET ? 0 : atom
.getImplicitHydrogenCount();
Double atomContainerBondOrderSum = atomContainer.getBondOrderSum(atom);
if (atomContainerBondOrderSum == CDKConstants.UNSET) atomContainerBondOrderSum = 0.0;
if (atomContainerBondOrderSum < bondOrderSum - hydrogenCount) {
logger.debug("Atom has " + atomContainerBondOrderSum + ", may have: " + bondOrderSum);
partners = atomContainer.getConnectedAtomsList(atom);
for (int g = 0; g < partners.size(); g++) {
partner = (IAtom) partners.get(g);
logger.debug("Atom has " + partners.size() + " partners");
atomTypes2 = getAtomTypeFactory(atom.getBuilder()).getAtomTypes(partner.getSymbol());
if (atomTypes2.length == 0) return;
Double bos2 = atomTypes2[0].getBondOrderSum();
Integer hc2 = partner.getImplicitHydrogenCount();
Double acbos2 = atomContainer.getBondOrderSum(partner);
if (bos2 == CDKConstants.UNSET) bos2 = 0.0;
if (hc2 == CDKConstants.UNSET) hc2 = 0;
if (acbos2 == CDKConstants.UNSET) acbos2 = 0.0;
if (acbos2 < bos2 - hc2) {
logger.debug("Partner has " + acbos2 + ", may have: " + bos2);
bond = atomContainer.getBond(atom, partner);
logger.debug("Bond order was " + bond.getOrder());
BondManipulator.increaseBondOrder(bond);
logger.debug("Bond order now " + bond.getOrder());
break;
}
}
}
}
}
}
}
}
public void saturateRingSystems(IAtomContainer atomContainer) throws CDKException {
IRingSet rs = Cycles.sssr(atomContainer.getBuilder().newInstance(IAtomContainer.class, atomContainer))
.toRingSet();
List ringSets = RingPartitioner.partitionRings(rs);
IAtomContainer ac = null;
IAtom atom = null;
int temp[];
for (int f = 0; f < ringSets.size(); f++) {
rs = ringSets.get(f);
List containers = RingSetManipulator.getAllAtomContainers(rs);
for (int counter = 0; counter < containers.size(); counter++) {
ac = containers.get(counter);
temp = new int[ac.getAtomCount()];
for (int g = 0; g < ac.getAtomCount(); g++) {
atom = ac.getAtom(g);
temp[g] = atom.getImplicitHydrogenCount();
atom.setImplicitHydrogenCount(atomContainer.getConnectedBondsCount(atom)
- ac.getConnectedBondsCount(atom) - temp[g]);
}
saturate(ac);
for (int g = 0; g < ac.getAtomCount(); g++) {
atom = ac.getAtom(g);
atom.setImplicitHydrogenCount(temp[g]);
}
}
}
}
/*
* Recursively fixes bond orders in a molecule for which only connectivities
* but no bond orders are know.
* @ param molecule The molecule to fix the bond orders for
* @ param bond The number of the bond to treat in this recursion step
* @ return true if the bond order which was implemented was ok.
*/
/*
* private boolean recursiveBondOrderFix(Molecule molecule, int bondNumber)
* { Atom partner = null; Atom atom = null; Atom[] partners = null;
* AtomType[] atomTypes1 = null; AtomType[] atomTypes2 = null; int
* maxBondOrder = 0; int oldBondOrder = 0; if (bondNumber <
* molecule.getBondCount()) { Bond bond = molecule.getBondAt(f); } else {
* return true; } atom = bond.getAtomAt(0); partner = bond.getAtomAt(1);
* atomTypes1 = atf.getAtomTypes(atom.getSymbol(),
* atf.ATOMTYPE_ID_STRUCTGEN); atomTypes2 =
* atf.getAtomTypes(partner.getSymbol(), atf.ATOMTYPE_ID_STRUCTGEN);
* maxBondOrder = Math.min(atomTypes1[0].getMaxBondOrder(),
* atomTypes2[0].getMaxBondOrder()); for (int f = 1; f <= maxBondOrder; f++)
* { oldBondOrder = bond.getOrder() bond.setOrder(f); if
* (!isOverSaturated(atom, molecule) && !isOverSaturated(partner, molecule))
* { if (!recursiveBondOrderFix(molecule, bondNumber + 1)) break; } else {
* bond.setOrder(oldBondOrder); return false; } } return true; }
*/
/**
* Calculate the number of missing hydrogens by subtracting the number of
* bonds for the atom from the expected number of bonds. Charges are included
* in the calculation. The number of expected bonds is defined by the AtomType
* generated with the AtomTypeFactory.
*
* @param atom Description of the Parameter
* @param container Description of the Parameter
* @return Description of the Return Value
* @see AtomTypeFactory
*/
public int calculateNumberOfImplicitHydrogens(IAtom atom, IAtomContainer container) throws CDKException {
return this.calculateNumberOfImplicitHydrogens(atom, container, false);
}
public int calculateNumberOfImplicitHydrogens(IAtom atom) throws CDKException {
List bonds = new ArrayList();
return this.calculateNumberOfImplicitHydrogens(atom, 0, 0, bonds, false);
}
public int calculateNumberOfImplicitHydrogens(IAtom atom, IAtomContainer container,
boolean throwExceptionForUnknowAtom) throws CDKException {
return this.calculateNumberOfImplicitHydrogens(atom, container.getBondOrderSum(atom),
container.getConnectedSingleElectronsCount(atom), container.getConnectedBondsList(atom),
throwExceptionForUnknowAtom);
}
/**
* Calculate the number of missing hydrogens by subtracting the number of
* bonds for the atom from the expected number of bonds. Charges are included
* in the calculation. The number of expected bonds is defined by the AtomType
* generated with the AtomTypeFactory.
*
* @param atom Description of the Parameter
* @param throwExceptionForUnknowAtom Should an exception be thrown if an unknown atomtype is found or 0 returned ?
* @return Description of the Return Value
* @see AtomTypeFactory
*/
public int calculateNumberOfImplicitHydrogens(IAtom atom, double bondOrderSum, double singleElectronSum,
List connectedBonds, boolean throwExceptionForUnknowAtom) throws CDKException {
int missingHydrogen = 0;
if (atom instanceof IPseudoAtom) {
// don't figure it out... it simply does not lack H's
} else if (atom.getAtomicNumber() != null && atom.getAtomicNumber() == 1 || atom.getSymbol().equals("H")) {
missingHydrogen = (int) (1 - bondOrderSum - singleElectronSum - atom.getFormalCharge());
} else {
logger.info("Calculating number of missing hydrogen atoms");
// get default atom
IAtomType[] atomTypes = getAtomTypeFactory(atom.getBuilder()).getAtomTypes(atom.getSymbol());
if (atomTypes.length == 0 && throwExceptionForUnknowAtom) return 0;
logger.debug("Found atomtypes: " + atomTypes.length);
if (atomTypes.length > 0) {
IAtomType defaultAtom = atomTypes[0];
logger.debug("DefAtom: ", defaultAtom);
Integer formalCharge = atom.getFormalCharge();
if (formalCharge == null) formalCharge = 0;
Double tmpBondOrderSum = defaultAtom.getBondOrderSum();
if (tmpBondOrderSum == null) tmpBondOrderSum = 0.0;
missingHydrogen = (int) (tmpBondOrderSum - bondOrderSum - singleElectronSum + formalCharge);
if (atom.getFlag(CDKConstants.ISAROMATIC)) {
boolean subtractOne = true;
for (int i = 0; i < connectedBonds.size(); i++) {
IBond conBond = (IBond) connectedBonds.get(i);
if (conBond.getOrder() == IBond.Order.DOUBLE || conBond.getFlag(CDKConstants.ISAROMATIC))
subtractOne = false;
}
if (subtractOne) missingHydrogen--;
}
logger.debug("Atom: ", atom.getSymbol());
logger.debug(" max bond order: " + tmpBondOrderSum);
logger.debug(" bond order sum: " + bondOrderSum);
logger.debug(" charge : " + formalCharge);
} else {
logger.warn("Could not find atom type for ", atom.getSymbol());
}
}
return missingHydrogen;
}
}