All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.xmlcml.cml.tools.AtomSetTool Maven / Gradle / Ivy

/**
 *    Copyright 2011 Peter Murray-Rust et. al.
 *
 *    Licensed under the Apache License, Version 2.0 (the "License");
 *    you may not use this file except in compliance with the License.
 *    You may obtain a copy of the License at
 *
 *        http://www.apache.org/licenses/LICENSE-2.0
 *
 *    Unless required by applicable law or agreed to in writing, software
 *    distributed under the License is distributed on an "AS IS" BASIS,
 *    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 *    See the License for the specific language governing permissions and
 *    limitations under the License.
 */

package org.xmlcml.cml.tools;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;

import nu.xom.Elements;
import nu.xom.Nodes;

import org.apache.log4j.Logger;
import org.xmlcml.cml.base.AbstractTool;
import org.xmlcml.cml.base.CMLConstants;
import org.xmlcml.cml.base.CMLElement;
import org.xmlcml.cml.base.CMLElements;
import org.xmlcml.cml.base.CMLElement.CoordinateType;
import org.xmlcml.cml.element.CMLAngle;
import org.xmlcml.cml.element.CMLAtom;
import org.xmlcml.cml.element.CMLAtomSet;
import org.xmlcml.cml.element.CMLBond;
import org.xmlcml.cml.element.CMLBondSet;
import org.xmlcml.cml.element.CMLLink;
import org.xmlcml.cml.element.CMLMap;
import org.xmlcml.cml.element.CMLMolecule;
import org.xmlcml.cml.element.CMLTorsion;
import org.xmlcml.cml.element.CMLTransform3;
import org.xmlcml.cml.element.CMLMap.Direction;
import org.xmlcml.euclid.Point3;
import org.xmlcml.euclid.Point3Vector;
import org.xmlcml.euclid.Real2;
import org.xmlcml.euclid.Real2Range;
import org.xmlcml.euclid.Real3Range;
import org.xmlcml.euclid.Transform3;
import org.xmlcml.euclid.Util;
import org.xmlcml.euclid.Vector3;
import org.xmlcml.molutil.ChemicalElement.AS;

/**
 * tool to support atom set. not sure if useful
 * 
 * @author pmr
 * 
 */
public class AtomSetTool extends AbstractTool {
	private static Logger LOG = Logger.getLogger(AtomSetTool.class);

	private CMLAtomSet atomSet;
	private CMLMolecule molecule = null;
	private Map parentTable = null;
	private AtomMatchObject atomMatchObject = null;
	
	static List getChildAtomSetList(CMLElement element) {
		Nodes nodes = element.query("./cml:atomSet", CMLConstants.CML_XPATH);
		 List  atomSetList = new ArrayList();
		 for (int i = 0; i < nodes.size(); i++) {
			 atomSetList.add((CMLAtomSet) nodes.get(i));
		 }
		return atomSetList;
	}
	    
	/**
	 * constructor.
	 * 
	 * @param atomSet
	 * @deprecated use getOrCreateTool
	 */
	public AtomSetTool(CMLAtomSet atomSet) {
		if (atomSet == null) {
			throw new RuntimeException("Null atomSet");
		}
		this.atomSet = atomSet;
		List atomList = atomSet.getAtoms();
		if (atomList.size() > 0) {
			molecule = atomList.get(0).getMolecule();
		}
	}

	/**
	 * gets AtomSetTool associated with atomSet. if null creates one and sets it
	 * in atomSet
	 * 
	 * @param atomSet
	 * @return tool
	 */
	public static AtomSetTool getOrCreateTool(CMLAtomSet atomSet) {
		AtomSetTool atomSetTool = null;
		if (atomSet != null) {
			atomSetTool = (AtomSetTool) atomSet.getTool();
			if (atomSetTool == null) {
				atomSetTool = new AtomSetTool(atomSet);
				atomSet.setTool(atomSetTool);
				if (atomSet.getAtoms().size() > 0) {
					atomSetTool.molecule = atomSetTool.getMoleculeOrAncestor();
				}
			}
		}
		return atomSetTool;
	}
	
    public CMLMolecule getMoleculeOrAncestor() {
        List atoms = atomSet.getAtoms();
        if (atoms.size() > 0) {
            molecule = CMLMolecule.getMoleculeAncestor(atoms.get(0));
        } else {
            throw new RuntimeException("NO atoms in set...");
        }
        return molecule;
    }

    /**
     * @param atomSetOld
     * @param atomIds if not found, skips without error
     * @return
     */
	public static CMLAtomSet createAtomSet(CMLAtomSet atomSetOld,
			String[] atomIds) {
		CMLAtomSet atomSet = new CMLAtomSet();
		for (String atomId : atomIds) {
			CMLAtom atom = atomSetOld.getAtomById(atomId);
			if (atom != null) {
				atomSet.addAtom(atom);
			}
		}
		return atomSet;
	}

	/**
	 * gets AtomSetTool associated with molecule. if null creates one and sets
	 * it in atomSet (currently always creates it as molecule does not remember
	 * atomSet)
	 * 
	 * @param molecule
	 * @return tool
	 */
	public static AtomSetTool getOrCreateTool(CMLMolecule molecule) {
		return getOrCreateTool(MoleculeTool.getOrCreateTool(molecule)
				.getAtomSet());
	}

	// =============== ATOMSET =========================

	/**
	 * gets bondset for all bonds in current atom set.
	 * slow.
	 * order of bond is
	 * undetermined but probably in atom document order and iteration through
	 * ligands
	 * 
	 * @return the bondSet
	 * @throws CMLException one or more bonds does not have an id
	 */
	public CMLBondSet extractBondSet() {
		List atoms = atomSet.getAtoms();
		CMLBondSet bondSet = new CMLBondSet();
		for (CMLAtom atom : atoms) {
			List ligandList = atom.getLigandAtoms();
			List ligandBondList = atom.getLigandBonds();
			for (int i = 0; i < ligandList.size(); i++) {
				CMLAtom ligandAtom = ligandList.get(i);
				CMLBond ligandBond = ligandBondList.get(i);
				if (atomSet.contains(ligandAtom) &&
					!bondSet.contains(ligandBond)) {
					bondSet.addBond(ligandBond);
				}
			}
		}
		return bondSet;
	}

	/**
	 * gets bondset for all bonds in current atom set. slow. order of bond is
	 * undetermined but probably in atom document order and iteration through
	 * ligands
	 * 
	 * @return the bondSet
	 * @throws RuntimeException
	 *             one or more bonds does not have an id
	 */
	@SuppressWarnings("unused")
	// FIXME don't think this works
	public CMLBondSet getBondSet() {
		List atoms = molecule.getAtoms();
		molecule.getBonds();
		CMLBondSet bondSet = new CMLBondSet();
		for (CMLAtom atom : atoms) {
			List ligandList = atom.getLigandAtoms();
			List ligandBondList = atom.getLigandBonds();
			// Iterator ita = atom.getLigandList().iterator();
			// Iterator itb = atom.getBondList().iterator();
			// loop through ligands and examine each for membership of this set;
			// if so add bond
			int i = 0;
			for (CMLAtom ligand : ligandList) {
				CMLBond ligandBond = ligandBondList.get(i++);
				// if (atomSet.contains(ligand)) {
				bondSet.addBond(ligandBond);
				// }
			}
		}
		return bondSet;
	}
	
    public CMLAtomSet getAtomSetIncludingElementTypes(String[] elementTypes) {
     	return createIncludedSet(elementTypes, true);
     }

    public CMLAtomSet getAtomSetExcludingElementTypes(String[] elementTypes) {
    	return createIncludedSet(elementTypes, false);
    }

	private CMLAtomSet createIncludedSet(String[] elementTypes, boolean include) {
		CMLAtomSet includedAtomSet = new CMLAtomSet();
		boolean ignoreCase = false;
    	List atoms = atomSet.getAtoms();
    	for (CMLAtom atom : atoms) {
    		String elementType = atom.getElementType();
    		if (include) {
    			if (Util.indexOf(elementType, elementTypes, ignoreCase) != -1) {
    				includedAtomSet.addAtom(atom);
    			}
    		} else {
    			if (Util.indexOf(elementType, elementTypes, ignoreCase) == -1) {
    				includedAtomSet.addAtom(atom);
    			}
    		} 
    	}
		return includedAtomSet;
	}

	/**
	 * create all valence angles for molecule.
	 * 
	 * @param atomSet
	 * @param calculate
	 *            false=> empty content; true=>calculated values (degrees) as
	 *            content
	 * @param add
	 *            array as childElements of molecule
	 * @return array of angles (zero length if none)
	 */
	public List createValenceAngles(CMLAtomSet atomSet,
			boolean calculate, boolean add) {
		List angleVector = new ArrayList();
		for (CMLAtom atomi : atomSet.getAtoms()) {
			Set usedAtomSetj = new HashSet();
			List ligandListI = atomi.getLigandAtoms();
			for (CMLAtom atomj : ligandListI) {
				usedAtomSetj.add(atomj);
				for (CMLAtom atomk : ligandListI) {
					if (usedAtomSetj.contains(atomk))
						continue;
					CMLAngle angle = new CMLAngle();
					// angle.setMolecule(molecule);
					angle.setAtomRefs3(new String[] { atomj.getId(),
							atomi.getId(), atomk.getId() });
					if (calculate) {
						double angleVal = angle.getCalculatedAngle(molecule);
						angle.setXMLContent(angleVal);
					}
					if (add) {
						try {
							molecule.appendChild(angle);
						} catch (Exception e) {
							e.printStackTrace();
						}
					}
					angleVector.add(angle);
				}
			}
		}
		return angleVector;
	}

	/**
	 * get all valence torsions for molecule.
	 * 
	 * @param atomSet
	 * @param calculate
	 *            false=> empty content; true=>calculated values (degrees) as
	 *            content
	 * @param add
	 *            array as childElements of molecule
	 * @return array of torsions (zero length if none)
	 */
	public List createValenceTorsions(CMLAtomSet atomSet,
			boolean calculate, boolean add) {
		List torsionVector = new ArrayList();
		for (CMLBond bond : molecule.getBonds()) {
			CMLAtom at0 = bond.getAtom(0);
			if (!atomSet.contains(at0)) {
				continue;
			}
			CMLAtom at1 = bond.getAtom(1);
			if (!atomSet.contains(at1)) {
				continue;
			}

			List ligandList0 = at1.getLigandAtoms();
			for (CMLAtom ligand0 : ligandList0) {
				if (!atomSet.contains(ligand0)) {
					continue;
				}
				if (ligand0.equals(at1)) {
					continue;
				}
				List ligandList1 = at1.getLigandAtoms();
				for (CMLAtom ligand1 : ligandList1) {
					if (!atomSet.contains(ligand1)) {
						continue;
					}
					if (ligand1.equals(at0)) {
						continue;
					}
					CMLTorsion torsion = new CMLTorsion();
					torsion.setAtomRefs4(new String[] { ligand0.getId(),
							at0.getId(), at1.getId(), ligand1.getId() });
					if (calculate) {
						double torsionVal = torsion
								.getCalculatedTorsion(molecule);
						torsion.setXMLContent(torsionVal);
					}
					if (add) {
						try {
							molecule.appendChild(torsion);
						} catch (Exception e) {
							e.printStackTrace();
						}
					}
					torsionVector.add(torsion);
				}
			}
		}
		return torsionVector;
	}

	/**
	 * Creates new AtomSet one ligand shell larger than this one.
	 * 
	 * @return set
	 */
	public CMLAtomSet sprout() {
		CMLAtomSet newAtomSet = new CMLAtomSet();
		List atoms = atomSet.getAtoms();
		for (CMLAtom atomi : atoms) {
			newAtomSet.addAtom(atomi);
			List ligandList = atomi.getLigandAtoms();
			for (CMLAtom ligandj : ligandList) {
				if (!atomSet.contains(ligandj)) {
					newAtomSet.addAtom(ligandj);
				}
			}
		}
		return newAtomSet;
	}

	/**
	 * tests whether an array of atomSets are all different.
	 * 
	 * assumes all sets are in prioritized order (i.e. simply compares each set
	 * with each other without normalization or sorting uses compareTo
	 * 
	 * @param atomSets
	 *            array of ordered atomSets (null returns true)
	 * 
	 * @return are all sets different (by ordered atom equality)
	 */
	public static boolean areOrderedAtomSetsDifferent(CMLAtomSet[] atomSets) {
		boolean different = true;
		if (atomSets != null) {
			for (int i = 0; i < atomSets.length; i++) {
				for (int j = i; j < atomSets.length; j++) {
					// boolean diff = false;
					if (atomSets[i] == null || atomSets[j] == null) {
						throw new RuntimeException("Null atom set component: "
								+ i + S_SLASH + j);
					} else if (atomSets[i].compareTo(atomSets[j]) == 0) {
						different = false;
						break;
					}
				}
				if (!different) {
					break;
				}
			}
		}
		return different;
	}

	/**
	 * create from an array of molecules. we seem not to be able to use two
	 * List<> constructors
	 * 
	 * @param molecules
	 *            the molecules
	 * @return atomSet
	 */
	// public CMLAtomSet(List molecule) {
	public static CMLAtomSet createAtomSet(List molecules) {
		CMLAtomSet atomSet = new CMLAtomSet();
		for (CMLMolecule mol : molecules) {
			for (CMLAtom atom : mol.getAtoms()) {
				atomSet.addAtom(atom);
			}
		}
		return atomSet;
	}

	/**
	 * Wipes parent atom references.
	 * 
	 * For use with spanning trees.
	 * 
	 */
	public void resetParents() {
		parentTable = new HashMap();
	}

	/**
	 * Sets parent atom reference.
	 * 
	 * For use with spanning trees.
	 * 
	 * @param atom
	 *            atom whose parent should be set
	 * @param parentAtom
	 * @throws RuntimeException
	 *             child atom not in atom set
	 */
	public void setParent(CMLAtom atom, CMLAtom parentAtom) {
		if (!atomSet.contains(atom)) {
			throw new RuntimeException("Child atom not in atom set");
		}
		if (parentTable == null) {
			parentTable = new HashMap();
		}
		parentTable.put(atom, parentAtom);
	}

	/**
	 * Gets parent atom reference.
	 * 
	 * For use with spanning trees. Returns null if parent atom is not known.
	 * 
	 * @param atom
	 *            atom whose parent should be found
	 * @return parent atom, or null
	 * @throws RuntimeException
	 *             child atom not in atom set
	 */
	public CMLAtom getParent(CMLAtom atom) throws RuntimeException {
		if (!atomSet.contains(atom)) {
			throw new RuntimeException("Child atom not in atom set");
		}
		if (parentTable == null) {
			return null;
		}
		CMLAtom parentAtom = parentTable.get(atom);
		return parentAtom;
	}

	/**
	 * find atom closest to point. uses 2D coordinates skips atoms without 2D
	 * coords
	 * 
	 * @param point
	 * @return atom or null
	 */
	public CMLAtom getNearestAtom(Real2 point) {
		CMLAtom closestAtom = null;
		double maxDist = 999999.;
		List thisAtoms = atomSet.getAtoms();
		for (int i = 0; i < thisAtoms.size(); i++) {
			CMLAtom thisAtom = thisAtoms.get(i);
			Real2 thisXY2 = thisAtom.getXY2();
			if (thisXY2 != null) {
				double dist = thisXY2.getDistance(point);
				if (dist < maxDist) {
					maxDist = dist;
					closestAtom = thisAtom;
				}
			}
		}
		return closestAtom;
	}

	/**
	 * find atom closest to point. uses 3D coordinates skips atoms without 3D
	 * coords
	 * 
	 * @param point
	 * @return atom or null
	 */
	public CMLAtom getNearestAtom(Point3 point) {
		CMLAtom closestAtom = null;
		double maxDist = 999999.;
		List thisAtoms = atomSet.getAtoms();
		for (int i = 0; i < thisAtoms.size(); i++) {
			CMLAtom thisAtom = thisAtoms.get(i);
			Point3 thisXYZ3 = thisAtom.getXYZ3();
			if (thisXYZ3 != null) {
				double dist = thisXYZ3.getDistanceFromPoint(point);
				if (dist < maxDist) {
					maxDist = dist;
					closestAtom = thisAtom;
				}
			}
		}
		return closestAtom;
	}

	/**
	 * get nearest atom.
	 * 
	 * iterates though this.atoms of same elementType as atom comparing 2D
	 * distance to atom.
	 * 
	 * @param atom
	 * @return nearest atom of same type as atom or null
	 */
	public CMLAtom getNearestAtom2OfSameElementType(CMLAtom atom) {
		CMLAtom closestAtom = null;
		if (atom != null) {
			String elementType = atom.getElementType();
			Real2 xy2 = atom.getXY2();
			if (xy2 != null) {
				closestAtom = getNearestAtomOfSameElementType(elementType, xy2);
			}
		}
		return closestAtom;
	}

	/**
	 * @param elementType
	 * @param xy2
	 * @return atom or null
	 */
	public CMLAtom getNearestAtomOfSameElementType(String elementType, Real2 xy2) {
		CMLAtom closestAtom = null;
		double maxDist = 999999.;
		List thisAtoms = atomSet.getAtoms();
		for (int i = 0; i < thisAtoms.size(); i++) {
			CMLAtom thisAtom = thisAtoms.get(i);
			if (elementType.equals(thisAtom.getElementType())) {
				Real2 thisXY2 = thisAtom.getXY2();
				double dist = thisXY2.getDistance(xy2);
				if (dist < maxDist) {
					maxDist = dist;
					closestAtom = thisAtom;
				}
			}
		}
		return closestAtom;
	}

	/**
	 * remove any links pointing to atoms which differ in generic ways. this
	 * looks awful
	 * 
	 * @param map
	 *            with from links may point
	 * @param toAtomSet
	 *            to which to links may point
	 * @param attribute
	 *            such as "elementType", "formalCharge",
	 */
	public void removeUnmatchedAtoms(CMLMap map, CMLAtomSet toAtomSet,
			String attribute) {
		Elements links = map.getChildElements(CMLLink.TAG, CMLConstants.CML_NS);
		for (int i = 0; i < links.size(); i++) {
			String fromId = ((CMLLink) links.get(i)).getFrom();
			String toId = ((CMLLink) links.get(i)).getTo();
			CMLAtom fromAtom = molecule.getAtomById(fromId);
			CMLAtom toAtom = toAtomSet.getAtomById(toId);
			String fromAttribute = (fromAtom == null) ? null : fromAtom
					.getAttribute(attribute).getValue();
			String toAttribute = (toAtom == null) ? null : toAtom.getAttribute(
					attribute).getValue();
			// no match
			if (toAttribute != null && fromAttribute != null
					&& !toAttribute.equals(fromAttribute)) {
				links.get(i).detach();
			}
		}
	}

	/**
	 * creates an atomSet of those atoms which have identical 3D coordinates.
	 * returns atoms in 'this', i.e. if a1 in this overlaps a2 in otherSet uses
	 * a1. If IDs in this clash with atomSet the serialization is fragile
	 * 
	 * @param otherSet
	 *            to compare to
	 * @param type
	 *            whether cartesian or fractional
	 * @return referring to atoms in this which overlap with otherSet. if empty
	 *         returns empty atomSet, not null
	 */
	public CMLAtomSet getOverlapping3DAtoms(CMLAtomSet otherSet,
			CoordinateType type) {
		CMLAtomSet newAtomSet = new CMLAtomSet();
		for (CMLAtom thisAtom : atomSet.getAtoms()) {
			Point3 thisPoint = thisAtom.getPoint3(type);
			if (thisPoint == null) {
				continue;
			}
			for (CMLAtom otherAtom : otherSet.getAtoms()) {
				Point3 otherPoint = otherAtom.getPoint3(type);
				if (otherPoint == null) {
					continue;
				}
				if (thisPoint.isEqualTo(otherPoint, EPS)) {
					newAtomSet.addAtom(thisAtom);
				}
			}
		}
		return newAtomSet;
	}

	/**
	 * get nearest atom in atomSet1 to this. may get 1:n and 0:n mappings
	 * 
	 * @param atomSet1
	 * @param delta
	 * @return mapping
	 * @throws RuntimeException
	 */
	public CMLMap matchNearestAtoms(CMLAtomSet atomSet1, double delta) {
		CMLMap map = new CMLMap();
		boolean match = false;
		Real2 centroid0 = atomSet.getCentroid2D();
		Real2 centroid1 = atomSet1.getCentroid2D();
		Real2 deltaM = centroid0.subtract(centroid1);
		atomSet1.translate2D(deltaM);
		List atoms0 = atomSet.getAtoms();
		List atoms1 = atomSet1.getAtoms();
		Map atomMatchTable = new HashMap();
		for (CMLAtom atom0 : atoms0) {
			Real2 point0 = atom0.getXY2();
			for (CMLAtom atom1 : atoms1) {
				if (atomMatchTable.containsKey(atom1)) {
					continue;
				}
				Real2 point1 = atom1.getXY2();
				double d = point1.getDistance(point0);
				if (d < delta) {
					atomMatchTable.put(atom0, atom1);
					break;
				}
			}
		}
		match = true;
		/**
		 * not sure this is required... for (int i = 0; i < atoms.size(); i++) {
		 * if (!atomMatchTable.containsKey(atoms[i])) { match = false; } } for
		 * (int i = 0; i < atoms.size(); i++) { if
		 * (!atomMatchTable.contains(atoms1[i])) { match = false; } } --
		 */
		if (match) {
			// FIXME
			// map = createMap(atomSet, atomSet1, atomMatchTable);
		} else {
			// System. out.println("FAILED to match unordered atoms");
		}

		return map;
	}

	   /** gets extend (bounding box).
	    * 
	    * @return real2range
	    */
	   public Real2Range getExtent2() {
		   Real2Range r2r = new Real2Range();
		   if (molecule != null) {
			   List atoms = molecule.getAtoms();
			   for (CMLAtom atom : atoms) {
				   Real2 xy = atom.getXY2();
				   if (xy != null) {
					   r2r.add(xy);
				   }
			   }
		   }
		   return r2r;
	   }
	    
	   /** gets extend (bounding box).
	    * 
	    * @return {@link Real3Range}
	    */
	   public Real3Range getExtent3() {
		   Real3Range r3r = new Real3Range();
		   List atoms = molecule.getAtoms();
		   for (CMLAtom atom : atoms) {
			   Point3 xyz = atom.getXYZ3();
			   if (xyz != null) {
				   r3r.add(xyz);
			   }
		   }
		   return r3r;
	   }
	public CMLAtomSet getAtomSet() {
		return atomSet;
	}

	public CMLMolecule getMolecule() {
		return molecule;
	}

	/**
	 * transform 3D cartesian coordinates. modifies this
	 * 
	 * @param transform
	 *            the transformation
	 */
	public void transformCartesians(Transform3 transform) {
		for (CMLAtom atom : atomSet.getAtoms()) {
			atom.transformCartesians(transform);
		}
	}

	/**
	 * transform 3D cartesian coordinates. modifies this
	 * 
	 * @param transform
	 *            the transformation
	 */
	public void transformCartesians(CMLTransform3 transform) {
		for (CMLAtom atom : atomSet.getAtoms()) {
			AtomTool.getOrCreateTool(atom).transformCartesians(transform);
		}
	}

    public void translateCentroidToOrigin3(CoordinateType type) {
    	Point3 centroid = this.getCentroid3(type);
    	if (centroid != null) {
	    	Vector3 v3 = new Vector3(centroid).multiplyBy(-1.0);
	    	this.translate3D(v3, type);
    	}
    }

    /**
     * translate molecule in 3D.
     *
     * @param delta3 add to all 3D coordinates
     */
    public void translate3D(Vector3 delta3, CoordinateType type) {
        List atoms = atomSet.getAtoms();
        for (int i = 0; i < atoms.size(); i++) {
            CMLAtom atom = atoms.get(i);
            if (type.equals(CoordinateType.CARTESIAN)) {
	            if (atom.getX3Attribute() != null && atom.getY3Attribute() != null
	                    && atom.getZ3Attribute() != null) {
	                atom.setX3(atom.getX3() + delta3.getArray()[0]);
	                atom.setY3(atom.getY3() + delta3.getArray()[1]);
	                atom.setZ3(atom.getZ3() + delta3.getArray()[2]);
	            }
            } else if (type.equals(CoordinateType.FRACTIONAL)) {
	            if (atom.getXFractAttribute() != null &&
	            		atom.getYFractAttribute() != null &&
	            		atom.getZFractAttribute() != null) {
	                atom.setXFract(atom.getXFract() + delta3.getArray()[0]);
	                atom.setYFract(atom.getYFract() + delta3.getArray()[1]);
	                atom.setZFract(atom.getZFract() + delta3.getArray()[2]);
	            }
            }
        }
    }

    /** get 3D centroid.
    *
    * @param type
    *            CARTESIAN or FRACTIONAL
    * @return centroid of 3D coords or null
    */
   public Point3 getCentroid3(CoordinateType type) {
       Point3 centroid3 = null;
       Point3Vector p3Vector = atomSet.getCoordinates3(type);
       if (p3Vector != null) {
           centroid3 = p3Vector.getCentroid();
       }
       return centroid3;
   }

	/**
	 * transform 3D fractional coordinates. modifies this does not affect x3,
	 * y3, z3 (may need to re-generate cartesians)
	 * 
	 * @param transform
	 *            the transformation
	 */
	public void transformFractionals(Transform3 transform) {
		for (CMLAtom atom : atomSet.getAtoms()) {
			atom.transformFractionals(transform);
		}
	}

	/**
	 * transform 3D fractional coordinates. modifies this does not affect x3,
	 * y3, z3 (may need to re-generate cartesians)
	 * 
	 * @param transform
	 *            the transformation
	 */
	public void transformFractionals(CMLTransform3 transform) {
		for (CMLAtom atom : atomSet.getAtoms()) {
			AtomTool.getOrCreateTool(atom).transformFractionals(transform);
		}
	}

	/**
	 * transform fractional and 3D coordinates. does NOT alter 2D coordinates
	 * transforms fractionals then applies orthogonalisation to result
	 * 
	 * @param transform
	 *            the fractional symmetry transformation
	 * @param orthTransform
	 *            orthogonalisation transform
	 */
	public void transformFractionalsAndCartesians(CMLTransform3 transform,
			Transform3 orthTransform) {
		for (CMLAtom atom : atomSet.getAtoms()) {
			AtomTool.getOrCreateTool(atom).transformFractionalsAndCartesians(
					transform, orthTransform);
		}
	}

	class SearchAtom {
		CMLAtom atom;
		List matchesInTarget = new ArrayList();

		public SearchAtom(CMLAtom atom) {
			this.atom = atom;
		}

		public void addMatchableAtoms(CMLAtomSet targetAtomSet,
				AtomMatcher atomMatcher) {
			for (CMLAtom targetAtom : targetAtomSet.getAtoms()) {
				if (atomMatcher.matches(atom, targetAtom)) {
					matchesInTarget.add(targetAtom);
				}
			}
		}

	}
	
	public void clean2D(double bondLength, int ncyc) {
		int count = 0;
		boolean converged = false;
		LOG.trace("clean "+atomSet.getSize());
		while (!converged && count < ncyc) {
	    	double modShift = 0.;
	    	Map shiftMap = new HashMap();
	    	for (CMLAtom atom : atomSet.getAtoms()) {
	    		LOG.trace(atom.getId());
	    		buildShiftFromLigands(bondLength, atom, shiftMap);
	    	}
	    	for (CMLAtom atom : atomSet.getAtoms()) {
				Real2 shift = shiftMap.get(atom);
				modShift += shift.getLength();
	    		LOG.trace("mod shift "+modShift);
	    	}
    		if (modShift < 0.01 * bondLength) {
    			converged = true;
    			LOG.trace("converged");
    			break;
    		}
	    	int i = 0;
	    	for (CMLAtom atom : shiftMap.keySet()) {
	    		Real2 shift = shiftMap.get(atom);
	    		LOG.trace("SHIFT "+shift);
	    		atom.increaseXY2(shift.getX(), shift.getY());
	    		LOG.trace("XY2 "+atom.getXY2());
	    		i++;
	    	}
	    	count++;
		}
		
	}
	

	private void buildShiftFromLigands(double bondLength, CMLAtom atom, Map shiftMap) {
		List ligands = atom.getLigandAtoms();
		for (int i = 0; i < ligands.size(); i++) {
			CMLAtom ligand = ligands.get(i);
			if (atomSet.contains(ligand)) {
				// only count each pair once. crude but works
	    		if (atom.hashCode() > ligand.hashCode()) {
	    			getaddShiftsToMap(bondLength, atom, ligand, shiftMap);
	    		}
	    		// make sure interaction only counted once
	    		//FIXME  fails for 4-rings...
				for (int j = i+1; j < ligands.size(); j++) {
					CMLAtom ligand2 = ligands.get(j);
					if (atomSet.contains(ligand2)) {
						getaddShiftsToMap(bondLength * Math.sqrt(3.), ligand, ligand2, shiftMap);
					}
				}
			}
		}
	}

	private void getaddShiftsToMap(double bondLength, CMLAtom atom1,
			CMLAtom atom2, Map shiftMap) {
		Real2 x0 = atom1.getXY2();
		Real2 xi = atom2.getXY2();
		LOG.trace(" "+atom1.getId()+" ... "+atom2.getId());
		if (atom1.equals(atom2)) {
			throw new RuntimeException("identical ligands");
		}
//		Real2 vi = xi.subtract(x0);
		Real2 x0i = xi.subtract(x0);
		Real2 x0in = x0i.getUnitVector().multiplyBy(bondLength);
		Real2 bondDelta = x0i.subtract(x0in).multiplyBy(0.5);
		LOG.trace("D+ "+bondDelta);
		addShift(shiftMap, atom1, bondDelta);
		bondDelta.multiplyEquals(-1.);
		LOG.trace("D- "+bondDelta);
		addShift(shiftMap, atom2, bondDelta);
	}
	
	private void addShift(Map map, CMLAtom atom, Real2 delta) {
		Real2 shift = map.get(atom);
		if (shift == null) {
			shift = new Real2();
			map.put(atom, shift);
		}
		shift.plusEquals(delta);
	}
	

	/**
	 * returns containing atomTree labelling (was in AtomSet) the atomTree is
	 * calculated for each atom and expanded until that atom can be seen to be
	 * unique in the atomSet or until maxAtomTreeLevel is reached For those
	 * atoms which have unique atomTreeStrings and are keyed by these. If there
	 * are sets of atoms which have the same string they are mapped to atomSets.
	 * Example (maxAtomTreeLevel = 1: O1-C2-O3 has map = {"C", "a2"}, {"O(C)",
	 * atomSet(a1, a3)}
	 * 
	 * terms:
	 *  atomSetValue "a1 a3 a4"
	 *  atomTreeString "C(N)(N(O))"
	 * 
	 * @param atomSet
	 * @param atomMatcher
	 *            to use
	 * @return the maps
	 */
	@SuppressWarnings("all")
	public Map createAtomSetByAtomTreeStringAtomTreeLabelling(AtomMatchObject atomMatchObject) {
		this.atomMatchObject = atomMatchObject;
		// iterate through levels
		List atoms = atomSet.getAtoms();
		int atomTreeLevel = atomMatchObject.getAtomTreeLevel(); // -1 by default
		boolean variableLevel = (atomTreeLevel < 0);
		int startLevel = (variableLevel) ? 0 : atomTreeLevel;
		int endLevel = (variableLevel) ? atomMatchObject.getMaximumAtomTreeLevel() : atomTreeLevel + 1;
		
		Map atomSetByAtomTreeString = 
			iterateThroughLevelsUntilPersistentNoChangeAndCreateAtomSetByAtomTreeString(
			atoms, variableLevel, startLevel, endLevel);
//		ToolUtils.debugMap("atomSetByAtomTreeString", atomSetByAtomTreeString);
		Map atomTreeStringByAtomSetValue = 
			createAtomTreeStringsIndexedByAtomSetValue(atomSetByAtomTreeString);
//		ToolUtils.debugMap("atomTreeStringByAtomSetValue", atomTreeStringByAtomSetValue);
		Map shortestAtomSetValueByAtomId = 
			createShortestAtomSetValueByAtomId(atomSetByAtomTreeString);
//		ToolUtils.debugMap("shortestAtomSetValueByAtomId", shortestAtomSetValueByAtomId);
		atomTreeStringByAtomSetValue = removeSuperAtomSetsInAtomTreeStringByAtomSetValue(atomTreeStringByAtomSetValue, shortestAtomSetValueByAtomId);
//		ToolUtils.debugMap("atomTreeStringByAtomSetValueShort", atomTreeStringByAtomSetValue);
		atomSetByAtomTreeString = removeSuperSetsInAtomSetByAtomTreeString(atomSetByAtomTreeString, atomTreeStringByAtomSetValue);
//		ToolUtils.debugMap("atomTreeStringByAtomSetValueShort", atomTreeStringByAtomSetValue);
		Map atomTreeStringByAtomId = createAtomTreeStringByAtomId(
				shortestAtomSetValueByAtomId, atomTreeStringByAtomSetValue);
//		ToolUtils.debugMap("atomTreeStringByAtomId", atomTreeStringByAtomId);
		return atomSetByAtomTreeString;
	}


	private Map removeSuperSetsInAtomSetByAtomTreeString(
			Map atomSetByAtomTreeString,
			Map atomTreeStringByAtomSetValue) {
		Set keysToDelete = new HashSet();
		for (String atomTreeString : atomSetByAtomTreeString.keySet()) {
			if (!atomTreeStringByAtomSetValue.containsValue(atomTreeString)) {
				keysToDelete.add(atomTreeString);
			}
		}
		for (String atomTreeString : keysToDelete) {
			atomSetByAtomTreeString.remove(atomTreeString);
		}
		return atomSetByAtomTreeString;
	}

	private static Map removeSuperAtomSetsInAtomTreeStringByAtomSetValue(
		Map atomTreeByAtomSetValue, Map shortestAtomSetValueByAtomId) {
		Set atomSetValueSet = new HashSet();
		for (String atomId : shortestAtomSetValueByAtomId.keySet()) {
			atomSetValueSet.add(shortestAtomSetValueByAtomId.get(atomId));
		}
		Set keysToRemove = new HashSet();
		for (String atomSetValue : atomTreeByAtomSetValue.keySet()) {
			if (!atomSetValueSet.contains(atomSetValue)) {
				keysToRemove.add(atomSetValue);
			}
		}
		for (String key : keysToRemove) {
			atomTreeByAtomSetValue.remove(key);
		}
		return atomTreeByAtomSetValue;
	}

	private Map iterateThroughLevelsUntilPersistentNoChangeAndCreateAtomSetByAtomTreeString(
			List atoms, boolean variableLevel, int startLevel,
			int endLevel) {
		Map atomSetByAtomTreeString = new HashMap();
		CMLAtomSet uniqueAtomSet = new CMLAtomSet();
		int maxWithoutChange = 3;
		int nunchanged = 0;
		for (int level = startLevel; level < endLevel; level++) {
			LOG.trace("START "+level);
			int nuniq = uniqueAtomSet.size();
			iterateThroughNonUniqueAtoms(atomSetByAtomTreeString, uniqueAtomSet, atoms, level);
			markUniqueAtomsInMap(atomSetByAtomTreeString, uniqueAtomSet);
			if (uniqueAtomSet.size() == nuniq) {
				if (nunchanged++ >= maxWithoutChange-1) break;
			} else {
				nunchanged = 0;
			}
		}
		return atomSetByAtomTreeString;
	}

	private void iterateThroughNonUniqueAtoms(Map atomSetByAtomTreeString, CMLAtomSet uniqueAtomSet,
			List atoms, int level) {
		for (int i = 0; i < atoms.size(); i++) {
			CMLAtom atom = atoms.get(i);
			boolean omit = false;
			String elementType = atom.getElementType();
			for (int j = 0; j < atomMatchObject.getExcludeElementTypes().length; j++) {
				if (atomMatchObject.getExcludeElementTypes()[j].equals(elementType)) {
					omit = true;
					break;
				}
			}
			if (!omit && !uniqueAtomSet.contains(atom)) {
				createNewAtomTree(atomSetByAtomTreeString, level, atom, elementType);
			}
		}
	}

	private void createNewAtomTree(Map atomSetByAtomTreeString, int level, CMLAtom atom, String elementType) {
		AtomTree atomTree = new AtomTree(atom);
		atomTree.setUseCharge(atomMatchObject.isUseCharge());
		atomTree.setUseLabel(atomMatchObject.isUseLabel());
		atomTree.setUseExplicitHydrogens(AS.H.equals(elementType));
		atomTree.expandTo(level);
		String atomTreeString = atomTree.toString();
		CMLAtomSet atomSet = atomSetByAtomTreeString.get(atomTreeString);
		if (atomSet == null) {
			atomSet = new CMLAtomSet();
			atomSetByAtomTreeString.put(atomTreeString, atomSet);
		}
		atomSet.addAtom(atom);
	}

	private Map createAtomTreeStringsIndexedByAtomSetValue(
			Map atomSetByAtomTreeString) {
		Map atomTreeStringsIndexedByAtomSetValue = new HashMap();
		for (String atomTreeString : atomSetByAtomTreeString.keySet()) {
			CMLAtomSet atomSet = atomSetByAtomTreeString.get(atomTreeString);
			atomTreeStringsIndexedByAtomSetValue.put(atomSet.getValue(), atomTreeString);
		}
		return atomTreeStringsIndexedByAtomSetValue;
	}

	private Map createAtomTreeStringByAtomId (
			Map atomSetValueIndexedByAtomId,
			Map atomTreeStringsIndexedByAtomSetValue) {
		Map atomTreeStringByAtomId = new HashMap();
		for (String atomId : atomSetValueIndexedByAtomId.keySet()) {
			String atomSetValue = atomSetValueIndexedByAtomId.get(atomId);
			String atomTreeValue = atomTreeStringsIndexedByAtomSetValue.get(atomSetValue);
			atomTreeStringByAtomId.put(atomId, atomTreeValue);
		}
		return atomTreeStringByAtomId;
	}

	private Map createShortestAtomSetValueByAtomId(
			Map atomTreeStringsByAtomSet) {
		Map>  atomSetValuesIndexedByAtomId = new HashMap>();
		indexAtomSetValuesByAtomId(atomTreeStringsByAtomSet,
				atomSetValuesIndexedByAtomId);
		Map atomSetValueIndexedByAtomId = 
			indexShortestAtomSetByAtomId(atomSetValuesIndexedByAtomId);
		return atomSetValueIndexedByAtomId;
	}

	private void indexAtomSetValuesByAtomId(
			Map atomTreeStringsByAtomSet,
			Map> atomSetValueIndexedByAtomId) {
		for (String atomTreeString : atomTreeStringsByAtomSet.keySet()) {
			CMLAtomSet atomSet = atomTreeStringsByAtomSet.get(atomTreeString);
			String atomSetValue = atomSet.getValue();
			String[] atomIds = atomSetValue.split(" ");
			for (String atomId : atomIds) {
				List atomSetValueList = atomSetValueIndexedByAtomId.get(atomId);
				if (atomSetValueList == null) {
					atomSetValueList = new ArrayList();
					atomSetValueIndexedByAtomId.put(atomId, atomSetValueList);
				}
				atomSetValueList.add(atomSetValue);
			}
		}
	}

	private Map indexShortestAtomSetByAtomId(
			Map> atomSetValueIndexedByAtomId) {
		for (String atomId : atomSetValueIndexedByAtomId.keySet()) {
			String[] shortestAtomSetValues = null;
			String shortestAtomSetValue = null;
			List atomSetValues = atomSetValueIndexedByAtomId.get(atomId);
			for (String atomSetValue : atomSetValues) {
				String[] newAtomSetValues = atomSetValue.split(CMLConstants.S_SPACE);
				if (shortestAtomSetValues == null) {
					shortestAtomSetValues = newAtomSetValues;
					shortestAtomSetValue = atomSetValue;
				} else if (shortestAtomSetValues.length > newAtomSetValues.length) {
					shortestAtomSetValues = newAtomSetValues;
					shortestAtomSetValue = atomSetValue;
				}
			}
			atomSetValues = new ArrayList();
			atomSetValues.add(shortestAtomSetValue);
			atomSetValueIndexedByAtomId.put(atomId, atomSetValues);
		}
		Map indexShortestAtomSet = new HashMap();
		for (String atomId : atomSetValueIndexedByAtomId.keySet()) {
			indexShortestAtomSet.put(atomId, atomSetValueIndexedByAtomId.get(atomId).get(0));
		}
		return indexShortestAtomSet;
	}
	

	private void markUniqueAtomsInMap(Map atomSetByAtomTreeString, CMLAtomSet uniqueAtomSet) {
		for (String atomTreeString : atomSetByAtomTreeString.keySet()) {
			CMLAtomSet atomSet2 = atomSetByAtomTreeString.get(atomTreeString);
			LOG.trace("A "+atomTreeString+"..."+atomSet2.getValue());
			// if only one element, mark as unique
			if (atomSet2.size() == 1) {
				uniqueAtomSet.addAtom(atomSet2.getAtom(0));
				LOG.trace("Unique: "+atomTreeString+"..."+atomSet2.getValue());
			}
		}
	}

	/** should move to AtomSetTool
	 * 
	 * @param direction
	 * @param cmlMap
	 * @param atomSet
	 * @return
	 */
	public CMLAtomSet getAtomSetFromMap(Direction direction, CMLMap cmlMap) {
		CMLAtomSet newAtomSet = new CMLAtomSet();
		CMLElements links = cmlMap.getLinkElements();
		for (CMLLink link : links) {
			String toFrom = null;
			if (direction.equals(CMLMap.Direction.FROM)) {
				toFrom = link.getAttributeValue("fromSet");
			} else if (direction.equals(CMLMap.Direction.TO)) {
				toFrom = link.getAttributeValue("toSet");
			}
			if (toFrom != null) {
				String[] refs = toFrom.split(CMLConstants.S_WHITEREGEX);
				for (String ref : refs) {
					CMLAtom atom = atomSet.getAtomById(ref);
					if (atom == null) {
//						LOG.error("Cannot find atom: "+ref);
//						return null;
						throw new RuntimeException("Cannot find atom: "+ref);
					}
					newAtomSet.addAtom(atom);
				}
			}
		}
		return newAtomSet;
	}



}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy