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 javax.vecmath.Point3d;
025
026import java.util.*;
027import java.util.Map.Entry;
028
029public class HelicalRepeatUnit {
030        private QuatSymmetrySubunits subunits = null;
031        private List<Point3d> repeatUnitCenters = new ArrayList<Point3d>();
032        private List<Point3d[]> repeatUnits = new ArrayList<Point3d[]>();
033        private List<List<Integer>> repeatUnitIndices = new ArrayList<>();
034        private Map<Integer[], Integer> interactingNeighbors = Collections.emptyMap();
035
036public HelicalRepeatUnit(QuatSymmetrySubunits subunits) {
037        this.subunits = subunits;
038}
039
040public List<Point3d> getRepeatUnitCenters() {
041        if (repeatUnitCenters.isEmpty()) {
042                run();
043        }
044        return repeatUnitCenters;
045}
046
047public List<Point3d[]> getRepeatUnits() {
048        if (repeatUnits.isEmpty()) {
049                run();
050        }
051        return repeatUnits;
052}
053
054public List<List<Integer>> getRepeatUnitIndices() {
055        return repeatUnitIndices;
056}
057
058public Map<Integer[], Integer> getInteractingRepeatUnits() {
059        if (interactingNeighbors.isEmpty()) {
060                run();
061        }
062        return interactingNeighbors;
063}
064
065private void run() {
066        this.repeatUnitCenters = calcRepeatUnitCenters();
067        if (this.repeatUnitCenters.size() == 0) {
068                return;
069        }
070        this.repeatUnits = calcRepeatUnits();
071        this.interactingNeighbors = findInteractingNeigbors();
072}
073
074private List<Point3d> calcRepeatUnitCenters() {
075
076        // TODO why do we use models here? it should not matter. Setting to 0 all
077        List<Integer> models = new ArrayList<>(subunits.getSubunitCount());
078        for (int s = 0; s <subunits.getSubunitCount(); s++)
079                models.add(0);
080        Set<Integer> uniqueModels = new HashSet<>(Arrays.asList(1));
081
082        int modelCount = uniqueModels.size();
083        List<Integer> folds = this.subunits.getFolds();
084        int maxFold = folds.get(folds.size()-1);
085
086        List<Point3d> repeatCenters = new ArrayList<Point3d>();
087        List<Point3d> centers = subunits.getCenters();
088
089//      if (modelCount == maxFold && subunits.getSubunitCount() > 3) {
090        if (maxFold%modelCount == 0 && modelCount > 1 && subunits.getSubunitCount() > 3) {
091//              System.out.println("calcRepeatUnitCenters case 1");
092                for (int i = 0; i < modelCount; i++) {
093                        List<Integer> subunitIndices = new ArrayList<>();
094                        Point3d p = new Point3d();
095                        int count = 0;
096//                      System.out.println("Models: " + models.size());
097                        for (int j = 0; j < models.size(); j++) {
098                                if (models.get(j) == i) {
099                                        p.add(centers.get(j));
100                                        subunitIndices.add(j);
101                                        count++;
102                                }
103                        }
104//                      System.out.println("count: " + count);
105                        p.scale(1.0/count);
106//                      System.out.println("Orig Repeat unit: " + p);
107                        repeatCenters.add(p);
108                        repeatUnitIndices.add(subunitIndices);
109                }
110        } else {
111//              System.out.println("calcRepeatUnitCenters case21");
112                // TODO need to group subunits into repeating groups
113                // Case of 3B5U: A14, but seems to form (A2)*7 and symmetry related subunits don't have direct contact
114                List<Integer> sequenceClusterIds = subunits.getClusterIds();
115                for (int i = 0; i < subunits.getSubunitCount(); i++) {
116                        List<Integer> subunitIndices = new ArrayList<>(1);
117                        if (sequenceClusterIds.get(i) == 0) {
118                                repeatCenters.add(new Point3d(centers.get(i)));
119//                              System.out.println("Orig Repeat unit: " + centers.get(i));
120                                subunitIndices.add(i);
121                                repeatUnitIndices.add(subunitIndices);
122                        }
123                }
124        }
125
126        // helixes should have at least 3 repeat centers
127//      System.out.println("Number of repeat centers: " + repeatCenters.size());
128        if (repeatCenters.size() < 3) {
129                repeatCenters.clear();
130        }
131
132        return repeatCenters;
133}
134
135private List<Point3d[]> calcRepeatUnits() {
136
137        // TODO why do we use models here? it should not matter. Setting to 0 all
138        List<Integer> models = new ArrayList<>(
139                        subunits.getSubunitCount());
140        for (int s = 0; s < subunits.getSubunitCount(); s++)
141                models.add(0);
142        Set<Integer> uniqueModels = new HashSet<>(Arrays.asList(1));
143
144        int modelCount = uniqueModels.size();
145        List<Integer> folds = this.subunits.getFolds();
146        int maxFold = folds.get(folds.size()-1);
147
148        List<Point3d[]> repeatTraces = new ArrayList<Point3d[]>();
149        List<Point3d[]> traces = subunits.getTraces();
150
151//      if (modelCount == maxFold && subunitCount > 3) {
152        if (maxFold%modelCount == 0 && modelCount > 1 && subunits.getSubunitCount() > 3) {
153                for (int i = 0; i < modelCount; i++) {
154                        List<Point3d> coords = new ArrayList<Point3d>();
155                        for (int j = 0; j < models.size(); j++) {
156                                if (models.get(j) == i) {
157                                        coords.addAll(Arrays.asList(traces.get(j)));
158                                }
159                        }
160                        Point3d[] x = new Point3d[coords.size()];
161                        coords.toArray(x);
162//                      repeatTraces.add(x); // make sure we don't accidently change the original coordinates
163                        repeatTraces.add(CalcPoint.clonePoint3dArray(x));
164                }
165        } else {
166                List<Integer> sequenceClusterIds = subunits.getClusterIds();
167                for (int i = 0; i < subunits.getSubunitCount(); i++) {
168                        if (sequenceClusterIds.get(i) == 0) {
169                                Point3d[] x = subunits.getTraces().get(i);
170                                repeatTraces.add(CalcPoint.clonePoint3dArray(x));
171                        }
172                }
173        }
174
175//      for (int i = 0; i < repeatTraces.size(); i++) {
176//              System.out.println("Repeat " + i);
177//              System.out.println(Arrays.toString(repeatTraces.get(i)));
178//      }
179        return repeatTraces;
180}
181
182private Map<Integer[], Integer> findInteractingNeigbors() {
183        Map<Integer[], Integer>  contactMap = new HashMap<>();
184
185        Map<Integer, List<Integer[]>> distanceMap = findClosestPairs(8);
186        for (List<Integer[]> pairs: distanceMap.values())
187        for (Integer[] pair: pairs) {
188                Integer contacts = calcContactNumber(repeatUnits.get(pair[0]), repeatUnits.get(pair[1]));
189//              System.out.println("contacts: " + pair[0] + "-" + pair[1] + ": " + contacts);
190                if (contacts > 0) {
191                        contactMap.put(pair, contacts);
192                }
193        }
194
195        return contactMap;
196}
197
198private Map<Integer, List<Integer[]>> findClosestPairs(int maxNeighbors) {
199        Map<Integer, List<Integer[]>>  reducedMap = new TreeMap<>();
200
201        Map<Integer, List<Integer[]>>  distanceMap = new TreeMap<>();
202        int nCenters = repeatUnitCenters.size();
203//      System.out.println("repeatUnitCenters: " + repeatUnitCenters);
204
205        for (int i = 0; i < nCenters-1; i++) {
206                for (int j = i+1; j < nCenters; j++) {
207                        float dist = (float)repeatUnitCenters.get(i).distance(repeatUnitCenters.get(j));
208                        // round to 2 digits precision
209//                      System.out.println("dist pair: " + i + "-" + j + ": " + dist);
210                        dist *= 100;
211                        int intDist = Math.round(dist);
212                        List<Integer[]> pairs = distanceMap.get(intDist);
213                        // save only one representative pair for each distance
214                        if (pairs == null) {
215                                pairs = new ArrayList<>();
216                        }
217                        Integer[] pair = new Integer[2];
218                        pair[0] = i;
219                        pair[1] = j;
220                        pairs.add(pair);
221                        distanceMap.put(intDist, pairs);
222                }
223        }
224
225        int count = 0;
226        for (Entry<Integer, List<Integer[]>> entry: distanceMap.entrySet()) {
227                if (! (reducedMap.containsKey(entry.getKey())) ) {
228                        reducedMap.put(entry.getKey(), entry.getValue());
229//                      System.out.println("dist pair: " + entry.getValue() + ": " + entry.getKey());
230                        count++;
231                        if (count == maxNeighbors) {
232                                break;
233                        }
234                }
235        }
236        distanceMap.clear();
237
238        return reducedMap;
239}
240
241private static int calcContactNumber(Point3d[] a, Point3d[] b) {
242        int contacts = 0;
243        for (Point3d pa : a) {
244                for (Point3d pb : b) {
245                        if (pa.distance(pb) < 10) {
246                                contacts++;
247                        }
248                }
249        }
250        return contacts;
251}
252}