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.internal; 022 023import static java.lang.Math.*; 024 025import org.biojava.nbio.structure.Atom; 026import org.biojava.nbio.structure.align.model.AFPChain; 027import org.biojava.nbio.structure.align.util.RotationAxis; 028import org.biojava.nbio.structure.symmetry.internal.OrderDetector; 029import org.biojava.nbio.structure.symmetry.internal.RefinerFailedException; 030 031/** 032 * Guesses an order of rotational symmetry from the angle. 033 * <p> 034 * Improves upon the AngleOrderDetector used in the paper by checking all 035 * rotations, not just the base one. 036 * 037 * @author Spencer Bliven 038 * @since 4.2.0 039 * 040 */ 041public class AngleOrderDetectorPlus implements OrderDetector { 042 043 private int maxOrder; 044 private final double error; 045 private boolean normalizeError; 046 047 /** 048 * @param angleError 049 * maximum angular error, in radians 050 */ 051 public AngleOrderDetectorPlus(double angleError) { 052 this(8, angleError, false); 053 } 054 055 public AngleOrderDetectorPlus(int maxOrder) { 056 // PI is the max error for C1 057 this(maxOrder, PI, false); 058 } 059 060 /** 061 * 062 * @param maxOrder 063 * maximum order to consider 064 * @param angleError 065 * maximum angular error, in radians 066 */ 067 public AngleOrderDetectorPlus(int maxOrder, double angleError) { 068 this(maxOrder, angleError, false); 069 } 070 071 /** 072 * Determine order by finding the order (up to the maxOrder) which has the 073 * closest rotation angle to the observed rotation. 074 * 075 * If normalized is false, then the error is taken to be the absolute error 076 * from the closest ideal angle (in radians). If normalized is true, error 077 * is taken to be relative to the fundamental rotation for a given order. 078 * For instance, for an error of .25, C2 order would be accepted for angles 079 * within .25*pi radians of 0 or pi, while C3 order would be acceptable 080 * within .25*2pi/3 radians of 0, 2pi/3, or 4pi/3. In the normalized case, 081 * numbers between 0 and .5 are sensible for error. 082 * 083 * @param maxOrder 084 * maximum order to consider 085 * @param angleError 086 * maximum angular error 087 * @param normalize 088 * indicates whether error should be normalized by the order 089 */ 090 public AngleOrderDetectorPlus(int maxOrder, double angleError, 091 boolean normalize) { 092 super(); 093 this.maxOrder = maxOrder; 094 this.error = angleError; 095 this.normalizeError = normalize; 096 } 097 098 @Override 099 public int calculateOrder(AFPChain afpChain, Atom[] ca) 100 throws RefinerFailedException { 101 final double tol = 1e-6; // tolerance to floating point errors 102 try { 103 RotationAxis axis = new RotationAxis(afpChain); 104 double theta = axis.getAngle(); 105 106 double bestDelta = error; 107 int bestOrder = 1; 108 for (int order = 1; order <= maxOrder; order++) { 109 // Triangle wave starting at 0 with period 2pi/order 110 double delta = abs(abs(theta * order / (2 * PI) - .5) % 1.0 - .5); 111 // Triangle waves have amplitude 1, so need to un-normalize 112 if (!normalizeError) 113 delta *= 2 * PI / order; 114 115 if (delta < bestDelta - tol) { 116 bestOrder = order; 117 bestDelta = delta; 118 } 119 } 120 return bestOrder; 121 } catch (Exception e) { 122 throw new RefinerFailedException(e); 123 } 124 } 125 126 @Override 127 public String toString() { 128 return getClass().getSimpleName() + "[maxOrder=" + maxOrder 129 + ", error=" + error + ", normalizeError=" + normalizeError 130 + "]"; 131 } 132 133}