001/* 002 * BioJava development code 003 * 004 * This code may be freely distributed and modified under the 005 * terms of the GNU Lesser General Public Licence. This should 006 * be distributed with the code. If you do not have a copy, 007 * see: 008 * 009 * http://www.gnu.org/copyleft/lesser.html 010 * 011 * Copyright for this code is held jointly by the individual 012 * authors. These should be listed in @author doc comments. 013 * 014 * For more information on the BioJava project and its aims, 015 * or to join the biojava-l mailing list, visit the home page 016 * at: 017 * 018 * http://www.biojava.org/ 019 * 020 */ 021 022package org.biojava.nbio.structure.symmetry.core; 023 024import org.biojava.nbio.structure.geometry.CalcPoint; 025import org.biojava.nbio.structure.geometry.UnitQuaternions; 026import org.biojava.nbio.structure.symmetry.utils.PermutationGenerator; 027 028import javax.vecmath.AxisAngle4d; 029import javax.vecmath.Matrix4d; 030import javax.vecmath.Point3d; 031import javax.vecmath.Quat4d; 032import javax.vecmath.Vector3d; 033import java.util.ArrayList; 034import java.util.HashSet; 035import java.util.List; 036import java.util.Objects; 037import java.util.Set; 038 039 040/** 041 * 042 * @author Peter 043 */ 044public class SystematicSolver implements QuatSymmetrySolver { 045 private QuatSymmetrySubunits subunits = null; 046 private QuatSymmetryParameters parameters = null; 047 048 private Point3d[] originalCoords = null; 049 private Point3d[] transformedCoords = null; 050 private RotationGroup rotations = new RotationGroup(); 051 private Vector3d centroid = new Vector3d(); 052 private Matrix4d centroidInverse = new Matrix4d(); 053 private Set<List<Integer>> hashCodes = new HashSet<>(); 054 055 public SystematicSolver(QuatSymmetrySubunits subunits, QuatSymmetryParameters parameters) { 056 if (subunits.getSubunitCount()== 2) { 057 throw new IllegalArgumentException("SystematicSolver cannot be applied to subunits with 2 centers"); 058 } 059 this.subunits = subunits; 060 this.parameters = parameters; 061 } 062 063 @Override 064 public RotationGroup getSymmetryOperations() { 065 if (rotations.getOrder() == 0) { 066 solve(); 067 rotations.complete(); 068 } 069 return rotations; 070 } 071 072 private void solve() { 073 initialize(); 074 int n = subunits.getSubunitCount(); 075 PermutationGenerator g = new PermutationGenerator(n); 076 077 // loop over all permutations 078 while (g.hasMore()) { 079 int[] perm = g.getNext(); 080 List<Integer> permutation = new ArrayList<>(perm.length); 081 for (int j = 0; j < n; j++) { 082 permutation.add(perm[j]); 083 } 084 085 if (! isValidPermutation(permutation)) { 086 continue; 087 } 088 089 boolean newPermutation = evaluatePermutation(permutation); 090 if (newPermutation) { 091 completeRotationGroup(); 092 } 093 094 if (rotations.getOrder() >= subunits.getSubunitCount()) { 095 return; 096 } 097 } 098 } 099 100 /** 101 * Adds translational component to rotation matrix 102 * @param rotTrans 103 * @param rotation 104 * @return 105 */ 106 private void combineWithTranslation(Matrix4d rotation) { 107 rotation.setTranslation(centroid); 108 rotation.mul(rotation, centroidInverse); 109 } 110 111 private Rotation createSymmetryOperation(List<Integer> permutation, Matrix4d transformation, AxisAngle4d axisAngle, int fold, QuatSymmetryScores scores) { 112 Rotation s = new Rotation(); 113 s.setPermutation(new ArrayList<Integer>(permutation)); 114 s.setTransformation(new Matrix4d(transformation)); 115 s.setAxisAngle(new AxisAngle4d(axisAngle)); 116 s.setFold(fold); 117 s.setScores(scores); 118 return s; 119 } 120 121 private void completeRotationGroup() { 122 PermutationGroup g = new PermutationGroup(); 123 for (int i = 0; i < rotations.getOrder(); i++) { 124 Rotation s = rotations.getRotation(i); 125 g.addPermutation(s.getPermutation()); 126 } 127 g.completeGroup(); 128 129// System.out.println("Completing rotation group from: " +symmetryOperations.getSymmetryOperationCount() + " to " + g.getPermutationCount()); 130 131 // the group is complete, nothing to do 132 if (g.getOrder() == rotations.getOrder()) { 133 return; 134 } 135 136// System.out.println("complete group: " + rotations.getOrder() +"/" + g.getOrder()); 137 // try to complete the group 138 for (int i = 0; i < g.getOrder(); i++) { 139 List<Integer> permutation = g.getPermutation(i); 140 if (isValidPermutation(permutation)) { 141 // perform permutation of subunits 142 evaluatePermutation(permutation); 143 } 144 } 145 } 146 147 private boolean isValidPermutation(List<Integer> permutation) { 148 if (permutation.size() == 0) { 149 return false; 150 } 151 152 // if this permutation is a duplicate, return false 153 if (hashCodes.contains(permutation)) { 154 return false; 155 } 156 157 // check if permutation is pseudosymmetric 158 if (! isAllowedPermuation(permutation)) { 159 return false; 160 } 161 162 // get fold and make sure there is only one E (fold=1) permutation 163 int fold = PermutationGroup.getOrder(permutation); 164 if (rotations.getOrder() > 1 && fold == 1) { 165 return false; 166 } 167 if (fold == 0 || subunits.getSubunitCount() % fold != 0) { 168 return false; 169 } 170 171 // if this permutation is a duplicate, returns false 172 return hashCodes.add(permutation); 173 } 174 175 private boolean isAllowedPermuation(List<Integer> permutation) { 176 List<Integer> seqClusterId = subunits.getClusterIds(); 177 for (int i = 0; i < permutation.size(); i++) { 178 int j = permutation.get(i); 179 if (!Objects.equals(seqClusterId.get(i), seqClusterId.get(j))) { 180 return false; 181 } 182 } 183 return true; 184 } 185 186 private boolean evaluatePermutation(List<Integer> permutation) { 187 // permutate subunits 188 for (int j = 0, n = subunits.getSubunitCount(); j < n; j++) { 189 transformedCoords[j].set(originalCoords[permutation.get(j)]); 190 } 191 192 int fold = PermutationGroup.getOrder(permutation); 193 194 // TODO implement this piece of code using at origin superposition 195 Quat4d quat = UnitQuaternions.relativeOrientation( 196 originalCoords, transformedCoords); 197 AxisAngle4d axisAngle = new AxisAngle4d(); 198 Matrix4d transformation = new Matrix4d(); 199 200 transformation.set(quat); 201 axisAngle.set(quat); 202 203 Vector3d axis = new Vector3d(axisAngle.x, axisAngle.y, axisAngle.z); 204 if (axis.lengthSquared() < 1.0E-6) { 205 axisAngle.x = 0; 206 axisAngle.y = 0; 207 axisAngle.z = 1; 208 axisAngle.angle = 0; 209 } else { 210 axis.normalize(); 211 axisAngle.x = axis.x; 212 axisAngle.y = axis.y; 213 axisAngle.z = axis.z; 214 } 215 216 CalcPoint.transform(transformation, transformedCoords); 217 218 double subunitRmsd = CalcPoint.rmsd(transformedCoords, originalCoords); 219 220 if (subunitRmsd <parameters.getRmsdThreshold()) { 221 // transform to original coordinate system 222 combineWithTranslation(transformation); 223 QuatSymmetryScores scores = QuatSuperpositionScorer.calcScores(subunits, transformation, permutation); 224 if (scores.getRmsd() < 0.0 || scores.getRmsd() > parameters.getRmsdThreshold()) { 225 return false; 226 } 227 228 scores.setRmsdCenters(subunitRmsd); 229 Rotation symmetryOperation = createSymmetryOperation(permutation, transformation, axisAngle, fold, scores); 230 rotations.addRotation(symmetryOperation); 231 return true; 232 } 233 return false; 234 } 235 236 private void initialize() { 237 // translation to centered coordinate system 238 centroid = new Vector3d(subunits.getCentroid()); 239 240 // translation back to original coordinate system 241 Vector3d reverse = new Vector3d(centroid); 242 reverse.negate(); 243 centroidInverse.set(reverse); 244 // Make sure matrix element m33 is 1.0. An old version vecmath did not set this element. 245 centroidInverse.setElement(3, 3, 1.0); 246 247 List<Point3d> centers = subunits.getCenters(); 248 int n = subunits.getSubunitCount(); 249 250 originalCoords = new Point3d[n]; 251 transformedCoords = new Point3d[n]; 252 253 for (int i = 0; i < n; i++) { 254 originalCoords[i] = centers.get(i); 255 transformedCoords[i] = new Point3d(); 256 } 257 } 258}