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.geometry;
023
024import org.biojava.nbio.structure.jama.EigenvalueDecomposition;
025import org.biojava.nbio.structure.jama.Matrix;
026
027import javax.vecmath.Matrix3d;
028import javax.vecmath.Point3d;
029import javax.vecmath.Vector3d;
030
031import java.util.ArrayList;
032import java.util.List;
033
034/**
035 * The moment of inertia, otherwise known as the angular mass or rotational
036 * inertia, of a rigid body determines the torque needed for a desired angular
037 * acceleration about a rotational axis. It depends on the body's mass
038 * distribution and the axis chosen, with larger moments requiring more torque
039 * to change the body's rotation.
040 * <p>
041 * More in https://en.wikipedia.org/wiki/Moment_of_inertia.
042 *
043 * @author Peter Rose
044 * @author Aleix Lafita
045 *
046 */
047public class MomentsOfInertia {
048
049        private List<Point3d> points = new ArrayList<Point3d>();
050        private List<Double> masses = new ArrayList<>();
051
052        private boolean modified = true;
053
054        private double[] principalMomentsOfInertia = new double[3];
055        private Vector3d[] principalAxes = new Vector3d[3];
056
057        private Matrix3d orientation = new Matrix3d();
058
059        public enum SymmetryClass {
060                LINEAR, PROLATE, OBLATE, SYMMETRIC, ASYMMETRIC
061        };
062
063        /** Creates a new empty instance of MomentsOfInertia */
064        public MomentsOfInertia() {
065        }
066
067        public void addPoint(Point3d point, double mass) {
068                points.add(point);
069                masses.add(mass);
070                modified = true;
071        }
072
073        public Point3d getCenterOfMass() {
074
075                if (points.size() == 0) {
076                        throw new IllegalStateException(
077                                        "MomentsOfInertia: no points defined");
078                }
079
080                Point3d center = new Point3d();
081                double totalMass = 0.0;
082
083                for (int i = 0, n = points.size(); i < n; i++) {
084                        double mass = masses.get(i);
085                        totalMass += mass;
086                        center.scaleAdd(mass, points.get(i), center);
087                }
088                center.scale(1.0 / totalMass);
089                return center;
090        }
091
092        public double[] getPrincipalMomentsOfInertia() {
093
094                if (modified) {
095                        diagonalizeTensor();
096                        modified = false;
097                }
098                return principalMomentsOfInertia;
099        }
100
101        /**
102         * The principal axes of intertia
103         *
104         * @return
105         */
106        public Vector3d[] getPrincipalAxes() {
107
108                if (modified) {
109                        diagonalizeTensor();
110                        modified = false;
111                }
112                return principalAxes;
113        }
114
115        /**
116         * The orientation Matrix is a 3x3 Matrix with a column for each principal
117         * axis. It represents the orientation (rotation) of the principal axes with
118         * respect to the axes of the coordinate system (unit vectors [1,0,0],
119         * [0,1,0] and [0,0,1]).
120         * <p>
121         * The orientation matrix indicates the rotation to bring the coordinate
122         * axes to the principal axes, in this direction.
123         *
124         * @return the orientation Matrix as a Matrix3d object
125         */
126        public Matrix3d getOrientationMatrix() {
127
128                if (modified) {
129                        diagonalizeTensor();
130                        modified = false;
131                }
132                return orientation;
133        }
134
135        /**
136         * The effective value of this distance for a certain body is known as its
137         * radius of / gyration with respect to the given axis. The radius of
138         * gyration corresponding to Ijj / is defined as /
139         * http://www.eng.auburn.edu/~marghitu/MECH2110/C_4.pdf / radius of gyration
140         * k(j) = sqrt(I(j)/m)
141         */
142        public double[] getElipsisRadii() {
143                if (modified) {
144                        diagonalizeTensor();
145                        modified = false;
146                }
147                double m = 0;
148                for (int i = 0, n = points.size(); i < n; i++) {
149                        m += masses.get(i);
150                }
151                double[] r = new double[3];
152                for (int i = 0; i < 3; i++) {
153                        r[i] = Math.sqrt(principalMomentsOfInertia[i] / m);
154                }
155                return r;
156        }
157
158        public double getRadiusOfGyration() {
159                Point3d c = getCenterOfMass();
160                Point3d t = new Point3d();
161                double sum = 0;
162                for (int i = 0, n = points.size(); i < n; i++) {
163                        t.set(points.get(i));
164                        sum += t.distanceSquared(c);
165                }
166                sum /= points.size();
167                return Math.sqrt(sum);
168        }
169
170        public SymmetryClass getSymmetryClass(double threshold) {
171                if (modified) {
172                        diagonalizeTensor();
173                        modified = false;
174                }
175                double ia = principalMomentsOfInertia[0];
176                double ib = principalMomentsOfInertia[1];
177                double ic = principalMomentsOfInertia[2];
178                boolean c1 = (ib - ia) / (ib + ia) < threshold;
179                boolean c2 = (ic - ib) / (ic + ib) < threshold;
180
181                if (c1 && c2) {
182                        return SymmetryClass.SYMMETRIC;
183                }
184                if (c1) {
185                        return SymmetryClass.OBLATE;
186                }
187                if (c2) {
188                        return SymmetryClass.PROLATE;
189                }
190                return SymmetryClass.ASYMMETRIC;
191        }
192
193        public double symmetryCoefficient() {
194                if (modified) {
195                        diagonalizeTensor();
196                        modified = false;
197                }
198                double ia = principalMomentsOfInertia[0];
199                double ib = principalMomentsOfInertia[1];
200                double ic = principalMomentsOfInertia[2];
201                double c1 = 1.0f - (ib - ia) / (ib + ia);
202                double c2 = 1.0f - (ic - ib) / (ic + ib);
203                return Math.max(c1, c2);
204        }
205
206        public double getAsymmetryParameter(double threshold) {
207                if (modified) {
208                        diagonalizeTensor();
209                        modified = false;
210                }
211                if (getSymmetryClass(threshold).equals(SymmetryClass.SYMMETRIC)) {
212                        return 0.0;
213                }
214                double a = 1.0 / principalMomentsOfInertia[0];
215                double b = 1.0 / principalMomentsOfInertia[1];
216                double c = 1.0 / principalMomentsOfInertia[2];
217                return (2 * b - a - c) / (a - c);
218        }
219
220        public double[][] getInertiaTensor() {
221
222                Point3d p = new Point3d();
223                double[][] tensor = new double[3][3];
224
225                // calculate the inertia tensor at center of mass
226                Point3d com = getCenterOfMass();
227
228                for (int i = 0, n = points.size(); i < n; i++) {
229                        double mass = masses.get(i);
230                        p.sub(points.get(i), com);
231
232                        tensor[0][0] += mass * (p.y * p.y + p.z * p.z);
233                        tensor[1][1] += mass * (p.x * p.x + p.z * p.z);
234                        tensor[2][2] += mass * (p.x * p.x + p.y * p.y);
235
236                        tensor[0][1] -= mass * p.x * p.y;
237                        tensor[0][2] -= mass * p.x * p.z;
238                        tensor[1][2] -= mass * p.y * p.z;
239                }
240
241                tensor[1][0] = tensor[0][1];
242                tensor[2][0] = tensor[0][2];
243                tensor[2][1] = tensor[1][2];
244
245                return tensor;
246        }
247
248        private void diagonalizeTensor() {
249
250                Matrix m = new Matrix(getInertiaTensor());
251                EigenvalueDecomposition eig = m.eig();
252
253                principalMomentsOfInertia = eig.getRealEigenvalues();
254                double[][] eigenVectors = eig.getV().getArray();
255
256                // Get the principal axes from the eigenVectors
257                principalAxes[0] = new Vector3d(eigenVectors[0][0], eigenVectors[1][0],
258                                eigenVectors[2][0]);
259                principalAxes[1] = new Vector3d(eigenVectors[0][1], eigenVectors[1][1],
260                                eigenVectors[2][1]);
261                principalAxes[2] = new Vector3d(eigenVectors[0][2], eigenVectors[1][2],
262                                eigenVectors[2][2]);
263
264                // Convert the principal axes into a rotation matrix
265                for (int i = 0; i < 3; i++)
266                        orientation.setColumn(i, principalAxes[i]);
267                orientation.negate();
268
269        }
270}