[Git][java-team/openchemlib][upstream] New upstream version 2021.10.1+dfsg

Andrius Merkys (@merkys) gitlab at salsa.debian.org
Tue Oct 26 07:21:13 BST 2021



Andrius Merkys pushed to branch upstream at Debian Java Maintainers / openchemlib


Commits:
5c2ee3a2 by Andrius Merkys at 2021-10-26T01:43:00-04:00
New upstream version 2021.10.1+dfsg
- - - - -


18 changed files:

- pom.xml
- src/main/java/com/actelion/research/chem/AromaticityResolver.java
- src/main/java/com/actelion/research/chem/Molecule.java
- src/main/java/com/actelion/research/chem/SmilesParser.java
- src/main/java/com/actelion/research/chem/alignment3d/KabschAlignment.java
- src/main/java/com/actelion/research/chem/docking/DockingEngine.java
- src/main/java/com/actelion/research/chem/docking/LigandPose.java
- src/main/java/com/actelion/research/chem/docking/receptorpharmacophore/NegativeReceptorImage.java
- src/main/java/com/actelion/research/chem/docking/scoring/AbstractScoringEngine.java
- + src/main/java/com/actelion/research/chem/forcefield/mmff/MMFFExternalPositionConstraint.java
- src/main/java/com/actelion/research/chem/forcefield/mmff/PositionConstraint.java → src/main/java/com/actelion/research/chem/forcefield/mmff/MMFFPositionConstraint.java
- src/main/java/com/actelion/research/chem/io/CompoundFileHelper.java
- src/main/java/com/actelion/research/chem/mcs/MCS.java
- − src/main/java/com/actelion/research/chem/mcs/SubStructSearchBondIndex.java
- src/main/java/com/actelion/research/chem/mcs/SubStructSearchExhaustiveTreeWalker.java
- src/main/java/com/actelion/research/chem/phesa/pharmacophore/PharmacophoreCalculator.java
- src/main/java/com/actelion/research/chem/phesaflex/FlexibleShapeAlignment.java
- src/main/java/com/actelion/research/chem/potentialenergy/PositionConstraint.java


Changes:

=====================================
pom.xml
=====================================
@@ -8,7 +8,7 @@
     Please follow the naming scheme YEAR.MONTH.RELEASE_NO_OF_MONTH
     (eg. 2016.4.1 for second release in Apr 2016)
     -->
-    <version>2021.10.0</version>
+    <version>2021.10.1</version>
 
     <name>OpenChemLib</name>
     <description>Open Source Chemistry Library</description>
@@ -195,7 +195,7 @@
         <connection>scm:git:git at github.com:Actelion/openchemlib.git</connection>
         <developerConnection>scm:git:git at github.com:Actelion/openchemlib.git</developerConnection>
         <url>https://github.com/Actelion/openchemlib</url>
-      <tag>openchemlib-2021.10.0</tag>
+      <tag>openchemlib-2021.10.1</tag>
   </scm>
 
     <distributionManagement>


=====================================
src/main/java/com/actelion/research/chem/AromaticityResolver.java
=====================================
@@ -332,8 +332,9 @@ public class AromaticityResolver {
 			if (mIsDelocalizedAtom[atom]
 			 && mMol.getLowestFreeValence(atom) == 0
 			 && (!mayChangeAtomCharges
-			  || !mMol.isElectronegative(atom)
-			  || mMol.getAtomCharge(atom) > 0))
+			  || (mMol.getAtomicNo(atom) == 5 && mMol.getAtomCharge(atom) < 0)
+			  || (mMol.getAtomicNo(atom) == 6 || mMol.getAtomicNo(atom) == 14)
+			  || (mMol.isElectronegative(atom) && mMol.getAtomCharge(atom) > 0)))
 				protectAtom(atom);
 		}
 
@@ -679,39 +680,23 @@ public class AromaticityResolver {
 	 */
 	private boolean checkAtomTypePi1(int atom, boolean correctCharge) {
 		int atomicNo = mMol.getAtomicNo(atom);
-		if ((atomicNo >=5 && atomicNo <= 8)
+		if ((atomicNo >= 5 && atomicNo <= 8)
 		 || atomicNo == 15 || atomicNo == 16 || atomicNo == 33 || atomicNo == 34 || atomicNo == 52) {	// P,S,As,Se,Te
 
-// Old logic seems fishy to me; TLS 10Dec2020
-//			int freeValence = mMol.getFreeValence(atom);
-//			if (freeValence == 1 || freeValence == 2)	// we allow one more free valence, because the atom may have a missing charge
-//  			return true;
-
 			int freeValence = mMol.getLowestFreeValence(atom);
 			if (freeValence != 0)
 				return true;
 
-			if (mMol.getAtomCharge(atom) == 0) {
-				if ((atomicNo == 15 || atomicNo == 33) /* && freeValence == 3 */) {
-					if (correctCharge)
-						mMol.setAtomCharge(atom, 1);
-					return true;
-					}
-				if ((atomicNo == 16 || atomicNo == 34 || atomicNo == 52) /* && freeValence == 4 */) {
-					if (correctCharge)
-						mMol.setAtomCharge(atom, 1);
-					return true;
-					}
-				if (atomicNo == 5 /* && freeValence == 0 */) {
-					if (correctCharge)
-						mMol.setAtomCharge(atom, -1);
-					return true;
-					}
-				if ((atomicNo == 7 || atomicNo == 8) /* && freeValence == 0 */) {
-					if (correctCharge)
-						mMol.setAtomCharge(atom, 1);
-					return true;
-					}
+			int charge = mMol.getAtomCharge(atom);
+			if (atomicNo == 5 && charge >= 0) {
+				if (correctCharge)
+					mMol.setAtomCharge(atom, charge-1);
+				return true;
+				}
+			if (atomicNo != 5 && charge <= 0) {
+				if (correctCharge)
+					mMol.setAtomCharge(atom, charge+1);
+				return true;
 				}
 			}
 


=====================================
src/main/java/com/actelion/research/chem/Molecule.java
=====================================
@@ -370,7 +370,7 @@ public class Molecule implements Serializable {
 	public static final int cDefaultAtomValence = 6;
 	private static final byte[] cDefaultAtomValences = { cDefaultAtomValence };
 	private static final byte[] cAminoAcidValences = { 2 };
-	private static final byte[][] cAtomValence = {null,
+	public static final byte[][] cAtomValence = {null,
 			{1}, {0}, {1}, {2}, {3}, {4}, {3}, {2}, {1}, {0},			// H to Ne
 			{1}, {2}, {3}, {4}, {3, 5}, {2, 4, 6}, {1, 3, 5, 7}, {0},	// Na to Ar
 			{1}, {2}, null, null, null, null, null, null, null, null,	// K to Ni


=====================================
src/main/java/com/actelion/research/chem/SmilesParser.java
=====================================
@@ -50,6 +50,7 @@ public class SmilesParser {
 	public static final int SMARTS_MODE_IS_SMARTS = 2;
 
 	public static final int MODE_SKIP_COORDINATE_TEMPLATES = 4;
+	public static final int MODE_MAKE_HYDROGEN_EXPLICIT = 8;
 
 	private static final int INITIAL_CONNECTIONS = 16;
 	private static final int MAX_CONNECTIONS = 100; // largest allowed one in SMILES is 99
@@ -65,7 +66,7 @@ public class SmilesParser {
 	private StereoMolecule mMol;
 	private boolean[] mIsAromaticBond;
 	private int mAromaticAtoms,mAromaticBonds,mSmartsMode,mCoordinateMode;
-	private boolean mCreateSmartsWarnings,mSkipTemplates;
+	private boolean mCreateSmartsWarnings, mMakeHydrogenExplicit;
 	private StringBuilder mSmartsWarningBuffer;
 
 	/**
@@ -91,11 +92,12 @@ public class SmilesParser {
 	public SmilesParser(int mode, boolean createSmartsWarnings) {
 		mSmartsMode = mode & SMARTS_MODE_MASK;
 		mCreateSmartsWarnings = createSmartsWarnings;
+		mMakeHydrogenExplicit = ((mode & MODE_MAKE_HYDROGEN_EXPLICIT) != 0);
 		mCoordinateMode = CoordinateInventor.MODE_DEFAULT;
 		if ((mode & MODE_SKIP_COORDINATE_TEMPLATES) != 0)
 			mCoordinateMode |= CoordinateInventor.MODE_SKIP_DEFAULT_TEMPLATES;
-
-		mSkipTemplates = ((mode & MODE_SKIP_COORDINATE_TEMPLATES) != 0);
+		if (mMakeHydrogenExplicit)
+			mCoordinateMode &= ~CoordinateInventor.MODE_REMOVE_HYDROGEN;
 		}
 
 	public StereoMolecule parseMolecule(String smiles) {
@@ -624,7 +626,7 @@ public class SmilesParser {
 
 				// mark aromatic atoms
 				if (Character.isLowerCase(theChar)) {
-					if (atomicNo != 5 && atomicNo != 6 && atomicNo != 7 && atomicNo != 8 && atomicNo != 15 &&atomicNo != 16)
+					if (atomicNo != 5 && atomicNo != 6 && atomicNo != 7 && atomicNo != 8 && atomicNo != 15 &&atomicNo != 16 && atomicNo != 33  && atomicNo != 34)
 						throw new Exception("SmilesParser: atomicNo "+atomicNo+" must not be aromatic");
 
 					mMol.setAtomMarker(atom, true);
@@ -637,7 +639,7 @@ public class SmilesParser {
 				// put explicitHydrogen into atomCustomLabel to keep atom-relation when hydrogens move to end of atom list in handleHydrogen()
 				if (explicitHydrogens != HYDROGEN_ANY && atomicNo != 1) {	// no custom labels for hydrogen to get useful results in getHandleHydrogenMap()
 					byte[] bytes = new byte[1];
-					bytes[0] = (byte)explicitHydrogens;
+					bytes[0] = (byte)(explicitHydrogens == HYDROGEN_IMPLICIT_ZERO ? 0 : explicitHydrogens);
 					mMol.setAtomCustomLabel(atom, bytes);
 					}
 
@@ -908,7 +910,11 @@ public class SmilesParser {
 			if (mMol.getAtomCustomLabel(atom) != null) {	// if we have the exact number of hydrogens
 				int explicitHydrogen = mMol.getAtomCustomLabelBytes(atom)[0];
 
-				if (smartsFeatureFound || mSmartsMode == SMARTS_MODE_IS_SMARTS) {
+				if (mMakeHydrogenExplicit) {
+					for (int i=0; i<explicitHydrogen; i++)
+						mMol.addBond(atom, mMol.addAtom(1), 1);
+					}
+				else if (smartsFeatureFound || mSmartsMode == SMARTS_MODE_IS_SMARTS) {
 					if (explicitHydrogen == 0)
 						mMol.setAtomQueryFeature(atom, Molecule.cAtomQFHydrogen & ~Molecule.cAtomQFNot0Hydrogen, true);
 					if (explicitHydrogen == 1)
@@ -919,9 +925,6 @@ public class SmilesParser {
 						mMol.setAtomQueryFeature(atom, Molecule.cAtomQFHydrogen & ~Molecule.cAtomQFNot3Hydrogen, true);
 					}
 				else {
-					if (explicitHydrogen == HYDROGEN_IMPLICIT_ZERO)
-						explicitHydrogen = 0;
-
 					if (!mMol.isMarkedAtom(atom)) {
 						// We don't correct aromatic atoms, because for aromatic atoms the number of
 						// explicit hydrogens encodes whether a pi-bond needs to be placed at the atom
@@ -954,8 +957,24 @@ public class SmilesParser {
 						}
 					}
 				}
+			else if (!mMakeHydrogenExplicit && (smartsFeatureFound || mSmartsMode == SMARTS_MODE_IS_SMARTS)) {
+				// if we don't have a hydrogen count on the atom, but we have explicit hydrogen atoms
+				// and if we decode a SMARTS, then we convert explicit hydrogens into an 'at least n hydrogen'
+				int explicitHydrogen = mMol.getExplicitHydrogens(atom);
+				if (explicitHydrogen >= 1)
+					mMol.setAtomQueryFeature(atom, Molecule.cAtomQFNot0Hydrogen, true);
+				if (explicitHydrogen >= 2)
+					mMol.setAtomQueryFeature(atom, Molecule.cAtomQFNot1Hydrogen, true);
+				if (explicitHydrogen >= 3)
+					mMol.setAtomQueryFeature(atom, Molecule.cAtomQFNot2Hydrogen, true);
+				if (explicitHydrogen >= 4)
+					mMol.setAtomQueryFeature(atom, Molecule.cAtomQFNot3Hydrogen, true);
+				}
 			}
 
+		if (!mMakeHydrogenExplicit && (smartsFeatureFound || mSmartsMode == SMARTS_MODE_IS_SMARTS))
+			mMol.removeExplicitHydrogens();
+
 		mMol.ensureHelperArrays(Molecule.cHelperNeighbours);
 
 		correctValenceExceededNitrogen();	// convert pyridine oxides and nitro into polar structures with valid nitrogen valences
@@ -1343,9 +1362,8 @@ public class SmilesParser {
 		 || !mMol.isMarkedAtom(atom))	// already marked as hetero-atom of another ring
 			return false;
 
-		int explicitHydrogens = (mMol.getAtomCustomLabel(atom) == null
-							  || mMol.getAtomCustomLabelBytes(atom)[0] == HYDROGEN_IMPLICIT_ZERO) ?
-											0 : mMol.getAtomCustomLabelBytes(atom)[0];
+		int explicitHydrogens = (mMol.getAtomCustomLabel(atom) == null) ?
+								0 : mMol.getAtomCustomLabelBytes(atom)[0];
 		int freeValence = mMol.getFreeValence(atom) - explicitHydrogens;
 		if (freeValence < 1)
 			return false;


=====================================
src/main/java/com/actelion/research/chem/alignment3d/KabschAlignment.java
=====================================
@@ -66,9 +66,14 @@ public class KabschAlignment {
 		for(Coordinates c2 : coords2)
 			c2.sub(com2);
 		
-		trans1 = com2.scaleC(-1.0);
-		trans2 = new Coordinates(com1);
-
+		Coordinates t1 = com2.scaleC(-1.0);
+		trans1.x = t1.x;
+		trans1.y = t1.y;
+		trans1.z = t1.z;
+		
+		trans2.x = com1.x;
+		trans2.y = com1.y;
+		trans2.z = com1.z;
 		
 		Matrix m = new Matrix(3,3);
 		double [][] c1 = Arrays.stream(coords1).map(e -> new double[] {e.x,e.y,e.z}).toArray(double[][]::new);
@@ -97,15 +102,15 @@ public class KabschAlignment {
 		ma.set(2,2,det);
 		
 		
-		rot = ma.multiply(ut);
-		rot = v.multiply(rot);
-		assert(rot.det()>0.0);
-		rot = rot.getTranspose();
+		Matrix r = ma.multiply(ut);
+		r = v.multiply(r);
+		assert(r.det()>0.0);
+		r = r.getTranspose();
 		
 		
 		
 	    for(Coordinates c : coords2) {
-	    	c.rotate(rot.getArray());
+	    	c.rotate(r.getArray());
 	    	c.add(com1);
 	    }
 	    
@@ -113,6 +118,8 @@ public class KabschAlignment {
 	    	c.add(com1);
 	    }
 	    
+	    rot.set(r.getArray());
+	    
 
 		
 	}


=====================================
src/main/java/com/actelion/research/chem/docking/DockingEngine.java
=====================================
@@ -5,15 +5,20 @@ import java.util.ArrayList;
 import java.util.Base64;
 import java.util.HashMap;
 import java.util.HashSet;
+import java.util.Iterator;
+import java.util.LinkedHashMap;
 import java.util.List;
 import java.util.Map;
 import java.util.Random;
 import java.util.Set;
 import java.util.Base64.Decoder;
 import java.util.Base64.Encoder;
+import java.util.Map.Entry;
+import java.util.stream.Collectors;
 
 import org.openmolecules.chem.conf.gen.ConformerGenerator;
 
+import com.actelion.research.calc.Matrix;
 import com.actelion.research.calc.ThreadMaster;
 import com.actelion.research.chem.Canonizer;
 import com.actelion.research.chem.Coordinates;
@@ -21,7 +26,9 @@ import com.actelion.research.chem.IDCodeParser;
 import com.actelion.research.chem.IDCodeParserWithoutCoordinateInvention;
 import com.actelion.research.chem.Molecule;
 import com.actelion.research.chem.Molecule3D;
+import com.actelion.research.chem.SSSearcher;
 import com.actelion.research.chem.StereoMolecule;
+import com.actelion.research.chem.alignment3d.KabschAlignment;
 import com.actelion.research.chem.alignment3d.transformation.ExponentialMap;
 import com.actelion.research.chem.alignment3d.transformation.Quaternion;
 import com.actelion.research.chem.alignment3d.transformation.Rotation;
@@ -35,15 +42,18 @@ import com.actelion.research.chem.docking.scoring.AbstractScoringEngine;
 import com.actelion.research.chem.docking.scoring.ChemPLP;
 import com.actelion.research.chem.docking.scoring.IdoScore;
 import com.actelion.research.chem.docking.shape.ShapeDocking;
+import com.actelion.research.chem.forcefield.mmff.MMFFExternalPositionConstraint;
 import com.actelion.research.chem.forcefield.mmff.ForceFieldMMFF94;
-import com.actelion.research.chem.forcefield.mmff.PositionConstraint;
+import com.actelion.research.chem.forcefield.mmff.MMFFPositionConstraint;
 import com.actelion.research.chem.interactionstatistics.InteractionAtomTypeCalculator;
 import com.actelion.research.chem.io.pdb.converter.MoleculeGrid;
+import com.actelion.research.chem.mcs.MCS;
 import com.actelion.research.chem.optimization.OptimizerLBFGS;
 import com.actelion.research.chem.phesa.EncodeFunctions;
 import com.actelion.research.chem.phesa.MolecularVolume;
 import com.actelion.research.chem.phesa.PheSAAlignment;
 import com.actelion.research.chem.phesa.ShapeVolume;
+import com.actelion.research.chem.potentialenergy.PositionConstraint;
 import com.actelion.research.chem.phesa.PheSAAlignment.PheSAResult;
 
 public class DockingEngine {
@@ -62,6 +72,7 @@ public class DockingEngine {
 	public static final double GRID_DIMENSION = 6.0;
 	public static final double GRID_RESOLUTION = 0.5;
 	public static final double MINI_CUTOFF = 100; //if energy higher after MC step, don't minimize
+	public static final int MCS_EXHAUSTIVENESS = 3; //MCS docking requires more starting positions, but less MC sampling (reduced DOF)
 	
 
 	private Rotation rotation; //for initial prealignment to native ligand
@@ -73,6 +84,9 @@ public class DockingEngine {
 	private StereoMolecule nativeLigand;
 	private ShapeDocking shapeDocking;
 	private ThreadMaster threadMaster;
+	private StereoMolecule mcsRef;
+	private List<Integer> mcsConstrainedBonds;
+	private List<Integer> mcsConstrainedAtoms;
 
 	
 	public DockingEngine(StereoMolecule rec, StereoMolecule nativeLig, int mcSteps, int startPositions,
@@ -122,20 +136,24 @@ public class DockingEngine {
 		threadMaster = tm;
 	}
 	
+	
+	/**
+	 * generate initial poses: 
+	 * 1) shape docking into the negative receptor image
+	 * 2) constrained optimization of initial poses to reduce strain energy
+	 * @param mol
+	 * @param initialPos
+	 * @return
+	 * @throws DockingFailedException
+	 */
 	private double getStartingPositions(StereoMolecule mol, List<Conformer> initialPos) throws DockingFailedException {
 
 		double eMin = Double.MAX_VALUE;
 		Map<String, Object> ffOptions = new HashMap<String, Object>();
 		ffOptions.put("dielectric constant", 80.0);
-		//ffOptions.put("angle bend", false);
-		//ffOptions.put("stretch bend", false);
-		//ffOptions.put("bond stretch", false);
-		//ffOptions.put("out of plane", false);
 
 		ConformerSet confSet = new ConformerSet();
-		long t0 = System.currentTimeMillis();
 		List<StereoMolecule> alignedMol = shapeDocking.dock(mol);
-		long t1 = System.currentTimeMillis();
 		alignedMol.stream().forEach(e -> confSet.add(new Conformer(e)));
 		for(Conformer c : confSet) {
 			if(c!=null) {
@@ -149,7 +167,7 @@ public class DockingEngine {
 				catch(Exception e) {
 					throw new DockingFailedException("could not assess atom types");
 				}
-				PositionConstraint constraint = new PositionConstraint(conf,50,0.2);
+				MMFFPositionConstraint constraint = new MMFFPositionConstraint(conf,50,0.2);
 				mmff.addEnergyTerm(constraint);
 				mmff.minimise();
 				Conformer ligConf = new Conformer(conf);
@@ -168,7 +186,136 @@ public class DockingEngine {
 		return eMin;
 		
 	}
+	/**
+	 * 1) find MCS between reference molecule and candidate molecule
+	 * 2) 
+	 * @param mol
+	 */
+	private double getStartingPositionsMCS(StereoMolecule mol, List<Conformer> initialPos) {
+		double eMin = Double.MAX_VALUE;
+		int rotBondsMol = mol.getRotatableBondCount();
+		int rotBondsRef = nativeLigand.getRotatableBondCount();
+		MCS mcs = new MCS();
+		//in MCS, the first molecule should have more rotatable bonds than the second 
+		boolean[] bondMCSMol = null;
+		boolean[] bondMCSFrag = null;
+		boolean isNativeLigFrag;//if native ligand is smaller than the candidate molecule
+		if(rotBondsMol > rotBondsRef) {
+			mcs.set(mol,nativeLigand);
+			bondMCSMol = new boolean[mol.getAllBonds()];
+			bondMCSFrag = new boolean[nativeLigand.getAllBonds()];
+			isNativeLigFrag = true;
+		}
+		else {
+			mcs.set(mol,nativeLigand);
+			bondMCSMol = new boolean[mol.getAllBonds()];
+			bondMCSFrag = new boolean[nativeLigand.getAllBonds()];
+			isNativeLigFrag = false;
+		}
+
+		StereoMolecule mcsMol = mcs.getMCS();
+		SSSearcher sss = new SSSearcher();
+		sss.setFragment(mcsMol);
+		sss.setMolecule(nativeLigand);
+		sss.findFragmentInMolecule(SSSearcher.cCountModeOverlapping, SSSearcher.cMatchDBondToDelocalized);
+		int[] map1 = sss.getMatchList().get(0);
+		sss.setMolecule(mol);
+		sss.findFragmentInMolecule(SSSearcher.cCountModeOverlapping, SSSearcher.cMatchDBondToDelocalized);
+		int[] map2 = sss.getMatchList().get(0);
+		mcsConstrainedAtoms = new ArrayList<>();
+		for(int a: map2) {
+			mcsConstrainedAtoms.add(a);
+		}
+		//map from reference to mol
+		Map<Integer,Integer> map = new HashMap<Integer,Integer>();
+		for(int i=0;i<map1.length;i++) {
+			map.put(map1[i], map2[i]);
+		}
+		//find bonds not part of MCS
+		mcs.getMCSBondArray(bondMCSMol, bondMCSFrag);
+		mcsConstrainedBonds = new ArrayList<>();
+		boolean[] bondArray = null;
+		if(isNativeLigFrag) 
+			bondArray = bondMCSMol;
+		else 
+			bondArray = bondMCSFrag;
+		for(int b=0;b<bondArray.length;b++) {
+				if(bondArray[b])
+					mcsConstrainedBonds.add(b);
+		}
+		ForceFieldMMFF94.initialize(ForceFieldMMFF94.MMFF94SPLUS);
+		Map<String, Object> ffOptions = new HashMap<String, Object>();
+		ffOptions.put("dielectric constant", 80.0);
+		ConformerSetGenerator csGen = new ConformerSetGenerator();
+		ConformerSet confSet = csGen.generateConformerSet(mol);
+		Map<Conformer,Double> confsWithRMSDs = new HashMap<Conformer,Double>();
+		for(Conformer conf : confSet) {
+			//align MCS using Kabsch algorithm
+			Coordinates[] coords1 = new Coordinates[map1.length];
+			Coordinates[] coords2 = new Coordinates[map1.length];
+			int counter = 0;
+			for(int key : map.keySet()) {
+				coords1[counter] = new Coordinates(nativeLigand.getCoordinates(key));
+				coords2[counter] = new Coordinates(conf.getCoordinates(map.get(key)));
+				counter++;
+			}
+			int[][] mapping = new int[coords1.length][2];
+			counter = 0;
+			for(int[] m : mapping) {
+				m[0] = counter;
+				m[1] = counter;
+				counter++;
+			}
+			Coordinates trans1 = new Coordinates();
+			Matrix rot = new Matrix(3,3); 
+			Coordinates trans2 = new Coordinates();
+			KabschAlignment alignment = new KabschAlignment(coords1,coords2,mapping);
+			alignment.align(trans1,rot,trans2);
+			double rmsd = getCoreRMSD(coords1,coords2);
+			for(Coordinates coord : conf.getCoordinates()) {
+				coord.add(trans1);
+				coord.rotate(rot.getArray());
+				coord.add(trans2);
+			}
+			confsWithRMSDs.put(conf, rmsd);
+		}
+		
+		LinkedHashMap<Conformer,Double> sortedConfs = confsWithRMSDs.entrySet().stream().sorted(Map.Entry.<Conformer,Double>comparingByValue()).collect(Collectors.toMap(Map.Entry::getKey,Map.Entry::getValue,
+				(e1, e2) -> e1, LinkedHashMap::new));
+		Iterator<Entry<Conformer,Double>> iterator = sortedConfs.entrySet().iterator();
+		boolean done = false;
+		Entry<Conformer,Double> entry;
+		int c=0;
+		while(!done && c<startPositions*MCS_EXHAUSTIVENESS) {
+			c++;
+			try	{
+				entry = iterator.next();
+			}
+			catch(Exception e) {
+				done = true;
+				continue;
+			}
+			Conformer conf = entry.getKey();
+			StereoMolecule aligned = new StereoMolecule();
+			aligned = conf.toMolecule(null);
+			aligned.ensureHelperArrays(Molecule.cHelperParities);
+			ForceFieldMMFF94 mmff;
+			List<MMFFExternalPositionConstraint> constraints = new ArrayList<>();
+			MMFFPositionConstraint constraint = new MMFFPositionConstraint(aligned,50,0.2);
+			mmff = new ForceFieldMMFF94(aligned, ForceFieldMMFF94.MMFF94SPLUS,ffOptions);
+			mmff.addEnergyTerm(constraint);
+			mmff.minimise();
+			double e = mmff.getTotalEnergy();
+			for(int a=0;a<aligned.getAllAtoms();a++) {
+				conf.setCoordinates(a, new Coordinates(aligned.getCoordinates(a)));
+			}
+			initialPos.add(conf);
+			if(e<eMin)
+				eMin = e;
+		}
+		return eMin;
 	
+	}
 	
 	
 	public DockingResult dockMolecule(StereoMolecule mol) throws DockingFailedException {
@@ -177,35 +324,38 @@ public class DockingEngine {
 		if(ForceFieldMMFF94.table(ForceFieldMMFF94.MMFF94SPLUS)==null)
 			ForceFieldMMFF94.initialize(ForceFieldMMFF94.MMFF94SPLUS);
 		List<Conformer> startPoints = new ArrayList<>();
-		double eMin = getStartingPositions(mol, startPoints);
+		double eMin = 0.0;
+		int steps = mcSteps;
+		if(mcsRef!=null) {
+			eMin = getStartingPositionsMCS(mol, startPoints);
+			steps=steps/MCS_EXHAUSTIVENESS;
+		}
+		else {
+			eMin = getStartingPositions(mol, startPoints);
+		}
+
 		Map<String,Double> contributions = null;
 		for(Conformer ligConf : startPoints) {
-			for(double[] transform : PheSAAlignment.initialTransform(0)) {
-				Conformer newLigConf = new Conformer(ligConf);
-				ExponentialMap eMap = new ExponentialMap(transform[0],transform[1],transform[2]);
-				Quaternion q = eMap.toQuaternion();
-				Coordinates com = DockingUtils.getCOM(newLigConf);
-				Rotation rot = new Rotation(q.getRotMatrix().getArray());
-				Translation trans = new Translation(new double[] {transform[3],transform[4],transform[5]});
-				TransformationSequence t = new TransformationSequence();
-				Translation t1 = new Translation(-com.x,-com.y,-com.z);
-				Translation t2 = new Translation(com.x,com.y,com.z);
-				t.addTransformation(t1);
-				t.addTransformation(rot);
-				t.addTransformation(t2);
-				t.addTransformation(trans);
-				t.apply(newLigConf);
-				LigandPose pose = initiate(newLigConf,eMin);
-				double energy = mcSearch(pose);
-				if(energy<bestEnergy) {
-					bestEnergy = energy;
-					bestPose = pose.getLigConf();
-					contributions = pose.getContributions();
+			Conformer newLigConf = new Conformer(ligConf);		
+			LigandPose pose = initiate(newLigConf,eMin);
+			if(mcsRef!=null) {
+				pose.setMCSBondConstraints(mcsConstrainedBonds);
+				for(int a : mcsConstrainedAtoms) {
+					PositionConstraint constr = new PositionConstraint(newLigConf,a,50,1.0);
+					pose.addConstraint(constr);
 				}
-				if(threadMaster!=null && threadMaster.threadMustDie())
-					break;
+				
 			}
+			double energy = mcSearch(pose,steps);
+			if(energy<bestEnergy) {
+				bestEnergy = energy;
+				bestPose = pose.getLigConf();
+				contributions = pose.getContributions();
+			}
+			if(threadMaster!=null && threadMaster.threadMustDie())
+				break;
 		}
+		
 		if(bestPose!=null) {
 			StereoMolecule best = bestPose.toMolecule();
 			Rotation rot = rotation.getInvert();
@@ -222,8 +372,13 @@ public class DockingEngine {
 	
 	
 
-	
-	private double mcSearch(LigandPose pose) {
+	/**
+	 * use monte carlo steps to permute molecular rotation, translation, torsion angles
+	 * promising poses (below a certain cutoff) are optimized
+	 * @param pose
+	 * @return
+	 */
+	private double mcSearch(LigandPose pose, int steps) {
 		double[] bestState = new double[pose.getState().length];
 		double[] oldState = new double[pose.getState().length];
 		double[] state = new double[pose.getState().length];
@@ -236,7 +391,7 @@ public class DockingEngine {
 		oldEnergy = pose.getFGValue(new double[bestState.length]);
 		bestEnergy = oldEnergy;
 	
-		for(int i=0;i<mcSteps;i++) {
+		for(int i=0;i<steps;i++) {
 			pose.randomPerturbation(random);
 			double energyMC = pose.getFGValue(new double[bestState.length]);
 			if(energyMC<MINI_CUTOFF) {
@@ -274,6 +429,8 @@ public class DockingEngine {
 		}
 
 		pose.setState(bestState);
+		pose.removeConstraints();
+		bestEnergy = pose.getFGValue(new double[bestState.length]);
 		return bestEnergy;
 		
 	}
@@ -286,6 +443,10 @@ public class DockingEngine {
 		
 	}
 	
+	public void setMCSReference(StereoMolecule referencePose) {
+		mcsRef = referencePose;
+	}
+	
 	public static void getBindingSiteAtoms(StereoMolecule receptor, Set<Integer> bindingSiteAtoms, MoleculeGrid grid,
 			boolean includeHydrogens) {
 		int[] gridSize = grid.getGridSize();
@@ -341,15 +502,12 @@ public class DockingEngine {
 	public double refineNativePose(double d, double[] coords) {
 		Map<String, Object> ffOptions = new HashMap<String, Object>();
 		ffOptions.put("dielectric constant", 80.0);
-		//ffOptions.put("angle bend", false);
-		//ffOptions.put("stretch bend", false);
-		//ffOptions.put("bond stretch", false);
 		if(ForceFieldMMFF94.table(ForceFieldMMFF94.MMFF94SPLUS)==null)
 			ForceFieldMMFF94.initialize(ForceFieldMMFF94.MMFF94SPLUS);
 		Molecule3D nativePose = new Molecule3D(nativeLigand);
 		new Canonizer(nativePose);
 		ForceFieldMMFF94 mmff = new ForceFieldMMFF94(nativePose, ForceFieldMMFF94.MMFF94SPLUS, ffOptions);
-		PositionConstraint constraint = new PositionConstraint(nativePose,50,0.2);
+		MMFFPositionConstraint constraint = new MMFFPositionConstraint(nativePose,50,0.2);
 		mmff.addEnergyTerm(constraint);
 		mmff.minimise();
 		ConformerSetGenerator confSetGen = new ConformerSetGenerator(100,ConformerGenerator.STRATEGY_LIKELY_RANDOM, false,
@@ -363,7 +521,7 @@ public class DockingEngine {
 				StereoMolecule conf = conformer.toMolecule();
 				conf.ensureHelperArrays(Molecule.cHelperParities);
 				mmff = new ForceFieldMMFF94(conf, ForceFieldMMFF94.MMFF94SPLUS, ffOptions);
-				constraint = new PositionConstraint(conf,50,0.2);
+				constraint = new MMFFPositionConstraint(conf,50,0.2);
 				mmff.addEnergyTerm(constraint);
 				mmff.minimise();
 				Conformer ligConf = new Conformer(conf);
@@ -396,6 +554,18 @@ public class DockingEngine {
 		
 	}
 	
+	private static double getCoreRMSD(Coordinates[] coords1, Coordinates[] coords2) {
+		double rmsd = 0.0;
+		for(int i=0;i<coords1.length;i++) { 
+			Coordinates c1 = coords1[i];
+			Coordinates c2 = coords2[i];
+			rmsd+=c1.distanceSquared(c2);
+		}
+		rmsd/=coords1.length;
+		rmsd = Math.sqrt(rmsd);
+		return rmsd;
+	}
+	
 	public static class DockingResult implements Comparable<DockingResult>  {
 		private double score;
 		private StereoMolecule pose;
@@ -486,6 +656,7 @@ public class DockingEngine {
             return Double.compare( this.score, o.score);
            
 		}
+		
 	}
 	
 


=====================================
src/main/java/com/actelion/research/chem/docking/LigandPose.java
=====================================
@@ -1,7 +1,9 @@
 package com.actelion.research.chem.docking;
 
+import java.util.ArrayList;
 import java.util.Arrays;
 import java.util.HashMap;
+import java.util.List;
 import java.util.Map;
 import java.util.Random;
 
@@ -21,6 +23,7 @@ import com.actelion.research.chem.conf.TorsionDB;
 import com.actelion.research.chem.docking.scoring.AbstractScoringEngine;
 import com.actelion.research.chem.optimization.Evaluable;
 import com.actelion.research.chem.potentialenergy.PositionConstraint;
+import com.actelion.research.chem.potentialenergy.PotentialEnergyTerm;
 
 
 public class LigandPose implements Evaluable{
@@ -39,6 +42,8 @@ public class LigandPose implements Evaluable{
 	private AbstractScoringEngine engine;
 	public static long SEED = 12345L;
 	private Coordinates origCOM;
+	private int[] mcsRotBondIndeces; //for MCS docking, only bonds not part of the MCS are sampled
+	
 	
 	public LigandPose(Conformer ligConf, AbstractScoringEngine engine, double e0) {
 		
@@ -47,6 +52,25 @@ public class LigandPose implements Evaluable{
 		init(e0);
 	}
 	
+	/**
+	 * for MCS docking:create array of bond indices that are allowed to be permuted
+	 * @param constraints
+	 */
+	public void setMCSBondConstraints(List<Integer> constraints) {
+		int[] rotBonds = torsionHelper.getRotatableBonds();
+		List<Integer> allowedIndices = new ArrayList<>();
+		for(int rbIndex=0;rbIndex<rotBonds.length;rbIndex++) {
+			int rb = rotBonds[rbIndex];
+			if(!constraints.contains(rb))
+				allowedIndices.add(rbIndex);
+		}
+		mcsRotBondIndeces = new int[allowedIndices.size()];
+		for(int i=0;i<allowedIndices.size();i++) {
+			mcsRotBondIndeces[i]=allowedIndices.get(i);
+		}
+		
+	}
+	
 	private void init(double e0) {
 		mol = ligConf.getMolecule();
 		//rotationCenter = calcCOM();
@@ -255,46 +279,67 @@ public class LigandPose implements Evaluable{
 	}
 	
 	public void randomPerturbation(Random random) {
-		int num = (int) (3*random.nextDouble());
-		if(num==0) { //translation
-			Coordinates shift = DockingUtils.randomVectorInSphere(random).scale(MOVE_AMPLITUDE);
-			state[0]+=shift.x;
-			state[1]+=shift.y;
-			state[2]+=shift.z;
-			/*
-			for(int a=0;a<ligConf.getMolecule().getAllAtoms();a++) { 
-				Coordinates c = ligConf.getCoordinates(a);
-				c.add(shift);
+		if(mcsRotBondIndeces==null) {
+			int num = (int) (3*random.nextDouble());
+			if(num==0) { //translation
+				Coordinates shift = DockingUtils.randomVectorInSphere(random).scale(MOVE_AMPLITUDE);
+				state[0]+=shift.x;
+				state[1]+=shift.y;
+				state[2]+=shift.z;
+				/*
+				for(int a=0;a<ligConf.getMolecule().getAllAtoms();a++) { 
+					Coordinates c = ligConf.getCoordinates(a);
+					c.add(shift);
+				}
+				*/
 			}
-			*/
-		}
-		else if(num==1) {
-			double r = getGyrationRadius();
-			Coordinates rot = DockingUtils.randomVectorInSphere(random).scale(MOVE_AMPLITUDE/r);
-			double angle = rot.dist();
-			Quaternion q = new Quaternion(1.0,0.0,0.0,0.0);
-			if(angle>0.0001) {
-				Coordinates axis = rot.scale(1.0/angle);
-				q = new Quaternion(axis,angle);
+			else if(num==1) {
+				double r = getGyrationRadius();
+				Coordinates rot = DockingUtils.randomVectorInSphere(random).scale(MOVE_AMPLITUDE/r);
+				double angle = rot.dist();
+				Quaternion q = new Quaternion(1.0,0.0,0.0,0.0);
+				if(angle>0.0001) {
+					Coordinates axis = rot.scale(1.0/angle);
+					q = new Quaternion(axis,angle);
+				}
+				ExponentialMap em = new ExponentialMap(state[3],state[4],state[5]);
+				Quaternion qOrig = em.toQuaternion();
+				q.multiply(qOrig);
+				ExponentialMap emNew = new ExponentialMap(q);
+				state[3] = emNew.getP().x;
+				state[4] = emNew.getP().y;
+				state[5] = emNew.getP().z;
+	
+				
+			}
+			else  {
+				if(torsionHelper.getRotatableBonds().length==0)
+					return;
+				double rnd = random.nextDouble();
+				rnd*=180.0;
+				double rotateBy = random.nextBoolean() ? rnd : -rnd;
+				rotateBy = rotateBy*Math.PI/180.0;
+				int bond = random.nextInt(torsionHelper.getRotatableBonds().length);
+				double previousAngle = state[6+bond];
+				double newAngle = previousAngle+rotateBy; 
+				if(newAngle>Math.PI) {
+					newAngle -= 2*Math.PI;
+				}
+				state[6+bond] = newAngle;
+				
 			}
-			ExponentialMap em = new ExponentialMap(state[3],state[4],state[5]);
-			Quaternion qOrig = em.toQuaternion();
-			q.multiply(qOrig);
-			ExponentialMap emNew = new ExponentialMap(q);
-			state[3] = emNew.getP().x;
-			state[4] = emNew.getP().y;
-			state[5] = emNew.getP().z;
-
-			
 		}
-		else  {
+		//mcs docking
+		else {
 			if(torsionHelper.getRotatableBonds().length==0)
 				return;
+			else if(mcsRotBondIndeces.length==0)
+				return;
 			double rnd = random.nextDouble();
 			rnd*=180.0;
 			double rotateBy = random.nextBoolean() ? rnd : -rnd;
 			rotateBy = rotateBy*Math.PI/180.0;
-			int bond = random.nextInt(torsionHelper.getRotatableBonds().length);
+			int bond = mcsRotBondIndeces[random.nextInt(mcsRotBondIndeces.length)];
 			double previousAngle = state[6+bond];
 			double newAngle = previousAngle+rotateBy; 
 			if(newAngle>Math.PI) {
@@ -302,7 +347,9 @@ public class LigandPose implements Evaluable{
 			}
 			state[6+bond] = newAngle;
 			
+			
 		}
+		
 		updateLigandCoordinates();
 		
 	}
@@ -314,20 +361,20 @@ public class LigandPose implements Evaluable{
 		}
 	}
 	
+	public void addConstraint(PositionConstraint constraint) {
+		engine.addConstraint(constraint);
+	}
+	
+	public void removeConstraints() {
+		engine.removeConstraints();
+	}
+	
+	
 	public Conformer getLigConf() {
 		return ligConf;
 	}
 	
-	private Coordinates calcCOM(){ 
-		Coordinates com = new Coordinates();
-		for(int a=0;a<ligConf.getMolecule().getAtoms();a++){
-			com.add(ligConf.getCoordinates(a));
-		}
-		com.scale(1.0/ligConf.getMolecule().getAtoms());
-		return com;
-		
 
-	}
 	
 
 }


=====================================
src/main/java/com/actelion/research/chem/docking/receptorpharmacophore/NegativeReceptorImage.java
=====================================
@@ -1,37 +1,23 @@
 package com.actelion.research.chem.docking.receptorpharmacophore;
 
-import java.util.ArrayList;
-import java.util.Arrays;
-import java.util.HashMap;
-import java.util.HashSet;
-import java.util.LinkedList;
-import java.util.List;
-import java.util.Map;
-import java.util.Queue;
-import java.util.Random;
-import java.util.Set;
-import java.util.stream.IntStream;
-
 import com.actelion.research.chem.Coordinates;
 import com.actelion.research.chem.StereoMolecule;
-import com.actelion.research.chem.descriptor.pharmacophoretree.IonizableGroupDetector2D;
 import com.actelion.research.chem.docking.DockingUtils;
-import com.actelion.research.chem.docking.ScoringTask;
 import com.actelion.research.chem.docking.scoring.ProbeScanning;
-import com.actelion.research.chem.interactionstatistics.InteractionAtomTypeCalculator;
 import com.actelion.research.chem.io.pdb.converter.MoleculeGrid;
-import com.actelion.research.chem.phesa.MolecularVolume;
 import com.actelion.research.chem.phesa.AtomicGaussian;
-import com.actelion.research.chem.phesa.ShapeVolume;
 import com.actelion.research.chem.phesa.Gaussian3D;
+import com.actelion.research.chem.phesa.MolecularVolume;
+import com.actelion.research.chem.phesa.ShapeVolume;
 import com.actelion.research.chem.phesa.pharmacophore.PharmacophoreCalculator;
-import com.actelion.research.chem.phesa.pharmacophore.pp.ChargePoint;
 import com.actelion.research.chem.phesa.pharmacophore.pp.IPharmacophorePoint;
 import com.actelion.research.chem.phesa.pharmacophore.pp.PPGaussian;
 import com.actelion.research.chem.phesa.pharmacophore.pp.SimplePharmacophorePoint;
-
 import smile.clustering.KMeans;
 
+import java.util.*;
+import java.util.stream.IntStream;
+
 
 public class NegativeReceptorImage extends MoleculeGrid {
 	
@@ -324,9 +310,19 @@ public class NegativeReceptorImage extends MoleculeGrid {
 	
 	private void analyzeBumps() {
 		double radiusSq = BUMP_RADIUS2*BUMP_RADIUS2;
-		for(int x=0;x<this.gridSize[0];x++) {
-			for(int y=0;y<this.gridSize[1];y++) {
-				for(int z=0;z<this.gridSize[2];z++) {
+
+// Replaced because of out-of-bounds exceptions; TLS 17Oct2021
+//		for(int x=0;x<this.gridSize[0];x++) {
+//			for(int y=0;y<this.gridSize[1];y++) {
+//				for(int z=0;z<this.gridSize[2];z++) {
+
+		int xmax = Math.min(this.gridSize[0], bumpGrid.length);
+		int ymax = Math.min(this.gridSize[1], bumpGrid[0].length);
+		int zmax = Math.min(this.gridSize[2], bumpGrid[0][0].length);
+		for(int x=0;x<xmax;x++) {
+			for(int y=0;y<ymax;y++) {
+				for(int z=0;z<zmax;z++) {
+
 					Coordinates probeCoords = this.getCartCoordinates(new int[] {x,y,z});
 					for(int atom=0;atom<receptor.getAtoms();atom++) {
 						Coordinates c = receptor.getCoordinates(atom);


=====================================
src/main/java/com/actelion/research/chem/docking/scoring/AbstractScoringEngine.java
=====================================
@@ -72,6 +72,10 @@ public abstract class AbstractScoringEngine  {
 	public void addConstraint(PotentialEnergyTerm constraint) {
 		this.constraints.add(constraint);
 	}
+	
+	public void removeConstraints() {
+		this.constraints = new ArrayList<>();
+	}
 
 	public abstract void init(LigandPose candidatePose, double e0);
 	


=====================================
src/main/java/com/actelion/research/chem/forcefield/mmff/MMFFExternalPositionConstraint.java
=====================================
@@ -0,0 +1,56 @@
+package com.actelion.research.chem.forcefield.mmff;
+
+import com.actelion.research.chem.StereoMolecule;
+
+public class MMFFExternalPositionConstraint implements EnergyTerm {
+	private double[] refPos;
+	private int constrainedAtom;
+	private double k;
+	private double d;
+	
+	
+	public MMFFExternalPositionConstraint (int atom, double[] refPos, double k, double d) {
+		this.refPos = refPos;
+		this.constrainedAtom = atom;
+		this.k = k;
+		this.d = d;
+	}
+	
+
+	@Override
+	public double getEnergy(double[] pos) {
+		double energy = 0.0;
+		double dx = pos[3*constrainedAtom]-refPos[0];
+		double dy = pos[3*constrainedAtom+1]-refPos[1];
+		double dz = pos[3*constrainedAtom+2]-refPos[2];
+		double dist = Math.sqrt(dx*dx + dy*dy + dz*dz);
+		double prefactor = 0.0;
+		if(dist>d)
+			prefactor = dist-d;
+		else 
+			prefactor = 0.0;
+		energy+=0.5*k*prefactor*prefactor;
+		
+		return energy;
+	}
+
+	@Override
+	public void getGradient(double[] pos, double[] grad) {
+			int a = constrainedAtom;
+			double dx = pos[3*a]-refPos[0];
+			double dy = pos[3*a+1]-refPos[1];
+			double dz = pos[3*a+2]-refPos[2];
+			double dist = Math.sqrt(dx*dx + dy*dy + dz*dz);
+			double prefactor = 0.0;
+			if(dist>d)
+				prefactor = dist-d;
+			else 
+				prefactor = 0.0;
+			grad[a] += prefactor*dx/Math.max(dist, 1E-08);
+			grad[a+1] += prefactor*dy/Math.max(dist, 1E-08);
+			grad[a+2] += prefactor*dz/Math.max(dist, 1E-08);
+	}
+		
+	
+
+}


=====================================
src/main/java/com/actelion/research/chem/forcefield/mmff/PositionConstraint.java → src/main/java/com/actelion/research/chem/forcefield/mmff/MMFFPositionConstraint.java
=====================================
@@ -2,14 +2,14 @@ package com.actelion.research.chem.forcefield.mmff;
 
 import com.actelion.research.chem.StereoMolecule;
 
-public class PositionConstraint implements EnergyTerm {
+public class MMFFPositionConstraint implements EnergyTerm {
 	private double[] refPos;
 	private boolean[] constrained;
 	private double k;
 	private double d;
 	
 	
-	public PositionConstraint (StereoMolecule mol, double k, double d) {
+	public MMFFPositionConstraint (StereoMolecule mol, double k, double d) {
 		refPos = new double[3*mol.getAllAtoms()];
 		constrained = new boolean[mol.getAllAtoms()];
 		for(int a=0;a<mol.getAllAtoms();a++) {


=====================================
src/main/java/com/actelion/research/chem/io/CompoundFileHelper.java
=====================================
@@ -69,7 +69,9 @@ public abstract class CompoundFileHelper {
 	public static final int cFileTypeMOL = 0x00010000;
 	public static final int cFileTypeMOL2 = 0x00020000;
 	public static final int cFileTypePDB = 0x00040000;
-	public static final int cFileTypeSDGZ = 0x00080000;
+	public static final int cFileTypeMMTF = 0x00080000;
+	public static final int cFileTypeProtein = 0x000C0000;
+	public static final int cFileTypeSDGZ = 0x00100000;
     public static final int cFileTypeUnknown = -1;
 	public static final int cFileTypeDirectory = -2;
 
@@ -379,6 +381,10 @@ public abstract class CompoundFileHelper {
 			filter.addExtension("pdb");
 			filter.addDescription("Protein Data Bank files");
 			}
+		if ((filetypes & cFileTypeMMTF) != 0) {
+			filter.addExtension("mmtf");
+			filter.addDescription("Binary Protein Data Bank files");
+			}
 		if ((filetypes & cFileTypeMOL) != 0) {
 			filter.addExtension("mol");
 			filter.addDescription("MDL Molfiles");
@@ -490,6 +496,8 @@ public abstract class CompoundFileHelper {
 			return cFileTypeMOL2;
 		if (extension.equals(".pdb"))
 			return cFileTypePDB;
+		if (extension.equals(".mmtf"))
+			return cFileTypeMMTF;
 
         return cFileTypeUnknown;
         }
@@ -574,7 +582,10 @@ public abstract class CompoundFileHelper {
 		case cFileTypePDB:
 			extension = ".pdb";
 			break;
-			case cFileTypeSDGZ:
+		case cFileTypeMMTF:
+			extension = ".mmtf";
+			break;
+		case cFileTypeSDGZ:
 			extension = ".sdf.gz";
 			break;
 			}


=====================================
src/main/java/com/actelion/research/chem/mcs/MCS.java
=====================================
@@ -37,13 +37,12 @@ import java.util.*;
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 *
 */
-public class MCS
-{
-	public static final int PAR_CLEAVE_RINGS=0; 
-	public static final int PAR_KEEP_RINGS=1;
-	public static final int PAR_KEEP_AROMATIC_RINGS=2; 
+public class MCS {
+	public static final int PAR_CLEAVE_RINGS = 0;
+	public static final int PAR_KEEP_RINGS = 1;
+	public static final int PAR_KEEP_AROMATIC_RINGS = 2;
 	private static final boolean DEBUG = false;
-	private static final int CAPACITY = (60+60)*10; 
+	private static final int CAPACITY = (60 + 60) * 10;
 	private StereoMolecule mol;
 	private StereoMolecule frag;
 	private HashSet<IntVec> hsIndexFragCandidates;
@@ -56,74 +55,71 @@ public class MCS
 	private boolean considerRings;
 	// The largest valid solutions.
 	private List<IntVec> liMCSSolutions;
-	private HashMap<Integer, List<int []>> hmRingBnd_ListRingBnds;
-	private HashMap<Integer, List<int []>> hmAromaticRingBnd_ListRingBnds;
+	private HashMap<Integer, List<int[]>> hmRingBnd_ListRingBnds;
+	private HashMap<Integer, List<int[]>> hmAromaticRingBnd_ListRingBnds;
 	private RingCollection ringCollection;
-    boolean excluded[] = null;
+	boolean excluded[] = null;
+	private int [] arrMatchListFrag2Mol;
 
-    public MCS()
-    {
-        this(PAR_CLEAVE_RINGS,null);
+	public MCS() {
+		this(PAR_CLEAVE_RINGS, null);
 	}
-	
-    public MCS(int ringStatus)
-    {
-        this(ringStatus,null);
-    }
-		
-    public MCS(int ringStatus,SSSearcher searcher)
-    {
-		considerRings=false;
-		considerAromaticRings=false;
+
+	public MCS(int ringStatus) {
+		this(ringStatus, null);
+	}
+
+	public MCS(int ringStatus, SSSearcher searcher) {
+		considerRings = false;
+		considerAromaticRings = false;
 		switch (ringStatus) {
-		case PAR_CLEAVE_RINGS:
-			break;
-		case PAR_KEEP_RINGS:
-			considerRings=true;
-			break;
-		case PAR_KEEP_AROMATIC_RINGS:
-			considerAromaticRings=true;
-			break;
-		default:
-			break;
+			case PAR_CLEAVE_RINGS:
+				break;
+			case PAR_KEEP_RINGS:
+				considerRings = true;
+				break;
+			case PAR_KEEP_AROMATIC_RINGS:
+				considerAromaticRings = true;
+				break;
+			default:
+				break;
 		}
 		hsIndexFragCandidates = new HashSet<IntVec>(CAPACITY);
 		hsIndexFragGarbage = new HashSet<IntVec>(CAPACITY);
 		hsIndexFragSolution = new HashSet<IntVec>(CAPACITY);
 		liMCSSolutions = new ArrayList<IntVec>(CAPACITY);
-        if (searcher == null)
-		sss = new SSSearcher();
-        else
-            sss = searcher;
+		if (searcher == null)
+			sss = new SSSearcher();
+		else
+			sss = searcher;
 		comparatorBitsSet = new ComparatorBitsSet();
-		hmRingBnd_ListRingBnds = new HashMap<Integer, List<int []>>();
-		hmAromaticRingBnd_ListRingBnds = new HashMap<Integer, List<int []>>();
+		hmRingBnd_ListRingBnds = new HashMap<Integer, List<int[]>>();
+		hmAromaticRingBnd_ListRingBnds = new HashMap<Integer, List<int[]>>();
+	}
+
+	public void setSSSearcher(SSSearcher sss) {
+		this.sss = sss;
 	}
-	
-    public void setSSSearcher(SSSearcher sss)
-    {
-        this.sss  = sss;
-    }
 
 	/**
 	 * mol should contain equal or more bonds than frag.
+	 *
 	 * @param mol
 	 * @param frag
 	 */
-    public void set(StereoMolecule mol, StereoMolecule frag)
-    {
-        set(mol,frag,null);
+	public void set(StereoMolecule mol, StereoMolecule frag) {
+		set(mol, frag, null);
 	}
 
 	/**
 	 * mol should contain equal or more bonds than frag.
 	 * If frag contains more than one molecule only the biggest one is considered.
+	 *
 	 * @param mol
 	 * @param frag
 	 * @param excluded
 	 */
-    public void set(StereoMolecule mol, StereoMolecule frag, boolean excluded[])
-    {
+	public void set(StereoMolecule mol, StereoMolecule frag, boolean excluded[]) {
 
 		StereoMolecule fragBiggestSub = new StereoMolecule(frag);
 
@@ -135,15 +131,14 @@ public class MCS
 
 		this.mol = mol;
 
-        this.frag = fragBiggestSub;
+		this.frag = fragBiggestSub;
 
 		this.excluded = excluded;
 
-        init();
-    }
+		init();
+	}
 
-    private void init()
-    {
+	private void init() {
 		hsIndexFragCandidates.clear();
 		hsIndexFragGarbage.clear();
 		hsIndexFragSolution.clear();
@@ -152,32 +147,31 @@ public class MCS
 		hmAromaticRingBnd_ListRingBnds.clear();
 		initCandidates();
 	}
-	
-    private void initCandidates()
-    {
+
+	private void initCandidates() {
 		ringCollection = frag.getRingSet();
 		int rings = ringCollection.getSize();
 		for (int i = 0; i < rings; i++) {
-			int [] arrIndexBnd = ringCollection.getRingBonds(i);
+			int[] arrIndexBnd = ringCollection.getRingBonds(i);
 			for (int j = 0; j < arrIndexBnd.length; j++) {
-				if(!hmRingBnd_ListRingBnds.containsKey(arrIndexBnd[j])){
-					hmRingBnd_ListRingBnds.put(arrIndexBnd[j], new ArrayList<int []>());
+				if (!hmRingBnd_ListRingBnds.containsKey(arrIndexBnd[j])) {
+					hmRingBnd_ListRingBnds.put(arrIndexBnd[j], new ArrayList<int[]>());
 				}
-				List<int []> li = hmRingBnd_ListRingBnds.get(arrIndexBnd[j]);
+				List<int[]> li = hmRingBnd_ListRingBnds.get(arrIndexBnd[j]);
 				li.add(arrIndexBnd);
 			}
-			if(ringCollection.isAromatic(i)){
+			if (ringCollection.isAromatic(i)) {
 				for (int j = 0; j < arrIndexBnd.length; j++) {
-					if(!hmAromaticRingBnd_ListRingBnds.containsKey(arrIndexBnd[j])){
-						hmAromaticRingBnd_ListRingBnds.put(arrIndexBnd[j], new ArrayList<int []>());
+					if (!hmAromaticRingBnd_ListRingBnds.containsKey(arrIndexBnd[j])) {
+						hmAromaticRingBnd_ListRingBnds.put(arrIndexBnd[j], new ArrayList<int[]>());
 					}
-					List<int []> li = hmAromaticRingBnd_ListRingBnds.get(arrIndexBnd[j]);
+					List<int[]> li = hmAromaticRingBnd_ListRingBnds.get(arrIndexBnd[j]);
 					li.add(arrIndexBnd);
 				}
 			}
 		}
 		// Array length for bit list.
-		int nInts = (int)(((double)frag.getBonds() / Integer.SIZE) + ((double)(Integer.SIZE-1)/Integer.SIZE));
+		int nInts = (int) (((double) frag.getBonds() / Integer.SIZE) + ((double) (Integer.SIZE - 1) / Integer.SIZE));
 		// The vector is used as bit vector.
 		// For each bond one bit.
 		for (int i = 0; i < frag.getBonds(); i++) {
@@ -186,28 +180,27 @@ public class MCS
 			hsIndexFragCandidates.add(iv);
 		}
 	}
-	
-    private void setBitAndAddRelatedRingBonds(int bit, IntVec iv)
-    {
-		if(!considerAromaticRings && !considerRings) {
+
+	private void setBitAndAddRelatedRingBonds(int bit, IntVec iv) {
+		if (!considerAromaticRings && !considerRings) {
 			iv.setBit(bit);
-		} else if(considerRings) {
+		} else if (considerRings) {
 			iv.setBit(bit);
-			if(hmRingBnd_ListRingBnds.containsKey(bit)){
-				List<int []> li = hmRingBnd_ListRingBnds.get(bit);
+			if (hmRingBnd_ListRingBnds.containsKey(bit)) {
+				List<int[]> li = hmRingBnd_ListRingBnds.get(bit);
 				for (int i = 0; i < li.size(); i++) {
-					int [] a = li.get(i);
+					int[] a = li.get(i);
 					for (int j = 0; j < a.length; j++) {
 						iv.setBit(a[j]);
 					}
 				}
 			}
-		} else if(considerAromaticRings) {
+		} else if (considerAromaticRings) {
 			iv.setBit(bit);
-			if(hmAromaticRingBnd_ListRingBnds.containsKey(bit)){
-				List<int []> li = hmAromaticRingBnd_ListRingBnds.get(bit);
+			if (hmAromaticRingBnd_ListRingBnds.containsKey(bit)) {
+				List<int[]> li = hmAromaticRingBnd_ListRingBnds.get(bit);
 				for (int i = 0; i < li.size(); i++) {
-					int [] a = li.get(i);
+					int[] a = li.get(i);
 					for (int j = 0; j < a.length; j++) {
 						iv.setBit(a[j]);
 					}
@@ -216,17 +209,16 @@ public class MCS
 		}
 		iv.calculateHashCode();
 	}
-	
+
 	/**
 	 * Checks first fragment for being sub structure of molecule. If true, frag is returned.
 	 * The returned IntVec contains the bond information of frag. The IntVec is used bit wise. Each bit corresponds to
 	 * one bond in the fragment. The information for the matching substructure in the molecule is not contained in the
 	 * IntVec.
-     *
+	 *
 	 * @return
 	 */
-    private List<IntVec> getAllSolutionsForCommonSubstructures()
-    {
+	private List<IntVec> getAllSolutionsForCommonSubstructures() {
 		sss.setMolecule(mol);
 		frag.setFragment(true);
 		sss.setFragment(frag);
@@ -234,10 +226,10 @@ public class MCS
 		// The MCS is the complete fragment.
 		//
 		try {
-            if (sss.findFragmentInMolecule(SSSearcher.cCountModeOverlapping, SSSearcher.cMatchDBondToDelocalized, excluded) > 0) {
+			if (sss.findFragmentInMolecule(SSSearcher.cCountModeOverlapping, SSSearcher.cMatchDBondToDelocalized, excluded) > 0) {
 				molMCS = frag;
 				List<IntVec> liIndexFragCandidates = new ArrayList<IntVec>(hsIndexFragCandidates);
-                if (liIndexFragCandidates != null && !liIndexFragCandidates.isEmpty()) {
+				if (liIndexFragCandidates != null && !liIndexFragCandidates.isEmpty()) {
 					IntVec iv = liIndexFragCandidates.get(0);
 					iv.setBits(0, iv.sizeBits());
 					iv.calculateHashCode();
@@ -245,70 +237,69 @@ public class MCS
 					li.add(iv);
 					return li;
 				}
-            }
+			}
 		} catch (Exception e) {
 			e.printStackTrace();
 		}
 		int maxSizeCandidates = 0;
-		while(!hsIndexFragCandidates.isEmpty()){
+		while (!hsIndexFragCandidates.isEmpty()) {
 			List<IntVec> liIndexFragCandidates = new ArrayList<IntVec>(hsIndexFragCandidates);
 			Collections.sort(liIndexFragCandidates, comparatorBitsSet);
 			// Get largest mcs.
-			IntVec iv = liIndexFragCandidates.get(liIndexFragCandidates.size()-1);
+			IntVec iv = liIndexFragCandidates.get(liIndexFragCandidates.size() - 1);
 			hsIndexFragCandidates.remove(iv);
-			if(DEBUG) {
+			if (DEBUG) {
 				System.out.println("Bits set " + iv.getBitsSet());
-				if(iv.getBitsSet() == frag.getBonds()){
+				if (iv.getBitsSet() == frag.getBonds()) {
 					System.out.println("Full structure in iv.");
 				}
 			}
 			StereoMolecule fragSub = getSubFrag(frag, iv);
 			sss.setFragment(fragSub);
-			if(sss.findFragmentInMolecule(SSSearcher.cCountModeOverlapping, SSSearcher.cDefaultMatchMode,excluded)>0){
+			if (sss.findFragmentInMolecule(SSSearcher.cCountModeOverlapping, SSSearcher.cDefaultMatchMode, excluded) > 0) {
 				hsIndexFragSolution.add(iv);
 				removeAllSubSolutions(iv);
-				if(iv.getBitsSet() != frag.getBonds()) {
+				if (iv.getBitsSet() != frag.getBonds()) {
 					List<IntVec> liIV = getAllPlusOneAtomCombinations(iv, frag);
 					for (IntVec ivPlus : liIV) {
-						if(DEBUG) {
-							if(ivPlus.getBitsSet() == frag.getBonds()){
+						if (DEBUG) {
+							if (ivPlus.getBitsSet() == frag.getBonds()) {
 								System.out.println("Full structure in ivPlus.");
 							}
 						}
-						if((!hsIndexFragGarbage.contains(ivPlus)) && (!hsIndexFragSolution.contains(ivPlus))){
-							hsIndexFragCandidates.add(ivPlus);	
+						if ((!hsIndexFragGarbage.contains(ivPlus)) && (!hsIndexFragSolution.contains(ivPlus))) {
+							hsIndexFragCandidates.add(ivPlus);
 						}
 					}
-                    if (DEBUG) {
+					if (DEBUG) {
 						System.out.println("tsIndexFragCandidates " + hsIndexFragCandidates.size());
+					}
 				}
-                }
-                if (maxSizeCandidates < hsIndexFragCandidates.size()) {
+				if (maxSizeCandidates < hsIndexFragCandidates.size()) {
 					maxSizeCandidates = hsIndexFragCandidates.size();
-                }
+				}
 			} else {
 				hsIndexFragGarbage.add(iv);
 			}
 		}
-		if(hsIndexFragSolution.size()==0){
+		if (hsIndexFragSolution.size() == 0) {
 			return null;
 		}
 		return getFinalSolutionSet(hsIndexFragSolution);
 	}
-	
+
 	/**
 	 * All molecules which are sub structures of an other molecule in the list are removed.
-     *
+	 *
 	 * @return
 	 */
-    public LinkedList<StereoMolecule> getAllCommonSubstructures()
-    {
+	public LinkedList<StereoMolecule> getAllCommonSubstructures() {
 		List<IntVec> liIndexFragSolution = getAllSolutionsForCommonSubstructures();
-		if(liIndexFragSolution==null){
+		if (liIndexFragSolution == null) {
 			return null;
 		}
 		Collections.sort(liIndexFragSolution, comparatorBitsSet);
-		IntVec ivMCS = liIndexFragSolution.get(liIndexFragSolution.size()-1);
+		IntVec ivMCS = liIndexFragSolution.get(liIndexFragSolution.size() - 1);
 		molMCS = getSubFrag(frag, ivMCS);
 		List<StereoMolecule> li = new ArrayList<StereoMolecule>();
 		for (IntVec iv : liIndexFragSolution) {
@@ -320,28 +311,27 @@ public class MCS
 	/**
 	 * @return maximum common substructure at top of list or null of none common MCS was found.
 	 */
-    public StereoMolecule getMCS()
-    {
-		List<IntVec> liIndexFragSolution = getAllSolutionsForCommonSubstructures ();
-		if(liIndexFragSolution==null){
+	public StereoMolecule getMCS() {
+		List<IntVec> liIndexFragSolution = getAllSolutionsForCommonSubstructures();
+		if (liIndexFragSolution == null) {
 			return null;
 		}
 		Collections.sort(liIndexFragSolution, comparatorBitsSet);
-		IntVec ivMCS = liIndexFragSolution.get(liIndexFragSolution.size()-1);
+		IntVec ivMCS = liIndexFragSolution.get(liIndexFragSolution.size() - 1);
 		molMCS = getSubFrag(frag, ivMCS);
 		return molMCS;
 	}
-	
+
 	/**
 	 * Calculates the bond arrays for molecule and fragment for their maximum common substructure.
-     *
-	 * @param arrBondMCSMol result array. Null or initialized with the length number of bonds in molecule. 
-	 * @param arrBondFrag result array. Null or initialized with the length number of bonds in fragment.
-	 * @return two arrays, in the first array the bits are set 'true' which corresponds to the bonds for maximum common substructure in the molecule. 
+	 *
+	 * @param arrBondMCSMol result array. Null or initialized with the length number of bonds in molecule.
+	 * @param arrBondFrag   result array. Null or initialized with the length number of bonds in fragment.
+	 * @return two arrays, in the first array the bits are set 'true' which corresponds to the bonds for maximum common substructure in the molecule..
 	 * In the second are the bits set 'true' which corresponds to the bonds for maximum common substructure in the fragment.
 	 */
-    public boolean[][] getMCSBondArray(boolean[] arrBondMCSMol, boolean[] arrBondFrag)
-    {
+
+    public boolean[][] getMCSBondArray(boolean[] arrBondMCSMol, boolean[] arrBondFrag) {
 		boolean [][] arrBondMol_Result = new boolean [2][];
 		List<IntVec> liIndexFragSolution = getAllSolutionsForCommonSubstructures();
 		if(liIndexFragSolution==null){
@@ -366,9 +356,10 @@ public class MCS
 		//
 		// The substructure has to be searched in the molecule because the mapping indices are not contained in the
 		// IntVec that is the solution from the MCS search.
+		arrMatchListFrag2Mol = null;
 		if(sss.findFragmentInMolecule(SSSearcher.cCountModeOverlapping, SSSearcher.cDefaultMatchMode,excluded)>0){
 			ArrayList<int[]> liMatchSubFrag2Mol = sss.getMatchList();
-			int [] arrMatchListFrag2Mol = getMappedMatchListFrag2Mol(fragSub, liMatchSubFrag2Mol.get(0));
+			arrMatchListFrag2Mol = getMappedMatchListFrag2Mol(fragSub, liMatchSubFrag2Mol.get(0));
 			int bondsMol = mol.getBonds();
 			if(arrBondMCSMol==null) {
 				arrBondMCSMol = new boolean [bondsMol];
@@ -382,7 +373,16 @@ public class MCS
 		return arrBondMol_Result;
 	}
 
-    private int[] getMappedMatchListFrag2Mol(StereoMolecule fragSub, int[] arrMatchListSubFrag2Mol)
+	/**
+	 * delivers the match list indices from getMCSBondArray(...)
+	 * index is the Fragment atom index, value is the matched atom index in molecule.
+	 * @return null if no match was found.
+	 */
+	public int[] getArrMatchListFrag2Mol() {
+		return arrMatchListFrag2Mol;
+	}
+
+	private int[] getMappedMatchListFrag2Mol(StereoMolecule fragSub, int[] arrMatchListSubFrag2Mol)
     {
 		int [] arrMatchListFrag2Mol = new int [frag.getAtoms()];
 //        SSSearcher sss = new SSSearcher();


=====================================
src/main/java/com/actelion/research/chem/mcs/SubStructSearchBondIndex.java deleted
=====================================
@@ -1,68 +0,0 @@
-package com.actelion.research.chem.mcs;
-
-import com.actelion.research.chem.StereoMolecule;
-import com.actelion.research.util.datamodel.IntVec;
-
-/*
-* Copyright (c) 1997 - 2016
-* Actelion Pharmaceuticals Ltd.
-* Gewerbestrasse 16
-* CH-4123 Allschwil, Switzerland
-*
-* All rights reserved.
-*
-* Redistribution and use in source and binary forms, with or without
-* modification, are permitted provided that the following conditions are met:
-*
-* 1. Redistributions of source code must retain the above copyright notice, this
-*    list of conditions and the following disclaimer.
-* 2. Redistributions in binary form must reproduce the above copyright notice,
-*    this list of conditions and the following disclaimer in the documentation
-*    and/or other materials provided with the distribution.
-* 3. Neither the name of the the copyright holder nor the
-*    names of its contributors may be used to endorse or promote products
-*    derived from this software without specific prior written permission.
-*
-* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
-* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
-* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
-* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
-* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
-* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
-* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
-* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
-* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
-* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
-*
-*/
-public class SubStructSearchBondIndex {
-	
-	
-	private StereoMolecule mol;
-	
-	private StereoMolecule frag;
-	
-	private IntVec ivBondIndex;
-	
-	public void setMolecule(StereoMolecule mol){
-		this.mol=mol;
-	}
-	
-	public void setFragment(StereoMolecule frag){
-		this.frag=frag;
-	}
-	
-	public void setBondIndex(IntVec ivBondIndex){
-		this.ivBondIndex=ivBondIndex;
-		
-		
-	}
-	
-	public int findFragmentInMolecule() {
-		
-		
-		
-		return -1;
-	}
-
-}


=====================================
src/main/java/com/actelion/research/chem/mcs/SubStructSearchExhaustiveTreeWalker.java
=====================================
@@ -281,7 +281,7 @@ public class SubStructSearchExhaustiveTreeWalker {
 	
 	/**
 	 * Generates a tree from the fragment
-	 * @param possMapFrag2MolStart
+	 * @param
 	 * @return an array with one field for each fragment atom. 
 	 */
 	private void determineProcessingArray(){


=====================================
src/main/java/com/actelion/research/chem/phesa/pharmacophore/PharmacophoreCalculator.java
=====================================
@@ -1,10 +1,5 @@
 package com.actelion.research.chem.phesa.pharmacophore;
 
-import java.util.ArrayList;
-import java.util.Arrays;
-import java.util.List;
-import java.util.stream.Collectors;
-
 import com.actelion.research.chem.AtomFunctionAnalyzer;
 import com.actelion.research.chem.RingCollection;
 import com.actelion.research.chem.StereoMolecule;
@@ -12,9 +7,13 @@ import com.actelion.research.chem.interactionstatistics.InteractionAtomTypeCalcu
 import com.actelion.research.chem.phesa.pharmacophore.pp.AcceptorPoint;
 import com.actelion.research.chem.phesa.pharmacophore.pp.AromRingPoint;
 import com.actelion.research.chem.phesa.pharmacophore.pp.DonorPoint;
-import com.actelion.research.chem.phesa.pharmacophore.pp.ExitVectorPoint;
 import com.actelion.research.chem.phesa.pharmacophore.pp.IPharmacophorePoint;
 
+import java.util.ArrayList;
+import java.util.Arrays;
+import java.util.List;
+import java.util.stream.Collectors;
+
 public class PharmacophoreCalculator {
 	
 	public static final int ACCEPTOR_ID = 0;
@@ -79,36 +78,36 @@ public class PharmacophoreCalculator {
 					}
 					if(mol.getAtomicNo(i)==8 && neighbours==1 && (mol.getConnBondOrder(i, 0)==2 || AtomFunctionAnalyzer.isAcidicOxygen(mol, i) || mol.getAtomCharge(i)==-1 )) {
 						int a1 = mol.getConnAtom(i,0);
-						if(!(mol.getAtomicNo(a1)==16 || mol.getAtomicNo(a1)==15)) {		
-							int aa1 = mol.getConnAtom(a1,0);
-							if(aa1==i) 
-								aa1 = mol.getConnAtom(a1,1);
-							neighbourList.add(aa1);
-							AcceptorPoint ap = new AcceptorPoint(mol,i,neighbourList,interactionClass,1);
+						if (mol.getConnAtoms(a1) > 1) { // added this check to prevent OOB exceptions with cropped proteins; TLS 17Oct2021
+							if(!(mol.getAtomicNo(a1)==16 || mol.getAtomicNo(a1)==15)) {
+								int aa1 = mol.getConnAtom(a1,0);
+								if(aa1==i)
+									aa1 = mol.getConnAtom(a1,1);
+								neighbourList.add(aa1);
+								AcceptorPoint ap = new AcceptorPoint(mol,i,neighbourList,interactionClass,1);
 
-							ppPoints.add(ap);
-							List<Integer> neighbourList2 = new ArrayList<Integer>();
-							for(int neighbour : neighbourList) {
-								neighbourList2.add(neighbour);
-							}
-							ap = new AcceptorPoint(mol,i,neighbourList2,interactionClass,2);
-							ppPoints.add(ap);
+								ppPoints.add(ap);
+								List<Integer> neighbourList2 = new ArrayList<Integer>();
+								for(int neighbour : neighbourList) {
+									neighbourList2.add(neighbour);
+								}
+								ap = new AcceptorPoint(mol,i,neighbourList2,interactionClass,2);
+								ppPoints.add(ap);
 
+								}
+							else {
+								AcceptorPoint ap = new AcceptorPoint(mol,i,neighbourList,interactionClass);
+								ppPoints.add(ap);
 							}
-						else { 
-							AcceptorPoint ap = new AcceptorPoint(mol,i,neighbourList,interactionClass);
-							ppPoints.add(ap);
 						}
-	
 					}
-					else {
-					AcceptorPoint ap = new AcceptorPoint(mol,i,neighbourList,interactionClass);
-					ppPoints.add(ap);
-
+					else if (neighbourList.size() != 0) { // added this check to prevent OOB exceptions with cropped proteins; TLS 17Oct2021
+						AcceptorPoint ap = new AcceptorPoint(mol,i,neighbourList,interactionClass);
+						ppPoints.add(ap);
 					}
+				}
 			}
 		}
-		}
 		return ppPoints;
 	}
 	
@@ -124,7 +123,7 @@ public class PharmacophoreCalculator {
 				}
 			}
 			else if (mol.getAtomicNo(a)==7){ // atom is not aromatic
-				if(mol.getConnBondOrder(a, 0)==3) //nitrile
+				if(mol.getConnAtoms(a) == 1 && mol.getConnBondOrder(a, 0)==3) //nitrile
 					return true;
 				if (mol.isFlatNitrogen(a)) { 
 					for(int b=0;b<mol.getConnAtoms(a);b++) {


=====================================
src/main/java/com/actelion/research/chem/phesaflex/FlexibleShapeAlignment.java
=====================================
@@ -7,7 +7,7 @@ import java.util.Map;
 import com.actelion.research.chem.StereoMolecule;
 import com.actelion.research.chem.alignment3d.transformation.TransformationSequence;
 import com.actelion.research.chem.forcefield.mmff.ForceFieldMMFF94;
-import com.actelion.research.chem.forcefield.mmff.PositionConstraint;
+import com.actelion.research.chem.forcefield.mmff.MMFFPositionConstraint;
 import com.actelion.research.chem.optimization.OptimizerLBFGS;
 import com.actelion.research.chem.phesa.MolecularVolume;
 import com.actelion.research.chem.phesa.PheSAAlignment;
@@ -166,7 +166,7 @@ public class FlexibleShapeAlignment {
 		int cycles = 0;
 		while(notRelaxed && cycles<maxCycles) {
 			ForceFieldMMFF94 forceField = new ForceFieldMMFF94(fitMol, ForceFieldMMFF94.MMFF94SPLUS, ffOptions);
-			PositionConstraint constraint = new PositionConstraint(fitMol,50,init);
+			MMFFPositionConstraint constraint = new MMFFPositionConstraint(fitMol,50,init);
 			forceField.addEnergyTerm(constraint);
 			forceField.minimise();
 			double e = forceField.getTotalEnergy();


=====================================
src/main/java/com/actelion/research/chem/potentialenergy/PositionConstraint.java
=====================================
@@ -15,9 +15,9 @@ public class PositionConstraint implements PotentialEnergyTerm {
 		refPos = new double[3];
 		this.atom = atom;
 		this.conf = conf;
-		refPos[0] = conf.getMolecule().getAtomX(atom);
-		refPos[1] = conf.getMolecule().getAtomY(atom);
-		refPos[2] = conf.getMolecule().getAtomZ(atom);
+		refPos[0] = conf.getX(atom);
+		refPos[1] = conf.getY(atom);
+		refPos[2] = conf.getZ(atom);
 		
 		this.k = k;
 		this.d = d;



View it on GitLab: https://salsa.debian.org/java-team/openchemlib/-/commit/5c2ee3a24208a07614270d7c768703ae682e2c06

-- 
View it on GitLab: https://salsa.debian.org/java-team/openchemlib/-/commit/5c2ee3a24208a07614270d7c768703ae682e2c06
You're receiving this email because of your account on salsa.debian.org.


-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://alioth-lists.debian.net/pipermail/pkg-java-commits/attachments/20211026/a992611a/attachment.htm>


More information about the pkg-java-commits mailing list