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 */
021package org.biojava.nbio.structure.symmetry.core;
022
023import org.biojava.nbio.structure.geometry.CalcPoint;
024import org.biojava.nbio.structure.geometry.SuperPositions;
025import org.slf4j.Logger;
026import org.slf4j.LoggerFactory;
027
028import javax.vecmath.AxisAngle4d;
029import javax.vecmath.Matrix4d;
030import javax.vecmath.Point3d;
031import javax.vecmath.Vector3d;
032
033import java.util.*;
034import java.util.Map.Entry;
035
036/**
037 *
038 *
039 * @author Peter Rose
040 *
041 */
042public class HelixSolver {
043
044        private static final Logger logger = LoggerFactory
045                        .getLogger(HelixSolver.class);
046
047        private QuatSymmetrySubunits subunits = null;
048        private int fold = 1;
049        private HelixLayers helixLayers = new HelixLayers();
050        private QuatSymmetryParameters parameters = null;
051        boolean modified = true;
052
053        public HelixSolver(QuatSymmetrySubunits subunits, int fold,
054                        QuatSymmetryParameters parameters) {
055                this.subunits = subunits;
056                this.fold = fold;
057                this.parameters = parameters;
058        }
059
060        public HelixLayers getSymmetryOperations() {
061                if (modified) {
062                        solve();
063                        modified = false;
064                }
065                return helixLayers;
066        }
067
068        private void solve() {
069                if (!preCheck()) {
070                        return;
071                }
072
073                HelicalRepeatUnit unit = new HelicalRepeatUnit(subunits);
074                List<Point3d> repeatUnitCenters = unit.getRepeatUnitCenters();
075                List<Point3d[]> repeatUnits = unit.getRepeatUnits();
076                Set<List<Integer>> permutations = new HashSet<>();
077
078                double minRise = parameters.getMinimumHelixRise() * fold; // for n-start
079                                                                                                                                        // helix,
080                                                                                                                                        // the rise
081                                                                                                                                        // must be
082                                                                                                                                        // steeper
083                Map<Integer[], Integer> interactionMap = unit
084                                .getInteractingRepeatUnits();
085
086                int maxLayerLineLength = 0;
087
088                for (Entry<Integer[], Integer> entry : interactionMap.entrySet()) {
089                        Integer[] pair = entry.getKey();
090                        logger.debug("HelixSolver: pair: {}", Arrays.toString(pair));
091
092                        int contacts = entry.getValue();
093                        Point3d[] h1 = CalcPoint.clonePoint3dArray(repeatUnits.get(pair[0]));
094                        Point3d[] h2 = CalcPoint.clonePoint3dArray(repeatUnits.get(pair[1]));
095
096                        // trial superposition of repeat unit pairs to get a seed
097                        // permutation
098                        Matrix4d transformation = SuperPositions.superposeAndTransform(h2, h1);
099
100                        double rmsd = CalcPoint.rmsd(h1, h2);
101                        double rise = getRise(transformation,
102                                        repeatUnitCenters.get(pair[0]),
103                                        repeatUnitCenters.get(pair[1]));
104                        double angle = getAngle(transformation);
105
106                        logger.debug(
107                                        "Original rmsd: {}, Original rise {}, Original angle: {}",
108                                        rmsd, rise, Math.toDegrees(angle));
109
110                        if (rmsd > parameters.getRmsdThreshold()) {
111                                continue;
112                        }
113
114                        if (Math.abs(rise) < minRise) {
115                                continue;
116                        }
117
118                        // determine which subunits are permuted by the transformation
119                        List<Integer> permutation = getPermutation(transformation);
120
121                        // check permutations for validity
122
123                        // don't save redundant permutations
124                        if (permutations.contains(permutation)) {
125                                continue;
126                        }
127                        permutations.add(permutation);
128                        logger.debug("Permutation: {}", permutation);
129
130
131                        // keep track of which subunits are permuted
132                        Set<Integer> permSet = new HashSet<>();
133                        int count = 0;
134                        boolean valid = true;
135                        for (int i = 0; i < permutation.size(); i++) {
136                                if (permutation.get(i) == i) {
137                                        valid = false;
138                                        break;
139                                }
140                                if (permutation.get(i) != -1) {
141                                        permSet.add(permutation.get(i));
142                                        permSet.add(i);
143                                        count++;
144                                }
145
146                        }
147
148                        // a helix a repeat unit cannot map onto itself
149                        if (!valid) {
150                                logger.debug("Invalid mapping");
151                                continue;
152                        }
153
154                        // all subunits must be involved in a permutation
155                        if (permSet.size() != subunits.getSubunitCount()) {
156                                logger.debug("Not all subunits involved in permutation");
157                                continue;
158                        }
159
160                        // if all subunit permutation values are set, then it can't be
161                        // helical symmetry (must be cyclic symmetry)
162                        if (count == permutation.size()) {
163                                continue;
164                        }
165
166                        // superpose all permuted subunits
167                        List<Point3d> point1 = new ArrayList<Point3d>();
168                        List<Point3d> point2 = new ArrayList<Point3d>();
169                        List<Point3d> centers = subunits.getOriginalCenters();
170                        for (int j = 0; j < permutation.size(); j++) {
171                                if (permutation.get(j) != -1) {
172                                        point1.add(new Point3d(centers.get(j)));
173                                        point2.add(new Point3d(centers.get(permutation.get(j))));
174                                }
175                        }
176
177                        h1 = new Point3d[point1.size()];
178                        h2 = new Point3d[point2.size()];
179                        point1.toArray(h1);
180                        point2.toArray(h2);
181
182                        // calculate subunit rmsd if at least 3 subunits are available
183                        double subunitRmsd = 0;
184                        if (point1.size() > 2) {
185                                transformation = SuperPositions.superposeAndTransform(h2, h1);
186
187                                subunitRmsd = CalcPoint.rmsd(h1, h2);
188                                rise = getRise(transformation, repeatUnitCenters.get(pair[0]),
189                                                repeatUnitCenters.get(pair[1]));
190                                angle = getAngle(transformation);
191
192                                logger.debug("Subunit rmsd: {}, Subunit rise: {}, Subunit angle: {}", subunitRmsd, rise, Math.toDegrees(angle));
193
194                                if (subunitRmsd > parameters.getRmsdThreshold()) {
195                                        continue;
196                                }
197
198                                if (Math.abs(rise) < minRise) {
199                                        continue;
200                                }
201
202                                if (subunitRmsd > parameters.getHelixRmsdToRiseRatio()
203                                                * Math.abs(rise)) {
204                                        continue;
205                                }
206                        }
207
208                        // superpose all C alpha traces
209                        point1.clear();
210                        point2.clear();
211                        List<Point3d[]> traces = subunits.getTraces();
212                        for (int j = 0; j < permutation.size(); j++) {
213                                if (permutation.get(j) == -1) {
214                                        continue;
215                                }
216                                for (Point3d p : traces.get(j)) {
217                                        point1.add(new Point3d(p));
218                                }
219                                for (Point3d p : traces.get(permutation.get(j))) {
220                                        point2.add(new Point3d(p));
221                                }
222                        }
223
224                        h1 = new Point3d[point1.size()];
225                        h2 = new Point3d[point2.size()];
226                        point1.toArray(h1);
227                        point2.toArray(h2);
228                        Point3d[] h3 = CalcPoint.clonePoint3dArray(h1);
229                        transformation = SuperPositions.superposeAndTransform(h2, h1);
230
231                        Point3d xtrans = CalcPoint.centroid(h3);
232
233                        xtrans.negate();
234
235                        double traceRmsd = CalcPoint.rmsd(h1, h2);
236
237                        rise = getRise(transformation, repeatUnitCenters.get(pair[0]),
238                                        repeatUnitCenters.get(pair[1]));
239                        angle = getAngle(transformation);
240
241                        logger.debug("Trace rmsd: {}", traceRmsd);
242                        logger.debug("Trace rise: {}", rise);
243                        logger.debug("Trace angle: {}", Math.toDegrees(angle));
244                        logger.debug("Permutation: {}", permutation);
245
246                        if (traceRmsd > parameters.getRmsdThreshold()) {
247                                continue;
248                        }
249
250                        if (Math.abs(rise) < minRise) {
251                                continue;
252                        }
253
254                        // This prevents translational repeats to be counted as helices
255                        if (angle < Math.toRadians(parameters.getMinimumHelixAngle())) {
256                                continue;
257                        }
258
259                        if (traceRmsd > parameters.getHelixRmsdToRiseRatio()
260                                        * Math.abs(rise)) {
261                                continue;
262                        }
263
264                        AxisAngle4d a1 = new AxisAngle4d();
265                        a1.set(transformation);
266
267                        // save this helix rot-translation
268                        Helix helix = new Helix();
269                        helix.setTransformation(transformation);
270                        helix.setPermutation(permutation);
271                        helix.setRise(rise);
272                        // Old version of Vecmath on LINUX doesn't set element m33 to 1.
273                        // Here we make sure it's 1.
274                        transformation.setElement(3, 3, 1.0);
275                        transformation.invert();
276                        QuatSymmetryScores scores = QuatSuperpositionScorer.calcScores(
277                                        subunits, transformation, permutation);
278                        scores.setRmsdCenters(subunitRmsd);
279                        helix.setScores(scores);
280                        helix.setFold(fold);
281                        helix.setContacts(contacts);
282                        helix.setRepeatUnits(unit.getRepeatUnitIndices());
283                        logger.debug("Layerlines: {}", helix.getLayerLines());
284
285                        for (List<Integer> line : helix.getLayerLines()) {
286                                maxLayerLineLength = Math.max(maxLayerLineLength, line.size());
287                        }
288
289                        // TODO
290                        // checkSelfLimitingHelix(helix);
291
292                        helixLayers.addHelix(helix);
293
294                }
295                if (maxLayerLineLength < 3) {
296                        // System.out.println("maxLayerLineLength: " + maxLayerLineLength);
297                        helixLayers.clear();
298                }
299
300                return;
301        }
302
303        @SuppressWarnings("unused")
304        private void checkSelfLimitingHelix(Helix helix) {
305                HelixExtender he = new HelixExtender(subunits, helix);
306                Point3d[] extendedHelix = he.extendHelix(1);
307
308                int overlap1 = 0;
309                for (Point3d[] trace : subunits.getTraces()) {
310                        for (Point3d pt : trace) {
311                                for (Point3d pe : extendedHelix) {
312                                        if (pt.distance(pe) < 5.0) {
313                                                overlap1++;
314                                        }
315                                }
316                        }
317                }
318
319                extendedHelix = he.extendHelix(-1);
320
321                int overlap2 = 0;
322                for (Point3d[] trace : subunits.getTraces()) {
323                        for (Point3d pt : trace) {
324                                for (Point3d pe : extendedHelix) {
325                                        if (pt.distance(pe) < 3.0) {
326                                                overlap2++;
327                                        }
328                                }
329                        }
330                }
331                System.out.println("SelfLimiting helix: " + overlap1 + ", " + overlap2);
332        }
333
334        private boolean preCheck() {
335                if (subunits.getSubunitCount() < 3) {
336                        return false;
337                }
338                List<Integer> folds = this.subunits.getFolds();
339                int maxFold = folds.get(folds.size() - 1);
340                return maxFold > 1;
341        }
342
343        /**
344         * Returns a permutation of subunit indices for the given helix
345         * transformation. An index of -1 is used to indicate subunits that do not
346         * superpose onto any other subunit.
347         *
348         * @param transformation
349         * @return
350         */
351        private List<Integer> getPermutation(Matrix4d transformation) {
352                double rmsdThresholdSq = Math
353                                .pow(this.parameters.getRmsdThreshold(), 2);
354
355                List<Point3d> centers = subunits.getOriginalCenters();
356                List<Integer> seqClusterId = subunits.getClusterIds();
357
358                List<Integer> permutations = new ArrayList<>(centers.size());
359                double[] dSqs = new double[centers.size()];
360                boolean[] used = new boolean[centers.size()];
361                Arrays.fill(used, false);
362
363                for (int i = 0; i < centers.size(); i++) {
364                        Point3d tCenter = new Point3d(centers.get(i));
365                        transformation.transform(tCenter);
366                        int permutation = -1;
367                        double minDistSq = Double.MAX_VALUE;
368                        for (int j = 0; j < centers.size(); j++) {
369                                if (Objects.equals(seqClusterId.get(i), seqClusterId.get(j))) {
370                                        if (!used[j]) {
371                                                double dSq = tCenter.distanceSquared(centers.get(j));
372                                                if (dSq < minDistSq && dSq <= rmsdThresholdSq) {
373                                                        minDistSq = dSq;
374                                                        permutation = j;
375                                                        dSqs[j] = dSq;
376                                                }
377                                        }
378                                }
379                        }
380                        // can't map to itself
381                        if (permutations.size() == permutation) {
382                                permutation = -1;
383                        }
384
385                        if (permutation != -1) {
386                                used[permutation] = true;
387                        }
388
389                        permutations.add(permutation);
390                }
391
392                return permutations;
393        }
394
395        /**
396         * Returns the rise of a helix given the subunit centers of two adjacent
397         * subunits and the helix transformation
398         *
399         * @param transformation
400         *            helix transformation
401         * @param p1
402         *            center of one subunit
403         * @param p2
404         *            center of an adjacent subunit
405         * @return
406         */
407        private static double getRise(Matrix4d transformation, Point3d p1,
408                        Point3d p2) {
409                AxisAngle4d axis = getAxisAngle(transformation);
410                Vector3d h = new Vector3d(axis.x, axis.y, axis.z);
411                Vector3d p = new Vector3d();
412                p.sub(p1, p2);
413                return p.dot(h);
414        }
415
416        /**
417         * Returns the pitch angle of the helix
418         *
419         * @param transformation
420         *            helix transformation
421         * @return
422         */
423        private static double getAngle(Matrix4d transformation) {
424                return getAxisAngle(transformation).angle;
425        }
426
427        /**
428         * Returns the AxisAngle of the helix transformation
429         *
430         * @param transformation
431         *            helix transformation
432         * @return
433         */
434        private static AxisAngle4d getAxisAngle(Matrix4d transformation) {
435                AxisAngle4d axis = new AxisAngle4d();
436                axis.set(transformation);
437                return axis;
438        }
439}