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/**
022 *
023 */
024package org.biojava.nbio.structure.symmetry.geometry;
025
026import javax.vecmath.Matrix3d;
027import javax.vecmath.Point3d;
028import java.util.ArrayList;
029import java.util.List;
030
031
032/**
033 * @author Peter
034 *
035 */
036public class Prism implements Polyhedron {
037        private int n = 3;
038        private double circumscribedRadius = 1.0;
039        private double height = 1.0;
040
041        public Prism(int n) {
042                this.n = n;
043        }
044
045        /**
046         * @return the height
047         */
048        public double getHeight() {
049                return height;
050        }
051
052        /**
053         * @param height the height to set
054         */
055        public void setHeight(double height) {
056                this.height = height;
057        }
058
059        /**
060         * Returns the radius of a circumscribed sphere, that goes
061         * through all vertices
062         * @return the cirumscribedRadius
063         */
064        @Override
065        public double getCirumscribedRadius() {
066                return circumscribedRadius;
067        }
068
069        /**
070         * Set the radius of a circumscribed sphere, that goes
071         * through all vertices
072         * @param cirumscribedRadius the cirumscribedRadius to set
073         */
074        public void setCirumscribedRadius(double cirumscribedRadius) {
075                this.circumscribedRadius = cirumscribedRadius;
076        }
077        /**
078         * Returns the radius of an inscribed sphere, that is tangent to each
079         * of the icosahedron's faces
080         * @return the inscribedRadius
081         */
082        public double getInscribedRadius() {
083                double side = getSideLengthFromCircumscribedRadius(circumscribedRadius, n);
084                return getInscribedRadiusFromSideLength(side, n);
085        }
086
087        /**
088         * Sets the radius of an inscribed sphere, that is tangent to each
089         * of the icosahedron's faces
090         * @param radius the inscribedRadius to set
091         */
092        public void setInscribedRadius(double radius) {
093                double side = getSideLengthFromInscribedRadius(radius, n);
094                this.circumscribedRadius = getCircumscribedRadiusFromSideLength(side, n);
095        }
096
097        /**
098         * Returns the vertices of an n-fold polygon of given radius and center
099         * @return
100         */
101        @Override
102        public Point3d[] getVertices() {
103                Point3d[] polygon = new Point3d[2*n];
104                Matrix3d m = new Matrix3d();
105
106                Point3d center = new Point3d(0, 0, height/2);
107
108                for (int i = 0; i < n; i++) {
109                        polygon[i] = new Point3d(0, circumscribedRadius, 0);
110                        m.rotZ(i*2*Math.PI/n);
111                        m.transform(polygon[i]);
112                        polygon[n+i] = new  Point3d(polygon[i]);
113                        polygon[i].sub(center);
114                        polygon[n+i].add(center);
115                }
116
117                return polygon;
118        };
119
120        @Override
121        public List<int[]> getLineLoops() {
122                List<int[]> list = new ArrayList<>();
123                int[] l1 = new int[2*n+2];
124                for (int i = 0; i < n; i++) {
125                        l1[i] = i;
126                }
127                l1[n] = 0;
128                for (int i = 0; i < n; i++) {
129                        l1[n+i+1] = n+i;
130                }
131                l1[2*n+1] = l1[n+1];
132                list.add(l1);
133
134                for (int i = 1; i < n; i++) {
135                        int[] l2 = new int[2];
136                        l2[0] = i;
137                        l2[1] = n+i;
138                        list.add(l2);
139                }
140                return list;
141        }
142
143        /**
144         * Returns the vertices of an n-fold polygon of given radius and center
145         * @return
146         */
147        public static Point3d[] getPolygonVertices(int n, double radius, Point3d center) {
148                Point3d[] polygon = new Point3d[n];
149                Matrix3d m = new Matrix3d();
150
151                for (int i = 0; i < n; i++) {
152                        polygon[i] = new Point3d(0, radius, 0);
153                        m.rotZ(i*2*Math.PI/n);
154                        m.transform(polygon[i]);
155                        polygon[i].add(center);
156                }
157                return polygon;
158        }
159
160        @Override
161        public int getViewCount() {
162                return 4;
163        }
164
165        @Override
166        public String getViewName(int index) {
167                String name;
168                switch (index) {
169                case 0:  name = "Front C" + n + " axis";
170                break;
171                case 1:  name = "Side edge-centered";
172                break;
173                case 2:  name = "Side face-centered";
174                break;
175                case 3:  name = "Back C" + n + " axis";
176                break;
177                default: throw new IllegalArgumentException("getViewMatrix: index out of range:" + index);
178                }
179                return name;
180        }
181
182        @Override
183        public Matrix3d getViewMatrix(int index) {
184                Matrix3d m = new Matrix3d();
185                switch (index) {
186                case 0:
187                        m.setIdentity(); // front
188                        break;
189                case 1:
190                        m.rotX(Math.PI/2); // side edge-centered
191                        break;
192                case 2:
193                        m.rotY(Math.PI/n); // side face-centered
194                        Matrix3d m1 = new Matrix3d();
195                        m1.rotX(Math.PI/2);
196                        m.mul(m1);
197                        break;
198                case 3:
199                        m.set(flipX()); // back
200                        break;
201                default:
202                        throw new IllegalArgumentException("getViewMatrix: index out of range:" + index);
203                }
204                return m;
205        }
206
207        // http://www.mathopenref.com/polygonincircle.html
208        private static double getSideLengthFromInscribedRadius(double radius, int n) {
209                if (n == 2) {
210                        return radius;
211                }
212                return radius * 2 * Math.tan(Math.PI/n);
213        }
214
215        private static double getInscribedRadiusFromSideLength(double length, int n) {
216                if (n == 2) {
217                        return length;
218                }
219                return length / (2 * Math.tan(Math.PI/n));
220        }
221
222        // http://www.mathopenref.com/polygonradius.html
223        private static double getSideLengthFromCircumscribedRadius(double radius, int n) {
224                if (n == 2) {
225                        return radius;
226                }
227                return radius * (2 * Math.sin(Math.PI/n));
228        }
229
230        private static double getCircumscribedRadiusFromSideLength(double length, int n) {
231                if (n == 2) {
232                        return length;
233                }
234                return length / (2 * Math.sin(Math.PI/n));
235        }
236
237        private static Matrix3d flipX() {
238                Matrix3d rot = new Matrix3d();
239                rot.m00 = 1;
240                rot.m11 = -1;
241                rot.m22 = -1;
242                return rot;
243        }
244
245}