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

org.openscience.cdk.hash.stereo.GeometricCumulativeDoubleBondFactory Maven / Gradle / Ivy

/*
 * Copyright (c) 2013 John May 
 *
 * 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 U
 */

package org.openscience.cdk.hash.stereo;

import org.openscience.cdk.interfaces.IAtom;
import org.openscience.cdk.interfaces.IAtomContainer;
import org.openscience.cdk.interfaces.IBond;

import javax.vecmath.Point2d;
import javax.vecmath.Point3d;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;

/**
 * Stereo encoder factory for 2D and 3D cumulative double bonds.
 *
 * @author John May
 * @cdk.module hash
 */
public class GeometricCumulativeDoubleBondFactory implements StereoEncoderFactory {

    /**
     * Create a stereo encoder for cumulative double bonds.
     *
     * @param container the container
     * @param graph     adjacency list representation of the container
     * @return a stereo encoder
     */
    @Override
    public StereoEncoder create(IAtomContainer container, int[][] graph) {

        int n = container.getAtomCount();
        BondMap map = new BondMap(n);

        List encoders = new ArrayList(1);

        // index double bonds by their atoms
        for (IBond bond : container.bonds()) {
            if (isDoubleBond(bond)) map.add(bond);
        }

        Set visited = new HashSet(n);

        // find atoms which are connected between two double bonds
        for (IAtom a : map.atoms()) {

            List bonds = map.bonds(a);
            if (bonds.size() == 2) {

                // (s)tart/(e)nd of cumulated system: -s=a=e-
                IAtom s = bonds.get(0).getOther(a);
                IAtom e = bonds.get(1).getOther(a);
                // need the parents to re-use the double bond encoder
                IAtom sParent = a;
                IAtom eParent = a;

                visited.add(a);
                visited.add(s);
                visited.add(e);

                int size = 2;

                // expand out from 'l'
                while (s != null && map.cumulated(s)) {
                    IAtom p = map.bonds(s).get(0).getOther(s);
                    IAtom q = map.bonds(s).get(1).getOther(s);
                    sParent = s;
                    s = visited.add(p) ? p : visited.add(q) ? q : null;
                    size++;
                }

                // expand from 'r'
                while (e != null && map.cumulated(e)) {
                    IAtom p = map.bonds(e).get(0).getOther(e);
                    IAtom q = map.bonds(e).get(1).getOther(e);
                    eParent = e;
                    e = visited.add(p) ? p : visited.add(q) ? q : null;
                    size++;
                }

                // s and e are null if we had a cumulative cycle...
                if (s != null && e != null) {

                    // system has now be expanded, size is the number of double
                    // bonds. For odd numbers we use E/Z whilst for even are
                    // axial M/P.
                    //  \           /
                    //   s = = = = e
                    //  /           \
                    if (isOdd(size)) {
                        StereoEncoder encoder = GeometricDoubleBondEncoderFactory.newEncoder(container, s, sParent, e,
                                eParent, graph);
                        if (encoder != null) {
                            encoders.add(encoder);
                        }
                    } else {
                        StereoEncoder encoder = axialEncoder(container, s, e);
                        if (encoder != null) {
                            encoders.add(encoder);
                        }
                    }
                }
            }
        }

        return encoders.isEmpty() ? StereoEncoder.EMPTY : new MultiStereoEncoder(encoders);
    }

    /**
     * Create an encoder for axial 2D stereochemistry for the given start and
     * end atoms.
     *
     * @param container the molecule
     * @param start     start of the cumulated system
     * @param end       end of the cumulated system
     * @return an encoder or null if there are no coordinated
     */
    static StereoEncoder axialEncoder(IAtomContainer container, IAtom start, IAtom end) {

        List startBonds = container.getConnectedBondsList(start);
        List endBonds = container.getConnectedBondsList(end);

        if (startBonds.size() < 2 || endBonds.size() < 2) return null;

        if (has2DCoordinates(startBonds) && has2DCoordinates(endBonds)) {
            return axial2DEncoder(container, start, startBonds, end, endBonds);
        } else if (has3DCoordinates(startBonds) && has3DCoordinates(endBonds)) {
            return axial3DEncoder(container, start, startBonds, end, endBonds);
        }

        return null;
    }

    /**
     * Create an encoder for axial 2D stereochemistry for the given start and
     * end atoms.
     *
     * @param container  the molecule
     * @param start      start of the cumulated system
     * @param startBonds bonds connected to the start
     * @param end        end of the cumulated system
     * @param endBonds   bonds connected to the end
     * @return an encoder
     */
    private static StereoEncoder axial2DEncoder(IAtomContainer container, IAtom start, List startBonds,
            IAtom end, List endBonds) {

        Point2d[] ps = new Point2d[4];
        int[] es = new int[4];

        PermutationParity perm = new CombinedPermutationParity(fill2DCoordinates(container, start, startBonds, ps, es,
                0), fill2DCoordinates(container, end, endBonds, ps, es, 2));

        GeometricParity geom = new Tetrahedral2DParity(ps, es);

        int u = container.indexOf(start);
        int v = container.indexOf(end);

        return new GeometryEncoder(new int[]{u, v}, perm, geom);
    }

    /**
     * Create an encoder for axial 3D stereochemistry for the given start and
     * end atoms.
     *
     * @param container  the molecule
     * @param start      start of the cumulated system
     * @param startBonds bonds connected to the start
     * @param end        end of the cumulated system
     * @param endBonds   bonds connected to the end
     * @return an encoder
     */
    private static StereoEncoder axial3DEncoder(IAtomContainer container, IAtom start, List startBonds,
            IAtom end, List endBonds) {

        Point3d[] coordinates = new Point3d[4];

        PermutationParity perm = new CombinedPermutationParity(fill3DCoordinates(container, start, startBonds,
                coordinates, 0), fill3DCoordinates(container, end, endBonds, coordinates, 2));

        GeometricParity geom = new Tetrahedral3DParity(coordinates);

        int u = container.indexOf(start);
        int v = container.indexOf(end);

        return new GeometryEncoder(new int[]{u, v}, perm, geom);
    }

    /**
     * Fill the {@literal coordinates} and {@literal elevation} from the given
     * offset index. If there is only one connection then the second entry (from
     * the offset) will use the coordinates of a. The permutation parity
     * is also built and returned.
     *
     * @param container   atom container
     * @param a           the central atom
     * @param connected   bonds connected to the central atom
     * @param coordinates the coordinates array to fill
     * @param elevations  the elevations of the connected atoms
     * @param offset      current location in the offset array
     * @return the permutation parity
     */
    private static PermutationParity fill2DCoordinates(IAtomContainer container, IAtom a, List connected,
            Point2d[] coordinates, int[] elevations, int offset) {

        int i = 0;
        coordinates[offset + 1] = a.getPoint2d();
        elevations[offset + 1] = 0;
        int[] indices = new int[2];

        for (IBond bond : connected) {
            if (!isDoubleBond(bond)) {
                IAtom other = bond.getOther(a);
                coordinates[i + offset] = other.getPoint2d();
                elevations[i + offset] = elevation(bond, a);
                indices[i] = container.indexOf(other);
                i++;
            }
        }

        if (i == 1) {
            return PermutationParity.IDENTITY;
        } else {
            return new BasicPermutationParity(indices);
        }

    }

    /**
     * Fill the {@literal coordinates} from the given offset index. If there is
     * only one connection then the second entry (from the offset) will use the
     * coordinates of a. The permutation parity is also built and
     * returned.
     *
     * @param container   atom container
     * @param a           the central atom
     * @param connected   bonds connected to the central atom
     * @param coordinates the coordinates array to fill
     * @param offset      current location in the offset array
     * @return the permutation parity
     */
    private static PermutationParity fill3DCoordinates(IAtomContainer container, IAtom a, List connected,
            Point3d[] coordinates, int offset) {

        int i = 0;
        int[] indices = new int[2];

        for (IBond bond : connected) {
            if (!isDoubleBond(bond)) {
                IAtom other = bond.getOther(a);
                coordinates[i + offset] = other.getPoint3d();
                indices[i] = container.indexOf(other);
                i++;
            }
        }

        // only one connection, use the coordinate of 'a'
        if (i == 1) {
            coordinates[offset + 1] = a.getPoint3d();
            return PermutationParity.IDENTITY;
        } else {
            return new BasicPermutationParity(indices);
        }
    }

    /**
     * Check if all atoms in the bond list have 2D coordinates. There is some
     * redundant checking but the list will typically be short.
     *
     * @param bonds the bonds to check
     * @return whether all atoms have 2D coordinates
     */
    private static boolean has2DCoordinates(List bonds) {
        for (IBond bond : bonds) {
            if (bond.getBegin().getPoint2d() == null || bond.getEnd().getPoint2d() == null) return false;
        }
        return true;
    }

    /**
     * Check if all atoms in the bond list have 3D coordinates. There is some
     * redundant checking but the list will typically be short.
     *
     * @param bonds the bonds to check
     * @return whether all atoms have 2D coordinates
     */
    private static boolean has3DCoordinates(List bonds) {
        for (IBond bond : bonds) {
            if (bond.getBegin().getPoint3d() == null || bond.getEnd().getPoint3d() == null) return false;
        }
        return true;
    }

    /**
     * Access the elevation of a bond relative to the given source atom. With a
     * wedge bond if the atom a is the point end then the bond
     * comes off the paper above the plane. If a is the fat
     * end then the bond from a goes below the plane.
     *
     * @param bond a bond
     * @param a    an atom
     * @return elevation of bond
     */
    static int elevation(IBond bond, IAtom a) {
        return bond.getBegin().equals(a) ? elevation(bond) : elevation(bond) * -1;
    }

    /**
     * Access the elevation of a bond.
     *
     * @param bond the bond
     * @return +1 above the plane, 0 in the plane (default) or -1 below the
     *         plane
     */
    static int elevation(IBond bond) {
        IBond.Stereo stereo = bond.getStereo();
        if (stereo == null) return 0;
        switch (stereo) {
            case UP:
            case DOWN_INVERTED:
                return +1;
            case DOWN:
            case UP_INVERTED:
                return -1;
            default:
                return 0;
        }
    }

    /**
     * Is the value x odd?
     *
     * @param x an int value
     * @return whether x is odd
     */
    private static boolean isOdd(int x) {
        return (x & 0x1) != 0;
    }

    /**
     * Determine whether the bond order is 'double'.
     *
     * @param bond a bond
     * @return the bond is a double bond.
     */
    private static boolean isDoubleBond(IBond bond) {
        return IBond.Order.DOUBLE.equals(bond.getOrder());
    }

    /**
     * Helper class for storing a lookup of atoms and their connected double
     * bonds.
     */
    private static class BondMap {

        private Map> bonds;

        /**
         * Create new bond map for the specified number of atoms.
         *
         * @param n atom count
         */
        BondMap(int n) {
            bonds = new HashMap>(n > 3 ? n + (n / 3) : n);
        }

        /**
         * List of bonds involving the atom.
         *
         * @param a atom
         * @return list of bonds, empty if none stored
         */
        public List bonds(IAtom a) {
            List bs = bonds.get(a);
            return bs != null ? bs : Collections. emptyList();
        }

        /**
         * Check whether the the atom is cumulated - two consecutive double
         * bonds.
         *
         * @param a an atom
         * @return whether the atom is cumulated
         */
        public boolean cumulated(IAtom a) {
            return bonds(a).size() == 2;
        }

        /**
         * Add a bond to the map.
         *
         * @param bond the bond to add
         */
        public void add(IBond bond) {
            add(bond.getBegin(), bond);
            add(bond.getEnd(), bond);
        }

        /**
         * Add the bond for the provided atom.
         *
         * @param a an atom of the bond
         * @param b the bond
         */
        private void add(IAtom a, IBond b) {
            if (bonds(a).isEmpty()) {
                bonds.put(a, new ArrayList(2));
            }
            bonds.get(a).add(b);
        }

        /**
         * Set of atoms which have double bonds.
         *
         * @return iterable set of atoms
         */
        public Iterable atoms() {
            return bonds.keySet();
        }
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy