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 java.util.ArrayList;
025import java.util.List;
026
027/**
028 *
029 * @author Peter
030 */
031public class HelixLayers {
032        private List<Helix> helices = new ArrayList<>();
033        private double symmetryDeviation = 0;
034
035        public int size() {
036                return helices.size();
037        }
038
039        public void addHelix(Helix helix) {
040                helices.add(helix);
041        }
042
043        public Helix getHelix(int index) {
044                return helices.get(index);
045        }
046
047        /*
048         * Returns Helix with lowest twist angle
049         */
050        public Helix getByLowestAngle() {
051                double angle = Double.MAX_VALUE;
052                Helix lowest = null;
053                for (Helix helix: helices) {
054                        if (helix.getAngle() < angle) {
055                                angle = helix.getAngle();
056                                lowest = helix;
057                        }
058                }
059                return lowest;
060        }
061
062        /*
063         * Returns Helix with largest number of intermolecular contacts
064         * between repeat units
065         */
066        public Helix getByLargestContacts() {
067                double contacts = 0;
068                Helix largest = null;
069                for (Helix helix: helices) {
070                        if (helix.getContacts() > contacts) {
071                                contacts = helix.getContacts();
072                                largest = helix;
073                        }
074                }
075                return largest;
076        }
077
078        /*
079         * Returns Helix that has the largest number of contacts, besides
080         * the Helix with the lowest twist angle
081         */
082        public Helix getByLargestContactsNotLowestAngle() {
083                double contacts = 0;
084                Helix lowest = getByLowestAngle();
085                // TODO why are there helices with almost identical helix parameters??
086                double angle = lowest.getAngle() + 0.05;
087                Helix largest = null;
088                for (Helix helix: helices) {
089                        if (helix == lowest) {
090                                continue;
091                        }
092                        if (helix.getContacts() > contacts && helix.getAngle() > angle) {
093                                contacts = helix.getContacts();
094                                largest = helix;
095                        }
096                }
097                if (largest == null) {
098                        return lowest;
099                }
100                return largest;
101        }
102
103        /**
104         * Returns QuatSymmetryScores averaged over all rotations
105         * (except the first rotation, which is the unit operation E)
106         * @return mean scores average over rotations
107         */
108        public QuatSymmetryScores getScores() {
109                QuatSymmetryScores scores = new QuatSymmetryScores();
110
111                double[] values = new double[helices.size()];
112
113                // minRmsd
114                for (int i = 0; i < helices.size(); i++) {
115                        values[i] = helices.get(i).getScores().getMinRmsd();
116                }
117                scores.setMinRmsd(minScores(values));
118
119                // maxRmsd
120                for (int i = 0; i < helices.size(); i++) {
121                        values[i] = helices.get(i).getScores().getMaxRmsd();
122                }
123                scores.setMaxRmsd(maxScores(values));
124
125                // Rmsd
126                for (int i = 0; i < helices.size(); i++) {
127                        values[i] = helices.get(i).getScores().getRmsd();
128                }
129                scores.setRmsd(averageScores(values));
130
131                // minTm
132                for (int i = 0; i < helices.size(); i++) {
133                        values[i] = helices.get(i).getScores().getMinTm();
134                }
135                scores.setMinTm(minScores(values));
136
137                // maxTm
138                for (int i = 0; i < helices.size(); i++) {
139                        values[i] = helices.get(i).getScores().getMaxTm();
140                }
141                scores.setMaxTm(maxScores(values));
142
143                // Tm
144                for (int i = 0; i < helices.size(); i++) {
145                        values[i] = helices.get(i).getScores().getTm();
146                }
147                scores.setTm(averageScores(values));
148
149                // Rmsd subunit centers
150                for (int i = 0; i < helices.size(); i++) {
151                        values[i] = helices.get(i).getScores().getRmsdCenters();
152                }
153                scores.setRmsdCenters(averageScores(values));
154
155                // TmIntra
156                for (int i = 0; i < helices.size(); i++) {
157                        values[i] = helices.get(i).getScores().getTmIntra();
158                }
159                scores.setTmIntra(averageScores(values));
160
161                // RmsdIntra
162                for (int i = 0; i < helices.size(); i++) {
163                        values[i] = helices.get(i).getScores().getRmsdIntra();
164                }
165                scores.setRmsdIntra(averageScores(values));
166
167                // SymDeviation
168                scores.setSymDeviation(symmetryDeviation);
169
170                return scores;
171        }
172
173        /**
174         * @param symmetryDeviation the symmetryDeviation to set
175         */
176        public void setSymmetryDeviation(double symmetryDeviation) {
177                this.symmetryDeviation = symmetryDeviation;
178        }
179
180        private double averageScores(double[] scores) {
181                double sum = 0;
182                for (double s: scores) {
183                        sum += s;
184                }
185                return sum/scores.length;
186        }
187
188        private double minScores(double[] scores) {
189                double score = Double.MAX_VALUE;
190                for (double s: scores) {
191                        score = Math.min(score, s);
192                }
193                return score;
194        }
195
196        private double maxScores(double[] scores) {
197                double score = Double.MIN_VALUE;
198                for (double s: scores) {
199                        score = Math.max(score, s);
200                }
201                return score;
202        }
203
204        public void clear() {
205                helices.clear();
206        }
207
208        @Override
209        public String toString() {
210                StringBuilder sb = new StringBuilder();
211                sb.append("Helices: ").append(size()).append("\n");
212                for (Helix s: helices) {
213                        sb.append(s.toString()).append("\n");
214                }
215                return sb.toString();
216        }
217
218
219
220}